corrrigi team_stats_page

This commit is contained in:
2026-01-21 10:41:14 +00:00
parent 05c4738ef2
commit d7742432f4
3 changed files with 57 additions and 54 deletions

View File

@@ -5,71 +5,52 @@ import '../models/person_model.dart';
class StatsController { class StatsController {
final SupabaseClient _supabase = Supabase.instance.client; final SupabaseClient _supabase = Supabase.instance.client;
// --- 1. LER DADOS (STREAM) --- // --- 1. LER MEMBROS (STREAM EM TEMPO REAL) ---
Stream<List<Person>> getMembers(String teamId) { Stream<List<Person>> getMembers(String teamId) {
return _supabase return _supabase
.from('members') .from('members') // Nome da tua tabela no Supabase
.stream(primaryKey: ['id']) .stream(primaryKey: ['id'])
.eq('team_id', teamId) .eq('team_id', teamId)
.order('name', ascending: true) // Ordena por nome .order('name', ascending: true)
.map((data) => data.map((json) => Person.fromMap(json)).toList()); .map((data) => data.map((json) => Person.fromMap(json)).toList());
} }
// --- 2. AÇÕES DE BASE DE DADOS --- // --- 2. APAGAR ---
// Adicionar
Future<void> _addPersonToSupabase(String teamId, String name, String type, String number) async {
await _supabase.from('members').insert({
'team_id': teamId,
'name': name,
'type': type,
'number': number,
});
}
// Editar
Future<void> _updatePersonInSupabase(String personId, String name, String type, String number) async {
await _supabase.from('members').update({
'name': name,
'type': type,
'number': number,
}).eq('id', personId);
}
// Apagar
Future<void> deletePerson(String teamId, String personId) async { Future<void> deletePerson(String teamId, String personId) async {
try { try {
await _supabase.from('members').delete().eq('id', personId); await _supabase.from('members').delete().eq('id', personId);
} catch (e) { } catch (e) {
debugPrint("Erro ao apagar: $e"); debugPrint("Erro ao eliminar: $e");
} }
} }
// --- 3. DIÁLOGOS (UI) --- // --- 3. DIÁLOGOS DE ADICIONAR / EDITAR ---
// Mostrar Diálogo de Adicionar // Abrir diálogo para criar novo
void showAddPersonDialog(BuildContext context, String teamId) { void showAddPersonDialog(BuildContext context, String teamId) {
_showPersonDialog(context, teamId: teamId); _showPersonForm(context, teamId: teamId);
} }
// Mostrar Diálogo de Editar // Abrir diálogo para editar existente
void showEditPersonDialog(BuildContext context, String teamId, Person person) { void showEditPersonDialog(BuildContext context, String teamId, Person person) {
_showPersonDialog(context, teamId: teamId, person: person); _showPersonForm(context, teamId: teamId, person: person);
} }
// Função Genérica para o Diálogo (Serve para criar e editar) // Lógica interna do formulário
void _showPersonDialog(BuildContext context, {required String teamId, Person? person}) { void _showPersonForm(BuildContext context, {required String teamId, Person? person}) {
final isEditing = person != null; final isEditing = person != null;
// Controladores de texto
final nameController = TextEditingController(text: person?.name ?? ''); final nameController = TextEditingController(text: person?.name ?? '');
final numberController = TextEditingController(text: person?.number ?? ''); final numberController = TextEditingController(text: person?.number ?? '');
// Valor inicial do dropdown ('Jogador' por defeito) // Variável para o Dropdown (Valor inicial)
String selectedType = person?.type ?? 'Jogador'; String selectedType = person?.type ?? 'Jogador';
showDialog( showDialog(
context: context, context: context,
builder: (context) { builder: (context) {
// Usamos StatefulBuilder para atualizar o Dropdown dentro do Dialog // StatefulBuilder serve para atualizar o Dropdown DENTRO do diálogo
return StatefulBuilder( return StatefulBuilder(
builder: (context, setState) { builder: (context, setState) {
return AlertDialog( return AlertDialog(
@@ -85,7 +66,7 @@ class StatsController {
), ),
const SizedBox(height: 10), const SizedBox(height: 10),
// Tipo (Jogador/Treinador) // Tipo (Jogador vs Treinador)
DropdownButtonFormField<String>( DropdownButtonFormField<String>(
value: selectedType, value: selectedType,
decoration: const InputDecoration(labelText: 'Função'), decoration: const InputDecoration(labelText: 'Função'),
@@ -101,7 +82,7 @@ class StatsController {
), ),
const SizedBox(height: 10), const SizedBox(height: 10),
// Número (Só aparece se for Jogador) // Número (Só mostramos se for Jogador)
if (selectedType == 'Jogador') if (selectedType == 'Jogador')
TextField( TextField(
controller: numberController, controller: numberController,
@@ -118,15 +99,26 @@ class StatsController {
ElevatedButton( ElevatedButton(
style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFF00C853)), style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFF00C853)),
onPressed: () async { onPressed: () async {
if (nameController.text.isEmpty) return; if (nameController.text.trim().isEmpty) return;
final name = nameController.text.trim(); final name = nameController.text.trim();
final number = numberController.text.trim(); final number = numberController.text.trim();
if (isEditing) { if (isEditing) {
await _updatePersonInSupabase(person!.id, name, selectedType, number); // ATUALIZAR
await _supabase.from('members').update({
'name': name,
'type': selectedType,
'number': number,
}).eq('id', person!.id);
} else { } else {
await _addPersonToSupabase(teamId, name, selectedType, number); // CRIAR NOVO
await _supabase.from('members').insert({
'team_id': teamId,
'name': name,
'type': selectedType,
'number': number,
});
} }
if (context.mounted) Navigator.pop(context); if (context.mounted) Navigator.pop(context);

View File

@@ -13,7 +13,7 @@ class Person {
required this.number, required this.number,
}); });
// Converter do Supabase (Map) para o Objeto // Converte o JSON do Supabase para o objeto Person
factory Person.fromMap(Map<String, dynamic> map) { factory Person.fromMap(Map<String, dynamic> map) {
return Person( return Person(
id: map['id'] ?? '', id: map['id'] ?? '',

View File

@@ -6,6 +6,7 @@ import '../widgets/stats_widgets.dart';
class TeamStatsPage extends StatelessWidget { class TeamStatsPage extends StatelessWidget {
final Team team; final Team team;
// Agora este controller já tem o método getMembers
final StatsController _controller = StatsController(); final StatsController _controller = StatsController();
TeamStatsPage({super.key, required this.team}); TeamStatsPage({super.key, required this.team});
@@ -16,16 +17,26 @@ class TeamStatsPage extends StatelessWidget {
backgroundColor: const Color(0xFFF5F7FA), backgroundColor: const Color(0xFFF5F7FA),
body: Column( body: Column(
children: [ children: [
// Certifica-te que tens o StatsHeader no ficheiro stats_widgets.dart
StatsHeader(team: team), StatsHeader(team: team),
Expanded( Expanded(
child: StreamBuilder<List<Person>>( child: StreamBuilder<List<Person>>(
// AGORA ISTO VAI FUNCIONAR:
stream: _controller.getMembers(team.id), stream: _controller.getMembers(team.id),
builder: (context, snapshot) { builder: (context, snapshot) {
if (snapshot.connectionState == ConnectionState.waiting) { if (snapshot.connectionState == ConnectionState.waiting) {
return const Center(child: CircularProgressIndicator()); return const Center(child: CircularProgressIndicator());
} }
if (snapshot.hasError) {
return Center(child: Text('Erro: ${snapshot.error}'));
}
final members = snapshot.data ?? []; final members = snapshot.data ?? [];
if (members.isEmpty) {
return const Center(child: Text("Sem membros nesta equipa."));
}
final coaches = members.where((m) => m.type == 'Treinador').toList(); final coaches = members.where((m) => m.type == 'Treinador').toList();
final players = members.where((m) => m.type == 'Jogador').toList(); final players = members.where((m) => m.type == 'Jogador').toList();
@@ -34,13 +45,17 @@ class TeamStatsPage extends StatelessWidget {
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
const SummaryCard(), const SummaryCard(), // Confirma se tens este widget
const SizedBox(height: 30), const SizedBox(height: 30),
if (coaches.isNotEmpty) ...[
_buildSectionTitle("Treinadores"), _buildSectionTitle("Treinadores"),
...coaches.map((c) => _buildPersonCard(context, c, isCoach: true)), ...coaches.map((c) => _buildPersonCard(context, c, isCoach: true)),
const SizedBox(height: 30), const SizedBox(height: 30),
],
if (players.isNotEmpty) ...[
_buildSectionTitle("Jogadores"), _buildSectionTitle("Jogadores"),
...players.map((p) => _buildPersonCard(context, p, isCoach: false)), ...players.map((p) => _buildPersonCard(context, p, isCoach: false)),
],
const SizedBox(height: 80), const SizedBox(height: 80),
], ],
), ),
@@ -58,7 +73,6 @@ class TeamStatsPage extends StatelessWidget {
); );
} }
// CORREÇÃO: Adicionado BuildContext context como argumento
Widget _buildPersonCard(BuildContext context, Person person, {required bool isCoach}) { Widget _buildPersonCard(BuildContext context, Person person, {required bool isCoach}) {
return Card( return Card(
margin: const EdgeInsets.only(top: 12), margin: const EdgeInsets.only(top: 12),
@@ -126,6 +140,3 @@ class TeamStatsPage extends StatelessWidget {
); );
} }
} }
class StatsController {
}