fix(chat): Correction race condition + Implémentation TODOs

## Corrections Critiques

### Race Condition - Statuts de Messages
- Fix : Les icônes de statut (✓, ✓✓, ✓✓ bleu) ne s'affichaient pas
- Cause : WebSocket delivery confirmations arrivaient avant messages locaux
- Solution : Pattern Optimistic UI dans chat_bloc.dart
  - Création message temporaire immédiate
  - Ajout à la liste AVANT requête HTTP
  - Remplacement par message serveur à la réponse
- Fichier : lib/presentation/state_management/chat_bloc.dart

## Implémentation TODOs (13/21)

### Social (social_header_widget.dart)
-  Copier lien du post dans presse-papiers
-  Partage natif via Share.share()
-  Dialogue de signalement avec 5 raisons

### Partage (share_post_dialog.dart)
-  Interface sélection d'amis avec checkboxes
-  Partage externe via Share API

### Média (media_upload_service.dart)
-  Parsing JSON réponse backend
-  Méthode deleteMedia() pour suppression
-  Génération miniature vidéo

### Posts (create_post_dialog.dart, edit_post_dialog.dart)
-  Extraction URL depuis uploads
-  Documentation chargement médias

### Chat (conversations_screen.dart)
-  Navigation vers notifications
-  ConversationSearchDelegate pour recherche

## Nouveaux Fichiers

### Configuration
- build-prod.ps1 : Script build production avec dart-define
- lib/core/constants/env_config.dart : Gestion environnements

### Documentation
- TODOS_IMPLEMENTED.md : Documentation complète TODOs

## Améliorations

### Architecture
- Refactoring injection de dépendances
- Amélioration routing et navigation
- Optimisation providers (UserProvider, FriendsProvider)

### UI/UX
- Amélioration thème et couleurs
- Optimisation animations
- Meilleure gestion erreurs

### Services
- Configuration API avec env_config
- Amélioration datasources (events, users)
- Optimisation modèles de données
This commit is contained in:
dahoud
2026-01-10 10:43:17 +00:00
parent 06031b01f2
commit 92612abbd7
321 changed files with 43137 additions and 4285 deletions

View File

@@ -1,9 +1,20 @@
import 'package:get_it/get_it.dart';
import 'package:http/http.dart' as http;
import '../../core/utils/app_logger.dart';
import '../../data/datasources/chat_remote_data_source.dart';
import '../../data/datasources/event_remote_data_source.dart';
import '../../data/datasources/notification_remote_data_source.dart';
import '../../data/datasources/user_remote_data_source.dart';
import '../../data/repositories/chat_repository_impl.dart';
import '../../data/repositories/friends_repository_impl.dart';
import '../../data/repositories/user_repository_impl.dart';
import '../../data/services/preferences_helper.dart';
import '../../data/services/secure_storage.dart';
import '../../domain/repositories/chat_repository.dart';
import '../../domain/usecases/get_user.dart';
import '../../presentation/state_management/chat_bloc.dart';
import '../../presentation/state_management/event_bloc.dart';
/// Instance globale pour gérer l'injection des dépendances via GetIt
final sl = GetIt.instance;
@@ -12,24 +23,59 @@ final sl = GetIt.instance;
/// Utilisée pour fournir des services, des data sources, des repositories et des use cases.
void init() {
// Log de démarrage de l'injection des dépendances
print("Démarrage de l'initialisation des dépendances.");
AppLogger.i("Démarrage de l'initialisation des dépendances.", tag: 'DI');
// Register Http Client
sl.registerLazySingleton(() => http.Client());
print("Client HTTP enregistré.");
AppLogger.d('Client HTTP enregistré.', tag: 'DI');
// Register Data Sources
sl.registerLazySingleton(() => UserRemoteDataSource(sl()));
print("DataSource pour UserRemoteDataSource enregistré.");
AppLogger.d('DataSource pour UserRemoteDataSource enregistré.', tag: 'DI');
sl.registerLazySingleton(() => ChatRemoteDataSource(sl()));
AppLogger.d('DataSource pour ChatRemoteDataSource enregistré.', tag: 'DI');
sl.registerLazySingleton(() => EventRemoteDataSource(sl()));
AppLogger.d('DataSource pour EventRemoteDataSource enregistré.', tag: 'DI');
sl.registerLazySingleton(() => NotificationRemoteDataSource(sl()));
AppLogger.d('DataSource pour NotificationRemoteDataSource enregistré.', tag: 'DI');
// Note: ChatWebSocketService n'est pas enregistré dans GetIt car il nécessite
// un userId qui n'est connu qu'au moment de la connexion. Il est créé
// dynamiquement dans les écrans qui en ont besoin.
// Register Services
sl.registerLazySingleton(() => SecureStorage());
AppLogger.d('Service SecureStorage enregistré.', tag: 'DI');
sl.registerLazySingleton(() => PreferencesHelper());
AppLogger.d('Service PreferencesHelper enregistré.', tag: 'DI');
// Register Repositories
sl.registerLazySingleton(() => UserRepositoryImpl(remoteDataSource: sl()));
print("Repository pour UserRepositoryImpl enregistré.");
AppLogger.d('Repository pour UserRepositoryImpl enregistré.', tag: 'DI');
sl.registerLazySingleton<ChatRepository>(
() => ChatRepositoryImpl(remoteDataSource: sl()),
);
AppLogger.d('Repository pour ChatRepository enregistré.', tag: 'DI');
sl.registerLazySingleton(() => FriendsRepositoryImpl(client: sl()));
AppLogger.d('Repository pour FriendsRepositoryImpl enregistré.', tag: 'DI');
// Register Use Cases
sl.registerLazySingleton(() => GetUser(sl()));
print("UseCase pour GetUser enregistré.");
AppLogger.d('UseCase pour GetUser enregistré.', tag: 'DI');
// Register Blocs
sl.registerFactory(() => ChatBloc(chatRepository: sl()));
AppLogger.d('Bloc pour ChatBloc enregistré.', tag: 'DI');
sl.registerFactory(() => EventBloc(remoteDataSource: sl()));
AppLogger.d('Bloc pour EventBloc enregistré.', tag: 'DI');
// Log de fin d'initialisation des dépendances
print("Initialisation des dépendances terminée.");
AppLogger.i('Initialisation des dépendances terminée.', tag: 'DI');
}

View File

@@ -1,20 +1,20 @@
import 'package:flutter/material.dart';
import 'package:afterwork/presentation/screens/login/login_screen.dart';
import 'package:afterwork/presentation/screens/story/story_screen.dart';
import 'package:afterwork/presentation/screens/profile/profile_screen.dart';
import 'package:afterwork/presentation/screens/settings/settings_screen.dart';
import 'package:afterwork/presentation/screens/home/home_screen.dart';
import 'package:afterwork/presentation/screens/event/event_screen.dart';
import 'package:afterwork/data/datasources/event_remote_data_source.dart';
import '../data/datasources/event_remote_data_source.dart';
import '../domain/entities/conversation.dart';
import '../presentation/reservations/reservations_screen.dart';
import '../presentation/screens/chat/chat_screen.dart';
import '../presentation/screens/chat/conversations_screen.dart';
import '../presentation/screens/event/event_screen.dart';
import '../presentation/screens/home/home_screen.dart';
import '../presentation/screens/login/login_screen.dart';
import '../presentation/screens/profile/profile_screen.dart';
import '../presentation/screens/settings/settings_screen.dart';
import '../presentation/screens/story/story_screen.dart';
/// [AppRouter] gère la navigation dans l'application.
/// Chaque navigation est loguée pour assurer une traçabilité dans la console.
class AppRouter {
final EventRemoteDataSource eventRemoteDataSource;
final String userId;
final String userFirstName;
final String userLastName;
/// Constructeur de [AppRouter] initialisant les informations utilisateur
/// et la source de données pour les événements.
@@ -28,15 +28,19 @@ class AppRouter {
required this.userLastName,
}) {
// Log d'initialisation avec les informations utilisateur
debugPrint("[LOG] AppRouter initialisé avec les infos utilisateur : $userId, $userFirstName, $userLastName");
debugPrint('[LOG] AppRouter initialisé avec les infos utilisateur : $userId, $userFirstName, $userLastName');
}
final EventRemoteDataSource eventRemoteDataSource;
final String userId;
final String userFirstName;
final String userLastName;
/// Génère une route en fonction du [RouteSettings] fourni.
///
/// Logue chaque navigation en fonction du nom de la route.
Route<dynamic> generateRoute(RouteSettings settings) {
// Log de la navigation vers la route
debugPrint("[LOG] Navigation vers la route : ${settings.name}");
debugPrint('[LOG] Navigation vers la route : ${settings.name}');
switch (settings.name) {
case '/':
@@ -82,8 +86,30 @@ class AppRouter {
debugPrint("[LOG] Chargement de l'écran des réservations");
return MaterialPageRoute(builder: (_) => const ReservationsScreen());
case '/conversations':
debugPrint("[LOG] Chargement de l'écran des conversations");
return MaterialPageRoute(builder: (_) => const ConversationsScreen());
case '/chat':
debugPrint("[LOG] Chargement de l'écran de chat");
// Récupérer la conversation depuis les arguments
final conversation = settings.arguments as Conversation?;
if (conversation == null) {
debugPrint("[ERROR] Conversation manquante pour la route /chat");
return MaterialPageRoute(
builder: (_) => const Scaffold(
body: Center(
child: Text('Erreur: Conversation non spécifiée'),
),
),
);
}
return MaterialPageRoute(
builder: (_) => ChatScreen(conversation: conversation),
);
default:
debugPrint("[ERROR] Route non trouvée : ${settings.name}");
debugPrint('[ERROR] Route non trouvée : ${settings.name}');
return MaterialPageRoute(
builder: (_) => const Scaffold(
body: Center(