feat: WebSocket temps réel + Finance Workflow + corrections

- Task #6: WebSocket /ws/dashboard + Kafka events (5 topics)
  * Backend: KafkaEventProducer, KafkaEventConsumer
  * Mobile: WebSocketService (reconnection, heartbeat, typed events)
  * DashboardBloc: Auto-refresh depuis WebSocket events

- Finance Workflow: approbations + budgets (backend + mobile)
  * Backend: entities, services, resources, migrations Flyway V6
  * Mobile: features finance_workflow complète avec BLoC

- Corrections DI: interfaces IRepository partout
  * IProfileRepository, IOrganizationRepository, IMembreRepository
  * GetIt configuré avec @injectable

- Spec-Kit: constitution + templates mis à jour
  * .specify/memory/constitution.md enrichie
  * Templates agent, plan, spec, tasks, checklist

- Nettoyage: fichiers temporaires supprimés

Signed-off-by: lions dev Team
This commit is contained in:
dahoud
2026-03-15 02:12:17 +00:00
parent bbc409de9d
commit e8ad874015
635 changed files with 58160 additions and 20674 deletions

View File

@@ -0,0 +1,23 @@
/// Use Case: Supprimer le compte utilisateur
library delete_account;
import 'package:injectable/injectable.dart';
import '../repositories/profile_repository.dart';
/// Supprime le compte utilisateur (soft delete via désactivation)
/// Endpoint: POST /api/membres/{id}/desactiver
/// Marque le compte comme inactif au lieu de supprimer les données
@injectable
class DeleteAccount {
final IProfileRepository _repository;
DeleteAccount(this._repository);
/// Exécute le use case
/// [id] : Identifiant du membre
/// Lance une suppression de compte (soft delete via désactivation)
/// L'utilisateur sera déconnecté après cette opération
Future<void> call(String id) async {
return _repository.deleteAccount(id);
}
}

View File

@@ -0,0 +1,20 @@
/// Use Case: Récupérer le profil du membre connecté
library get_profile;
import 'package:injectable/injectable.dart';
import '../../../members/data/models/membre_complete_model.dart';
import '../repositories/profile_repository.dart';
/// Récupère le profil du membre connecté via l'endpoint /api/membres/me
@injectable
class GetProfile {
final IProfileRepository _repository;
GetProfile(this._repository);
/// Exécute le use case
/// Retourne le profil du membre connecté ou null si non trouvé
Future<MembreCompletModel?> call() async {
return _repository.getMe();
}
}

View File

@@ -0,0 +1,23 @@
/// Use Case: Mettre à jour la photo de profil
library update_avatar;
import 'package:injectable/injectable.dart';
import '../../../members/data/models/membre_complete_model.dart';
import '../repositories/profile_repository.dart';
/// Met à jour la photo de profil (avatar) d'un membre
/// Utilise l'endpoint de mise à jour générale avec copyWith
@injectable
class UpdateAvatar {
final IProfileRepository _repository;
UpdateAvatar(this._repository);
/// Exécute le use case
/// [id] : Identifiant du membre
/// [photoUrl] : URL de la nouvelle photo de profil
/// Retourne le profil mis à jour avec la nouvelle photo
Future<MembreCompletModel> call(String id, String photoUrl) async {
return _repository.updateAvatar(id, photoUrl);
}
}

View File

@@ -0,0 +1,24 @@
/// Use Case: Mettre à jour les préférences utilisateur
library update_preferences;
import 'package:injectable/injectable.dart';
import '../repositories/profile_repository.dart';
/// Met à jour les préférences utilisateur (langue, notifications, thème, etc.)
/// Endpoint: PUT /api/membres/{id}/preferences
/// Fallback sur stockage local si endpoint non disponible
@injectable
class UpdatePreferences {
final IProfileRepository _repository;
UpdatePreferences(this._repository);
/// Exécute le use case
/// [id] : Identifiant du membre
/// [preferences] : Map contenant les préférences à mettre à jour
/// Exemple: { "langue": "fr", "notificationsEmail": true, "theme": "dark" }
/// Retourne les préférences mises à jour
Future<Map<String, dynamic>> call(String id, Map<String, dynamic> preferences) async {
return _repository.updatePreferences(id, preferences);
}
}

View File

@@ -0,0 +1,22 @@
/// Use Case: Mettre à jour le profil utilisateur
library update_profile;
import 'package:injectable/injectable.dart';
import '../../../members/data/models/membre_complete_model.dart';
import '../repositories/profile_repository.dart';
/// Met à jour les informations du profil utilisateur
@injectable
class UpdateProfile {
final IProfileRepository _repository;
UpdateProfile(this._repository);
/// Exécute le use case
/// [id] : Identifiant du membre
/// [membre] : Modèle avec les données mises à jour
/// Retourne le profil mis à jour
Future<MembreCompletModel> call(String id, MembreCompletModel membre) async {
return _repository.updateProfile(id, membre);
}
}