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,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(