corrrigi team_stats_page
This commit is contained in:
@@ -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,17 +99,28 @@ 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);
|
||||||
},
|
},
|
||||||
child: Text(isEditing ? 'Guardar' : 'Adicionar', style: const TextStyle(color: Colors.white)),
|
child: Text(isEditing ? 'Guardar' : 'Adicionar', style: const TextStyle(color: Colors.white)),
|
||||||
|
|||||||
@@ -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'] ?? '',
|
||||||
|
|||||||
@@ -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: [
|
||||||
StatsHeader(team: team),
|
// Certifica-te que tens o StatsHeader no ficheiro stats_widgets.dart
|
||||||
|
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),
|
||||||
_buildSectionTitle("Treinadores"),
|
if (coaches.isNotEmpty) ...[
|
||||||
...coaches.map((c) => _buildPersonCard(context, c, isCoach: true)),
|
_buildSectionTitle("Treinadores"),
|
||||||
const SizedBox(height: 30),
|
...coaches.map((c) => _buildPersonCard(context, c, isCoach: true)),
|
||||||
_buildSectionTitle("Jogadores"),
|
const SizedBox(height: 30),
|
||||||
...players.map((p) => _buildPersonCard(context, p, isCoach: false)),
|
],
|
||||||
|
if (players.isNotEmpty) ...[
|
||||||
|
_buildSectionTitle("Jogadores"),
|
||||||
|
...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),
|
||||||
@@ -125,7 +139,4 @@ class TeamStatsPage extends StatelessWidget {
|
|||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class StatsController {
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user