Cette mise à jour majeure introduit un système de notifications robuste, centré sur la création d'alertes et l'envoi d'emails via des Cloud Functions. Elle inclut la gestion des préférences utilisateur, la création automatique d'alertes lors d'événements critiques et une nouvelle interface dédiée.
**Backend (Cloud Functions) :**
- **Nouveau service d'alerting (`createAlert`, `processEquipmentValidation`) :**
- `createAlert` : Nouvelle fonction pour créer une alerte. Elle détermine les utilisateurs à notifier (admins, workforce d'événement) et gère la persistance dans Firestore.
- `processEquipmentValidation` : Endpoint appelé lors de la validation du matériel (chargement/déchargement). Il analyse l'état de l'équipement (`LOST`, `MISSING`, `DAMAGED`) et crée automatiquement les alertes correspondantes.
- **Système d'envoi d'emails (`sendAlertEmail`, `sendDailyDigest`) :**
- `sendAlertEmail` : Cloud Function `onCall` pour envoyer un email d'alerte individuel. Elle respecte les préférences de notification de l'utilisateur (canal email, type d'alerte).
- `sendDailyDigest` : Tâche planifiée (tous les jours à 8h) qui envoie un email récapitulatif des alertes non lues des dernières 24 heures aux utilisateurs concernés.
- Ajout de templates HTML (`base-template`, `alert-individual`, `alert-digest`) avec `Handlebars` pour des emails riches.
- Configuration centralisée du SMTP via des variables d'environnement (`.env`).
- **Triggers Firestore (`onEventCreated`, `onEventUpdated`) :**
- Des triggers créent désormais des alertes d'information lorsqu'un événement est créé ou que de nouveaux membres sont ajoutés à la workforce.
- **Règles Firestore :**
- Mises à jour pour autoriser les utilisateurs authentifiés à créer et modifier leurs propres alertes (marquer comme lue, supprimer), tout en sécurisant les accès.
**Frontend (Flutter) :**
- **Nouvel `AlertService` et `EmailService` :**
- `AlertService` : Centralise la logique de création, lecture et gestion des alertes côté client en appelant les nouvelles Cloud Functions.
- `EmailService` : Service pour déclencher l'envoi d'emails via la fonction `sendAlertEmail`. Il contient la logique pour déterminer si une notification doit être immédiate (critique) ou différée (digest).
- **Nouvelle page de Notifications (`/alerts`) :**
- Interface dédiée pour lister toutes les alertes de l'utilisateur, avec des onglets pour filtrer par catégorie (Toutes, Événement, Maintenance, Équipement).
- Permet de marquer les alertes comme lues, de les supprimer et de tout marquer comme lu.
- **Intégration dans l'UI :**
- Ajout d'un badge de notification dans la `CustomAppBar` affichant le nombre d'alertes non lues en temps réel.
- Le `AutoLoginWrapper` gère désormais la redirection vers des routes profondes (ex: `/alerts`) depuis une URL.
- **Gestion des Préférences de Notification :**
- Ajout d'un widget `NotificationPreferencesWidget` dans la page "Mon Compte".
- Les utilisateurs peuvent désormais activer/désactiver les notifications par email, ainsi que filtrer par type d'alerte (événements, maintenance, etc.).
- Le `UserModel` et `LocalUserProvider` ont été étendus pour gérer ce nouveau modèle de préférences.
- **Création d'alertes contextuelles :**
- Le service `EventFormService` crée maintenant automatiquement une alerte lorsqu'un événement est créé ou modifié.
- La page de préparation d'événement (`EventPreparationPage`) appelle `processEquipmentValidation` à la fin de chaque étape pour une détection automatisée des anomalies.
**Dépendances et CI/CD :**
- Ajout des dépendances `cloud_functions` et `timeago` (Flutter), et `nodemailer`, `handlebars`, `dotenv` (Node.js).
- Ajout de scripts de déploiement PowerShell (`deploy_functions.ps1`, `deploy_firestore_rules.ps1`) pour simplifier les mises en production.
254 lines
8.8 KiB
Dart
254 lines
8.8 KiB
Dart
import 'package:em2rp/providers/users_provider.dart';
|
|
import 'package:em2rp/providers/event_provider.dart';
|
|
import 'package:em2rp/providers/equipment_provider.dart';
|
|
import 'package:em2rp/providers/container_provider.dart';
|
|
import 'package:em2rp/providers/maintenance_provider.dart';
|
|
import 'package:em2rp/providers/alert_provider.dart';
|
|
import 'package:em2rp/utils/auth_guard_widget.dart';
|
|
import 'package:em2rp/views/alerts_page.dart';
|
|
import 'package:em2rp/views/calendar_page.dart';
|
|
import 'package:em2rp/views/login_page.dart';
|
|
import 'package:em2rp/views/equipment_management_page.dart';
|
|
import 'package:em2rp/views/container_management_page.dart';
|
|
import 'package:em2rp/views/container_form_page.dart';
|
|
import 'package:em2rp/views/container_detail_page.dart';
|
|
import 'package:em2rp/views/event_preparation_page.dart';
|
|
import 'package:em2rp/models/container_model.dart';
|
|
import 'package:em2rp/models/event_model.dart';
|
|
import 'package:firebase_auth/firebase_auth.dart';
|
|
import 'package:cloud_firestore/cloud_firestore.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:firebase_core/firebase_core.dart';
|
|
import 'firebase_options.dart';
|
|
import 'utils/colors.dart';
|
|
import 'views/my_account_page.dart';
|
|
import 'views/user_management_page.dart';
|
|
import 'package:provider/provider.dart';
|
|
import 'providers/local_user_provider.dart';
|
|
import 'views/reset_password_page.dart';
|
|
import 'config/env.dart';
|
|
import 'config/api_config.dart';
|
|
import 'package:flutter_localizations/flutter_localizations.dart';
|
|
|
|
void main() async {
|
|
WidgetsFlutterBinding.ensureInitialized();
|
|
await Firebase.initializeApp(
|
|
options: DefaultFirebaseOptions.currentPlatform,
|
|
);
|
|
|
|
// Configuration des émulateurs en mode développement
|
|
if (ApiConfig.isDevelopment) {
|
|
print('🔧 Mode développement activé - Utilisation des émulateurs');
|
|
|
|
// Configurer l'émulateur Auth
|
|
await FirebaseAuth.instance.useAuthEmulator('localhost', 9199);
|
|
print('✓ Auth émulateur configuré: localhost:9199');
|
|
|
|
// Configurer l'émulateur Firestore
|
|
FirebaseFirestore.instance.useFirestoreEmulator('localhost', 8088);
|
|
print('✓ Firestore émulateur configuré: localhost:8088');
|
|
}
|
|
|
|
await FirebaseAuth.instance.setPersistence(Persistence.LOCAL);
|
|
|
|
runApp(
|
|
MultiProvider(
|
|
providers: [
|
|
// LocalUserProvider pour la gestion de l'authentification
|
|
ChangeNotifierProvider<LocalUserProvider>(
|
|
create: (context) => LocalUserProvider()),
|
|
|
|
// UsersProvider migré vers l'API
|
|
ChangeNotifierProvider<UsersProvider>(
|
|
create: (context) => UsersProvider(),
|
|
),
|
|
|
|
// EventProvider migré vers l'API
|
|
ChangeNotifierProvider<EventProvider>(
|
|
create: (context) => EventProvider(),
|
|
),
|
|
|
|
// EquipmentProvider migré vers l'API
|
|
ChangeNotifierProvider<EquipmentProvider>(
|
|
create: (context) => EquipmentProvider(),
|
|
),
|
|
|
|
// ContainerProvider migré vers l'API
|
|
ChangeNotifierProvider<ContainerProvider>(
|
|
create: (context) => ContainerProvider(),
|
|
),
|
|
|
|
// MaintenanceProvider migré vers l'API
|
|
ChangeNotifierProvider<MaintenanceProvider>(
|
|
create: (context) => MaintenanceProvider(),
|
|
),
|
|
ChangeNotifierProvider<AlertProvider>(
|
|
create: (context) => AlertProvider(),
|
|
),
|
|
],
|
|
child: const MyApp(),
|
|
),
|
|
);
|
|
}
|
|
|
|
class MyApp extends StatelessWidget {
|
|
const MyApp({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return MaterialApp(
|
|
title: 'EM2 ERP',
|
|
theme: ThemeData(
|
|
primarySwatch: Colors.red,
|
|
primaryColor: AppColors.noir,
|
|
colorScheme:
|
|
ColorScheme.fromSwatch().copyWith(secondary: AppColors.rouge),
|
|
textTheme: const TextTheme(
|
|
bodyMedium: TextStyle(color: AppColors.noir),
|
|
),
|
|
inputDecorationTheme: const InputDecorationTheme(
|
|
focusedBorder: OutlineInputBorder(
|
|
borderSide: BorderSide(color: AppColors.noir),
|
|
),
|
|
enabledBorder: OutlineInputBorder(
|
|
borderSide: BorderSide(color: AppColors.gris),
|
|
),
|
|
labelStyle: TextStyle(color: AppColors.noir),
|
|
hintStyle: TextStyle(color: AppColors.gris),
|
|
),
|
|
elevatedButtonTheme: ElevatedButtonThemeData(
|
|
style: ElevatedButton.styleFrom(
|
|
foregroundColor: AppColors.blanc,
|
|
backgroundColor: AppColors.noir,
|
|
),
|
|
),
|
|
),
|
|
locale: const Locale('fr', 'FR'),
|
|
supportedLocales: const [
|
|
Locale('fr', 'FR'),
|
|
],
|
|
localizationsDelegates: const [
|
|
GlobalMaterialLocalizations.delegate,
|
|
GlobalWidgetsLocalizations.delegate,
|
|
GlobalCupertinoLocalizations.delegate,
|
|
],
|
|
initialRoute: '/',
|
|
routes: {
|
|
'/': (context) => const AutoLoginWrapper(),
|
|
'/login': (context) => const LoginPage(),
|
|
'/alerts': (context) => const AuthGuard(child: AlertsPage()),
|
|
'/calendar': (context) => const AuthGuard(child: CalendarPage()),
|
|
'/my_account': (context) => const AuthGuard(child: MyAccountPage()),
|
|
'/user_management': (context) => const AuthGuard(
|
|
requiredPermission: "view_all_users", child: UserManagementPage()),
|
|
'/reset_password': (context) {
|
|
final args = ModalRoute.of(context)!.settings.arguments
|
|
as Map<String, dynamic>;
|
|
return ResetPasswordPage(
|
|
email: args['email'] as String,
|
|
actionCode: args['actionCode'] as String,
|
|
);
|
|
},
|
|
'/equipment_management': (context) => const AuthGuard(
|
|
requiredPermission: "view_equipment",
|
|
child: EquipmentManagementPage()),
|
|
'/container_management': (context) => const AuthGuard(
|
|
requiredPermission: "view_equipment",
|
|
child: ContainerManagementPage()),
|
|
'/container_form': (context) {
|
|
final args = ModalRoute.of(context)?.settings.arguments;
|
|
return AuthGuard(
|
|
requiredPermission: "manage_equipment",
|
|
child: ContainerFormPage(
|
|
container: args as ContainerModel?,
|
|
),
|
|
);
|
|
},
|
|
'/container_detail': (context) {
|
|
final container = ModalRoute.of(context)!.settings.arguments as ContainerModel;
|
|
return AuthGuard(
|
|
requiredPermission: "view_equipment",
|
|
child: ContainerDetailPage(container: container),
|
|
);
|
|
},
|
|
'/event_preparation': (context) {
|
|
final args = ModalRoute.of(context)!.settings.arguments as Map<String, dynamic>;
|
|
final event = args['event'] as EventModel;
|
|
return AuthGuard(
|
|
child: EventPreparationPage(
|
|
initialEvent: event,
|
|
),
|
|
);
|
|
},
|
|
},
|
|
);
|
|
}
|
|
}
|
|
|
|
class AutoLoginWrapper extends StatefulWidget {
|
|
const AutoLoginWrapper({super.key});
|
|
|
|
@override
|
|
State<AutoLoginWrapper> createState() => _AutoLoginWrapperState();
|
|
}
|
|
|
|
class _AutoLoginWrapperState extends State<AutoLoginWrapper> {
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
_autoLogin();
|
|
}
|
|
|
|
Future<void> _autoLogin() async {
|
|
try {
|
|
final localAuthProvider =
|
|
Provider.of<LocalUserProvider>(context, listen: false);
|
|
|
|
// Vérifier si l'utilisateur est déjà connecté
|
|
if (FirebaseAuth.instance.currentUser == null && Env.isDevelopment) {
|
|
// Connexion automatique en mode développement
|
|
await localAuthProvider.signInWithEmailAndPassword(
|
|
Env.devAdminEmail,
|
|
Env.devAdminPassword,
|
|
);
|
|
}
|
|
|
|
// Charger les données utilisateur
|
|
await localAuthProvider.loadUserData();
|
|
|
|
if (mounted) {
|
|
// MODIFIÉ : Vérifier si une route spécifique est demandée dans l'URL
|
|
// En Flutter Web, on peut vérifier window.location.hash
|
|
final currentUri = Uri.base;
|
|
final fragment = currentUri.fragment; // Ex: "/alerts" si URL est /#/alerts
|
|
|
|
print('[AutoLoginWrapper] Fragment URL: $fragment');
|
|
|
|
// Si une route spécifique est demandée (autre que / ou vide)
|
|
if (fragment.isNotEmpty && fragment != '/' && fragment != '/calendar') {
|
|
print('[AutoLoginWrapper] Redirection vers: $fragment');
|
|
Navigator.of(context).pushReplacementNamed(fragment);
|
|
} else {
|
|
// Route par défaut : calendrier
|
|
print('[AutoLoginWrapper] Redirection vers: /calendar (défaut)');
|
|
Navigator.of(context).pushReplacementNamed('/calendar');
|
|
}
|
|
}
|
|
} catch (e) {
|
|
print('Auto login failed: $e');
|
|
if (mounted) {
|
|
Navigator.of(context).pushReplacementNamed('/login');
|
|
}
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return const Scaffold(
|
|
body: Center(
|
|
child: CircularProgressIndicator(),
|
|
),
|
|
);
|
|
}
|
|
}
|