From 0772f83369d5b20c6fd7d3e567e4b15d575ddf74 Mon Sep 17 00:00:00 2001 From: Julien Martin Date: Wed, 7 May 2025 17:09:06 +0200 Subject: [PATCH] =?UTF-8?q?feat(auth):=20am=C3=A9lioration=20UI=20et=20UX?= =?UTF-8?q?=20=C3=A9tape=204=20inscription=20parent?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- frontend/lib/navigation/app_router.dart | 6 + .../auth/parent_register_step3_screen.dart | 4 +- .../auth/parent_register_step4_screen.dart | 200 ++++++++++++++++++ .../widgets/custom_decorated_text_field.dart | 56 +++++ 4 files changed, 264 insertions(+), 2 deletions(-) create mode 100644 frontend/lib/screens/auth/parent_register_step4_screen.dart create mode 100644 frontend/lib/widgets/custom_decorated_text_field.dart diff --git a/frontend/lib/navigation/app_router.dart b/frontend/lib/navigation/app_router.dart index 70fd225..0e27700 100644 --- a/frontend/lib/navigation/app_router.dart +++ b/frontend/lib/navigation/app_router.dart @@ -5,6 +5,7 @@ import '../screens/auth/register_choice_screen.dart'; import '../screens/auth/parent_register_step1_screen.dart'; import '../screens/auth/parent_register_step2_screen.dart'; import '../screens/auth/parent_register_step3_screen.dart'; +import '../screens/auth/parent_register_step4_screen.dart'; import '../screens/home/home_screen.dart'; class AppRouter { @@ -13,6 +14,7 @@ class AppRouter { static const String parentRegisterStep1 = '/parent-register/step1'; static const String parentRegisterStep2 = '/parent-register/step2'; static const String parentRegisterStep3 = '/parent-register/step3'; + static const String parentRegisterStep4 = '/parent-register/step4'; static const String home = '/home'; static Route generateRoute(RouteSettings settings) { @@ -39,6 +41,10 @@ class AppRouter { screen = const ParentRegisterStep3Screen(); slideTransition = true; break; + case parentRegisterStep4: + screen = const ParentRegisterStep4Screen(); + slideTransition = true; + break; case home: screen = const HomeScreen(); break; diff --git a/frontend/lib/screens/auth/parent_register_step3_screen.dart b/frontend/lib/screens/auth/parent_register_step3_screen.dart index 8ebdfca..0fcd231 100644 --- a/frontend/lib/screens/auth/parent_register_step3_screen.dart +++ b/frontend/lib/screens/auth/parent_register_step3_screen.dart @@ -330,8 +330,8 @@ class _ParentRegisterStep3ScreenState extends State { child: IconButton( icon: Image.asset('assets/images/chevron_right.png', height: 40), onPressed: () { - print('Passer à l\'étape 4 (Situation familiale)'); - // Navigator.pushNamed(context, '/parent-register/step4'); + print('Passer à l\'étape 4 (Situation familiale et CGU)'); + Navigator.pushNamed(context, '/parent-register/step4'); }, tooltip: 'Suivant', ), diff --git a/frontend/lib/screens/auth/parent_register_step4_screen.dart b/frontend/lib/screens/auth/parent_register_step4_screen.dart new file mode 100644 index 0000000..c9e776e --- /dev/null +++ b/frontend/lib/screens/auth/parent_register_step4_screen.dart @@ -0,0 +1,200 @@ +import 'package:flutter/material.dart'; +import 'package:google_fonts/google_fonts.dart'; +import 'package:p_tits_pas/widgets/custom_decorated_text_field.dart'; // Import du nouveau widget +import 'dart:math' as math; // Pour la rotation du chevron +import 'package:p_tits_pas/widgets/app_custom_checkbox.dart'; // Import de la checkbox personnalisée + +class ParentRegisterStep4Screen extends StatefulWidget { + const ParentRegisterStep4Screen({super.key}); + + @override + State createState() => _ParentRegisterStep4ScreenState(); +} + +class _ParentRegisterStep4ScreenState extends State { + final _motivationController = TextEditingController(); + bool _cguAccepted = false; + + @override + void dispose() { + _motivationController.dispose(); + super.dispose(); + } + + void _showCGUModal() { + // Un long texte Lorem Ipsum pour simuler les CGU + const String loremIpsumText = ''' +Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed non risus. Suspendisse lectus tortor, dignissim sit amet, adipiscing nec, ultricies sed, dolor. Cras elementum ultrices diam. Maecenas ligula massa, varius a, semper congue, euismod non, mi. Proin porttitor, orci nec nonummy molestie, enim est eleifend mi, non fermentum diam nisl sit amet erat. Duis semper. Duis arcu massa, scelerisque vitae, consequat in, pretium a, enim. Pellentesque congue. Ut in risus volutpat libero pharetra tempor. Cras vestibulum bibendum augue. Praesent egestas leo in pede. Praesent blandit odio eu enim. Pellentesque sed dui ut augue blandit sodales. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; Aliquam nibh. Mauris ac mauris sed pede pellentesque fermentum. Maecenas adipiscing ante non diam sodales hendrerit. + +Ut velit mauris, egestas sed, gravida nec, ornare ut, mi. Aenean ut orci vel massa suscipit pulvinar. Nulla sollicitudin. Fusce varius, ligula non tempus aliquam, nunc turpis ullamcorper nibh, in tempus sapien eros vitae ligula. Pellentesque rhoncus nunc et augue. Integer id felis. Curabitur aliquet pellentesque diam. Integer quis metus vitae elit lobortis egestas. Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Morbi vel erat non mauris convallis vehicula. Nulla et sapien. Integer tortor tellus, aliquam faucibus, convallis id, congue eu, quam. Mauris ullamcorper felis vitae erat. Proin feugiat, augue non elementum posuere, metus purus iaculis lectus, et tristique ligula justo vitae magna. + +Aliquam convallis sollicitudin purus. Praesent aliquam, enim at fermentum mollis, ligula massa adipiscing nisl, ac euismod nibh nisl eu lectus. Fusce vulputate sem at sapien. Vivamus leo. Aliquam euismod libero eu enim. Nulla nec felis sed leo placerat imperdiet. Aenean suscipit nulla in justo. Suspendisse cursus rutrum augue. Nulla tincidunt tincidunt mi. Curabitur iaculis, lorem vel rhoncus faucibus, felis magna fermentum augue, et ultricies lacus lorem varius purus. Curabitur eu amet. + +Sed non risus. Suspendisse lectus tortor, dignissim sit amet, adipiscing nec, ultricies sed, dolor. Cras elementum ultrices diam. Maecenas ligula massa, varius a, semper congue, euismod non, mi. Proin porttitor, orci nec nonummy molestie, enim est eleifend mi, non fermentum diam nisl sit amet erat. Duis semper. Duis arcu massa, scelerisque vitae, consequat in, pretium a, enim. Pellentesque congue. Ut in risus volutpat libero pharetra tempor. Cras vestibulum bibendum augue. Praesent egestas leo in pede. Praesent blandit odio eu enim. Pellentesque sed dui ut augue blandit sodales. Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia Curae; Aliquam nibh. Mauris ac mauris sed pede pellentesque fermentum. Maecenas adipiscing ante non diam sodales hendrerit. + +Ut velit mauris, egestas sed, gravida nec, ornare ut, mi. Aenean ut orci vel massa suscipit pulvinar. Nulla sollicitudin. Fusce varius, ligula non tempus aliquam, nunc turpis ullamcorper nibh, in tempus sapien eros vitae ligula. Pellentesque rhoncus nunc et augue. Integer id felis. Curabitur aliquet pellentesque diam. Integer quis metus vitae elit lobortis egestas. Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Morbi vel erat non mauris convallis vehicula. Nulla et sapien. Integer tortor tellus, aliquam faucibus, convallis id, congue eu, quam. Mauris ullamcorper felis vitae erat. Proin feugiat, augue non elementum posuere, metus purus iaculis lectus, et tristique ligula justo vitae magna. Etiam et felis dolor. + +Praesent aliquam, enim at fermentum mollis, ligula massa adipiscing nisl, ac euismod nibh nisl eu lectus. Fusce vulputate sem at sapien. Vivamus leo. Aliquam euismod libero eu enim. Nulla nec felis sed leo placerat imperdiet. Aenean suscipit nulla in justo. Suspendisse cursus rutrum augue. Nulla tincidunt tincidunt mi. Curabitur iaculis, lorem vel rhoncus faucibus, felis magna fermentum augue, et ultricies lacus lorem varius purus. Curabitur eu amet. Duis autem vel eum iriure dolor in hendrerit in vulputate velit esse molestie consequat, vel illum dolore eu feugiat nulla facilisis at vero eros et accumsan et iusto odio dignissim qui blandit praesent luptatum zzril delenit augue duis dolore te feugait nulla facilisi. Lorem ipsum dolor sit amet, consectetuer adipiscing elit, sed diam nonummy nibh euismod tincidunt ut laoreet dolore magna aliquam erat volutpat. + +Ut wisi enim ad minim veniam, quis nostrud exerci tation ullamcorper suscipit lobortis nisl ut aliquip ex ea commodo consequat. Duis autem vel eum iriure dolor in hendrerit in vulputate velit esse molestie consequat, vel illum dolore eu feugiat nulla facilisis at vero eros et accumsan et iusto odio dignissim qui blandit praesent luptatum zzril delenit augue duis dolore te feugait nulla facilisi. Nam liber tempor cum soluta nobis eleifend option congue nihil imperdiet doming id quod mazim placerat facer possim assum. Lorem ipsum dolor sit amet, consectetuer adipiscing elit, sed diam nonummy nibh euismod tincidunt ut laoreet dolore magna aliquam erat volutpat. Ut wisi enim ad minim veniam, quis nostrud exerci tation ullamcorper suscipit lobortis nisl ut aliquip ex ea commodo consequat. +'''; + + showDialog( + context: context, + barrierDismissible: false, // L'utilisateur doit utiliser le bouton + builder: (BuildContext dialogContext) { + return AlertDialog( + title: Text( + 'Conditions Générales d\'Utilisation', + style: GoogleFonts.merienda(fontWeight: FontWeight.bold), + ), + content: SizedBox( + width: MediaQuery.of(dialogContext).size.width * 0.7, // 70% de la largeur de l'écran + height: MediaQuery.of(dialogContext).size.height * 0.6, // 60% de la hauteur de l'écran + child: SingleChildScrollView( + child: Text( + loremIpsumText, + style: GoogleFonts.merienda(fontSize: 13), + textAlign: TextAlign.justify, + ), + ), + ), + actionsPadding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 10.0), + actionsAlignment: MainAxisAlignment.center, + actions: [ + ElevatedButton( + style: ElevatedButton.styleFrom( + backgroundColor: Theme.of(dialogContext).primaryColor, + padding: const EdgeInsets.symmetric(horizontal: 30, vertical: 15), + ), + child: Text( + 'Valider et Accepter', + style: GoogleFonts.merienda(fontSize: 15, color: Colors.white, fontWeight: FontWeight.bold), + ), + onPressed: () { + Navigator.of(dialogContext).pop(); // Ferme la modale + setState(() { + _cguAccepted = true; // Met à jour l'état + }); + }, + ), + ], + ); + }, + ); + } + + @override + Widget build(BuildContext context) { + final screenSize = MediaQuery.of(context).size; + // Calculer la largeur disponible pour le contenu principal de la colonne + // La SingleChildScrollView a un padding horizontal de 50.0 de chaque côté. + final availableContentWidth = screenSize.width - (50.0 * 2); + // Définir la taille du champ de texte carré comme un pourcentage de cette largeur disponible + final textFieldSize = (availableContentWidth * 0.65) / 2; // Réduction de la taille par deux + + return Scaffold( + body: Stack( + children: [ + Positioned.fill( + child: Image.asset('assets/images/paper2.png', fit: BoxFit.cover, repeat: ImageRepeat.repeat), + ), + Center( + child: SingleChildScrollView( + padding: const EdgeInsets.symmetric(vertical: 40.0, horizontal: 50.0), + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Text( + 'Étape 4/5', // Supposant 5 étapes au total pour l'instant + style: GoogleFonts.merienda(fontSize: 16, color: Colors.black54), + ), + const SizedBox(height: 20), + Text( + 'Merci de motiver votre demande création de compte :', + style: GoogleFonts.merienda(fontSize: 22, fontWeight: FontWeight.bold, color: Colors.black87), + textAlign: TextAlign.center, + ), + const SizedBox(height: 30), + SizedBox( + width: textFieldSize, + height: textFieldSize, + child: CustomDecoratedTextField( + controller: _motivationController, + hintText: 'Écrivez ici pour motiver votre demande...', + fieldHeight: textFieldSize, + maxLines: 10, + expandDynamically: true, + ), + ), + const SizedBox(height: 30), + GestureDetector( + onTap: () { + // Si on clique sur la zone et que ce n'est pas encore accepté, + // ou si c'est déjà accepté (l'utilisateur veut peut-être revoir les CGU) + _showCGUModal(); + }, + child: AppCustomCheckbox( + label: 'J\'accepte les conditions générales d\'utilisation', + value: _cguAccepted, + onChanged: (newValue) { + // La logique d'ouverture de la modale est déclenchée par le GestureDetector externe. + // La modale mettra à jour _cguAccepted et reconstruira le widget. + // Si newValue est true (ce qui signifie que la modale a été acceptée et _cguAccepted mis à jour), + // on n'a rien à faire de plus ici. + // Si newValue est false (l'utilisateur a réussi à la décocher d'une manière ou d'une autre, + // ce qui ne devrait pas arriver car on ouvre toujours la modale), + // on pourrait vouloir forcer l'affichage de la modale aussi. + // Pour l'instant, on se fie au fait que la modale gère l'acceptation. + if (!_cguAccepted) { // Si pas encore accepté, la modale doit s'ouvrir + _showCGUModal(); + } else if (newValue == false) { // Si on essaie de décocher + // Optionnel: Forcer la non-acceptation et rouvrir la modale ? + // Pour l'instant, on ne fait rien, la modale gère. + } + }, + // Vous pouvez ajuster checkboxSize et checkmarkSizeFactor si nécessaire + ), + ), + const SizedBox(height: 40), + // On ajoutera un Form et un bouton de soumission principal plus tard + // Pour l'instant, les chevrons servent à la navigation simple + ], + ), + ), + ), + // Chevrons de navigation + Positioned( + top: screenSize.height / 2 - 20, + left: 40, + child: IconButton( + icon: Transform(alignment: Alignment.center, transform: Matrix4.rotationY(math.pi), child: Image.asset('assets/images/chevron_right.png', height: 40)), + onPressed: () => Navigator.pop(context), + tooltip: 'Retour', + ), + ), + Positioned( + top: screenSize.height / 2 - 20, + right: 40, + child: IconButton( + icon: Image.asset('assets/images/chevron_right.png', height: 40), + onPressed: _cguAccepted + ? () { + print('Motivation: ${_motivationController.text}'); + print('CGU acceptées: $_cguAccepted'); + // TODO: Naviguer vers l'étape 5 + // Navigator.pushNamed(context, '/parent-register/step5'); + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar(content: Text('Navigation vers Étape 5 (TODO)')), + ); + } + : null, // Désactiver si les CGU ne sont pas acceptées + tooltip: 'Suivant', + ), + ), + ], + ), + ); + } +} \ No newline at end of file diff --git a/frontend/lib/widgets/custom_decorated_text_field.dart b/frontend/lib/widgets/custom_decorated_text_field.dart new file mode 100644 index 0000000..9f7f52c --- /dev/null +++ b/frontend/lib/widgets/custom_decorated_text_field.dart @@ -0,0 +1,56 @@ +import 'package:flutter/material.dart'; +import 'package:google_fonts/google_fonts.dart'; + +class CustomDecoratedTextField extends StatelessWidget { + final TextEditingController controller; + final String hintText; + final int maxLines; + final double? fieldHeight; // Hauteur optionnelle pour le champ + final bool expandDynamically; // Nouvelle propriété + + const CustomDecoratedTextField({ + super.key, + required this.controller, + this.hintText = 'Écrire votre texte ici...', + this.maxLines = 10, // Un nombre raisonnable de lignes par défaut si non dynamique + this.fieldHeight, // Si non fourni, la hauteur sera intrinsèque ou définie par l'image + this.expandDynamically = false, // Par défaut, non dynamique + }); + + @override + Widget build(BuildContext context) { + return SizedBox( + height: fieldHeight, // Permet de forcer une hauteur si besoin + child: Stack( + alignment: Alignment.topLeft, + children: [ + Image.asset( + 'assets/images/square.png', // L'image de fond + fit: BoxFit.fill, // Pour remplir l'espace du Stack/SizedBox + width: double.infinity, // S'assurer qu'elle prend toute la largeur disponible + height: fieldHeight != null ? double.infinity : null, // Et toute la hauteur si fieldHeight est spécifié + ), + Padding( + // Ajouter un padding interne pour que le texte ne colle pas aux bords de l'image + padding: const EdgeInsets.only(top: 25.0, bottom: 15.0, left: 20.0, right: 20.0), // Augmentation de la marge supérieure + child: TextFormField( + controller: controller, + keyboardType: TextInputType.multiline, + maxLines: expandDynamically ? null : maxLines, // S'étend dynamiquement si expandDynamically est true + style: GoogleFonts.merienda(fontSize: 15, color: Colors.black87), + textAlignVertical: TextAlignVertical.top, + decoration: InputDecoration( + hintText: hintText, + hintStyle: GoogleFonts.merienda(fontSize: 15, color: Colors.black54.withOpacity(0.7)), + border: InputBorder.none, // Pas de bordure pour le TextFormField lui-même + contentPadding: EdgeInsets.zero, // Le padding est géré par le widget Padding externe + // Pour aligner le hintText en haut à gauche + alignLabelWithHint: true, + ), + ), + ), + ], + ), + ); + } +} \ No newline at end of file