feat: Mise à jour des interfaces mobiles v3.2.3
- Amélioration des interfaces utilisateur sur mobile - Optimisation de la responsivité des composants Flutter - Mise à jour des widgets de chat et communication - Amélioration des formulaires et tableaux - Ajout de nouveaux composants pour l'administration - Optimisation des thèmes et styles visuels 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -37,9 +37,6 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
final List<Map<String, dynamic>> _sectors = [];
|
||||
final List<Map<String, dynamic>> _passages = [];
|
||||
|
||||
// État du plein écran
|
||||
bool _isFullScreen = false;
|
||||
|
||||
// Items pour la combobox de secteurs
|
||||
List<DropdownMenuItem<int?>> _sectorItems = [];
|
||||
|
||||
@@ -567,32 +564,12 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final theme = Theme.of(context);
|
||||
final size = MediaQuery.of(context).size;
|
||||
final isDesktop = size.width > 900;
|
||||
|
||||
return Scaffold(
|
||||
backgroundColor: Colors.transparent,
|
||||
body: SafeArea(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
// En-tête - affiché uniquement si pas en plein écran
|
||||
if (!_isFullScreen)
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: Text(
|
||||
'Carte des passages',
|
||||
style: theme.textTheme.headlineMedium?.copyWith(
|
||||
fontWeight: FontWeight.bold,
|
||||
color: theme.colorScheme.primary,
|
||||
),
|
||||
),
|
||||
),
|
||||
|
||||
// Filtres - affichés uniquement si pas en plein écran
|
||||
if (!_isFullScreen) _buildFilters(theme, isDesktop),
|
||||
|
||||
// Carte
|
||||
Expanded(
|
||||
child: Stack(
|
||||
@@ -606,7 +583,7 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
useOpenStreetMap: !kIsWeb,
|
||||
markers: _buildPassageMarkers(),
|
||||
polygons: _buildSectorPolygons(),
|
||||
showControls: true,
|
||||
showControls: false, // Désactiver les contrôles par défaut pour éviter la duplication
|
||||
onMapEvent: (event) {
|
||||
if (event is MapEventMove) {
|
||||
// Mettre à jour la position et le zoom actuels
|
||||
@@ -632,7 +609,7 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
width:
|
||||
220, // Largeur fixe pour accommoder les noms longs
|
||||
decoration: BoxDecoration(
|
||||
color: Colors.white.withOpacity(0.95),
|
||||
color: Colors.white.withValues(alpha: 0.95),
|
||||
borderRadius: BorderRadius.circular(8),
|
||||
),
|
||||
child: Row(
|
||||
@@ -673,31 +650,148 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
),
|
||||
),
|
||||
|
||||
// Bouton de plein écran (les autres contrôles sont gérés par MapboxMap)
|
||||
// Contrôles de zoom et localisation en bas à droite
|
||||
Positioned(
|
||||
bottom: 16.0,
|
||||
right: 16.0,
|
||||
child: _buildMapButton(
|
||||
icon: _isFullScreen
|
||||
? Icons.fullscreen_exit
|
||||
: Icons.fullscreen,
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
_isFullScreen = !_isFullScreen;
|
||||
});
|
||||
},
|
||||
child: Column(
|
||||
children: [
|
||||
// Bouton zoom +
|
||||
_buildMapButton(
|
||||
icon: Icons.add,
|
||||
onPressed: () {
|
||||
final newZoom = _currentZoom + 1;
|
||||
_mapController.move(_currentPosition, newZoom);
|
||||
setState(() {
|
||||
_currentZoom = newZoom;
|
||||
});
|
||||
_saveSettings();
|
||||
},
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
// Bouton zoom -
|
||||
_buildMapButton(
|
||||
icon: Icons.remove,
|
||||
onPressed: () {
|
||||
final newZoom = _currentZoom - 1;
|
||||
_mapController.move(_currentPosition, newZoom);
|
||||
setState(() {
|
||||
_currentZoom = newZoom;
|
||||
});
|
||||
_saveSettings();
|
||||
},
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
// Bouton de localisation
|
||||
_buildMapButton(
|
||||
icon: Icons.my_location,
|
||||
onPressed: () {
|
||||
_getUserLocation();
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
|
||||
// Bouton de localisation personnalisé (pour utiliser notre propre logique)
|
||||
// Filtres de type de passage en bas à gauche
|
||||
Positioned(
|
||||
bottom: 80.0, // Positionné au-dessus du bouton plein écran
|
||||
right: 16.0,
|
||||
child: _buildMapButton(
|
||||
icon: Icons.my_location,
|
||||
onPressed: () {
|
||||
_getUserLocation();
|
||||
},
|
||||
bottom: 16.0,
|
||||
left: 16.0,
|
||||
child: Container(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 6),
|
||||
decoration: BoxDecoration(
|
||||
color: Colors.white.withValues(alpha: 0.7),
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Colors.black.withValues(alpha: 0.2),
|
||||
blurRadius: 6,
|
||||
offset: const Offset(0, 3),
|
||||
),
|
||||
],
|
||||
),
|
||||
child: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
// Filtre Effectués (type 1)
|
||||
_buildFilterDot(
|
||||
color: Color(AppKeys.typesPassages[1]?['couleur2'] as int),
|
||||
selected: _showEffectues,
|
||||
onTap: () {
|
||||
setState(() {
|
||||
_showEffectues = !_showEffectues;
|
||||
_loadPassages();
|
||||
_saveSettings();
|
||||
});
|
||||
},
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
// Filtre À finaliser (type 2)
|
||||
_buildFilterDot(
|
||||
color: Color(AppKeys.typesPassages[2]?['couleur2'] as int),
|
||||
selected: _showAFinaliser,
|
||||
onTap: () {
|
||||
setState(() {
|
||||
_showAFinaliser = !_showAFinaliser;
|
||||
_loadPassages();
|
||||
_saveSettings();
|
||||
});
|
||||
},
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
// Filtre Refusés (type 3)
|
||||
_buildFilterDot(
|
||||
color: Color(AppKeys.typesPassages[3]?['couleur2'] as int),
|
||||
selected: _showRefuses,
|
||||
onTap: () {
|
||||
setState(() {
|
||||
_showRefuses = !_showRefuses;
|
||||
_loadPassages();
|
||||
_saveSettings();
|
||||
});
|
||||
},
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
// Filtre Dons (type 4)
|
||||
_buildFilterDot(
|
||||
color: Color(AppKeys.typesPassages[4]?['couleur2'] as int),
|
||||
selected: _showDons,
|
||||
onTap: () {
|
||||
setState(() {
|
||||
_showDons = !_showDons;
|
||||
_loadPassages();
|
||||
_saveSettings();
|
||||
});
|
||||
},
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
// Filtre Lots (type 5)
|
||||
_buildFilterDot(
|
||||
color: Color(AppKeys.typesPassages[5]?['couleur2'] as int),
|
||||
selected: _showLots,
|
||||
onTap: () {
|
||||
setState(() {
|
||||
_showLots = !_showLots;
|
||||
_loadPassages();
|
||||
_saveSettings();
|
||||
});
|
||||
},
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
// Filtre Maisons vides (type 6)
|
||||
_buildFilterDot(
|
||||
color: Color(AppKeys.typesPassages[6]?['couleur2'] as int),
|
||||
selected: _showMaisonsVides,
|
||||
onTap: () {
|
||||
setState(() {
|
||||
_showMaisonsVides = !_showMaisonsVides;
|
||||
_loadPassages();
|
||||
_saveSettings();
|
||||
});
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
@@ -709,145 +803,26 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
);
|
||||
}
|
||||
|
||||
// Construire les filtres pour les passages
|
||||
Widget _buildFilters(ThemeData theme, bool isDesktop) {
|
||||
return Padding(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Wrap(
|
||||
spacing: 8.0,
|
||||
runSpacing: 8.0,
|
||||
children: [
|
||||
// Filtre pour les passages effectués
|
||||
_buildFilterChip(
|
||||
label: AppKeys.typesPassages[1]?['titres'] as String? ??
|
||||
'Effectués',
|
||||
selected: _showEffectues,
|
||||
color: Color(AppKeys.typesPassages[1]?['couleur2'] as int),
|
||||
onSelected: (selected) {
|
||||
setState(() {
|
||||
_showEffectues = selected;
|
||||
_loadPassages(); // Recharger les passages avec le nouveau filtre
|
||||
_saveSettings(); // Sauvegarder les préférences
|
||||
});
|
||||
},
|
||||
),
|
||||
|
||||
// Filtre pour les passages à finaliser
|
||||
_buildFilterChip(
|
||||
label: AppKeys.typesPassages[2]?['titres'] as String? ??
|
||||
'À finaliser',
|
||||
selected: _showAFinaliser,
|
||||
color: Color(AppKeys.typesPassages[2]?['couleur2'] as int),
|
||||
onSelected: (selected) {
|
||||
setState(() {
|
||||
_showAFinaliser = selected;
|
||||
_loadPassages(); // Recharger les passages avec le nouveau filtre
|
||||
_saveSettings(); // Sauvegarder les préférences
|
||||
});
|
||||
},
|
||||
),
|
||||
|
||||
// Filtre pour les passages refusés
|
||||
_buildFilterChip(
|
||||
label:
|
||||
AppKeys.typesPassages[3]?['titres'] as String? ?? 'Refusés',
|
||||
selected: _showRefuses,
|
||||
color: Color(AppKeys.typesPassages[3]?['couleur2'] as int),
|
||||
onSelected: (selected) {
|
||||
setState(() {
|
||||
_showRefuses = selected;
|
||||
_loadPassages(); // Recharger les passages avec le nouveau filtre
|
||||
_saveSettings(); // Sauvegarder les préférences
|
||||
});
|
||||
},
|
||||
),
|
||||
|
||||
// Filtre pour les dons
|
||||
_buildFilterChip(
|
||||
label: AppKeys.typesPassages[4]?['titres'] as String? ?? 'Dons',
|
||||
selected: _showDons,
|
||||
color: Color(AppKeys.typesPassages[4]?['couleur2'] as int),
|
||||
onSelected: (selected) {
|
||||
setState(() {
|
||||
_showDons = selected;
|
||||
_loadPassages(); // Recharger les passages avec le nouveau filtre
|
||||
_saveSettings(); // Sauvegarder les préférences
|
||||
});
|
||||
},
|
||||
),
|
||||
|
||||
// Filtre pour les lots
|
||||
_buildFilterChip(
|
||||
label: AppKeys.typesPassages[5]?['titres'] as String? ?? 'Lots',
|
||||
selected: _showLots,
|
||||
color: Color(AppKeys.typesPassages[5]?['couleur2'] as int),
|
||||
onSelected: (selected) {
|
||||
setState(() {
|
||||
_showLots = selected;
|
||||
_loadPassages(); // Recharger les passages avec le nouveau filtre
|
||||
_saveSettings(); // Sauvegarder les préférences
|
||||
});
|
||||
},
|
||||
),
|
||||
|
||||
// Filtre pour les maisons vides
|
||||
_buildFilterChip(
|
||||
label: AppKeys.typesPassages[6]?['titres'] as String? ??
|
||||
'Maisons vides',
|
||||
selected: _showMaisonsVides,
|
||||
color: Color(AppKeys.typesPassages[6]?['couleur2'] as int),
|
||||
onSelected: (selected) {
|
||||
setState(() {
|
||||
_showMaisonsVides = selected;
|
||||
_loadPassages(); // Recharger les passages avec le nouveau filtre
|
||||
_saveSettings(); // Sauvegarder les préférences
|
||||
});
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
// Construire un chip de filtre
|
||||
Widget _buildFilterChip({
|
||||
required String label,
|
||||
required bool selected,
|
||||
// Construire une pastille de filtre pour la carte
|
||||
Widget _buildFilterDot({
|
||||
required Color color,
|
||||
required Function(bool) onSelected,
|
||||
required bool selected,
|
||||
required VoidCallback onTap,
|
||||
}) {
|
||||
// Utiliser la couleur vive pour les boutons sélectionnés et une version plus terne pour les désélectionnés
|
||||
final Color avatarColor = selected ? color : color.withOpacity(0.4);
|
||||
final Color chipColor =
|
||||
selected ? color.withOpacity(0.2) : Colors.grey.withOpacity(0.1);
|
||||
|
||||
return FilterChip(
|
||||
label: Text(
|
||||
label,
|
||||
style: TextStyle(
|
||||
fontWeight: selected ? FontWeight.bold : FontWeight.normal,
|
||||
color: selected ? Colors.black : Colors.black54,
|
||||
return GestureDetector(
|
||||
onTap: onTap,
|
||||
child: Container(
|
||||
width: 24,
|
||||
height: 24,
|
||||
decoration: BoxDecoration(
|
||||
color: selected ? color : color.withValues(alpha: 0.3),
|
||||
shape: BoxShape.circle,
|
||||
border: Border.all(
|
||||
color: selected ? Colors.white : Colors.white.withValues(alpha: 0.5),
|
||||
width: 1.5,
|
||||
),
|
||||
),
|
||||
),
|
||||
selected: selected,
|
||||
showCheckmark: false,
|
||||
avatar: CircleAvatar(
|
||||
backgroundColor: avatarColor,
|
||||
radius: 10.0,
|
||||
),
|
||||
backgroundColor: Colors.white,
|
||||
selectedColor: chipColor,
|
||||
side: BorderSide(
|
||||
color: selected ? color : Colors.grey.withOpacity(0.3),
|
||||
width: selected ? 1.5 : 1.0,
|
||||
),
|
||||
padding: const EdgeInsets.symmetric(horizontal: 8.0, vertical: 4.0),
|
||||
onSelected: onSelected,
|
||||
);
|
||||
}
|
||||
|
||||
@@ -864,7 +839,7 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
shape: BoxShape.circle,
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Colors.black.withOpacity(0.2),
|
||||
color: Colors.black.withValues(alpha: 0.2),
|
||||
blurRadius: 6,
|
||||
offset: const Offset(0, 3),
|
||||
),
|
||||
@@ -918,8 +893,8 @@ class _UserMapPageState extends State<UserMapPage> {
|
||||
return _sectors.map((sector) {
|
||||
return Polygon(
|
||||
points: sector['points'] as List<LatLng>,
|
||||
color: (sector['color'] as Color).withOpacity(0.3),
|
||||
borderColor: (sector['color'] as Color).withOpacity(1.0),
|
||||
color: (sector['color'] as Color).withValues(alpha: 0.3),
|
||||
borderColor: (sector['color'] as Color).withValues(alpha: 1.0),
|
||||
borderStrokeWidth: 2.0,
|
||||
);
|
||||
}).toList();
|
||||
|
||||
Reference in New Issue
Block a user