feat: ajout du parcours complet d'inscription parent avec UI harmonisée et gestion centralisée des données
This commit is contained in:
parent
5156f4fefb
commit
d6ba6019fb
Binary file not shown.
|
Before Width: | Height: | Size: 1.8 MiB |
Binary file not shown.
|
Before Width: | Height: | Size: 84 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 67 KiB |
@ -1,73 +0,0 @@
|
||||
class Child {
|
||||
final String id;
|
||||
final String firstName;
|
||||
final String lastName;
|
||||
final DateTime? birthDate;
|
||||
final DateTime? expectedBirthDate;
|
||||
final String? photoUrl;
|
||||
final bool hasPhotoConsent;
|
||||
final DateTime? photoConsentDate;
|
||||
final String status; // 'unborn', 'active', 'schooled'
|
||||
final List<String> parentIds;
|
||||
final bool isMultipleBirth; // true pour jumeaux, triplés, etc.
|
||||
final DateTime createdAt;
|
||||
final DateTime updatedAt;
|
||||
|
||||
Child({
|
||||
required this.id,
|
||||
required this.firstName,
|
||||
required this.lastName,
|
||||
this.birthDate,
|
||||
this.expectedBirthDate,
|
||||
this.photoUrl,
|
||||
required this.hasPhotoConsent,
|
||||
this.photoConsentDate,
|
||||
required this.status,
|
||||
required this.parentIds,
|
||||
required this.isMultipleBirth,
|
||||
required this.createdAt,
|
||||
required this.updatedAt,
|
||||
});
|
||||
|
||||
factory Child.fromJson(Map<String, dynamic> json) {
|
||||
return Child(
|
||||
id: json['id'],
|
||||
firstName: json['firstName'],
|
||||
lastName: json['lastName'],
|
||||
birthDate: json['birthDate'] != null
|
||||
? DateTime.parse(json['birthDate'])
|
||||
: null,
|
||||
expectedBirthDate: json['expectedBirthDate'] != null
|
||||
? DateTime.parse(json['expectedBirthDate'])
|
||||
: null,
|
||||
photoUrl: json['photoUrl'],
|
||||
hasPhotoConsent: json['hasPhotoConsent'] ?? false,
|
||||
photoConsentDate: json['photoConsentDate'] != null
|
||||
? DateTime.parse(json['photoConsentDate'])
|
||||
: null,
|
||||
status: json['status'] ?? 'unborn',
|
||||
parentIds: List<String>.from(json['parentIds'] ?? []),
|
||||
isMultipleBirth: json['isMultipleBirth'] ?? false,
|
||||
createdAt: DateTime.parse(json['createdAt']),
|
||||
updatedAt: DateTime.parse(json['updatedAt']),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'firstName': firstName,
|
||||
'lastName': lastName,
|
||||
'birthDate': birthDate?.toIso8601String(),
|
||||
'expectedBirthDate': expectedBirthDate?.toIso8601String(),
|
||||
'photoUrl': photoUrl,
|
||||
'hasPhotoConsent': hasPhotoConsent,
|
||||
'photoConsentDate': photoConsentDate?.toIso8601String(),
|
||||
'status': status,
|
||||
'parentIds': parentIds,
|
||||
'isMultipleBirth': isMultipleBirth,
|
||||
'createdAt': createdAt.toIso8601String(),
|
||||
'updatedAt': updatedAt.toIso8601String(),
|
||||
};
|
||||
}
|
||||
}
|
||||
@ -1,63 +0,0 @@
|
||||
class Parent {
|
||||
final String id;
|
||||
final String userId;
|
||||
final String firstName;
|
||||
final String lastName;
|
||||
final String email;
|
||||
final String phoneNumber;
|
||||
final String address;
|
||||
final String city;
|
||||
final String postalCode;
|
||||
final List<String> childrenIds;
|
||||
final DateTime createdAt;
|
||||
final DateTime updatedAt;
|
||||
|
||||
Parent({
|
||||
required this.id,
|
||||
required this.userId,
|
||||
required this.firstName,
|
||||
required this.lastName,
|
||||
required this.email,
|
||||
required this.phoneNumber,
|
||||
required this.address,
|
||||
required this.city,
|
||||
required this.postalCode,
|
||||
required this.childrenIds,
|
||||
required this.createdAt,
|
||||
required this.updatedAt,
|
||||
});
|
||||
|
||||
factory Parent.fromJson(Map<String, dynamic> json) {
|
||||
return Parent(
|
||||
id: json['id'],
|
||||
userId: json['userId'],
|
||||
firstName: json['firstName'],
|
||||
lastName: json['lastName'],
|
||||
email: json['email'],
|
||||
phoneNumber: json['phoneNumber'],
|
||||
address: json['address'],
|
||||
city: json['city'],
|
||||
postalCode: json['postalCode'],
|
||||
childrenIds: List<String>.from(json['childrenIds']),
|
||||
createdAt: DateTime.parse(json['createdAt']),
|
||||
updatedAt: DateTime.parse(json['updatedAt']),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'id': id,
|
||||
'userId': userId,
|
||||
'firstName': firstName,
|
||||
'lastName': lastName,
|
||||
'email': email,
|
||||
'phoneNumber': phoneNumber,
|
||||
'address': address,
|
||||
'city': city,
|
||||
'postalCode': postalCode,
|
||||
'childrenIds': childrenIds,
|
||||
'createdAt': createdAt.toIso8601String(),
|
||||
'updatedAt': updatedAt.toIso8601String(),
|
||||
};
|
||||
}
|
||||
}
|
||||
@ -1,752 +0,0 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:image_picker/image_picker.dart';
|
||||
import 'package:go_router/go_router.dart';
|
||||
import '../../services/auth_service.dart';
|
||||
import '../../theme/app_theme.dart';
|
||||
import 'dart:convert';
|
||||
|
||||
class ChildData {
|
||||
final TextEditingController firstNameController = TextEditingController();
|
||||
final TextEditingController lastNameController = TextEditingController();
|
||||
DateTime? birthDate;
|
||||
DateTime? expectedBirthDate;
|
||||
XFile? photo;
|
||||
bool hasPhotoConsent = false;
|
||||
bool isMultipleBirth = false;
|
||||
bool isUnborn = false;
|
||||
}
|
||||
|
||||
class ParentRegisterScreen extends StatefulWidget {
|
||||
const ParentRegisterScreen({super.key});
|
||||
|
||||
@override
|
||||
State<ParentRegisterScreen> createState() => _ParentRegisterScreenState();
|
||||
}
|
||||
|
||||
class _ParentRegisterScreenState extends State<ParentRegisterScreen> {
|
||||
final _formKey = GlobalKey<FormState>();
|
||||
final _authService = AuthService();
|
||||
int _currentStep = 0;
|
||||
bool _isLoading = false;
|
||||
bool _hasPartner = false;
|
||||
bool _hasAcceptedCGU = false;
|
||||
bool _partnerSameAddress = false;
|
||||
|
||||
// Contrôleurs pour le parent 1
|
||||
final _emailController = TextEditingController();
|
||||
final _passwordController = TextEditingController();
|
||||
final _firstNameController = TextEditingController();
|
||||
final _lastNameController = TextEditingController();
|
||||
final _phoneController = TextEditingController();
|
||||
final _addressController = TextEditingController();
|
||||
final _cityController = TextEditingController();
|
||||
final _postalCodeController = TextEditingController();
|
||||
final _presentationController = TextEditingController();
|
||||
|
||||
// Contrôleurs pour le parent 2
|
||||
final _partnerFirstNameController = TextEditingController();
|
||||
final _partnerLastNameController = TextEditingController();
|
||||
final _partnerEmailController = TextEditingController();
|
||||
final _partnerPhoneController = TextEditingController();
|
||||
final _partnerAddressController = TextEditingController();
|
||||
final _partnerCityController = TextEditingController();
|
||||
final _partnerPostalCodeController = TextEditingController();
|
||||
|
||||
// Liste des enfants
|
||||
final List<ChildData> _children = [ChildData()];
|
||||
|
||||
final _motivationController = TextEditingController();
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_emailController.dispose();
|
||||
_passwordController.dispose();
|
||||
_firstNameController.dispose();
|
||||
_lastNameController.dispose();
|
||||
_phoneController.dispose();
|
||||
_addressController.dispose();
|
||||
_cityController.dispose();
|
||||
_postalCodeController.dispose();
|
||||
_presentationController.dispose();
|
||||
_partnerFirstNameController.dispose();
|
||||
_partnerLastNameController.dispose();
|
||||
_partnerEmailController.dispose();
|
||||
_partnerPhoneController.dispose();
|
||||
_partnerAddressController.dispose();
|
||||
_partnerCityController.dispose();
|
||||
_partnerPostalCodeController.dispose();
|
||||
for (var child in _children) {
|
||||
child.firstNameController.dispose();
|
||||
child.lastNameController.dispose();
|
||||
}
|
||||
_motivationController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
Future<void> _pickImage(ChildData child) async {
|
||||
final ImagePicker picker = ImagePicker();
|
||||
final XFile? image = await picker.pickImage(source: ImageSource.gallery);
|
||||
|
||||
if (image != null) {
|
||||
setState(() {
|
||||
child.photo = image;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
Future<String?> _uploadImage(XFile image, String userId) async {
|
||||
// En mode démonstration, on retourne juste un chemin local
|
||||
return image.path;
|
||||
}
|
||||
|
||||
Future<void> _register() async {
|
||||
if (!_formKey.currentState!.validate()) return;
|
||||
|
||||
setState(() => _isLoading = true);
|
||||
|
||||
try {
|
||||
final List<Map<String, dynamic>> childrenData = [];
|
||||
|
||||
for (var child in _children) {
|
||||
childrenData.add({
|
||||
'firstName': child.firstNameController.text,
|
||||
'lastName': child.lastNameController.text,
|
||||
'birthDate': child.isUnborn ? null : child.birthDate,
|
||||
'expectedBirthDate': child.isUnborn ? child.expectedBirthDate : null,
|
||||
'photo': child.photo != null ? base64Encode(child.photo!.readAsBytesSync()) : null,
|
||||
'hasPhotoConsent': child.hasPhotoConsent,
|
||||
'isMultipleBirth': child.isMultipleBirth,
|
||||
});
|
||||
}
|
||||
|
||||
await _authService.registerParent(
|
||||
email: _emailController.text,
|
||||
password: _passwordController.text,
|
||||
firstName: _firstNameController.text,
|
||||
lastName: _lastNameController.text,
|
||||
phoneNumber: _phoneController.text,
|
||||
address: _addressController.text,
|
||||
city: _cityController.text,
|
||||
postalCode: _postalCodeController.text,
|
||||
presentation: _presentationController.text,
|
||||
hasAcceptedCGU: _hasAcceptedCGU,
|
||||
partnerFirstName: _hasPartner ? _partnerFirstNameController.text : null,
|
||||
partnerLastName: _hasPartner ? _partnerLastNameController.text : null,
|
||||
partnerEmail: _hasPartner ? _partnerEmailController.text : null,
|
||||
partnerPhoneNumber: _hasPartner ? _partnerPhoneController.text : null,
|
||||
partnerAddress: _hasPartner
|
||||
? (_partnerSameAddress
|
||||
? _addressController.text
|
||||
: _partnerAddressController.text)
|
||||
: null,
|
||||
partnerCity: _hasPartner
|
||||
? (_partnerSameAddress ? _cityController.text : _partnerCityController.text)
|
||||
: null,
|
||||
partnerPostalCode: _hasPartner
|
||||
? (_partnerSameAddress ? _postalCodeController.text : _partnerPostalCodeController.text)
|
||||
: null,
|
||||
children: childrenData,
|
||||
motivation: _motivationController.text,
|
||||
);
|
||||
|
||||
if (mounted) {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
const SnackBar(
|
||||
content: Text('Inscription réussie ! Votre compte est en attente de validation.'),
|
||||
backgroundColor: Colors.green,
|
||||
),
|
||||
);
|
||||
context.pop();
|
||||
}
|
||||
} catch (e) {
|
||||
if (mounted) {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(
|
||||
content: Text('Erreur lors de l\'inscription: $e'),
|
||||
backgroundColor: Colors.red,
|
||||
),
|
||||
);
|
||||
}
|
||||
} finally {
|
||||
if (mounted) {
|
||||
setState(() => _isLoading = false);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Widget _buildChildForm(ChildData child, int index) {
|
||||
return Card(
|
||||
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
Text(
|
||||
'Enfant ${index + 1}',
|
||||
style: Theme.of(context).textTheme.titleMedium,
|
||||
),
|
||||
if (index > 0)
|
||||
IconButton(
|
||||
icon: const Icon(Icons.delete),
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
_children.removeAt(index);
|
||||
});
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
if (child.photo != null)
|
||||
CircleAvatar(
|
||||
radius: 50,
|
||||
backgroundImage: NetworkImage(child.photo!.path),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => _pickImage(child),
|
||||
child: const Text('Ajouter une photo'),
|
||||
),
|
||||
SwitchListTile(
|
||||
title: const Text('Enfant à naître'),
|
||||
value: child.isUnborn,
|
||||
onChanged: (value) => setState(() => child.isUnborn = value),
|
||||
),
|
||||
TextFormField(
|
||||
controller: child.firstNameController,
|
||||
decoration: const InputDecoration(labelText: 'Prénom de l\'enfant'),
|
||||
),
|
||||
TextFormField(
|
||||
controller: child.lastNameController,
|
||||
decoration: const InputDecoration(labelText: 'Nom de l\'enfant'),
|
||||
),
|
||||
if (!child.isUnborn)
|
||||
ListTile(
|
||||
title: const Text('Date de naissance'),
|
||||
subtitle: Text(child.birthDate != null
|
||||
? '${child.birthDate!.day}/${child.birthDate!.month}/${child.birthDate!.year}'
|
||||
: 'Non définie'),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.calendar_today),
|
||||
onPressed: () async {
|
||||
final date = await showDatePicker(
|
||||
context: context,
|
||||
initialDate: DateTime.now(),
|
||||
firstDate: DateTime(2000),
|
||||
lastDate: DateTime.now(),
|
||||
);
|
||||
if (date != null) {
|
||||
setState(() => child.birthDate = date);
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
if (child.isUnborn)
|
||||
ListTile(
|
||||
title: const Text('Date prévue'),
|
||||
subtitle: Text(child.expectedBirthDate != null
|
||||
? '${child.expectedBirthDate!.day}/${child.expectedBirthDate!.month}/${child.expectedBirthDate!.year}'
|
||||
: 'Non définie'),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.calendar_today),
|
||||
onPressed: () async {
|
||||
final date = await showDatePicker(
|
||||
context: context,
|
||||
initialDate: DateTime.now(),
|
||||
firstDate: DateTime.now(),
|
||||
lastDate: DateTime.now().add(const Duration(days: 365)),
|
||||
);
|
||||
if (date != null) {
|
||||
setState(() => child.expectedBirthDate = date);
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
SwitchListTile(
|
||||
title: const Text('Naissance multiple'),
|
||||
subtitle: const Text('Jumeaux, triplés, etc.'),
|
||||
value: child.isMultipleBirth,
|
||||
onChanged: (value) => setState(() => child.isMultipleBirth = value),
|
||||
),
|
||||
if (child.photo != null)
|
||||
SwitchListTile(
|
||||
title: const Text('Consentement photo'),
|
||||
subtitle: const Text('J\'autorise l\'utilisation de la photo de mon enfant'),
|
||||
value: child.hasPhotoConsent,
|
||||
onChanged: (value) => setState(() => child.hasPhotoConsent = value),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
List<Step> _getSteps() {
|
||||
return [
|
||||
// Étape 1 : Parent 1
|
||||
Step(
|
||||
title: const Text('Informations parent 1'),
|
||||
content: Column(
|
||||
children: [
|
||||
TextFormField(
|
||||
controller: _emailController,
|
||||
decoration: const InputDecoration(labelText: 'Email'),
|
||||
keyboardType: TextInputType.emailAddress,
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer votre email';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _passwordController,
|
||||
decoration: const InputDecoration(labelText: 'Mot de passe'),
|
||||
obscureText: true,
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer un mot de passe';
|
||||
}
|
||||
if (value.length < 6) {
|
||||
return 'Le mot de passe doit contenir au moins 6 caractères';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _firstNameController,
|
||||
decoration: const InputDecoration(labelText: 'Prénom'),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer votre prénom';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _lastNameController,
|
||||
decoration: const InputDecoration(labelText: 'Nom'),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer votre nom';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _phoneController,
|
||||
decoration: const InputDecoration(labelText: 'Téléphone'),
|
||||
keyboardType: TextInputType.phone,
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer votre numéro de téléphone';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _addressController,
|
||||
decoration: const InputDecoration(labelText: 'Adresse'),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer votre adresse';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _cityController,
|
||||
decoration: const InputDecoration(labelText: 'Ville'),
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer votre ville';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _postalCodeController,
|
||||
decoration: const InputDecoration(labelText: 'Code postal'),
|
||||
keyboardType: TextInputType.number,
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez entrer votre code postal';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _presentationController,
|
||||
decoration: const InputDecoration(labelText: 'Présentation'),
|
||||
maxLines: 3,
|
||||
),
|
||||
],
|
||||
),
|
||||
isActive: _currentStep >= 0,
|
||||
),
|
||||
// Étape 2 : Parent 2
|
||||
Step(
|
||||
title: const Text('Parent 2'),
|
||||
content: Column(
|
||||
children: [
|
||||
SwitchListTile(
|
||||
title: const Text('Ajouter un deuxième parent'),
|
||||
value: _hasPartner,
|
||||
onChanged: (value) => setState(() => _hasPartner = value),
|
||||
),
|
||||
if (_hasPartner) ...[
|
||||
SwitchListTile(
|
||||
title: const Text('Adresse identique au parent 1'),
|
||||
value: _partnerSameAddress,
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_partnerSameAddress = value;
|
||||
if (value) {
|
||||
_partnerAddressController.text = _addressController.text;
|
||||
_partnerCityController.text = _cityController.text;
|
||||
_partnerPostalCodeController.text = _postalCodeController.text;
|
||||
} else {
|
||||
_partnerAddressController.clear();
|
||||
_partnerCityController.clear();
|
||||
_partnerPostalCodeController.clear();
|
||||
}
|
||||
});
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _partnerFirstNameController,
|
||||
decoration: const InputDecoration(labelText: 'Prénom du deuxième parent'),
|
||||
validator: (value) {
|
||||
if (_hasPartner && (value == null || value.isEmpty)) {
|
||||
return 'Veuillez entrer le prénom';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _partnerLastNameController,
|
||||
decoration: const InputDecoration(labelText: 'Nom du deuxième parent'),
|
||||
validator: (value) {
|
||||
if (_hasPartner && (value == null || value.isEmpty)) {
|
||||
return 'Veuillez entrer le nom';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _partnerEmailController,
|
||||
decoration: const InputDecoration(labelText: 'Email du deuxième parent'),
|
||||
keyboardType: TextInputType.emailAddress,
|
||||
validator: (value) {
|
||||
if (_hasPartner && (value == null || value.isEmpty)) {
|
||||
return 'Veuillez entrer l\'email';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _partnerPhoneController,
|
||||
decoration: const InputDecoration(labelText: 'Téléphone du deuxième parent'),
|
||||
keyboardType: TextInputType.phone,
|
||||
),
|
||||
if (!_partnerSameAddress) ...[
|
||||
TextFormField(
|
||||
controller: _partnerAddressController,
|
||||
decoration: const InputDecoration(labelText: 'Adresse du deuxième parent'),
|
||||
validator: (value) {
|
||||
if (_hasPartner && !_partnerSameAddress && (value == null || value.isEmpty)) {
|
||||
return 'Veuillez entrer l\'adresse';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _partnerCityController,
|
||||
decoration: const InputDecoration(labelText: 'Ville du deuxième parent'),
|
||||
validator: (value) {
|
||||
if (_hasPartner && !_partnerSameAddress && (value == null || value.isEmpty)) {
|
||||
return 'Veuillez entrer la ville';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _partnerPostalCodeController,
|
||||
decoration: const InputDecoration(labelText: 'Code postal du deuxième parent'),
|
||||
keyboardType: TextInputType.number,
|
||||
validator: (value) {
|
||||
if (_hasPartner && !_partnerSameAddress && (value == null || value.isEmpty)) {
|
||||
return 'Veuillez entrer le code postal';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
],
|
||||
],
|
||||
],
|
||||
),
|
||||
isActive: _currentStep >= 1,
|
||||
),
|
||||
// Étape 3 : Enfants
|
||||
Step(
|
||||
title: const Text('Enfants'),
|
||||
content: Column(
|
||||
children: [
|
||||
..._children.asMap().entries.map((entry) => _buildChildForm(entry.value, entry.key)),
|
||||
const SizedBox(height: 16),
|
||||
ElevatedButton.icon(
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
_children.add(ChildData());
|
||||
});
|
||||
},
|
||||
icon: const Icon(Icons.add),
|
||||
label: const Text('Ajouter un autre enfant'),
|
||||
),
|
||||
],
|
||||
),
|
||||
isActive: _currentStep >= 2,
|
||||
),
|
||||
// Étape 4 : Description de la situation
|
||||
Step(
|
||||
title: const Text('Description de votre situation'),
|
||||
content: Column(
|
||||
children: [
|
||||
TextFormField(
|
||||
controller: _motivationController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Décrivez votre situation',
|
||||
hintText: 'Expliquez-nous votre situation familiale et vos besoins...',
|
||||
),
|
||||
maxLines: 5,
|
||||
validator: (value) {
|
||||
if (value == null || value.isEmpty) {
|
||||
return 'Veuillez nous décrire votre situation';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
isActive: _currentStep >= 3,
|
||||
),
|
||||
// Étape 5 : CGU
|
||||
Step(
|
||||
title: const Text('Conditions générales'),
|
||||
content: Column(
|
||||
children: [
|
||||
SwitchListTile(
|
||||
title: const Text('Conditions générales'),
|
||||
subtitle: const Text('J\'accepte les conditions générales d\'utilisation'),
|
||||
value: _hasAcceptedCGU,
|
||||
onChanged: (value) => setState(() => _hasAcceptedCGU = value),
|
||||
),
|
||||
if (!_hasAcceptedCGU)
|
||||
const Text(
|
||||
'Vous devez accepter les conditions générales pour continuer',
|
||||
style: TextStyle(color: Colors.red),
|
||||
),
|
||||
],
|
||||
),
|
||||
isActive: _currentStep >= 4,
|
||||
),
|
||||
// Étape 6 : Résumé
|
||||
Step(
|
||||
title: const Text('Résumé'),
|
||||
content: SingleChildScrollView(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
const Text(
|
||||
'Veuillez vérifier vos informations avant validation :',
|
||||
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16),
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
// Parent 1
|
||||
const Text('Parent 1', style: TextStyle(fontWeight: FontWeight.bold)),
|
||||
ListTile(
|
||||
title: const Text('Email'),
|
||||
subtitle: Text(_emailController.text),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 0),
|
||||
),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Nom complet'),
|
||||
subtitle: Text('${_firstNameController.text} ${_lastNameController.text}'),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 0),
|
||||
),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Adresse'),
|
||||
subtitle: Text('${_addressController.text}\n${_postalCodeController.text} ${_cityController.text}'),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 0),
|
||||
),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Téléphone'),
|
||||
subtitle: Text(_phoneController.text),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 0),
|
||||
),
|
||||
),
|
||||
if (_presentationController.text.isNotEmpty)
|
||||
ListTile(
|
||||
title: const Text('Présentation'),
|
||||
subtitle: Text(_presentationController.text),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 0),
|
||||
),
|
||||
),
|
||||
|
||||
// Parent 2
|
||||
if (_hasPartner) ...[
|
||||
const SizedBox(height: 16),
|
||||
const Text('Parent 2', style: TextStyle(fontWeight: FontWeight.bold)),
|
||||
ListTile(
|
||||
title: const Text('Email'),
|
||||
subtitle: Text(_partnerEmailController.text),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 1),
|
||||
),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Nom complet'),
|
||||
subtitle: Text('${_partnerFirstNameController.text} ${_partnerLastNameController.text}'),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 1),
|
||||
),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Téléphone'),
|
||||
subtitle: Text(_partnerPhoneController.text),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 1),
|
||||
),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Adresse'),
|
||||
subtitle: _partnerSameAddress
|
||||
? const Text('Identique au parent 1')
|
||||
: Text('${_partnerAddressController.text}\n${_partnerPostalCodeController.text} ${_partnerCityController.text}'),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 1),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
||||
// Enfants
|
||||
const SizedBox(height: 16),
|
||||
const Text('Enfants', style: TextStyle(fontWeight: FontWeight.bold)),
|
||||
..._children.asMap().entries.map((entry) {
|
||||
final child = entry.value;
|
||||
return ListTile(
|
||||
title: Text('Enfant ${entry.key + 1}'),
|
||||
subtitle: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text('Prénom : ${child.firstNameController.text} ${child.lastNameController.text}'),
|
||||
if (child.isUnborn)
|
||||
Text('Date prévue : ${child.expectedBirthDate?.day}/${child.expectedBirthDate?.month}/${child.expectedBirthDate?.year}')
|
||||
else
|
||||
Text('Date de naissance : ${child.birthDate?.day}/${child.birthDate?.month}/${child.birthDate?.year}'),
|
||||
if (child.isMultipleBirth)
|
||||
const Text('Naissance multiple'),
|
||||
],
|
||||
),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 2),
|
||||
),
|
||||
);
|
||||
}),
|
||||
|
||||
// Motivation
|
||||
const SizedBox(height: 16),
|
||||
const Text('Motivation', style: TextStyle(fontWeight: FontWeight.bold)),
|
||||
ListTile(
|
||||
title: const Text('Votre message'),
|
||||
subtitle: Text(_motivationController.text),
|
||||
trailing: IconButton(
|
||||
icon: const Icon(Icons.edit),
|
||||
onPressed: () => setState(() => _currentStep = 3),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
isActive: _currentStep >= 5,
|
||||
),
|
||||
];
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Inscription Parent'),
|
||||
),
|
||||
body: Form(
|
||||
key: _formKey,
|
||||
child: Stepper(
|
||||
currentStep: _currentStep,
|
||||
onStepContinue: () {
|
||||
if (_currentStep < _getSteps().length - 1) {
|
||||
setState(() => _currentStep++);
|
||||
} else if (_hasAcceptedCGU) {
|
||||
_register();
|
||||
}
|
||||
},
|
||||
onStepCancel: () {
|
||||
if (_currentStep > 0) {
|
||||
setState(() => _currentStep--);
|
||||
} else {
|
||||
Navigator.pop(context);
|
||||
}
|
||||
},
|
||||
controlsBuilder: (context, details) {
|
||||
return Padding(
|
||||
padding: const EdgeInsets.symmetric(vertical: 16.0),
|
||||
child: Row(
|
||||
children: [
|
||||
if (_currentStep > 0)
|
||||
OutlinedButton(
|
||||
onPressed: details.onStepCancel,
|
||||
child: const Text('Retour'),
|
||||
),
|
||||
const SizedBox(width: 16),
|
||||
if (_currentStep < _getSteps().length - 1)
|
||||
ElevatedButton(
|
||||
onPressed: details.onStepContinue,
|
||||
child: const Text('Suivant'),
|
||||
)
|
||||
else
|
||||
ElevatedButton(
|
||||
onPressed: _hasAcceptedCGU ? details.onStepContinue : null,
|
||||
child: _isLoading
|
||||
? const CircularProgressIndicator()
|
||||
: const Text('S\'inscrire'),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
},
|
||||
steps: _getSteps(),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user