Refactoring

This commit is contained in:
DahoudG
2025-09-17 17:54:06 +00:00
parent 12d514d866
commit 63fe107f98
165 changed files with 54220 additions and 276 deletions

View File

@@ -0,0 +1,414 @@
import 'package:equatable/equatable.dart';
import 'package:json_annotation/json_annotation.dart';
part 'notification.g.dart';
/// Énumération des types de notification
enum TypeNotification {
// Événements
@JsonValue('NOUVEL_EVENEMENT')
nouvelEvenement('Nouvel événement', 'evenements', 'info', 'event', '#FF9800'),
@JsonValue('RAPPEL_EVENEMENT')
rappelEvenement('Rappel d\'événement', 'evenements', 'reminder', 'schedule', '#2196F3'),
@JsonValue('EVENEMENT_ANNULE')
evenementAnnule('Événement annulé', 'evenements', 'warning', 'event_busy', '#F44336'),
@JsonValue('INSCRIPTION_CONFIRMEE')
inscriptionConfirmee('Inscription confirmée', 'evenements', 'success', 'check_circle', '#4CAF50'),
// Cotisations
@JsonValue('COTISATION_DUE')
cotisationDue('Cotisation due', 'cotisations', 'reminder', 'payment', '#FF5722'),
@JsonValue('COTISATION_PAYEE')
cotisationPayee('Cotisation payée', 'cotisations', 'success', 'paid', '#4CAF50'),
@JsonValue('PAIEMENT_CONFIRME')
paiementConfirme('Paiement confirmé', 'cotisations', 'success', 'check_circle', '#4CAF50'),
@JsonValue('PAIEMENT_ECHOUE')
paiementEchoue('Paiement échoué', 'cotisations', 'error', 'error', '#F44336'),
// Solidarité
@JsonValue('NOUVELLE_DEMANDE_AIDE')
nouvelleDemandeAide('Nouvelle demande d\'aide', 'solidarite', 'info', 'help', '#E91E63'),
@JsonValue('DEMANDE_AIDE_APPROUVEE')
demandeAideApprouvee('Demande d\'aide approuvée', 'solidarite', 'success', 'thumb_up', '#4CAF50'),
@JsonValue('AIDE_DISPONIBLE')
aideDisponible('Aide disponible', 'solidarite', 'info', 'volunteer_activism', '#E91E63'),
// Membres
@JsonValue('NOUVEAU_MEMBRE')
nouveauMembre('Nouveau membre', 'membres', 'info', 'person_add', '#2196F3'),
@JsonValue('ANNIVERSAIRE_MEMBRE')
anniversaireMembre('Anniversaire de membre', 'membres', 'celebration', 'cake', '#FF9800'),
// Organisation
@JsonValue('ANNONCE_GENERALE')
annonceGenerale('Annonce générale', 'organisation', 'info', 'campaign', '#2196F3'),
@JsonValue('REUNION_PROGRAMMEE')
reunionProgrammee('Réunion programmée', 'organisation', 'info', 'groups', '#2196F3'),
// Messages
@JsonValue('MESSAGE_PRIVE')
messagePrive('Message privé', 'messages', 'info', 'mail', '#2196F3'),
@JsonValue('MENTION')
mention('Mention', 'messages', 'info', 'alternate_email', '#FF9800'),
// Système
@JsonValue('MISE_A_JOUR_APP')
miseAJourApp('Mise à jour disponible', 'systeme', 'info', 'system_update', '#2196F3'),
@JsonValue('MAINTENANCE_PROGRAMMEE')
maintenanceProgrammee('Maintenance programmée', 'systeme', 'warning', 'build', '#FF9800');
const TypeNotification(this.libelle, this.categorie, this.priorite, this.icone, this.couleur);
final String libelle;
final String categorie;
final String priorite;
final String icone;
final String couleur;
bool get isCritique => priorite == 'urgent' || priorite == 'error';
bool get isRappel => priorite == 'reminder';
bool get isPositive => priorite == 'success' || priorite == 'celebration';
int get niveauPriorite {
switch (priorite) {
case 'urgent': return 1;
case 'error': return 2;
case 'warning': return 3;
case 'important': return 4;
case 'reminder': return 5;
case 'info': return 6;
case 'success': return 7;
case 'celebration': return 8;
default: return 6;
}
}
}
/// Énumération des statuts de notification
enum StatutNotification {
@JsonValue('BROUILLON')
brouillon('Brouillon', 'draft', '#9E9E9E'),
@JsonValue('PROGRAMMEE')
programmee('Programmée', 'scheduled', '#FF9800'),
@JsonValue('ENVOYEE')
envoyee('Envoyée', 'sent', '#4CAF50'),
@JsonValue('RECUE')
recue('Reçue', 'received', '#4CAF50'),
@JsonValue('AFFICHEE')
affichee('Affichée', 'displayed', '#2196F3'),
@JsonValue('OUVERTE')
ouverte('Ouverte', 'opened', '#4CAF50'),
@JsonValue('LUE')
lue('Lue', 'read', '#4CAF50'),
@JsonValue('NON_LUE')
nonLue('Non lue', 'unread', '#FF9800'),
@JsonValue('MARQUEE_IMPORTANTE')
marqueeImportante('Marquée importante', 'starred', '#FF9800'),
@JsonValue('SUPPRIMEE')
supprimee('Supprimée', 'deleted', '#F44336'),
@JsonValue('ARCHIVEE')
archivee('Archivée', 'archived', '#9E9E9E'),
@JsonValue('ECHEC_ENVOI')
echecEnvoi('Échec d\'envoi', 'failed', '#F44336');
const StatutNotification(this.libelle, this.code, this.couleur);
final String libelle;
final String code;
final String couleur;
bool get isSucces => this == envoyee || this == recue || this == affichee || this == ouverte || this == lue;
bool get isErreur => this == echecEnvoi;
bool get isFinal => this == supprimee || this == archivee || isErreur;
}
/// Action rapide de notification
@JsonSerializable()
class ActionNotification extends Equatable {
const ActionNotification({
required this.id,
required this.libelle,
required this.typeAction,
this.description,
this.icone,
this.couleur,
this.url,
this.route,
this.parametres,
this.fermeNotification = true,
this.necessiteConfirmation = false,
this.estDestructive = false,
this.ordre = 0,
this.estActivee = true,
});
final String id;
final String libelle;
final String? description;
final String typeAction;
final String? icone;
final String? couleur;
final String? url;
final String? route;
final Map<String, String>? parametres;
final bool fermeNotification;
final bool necessiteConfirmation;
final bool estDestructive;
final int ordre;
final bool estActivee;
factory ActionNotification.fromJson(Map<String, dynamic> json) =>
_$ActionNotificationFromJson(json);
Map<String, dynamic> toJson() => _$ActionNotificationToJson(this);
@override
List<Object?> get props => [
id, libelle, description, typeAction, icone, couleur,
url, route, parametres, fermeNotification, necessiteConfirmation,
estDestructive, ordre, estActivee,
];
ActionNotification copyWith({
String? id,
String? libelle,
String? description,
String? typeAction,
String? icone,
String? couleur,
String? url,
String? route,
Map<String, String>? parametres,
bool? fermeNotification,
bool? necessiteConfirmation,
bool? estDestructive,
int? ordre,
bool? estActivee,
}) {
return ActionNotification(
id: id ?? this.id,
libelle: libelle ?? this.libelle,
description: description ?? this.description,
typeAction: typeAction ?? this.typeAction,
icone: icone ?? this.icone,
couleur: couleur ?? this.couleur,
url: url ?? this.url,
route: route ?? this.route,
parametres: parametres ?? this.parametres,
fermeNotification: fermeNotification ?? this.fermeNotification,
necessiteConfirmation: necessiteConfirmation ?? this.necessiteConfirmation,
estDestructive: estDestructive ?? this.estDestructive,
ordre: ordre ?? this.ordre,
estActivee: estActivee ?? this.estActivee,
);
}
}
/// Entité principale de notification
@JsonSerializable()
class NotificationEntity extends Equatable {
const NotificationEntity({
required this.id,
required this.typeNotification,
required this.statut,
required this.titre,
required this.message,
this.messageCourt,
this.expediteurId,
this.expediteurNom,
required this.destinatairesIds,
this.organisationId,
this.donneesPersonnalisees,
this.imageUrl,
this.iconeUrl,
this.actionClic,
this.parametresAction,
this.actionsRapides,
required this.dateCreation,
this.dateEnvoiProgramme,
this.dateEnvoi,
this.dateExpiration,
this.dateDerniereLecture,
this.priorite = 3,
this.estLue = false,
this.estImportante = false,
this.estArchivee = false,
this.nombreAffichages = 0,
this.nombreClics = 0,
this.tags,
this.campagneId,
this.plateforme,
this.tokenFCM,
});
final String id;
final TypeNotification typeNotification;
final StatutNotification statut;
final String titre;
final String message;
final String? messageCourt;
final String? expediteurId;
final String? expediteurNom;
final List<String> destinatairesIds;
final String? organisationId;
final Map<String, dynamic>? donneesPersonnalisees;
final String? imageUrl;
final String? iconeUrl;
final String? actionClic;
final Map<String, String>? parametresAction;
final List<ActionNotification>? actionsRapides;
final DateTime dateCreation;
final DateTime? dateEnvoiProgramme;
final DateTime? dateEnvoi;
final DateTime? dateExpiration;
final DateTime? dateDerniereLecture;
final int priorite;
final bool estLue;
final bool estImportante;
final bool estArchivee;
final int nombreAffichages;
final int nombreClics;
final List<String>? tags;
final String? campagneId;
final String? plateforme;
final String? tokenFCM;
factory NotificationEntity.fromJson(Map<String, dynamic> json) =>
_$NotificationEntityFromJson(json);
Map<String, dynamic> toJson() => _$NotificationEntityToJson(this);
@override
List<Object?> get props => [
id, typeNotification, statut, titre, message, messageCourt,
expediteurId, expediteurNom, destinatairesIds, organisationId,
donneesPersonnalisees, imageUrl, iconeUrl, actionClic, parametresAction,
actionsRapides, dateCreation, dateEnvoiProgramme, dateEnvoi,
dateExpiration, dateDerniereLecture, priorite, estLue, estImportante,
estArchivee, nombreAffichages, nombreClics, tags, campagneId,
plateforme, tokenFCM,
];
NotificationEntity copyWith({
String? id,
TypeNotification? typeNotification,
StatutNotification? statut,
String? titre,
String? message,
String? messageCourt,
String? expediteurId,
String? expediteurNom,
List<String>? destinatairesIds,
String? organisationId,
Map<String, dynamic>? donneesPersonnalisees,
String? imageUrl,
String? iconeUrl,
String? actionClic,
Map<String, String>? parametresAction,
List<ActionNotification>? actionsRapides,
DateTime? dateCreation,
DateTime? dateEnvoiProgramme,
DateTime? dateEnvoi,
DateTime? dateExpiration,
DateTime? dateDerniereLecture,
int? priorite,
bool? estLue,
bool? estImportante,
bool? estArchivee,
int? nombreAffichages,
int? nombreClics,
List<String>? tags,
String? campagneId,
String? plateforme,
String? tokenFCM,
}) {
return NotificationEntity(
id: id ?? this.id,
typeNotification: typeNotification ?? this.typeNotification,
statut: statut ?? this.statut,
titre: titre ?? this.titre,
message: message ?? this.message,
messageCourt: messageCourt ?? this.messageCourt,
expediteurId: expediteurId ?? this.expediteurId,
expediteurNom: expediteurNom ?? this.expediteurNom,
destinatairesIds: destinatairesIds ?? this.destinatairesIds,
organisationId: organisationId ?? this.organisationId,
donneesPersonnalisees: donneesPersonnalisees ?? this.donneesPersonnalisees,
imageUrl: imageUrl ?? this.imageUrl,
iconeUrl: iconeUrl ?? this.iconeUrl,
actionClic: actionClic ?? this.actionClic,
parametresAction: parametresAction ?? this.parametresAction,
actionsRapides: actionsRapides ?? this.actionsRapides,
dateCreation: dateCreation ?? this.dateCreation,
dateEnvoiProgramme: dateEnvoiProgramme ?? this.dateEnvoiProgramme,
dateEnvoi: dateEnvoi ?? this.dateEnvoi,
dateExpiration: dateExpiration ?? this.dateExpiration,
dateDerniereLecture: dateDerniereLecture ?? this.dateDerniereLecture,
priorite: priorite ?? this.priorite,
estLue: estLue ?? this.estLue,
estImportante: estImportante ?? this.estImportante,
estArchivee: estArchivee ?? this.estArchivee,
nombreAffichages: nombreAffichages ?? this.nombreAffichages,
nombreClics: nombreClics ?? this.nombreClics,
tags: tags ?? this.tags,
campagneId: campagneId ?? this.campagneId,
plateforme: plateforme ?? this.plateforme,
tokenFCM: tokenFCM ?? this.tokenFCM,
);
}
/// Vérifie si la notification est expirée
bool get isExpiree {
if (dateExpiration == null) return false;
return DateTime.now().isAfter(dateExpiration!);
}
/// Vérifie si la notification est récente (moins de 24h)
bool get isRecente {
final maintenant = DateTime.now();
final difference = maintenant.difference(dateCreation);
return difference.inHours < 24;
}
/// Retourne le temps écoulé depuis la création
String get tempsEcoule {
final maintenant = DateTime.now();
final difference = maintenant.difference(dateCreation);
if (difference.inMinutes < 1) {
return 'À l\'instant';
} else if (difference.inMinutes < 60) {
return 'Il y a ${difference.inMinutes}min';
} else if (difference.inHours < 24) {
return 'Il y a ${difference.inHours}h';
} else if (difference.inDays < 7) {
return 'Il y a ${difference.inDays}j';
} else {
return 'Il y a ${(difference.inDays / 7).floor()}sem';
}
}
/// Retourne le message à afficher (court ou complet)
String get messageAffichage => messageCourt ?? message;
/// Retourne la couleur du type de notification
String get couleurType => typeNotification.couleur;
/// Retourne l'icône du type de notification
String get iconeType => typeNotification.icone;
/// Vérifie si la notification a des actions rapides
bool get hasActionsRapides => actionsRapides != null && actionsRapides!.isNotEmpty;
/// Retourne les actions rapides actives
List<ActionNotification> get actionsRapidesActives {
if (actionsRapides == null) return [];
return actionsRapides!.where((action) => action.estActivee).toList();
}
/// Calcule le taux d'engagement
double get tauxEngagement {
if (nombreAffichages == 0) return 0.0;
return (nombreClics / nombreAffichages) * 100;
}
}

View File

@@ -0,0 +1,451 @@
import 'package:equatable/equatable.dart';
import 'package:json_annotation/json_annotation.dart';
import 'notification.dart';
part 'preferences_notification.g.dart';
/// Énumération des canaux de notification
enum CanalNotification {
@JsonValue('URGENT_CHANNEL')
urgent('urgent', 'Notifications urgentes', 5, true, true, '#F44336'),
@JsonValue('ERROR_CHANNEL')
error('error', 'Erreurs système', 4, true, true, '#F44336'),
@JsonValue('WARNING_CHANNEL')
warning('warning', 'Avertissements', 4, true, true, '#FF9800'),
@JsonValue('IMPORTANT_CHANNEL')
important('important', 'Notifications importantes', 4, true, true, '#FF5722'),
@JsonValue('REMINDER_CHANNEL')
reminder('reminder', 'Rappels', 3, true, true, '#2196F3'),
@JsonValue('SUCCESS_CHANNEL')
success('success', 'Confirmations', 2, false, false, '#4CAF50'),
@JsonValue('DEFAULT_CHANNEL')
defaultChannel('default', 'Notifications générales', 2, false, false, '#2196F3'),
@JsonValue('EVENTS_CHANNEL')
events('events', 'Événements', 3, true, false, '#2196F3'),
@JsonValue('PAYMENTS_CHANNEL')
payments('payments', 'Paiements', 4, true, true, '#4CAF50'),
@JsonValue('SOLIDARITY_CHANNEL')
solidarity('solidarity', 'Solidarité', 3, true, false, '#E91E63'),
@JsonValue('MEMBERS_CHANNEL')
members('members', 'Membres', 2, false, false, '#2196F3'),
@JsonValue('ORGANIZATION_CHANNEL')
organization('organization', 'Organisation', 3, true, false, '#2196F3'),
@JsonValue('SYSTEM_CHANNEL')
system('system', 'Système', 2, false, false, '#607D8B'),
@JsonValue('MESSAGES_CHANNEL')
messages('messages', 'Messages', 3, true, false, '#2196F3');
const CanalNotification(this.id, this.nom, this.importance, this.sonActive,
this.vibrationActive, this.couleur);
final String id;
final String nom;
final int importance;
final bool sonActive;
final bool vibrationActive;
final String couleur;
bool get isCritique => importance >= 4;
bool get isSilencieux => !sonActive && !vibrationActive;
}
/// Préférences spécifiques à un type de notification
@JsonSerializable()
class PreferenceTypeNotification extends Equatable {
const PreferenceTypeNotification({
this.active = true,
this.priorite,
this.sonPersonnalise,
this.patternVibration,
this.couleurLED,
this.dureeAffichageSecondes,
this.doitVibrer,
this.doitEmettreSon,
this.doitAllumerLED,
this.ignoreModesilencieux = false,
});
final bool active;
final int? priorite;
final String? sonPersonnalise;
final List<int>? patternVibration;
final String? couleurLED;
final int? dureeAffichageSecondes;
final bool? doitVibrer;
final bool? doitEmettreSon;
final bool? doitAllumerLED;
final bool ignoreModesilencieux;
factory PreferenceTypeNotification.fromJson(Map<String, dynamic> json) =>
_$PreferenceTypeNotificationFromJson(json);
Map<String, dynamic> toJson() => _$PreferenceTypeNotificationToJson(this);
@override
List<Object?> get props => [
active, priorite, sonPersonnalise, patternVibration, couleurLED,
dureeAffichageSecondes, doitVibrer, doitEmettreSon, doitAllumerLED,
ignoreModesilencieux,
];
PreferenceTypeNotification copyWith({
bool? active,
int? priorite,
String? sonPersonnalise,
List<int>? patternVibration,
String? couleurLED,
int? dureeAffichageSecondes,
bool? doitVibrer,
bool? doitEmettreSon,
bool? doitAllumerLED,
bool? ignoreModesilencieux,
}) {
return PreferenceTypeNotification(
active: active ?? this.active,
priorite: priorite ?? this.priorite,
sonPersonnalise: sonPersonnalise ?? this.sonPersonnalise,
patternVibration: patternVibration ?? this.patternVibration,
couleurLED: couleurLED ?? this.couleurLED,
dureeAffichageSecondes: dureeAffichageSecondes ?? this.dureeAffichageSecondes,
doitVibrer: doitVibrer ?? this.doitVibrer,
doitEmettreSon: doitEmettreSon ?? this.doitEmettreSon,
doitAllumerLED: doitAllumerLED ?? this.doitAllumerLED,
ignoreModesilencieux: ignoreModesilencieux ?? this.ignoreModesilencieux,
);
}
}
/// Préférences spécifiques à un canal de notification
@JsonSerializable()
class PreferenceCanalNotification extends Equatable {
const PreferenceCanalNotification({
this.active = true,
this.importance,
this.sonPersonnalise,
this.patternVibration,
this.couleurLED,
this.sonActive,
this.vibrationActive,
this.ledActive,
this.peutEtreDesactive = true,
});
final bool active;
final int? importance;
final String? sonPersonnalise;
final List<int>? patternVibration;
final String? couleurLED;
final bool? sonActive;
final bool? vibrationActive;
final bool? ledActive;
final bool peutEtreDesactive;
factory PreferenceCanalNotification.fromJson(Map<String, dynamic> json) =>
_$PreferenceCanalNotificationFromJson(json);
Map<String, dynamic> toJson() => _$PreferenceCanalNotificationToJson(this);
@override
List<Object?> get props => [
active, importance, sonPersonnalise, patternVibration, couleurLED,
sonActive, vibrationActive, ledActive, peutEtreDesactive,
];
PreferenceCanalNotification copyWith({
bool? active,
int? importance,
String? sonPersonnalise,
List<int>? patternVibration,
String? couleurLED,
bool? sonActive,
bool? vibrationActive,
bool? ledActive,
bool? peutEtreDesactive,
}) {
return PreferenceCanalNotification(
active: active ?? this.active,
importance: importance ?? this.importance,
sonPersonnalise: sonPersonnalise ?? this.sonPersonnalise,
patternVibration: patternVibration ?? this.patternVibration,
couleurLED: couleurLED ?? this.couleurLED,
sonActive: sonActive ?? this.sonActive,
vibrationActive: vibrationActive ?? this.vibrationActive,
ledActive: ledActive ?? this.ledActive,
peutEtreDesactive: peutEtreDesactive ?? this.peutEtreDesactive,
);
}
}
/// Entité principale des préférences de notification
@JsonSerializable()
class PreferencesNotificationEntity extends Equatable {
const PreferencesNotificationEntity({
required this.id,
required this.utilisateurId,
this.organisationId,
this.notificationsActivees = true,
this.pushActivees = true,
this.emailActivees = true,
this.smsActivees = false,
this.inAppActivees = true,
this.typesActives,
this.typesDesactivees,
this.canauxActifs,
this.canauxDesactives,
this.modeSilencieux = false,
this.heureDebutSilencieux,
this.heureFinSilencieux,
this.joursSilencieux,
this.urgentesIgnorentSilencieux = true,
this.frequenceRegroupementMinutes = 5,
this.maxNotificationsSimultanees = 10,
this.dureeAffichageSecondes = 10,
this.vibrationActivee = true,
this.sonActive = true,
this.ledActivee = true,
this.sonPersonnalise,
this.patternVibrationPersonnalise,
this.couleurLEDPersonnalisee,
this.apercuEcranVerrouillage = true,
this.affichageHistorique = true,
this.dureeConservationJours = 30,
this.marquageLectureAutomatique = false,
this.delaiMarquageLectureSecondes,
this.archivageAutomatique = true,
this.delaiArchivageHeures = 168,
this.preferencesParType,
this.preferencesParCanal,
this.motsClesFiltre,
this.expediteursBloques,
this.expediteursPrioritaires,
this.notificationsTestActivees = false,
this.niveauLog = 'INFO',
this.tokenFCM,
this.plateforme,
this.versionApp,
this.langue = 'fr',
this.fuseauHoraire,
this.metadonnees,
});
final String id;
final String utilisateurId;
final String? organisationId;
final bool notificationsActivees;
final bool pushActivees;
final bool emailActivees;
final bool smsActivees;
final bool inAppActivees;
final Set<TypeNotification>? typesActives;
final Set<TypeNotification>? typesDesactivees;
final Set<CanalNotification>? canauxActifs;
final Set<CanalNotification>? canauxDesactives;
final bool modeSilencieux;
final String? heureDebutSilencieux; // Format HH:mm
final String? heureFinSilencieux; // Format HH:mm
final Set<int>? joursSilencieux; // 1=Lundi, 7=Dimanche
final bool urgentesIgnorentSilencieux;
final int frequenceRegroupementMinutes;
final int maxNotificationsSimultanees;
final int dureeAffichageSecondes;
final bool vibrationActivee;
final bool sonActive;
final bool ledActivee;
final String? sonPersonnalise;
final List<int>? patternVibrationPersonnalise;
final String? couleurLEDPersonnalisee;
final bool apercuEcranVerrouillage;
final bool affichageHistorique;
final int dureeConservationJours;
final bool marquageLectureAutomatique;
final int? delaiMarquageLectureSecondes;
final bool archivageAutomatique;
final int delaiArchivageHeures;
final Map<TypeNotification, PreferenceTypeNotification>? preferencesParType;
final Map<CanalNotification, PreferenceCanalNotification>? preferencesParCanal;
final Set<String>? motsClesFiltre;
final Set<String>? expediteursBloques;
final Set<String>? expediteursPrioritaires;
final bool notificationsTestActivees;
final String niveauLog;
final String? tokenFCM;
final String? plateforme;
final String? versionApp;
final String langue;
final String? fuseauHoraire;
final Map<String, dynamic>? metadonnees;
factory PreferencesNotificationEntity.fromJson(Map<String, dynamic> json) =>
_$PreferencesNotificationEntityFromJson(json);
Map<String, dynamic> toJson() => _$PreferencesNotificationEntityToJson(this);
@override
List<Object?> get props => [
id, utilisateurId, organisationId, notificationsActivees, pushActivees,
emailActivees, smsActivees, inAppActivees, typesActives, typesDesactivees,
canauxActifs, canauxDesactives, modeSilencieux, heureDebutSilencieux,
heureFinSilencieux, joursSilencieux, urgentesIgnorentSilencieux,
frequenceRegroupementMinutes, maxNotificationsSimultanees,
dureeAffichageSecondes, vibrationActivee, sonActive, ledActivee,
sonPersonnalise, patternVibrationPersonnalise, couleurLEDPersonnalisee,
apercuEcranVerrouillage, affichageHistorique, dureeConservationJours,
marquageLectureAutomatique, delaiMarquageLectureSecondes,
archivageAutomatique, delaiArchivageHeures, preferencesParType,
preferencesParCanal, motsClesFiltre, expediteursBloques,
expediteursPrioritaires, notificationsTestActivees, niveauLog,
tokenFCM, plateforme, versionApp, langue, fuseauHoraire, metadonnees,
];
PreferencesNotificationEntity copyWith({
String? id,
String? utilisateurId,
String? organisationId,
bool? notificationsActivees,
bool? pushActivees,
bool? emailActivees,
bool? smsActivees,
bool? inAppActivees,
Set<TypeNotification>? typesActives,
Set<TypeNotification>? typesDesactivees,
Set<CanalNotification>? canauxActifs,
Set<CanalNotification>? canauxDesactives,
bool? modeSilencieux,
String? heureDebutSilencieux,
String? heureFinSilencieux,
Set<int>? joursSilencieux,
bool? urgentesIgnorentSilencieux,
int? frequenceRegroupementMinutes,
int? maxNotificationsSimultanees,
int? dureeAffichageSecondes,
bool? vibrationActivee,
bool? sonActive,
bool? ledActivee,
String? sonPersonnalise,
List<int>? patternVibrationPersonnalise,
String? couleurLEDPersonnalisee,
bool? apercuEcranVerrouillage,
bool? affichageHistorique,
int? dureeConservationJours,
bool? marquageLectureAutomatique,
int? delaiMarquageLectureSecondes,
bool? archivageAutomatique,
int? delaiArchivageHeures,
Map<TypeNotification, PreferenceTypeNotification>? preferencesParType,
Map<CanalNotification, PreferenceCanalNotification>? preferencesParCanal,
Set<String>? motsClesFiltre,
Set<String>? expediteursBloques,
Set<String>? expediteursPrioritaires,
bool? notificationsTestActivees,
String? niveauLog,
String? tokenFCM,
String? plateforme,
String? versionApp,
String? langue,
String? fuseauHoraire,
Map<String, dynamic>? metadonnees,
}) {
return PreferencesNotificationEntity(
id: id ?? this.id,
utilisateurId: utilisateurId ?? this.utilisateurId,
organisationId: organisationId ?? this.organisationId,
notificationsActivees: notificationsActivees ?? this.notificationsActivees,
pushActivees: pushActivees ?? this.pushActivees,
emailActivees: emailActivees ?? this.emailActivees,
smsActivees: smsActivees ?? this.smsActivees,
inAppActivees: inAppActivees ?? this.inAppActivees,
typesActives: typesActives ?? this.typesActives,
typesDesactivees: typesDesactivees ?? this.typesDesactivees,
canauxActifs: canauxActifs ?? this.canauxActifs,
canauxDesactives: canauxDesactives ?? this.canauxDesactives,
modeSilencieux: modeSilencieux ?? this.modeSilencieux,
heureDebutSilencieux: heureDebutSilencieux ?? this.heureDebutSilencieux,
heureFinSilencieux: heureFinSilencieux ?? this.heureFinSilencieux,
joursSilencieux: joursSilencieux ?? this.joursSilencieux,
urgentesIgnorentSilencieux: urgentesIgnorentSilencieux ?? this.urgentesIgnorentSilencieux,
frequenceRegroupementMinutes: frequenceRegroupementMinutes ?? this.frequenceRegroupementMinutes,
maxNotificationsSimultanees: maxNotificationsSimultanees ?? this.maxNotificationsSimultanees,
dureeAffichageSecondes: dureeAffichageSecondes ?? this.dureeAffichageSecondes,
vibrationActivee: vibrationActivee ?? this.vibrationActivee,
sonActive: sonActive ?? this.sonActive,
ledActivee: ledActivee ?? this.ledActivee,
sonPersonnalise: sonPersonnalise ?? this.sonPersonnalise,
patternVibrationPersonnalise: patternVibrationPersonnalise ?? this.patternVibrationPersonnalise,
couleurLEDPersonnalisee: couleurLEDPersonnalisee ?? this.couleurLEDPersonnalisee,
apercuEcranVerrouillage: apercuEcranVerrouillage ?? this.apercuEcranVerrouillage,
affichageHistorique: affichageHistorique ?? this.affichageHistorique,
dureeConservationJours: dureeConservationJours ?? this.dureeConservationJours,
marquageLectureAutomatique: marquageLectureAutomatique ?? this.marquageLectureAutomatique,
delaiMarquageLectureSecondes: delaiMarquageLectureSecondes ?? this.delaiMarquageLectureSecondes,
archivageAutomatique: archivageAutomatique ?? this.archivageAutomatique,
delaiArchivageHeures: delaiArchivageHeures ?? this.delaiArchivageHeures,
preferencesParType: preferencesParType ?? this.preferencesParType,
preferencesParCanal: preferencesParCanal ?? this.preferencesParCanal,
motsClesFiltre: motsClesFiltre ?? this.motsClesFiltre,
expediteursBloques: expediteursBloques ?? this.expediteursBloques,
expediteursPrioritaires: expediteursPrioritaires ?? this.expediteursPrioritaires,
notificationsTestActivees: notificationsTestActivees ?? this.notificationsTestActivees,
niveauLog: niveauLog ?? this.niveauLog,
tokenFCM: tokenFCM ?? this.tokenFCM,
plateforme: plateforme ?? this.plateforme,
versionApp: versionApp ?? this.versionApp,
langue: langue ?? this.langue,
fuseauHoraire: fuseauHoraire ?? this.fuseauHoraire,
metadonnees: metadonnees ?? this.metadonnees,
);
}
/// Vérifie si un type de notification est activé
bool isTypeActive(TypeNotification type) {
if (!notificationsActivees) return false;
if (typesDesactivees?.contains(type) == true) return false;
if (typesActives != null) return typesActives!.contains(type);
return true; // Activé par défaut
}
/// Vérifie si un canal de notification est activé
bool isCanalActif(CanalNotification canal) {
if (!notificationsActivees) return false;
if (canauxDesactives?.contains(canal) == true) return false;
if (canauxActifs != null) return canauxActifs!.contains(canal);
return true; // Activé par défaut
}
/// Vérifie si on est en mode silencieux actuellement
bool get isEnModeSilencieux {
if (!modeSilencieux) return false;
if (heureDebutSilencieux == null || heureFinSilencieux == null) return false;
final maintenant = DateTime.now();
final heureActuelle = '${maintenant.hour.toString().padLeft(2, '0')}:${maintenant.minute.toString().padLeft(2, '0')}';
// Gestion du cas où la période traverse minuit
if (heureDebutSilencieux!.compareTo(heureFinSilencieux!) > 0) {
return heureActuelle.compareTo(heureDebutSilencieux!) >= 0 ||
heureActuelle.compareTo(heureFinSilencieux!) <= 0;
} else {
return heureActuelle.compareTo(heureDebutSilencieux!) >= 0 &&
heureActuelle.compareTo(heureFinSilencieux!) <= 0;
}
}
/// Vérifie si un expéditeur est bloqué
bool isExpediteurBloque(String? expediteurId) {
if (expediteurId == null) return false;
return expediteursBloques?.contains(expediteurId) == true;
}
/// Vérifie si un expéditeur est prioritaire
bool isExpediteurPrioritaire(String? expediteurId) {
if (expediteurId == null) return false;
return expediteursPrioritaires?.contains(expediteurId) == true;
}
/// Crée des préférences par défaut pour un utilisateur
static PreferencesNotificationEntity creerDefaut(String utilisateurId) {
return PreferencesNotificationEntity(
id: 'pref_$utilisateurId',
utilisateurId: utilisateurId,
);
}
}

View File

@@ -0,0 +1,310 @@
import 'package:dartz/dartz.dart';
import '../../../../core/error/failures.dart';
import '../entities/notification.dart';
import '../entities/preferences_notification.dart';
/// Repository abstrait pour la gestion des notifications
abstract class NotificationsRepository {
// === GESTION DES NOTIFICATIONS ===
/// Récupère les notifications d'un utilisateur
///
/// [utilisateurId] ID de l'utilisateur
/// [includeArchivees] Inclure les notifications archivées
/// [limite] Nombre maximum de notifications à retourner
/// [offset] Décalage pour la pagination
Future<Either<Failure, List<NotificationEntity>>> obtenirNotifications({
required String utilisateurId,
bool includeArchivees = false,
int limite = 50,
int offset = 0,
});
/// Récupère une notification spécifique
///
/// [notificationId] ID de la notification
Future<Either<Failure, NotificationEntity>> obtenirNotification(String notificationId);
/// Marque une notification comme lue
///
/// [notificationId] ID de la notification
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> marquerCommeLue(String notificationId, String utilisateurId);
/// Marque toutes les notifications comme lues
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> marquerToutesCommeLues(String utilisateurId);
/// Marque une notification comme importante
///
/// [notificationId] ID de la notification
/// [utilisateurId] ID de l'utilisateur
/// [importante] true pour marquer comme importante, false pour enlever
Future<Either<Failure, void>> marquerCommeImportante(
String notificationId,
String utilisateurId,
bool importante,
);
/// Archive une notification
///
/// [notificationId] ID de la notification
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> archiverNotification(String notificationId, String utilisateurId);
/// Archive toutes les notifications lues
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> archiverToutesLues(String utilisateurId);
/// Supprime une notification
///
/// [notificationId] ID de la notification
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> supprimerNotification(String notificationId, String utilisateurId);
/// Supprime toutes les notifications archivées
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> supprimerToutesArchivees(String utilisateurId);
// === FILTRAGE ET RECHERCHE ===
/// Recherche des notifications par critères
///
/// [utilisateurId] ID de l'utilisateur
/// [query] Texte de recherche
/// [types] Types de notifications à inclure
/// [statuts] Statuts de notifications à inclure
/// [dateDebut] Date de début de la période
/// [dateFin] Date de fin de la période
/// [limite] Nombre maximum de résultats
Future<Either<Failure, List<NotificationEntity>>> rechercherNotifications({
required String utilisateurId,
String? query,
List<TypeNotification>? types,
List<StatutNotification>? statuts,
DateTime? dateDebut,
DateTime? dateFin,
int limite = 50,
});
/// Récupère les notifications par type
///
/// [utilisateurId] ID de l'utilisateur
/// [type] Type de notification
/// [limite] Nombre maximum de notifications
Future<Either<Failure, List<NotificationEntity>>> obtenirNotificationsParType(
String utilisateurId,
TypeNotification type, {
int limite = 50,
});
/// Récupère les notifications non lues
///
/// [utilisateurId] ID de l'utilisateur
/// [limite] Nombre maximum de notifications
Future<Either<Failure, List<NotificationEntity>>> obtenirNotificationsNonLues(
String utilisateurId, {
int limite = 50,
});
/// Récupère les notifications importantes
///
/// [utilisateurId] ID de l'utilisateur
/// [limite] Nombre maximum de notifications
Future<Either<Failure, List<NotificationEntity>>> obtenirNotificationsImportantes(
String utilisateurId, {
int limite = 50,
});
// === STATISTIQUES ===
/// Récupère le nombre de notifications non lues
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, int>> obtenirNombreNonLues(String utilisateurId);
/// Récupère les statistiques des notifications
///
/// [utilisateurId] ID de l'utilisateur
/// [periode] Période d'analyse (en jours)
Future<Either<Failure, Map<String, dynamic>>> obtenirStatistiques(
String utilisateurId, {
int periode = 30,
});
// === ACTIONS SUR LES NOTIFICATIONS ===
/// Exécute une action rapide sur une notification
///
/// [notificationId] ID de la notification
/// [actionId] ID de l'action à exécuter
/// [utilisateurId] ID de l'utilisateur
/// [parametres] Paramètres additionnels pour l'action
Future<Either<Failure, Map<String, dynamic>>> executerActionRapide(
String notificationId,
String actionId,
String utilisateurId, {
Map<String, dynamic>? parametres,
});
/// Signale une notification comme spam
///
/// [notificationId] ID de la notification
/// [utilisateurId] ID de l'utilisateur
/// [raison] Raison du signalement
Future<Either<Failure, void>> signalerSpam(
String notificationId,
String utilisateurId,
String raison,
);
// === PRÉFÉRENCES DE NOTIFICATION ===
/// Récupère les préférences de notification d'un utilisateur
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, PreferencesNotificationEntity>> obtenirPreferences(String utilisateurId);
/// Met à jour les préférences de notification
///
/// [preferences] Nouvelles préférences
Future<Either<Failure, void>> mettreAJourPreferences(PreferencesNotificationEntity preferences);
/// Réinitialise les préférences aux valeurs par défaut
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, PreferencesNotificationEntity>> reinitialiserPreferences(String utilisateurId);
/// Active/désactive un type de notification
///
/// [utilisateurId] ID de l'utilisateur
/// [type] Type de notification
/// [active] true pour activer, false pour désactiver
Future<Either<Failure, void>> toggleTypeNotification(
String utilisateurId,
TypeNotification type,
bool active,
);
/// Active/désactive un canal de notification
///
/// [utilisateurId] ID de l'utilisateur
/// [canal] Canal de notification
/// [active] true pour activer, false pour désactiver
Future<Either<Failure, void>> toggleCanalNotification(
String utilisateurId,
CanalNotification canal,
bool active,
);
/// Configure le mode silencieux
///
/// [utilisateurId] ID de l'utilisateur
/// [active] true pour activer le mode silencieux
/// [heureDebut] Heure de début (format HH:mm)
/// [heureFin] Heure de fin (format HH:mm)
/// [jours] Jours de la semaine (1=Lundi, 7=Dimanche)
Future<Either<Failure, void>> configurerModeSilencieux(
String utilisateurId,
bool active, {
String? heureDebut,
String? heureFin,
Set<int>? jours,
});
// === GESTION DES TOKENS FCM ===
/// Enregistre ou met à jour le token FCM
///
/// [utilisateurId] ID de l'utilisateur
/// [token] Token FCM
/// [plateforme] Plateforme (android, ios)
Future<Either<Failure, void>> enregistrerTokenFCM(
String utilisateurId,
String token,
String plateforme,
);
/// Supprime le token FCM
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> supprimerTokenFCM(String utilisateurId);
// === NOTIFICATIONS DE TEST ===
/// Envoie une notification de test
///
/// [utilisateurId] ID de l'utilisateur
/// [type] Type de notification à tester
Future<Either<Failure, NotificationEntity>> envoyerNotificationTest(
String utilisateurId,
TypeNotification type,
);
// === CACHE ET SYNCHRONISATION ===
/// Synchronise les notifications avec le serveur
///
/// [utilisateurId] ID de l'utilisateur
/// [forceSync] Force la synchronisation même si le cache est récent
Future<Either<Failure, void>> synchroniser(String utilisateurId, {bool forceSync = false});
/// Vide le cache des notifications
///
/// [utilisateurId] ID de l'utilisateur (optionnel, vide tout si null)
Future<Either<Failure, void>> viderCache([String? utilisateurId]);
/// Vérifie si les données sont en cache et récentes
///
/// [utilisateurId] ID de l'utilisateur
/// [maxAgeMinutes] Âge maximum du cache en minutes
Future<bool> isCacheValide(String utilisateurId, {int maxAgeMinutes = 5});
// === ABONNEMENTS ET TOPICS ===
/// S'abonne à un topic de notifications
///
/// [utilisateurId] ID de l'utilisateur
/// [topic] Nom du topic
Future<Either<Failure, void>> abonnerAuTopic(String utilisateurId, String topic);
/// Se désabonne d'un topic de notifications
///
/// [utilisateurId] ID de l'utilisateur
/// [topic] Nom du topic
Future<Either<Failure, void>> desabonnerDuTopic(String utilisateurId, String topic);
/// Récupère la liste des topics auxquels l'utilisateur est abonné
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, List<String>>> obtenirTopicsAbornes(String utilisateurId);
// === EXPORT ET SAUVEGARDE ===
/// Exporte les notifications vers un fichier
///
/// [utilisateurId] ID de l'utilisateur
/// [format] Format d'export (json, csv)
/// [dateDebut] Date de début de la période
/// [dateFin] Date de fin de la période
Future<Either<Failure, String>> exporterNotifications(
String utilisateurId,
String format, {
DateTime? dateDebut,
DateTime? dateFin,
});
/// Sauvegarde les notifications localement
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> sauvegarderLocalement(String utilisateurId);
/// Restaure les notifications depuis une sauvegarde locale
///
/// [utilisateurId] ID de l'utilisateur
Future<Either<Failure, void>> restaurerDepuisSauvegarde(String utilisateurId);
}

View File

@@ -0,0 +1,388 @@
import 'package:dartz/dartz.dart';
import '../../../../core/error/failures.dart';
import '../../../../core/usecases/usecase.dart';
import '../entities/notification.dart';
import '../repositories/notifications_repository.dart';
/// Use case pour marquer une notification comme lue
class MarquerCommeLueUseCase implements UseCase<void, MarquerCommeLueParams> {
final NotificationsRepository repository;
MarquerCommeLueUseCase(this.repository);
@override
Future<Either<Failure, void>> call(MarquerCommeLueParams params) async {
return await repository.marquerCommeLue(
params.notificationId,
params.utilisateurId,
);
}
}
/// Paramètres pour marquer comme lue
class MarquerCommeLueParams {
final String notificationId;
final String utilisateurId;
const MarquerCommeLueParams({
required this.notificationId,
required this.utilisateurId,
});
@override
String toString() {
return 'MarquerCommeLueParams{notificationId: $notificationId, utilisateurId: $utilisateurId}';
}
}
/// Use case pour marquer toutes les notifications comme lues
class MarquerToutesCommeLuesUseCase implements UseCase<void, String> {
final NotificationsRepository repository;
MarquerToutesCommeLuesUseCase(this.repository);
@override
Future<Either<Failure, void>> call(String utilisateurId) async {
return await repository.marquerToutesCommeLues(utilisateurId);
}
}
/// Use case pour marquer une notification comme importante
class MarquerCommeImportanteUseCase implements UseCase<void, MarquerCommeImportanteParams> {
final NotificationsRepository repository;
MarquerCommeImportanteUseCase(this.repository);
@override
Future<Either<Failure, void>> call(MarquerCommeImportanteParams params) async {
return await repository.marquerCommeImportante(
params.notificationId,
params.utilisateurId,
params.importante,
);
}
}
/// Paramètres pour marquer comme importante
class MarquerCommeImportanteParams {
final String notificationId;
final String utilisateurId;
final bool importante;
const MarquerCommeImportanteParams({
required this.notificationId,
required this.utilisateurId,
required this.importante,
});
@override
String toString() {
return 'MarquerCommeImportanteParams{notificationId: $notificationId, utilisateurId: $utilisateurId, importante: $importante}';
}
}
/// Use case pour archiver une notification
class ArchiverNotificationUseCase implements UseCase<void, ArchiverNotificationParams> {
final NotificationsRepository repository;
ArchiverNotificationUseCase(this.repository);
@override
Future<Either<Failure, void>> call(ArchiverNotificationParams params) async {
return await repository.archiverNotification(
params.notificationId,
params.utilisateurId,
);
}
}
/// Paramètres pour archiver une notification
class ArchiverNotificationParams {
final String notificationId;
final String utilisateurId;
const ArchiverNotificationParams({
required this.notificationId,
required this.utilisateurId,
});
@override
String toString() {
return 'ArchiverNotificationParams{notificationId: $notificationId, utilisateurId: $utilisateurId}';
}
}
/// Use case pour archiver toutes les notifications lues
class ArchiverToutesLuesUseCase implements UseCase<void, String> {
final NotificationsRepository repository;
ArchiverToutesLuesUseCase(this.repository);
@override
Future<Either<Failure, void>> call(String utilisateurId) async {
return await repository.archiverToutesLues(utilisateurId);
}
}
/// Use case pour supprimer une notification
class SupprimerNotificationUseCase implements UseCase<void, SupprimerNotificationParams> {
final NotificationsRepository repository;
SupprimerNotificationUseCase(this.repository);
@override
Future<Either<Failure, void>> call(SupprimerNotificationParams params) async {
return await repository.supprimerNotification(
params.notificationId,
params.utilisateurId,
);
}
}
/// Paramètres pour supprimer une notification
class SupprimerNotificationParams {
final String notificationId;
final String utilisateurId;
const SupprimerNotificationParams({
required this.notificationId,
required this.utilisateurId,
});
@override
String toString() {
return 'SupprimerNotificationParams{notificationId: $notificationId, utilisateurId: $utilisateurId}';
}
}
/// Use case pour supprimer toutes les notifications archivées
class SupprimerToutesArchiveesUseCase implements UseCase<void, String> {
final NotificationsRepository repository;
SupprimerToutesArchiveesUseCase(this.repository);
@override
Future<Either<Failure, void>> call(String utilisateurId) async {
return await repository.supprimerToutesArchivees(utilisateurId);
}
}
/// Use case pour exécuter une action rapide
class ExecuterActionRapideUseCase implements UseCase<Map<String, dynamic>, ExecuterActionRapideParams> {
final NotificationsRepository repository;
ExecuterActionRapideUseCase(this.repository);
@override
Future<Either<Failure, Map<String, dynamic>>> call(ExecuterActionRapideParams params) async {
return await repository.executerActionRapide(
params.notificationId,
params.actionId,
params.utilisateurId,
parametres: params.parametres,
);
}
}
/// Paramètres pour exécuter une action rapide
class ExecuterActionRapideParams {
final String notificationId;
final String actionId;
final String utilisateurId;
final Map<String, dynamic>? parametres;
const ExecuterActionRapideParams({
required this.notificationId,
required this.actionId,
required this.utilisateurId,
this.parametres,
});
ExecuterActionRapideParams copyWith({
String? notificationId,
String? actionId,
String? utilisateurId,
Map<String, dynamic>? parametres,
}) {
return ExecuterActionRapideParams(
notificationId: notificationId ?? this.notificationId,
actionId: actionId ?? this.actionId,
utilisateurId: utilisateurId ?? this.utilisateurId,
parametres: parametres ?? this.parametres,
);
}
@override
String toString() {
return 'ExecuterActionRapideParams{notificationId: $notificationId, actionId: $actionId, utilisateurId: $utilisateurId, parametres: $parametres}';
}
}
/// Use case pour signaler une notification comme spam
class SignalerSpamUseCase implements UseCase<void, SignalerSpamParams> {
final NotificationsRepository repository;
SignalerSpamUseCase(this.repository);
@override
Future<Either<Failure, void>> call(SignalerSpamParams params) async {
return await repository.signalerSpam(
params.notificationId,
params.utilisateurId,
params.raison,
);
}
}
/// Paramètres pour signaler comme spam
class SignalerSpamParams {
final String notificationId;
final String utilisateurId;
final String raison;
const SignalerSpamParams({
required this.notificationId,
required this.utilisateurId,
required this.raison,
});
@override
String toString() {
return 'SignalerSpamParams{notificationId: $notificationId, utilisateurId: $utilisateurId, raison: $raison}';
}
}
/// Use case pour synchroniser les notifications
class SynchroniserNotificationsUseCase implements UseCase<void, SynchroniserNotificationsParams> {
final NotificationsRepository repository;
SynchroniserNotificationsUseCase(this.repository);
@override
Future<Either<Failure, void>> call(SynchroniserNotificationsParams params) async {
return await repository.synchroniser(
params.utilisateurId,
forceSync: params.forceSync,
);
}
}
/// Paramètres pour synchroniser les notifications
class SynchroniserNotificationsParams {
final String utilisateurId;
final bool forceSync;
const SynchroniserNotificationsParams({
required this.utilisateurId,
this.forceSync = false,
});
SynchroniserNotificationsParams copyWith({
String? utilisateurId,
bool? forceSync,
}) {
return SynchroniserNotificationsParams(
utilisateurId: utilisateurId ?? this.utilisateurId,
forceSync: forceSync ?? this.forceSync,
);
}
@override
String toString() {
return 'SynchroniserNotificationsParams{utilisateurId: $utilisateurId, forceSync: $forceSync}';
}
}
/// Use case pour vider le cache des notifications
class ViderCacheNotificationsUseCase implements UseCase<void, String?> {
final NotificationsRepository repository;
ViderCacheNotificationsUseCase(this.repository);
@override
Future<Either<Failure, void>> call(String? utilisateurId) async {
return await repository.viderCache(utilisateurId);
}
}
/// Use case pour envoyer une notification de test
class EnvoyerNotificationTestUseCase implements UseCase<NotificationEntity, EnvoyerNotificationTestParams> {
final NotificationsRepository repository;
EnvoyerNotificationTestUseCase(this.repository);
@override
Future<Either<Failure, NotificationEntity>> call(EnvoyerNotificationTestParams params) async {
return await repository.envoyerNotificationTest(
params.utilisateurId,
params.type,
);
}
}
/// Paramètres pour envoyer une notification de test
class EnvoyerNotificationTestParams {
final String utilisateurId;
final TypeNotification type;
const EnvoyerNotificationTestParams({
required this.utilisateurId,
required this.type,
});
@override
String toString() {
return 'EnvoyerNotificationTestParams{utilisateurId: $utilisateurId, type: $type}';
}
}
/// Use case pour exporter les notifications
class ExporterNotificationsUseCase implements UseCase<String, ExporterNotificationsParams> {
final NotificationsRepository repository;
ExporterNotificationsUseCase(this.repository);
@override
Future<Either<Failure, String>> call(ExporterNotificationsParams params) async {
return await repository.exporterNotifications(
params.utilisateurId,
params.format,
dateDebut: params.dateDebut,
dateFin: params.dateFin,
);
}
}
/// Paramètres pour exporter les notifications
class ExporterNotificationsParams {
final String utilisateurId;
final String format;
final DateTime? dateDebut;
final DateTime? dateFin;
const ExporterNotificationsParams({
required this.utilisateurId,
required this.format,
this.dateDebut,
this.dateFin,
});
ExporterNotificationsParams copyWith({
String? utilisateurId,
String? format,
DateTime? dateDebut,
DateTime? dateFin,
}) {
return ExporterNotificationsParams(
utilisateurId: utilisateurId ?? this.utilisateurId,
format: format ?? this.format,
dateDebut: dateDebut ?? this.dateDebut,
dateFin: dateFin ?? this.dateFin,
);
}
@override
String toString() {
return 'ExporterNotificationsParams{utilisateurId: $utilisateurId, format: $format, dateDebut: $dateDebut, dateFin: $dateFin}';
}
}

View File

@@ -0,0 +1,369 @@
import 'package:dartz/dartz.dart';
import '../../../../core/error/failures.dart';
import '../../../../core/usecases/usecase.dart';
import '../entities/notification.dart';
import '../entities/preferences_notification.dart';
import '../repositories/notifications_repository.dart';
/// Use case pour obtenir les préférences de notification
class ObtenirPreferencesUseCase implements UseCase<PreferencesNotificationEntity, String> {
final NotificationsRepository repository;
ObtenirPreferencesUseCase(this.repository);
@override
Future<Either<Failure, PreferencesNotificationEntity>> call(String utilisateurId) async {
return await repository.obtenirPreferences(utilisateurId);
}
}
/// Use case pour mettre à jour les préférences de notification
class MettreAJourPreferencesUseCase implements UseCase<void, PreferencesNotificationEntity> {
final NotificationsRepository repository;
MettreAJourPreferencesUseCase(this.repository);
@override
Future<Either<Failure, void>> call(PreferencesNotificationEntity preferences) async {
return await repository.mettreAJourPreferences(preferences);
}
}
/// Use case pour réinitialiser les préférences
class ReinitialiserPreferencesUseCase implements UseCase<PreferencesNotificationEntity, String> {
final NotificationsRepository repository;
ReinitialiserPreferencesUseCase(this.repository);
@override
Future<Either<Failure, PreferencesNotificationEntity>> call(String utilisateurId) async {
return await repository.reinitialiserPreferences(utilisateurId);
}
}
/// Use case pour activer/désactiver un type de notification
class ToggleTypeNotificationUseCase implements UseCase<void, ToggleTypeNotificationParams> {
final NotificationsRepository repository;
ToggleTypeNotificationUseCase(this.repository);
@override
Future<Either<Failure, void>> call(ToggleTypeNotificationParams params) async {
return await repository.toggleTypeNotification(
params.utilisateurId,
params.type,
params.active,
);
}
}
/// Paramètres pour activer/désactiver un type de notification
class ToggleTypeNotificationParams {
final String utilisateurId;
final TypeNotification type;
final bool active;
const ToggleTypeNotificationParams({
required this.utilisateurId,
required this.type,
required this.active,
});
@override
String toString() {
return 'ToggleTypeNotificationParams{utilisateurId: $utilisateurId, type: $type, active: $active}';
}
}
/// Use case pour activer/désactiver un canal de notification
class ToggleCanalNotificationUseCase implements UseCase<void, ToggleCanalNotificationParams> {
final NotificationsRepository repository;
ToggleCanalNotificationUseCase(this.repository);
@override
Future<Either<Failure, void>> call(ToggleCanalNotificationParams params) async {
return await repository.toggleCanalNotification(
params.utilisateurId,
params.canal,
params.active,
);
}
}
/// Paramètres pour activer/désactiver un canal de notification
class ToggleCanalNotificationParams {
final String utilisateurId;
final CanalNotification canal;
final bool active;
const ToggleCanalNotificationParams({
required this.utilisateurId,
required this.canal,
required this.active,
});
@override
String toString() {
return 'ToggleCanalNotificationParams{utilisateurId: $utilisateurId, canal: $canal, active: $active}';
}
}
/// Use case pour configurer le mode silencieux
class ConfigurerModeSilencieuxUseCase implements UseCase<void, ConfigurerModeSilencieuxParams> {
final NotificationsRepository repository;
ConfigurerModeSilencieuxUseCase(this.repository);
@override
Future<Either<Failure, void>> call(ConfigurerModeSilencieuxParams params) async {
return await repository.configurerModeSilencieux(
params.utilisateurId,
params.active,
heureDebut: params.heureDebut,
heureFin: params.heureFin,
jours: params.jours,
);
}
}
/// Paramètres pour configurer le mode silencieux
class ConfigurerModeSilencieuxParams {
final String utilisateurId;
final bool active;
final String? heureDebut;
final String? heureFin;
final Set<int>? jours;
const ConfigurerModeSilencieuxParams({
required this.utilisateurId,
required this.active,
this.heureDebut,
this.heureFin,
this.jours,
});
ConfigurerModeSilencieuxParams copyWith({
String? utilisateurId,
bool? active,
String? heureDebut,
String? heureFin,
Set<int>? jours,
}) {
return ConfigurerModeSilencieuxParams(
utilisateurId: utilisateurId ?? this.utilisateurId,
active: active ?? this.active,
heureDebut: heureDebut ?? this.heureDebut,
heureFin: heureFin ?? this.heureFin,
jours: jours ?? this.jours,
);
}
@override
String toString() {
return 'ConfigurerModeSilencieuxParams{utilisateurId: $utilisateurId, active: $active, heureDebut: $heureDebut, heureFin: $heureFin, jours: $jours}';
}
}
/// Use case pour enregistrer le token FCM
class EnregistrerTokenFCMUseCase implements UseCase<void, EnregistrerTokenFCMParams> {
final NotificationsRepository repository;
EnregistrerTokenFCMUseCase(this.repository);
@override
Future<Either<Failure, void>> call(EnregistrerTokenFCMParams params) async {
return await repository.enregistrerTokenFCM(
params.utilisateurId,
params.token,
params.plateforme,
);
}
}
/// Paramètres pour enregistrer le token FCM
class EnregistrerTokenFCMParams {
final String utilisateurId;
final String token;
final String plateforme;
const EnregistrerTokenFCMParams({
required this.utilisateurId,
required this.token,
required this.plateforme,
});
@override
String toString() {
return 'EnregistrerTokenFCMParams{utilisateurId: $utilisateurId, token: $token, plateforme: $plateforme}';
}
}
/// Use case pour supprimer le token FCM
class SupprimerTokenFCMUseCase implements UseCase<void, String> {
final NotificationsRepository repository;
SupprimerTokenFCMUseCase(this.repository);
@override
Future<Either<Failure, void>> call(String utilisateurId) async {
return await repository.supprimerTokenFCM(utilisateurId);
}
}
/// Use case pour s'abonner à un topic
class AbonnerAuTopicUseCase implements UseCase<void, AbonnerAuTopicParams> {
final NotificationsRepository repository;
AbonnerAuTopicUseCase(this.repository);
@override
Future<Either<Failure, void>> call(AbonnerAuTopicParams params) async {
return await repository.abonnerAuTopic(
params.utilisateurId,
params.topic,
);
}
}
/// Paramètres pour s'abonner à un topic
class AbonnerAuTopicParams {
final String utilisateurId;
final String topic;
const AbonnerAuTopicParams({
required this.utilisateurId,
required this.topic,
});
@override
String toString() {
return 'AbonnerAuTopicParams{utilisateurId: $utilisateurId, topic: $topic}';
}
}
/// Use case pour se désabonner d'un topic
class DesabonnerDuTopicUseCase implements UseCase<void, DesabonnerDuTopicParams> {
final NotificationsRepository repository;
DesabonnerDuTopicUseCase(this.repository);
@override
Future<Either<Failure, void>> call(DesabonnerDuTopicParams params) async {
return await repository.desabonnerDuTopic(
params.utilisateurId,
params.topic,
);
}
}
/// Paramètres pour se désabonner d'un topic
class DesabonnerDuTopicParams {
final String utilisateurId;
final String topic;
const DesabonnerDuTopicParams({
required this.utilisateurId,
required this.topic,
});
@override
String toString() {
return 'DesabonnerDuTopicParams{utilisateurId: $utilisateurId, topic: $topic}';
}
}
/// Use case pour obtenir les topics auxquels l'utilisateur est abonné
class ObtenirTopicsAbornesUseCase implements UseCase<List<String>, String> {
final NotificationsRepository repository;
ObtenirTopicsAbornesUseCase(this.repository);
@override
Future<Either<Failure, List<String>>> call(String utilisateurId) async {
return await repository.obtenirTopicsAbornes(utilisateurId);
}
}
/// Use case pour configurer les préférences avancées
class ConfigurerPreferencesAvanceesUseCase implements UseCase<void, ConfigurerPreferencesAvanceesParams> {
final NotificationsRepository repository;
ConfigurerPreferencesAvanceesUseCase(this.repository);
@override
Future<Either<Failure, void>> call(ConfigurerPreferencesAvanceesParams params) async {
// Récupération des préférences actuelles
final preferencesResult = await repository.obtenirPreferences(params.utilisateurId);
return preferencesResult.fold(
(failure) => Left(failure),
(preferences) async {
// Mise à jour des préférences avec les nouveaux paramètres
final preferencesModifiees = preferences.copyWith(
vibrationActivee: params.vibrationActivee ?? preferences.vibrationActivee,
sonActive: params.sonActive ?? preferences.sonActive,
ledActivee: params.ledActivee ?? preferences.ledActivee,
sonPersonnalise: params.sonPersonnalise ?? preferences.sonPersonnalise,
patternVibrationPersonnalise: params.patternVibrationPersonnalise ?? preferences.patternVibrationPersonnalise,
couleurLEDPersonnalisee: params.couleurLEDPersonnalisee ?? preferences.couleurLEDPersonnalisee,
apercuEcranVerrouillage: params.apercuEcranVerrouillage ?? preferences.apercuEcranVerrouillage,
dureeAffichageSecondes: params.dureeAffichageSecondes ?? preferences.dureeAffichageSecondes,
frequenceRegroupementMinutes: params.frequenceRegroupementMinutes ?? preferences.frequenceRegroupementMinutes,
maxNotificationsSimultanees: params.maxNotificationsSimultanees ?? preferences.maxNotificationsSimultanees,
marquageLectureAutomatique: params.marquageLectureAutomatique ?? preferences.marquageLectureAutomatique,
delaiMarquageLectureSecondes: params.delaiMarquageLectureSecondes ?? preferences.delaiMarquageLectureSecondes,
archivageAutomatique: params.archivageAutomatique ?? preferences.archivageAutomatique,
delaiArchivageHeures: params.delaiArchivageHeures ?? preferences.delaiArchivageHeures,
dureeConservationJours: params.dureeConservationJours ?? preferences.dureeConservationJours,
);
return await repository.mettreAJourPreferences(preferencesModifiees);
},
);
}
}
/// Paramètres pour configurer les préférences avancées
class ConfigurerPreferencesAvanceesParams {
final String utilisateurId;
final bool? vibrationActivee;
final bool? sonActive;
final bool? ledActivee;
final String? sonPersonnalise;
final List<int>? patternVibrationPersonnalise;
final String? couleurLEDPersonnalisee;
final bool? apercuEcranVerrouillage;
final int? dureeAffichageSecondes;
final int? frequenceRegroupementMinutes;
final int? maxNotificationsSimultanees;
final bool? marquageLectureAutomatique;
final int? delaiMarquageLectureSecondes;
final bool? archivageAutomatique;
final int? delaiArchivageHeures;
final int? dureeConservationJours;
const ConfigurerPreferencesAvanceesParams({
required this.utilisateurId,
this.vibrationActivee,
this.sonActive,
this.ledActivee,
this.sonPersonnalise,
this.patternVibrationPersonnalise,
this.couleurLEDPersonnalisee,
this.apercuEcranVerrouillage,
this.dureeAffichageSecondes,
this.frequenceRegroupementMinutes,
this.maxNotificationsSimultanees,
this.marquageLectureAutomatique,
this.delaiMarquageLectureSecondes,
this.archivageAutomatique,
this.delaiArchivageHeures,
this.dureeConservationJours,
});
@override
String toString() {
return 'ConfigurerPreferencesAvanceesParams{utilisateurId: $utilisateurId, vibrationActivee: $vibrationActivee, sonActive: $sonActive, ledActivee: $ledActivee, ...}';
}
}

View File

@@ -0,0 +1,274 @@
import 'package:dartz/dartz.dart';
import '../../../../core/error/failures.dart';
import '../../../../core/usecases/usecase.dart';
import '../entities/notification.dart';
import '../repositories/notifications_repository.dart';
/// Use case pour obtenir les notifications d'un utilisateur
class ObtenirNotificationsUseCase implements UseCase<List<NotificationEntity>, ObtenirNotificationsParams> {
final NotificationsRepository repository;
ObtenirNotificationsUseCase(this.repository);
@override
Future<Either<Failure, List<NotificationEntity>>> call(ObtenirNotificationsParams params) async {
// Vérification du cache en premier
final cacheValide = await repository.isCacheValide(
params.utilisateurId,
maxAgeMinutes: params.maxAgeCacheMinutes,
);
if (!cacheValide || params.forceRefresh) {
// Synchronisation avec le serveur si nécessaire
final syncResult = await repository.synchroniser(
params.utilisateurId,
forceSync: params.forceRefresh,
);
// On continue même si la sync échoue (mode offline)
if (syncResult.isLeft()) {
// Log de l'erreur mais on continue avec les données en cache
print('Erreur de synchronisation: ${syncResult.fold((l) => l.toString(), (r) => '')}');
}
}
// Récupération des notifications
return await repository.obtenirNotifications(
utilisateurId: params.utilisateurId,
includeArchivees: params.includeArchivees,
limite: params.limite,
offset: params.offset,
);
}
}
/// Paramètres pour obtenir les notifications
class ObtenirNotificationsParams {
final String utilisateurId;
final bool includeArchivees;
final int limite;
final int offset;
final bool forceRefresh;
final int maxAgeCacheMinutes;
const ObtenirNotificationsParams({
required this.utilisateurId,
this.includeArchivees = false,
this.limite = 50,
this.offset = 0,
this.forceRefresh = false,
this.maxAgeCacheMinutes = 5,
});
ObtenirNotificationsParams copyWith({
String? utilisateurId,
bool? includeArchivees,
int? limite,
int? offset,
bool? forceRefresh,
int? maxAgeCacheMinutes,
}) {
return ObtenirNotificationsParams(
utilisateurId: utilisateurId ?? this.utilisateurId,
includeArchivees: includeArchivees ?? this.includeArchivees,
limite: limite ?? this.limite,
offset: offset ?? this.offset,
forceRefresh: forceRefresh ?? this.forceRefresh,
maxAgeCacheMinutes: maxAgeCacheMinutes ?? this.maxAgeCacheMinutes,
);
}
@override
String toString() {
return 'ObtenirNotificationsParams{utilisateurId: $utilisateurId, includeArchivees: $includeArchivees, limite: $limite, offset: $offset, forceRefresh: $forceRefresh}';
}
}
/// Use case pour obtenir les notifications non lues
class ObtenirNotificationsNonLuesUseCase implements UseCase<List<NotificationEntity>, String> {
final NotificationsRepository repository;
ObtenirNotificationsNonLuesUseCase(this.repository);
@override
Future<Either<Failure, List<NotificationEntity>>> call(String utilisateurId) async {
return await repository.obtenirNotificationsNonLues(utilisateurId);
}
}
/// Use case pour obtenir le nombre de notifications non lues
class ObtenirNombreNonLuesUseCase implements UseCase<int, String> {
final NotificationsRepository repository;
ObtenirNombreNonLuesUseCase(this.repository);
@override
Future<Either<Failure, int>> call(String utilisateurId) async {
return await repository.obtenirNombreNonLues(utilisateurId);
}
}
/// Use case pour rechercher des notifications
class RechercherNotificationsUseCase implements UseCase<List<NotificationEntity>, RechercherNotificationsParams> {
final NotificationsRepository repository;
RechercherNotificationsUseCase(this.repository);
@override
Future<Either<Failure, List<NotificationEntity>>> call(RechercherNotificationsParams params) async {
return await repository.rechercherNotifications(
utilisateurId: params.utilisateurId,
query: params.query,
types: params.types,
statuts: params.statuts,
dateDebut: params.dateDebut,
dateFin: params.dateFin,
limite: params.limite,
);
}
}
/// Paramètres pour la recherche de notifications
class RechercherNotificationsParams {
final String utilisateurId;
final String? query;
final List<TypeNotification>? types;
final List<StatutNotification>? statuts;
final DateTime? dateDebut;
final DateTime? dateFin;
final int limite;
const RechercherNotificationsParams({
required this.utilisateurId,
this.query,
this.types,
this.statuts,
this.dateDebut,
this.dateFin,
this.limite = 50,
});
RechercherNotificationsParams copyWith({
String? utilisateurId,
String? query,
List<TypeNotification>? types,
List<StatutNotification>? statuts,
DateTime? dateDebut,
DateTime? dateFin,
int? limite,
}) {
return RechercherNotificationsParams(
utilisateurId: utilisateurId ?? this.utilisateurId,
query: query ?? this.query,
types: types ?? this.types,
statuts: statuts ?? this.statuts,
dateDebut: dateDebut ?? this.dateDebut,
dateFin: dateFin ?? this.dateFin,
limite: limite ?? this.limite,
);
}
@override
String toString() {
return 'RechercherNotificationsParams{utilisateurId: $utilisateurId, query: $query, types: $types, statuts: $statuts, dateDebut: $dateDebut, dateFin: $dateFin, limite: $limite}';
}
}
/// Use case pour obtenir les notifications par type
class ObtenirNotificationsParTypeUseCase implements UseCase<List<NotificationEntity>, ObtenirNotificationsParTypeParams> {
final NotificationsRepository repository;
ObtenirNotificationsParTypeUseCase(this.repository);
@override
Future<Either<Failure, List<NotificationEntity>>> call(ObtenirNotificationsParTypeParams params) async {
return await repository.obtenirNotificationsParType(
params.utilisateurId,
params.type,
limite: params.limite,
);
}
}
/// Paramètres pour obtenir les notifications par type
class ObtenirNotificationsParTypeParams {
final String utilisateurId;
final TypeNotification type;
final int limite;
const ObtenirNotificationsParTypeParams({
required this.utilisateurId,
required this.type,
this.limite = 50,
});
ObtenirNotificationsParTypeParams copyWith({
String? utilisateurId,
TypeNotification? type,
int? limite,
}) {
return ObtenirNotificationsParTypeParams(
utilisateurId: utilisateurId ?? this.utilisateurId,
type: type ?? this.type,
limite: limite ?? this.limite,
);
}
@override
String toString() {
return 'ObtenirNotificationsParTypeParams{utilisateurId: $utilisateurId, type: $type, limite: $limite}';
}
}
/// Use case pour obtenir les notifications importantes
class ObtenirNotificationsImportantesUseCase implements UseCase<List<NotificationEntity>, String> {
final NotificationsRepository repository;
ObtenirNotificationsImportantesUseCase(this.repository);
@override
Future<Either<Failure, List<NotificationEntity>>> call(String utilisateurId) async {
return await repository.obtenirNotificationsImportantes(utilisateurId);
}
}
/// Use case pour obtenir les statistiques des notifications
class ObtenirStatistiquesNotificationsUseCase implements UseCase<Map<String, dynamic>, ObtenirStatistiquesParams> {
final NotificationsRepository repository;
ObtenirStatistiquesNotificationsUseCase(this.repository);
@override
Future<Either<Failure, Map<String, dynamic>>> call(ObtenirStatistiquesParams params) async {
return await repository.obtenirStatistiques(
params.utilisateurId,
periode: params.periode,
);
}
}
/// Paramètres pour obtenir les statistiques
class ObtenirStatistiquesParams {
final String utilisateurId;
final int periode;
const ObtenirStatistiquesParams({
required this.utilisateurId,
this.periode = 30,
});
ObtenirStatistiquesParams copyWith({
String? utilisateurId,
int? periode,
}) {
return ObtenirStatistiquesParams(
utilisateurId: utilisateurId ?? this.utilisateurId,
periode: periode ?? this.periode,
);
}
@override
String toString() {
return 'ObtenirStatistiquesParams{utilisateurId: $utilisateurId, periode: $periode}';
}
}