feat: Introduce memberIds for efficient group querying and management, updating related UI components and .gitignore.
This commit is contained in:
4
.gitignore
vendored
4
.gitignore
vendored
@@ -44,7 +44,9 @@ app.*.map.json
|
|||||||
/android/app/profile
|
/android/app/profile
|
||||||
/android/app/release
|
/android/app/release
|
||||||
.vscode
|
.vscode
|
||||||
.VSCodeCounter/
|
.VSCodeCounter/*
|
||||||
.env
|
.env
|
||||||
.env.local
|
.env.local
|
||||||
.env.*.local
|
.env.*.local
|
||||||
|
firestore.rules
|
||||||
|
storage.rules
|
||||||
|
|||||||
@@ -36,6 +36,37 @@ class _CalendarPageState extends State<CalendarPage> {
|
|||||||
}).toList();
|
}).toList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> _selectTimeAndSchedule(Activity activity, DateTime date) async {
|
||||||
|
final TimeOfDay? pickedTime = await showTimePicker(
|
||||||
|
context: context,
|
||||||
|
initialTime: TimeOfDay.now(),
|
||||||
|
builder: (BuildContext context, Widget? child) {
|
||||||
|
return MediaQuery(
|
||||||
|
data: MediaQuery.of(context).copyWith(alwaysUse24HourFormat: true),
|
||||||
|
child: child!,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if (pickedTime != null && mounted) {
|
||||||
|
final scheduledDate = DateTime(
|
||||||
|
date.year,
|
||||||
|
date.month,
|
||||||
|
date.day,
|
||||||
|
pickedTime.hour,
|
||||||
|
pickedTime.minute,
|
||||||
|
);
|
||||||
|
|
||||||
|
context.read<ActivityBloc>().add(
|
||||||
|
UpdateActivityDate(
|
||||||
|
tripId: widget.trip.id!,
|
||||||
|
activityId: activity.id,
|
||||||
|
date: scheduledDate,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final theme = Theme.of(context);
|
final theme = Theme.of(context);
|
||||||
@@ -172,7 +203,9 @@ class _CalendarPageState extends State<CalendarPage> {
|
|||||||
selectedActivities[index];
|
selectedActivities[index];
|
||||||
return ListTile(
|
return ListTile(
|
||||||
title: Text(activity.name),
|
title: Text(activity.name),
|
||||||
subtitle: Text(activity.category),
|
subtitle: Text(
|
||||||
|
'${activity.category} - ${DateFormat('HH:mm').format(activity.date!)}',
|
||||||
|
),
|
||||||
trailing: IconButton(
|
trailing: IconButton(
|
||||||
icon: const Icon(Icons.close),
|
icon: const Icon(Icons.close),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
@@ -243,15 +276,10 @@ class _CalendarPageState extends State<CalendarPage> {
|
|||||||
icon: const Icon(Icons.arrow_back),
|
icon: const Icon(Icons.arrow_back),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
if (_selectedDay != null) {
|
if (_selectedDay != null) {
|
||||||
context
|
_selectTimeAndSchedule(
|
||||||
.read<ActivityBloc>()
|
activity,
|
||||||
.add(
|
_selectedDay!,
|
||||||
UpdateActivityDate(
|
);
|
||||||
tripId: widget.trip.id!,
|
|
||||||
activityId: activity.id,
|
|
||||||
date: _selectedDay,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
@@ -260,6 +288,33 @@ class _CalendarPageState extends State<CalendarPage> {
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
// Zone de drop pour le calendrier
|
||||||
|
DragTarget<Activity>(
|
||||||
|
onWillAccept: (data) => true,
|
||||||
|
onAccept: (activity) {
|
||||||
|
if (_selectedDay != null) {
|
||||||
|
_selectTimeAndSchedule(activity, _selectedDay!);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
builder: (context, candidateData, rejectedData) {
|
||||||
|
return Container(
|
||||||
|
height: 50,
|
||||||
|
color: candidateData.isNotEmpty
|
||||||
|
? theme.colorScheme.primary.withValues(
|
||||||
|
alpha: 0.1,
|
||||||
|
)
|
||||||
|
: null,
|
||||||
|
child: Center(
|
||||||
|
child: Text(
|
||||||
|
'Glisser ici pour planifier',
|
||||||
|
style: TextStyle(
|
||||||
|
color: theme.colorScheme.primary,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -455,6 +455,9 @@ class _ShowTripDetailsContentState extends State<ShowTripDetailsContent> {
|
|||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
const SizedBox(height: 32),
|
||||||
|
_buildNextActivitiesSection(),
|
||||||
|
_buildExpensesCard(),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -654,52 +657,63 @@ class _ShowTripDetailsContentState extends State<ShowTripDetailsContent> {
|
|||||||
|
|
||||||
return FutureBuilder(
|
return FutureBuilder(
|
||||||
future: _groupRepository.getGroupByTripId(widget.trip.id!),
|
future: _groupRepository.getGroupByTripId(widget.trip.id!),
|
||||||
builder: (context, snapshot) {
|
builder: (context, groupSnapshot) {
|
||||||
// En attente
|
if (groupSnapshot.connectionState == ConnectionState.waiting) {
|
||||||
if (snapshot.connectionState == ConnectionState.waiting) {
|
|
||||||
return const Center(child: CircularProgressIndicator());
|
return const Center(child: CircularProgressIndicator());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Erreur
|
if (groupSnapshot.hasError ||
|
||||||
if (snapshot.hasError) {
|
!groupSnapshot.hasData ||
|
||||||
return Center(
|
groupSnapshot.data == null) {
|
||||||
child: Text(
|
|
||||||
'Erreur: ${snapshot.error}',
|
|
||||||
style: TextStyle(color: Colors.red),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Pas de groupe trouvé
|
|
||||||
if (!snapshot.hasData || snapshot.data == null) {
|
|
||||||
return const Center(child: Text('Aucun participant'));
|
return const Center(child: Text('Aucun participant'));
|
||||||
}
|
}
|
||||||
|
|
||||||
final group = snapshot.data!;
|
final groupId = groupSnapshot.data!.id;
|
||||||
final members = group.members;
|
|
||||||
|
|
||||||
if (members.isEmpty) {
|
return StreamBuilder<List<GroupMember>>(
|
||||||
return const Center(child: Text('Aucun participant'));
|
stream: _groupRepository.watchGroupMembers(groupId),
|
||||||
}
|
builder: (context, snapshot) {
|
||||||
|
// En attente
|
||||||
|
if (snapshot.connectionState == ConnectionState.waiting) {
|
||||||
|
return const Center(child: CircularProgressIndicator());
|
||||||
|
}
|
||||||
|
|
||||||
return SingleChildScrollView(
|
// Erreur
|
||||||
scrollDirection: Axis.horizontal,
|
if (snapshot.hasError) {
|
||||||
child: Row(
|
return Center(
|
||||||
children: [
|
child: Text(
|
||||||
...List.generate(members.length, (index) {
|
'Erreur: ${snapshot.error}',
|
||||||
final member = members[index];
|
style: TextStyle(color: Colors.red),
|
||||||
return Padding(
|
),
|
||||||
padding: const EdgeInsets.only(right: 12),
|
);
|
||||||
child: _buildParticipantAvatar(member),
|
}
|
||||||
);
|
|
||||||
}),
|
final members = snapshot.data ?? [];
|
||||||
// Bouton "+" pour ajouter un participant
|
|
||||||
Padding(
|
if (members.isEmpty) {
|
||||||
padding: const EdgeInsets.only(right: 12),
|
return const Center(child: Text('Aucun participant'));
|
||||||
child: _buildAddParticipantButton(),
|
}
|
||||||
|
|
||||||
|
return SingleChildScrollView(
|
||||||
|
scrollDirection: Axis.horizontal,
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
...List.generate(members.length, (index) {
|
||||||
|
final member = members[index];
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 12),
|
||||||
|
child: _buildParticipantAvatar(member),
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
// Bouton "+" pour ajouter un participant
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 12),
|
||||||
|
child: _buildAddParticipantButton(),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
],
|
);
|
||||||
),
|
},
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@@ -950,4 +964,175 @@ class _ShowTripDetailsContentState extends State<ShowTripDetailsContent> {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildNextActivitiesSection() {
|
||||||
|
final theme = Theme.of(context);
|
||||||
|
return Column(
|
||||||
|
children: [
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
'Prochaines activités',
|
||||||
|
style: theme.textTheme.titleMedium?.copyWith(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: theme.colorScheme.onSurface,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => _navigateToActivities(),
|
||||||
|
child: Text(
|
||||||
|
'Tout voir',
|
||||||
|
style: TextStyle(
|
||||||
|
color: Colors.teal,
|
||||||
|
fontWeight: FontWeight.w600,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
_buildActivityCard(
|
||||||
|
title: 'Visite du Colisée',
|
||||||
|
date: '11 août, 10:00',
|
||||||
|
icon: Icons.museum,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 12),
|
||||||
|
_buildActivityCard(
|
||||||
|
title: 'Dîner à Trastevere',
|
||||||
|
date: '11 août, 20:30',
|
||||||
|
icon: Icons.restaurant,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildActivityCard({
|
||||||
|
required String title,
|
||||||
|
required String date,
|
||||||
|
required IconData icon,
|
||||||
|
}) {
|
||||||
|
final theme = Theme.of(context);
|
||||||
|
final isDarkMode = theme.brightness == Brightness.dark;
|
||||||
|
|
||||||
|
return Container(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: theme.cardColor,
|
||||||
|
borderRadius: BorderRadius.circular(16),
|
||||||
|
border: Border.all(
|
||||||
|
color: isDarkMode
|
||||||
|
? Colors.white.withValues(alpha: 0.1)
|
||||||
|
: Colors.black.withValues(alpha: 0.05),
|
||||||
|
width: 1,
|
||||||
|
),
|
||||||
|
boxShadow: [
|
||||||
|
BoxShadow(
|
||||||
|
color: Colors.black.withValues(alpha: 0.05),
|
||||||
|
blurRadius: 10,
|
||||||
|
offset: const Offset(0, 2),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
padding: const EdgeInsets.all(12),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.teal.withValues(alpha: 0.1),
|
||||||
|
borderRadius: BorderRadius.circular(12),
|
||||||
|
),
|
||||||
|
child: Icon(icon, color: Colors.teal, size: 24),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 16),
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
title,
|
||||||
|
style: theme.textTheme.bodyLarge?.copyWith(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: theme.colorScheme.onSurface,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Text(
|
||||||
|
date,
|
||||||
|
style: theme.textTheme.bodyMedium?.copyWith(
|
||||||
|
color: theme.colorScheme.onSurface.withValues(alpha: 0.6),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Icon(
|
||||||
|
Icons.chevron_right,
|
||||||
|
color: theme.colorScheme.onSurface.withValues(alpha: 0.4),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildExpensesCard() {
|
||||||
|
final theme = Theme.of(context);
|
||||||
|
|
||||||
|
return Container(
|
||||||
|
margin: const EdgeInsets.only(top: 24),
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: const Color(0xFFFDF4E3), // Light beige background
|
||||||
|
borderRadius: BorderRadius.circular(16),
|
||||||
|
),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
padding: const EdgeInsets.all(10),
|
||||||
|
decoration: const BoxDecoration(
|
||||||
|
color: Colors.orange,
|
||||||
|
shape: BoxShape.circle,
|
||||||
|
),
|
||||||
|
child: const Icon(
|
||||||
|
Icons.warning_amber_rounded,
|
||||||
|
color: Colors.white,
|
||||||
|
size: 24,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 16),
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
'Dépenses',
|
||||||
|
style: theme.textTheme.bodyLarge?.copyWith(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: const Color(0xFF5D4037), // Brown text
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Text(
|
||||||
|
'Vous devez 25€ à Clara',
|
||||||
|
style: theme.textTheme.bodyMedium?.copyWith(
|
||||||
|
color: const Color(0xFF8D6E63), // Lighter brown
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => _showComingSoon('Régler les dépenses'),
|
||||||
|
child: Text(
|
||||||
|
'Régler',
|
||||||
|
style: TextStyle(
|
||||||
|
color: const Color(0xFF5D4037),
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,6 +27,9 @@ class Group {
|
|||||||
/// List of members in this group
|
/// List of members in this group
|
||||||
final List<GroupMember> members;
|
final List<GroupMember> members;
|
||||||
|
|
||||||
|
/// List of member IDs for efficient querying and security rules
|
||||||
|
final List<String> memberIds;
|
||||||
|
|
||||||
/// Creates a new [Group] instance.
|
/// Creates a new [Group] instance.
|
||||||
///
|
///
|
||||||
/// [id], [name], [tripId], and [createdBy] are required.
|
/// [id], [name], [tripId], and [createdBy] are required.
|
||||||
@@ -40,9 +43,11 @@ class Group {
|
|||||||
DateTime? createdAt,
|
DateTime? createdAt,
|
||||||
DateTime? updatedAt,
|
DateTime? updatedAt,
|
||||||
List<GroupMember>? members,
|
List<GroupMember>? members,
|
||||||
}) : createdAt = createdAt ?? DateTime.now(),
|
List<String>? memberIds,
|
||||||
updatedAt = updatedAt ?? DateTime.now(),
|
}) : createdAt = createdAt ?? DateTime.now(),
|
||||||
members = members ?? [];
|
updatedAt = updatedAt ?? DateTime.now(),
|
||||||
|
members = members ?? [],
|
||||||
|
memberIds = memberIds ?? [];
|
||||||
|
|
||||||
/// Creates a [Group] instance from a Firestore document map.
|
/// Creates a [Group] instance from a Firestore document map.
|
||||||
///
|
///
|
||||||
@@ -59,6 +64,7 @@ class Group {
|
|||||||
createdAt: DateTime.fromMillisecondsSinceEpoch(map['createdAt'] ?? 0),
|
createdAt: DateTime.fromMillisecondsSinceEpoch(map['createdAt'] ?? 0),
|
||||||
updatedAt: DateTime.fromMillisecondsSinceEpoch(map['updatedAt'] ?? 0),
|
updatedAt: DateTime.fromMillisecondsSinceEpoch(map['updatedAt'] ?? 0),
|
||||||
members: [],
|
members: [],
|
||||||
|
memberIds: List<String>.from(map['memberIds'] ?? []),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,6 +75,7 @@ class Group {
|
|||||||
'createdBy': createdBy,
|
'createdBy': createdBy,
|
||||||
'createdAt': createdAt.millisecondsSinceEpoch,
|
'createdAt': createdAt.millisecondsSinceEpoch,
|
||||||
'updatedAt': updatedAt.millisecondsSinceEpoch,
|
'updatedAt': updatedAt.millisecondsSinceEpoch,
|
||||||
|
'memberIds': memberIds,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -80,6 +87,7 @@ class Group {
|
|||||||
DateTime? createdAt,
|
DateTime? createdAt,
|
||||||
DateTime? updatedAt,
|
DateTime? updatedAt,
|
||||||
List<GroupMember>? members,
|
List<GroupMember>? members,
|
||||||
|
List<String>? memberIds,
|
||||||
}) {
|
}) {
|
||||||
return Group(
|
return Group(
|
||||||
id: id ?? this.id,
|
id: id ?? this.id,
|
||||||
@@ -89,6 +97,7 @@ class Group {
|
|||||||
createdAt: createdAt ?? this.createdAt,
|
createdAt: createdAt ?? this.createdAt,
|
||||||
updatedAt: updatedAt ?? this.updatedAt,
|
updatedAt: updatedAt ?? this.updatedAt,
|
||||||
members: members ?? this.members,
|
members: members ?? this.members,
|
||||||
|
memberIds: memberIds ?? this.memberIds,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
import 'package:cloud_firestore/cloud_firestore.dart';
|
import 'package:cloud_firestore/cloud_firestore.dart';
|
||||||
|
import 'package:firebase_auth/firebase_auth.dart';
|
||||||
import 'package:travel_mate/services/error_service.dart';
|
import 'package:travel_mate/services/error_service.dart';
|
||||||
import '../models/group.dart';
|
import '../models/group.dart';
|
||||||
import '../models/group_member.dart';
|
import '../models/group_member.dart';
|
||||||
|
|
||||||
class GroupRepository {
|
class GroupRepository {
|
||||||
final FirebaseFirestore _firestore = FirebaseFirestore.instance;
|
final FirebaseFirestore _firestore = FirebaseFirestore.instance;
|
||||||
|
final FirebaseAuth _auth = FirebaseAuth.instance;
|
||||||
final _errorService = ErrorService();
|
final _errorService = ErrorService();
|
||||||
|
|
||||||
CollectionReference get _groupsCollection => _firestore.collection('groups');
|
CollectionReference get _groupsCollection => _firestore.collection('groups');
|
||||||
@@ -21,7 +23,9 @@ class GroupRepository {
|
|||||||
return await _firestore.runTransaction<String>((transaction) async {
|
return await _firestore.runTransaction<String>((transaction) async {
|
||||||
final groupRef = _groupsCollection.doc();
|
final groupRef = _groupsCollection.doc();
|
||||||
|
|
||||||
final groupData = group.toMap();
|
// Ajouter les IDs des membres à la liste memberIds
|
||||||
|
final memberIds = members.map((m) => m.userId).toList();
|
||||||
|
final groupData = group.copyWith(memberIds: memberIds).toMap();
|
||||||
transaction.set(groupRef, groupData);
|
transaction.set(groupRef, groupData);
|
||||||
|
|
||||||
for (var member in members) {
|
for (var member in members) {
|
||||||
@@ -38,50 +42,13 @@ class GroupRepository {
|
|||||||
|
|
||||||
Stream<List<Group>> getGroupsByUserId(String userId) {
|
Stream<List<Group>> getGroupsByUserId(String userId) {
|
||||||
return _groupsCollection
|
return _groupsCollection
|
||||||
|
.where('memberIds', arrayContains: userId)
|
||||||
.snapshots()
|
.snapshots()
|
||||||
.asyncMap((snapshot) async {
|
.map((snapshot) {
|
||||||
|
return snapshot.docs.map((doc) {
|
||||||
List<Group> userGroups = [];
|
final groupData = doc.data() as Map<String, dynamic>;
|
||||||
|
return Group.fromMap(groupData, doc.id);
|
||||||
for (var groupDoc in snapshot.docs) {
|
}).toList();
|
||||||
try {
|
|
||||||
final groupId = groupDoc.id;
|
|
||||||
|
|
||||||
// Vérifier si l'utilisateur est membre
|
|
||||||
final memberDoc = await groupDoc.reference
|
|
||||||
.collection('members')
|
|
||||||
.doc(userId)
|
|
||||||
.get();
|
|
||||||
|
|
||||||
if (memberDoc.exists) {
|
|
||||||
final groupData = groupDoc.data() as Map<String, dynamic>;
|
|
||||||
final group = Group.fromMap(groupData, groupId);
|
|
||||||
|
|
||||||
final members = await getGroupMembers(groupId);
|
|
||||||
userGroups.add(group.copyWith(members: members));
|
|
||||||
} else {
|
|
||||||
_errorService.logInfo('group_repository.dart','Utilisateur NON membre de $groupId');
|
|
||||||
}
|
|
||||||
} catch (e, stackTrace) {
|
|
||||||
_errorService.logError(e.toString(), stackTrace);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return userGroups;
|
|
||||||
})
|
|
||||||
.distinct((prev, next) {
|
|
||||||
// Comparer les listes pour éviter les doublons
|
|
||||||
if (prev.length != next.length) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Vérifier si les IDs sont identiques
|
|
||||||
final prevIds = prev.map((g) => g.id).toSet();
|
|
||||||
final nextIds = next.map((g) => g.id).toSet();
|
|
||||||
|
|
||||||
final identical = prevIds.difference(nextIds).isEmpty &&
|
|
||||||
nextIds.difference(prevIds).isEmpty;
|
|
||||||
|
|
||||||
return identical;
|
|
||||||
})
|
})
|
||||||
.handleError((error, stackTrace) {
|
.handleError((error, stackTrace) {
|
||||||
_errorService.logError(error, stackTrace);
|
_errorService.logError(error, stackTrace);
|
||||||
@@ -106,11 +73,30 @@ class GroupRepository {
|
|||||||
|
|
||||||
Future<Group?> getGroupByTripId(String tripId) async {
|
Future<Group?> getGroupByTripId(String tripId) async {
|
||||||
try {
|
try {
|
||||||
final querySnapshot = await _groupsCollection
|
final userId = _auth.currentUser?.uid;
|
||||||
|
if (userId == null) return null;
|
||||||
|
|
||||||
|
// Tentative 1: Requête optimisée avec memberIds
|
||||||
|
var querySnapshot = await _groupsCollection
|
||||||
.where('tripId', isEqualTo: tripId)
|
.where('tripId', isEqualTo: tripId)
|
||||||
|
.where('memberIds', arrayContains: userId)
|
||||||
.limit(1)
|
.limit(1)
|
||||||
.get();
|
.get();
|
||||||
|
|
||||||
|
// Tentative 2: Fallback pour le créateur (si memberIds est manquant - anciennes données)
|
||||||
|
if (querySnapshot.docs.isEmpty) {
|
||||||
|
querySnapshot = await _groupsCollection
|
||||||
|
.where('tripId', isEqualTo: tripId)
|
||||||
|
.where('createdBy', isEqualTo: userId)
|
||||||
|
.limit(1)
|
||||||
|
.get();
|
||||||
|
|
||||||
|
// Si on trouve le groupe via le fallback, on lance une migration
|
||||||
|
if (querySnapshot.docs.isNotEmpty) {
|
||||||
|
_migrateGroupData(querySnapshot.docs.first.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (querySnapshot.docs.isEmpty) return null;
|
if (querySnapshot.docs.isEmpty) return null;
|
||||||
|
|
||||||
final doc = querySnapshot.docs.first;
|
final doc = querySnapshot.docs.first;
|
||||||
@@ -123,17 +109,27 @@ class GroupRepository {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Méthode utilitaire pour migrer les anciennes données
|
||||||
|
Future<void> _migrateGroupData(String groupId) async {
|
||||||
|
try {
|
||||||
|
final members = await getGroupMembers(groupId);
|
||||||
|
final memberIds = members.map((m) => m.userId).toList();
|
||||||
|
|
||||||
|
if (memberIds.isNotEmpty) {
|
||||||
|
await _groupsCollection.doc(groupId).update({'memberIds': memberIds});
|
||||||
|
print('Migration réussie pour le groupe $groupId');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
print('Erreur de migration pour le groupe $groupId: $e');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Future<List<GroupMember>> getGroupMembers(String groupId) async {
|
Future<List<GroupMember>> getGroupMembers(String groupId) async {
|
||||||
try {
|
try {
|
||||||
final snapshot = await _membersCollection(groupId).get();
|
final snapshot = await _membersCollection(groupId).get();
|
||||||
return snapshot.docs
|
return snapshot.docs.map((doc) {
|
||||||
.map((doc) {
|
return GroupMember.fromMap(doc.data() as Map<String, dynamic>, doc.id);
|
||||||
return GroupMember.fromMap(
|
}).toList();
|
||||||
doc.data() as Map<String, dynamic>,
|
|
||||||
doc.id,
|
|
||||||
);
|
|
||||||
})
|
|
||||||
.toList();
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
throw Exception('Erreur lors de la récupération des membres: $e');
|
throw Exception('Erreur lors de la récupération des membres: $e');
|
||||||
}
|
}
|
||||||
@@ -141,10 +137,22 @@ class GroupRepository {
|
|||||||
|
|
||||||
Future<void> addMember(String groupId, GroupMember member) async {
|
Future<void> addMember(String groupId, GroupMember member) async {
|
||||||
try {
|
try {
|
||||||
|
// 1. Récupérer le groupe pour avoir le tripId
|
||||||
|
final group = await getGroupById(groupId);
|
||||||
|
if (group == null) throw Exception('Groupe introuvable');
|
||||||
|
|
||||||
|
// 2. Ajouter le membre dans la sous-collection members du groupe
|
||||||
await _membersCollection(groupId).doc(member.userId).set(member.toMap());
|
await _membersCollection(groupId).doc(member.userId).set(member.toMap());
|
||||||
|
|
||||||
|
// 3. Mettre à jour la liste memberIds du groupe
|
||||||
await _groupsCollection.doc(groupId).update({
|
await _groupsCollection.doc(groupId).update({
|
||||||
'updatedAt': DateTime.now().millisecondsSinceEpoch,
|
'updatedAt': DateTime.now().millisecondsSinceEpoch,
|
||||||
|
'memberIds': FieldValue.arrayUnion([member.userId]),
|
||||||
|
});
|
||||||
|
|
||||||
|
// 4. Mettre à jour la liste participants du voyage
|
||||||
|
await _firestore.collection('trips').doc(group.tripId).update({
|
||||||
|
'participants': FieldValue.arrayUnion([member.userId]),
|
||||||
});
|
});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
throw Exception('Erreur lors de l\'ajout du membre: $e');
|
throw Exception('Erreur lors de l\'ajout du membre: $e');
|
||||||
@@ -153,10 +161,22 @@ class GroupRepository {
|
|||||||
|
|
||||||
Future<void> removeMember(String groupId, String userId) async {
|
Future<void> removeMember(String groupId, String userId) async {
|
||||||
try {
|
try {
|
||||||
|
// 1. Récupérer le groupe pour avoir le tripId
|
||||||
|
final group = await getGroupById(groupId);
|
||||||
|
if (group == null) throw Exception('Groupe introuvable');
|
||||||
|
|
||||||
|
// 2. Supprimer le membre de la sous-collection members du groupe
|
||||||
await _membersCollection(groupId).doc(userId).delete();
|
await _membersCollection(groupId).doc(userId).delete();
|
||||||
|
|
||||||
|
// 3. Mettre à jour la liste memberIds du groupe
|
||||||
await _groupsCollection.doc(groupId).update({
|
await _groupsCollection.doc(groupId).update({
|
||||||
'updatedAt': DateTime.now().millisecondsSinceEpoch,
|
'updatedAt': DateTime.now().millisecondsSinceEpoch,
|
||||||
|
'memberIds': FieldValue.arrayRemove([userId]),
|
||||||
|
});
|
||||||
|
|
||||||
|
// 4. Mettre à jour la liste participants du voyage
|
||||||
|
await _firestore.collection('trips').doc(group.tripId).update({
|
||||||
|
'participants': FieldValue.arrayRemove([userId]),
|
||||||
});
|
});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
throw Exception('Erreur lors de la suppression du membre: $e');
|
throw Exception('Erreur lors de la suppression du membre: $e');
|
||||||
@@ -165,8 +185,11 @@ class GroupRepository {
|
|||||||
|
|
||||||
Future<void> updateGroup(String groupId, Group group) async {
|
Future<void> updateGroup(String groupId, Group group) async {
|
||||||
try {
|
try {
|
||||||
await _groupsCollection.doc(groupId).update(
|
await _groupsCollection
|
||||||
group.toMap()..['updatedAt'] = DateTime.now().millisecondsSinceEpoch,
|
.doc(groupId)
|
||||||
|
.update(
|
||||||
|
group.toMap()
|
||||||
|
..['updatedAt'] = DateTime.now().millisecondsSinceEpoch,
|
||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
throw Exception('Erreur lors de la mise à jour du groupe: $e');
|
throw Exception('Erreur lors de la mise à jour du groupe: $e');
|
||||||
@@ -174,38 +197,42 @@ class GroupRepository {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<void> deleteGroup(String tripId) async {
|
Future<void> deleteGroup(String tripId) async {
|
||||||
try {
|
try {
|
||||||
final querySnapshot = await _groupsCollection
|
final userId = _auth.currentUser?.uid;
|
||||||
.where('tripId', isEqualTo: tripId)
|
if (userId == null) throw Exception('Utilisateur non connecté');
|
||||||
.limit(1)
|
|
||||||
.get();
|
|
||||||
|
|
||||||
if (querySnapshot.docs.isEmpty) {
|
final querySnapshot = await _groupsCollection
|
||||||
throw Exception('Aucun groupe trouvé pour ce voyage');
|
.where('tripId', isEqualTo: tripId)
|
||||||
|
.where('createdBy', isEqualTo: userId)
|
||||||
|
.limit(1)
|
||||||
|
.get();
|
||||||
|
|
||||||
|
if (querySnapshot.docs.isEmpty) {
|
||||||
|
throw Exception('Aucun groupe trouvé pour ce voyage');
|
||||||
|
}
|
||||||
|
|
||||||
|
final groupDoc = querySnapshot.docs.first;
|
||||||
|
final groupId = groupDoc.id;
|
||||||
|
|
||||||
|
final membersSnapshot = await _membersCollection(groupId).get();
|
||||||
|
for (var doc in membersSnapshot.docs) {
|
||||||
|
await doc.reference.delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
await _groupsCollection.doc(groupId).delete();
|
||||||
|
} catch (e) {
|
||||||
|
throw Exception('Erreur lors de la suppression du groupe: $e');
|
||||||
}
|
}
|
||||||
|
|
||||||
final groupDoc = querySnapshot.docs.first;
|
|
||||||
final groupId = groupDoc.id;
|
|
||||||
|
|
||||||
final membersSnapshot = await _membersCollection(groupId).get();
|
|
||||||
for (var doc in membersSnapshot.docs) {
|
|
||||||
await doc.reference.delete();
|
|
||||||
}
|
|
||||||
|
|
||||||
await _groupsCollection.doc(groupId).delete();
|
|
||||||
} catch (e) {
|
|
||||||
throw Exception('Erreur lors de la suppression du groupe: $e');
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
Stream<List<GroupMember>> watchGroupMembers(String groupId) {
|
Stream<List<GroupMember>> watchGroupMembers(String groupId) {
|
||||||
return _membersCollection(groupId).snapshots().map(
|
return _membersCollection(groupId).snapshots().map(
|
||||||
(snapshot) => snapshot.docs
|
(snapshot) => snapshot.docs
|
||||||
.map((doc) => GroupMember.fromMap(
|
.map(
|
||||||
doc.data() as Map<String, dynamic>,
|
(doc) =>
|
||||||
doc.id,
|
GroupMember.fromMap(doc.data() as Map<String, dynamic>, doc.id),
|
||||||
))
|
)
|
||||||
.toList(),
|
.toList(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user