feat: optimisation du démarrage de l'application et de la gestion de l'authentification
- **Refonte du démarrage** : Mise en place d'un `AppInitializer` pour gérer l'initialisation asynchrone de Firebase et du cache en arrière-plan, réduisant le travail synchrone au lancement.
- **Sécurisation de l'authentification** :
- Création d'un `AppStartGate` pour gérer proprement la restauration de la session Firebase Auth et les erreurs potentielles sur le Web.
- Amélioration du `LocalUserProvider` avec un "bootstrap léger" permettant de rendre l'UID disponible immédiatement avant le chargement complet du profil.
- Ajout de protections contre les erreurs d'accès à `FirebaseAuth.instance` (notamment pour les problèmes d'interop JS sur le Web).
- **Optimisation de l'UI** :
- Remplacement du `AutoLoginWrapper` par une gestion plus robuste de la navigation post-authentification.
- Amélioration de l'`AuthGuard` pour permettre l'affichage de certains écrans (comme le calendrier) pendant le chargement des données utilisateur (`allowWhileLoading`).
- Ajout d'un écran de splash screen uniformisé (`StartupSplashScreen`).
- **Services & Cache** :
- Introduction de `CacheService` utilisant `shared_preferences` pour le stockage local léger.
- Refactoring des services (`AlertService`, `EmailService`, `FirebaseStorageManager`) pour accéder aux instances Firebase de manière plus flexible via des getters.
- Mise à jour des dépendances dans `pubspec.yaml` pour inclure `shared_preferences`.
- **Calendrier** : Ajout d'une logique de chargement initial différé des événements (`_scheduleInitialEventsLoad`) pour éviter les appels redondants au démarrage.
- **Maintenance** : Mise à jour de la version de l'application à `1.1.23` et nettoyage des fichiers de cache de déploiement.
This commit is contained in:
@@ -7,8 +7,8 @@ import 'api_service.dart' show FirebaseFunctionsApiService;
|
||||
/// Architecture simplifiée : le client appelle uniquement les Cloud Functions
|
||||
/// Toute la logique métier est gérée côté backend
|
||||
class AlertService {
|
||||
final FirebaseFirestore _firestore = FirebaseFirestore.instance;
|
||||
final FirebaseAuth _auth = FirebaseAuth.instance;
|
||||
FirebaseFirestore get _firestore => FirebaseFirestore.instance;
|
||||
FirebaseAuth get _auth => FirebaseAuth.instance;
|
||||
|
||||
/// Stream des alertes pour l'utilisateur connecté
|
||||
Stream<List<AlertModel>> getAlertsStream() {
|
||||
|
||||
@@ -0,0 +1,79 @@
|
||||
import 'dart:async';
|
||||
|
||||
import 'package:cloud_firestore/cloud_firestore.dart';
|
||||
import 'package:firebase_auth/firebase_auth.dart';
|
||||
import 'package:firebase_core/firebase_core.dart';
|
||||
import 'package:flutter/foundation.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
import 'package:flutter/widgets.dart';
|
||||
|
||||
import '../firebase_options.dart';
|
||||
import '../config/api_config.dart';
|
||||
import 'cache_service.dart';
|
||||
|
||||
/// Service responsable des initialisations lourdes en tâche de fond.
|
||||
///
|
||||
/// Objectif : réduire au maximum le travail synchrone dans main(),
|
||||
/// afficher immédiatement une UI minimale, puis effectuer l'init asynchrone.
|
||||
class AppInitializer with ChangeNotifier {
|
||||
bool _isInitialized = false;
|
||||
bool _isInitializing = false;
|
||||
|
||||
bool get isInitialized => _isInitialized;
|
||||
bool get isInitializing => _isInitializing;
|
||||
|
||||
final CacheService cacheService = CacheService();
|
||||
|
||||
/// Démarre l'initialisation asynchrone. Idempotent.
|
||||
Future<void> initialize() async {
|
||||
if (_isInitialized || _isInitializing) return;
|
||||
_isInitializing = true;
|
||||
notifyListeners();
|
||||
|
||||
try {
|
||||
// Initialiser Firebase
|
||||
await Firebase.initializeApp(
|
||||
options: DefaultFirebaseOptions.currentPlatform,
|
||||
);
|
||||
|
||||
// Configurer les émulateurs en dev si demandé
|
||||
if (ApiConfig.isDevelopment) {
|
||||
try {
|
||||
await FirebaseAuth.instance.useAuthEmulator('localhost', 9199);
|
||||
FirebaseFirestore.instance.useFirestoreEmulator('localhost', 8088);
|
||||
} catch (e) {
|
||||
// Ignorer si non supporté
|
||||
if (kDebugMode) print('Emulator setup failed: $e');
|
||||
}
|
||||
}
|
||||
|
||||
// Initialiser le cache local sans bloquer l'écran de démarrage.
|
||||
unawaited(cacheService.init());
|
||||
|
||||
// Précharger des assets critiques de façon asynchrone
|
||||
unawaited(_preloadAssets());
|
||||
|
||||
// TODO: lancer ici d'autres initialisations non bloquantes
|
||||
|
||||
_isInitialized = true;
|
||||
_isInitializing = false;
|
||||
notifyListeners();
|
||||
} catch (e, st) {
|
||||
if (kDebugMode) print('AppInitializer failed: $e\n$st');
|
||||
_isInitializing = false;
|
||||
// Ne rethrow pas pour éviter de planter l'app; laisser l'UI gérer les erreurs.
|
||||
notifyListeners();
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _preloadAssets() async {
|
||||
try {
|
||||
// Charger quelques assets en mémoire pour rendre l'affichage initial fluide
|
||||
await rootBundle.load('assets/logos/RectangleLogoBlack.png');
|
||||
await rootBundle.load('assets/logos/SquareLogoWhite.png');
|
||||
} catch (e) {
|
||||
if (kDebugMode) print('Preload assets failed: $e');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,44 @@
|
||||
import 'dart:convert';
|
||||
|
||||
import 'package:flutter/foundation.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
/// Service simple de cache local basé sur SharedPreferences.
|
||||
///
|
||||
/// Fonctionne sur mobile et sur Flutter Web pour conserver des données
|
||||
/// locales légères quand cela apporte une vraie valeur.
|
||||
class CacheService {
|
||||
SharedPreferences? _prefs;
|
||||
|
||||
Future<void> init() async {
|
||||
_prefs = await SharedPreferences.getInstance();
|
||||
}
|
||||
|
||||
bool ready() => _prefs != null;
|
||||
|
||||
Future<void> setJson(String key, Map<String, dynamic> value) async {
|
||||
if (_prefs == null) return;
|
||||
await _prefs!.setString(key, jsonEncode(value));
|
||||
}
|
||||
|
||||
Map<String, dynamic>? getJson(String key) {
|
||||
if (_prefs == null) return null;
|
||||
final s = _prefs!.getString(key);
|
||||
if (s == null) return null;
|
||||
try {
|
||||
return jsonDecode(s) as Map<String, dynamic>;
|
||||
} catch (e) {
|
||||
if (kDebugMode) print('CacheService getJson error: $e');
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> setString(String key, String value) async {
|
||||
if (_prefs == null) return;
|
||||
await _prefs!.setString(key, value);
|
||||
}
|
||||
|
||||
String? getString(String key) => _prefs?.getString(key);
|
||||
}
|
||||
|
||||
|
||||
@@ -5,7 +5,7 @@ import 'package:firebase_auth/firebase_auth.dart';
|
||||
|
||||
/// Service d'envoi d'emails via Cloud Functions
|
||||
class EmailService {
|
||||
final FirebaseFunctions _functions = FirebaseFunctions.instanceFor(region: 'europe-west9');
|
||||
FirebaseFunctions get _functions => FirebaseFunctions.instanceFor(region: 'europe-west9');
|
||||
|
||||
/// Envoie un email d'alerte à un utilisateur
|
||||
///
|
||||
|
||||
Reference in New Issue
Block a user