feat(auth): amélioration UI et UX étape 4 inscription parent

This commit is contained in:
Julien Martin 2025-05-07 17:09:06 +02:00
parent 42d147c273
commit 0772f83369
4 changed files with 264 additions and 2 deletions

View File

@ -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<dynamic> 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;

View File

@ -330,8 +330,8 @@ class _ParentRegisterStep3ScreenState extends State<ParentRegisterStep3Screen> {
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',
),

View File

@ -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<ParentRegisterStep4Screen> createState() => _ParentRegisterStep4ScreenState();
}
class _ParentRegisterStep4ScreenState extends State<ParentRegisterStep4Screen> {
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<void>(
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: <Widget>[
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 é 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',
),
),
],
),
);
}
}

View File

@ -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,
),
),
),
],
),
);
}
}