Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 5b968680be |
@@ -9,6 +9,7 @@ android {
|
|||||||
namespace = "com.example.playmaker"
|
namespace = "com.example.playmaker"
|
||||||
compileSdk = flutter.compileSdkVersion
|
compileSdk = flutter.compileSdkVersion
|
||||||
//ndkVersion = flutter.ndkVersion
|
//ndkVersion = flutter.ndkVersion
|
||||||
|
ndkVersion = "27.0.12077973"
|
||||||
compileOptions {
|
compileOptions {
|
||||||
sourceCompatibility = JavaVersion.VERSION_11
|
sourceCompatibility = JavaVersion.VERSION_11
|
||||||
targetCompatibility = JavaVersion.VERSION_11
|
targetCompatibility = JavaVersion.VERSION_11
|
||||||
|
|||||||
@@ -1,18 +1,8 @@
|
|||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
|
||||||
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
|
|
||||||
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES"/>
|
|
||||||
<uses-permission android:name="android.permission.INTERNET"/>
|
|
||||||
<application
|
<application
|
||||||
android:label="playmaker"
|
android:label="playmaker"
|
||||||
android:name="${applicationName}"
|
android:name="${applicationName}"
|
||||||
android:icon="@mipmap/ic_launcher">
|
android:icon="@mipmap/ic_launcher">
|
||||||
|
|
||||||
<activity
|
|
||||||
android:name="com.yalantis.ucrop.UCropActivity"
|
|
||||||
android:screenOrientation="portrait"
|
|
||||||
android:theme="@style/Theme.AppCompat.Light.NoActionBar"/>
|
|
||||||
|
|
||||||
<activity
|
<activity
|
||||||
android:name=".MainActivity"
|
android:name=".MainActivity"
|
||||||
android:exported="true"
|
android:exported="true"
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 564 KiB After Width: | Height: | Size: 2.1 MiB |
Binary file not shown.
|
Before Width: | Height: | Size: 1.8 MiB After Width: | Height: | Size: 268 KiB |
@@ -45,12 +45,5 @@
|
|||||||
<true/>
|
<true/>
|
||||||
<key>UIApplicationSupportsIndirectInputEvents</key>
|
<key>UIApplicationSupportsIndirectInputEvents</key>
|
||||||
<true/>
|
<true/>
|
||||||
|
|
||||||
<key>NSPhotoLibraryUsageDescription</key>
|
|
||||||
<string>A PlayMaker precisa de aceder à tua galeria para poderes escolher uma foto de perfil.</string>
|
|
||||||
<key>NSCameraUsageDescription</key>
|
|
||||||
<string>A PlayMaker precisa de aceder à câmara para poderes tirar uma foto de perfil.</string>
|
|
||||||
</dict>
|
|
||||||
|
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
|||||||
208
lib/calibrador_page.dart
Normal file
208
lib/calibrador_page.dart
Normal file
@@ -0,0 +1,208 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
|
import 'dart:math' as math;
|
||||||
|
|
||||||
|
class CalibradorPage extends StatefulWidget {
|
||||||
|
const CalibradorPage({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<CalibradorPage> createState() => _CalibradorPageState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CalibradorPageState extends State<CalibradorPage> {
|
||||||
|
// --- 👇 VALORES INICIAIS 👇 ---
|
||||||
|
double hoopBaseX = 0.08;
|
||||||
|
double arcRadius = 0.28;
|
||||||
|
double cornerY = 0.40;
|
||||||
|
// -----------------------------------------------------
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
SystemChrome.setPreferredOrientations([
|
||||||
|
DeviceOrientation.landscapeRight,
|
||||||
|
DeviceOrientation.landscapeLeft,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
SystemChrome.setPreferredOrientations([DeviceOrientation.portraitUp]);
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final double wScreen = MediaQuery.of(context).size.width;
|
||||||
|
final double hScreen = MediaQuery.of(context).size.height;
|
||||||
|
|
||||||
|
// O MESMO CÁLCULO EXATO DO PLACAR
|
||||||
|
final double sf = math.min(wScreen / 1150, hScreen / 720);
|
||||||
|
|
||||||
|
return Scaffold(
|
||||||
|
backgroundColor: const Color(0xFF266174),
|
||||||
|
body: SafeArea(
|
||||||
|
top: false,
|
||||||
|
bottom: false,
|
||||||
|
child: Stack(
|
||||||
|
children: [
|
||||||
|
// 👇 1. O CAMPO COM AS MARGENS EXATAS DO PLACAR 👇
|
||||||
|
Container(
|
||||||
|
margin: EdgeInsets.only(left: 65 * sf, right: 65 * sf, bottom: 55 * sf),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
border: Border.all(color: Colors.white, width: 2.5),
|
||||||
|
image: const DecorationImage(
|
||||||
|
image: AssetImage('assets/campo.png'),
|
||||||
|
fit: BoxFit.fill,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
child: LayoutBuilder(
|
||||||
|
builder: (context, constraints) {
|
||||||
|
return CustomPaint(
|
||||||
|
painter: LinePainter(
|
||||||
|
hoopBaseX: hoopBaseX,
|
||||||
|
arcRadius: arcRadius,
|
||||||
|
cornerY: cornerY,
|
||||||
|
color: Colors.redAccent,
|
||||||
|
width: constraints.maxWidth,
|
||||||
|
height: constraints.maxHeight,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// 👇 2. TOPO: MOSTRADORES DE VALORES COM FITTEDBOX (Não transborda) 👇
|
||||||
|
Positioned(
|
||||||
|
top: 0, left: 0, right: 0,
|
||||||
|
child: Container(
|
||||||
|
color: Colors.black87.withOpacity(0.8),
|
||||||
|
padding: EdgeInsets.symmetric(vertical: 5 * sf, horizontal: 15 * sf),
|
||||||
|
child: FittedBox( // Isto impede o ecrã de dar o erro dos 179 pixels!
|
||||||
|
fit: BoxFit.scaleDown,
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
_buildValueDisplay("Aro X", hoopBaseX, sf),
|
||||||
|
SizedBox(width: 20 * sf),
|
||||||
|
_buildValueDisplay("Raio", arcRadius, sf),
|
||||||
|
SizedBox(width: 20 * sf),
|
||||||
|
_buildValueDisplay("Canto", cornerY, sf),
|
||||||
|
SizedBox(width: 30 * sf),
|
||||||
|
ElevatedButton.icon(
|
||||||
|
onPressed: () => Navigator.pop(context),
|
||||||
|
icon: Icon(Icons.check, size: 18 * sf),
|
||||||
|
label: Text("FECHAR", style: TextStyle(fontSize: 14 * sf, fontWeight: FontWeight.bold)),
|
||||||
|
style: ElevatedButton.styleFrom(backgroundColor: Colors.green),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// 👇 3. FUNDO: SLIDERS (Com altura fixa para não dar o erro "hasSize") 👇
|
||||||
|
Positioned(
|
||||||
|
bottom: 0, left: 0, right: 0,
|
||||||
|
child: Container(
|
||||||
|
color: Colors.black87.withOpacity(0.8),
|
||||||
|
height: 80 * sf, // Altura segura para os sliders
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Expanded(child: _buildSlider("Pos. do Aro", hoopBaseX, 0.0, 0.25, (val) => setState(() => hoopBaseX = val), sf)),
|
||||||
|
Expanded(child: _buildSlider("Tam. da Curva", arcRadius, 0.1, 0.5, (val) => setState(() => arcRadius = val), sf)),
|
||||||
|
Expanded(child: _buildSlider("Pos. do Canto", cornerY, 0.2, 0.5, (val) => setState(() => cornerY = val), sf)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildValueDisplay(String label, double value, double sf) {
|
||||||
|
return Row(
|
||||||
|
children: [
|
||||||
|
Text("$label: ", style: TextStyle(color: Colors.white70, fontSize: 16 * sf)),
|
||||||
|
Text(value.toStringAsFixed(3), style: TextStyle(color: Colors.yellow, fontSize: 20 * sf, fontWeight: FontWeight.bold)),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildSlider(String label, double value, double min, double max, ValueChanged<double> onChanged, double sf) {
|
||||||
|
return Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
Text(label, style: TextStyle(color: Colors.white, fontSize: 12 * sf)),
|
||||||
|
SizedBox(
|
||||||
|
height: 40 * sf, // Altura exata para o Slider não crashar
|
||||||
|
child: Slider(
|
||||||
|
value: value, min: min, max: max,
|
||||||
|
activeColor: Colors.yellow, inactiveColor: Colors.white24,
|
||||||
|
onChanged: onChanged,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ==============================================================
|
||||||
|
// 📐 PINTOR: DESENHA A LINHA MATEMÁTICA NA TELA
|
||||||
|
// ==============================================================
|
||||||
|
class LinePainter extends CustomPainter {
|
||||||
|
final double hoopBaseX;
|
||||||
|
final double arcRadius;
|
||||||
|
final double cornerY;
|
||||||
|
final Color color;
|
||||||
|
final double width;
|
||||||
|
final double height;
|
||||||
|
|
||||||
|
LinePainter({
|
||||||
|
required this.hoopBaseX, required this.arcRadius, required this.cornerY,
|
||||||
|
required this.color, required this.width, required this.height,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
void paint(Canvas canvas, Size size) {
|
||||||
|
final paint = Paint()
|
||||||
|
..color = color
|
||||||
|
..style = PaintingStyle.stroke
|
||||||
|
..strokeWidth = 4;
|
||||||
|
|
||||||
|
double aspectRatio = width / height;
|
||||||
|
double hoopY = 0.50 * height;
|
||||||
|
|
||||||
|
// O cornerY controla a que distância do meio (50%) estão as linhas retas
|
||||||
|
double cornerDistY = cornerY * height;
|
||||||
|
|
||||||
|
// --- CESTO ESQUERDO ---
|
||||||
|
double hoopLX = hoopBaseX * width;
|
||||||
|
|
||||||
|
canvas.drawLine(Offset(0, hoopY - cornerDistY), Offset(width * 0.35, hoopY - cornerDistY), paint); // Cima
|
||||||
|
canvas.drawLine(Offset(0, hoopY + cornerDistY), Offset(width * 0.35, hoopY + cornerDistY), paint); // Baixo
|
||||||
|
|
||||||
|
canvas.drawArc(
|
||||||
|
Rect.fromCenter(center: Offset(hoopLX, hoopY), width: arcRadius * width * 2 / aspectRatio, height: arcRadius * height * 2),
|
||||||
|
-math.pi / 2, math.pi, false, paint,
|
||||||
|
);
|
||||||
|
|
||||||
|
// --- CESTO DIREITO ---
|
||||||
|
double hoopRX = (1.0 - hoopBaseX) * width;
|
||||||
|
|
||||||
|
canvas.drawLine(Offset(width, hoopY - cornerDistY), Offset(width * 0.65, hoopY - cornerDistY), paint); // Cima
|
||||||
|
canvas.drawLine(Offset(width, hoopY + cornerDistY), Offset(width * 0.65, hoopY + cornerDistY), paint); // Baixo
|
||||||
|
|
||||||
|
canvas.drawArc(
|
||||||
|
Rect.fromCenter(center: Offset(hoopRX, hoopY), width: arcRadius * width * 2 / aspectRatio, height: arcRadius * height * 2),
|
||||||
|
math.pi / 2, math.pi, false, paint,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool shouldRepaint(covariant LinePainter oldDelegate) {
|
||||||
|
return oldDelegate.hoopBaseX != hoopBaseX || oldDelegate.arcRadius != arcRadius || oldDelegate.cornerY != cornerY;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,57 +4,37 @@ import '../models/game_model.dart';
|
|||||||
class GameController {
|
class GameController {
|
||||||
final _supabase = Supabase.instance.client;
|
final _supabase = Supabase.instance.client;
|
||||||
|
|
||||||
String get myUserId => _supabase.auth.currentUser?.id ?? '';
|
// 1. LER JOGOS (Com Filtros Opcionais)
|
||||||
|
Stream<List<Game>> getFilteredGames({String? teamFilter, String? seasonFilter}) {
|
||||||
// LER JOGOS
|
|
||||||
Stream<List<Game>> get gamesStream {
|
|
||||||
return _supabase
|
return _supabase
|
||||||
.from('games')
|
.from('games')
|
||||||
.stream(primaryKey: ['id'])
|
.stream(primaryKey: ['id'])
|
||||||
.eq('user_id', myUserId)
|
|
||||||
.asyncMap((event) async {
|
|
||||||
final data = await _supabase
|
|
||||||
.from('games')
|
|
||||||
.select()
|
|
||||||
.eq('user_id', myUserId)
|
|
||||||
.order('game_date', ascending: false);
|
|
||||||
|
|
||||||
// O Game.fromMap agora faz o trabalho sujo todo!
|
|
||||||
return data.map((json) => Game.fromMap(json)).toList();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// LER JOGOS COM FILTROS
|
|
||||||
Stream<List<Game>> getFilteredGames({required String teamFilter, required String seasonFilter}) {
|
|
||||||
return _supabase
|
|
||||||
.from('games')
|
|
||||||
.stream(primaryKey: ['id'])
|
|
||||||
.eq('user_id', myUserId)
|
|
||||||
.asyncMap((event) async {
|
.asyncMap((event) async {
|
||||||
|
|
||||||
var query = _supabase.from('games').select().eq('user_id', myUserId);
|
// 👇 A CORREÇÃO ESTÁ AQUI: Lê diretamente da tabela 'games'
|
||||||
|
var query = _supabase.from('games').select();
|
||||||
|
|
||||||
if (seasonFilter != 'Todas') {
|
// Aplica o filtro de Temporada
|
||||||
|
if (seasonFilter != null && seasonFilter.isNotEmpty && seasonFilter != 'Todas') {
|
||||||
query = query.eq('season', seasonFilter);
|
query = query.eq('season', seasonFilter);
|
||||||
}
|
}
|
||||||
|
|
||||||
final data = await query.order('game_date', ascending: false);
|
// Aplica o filtro de Equipa (Procura em casa ou fora)
|
||||||
|
if (teamFilter != null && teamFilter.isNotEmpty && teamFilter != 'Todas') {
|
||||||
List<Game> games = data.map((json) => Game.fromMap(json)).toList();
|
query = query.or('my_team.eq.$teamFilter,opponent_team.eq.$teamFilter');
|
||||||
|
|
||||||
if (teamFilter != 'Todas') {
|
|
||||||
games = games.where((g) => g.myTeam == teamFilter || g.opponentTeam == teamFilter).toList();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return games;
|
// Executa a query com a ordenação por data
|
||||||
|
final viewData = await query.order('game_date', ascending: false);
|
||||||
|
|
||||||
|
return viewData.map((json) => Game.fromMap(json)).toList();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// CRIAR JOGO
|
// 2. CRIAR JOGO
|
||||||
Future<String?> createGame(String myTeam, String opponent, String season) async {
|
Future<String?> createGame(String myTeam, String opponent, String season) async {
|
||||||
try {
|
try {
|
||||||
final response = await _supabase.from('games').insert({
|
final response = await _supabase.from('games').insert({
|
||||||
'user_id': myUserId,
|
|
||||||
'my_team': myTeam,
|
'my_team': myTeam,
|
||||||
'opponent_team': opponent,
|
'opponent_team': opponent,
|
||||||
'season': season,
|
'season': season,
|
||||||
@@ -62,36 +42,14 @@ class GameController {
|
|||||||
'opponent_score': 0,
|
'opponent_score': 0,
|
||||||
'status': 'Decorrer',
|
'status': 'Decorrer',
|
||||||
'game_date': DateTime.now().toIso8601String(),
|
'game_date': DateTime.now().toIso8601String(),
|
||||||
// 👇 Preenchemos logo com os valores iniciais da tua Base de Dados
|
|
||||||
'remaining_seconds': 600, // Assume 10 minutos (600s)
|
|
||||||
'my_timeouts': 0,
|
|
||||||
'opp_timeouts': 0,
|
|
||||||
'current_quarter': 1,
|
|
||||||
'top_pts_name': '---',
|
|
||||||
'top_ast_name': '---',
|
|
||||||
'top_rbs_name': '---',
|
|
||||||
'top_def_name': '---',
|
|
||||||
'mvp_name': '---',
|
|
||||||
}).select().single();
|
}).select().single();
|
||||||
|
|
||||||
return response['id']?.toString();
|
return response['id'];
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print("Erro ao criar jogo: $e");
|
print("Erro ao criar jogo: $e");
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
|
||||||
// ELIMINAR JOGO
|
|
||||||
Future<bool> deleteGame(String gameId) async {
|
|
||||||
try {
|
|
||||||
await _supabase.from('games').delete().eq('id', gameId);
|
|
||||||
// Como o Supabase tem Cascade Delete (se configurado), vai apagar também
|
|
||||||
// as stats e shot_locations associadas a este game_id automaticamente.
|
|
||||||
return true;
|
|
||||||
} catch (e) {
|
|
||||||
print("Erro ao eliminar jogo: $e");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void dispose() {}
|
void dispose() {}
|
||||||
}
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
class HomeController extends ChangeNotifier {
|
class HomeController extends ChangeNotifier {
|
||||||
|
// Se precisar de estado para a home screen
|
||||||
int _selectedCardIndex = 0;
|
int _selectedCardIndex = 0;
|
||||||
|
|
||||||
int get selectedCardIndex => _selectedCardIndex;
|
int get selectedCardIndex => _selectedCardIndex;
|
||||||
@@ -10,8 +11,10 @@ class HomeController extends ChangeNotifier {
|
|||||||
notifyListeners();
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Métodos adicionais para lógica da home
|
||||||
void navigateToDetails(String playerName) {
|
void navigateToDetails(String playerName) {
|
||||||
print('Navegando para detalhes de $playerName');
|
print('Navegando para detalhes de $playerName');
|
||||||
|
// Implementar navegação
|
||||||
}
|
}
|
||||||
|
|
||||||
void refreshData() {
|
void refreshData() {
|
||||||
|
|||||||
@@ -1,53 +1,38 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
import 'dart:math' as math;
|
import 'dart:math' as math;
|
||||||
import 'dart:convert';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
|
||||||
|
|
||||||
class ShotRecord {
|
class ShotRecord {
|
||||||
final double relativeX;
|
final double relativeX;
|
||||||
final double relativeY;
|
final double relativeY;
|
||||||
final bool isMake;
|
final bool isMake;
|
||||||
final String playerId;
|
|
||||||
final String playerName;
|
final String playerName;
|
||||||
final String? zone;
|
|
||||||
final int? points;
|
|
||||||
|
|
||||||
ShotRecord({
|
ShotRecord({
|
||||||
required this.relativeX,
|
required this.relativeX,
|
||||||
required this.relativeY,
|
required this.relativeY,
|
||||||
required this.isMake,
|
required this.isMake,
|
||||||
required this.playerId,
|
required this.playerName
|
||||||
required this.playerName,
|
|
||||||
this.zone,
|
|
||||||
this.points,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
Map<String, dynamic> toJson() => {
|
|
||||||
'relativeX': relativeX, 'relativeY': relativeY, 'isMake': isMake,
|
|
||||||
'playerId': playerId, 'playerName': playerName, 'zone': zone, 'points': points,
|
|
||||||
};
|
|
||||||
|
|
||||||
factory ShotRecord.fromJson(Map<String, dynamic> json) => ShotRecord(
|
|
||||||
relativeX: json['relativeX'], relativeY: json['relativeY'], isMake: json['isMake'],
|
|
||||||
playerId: json['playerId'], playerName: json['playerName'], zone: json['zone'], points: json['points'],
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class PlacarController extends ChangeNotifier {
|
class PlacarController {
|
||||||
final String gameId;
|
final String gameId;
|
||||||
final String myTeam;
|
final String myTeam;
|
||||||
final String opponentTeam;
|
final String opponentTeam;
|
||||||
|
final VoidCallback onUpdate;
|
||||||
|
|
||||||
PlacarController({
|
PlacarController({
|
||||||
required this.gameId,
|
required this.gameId,
|
||||||
required this.myTeam,
|
required this.myTeam,
|
||||||
required this.opponentTeam,
|
required this.opponentTeam,
|
||||||
|
required this.onUpdate
|
||||||
});
|
});
|
||||||
|
|
||||||
bool isLoading = true;
|
bool isLoading = true;
|
||||||
bool isSaving = false;
|
bool isSaving = false;
|
||||||
|
|
||||||
bool gameWasAlreadyFinished = false;
|
bool gameWasAlreadyFinished = false;
|
||||||
|
|
||||||
int myScore = 0;
|
int myScore = 0;
|
||||||
@@ -66,24 +51,23 @@ class PlacarController extends ChangeNotifier {
|
|||||||
List<String> oppCourt = [];
|
List<String> oppCourt = [];
|
||||||
List<String> oppBench = [];
|
List<String> oppBench = [];
|
||||||
|
|
||||||
Map<String, String> playerNames = {};
|
|
||||||
Map<String, String> playerNumbers = {};
|
Map<String, String> playerNumbers = {};
|
||||||
Map<String, Map<String, int>> playerStats = {};
|
Map<String, Map<String, int>> playerStats = {};
|
||||||
|
Map<String, String> playerDbIds = {};
|
||||||
|
|
||||||
bool showMyBench = false;
|
bool showMyBench = false;
|
||||||
bool showOppBench = false;
|
bool showOppBench = false;
|
||||||
|
|
||||||
bool isSelectingShotLocation = false;
|
bool isSelectingShotLocation = false;
|
||||||
String? pendingAction;
|
String? pendingAction;
|
||||||
String? pendingPlayerId;
|
String? pendingPlayer;
|
||||||
List<ShotRecord> matchShots = [];
|
List<ShotRecord> matchShots = [];
|
||||||
|
|
||||||
List<String> playByPlay = [];
|
Duration duration = const Duration(minutes: 10);
|
||||||
|
|
||||||
ValueNotifier<Duration> durationNotifier = ValueNotifier(const Duration(minutes: 10));
|
|
||||||
Timer? timer;
|
Timer? timer;
|
||||||
bool isRunning = false;
|
bool isRunning = false;
|
||||||
|
|
||||||
|
// 👇 VARIÁVEIS DE CALIBRAÇÃO DO CAMPO (OS TEUS NÚMEROS!) 👇
|
||||||
bool isCalibrating = false;
|
bool isCalibrating = false;
|
||||||
double hoopBaseX = 0.088;
|
double hoopBaseX = 0.088;
|
||||||
double arcRadius = 0.459;
|
double arcRadius = 0.459;
|
||||||
@@ -94,9 +78,15 @@ class PlacarController extends ChangeNotifier {
|
|||||||
try {
|
try {
|
||||||
await Future.delayed(const Duration(milliseconds: 1500));
|
await Future.delayed(const Duration(milliseconds: 1500));
|
||||||
|
|
||||||
myCourt.clear(); myBench.clear(); oppCourt.clear(); oppBench.clear();
|
myCourt.clear();
|
||||||
playerNames.clear(); playerStats.clear(); playerNumbers.clear();
|
myBench.clear();
|
||||||
matchShots.clear(); playByPlay.clear(); myFouls = 0; opponentFouls = 0;
|
oppCourt.clear();
|
||||||
|
oppBench.clear();
|
||||||
|
playerStats.clear();
|
||||||
|
playerNumbers.clear();
|
||||||
|
playerDbIds.clear();
|
||||||
|
myFouls = 0;
|
||||||
|
opponentFouls = 0;
|
||||||
|
|
||||||
final gameResponse = await supabase.from('games').select().eq('id', gameId).single();
|
final gameResponse = await supabase.from('games').select().eq('id', gameId).single();
|
||||||
|
|
||||||
@@ -104,7 +94,7 @@ class PlacarController extends ChangeNotifier {
|
|||||||
opponentScore = int.tryParse(gameResponse['opponent_score']?.toString() ?? '0') ?? 0;
|
opponentScore = int.tryParse(gameResponse['opponent_score']?.toString() ?? '0') ?? 0;
|
||||||
|
|
||||||
int totalSeconds = int.tryParse(gameResponse['remaining_seconds']?.toString() ?? '600') ?? 600;
|
int totalSeconds = int.tryParse(gameResponse['remaining_seconds']?.toString() ?? '600') ?? 600;
|
||||||
durationNotifier.value = Duration(seconds: totalSeconds);
|
duration = Duration(seconds: totalSeconds);
|
||||||
|
|
||||||
myTimeoutsUsed = int.tryParse(gameResponse['my_timeouts']?.toString() ?? '0') ?? 0;
|
myTimeoutsUsed = int.tryParse(gameResponse['my_timeouts']?.toString() ?? '0') ?? 0;
|
||||||
opponentTimeoutsUsed = int.tryParse(gameResponse['opp_timeouts']?.toString() ?? '0') ?? 0;
|
opponentTimeoutsUsed = int.tryParse(gameResponse['opp_timeouts']?.toString() ?? '0') ?? 0;
|
||||||
@@ -112,12 +102,6 @@ class PlacarController extends ChangeNotifier {
|
|||||||
|
|
||||||
gameWasAlreadyFinished = gameResponse['status'] == 'Terminado';
|
gameWasAlreadyFinished = gameResponse['status'] == 'Terminado';
|
||||||
|
|
||||||
if (gameResponse['play_by_play'] != null) {
|
|
||||||
playByPlay = List<String>.from(gameResponse['play_by_play']);
|
|
||||||
} else {
|
|
||||||
playByPlay = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
final teamsResponse = await supabase.from('teams').select('id, name').inFilter('name', [myTeam, opponentTeam]);
|
final teamsResponse = await supabase.from('teams').select('id, name').inFilter('name', [myTeam, opponentTeam]);
|
||||||
for (var t in teamsResponse) {
|
for (var t in teamsResponse) {
|
||||||
if (t['name'] == myTeam) myTeamDbId = t['id'];
|
if (t['name'] == myTeam) myTeamDbId = t['id'];
|
||||||
@@ -140,7 +124,12 @@ class PlacarController extends ChangeNotifier {
|
|||||||
|
|
||||||
if (savedStats.containsKey(dbId)) {
|
if (savedStats.containsKey(dbId)) {
|
||||||
var s = savedStats[dbId];
|
var s = savedStats[dbId];
|
||||||
_loadSavedPlayerStats(dbId, s);
|
playerStats[name] = {
|
||||||
|
"pts": s['pts'] ?? 0, "rbs": s['rbs'] ?? 0, "ast": s['ast'] ?? 0,
|
||||||
|
"stl": s['stl'] ?? 0, "tov": s['tov'] ?? 0, "blk": s['blk'] ?? 0,
|
||||||
|
"fls": s['fls'] ?? 0, "fgm": s['fgm'] ?? 0, "fga": s['fga'] ?? 0,
|
||||||
|
"ftm": s['ftm'] ?? 0, "fta": s['fta'] ?? 0, "orb": s['orb'] ?? 0, "drb": s['drb'] ?? 0,
|
||||||
|
};
|
||||||
myFouls += (s['fls'] as int? ?? 0);
|
myFouls += (s['fls'] as int? ?? 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -154,65 +143,42 @@ class PlacarController extends ChangeNotifier {
|
|||||||
|
|
||||||
if (savedStats.containsKey(dbId)) {
|
if (savedStats.containsKey(dbId)) {
|
||||||
var s = savedStats[dbId];
|
var s = savedStats[dbId];
|
||||||
_loadSavedPlayerStats(dbId, s);
|
playerStats[name] = {
|
||||||
|
"pts": s['pts'] ?? 0, "rbs": s['rbs'] ?? 0, "ast": s['ast'] ?? 0,
|
||||||
|
"stl": s['stl'] ?? 0, "tov": s['tov'] ?? 0, "blk": s['blk'] ?? 0,
|
||||||
|
"fls": s['fls'] ?? 0, "fgm": s['fgm'] ?? 0, "fga": s['fga'] ?? 0,
|
||||||
|
"ftm": s['ftm'] ?? 0, "fta": s['fta'] ?? 0, "orb": s['orb'] ?? 0, "drb": s['drb'] ?? 0,
|
||||||
|
};
|
||||||
opponentFouls += (s['fls'] as int? ?? 0);
|
opponentFouls += (s['fls'] as int? ?? 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_padTeam(oppCourt, oppBench, "Adversário", isMyTeam: false);
|
_padTeam(oppCourt, oppBench, "Adversário", isMyTeam: false);
|
||||||
|
|
||||||
final shotsResponse = await supabase.from('shot_locations').select().eq('game_id', gameId);
|
|
||||||
for (var shotData in shotsResponse) {
|
|
||||||
matchShots.add(ShotRecord(
|
|
||||||
relativeX: double.parse(shotData['relative_x'].toString()),
|
|
||||||
relativeY: double.parse(shotData['relative_y'].toString()),
|
|
||||||
isMake: shotData['is_make'] == true,
|
|
||||||
playerId: shotData['member_id'].toString(),
|
|
||||||
playerName: shotData['player_name'].toString(),
|
|
||||||
zone: shotData['zone']?.toString(),
|
|
||||||
points: shotData['points'] != null ? int.parse(shotData['points'].toString()) : null,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
await _loadLocalBackup();
|
|
||||||
|
|
||||||
isLoading = false;
|
isLoading = false;
|
||||||
notifyListeners();
|
onUpdate();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
debugPrint("Erro ao retomar jogo: $e");
|
debugPrint("Erro ao retomar jogo: $e");
|
||||||
|
_padTeam(myCourt, myBench, "Falha", isMyTeam: true);
|
||||||
|
_padTeam(oppCourt, oppBench, "Falha Opp", isMyTeam: false);
|
||||||
isLoading = false;
|
isLoading = false;
|
||||||
notifyListeners();
|
onUpdate();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void _loadSavedPlayerStats(String dbId, Map<String, dynamic> s) {
|
|
||||||
playerStats[dbId] = {
|
|
||||||
"pts": s['pts'] ?? 0, "rbs": s['rbs'] ?? 0, "ast": s['ast'] ?? 0,
|
|
||||||
"stl": s['stl'] ?? 0, "tov": s['tov'] ?? 0, "blk": s['blk'] ?? 0,
|
|
||||||
"fls": s['fls'] ?? 0, "fgm": s['fgm'] ?? 0, "fga": s['fga'] ?? 0,
|
|
||||||
"ftm": s['ftm'] ?? 0, "fta": s['fta'] ?? 0, "orb": s['orb'] ?? 0, "drb": s['drb'] ?? 0,
|
|
||||||
"p2m": s['p2m'] ?? 0, "p2a": s['p2a'] ?? 0, "p3m": s['p3m'] ?? 0, "p3a": s['p3a'] ?? 0,
|
|
||||||
"so": s['so'] ?? 0, "il": s['il'] ?? 0, "li": s['li'] ?? 0,
|
|
||||||
"pa": s['pa'] ?? 0, "tres_s": s['tres_s'] ?? 0, "dr": s['dr'] ?? 0, "min": s['min'] ?? 0,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
void _registerPlayer({required String name, required String number, String? dbId, required bool isMyTeam, required bool isCourt}) {
|
void _registerPlayer({required String name, required String number, String? dbId, required bool isMyTeam, required bool isCourt}) {
|
||||||
String id = dbId ?? "fake_${DateTime.now().millisecondsSinceEpoch}_${math.Random().nextInt(9999)}";
|
if (playerNumbers.containsKey(name)) name = "$name (Opp)";
|
||||||
|
playerNumbers[name] = number;
|
||||||
|
if (dbId != null) playerDbIds[name] = dbId;
|
||||||
|
|
||||||
playerNames[id] = name;
|
playerStats[name] = {
|
||||||
playerNumbers[id] = number;
|
|
||||||
|
|
||||||
playerStats[id] = {
|
|
||||||
"pts": 0, "rbs": 0, "ast": 0, "stl": 0, "tov": 0, "blk": 0,
|
"pts": 0, "rbs": 0, "ast": 0, "stl": 0, "tov": 0, "blk": 0,
|
||||||
"fls": 0, "fgm": 0, "fga": 0, "ftm": 0, "fta": 0, "orb": 0, "drb": 0,
|
"fls": 0, "fgm": 0, "fga": 0, "ftm": 0, "fta": 0, "orb": 0, "drb": 0
|
||||||
"p2m": 0, "p2a": 0, "p3m": 0, "p3a": 0,
|
|
||||||
"so": 0, "il": 0, "li": 0, "pa": 0, "tres_s": 0, "dr": 0, "min": 0
|
|
||||||
};
|
};
|
||||||
|
|
||||||
if (isMyTeam) {
|
if (isMyTeam) {
|
||||||
if (isCourt) myCourt.add(id); else myBench.add(id);
|
if (isCourt) myCourt.add(name); else myBench.add(name);
|
||||||
} else {
|
} else {
|
||||||
if (isCourt) oppCourt.add(id); else oppBench.add(id);
|
if (isCourt) oppCourt.add(name); else oppBench.add(name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,78 +188,33 @@ class PlacarController extends ChangeNotifier {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _saveLocalBackup() async {
|
|
||||||
try {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final backupData = {
|
|
||||||
'myScore': myScore, 'opponentScore': opponentScore,
|
|
||||||
'myFouls': myFouls, 'opponentFouls': opponentFouls,
|
|
||||||
'currentQuarter': currentQuarter, 'duration': durationNotifier.value.inSeconds,
|
|
||||||
'myTimeoutsUsed': myTimeoutsUsed, 'opponentTimeoutsUsed': opponentTimeoutsUsed,
|
|
||||||
'playerStats': playerStats,
|
|
||||||
'myCourt': myCourt, 'myBench': myBench, 'oppCourt': oppCourt, 'oppBench': oppBench,
|
|
||||||
'matchShots': matchShots.map((s) => s.toJson()).toList(),
|
|
||||||
'playByPlay': playByPlay,
|
|
||||||
};
|
|
||||||
await prefs.setString('backup_$gameId', jsonEncode(backupData));
|
|
||||||
} catch (e) {
|
|
||||||
debugPrint("Erro no Auto-Save: $e");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _loadLocalBackup() async {
|
|
||||||
try {
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final String? backupString = prefs.getString('backup_$gameId');
|
|
||||||
|
|
||||||
if (backupString != null) {
|
|
||||||
final data = jsonDecode(backupString);
|
|
||||||
|
|
||||||
myScore = data['myScore']; opponentScore = data['opponentScore'];
|
|
||||||
myFouls = data['myFouls']; opponentFouls = data['opponentFouls'];
|
|
||||||
currentQuarter = data['currentQuarter']; durationNotifier.value = Duration(seconds: data['duration']);
|
|
||||||
myTimeoutsUsed = data['myTimeoutsUsed']; opponentTimeoutsUsed = data['opponentTimeoutsUsed'];
|
|
||||||
|
|
||||||
myCourt = List<String>.from(data['myCourt']); myBench = List<String>.from(data['myBench']);
|
|
||||||
oppCourt = List<String>.from(data['oppCourt']); oppBench = List<String>.from(data['oppBench']);
|
|
||||||
|
|
||||||
Map<String, dynamic> decodedStats = data['playerStats'];
|
|
||||||
playerStats = decodedStats.map((k, v) => MapEntry(k, Map<String, int>.from(v)));
|
|
||||||
|
|
||||||
List<dynamic> decodedShots = data['matchShots'];
|
|
||||||
matchShots = decodedShots.map((s) => ShotRecord.fromJson(s)).toList();
|
|
||||||
|
|
||||||
playByPlay = List<String>.from(data['playByPlay'] ?? []);
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
debugPrint("Erro ao carregar Auto-Save: $e");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void toggleTimer(BuildContext context) {
|
void toggleTimer(BuildContext context) {
|
||||||
if (isRunning) {
|
if (isRunning) {
|
||||||
timer?.cancel();
|
timer?.cancel();
|
||||||
_saveLocalBackup();
|
|
||||||
} else {
|
} else {
|
||||||
timer = Timer.periodic(const Duration(seconds: 1), (timer) {
|
timer = Timer.periodic(const Duration(seconds: 1), (timer) {
|
||||||
if (durationNotifier.value.inSeconds > 0) {
|
if (duration.inSeconds > 0) {
|
||||||
durationNotifier.value -= const Duration(seconds: 1);
|
duration -= const Duration(seconds: 1);
|
||||||
} else {
|
} else {
|
||||||
timer.cancel();
|
timer.cancel();
|
||||||
isRunning = false;
|
isRunning = false;
|
||||||
if (currentQuarter < 4) {
|
if (currentQuarter < 4) {
|
||||||
currentQuarter++;
|
currentQuarter++;
|
||||||
durationNotifier.value = const Duration(minutes: 10);
|
duration = const Duration(minutes: 10);
|
||||||
myFouls = 0; opponentFouls = 0;
|
myFouls = 0;
|
||||||
myTimeoutsUsed = 0; opponentTimeoutsUsed = 0;
|
opponentFouls = 0;
|
||||||
_saveLocalBackup();
|
myTimeoutsUsed = 0;
|
||||||
|
opponentTimeoutsUsed = 0;
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('Período $currentQuarter iniciado. Faltas e Timeouts resetados!'), backgroundColor: Colors.blue));
|
||||||
|
} else {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(const SnackBar(content: Text('FIM DO JOGO! Clica em Guardar para fechar a partida.'), backgroundColor: Colors.red));
|
||||||
}
|
}
|
||||||
notifyListeners();
|
|
||||||
}
|
}
|
||||||
|
onUpdate();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
isRunning = !isRunning;
|
isRunning = !isRunning;
|
||||||
notifyListeners();
|
onUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
void useTimeout(bool isOpponent) {
|
void useTimeout(bool isOpponent) {
|
||||||
@@ -304,14 +225,14 @@ class PlacarController extends ChangeNotifier {
|
|||||||
}
|
}
|
||||||
isRunning = false;
|
isRunning = false;
|
||||||
timer?.cancel();
|
timer?.cancel();
|
||||||
_saveLocalBackup();
|
onUpdate();
|
||||||
notifyListeners();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String formatTime() => "${duration.inMinutes.toString().padLeft(2, '0')}:${duration.inSeconds.remainder(60).toString().padLeft(2, '0')}";
|
||||||
|
|
||||||
void handleActionDrag(BuildContext context, String action, String playerData) {
|
void handleActionDrag(BuildContext context, String action, String playerData) {
|
||||||
String playerId = playerData.replaceAll("player_my_", "").replaceAll("player_opp_", "");
|
String name = playerData.replaceAll("player_my_", "").replaceAll("player_opp_", "");
|
||||||
final stats = playerStats[playerId]!;
|
final stats = playerStats[name]!;
|
||||||
final name = playerNames[playerId]!;
|
|
||||||
|
|
||||||
if (stats["fls"]! >= 5 && action != "sub_foul") {
|
if (stats["fls"]! >= 5 && action != "sub_foul") {
|
||||||
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('🛑 $name atingiu 5 faltas e está expulso!'), backgroundColor: Colors.red));
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('🛑 $name atingiu 5 faltas e está expulso!'), backgroundColor: Colors.red));
|
||||||
@@ -320,76 +241,79 @@ class PlacarController extends ChangeNotifier {
|
|||||||
|
|
||||||
if (action == "add_pts_2" || action == "add_pts_3" || action == "miss_2" || action == "miss_3") {
|
if (action == "add_pts_2" || action == "add_pts_3" || action == "miss_2" || action == "miss_3") {
|
||||||
pendingAction = action;
|
pendingAction = action;
|
||||||
pendingPlayerId = playerData;
|
pendingPlayer = playerData;
|
||||||
isSelectingShotLocation = true;
|
isSelectingShotLocation = true;
|
||||||
} else {
|
} else {
|
||||||
commitStat(action, playerData);
|
commitStat(action, playerData);
|
||||||
}
|
}
|
||||||
notifyListeners();
|
onUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
void handleSubbing(BuildContext context, String action, String courtPlayerId, bool isOpponent) {
|
void handleSubbing(BuildContext context, String action, String courtPlayerName, bool isOpponent) {
|
||||||
if (action.startsWith("bench_my_") && !isOpponent) {
|
if (action.startsWith("bench_my_") && !isOpponent) {
|
||||||
String benchPlayerId = action.replaceAll("bench_my_", "");
|
String benchPlayer = action.replaceAll("bench_my_", "");
|
||||||
if (playerStats[benchPlayerId]!["fls"]! >= 5) return;
|
if (playerStats[benchPlayer]!["fls"]! >= 5) return;
|
||||||
int courtIndex = myCourt.indexOf(courtPlayerId);
|
int courtIndex = myCourt.indexOf(courtPlayerName);
|
||||||
int benchIndex = myBench.indexOf(benchPlayerId);
|
int benchIndex = myBench.indexOf(benchPlayer);
|
||||||
myCourt[courtIndex] = benchPlayerId;
|
myCourt[courtIndex] = benchPlayer;
|
||||||
myBench[benchIndex] = courtPlayerId;
|
myBench[benchIndex] = courtPlayerName;
|
||||||
showMyBench = false;
|
showMyBench = false;
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('Sai $courtPlayerName, Entra $benchPlayer')));
|
||||||
}
|
}
|
||||||
if (action.startsWith("bench_opp_") && isOpponent) {
|
if (action.startsWith("bench_opp_") && isOpponent) {
|
||||||
String benchPlayerId = action.replaceAll("bench_opp_", "");
|
String benchPlayer = action.replaceAll("bench_opp_", "");
|
||||||
if (playerStats[benchPlayerId]!["fls"]! >= 5) return;
|
if (playerStats[benchPlayer]!["fls"]! >= 5) return;
|
||||||
int courtIndex = oppCourt.indexOf(courtPlayerId);
|
int courtIndex = oppCourt.indexOf(courtPlayerName);
|
||||||
int benchIndex = oppBench.indexOf(benchPlayerId);
|
int benchIndex = oppBench.indexOf(benchPlayer);
|
||||||
oppCourt[courtIndex] = benchPlayerId;
|
oppCourt[courtIndex] = benchPlayer;
|
||||||
oppBench[benchIndex] = courtPlayerId;
|
oppBench[benchIndex] = courtPlayerName;
|
||||||
showOppBench = false;
|
showOppBench = false;
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('Sai $courtPlayerName, Entra $benchPlayer')));
|
||||||
}
|
}
|
||||||
_saveLocalBackup();
|
onUpdate();
|
||||||
notifyListeners();
|
|
||||||
}
|
|
||||||
|
|
||||||
void registerShotFromPopup(BuildContext context, String action, String targetPlayer, String zone, int points, double relativeX, double relativeY) {
|
|
||||||
String playerId = targetPlayer.replaceAll("player_my_", "").replaceAll("player_opp_", "");
|
|
||||||
bool isMake = action.startsWith("add_");
|
|
||||||
String name = playerNames[playerId] ?? "Jogador";
|
|
||||||
|
|
||||||
matchShots.add(ShotRecord(
|
|
||||||
relativeX: relativeX, relativeY: relativeY, isMake: isMake,
|
|
||||||
playerId: playerId, playerName: name, zone: zone, points: points
|
|
||||||
));
|
|
||||||
|
|
||||||
String finalAction = isMake ? "add_pts_$points" : "miss_$points";
|
|
||||||
commitStat(finalAction, targetPlayer);
|
|
||||||
|
|
||||||
notifyListeners();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// =========================================================================
|
||||||
|
// 👇 A MÁGICA DOS PONTOS ACONTECE AQUI 👇
|
||||||
|
// =========================================================================
|
||||||
void registerShotLocation(BuildContext context, Offset position, Size size) {
|
void registerShotLocation(BuildContext context, Offset position, Size size) {
|
||||||
if (pendingAction == null || pendingPlayerId == null) return;
|
if (pendingAction == null || pendingPlayer == null) return;
|
||||||
|
|
||||||
bool is3Pt = pendingAction!.contains("_3");
|
bool is3Pt = pendingAction!.contains("_3");
|
||||||
bool is2Pt = pendingAction!.contains("_2");
|
bool is2Pt = pendingAction!.contains("_2");
|
||||||
|
|
||||||
|
// O ÁRBITRO MATEMÁTICO COM AS TUAS VARIÁVEIS CALIBRADAS
|
||||||
if (is3Pt || is2Pt) {
|
if (is3Pt || is2Pt) {
|
||||||
bool isValid = _validateShotZone(position, size, is3Pt);
|
bool isValid = _validateShotZone(position, size, is3Pt);
|
||||||
if (!isValid) return;
|
|
||||||
|
// SE A JOGADA FOI NO SÍTIO ERRADO
|
||||||
|
if (!isValid) {
|
||||||
|
|
||||||
|
return; // <-- ESTE RETURN BLOQUEIA A GRAVAÇÃO DO PONTO!
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SE A JOGADA FOI VÁLIDA:
|
||||||
bool isMake = pendingAction!.startsWith("add_pts_");
|
bool isMake = pendingAction!.startsWith("add_pts_");
|
||||||
|
|
||||||
double relX = position.dx / size.width;
|
double relX = position.dx / size.width;
|
||||||
double relY = position.dy / size.height;
|
double relY = position.dy / size.height;
|
||||||
String pId = pendingPlayerId!.replaceAll("player_my_", "").replaceAll("player_opp_", "");
|
|
||||||
|
|
||||||
matchShots.add(ShotRecord(relativeX: relX, relativeY: relY, isMake: isMake, playerId: pId, playerName: playerNames[pId]!));
|
String name = pendingPlayer!.replaceAll("player_my_", "").replaceAll("player_opp_", "");
|
||||||
|
|
||||||
commitStat(pendingAction!, pendingPlayerId!);
|
matchShots.add(ShotRecord(
|
||||||
|
relativeX: relX,
|
||||||
|
relativeY: relY,
|
||||||
|
isMake: isMake,
|
||||||
|
playerName: name
|
||||||
|
));
|
||||||
|
|
||||||
isSelectingShotLocation = false; pendingAction = null; pendingPlayerId = null;
|
commitStat(pendingAction!, pendingPlayer!);
|
||||||
_saveLocalBackup();
|
|
||||||
notifyListeners();
|
isSelectingShotLocation = false;
|
||||||
|
pendingAction = null;
|
||||||
|
pendingPlayer = null;
|
||||||
|
onUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _validateShotZone(Offset position, Size size, bool is3Pt) {
|
bool _validateShotZone(Offset position, Size size, bool is3Pt) {
|
||||||
@@ -405,10 +329,13 @@ class PlacarController extends ChangeNotifier {
|
|||||||
|
|
||||||
bool isInside2Pts;
|
bool isInside2Pts;
|
||||||
|
|
||||||
|
// Lógica das laterais (Cantos)
|
||||||
if (distFromCenterY > cornerY) {
|
if (distFromCenterY > cornerY) {
|
||||||
double distToBaseline = isLeftHalf ? relX : (1.0 - relX);
|
double distToBaseline = isLeftHalf ? relX : (1.0 - relX);
|
||||||
isInside2Pts = distToBaseline <= hoopBaseX;
|
isInside2Pts = distToBaseline <= hoopBaseX;
|
||||||
} else {
|
}
|
||||||
|
// Lógica da Curva Frontal
|
||||||
|
else {
|
||||||
double dx = (relX - hoopX) * aspectRatio;
|
double dx = (relX - hoopX) * aspectRatio;
|
||||||
double dy = (relY - hoopY);
|
double dy = (relY - hoopY);
|
||||||
double distanceToHoop = math.sqrt((dx * dx) + (dy * dy));
|
double distanceToHoop = math.sqrt((dx * dx) + (dy * dy));
|
||||||
@@ -418,146 +345,170 @@ class PlacarController extends ChangeNotifier {
|
|||||||
if (is3Pt) return !isInside2Pts;
|
if (is3Pt) return !isInside2Pts;
|
||||||
return isInside2Pts;
|
return isInside2Pts;
|
||||||
}
|
}
|
||||||
|
// 👆 ===================================================================== 👆
|
||||||
|
|
||||||
void cancelShotLocation() {
|
void cancelShotLocation() {
|
||||||
isSelectingShotLocation = false; pendingAction = null; pendingPlayerId = null; notifyListeners();
|
isSelectingShotLocation = false; pendingAction = null; pendingPlayer = null; onUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
void commitStat(String action, String playerData) {
|
void commitStat(String action, String playerData) {
|
||||||
bool isOpponent = playerData.startsWith("player_opp_");
|
bool isOpponent = playerData.startsWith("player_opp_");
|
||||||
String playerId = playerData.replaceAll("player_my_", "").replaceAll("player_opp_", "");
|
String name = playerData.replaceAll("player_my_", "").replaceAll("player_opp_", "");
|
||||||
final stats = playerStats[playerId]!;
|
final stats = playerStats[name]!;
|
||||||
final name = playerNames[playerId] ?? "Jogador";
|
|
||||||
|
|
||||||
String logText = "";
|
|
||||||
|
|
||||||
if (action.startsWith("add_pts_")) {
|
if (action.startsWith("add_pts_")) {
|
||||||
int pts = int.parse(action.split("_").last);
|
int pts = int.parse(action.split("_").last);
|
||||||
if (isOpponent) opponentScore += pts; else myScore += pts;
|
if (isOpponent) opponentScore += pts; else myScore += pts;
|
||||||
stats["pts"] = stats["pts"]! + pts;
|
stats["pts"] = stats["pts"]! + pts;
|
||||||
if (pts == 2) { stats["fgm"] = stats["fgm"]! + 1; stats["fga"] = stats["fga"]! + 1; stats["p2m"] = stats["p2m"]! + 1; stats["p2a"] = stats["p2a"]! + 1; }
|
if (pts == 2 || pts == 3) { stats["fgm"] = stats["fgm"]! + 1; stats["fga"] = stats["fga"]! + 1; }
|
||||||
if (pts == 3) { stats["fgm"] = stats["fgm"]! + 1; stats["fga"] = stats["fga"]! + 1; stats["p3m"] = stats["p3m"]! + 1; stats["p3a"] = stats["p3a"]! + 1; }
|
|
||||||
if (pts == 1) { stats["ftm"] = stats["ftm"]! + 1; stats["fta"] = stats["fta"]! + 1; }
|
if (pts == 1) { stats["ftm"] = stats["ftm"]! + 1; stats["fta"] = stats["fta"]! + 1; }
|
||||||
logText = "marcou $pts pontos 🏀";
|
|
||||||
}
|
}
|
||||||
else if (action == "miss_1") { stats["fta"] = stats["fta"]! + 1; logText = "falhou lance livre ❌"; }
|
else if (action.startsWith("sub_pts_")) {
|
||||||
else if (action == "miss_2") { stats["fga"] = stats["fga"]! + 1; stats["p2a"] = stats["p2a"]! + 1; logText = "falhou lançamento de 2 ❌"; }
|
int pts = int.parse(action.split("_").last);
|
||||||
else if (action == "miss_3") { stats["fga"] = stats["fga"]! + 1; stats["p3a"] = stats["p3a"]! + 1; logText = "falhou lançamento de 3 ❌"; }
|
if (isOpponent) { opponentScore = (opponentScore - pts < 0) ? 0 : opponentScore - pts; }
|
||||||
else if (action == "add_orb") { stats["orb"] = stats["orb"]! + 1; stats["rbs"] = stats["rbs"]! + 1; logText = "ganhou ressalto ofensivo 🔄"; }
|
else { myScore = (myScore - pts < 0) ? 0 : myScore - pts; }
|
||||||
else if (action == "add_drb") { stats["drb"] = stats["drb"]! + 1; stats["rbs"] = stats["rbs"]! + 1; logText = "ganhou ressalto defensivo 🛡️"; }
|
stats["pts"] = (stats["pts"]! - pts < 0) ? 0 : stats["pts"]! - pts;
|
||||||
else if (action == "add_ast") { stats["ast"] = stats["ast"]! + 1; logText = "fez uma assistência 🤝"; }
|
if (pts == 2 || pts == 3) {
|
||||||
else if (action == "add_stl") { stats["stl"] = stats["stl"]! + 1; logText = "roubou a bola 🥷"; }
|
if (stats["fgm"]! > 0) stats["fgm"] = stats["fgm"]! - 1;
|
||||||
else if (action == "add_blk") { stats["blk"] = stats["blk"]! + 1; logText = "fez um desarme (bloco) ✋"; }
|
if (stats["fga"]! > 0) stats["fga"] = stats["fga"]! - 1;
|
||||||
|
}
|
||||||
|
if (pts == 1) {
|
||||||
|
if (stats["ftm"]! > 0) stats["ftm"] = stats["ftm"]! - 1;
|
||||||
|
if (stats["fta"]! > 0) stats["fta"] = stats["fta"]! - 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (action == "miss_1") { stats["fta"] = stats["fta"]! + 1; }
|
||||||
|
else if (action == "miss_2" || action == "miss_3") { stats["fga"] = stats["fga"]! + 1; }
|
||||||
|
else if (action == "add_orb") { stats["orb"] = stats["orb"]! + 1; stats["rbs"] = stats["rbs"]! + 1; }
|
||||||
|
else if (action == "add_drb") { stats["drb"] = stats["drb"]! + 1; stats["rbs"] = stats["rbs"]! + 1; }
|
||||||
|
else if (action == "add_ast") { stats["ast"] = stats["ast"]! + 1; }
|
||||||
|
else if (action == "add_stl") { stats["stl"] = stats["stl"]! + 1; }
|
||||||
|
else if (action == "add_tov") { stats["tov"] = stats["tov"]! + 1; }
|
||||||
|
else if (action == "add_blk") { stats["blk"] = stats["blk"]! + 1; }
|
||||||
else if (action == "add_foul") {
|
else if (action == "add_foul") {
|
||||||
stats["fls"] = stats["fls"]! + 1;
|
stats["fls"] = stats["fls"]! + 1;
|
||||||
if (isOpponent) opponentFouls++; else myFouls++;
|
if (isOpponent) { opponentFouls++; } else { myFouls++; }
|
||||||
logText = "cometeu falta ⚠️";
|
|
||||||
}
|
}
|
||||||
else if (action == "add_so") { stats["so"] = stats["so"]! + 1; logText = "sofreu uma falta 🤕"; }
|
else if (action == "sub_foul") {
|
||||||
else if (action == "add_il") { stats["il"] = stats["il"]! + 1; logText = "intercetou um lançamento 🛑"; }
|
if (stats["fls"]! > 0) stats["fls"] = stats["fls"]! - 1;
|
||||||
else if (action == "add_li") { stats["li"] = stats["li"]! + 1; logText = "teve o lançamento intercetado 🚫"; }
|
if (isOpponent) { if (opponentFouls > 0) opponentFouls--; } else { if (myFouls > 0) myFouls--; }
|
||||||
|
|
||||||
// Registo avançado de Bolas Perdidas (TOV)
|
|
||||||
else if (action == "add_tov") { stats["tov"] = stats["tov"]! + 1; logText = "fez um passe ruim 🤦"; }
|
|
||||||
else if (action == "add_pa") { stats["pa"] = stats["pa"]! + 1; stats["tov"] = stats["tov"]! + 1; logText = "cometeu passos 🚶"; }
|
|
||||||
else if (action == "add_3s") { stats["tres_s"] = stats["tres_s"]! + 1; stats["tov"] = stats["tov"]! + 1; logText = "violação de 3 segundos ⏱️"; }
|
|
||||||
else if (action == "add_24s") { stats["tov"] = stats["tov"]! + 1; logText = "violação de 24 segundos ⏱️"; }
|
|
||||||
else if (action == "add_dr") { stats["dr"] = stats["dr"]! + 1; stats["tov"] = stats["tov"]! + 1; logText = "fez drible duplo 🏀"; }
|
|
||||||
|
|
||||||
if (logText.isNotEmpty) {
|
|
||||||
String time = "${durationNotifier.value.inMinutes.toString().padLeft(2, '0')}:${durationNotifier.value.inSeconds.remainder(60).toString().padLeft(2, '0')}";
|
|
||||||
playByPlay.insert(0, "P$currentQuarter - $time: $name $logText");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_saveLocalBackup();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> saveGameStats(BuildContext context) async {
|
Future<void> saveGameStats(BuildContext context) async {
|
||||||
final supabase = Supabase.instance.client;
|
final supabase = Supabase.instance.client;
|
||||||
isSaving = true;
|
isSaving = true;
|
||||||
notifyListeners();
|
onUpdate();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
bool isGameFinishedNow = currentQuarter >= 4 && durationNotifier.value.inSeconds == 0;
|
bool isGameFinishedNow = currentQuarter >= 4 && duration.inSeconds == 0;
|
||||||
String newStatus = isGameFinishedNow ? 'Terminado' : 'Pausado';
|
String newStatus = isGameFinishedNow ? 'Terminado' : 'Pausado';
|
||||||
|
|
||||||
String topPtsName = '---'; int maxPts = -1;
|
String topPtsName = '---'; int maxPts = -1;
|
||||||
String topAstName = '---'; int maxAst = -1;
|
String topAstName = '---'; int maxAst = -1;
|
||||||
String topRbsName = '---'; int maxRbs = -1;
|
String topRbsName = '---'; int maxRbs = -1;
|
||||||
String mvpName = '---'; double maxMvpScore = -999.0;
|
String topDefName = '---'; int maxDef = -1;
|
||||||
|
String mvpName = '---'; int maxMvpScore = -1;
|
||||||
|
|
||||||
playerStats.forEach((playerId, stats) {
|
playerStats.forEach((playerName, stats) {
|
||||||
int pts = stats['pts'] ?? 0;
|
int pts = stats['pts'] ?? 0;
|
||||||
int ast = stats['ast'] ?? 0;
|
int ast = stats['ast'] ?? 0;
|
||||||
int rbs = stats['rbs'] ?? 0;
|
int rbs = stats['rbs'] ?? 0;
|
||||||
|
int stl = stats['stl'] ?? 0;
|
||||||
|
int blk = stats['blk'] ?? 0;
|
||||||
|
|
||||||
int minJogados = (stats['min'] ?? 0) > 0 ? stats['min']! : 40;
|
int defScore = stl + blk;
|
||||||
int tr = rbs;
|
int mvpScore = pts + ast + rbs + defScore;
|
||||||
int br = stats['stl'] ?? 0;
|
|
||||||
int bp = stats['tov'] ?? 0;
|
|
||||||
int lFalhados = (stats['fga'] ?? 0) - (stats['fgm'] ?? 0);
|
|
||||||
int llFalhados = (stats['fta'] ?? 0) - (stats['ftm'] ?? 0);
|
|
||||||
|
|
||||||
double mvpScore = ((pts * 0.30) + (tr * 0.20) + (ast * 0.35) + (br * 0.15)) -
|
if (pts > maxPts && pts > 0) { maxPts = pts; topPtsName = '$playerName ($pts)'; }
|
||||||
((bp * 0.35) + (lFalhados * 0.30) + (llFalhados * 0.35));
|
if (ast > maxAst && ast > 0) { maxAst = ast; topAstName = '$playerName ($ast)'; }
|
||||||
mvpScore = mvpScore * (minJogados / 40.0);
|
if (rbs > maxRbs && rbs > 0) { maxRbs = rbs; topRbsName = '$playerName ($rbs)'; }
|
||||||
|
if (defScore > maxDef && defScore > 0) { maxDef = defScore; topDefName = '$playerName ($defScore)'; }
|
||||||
String pName = playerNames[playerId] ?? '---';
|
if (mvpScore > maxMvpScore && mvpScore > 0) { maxMvpScore = mvpScore; mvpName = playerName; }
|
||||||
|
|
||||||
if (pts > maxPts && pts > 0) { maxPts = pts; topPtsName = '$pName ($pts)'; }
|
|
||||||
if (ast > maxAst && ast > 0) { maxAst = ast; topAstName = '$pName ($ast)'; }
|
|
||||||
if (rbs > maxRbs && rbs > 0) { maxRbs = rbs; topRbsName = '$pName ($rbs)'; }
|
|
||||||
if (mvpScore > maxMvpScore) { maxMvpScore = mvpScore; mvpName = '$pName (${mvpScore.toStringAsFixed(1)})'; }
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await supabase.from('games').update({
|
await supabase.from('games').update({
|
||||||
'my_score': myScore, 'opponent_score': opponentScore,
|
'my_score': myScore,
|
||||||
'remaining_seconds': durationNotifier.value.inSeconds,
|
'opponent_score': opponentScore,
|
||||||
'my_timeouts': myTimeoutsUsed, 'opp_timeouts': opponentTimeoutsUsed,
|
'remaining_seconds': duration.inSeconds,
|
||||||
'current_quarter': currentQuarter, 'status': newStatus,
|
'my_timeouts': myTimeoutsUsed,
|
||||||
'top_pts_name': topPtsName, 'top_ast_name': topAstName,
|
'opp_timeouts': opponentTimeoutsUsed,
|
||||||
'top_rbs_name': topRbsName, 'mvp_name': mvpName,
|
'current_quarter': currentQuarter,
|
||||||
'play_by_play': playByPlay,
|
'status': newStatus,
|
||||||
|
'top_pts_name': topPtsName,
|
||||||
|
'top_ast_name': topAstName,
|
||||||
|
'top_rbs_name': topRbsName,
|
||||||
|
'top_def_name': topDefName,
|
||||||
|
'mvp_name': mvpName,
|
||||||
}).eq('id', gameId);
|
}).eq('id', gameId);
|
||||||
|
|
||||||
|
if (isGameFinishedNow && !gameWasAlreadyFinished && myTeamDbId != null && oppTeamDbId != null) {
|
||||||
|
|
||||||
|
final teamsData = await supabase.from('teams').select('id, wins, losses, draws').inFilter('id', [myTeamDbId, oppTeamDbId]);
|
||||||
|
|
||||||
|
Map<String, dynamic> myTeamUpdate = {};
|
||||||
|
Map<String, dynamic> oppTeamUpdate = {};
|
||||||
|
|
||||||
|
for(var t in teamsData) {
|
||||||
|
if(t['id'].toString() == myTeamDbId) myTeamUpdate = Map.from(t);
|
||||||
|
if(t['id'].toString() == oppTeamDbId) oppTeamUpdate = Map.from(t);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (myScore > opponentScore) {
|
||||||
|
myTeamUpdate['wins'] = (myTeamUpdate['wins'] ?? 0) + 1;
|
||||||
|
oppTeamUpdate['losses'] = (oppTeamUpdate['losses'] ?? 0) + 1;
|
||||||
|
} else if (myScore < opponentScore) {
|
||||||
|
myTeamUpdate['losses'] = (myTeamUpdate['losses'] ?? 0) + 1;
|
||||||
|
oppTeamUpdate['wins'] = (oppTeamUpdate['wins'] ?? 0) + 1;
|
||||||
|
} else {
|
||||||
|
myTeamUpdate['draws'] = (myTeamUpdate['draws'] ?? 0) + 1;
|
||||||
|
oppTeamUpdate['draws'] = (oppTeamUpdate['draws'] ?? 0) + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
await supabase.from('teams').update({
|
||||||
|
'wins': myTeamUpdate['wins'], 'losses': myTeamUpdate['losses'], 'draws': myTeamUpdate['draws']
|
||||||
|
}).eq('id', myTeamDbId!);
|
||||||
|
|
||||||
|
await supabase.from('teams').update({
|
||||||
|
'wins': oppTeamUpdate['wins'], 'losses': oppTeamUpdate['losses'], 'draws': oppTeamUpdate['draws']
|
||||||
|
}).eq('id', oppTeamDbId!);
|
||||||
|
|
||||||
|
gameWasAlreadyFinished = true;
|
||||||
|
}
|
||||||
|
|
||||||
List<Map<String, dynamic>> batchStats = [];
|
List<Map<String, dynamic>> batchStats = [];
|
||||||
playerStats.forEach((playerId, stats) {
|
playerStats.forEach((playerName, stats) {
|
||||||
if (!playerId.startsWith("fake_")) {
|
String? memberDbId = playerDbIds[playerName];
|
||||||
bool isMyTeamPlayer = myCourt.contains(playerId) || myBench.contains(playerId);
|
if (memberDbId != null && stats.values.any((val) => val > 0)) {
|
||||||
|
bool isMyTeamPlayer = myCourt.contains(playerName) || myBench.contains(playerName);
|
||||||
batchStats.add({
|
batchStats.add({
|
||||||
'game_id': gameId, 'member_id': playerId, 'team_id': isMyTeamPlayer ? myTeamDbId! : oppTeamDbId!,
|
'game_id': gameId, 'member_id': memberDbId, 'team_id': isMyTeamPlayer ? myTeamDbId! : oppTeamDbId!,
|
||||||
'pts': stats['pts'], 'rbs': stats['rbs'], 'ast': stats['ast'], 'stl': stats['stl'], 'blk': stats['blk'],
|
'pts': stats['pts'], 'rbs': stats['rbs'], 'ast': stats['ast'], 'stl': stats['stl'], 'blk': stats['blk'], 'tov': stats['tov'], 'fls': stats['fls'], 'fgm': stats['fgm'], 'fga': stats['fga'], 'ftm': stats['ftm'], 'fta': stats['fta'], 'orb': stats['orb'], 'drb': stats['drb'],
|
||||||
'tov': stats['tov'], 'fls': stats['fls'], 'fgm': stats['fgm'], 'fga': stats['fga'], 'ftm': stats['ftm'],
|
|
||||||
'fta': stats['fta'], 'orb': stats['orb'], 'drb': stats['drb'], 'p2m': stats['p2m'], 'p2a': stats['p2a'],
|
|
||||||
'p3m': stats['p3m'], 'p3a': stats['p3a'],
|
|
||||||
'so': stats['so'], 'il': stats['il'], 'li': stats['li'], 'pa': stats['pa'], 'tres_s': stats['tres_s'],
|
|
||||||
'dr': stats['dr'], 'min': stats['min'],
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
await supabase.from('player_stats').delete().eq('game_id', gameId);
|
await supabase.from('player_stats').delete().eq('game_id', gameId);
|
||||||
if (batchStats.isNotEmpty) await supabase.from('player_stats').insert(batchStats);
|
if (batchStats.isNotEmpty) {
|
||||||
|
await supabase.from('player_stats').insert(batchStats);
|
||||||
|
}
|
||||||
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
if (context.mounted) {
|
||||||
await prefs.remove('backup_$gameId');
|
ScaffoldMessenger.of(context).showSnackBar(const SnackBar(content: Text('Estatísticas e Resultados guardados com Sucesso!'), backgroundColor: Colors.green));
|
||||||
|
}
|
||||||
if (context.mounted) ScaffoldMessenger.of(context).showSnackBar(const SnackBar(content: Text('Guardado com Sucesso!'), backgroundColor: Colors.green));
|
|
||||||
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
debugPrint("Erro ao gravar estatísticas: $e");
|
debugPrint("Erro ao gravar estatísticas: $e");
|
||||||
if (context.mounted) ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('Erro ao guardar: $e'), backgroundColor: Colors.red));
|
if (context.mounted) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('Erro ao guardar: $e'), backgroundColor: Colors.red));
|
||||||
|
}
|
||||||
} finally {
|
} finally {
|
||||||
isSaving = false;
|
isSaving = false;
|
||||||
notifyListeners();
|
onUpdate();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
void dispose() {
|
||||||
timer?.cancel();
|
timer?.cancel();
|
||||||
super.dispose();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void registerFoul(String s, String foulType, String t) {}
|
|
||||||
}
|
}
|
||||||
@@ -1,68 +1,50 @@
|
|||||||
import 'dart:io';
|
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
|
|
||||||
class TeamController {
|
class TeamController {
|
||||||
final _supabase = Supabase.instance.client;
|
final _supabase = Supabase.instance.client;
|
||||||
|
|
||||||
// 1. STREAM (Realtime)
|
// 1. Variável fixa para guardar o Stream principal
|
||||||
Stream<List<Map<String, dynamic>>> get teamsStream {
|
late final Stream<List<Map<String, dynamic>>> teamsStream;
|
||||||
final userId = _supabase.auth.currentUser?.id;
|
|
||||||
|
|
||||||
if (userId == null) return const Stream.empty();
|
// 2. Dicionário (Cache) para não recriar Streams de contagem repetidos
|
||||||
|
final Map<String, Stream<int>> _playerCountStreams = {};
|
||||||
|
|
||||||
return _supabase
|
TeamController() {
|
||||||
|
// INICIALIZAÇÃO: O stream é criado APENAS UMA VEZ quando abres a página!
|
||||||
|
teamsStream = _supabase
|
||||||
.from('teams')
|
.from('teams')
|
||||||
.stream(primaryKey: ['id'])
|
.stream(primaryKey: ['id'])
|
||||||
.eq('user_id', userId); // ✅ Bem feito, este já estava certo!
|
.order('name', ascending: true)
|
||||||
|
.map((data) => List<Map<String, dynamic>>.from(data));
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. CRIAR (Agora guarda o dono da equipa!)
|
// CRIAR
|
||||||
Future<void> createTeam(String name, String season, File? imageFile) async {
|
Future<void> createTeam(String name, String season, String? imageUrl) async {
|
||||||
try {
|
try {
|
||||||
final userId = _supabase.auth.currentUser?.id;
|
|
||||||
if (userId == null) throw Exception("Utilizador não autenticado.");
|
|
||||||
|
|
||||||
String? uploadedImageUrl;
|
|
||||||
|
|
||||||
// Se o utilizador escolheu uma imagem, fazemos o upload primeiro
|
|
||||||
if (imageFile != null) {
|
|
||||||
final fileName = '${userId}_${DateTime.now().millisecondsSinceEpoch}.png';
|
|
||||||
final storagePath = 'teams/$fileName';
|
|
||||||
|
|
||||||
await _supabase.storage.from('avatars').upload(
|
|
||||||
storagePath,
|
|
||||||
imageFile,
|
|
||||||
fileOptions: const FileOptions(cacheControl: '3600', upsert: true)
|
|
||||||
);
|
|
||||||
|
|
||||||
uploadedImageUrl = _supabase.storage.from('avatars').getPublicUrl(storagePath);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Agora insere a equipa na base de dados com o ID DO DONO!
|
|
||||||
await _supabase.from('teams').insert({
|
await _supabase.from('teams').insert({
|
||||||
'user_id': userId, // 👈 CRUCIAL: Diz à base de dados de quem é esta equipa!
|
|
||||||
'name': name,
|
'name': name,
|
||||||
'season': season,
|
'season': season,
|
||||||
'image_url': uploadedImageUrl ?? '',
|
'image_url': imageUrl,
|
||||||
'is_favorite': false,
|
'is_favorite': false,
|
||||||
});
|
});
|
||||||
print("✅ Equipa guardada no Supabase com dono associado!");
|
print("✅ Equipa guardada no Supabase!");
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print("❌ Erro ao criar equipa: $e");
|
print("❌ Erro ao criar: $e");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. ELIMINAR
|
// ELIMINAR
|
||||||
Future<void> deleteTeam(String id) async {
|
Future<void> deleteTeam(String id) async {
|
||||||
try {
|
try {
|
||||||
// Como segurança extra, podemos garantir que só apaga se for o dono (opcional se tiveres RLS no Supabase)
|
|
||||||
await _supabase.from('teams').delete().eq('id', id);
|
await _supabase.from('teams').delete().eq('id', id);
|
||||||
|
// Limpa o cache deste teamId se a equipa for apagada
|
||||||
|
_playerCountStreams.remove(id);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print("❌ Erro ao eliminar: $e");
|
print("❌ Erro ao eliminar: $e");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. FAVORITAR
|
// FAVORITAR
|
||||||
Future<void> toggleFavorite(String teamId, bool currentStatus) async {
|
Future<void> toggleFavorite(String teamId, bool currentStatus) async {
|
||||||
try {
|
try {
|
||||||
await _supabase
|
await _supabase
|
||||||
@@ -74,29 +56,27 @@ class TeamController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. CONTAR JOGADORES (LEITURA ÚNICA)
|
// CONTAR JOGADORES (AGORA COM CACHE DE MEMÓRIA!)
|
||||||
Future<int> getPlayerCount(String teamId) async {
|
Stream<int> getPlayerCountStream(String teamId) {
|
||||||
try {
|
// Se já criámos um "Tubo de ligação" para esta equipa, REUTILIZA-O!
|
||||||
final count = await _supabase.from('members').count().eq('team_id', teamId);
|
if (_playerCountStreams.containsKey(teamId)) {
|
||||||
return count;
|
return _playerCountStreams[teamId]!;
|
||||||
} catch (e) {
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Se é a primeira vez que pede esta equipa, cria a ligação e guarda na memória
|
||||||
|
final newStream = _supabase
|
||||||
|
.from('members')
|
||||||
|
.stream(primaryKey: ['id'])
|
||||||
|
.eq('team_id', teamId)
|
||||||
|
.map((data) => data.length);
|
||||||
|
|
||||||
|
_playerCountStreams[teamId] = newStream; // Guarda no dicionário
|
||||||
|
return newStream;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. VIEW DAS EQUIPAS (AQUI ESTAVA O TEU ERRO DE LISTAGEM!)
|
// LIMPEZA FINAL QUANDO SAÍMOS DA PÁGINA
|
||||||
Future<List<Map<String, dynamic>>> getTeamsWithStats() async {
|
void dispose() {
|
||||||
final userId = _supabase.auth.currentUser?.id;
|
// Limpamos o dicionário de streams para libertar memória RAM
|
||||||
if (userId == null) return []; // Retorna lista vazia se não houver login
|
_playerCountStreams.clear();
|
||||||
|
|
||||||
final data = await _supabase
|
|
||||||
.from('teams_with_stats')
|
|
||||||
.select('*')
|
|
||||||
.eq('user_id', userId) // 👈 CRUCIAL: Só puxa as estatísticas das tuas equipas!
|
|
||||||
.order('name', ascending: true);
|
|
||||||
|
|
||||||
return List<Map<String, dynamic>>.from(data);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void dispose() {}
|
|
||||||
}
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import '../dados_grafico.dart';
|
import '../dados_grafico.dart'; // Ajusta o caminho se der erro de import
|
||||||
|
|
||||||
class PieChartController extends ChangeNotifier {
|
class PieChartController extends ChangeNotifier {
|
||||||
PieChartData _chartData = const PieChartData(victories: 0, defeats: 0, draws: 0);
|
PieChartData _chartData = const PieChartData(victories: 0, defeats: 0, draws: 0);
|
||||||
@@ -10,7 +10,7 @@ class PieChartController extends ChangeNotifier {
|
|||||||
_chartData = PieChartData(
|
_chartData = PieChartData(
|
||||||
victories: victories ?? _chartData.victories,
|
victories: victories ?? _chartData.victories,
|
||||||
defeats: defeats ?? _chartData.defeats,
|
defeats: defeats ?? _chartData.defeats,
|
||||||
draws: draws ?? _chartData.draws,
|
draws: draws ?? _chartData.draws, // 👇 AGORA ELE ACEITA OS EMPATES
|
||||||
);
|
);
|
||||||
notifyListeners();
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
class PieChartData {
|
class PieChartData {
|
||||||
final int victories;
|
final int victories;
|
||||||
final int defeats;
|
final int defeats;
|
||||||
final int draws;
|
final int draws; // 👇 AQUI ESTÃO OS EMPATES
|
||||||
|
|
||||||
const PieChartData({
|
const PieChartData({
|
||||||
required this.victories,
|
required this.victories,
|
||||||
@@ -9,6 +9,7 @@ class PieChartData {
|
|||||||
this.draws = 0,
|
this.draws = 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 👇 MATEMÁTICA ATUALIZADA 👇
|
||||||
int get total => victories + defeats + draws;
|
int get total => victories + defeats + draws;
|
||||||
|
|
||||||
double get victoryPercentage => total > 0 ? victories / total : 0;
|
double get victoryPercentage => total > 0 ? victories / total : 0;
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/grafico%20de%20pizza/widgets/grafico_widgets.dart';
|
import 'package:playmaker/grafico%20de%20pizza/widgets/grafico_widgets.dart';
|
||||||
import 'dados_grafico.dart';
|
import 'dados_grafico.dart';
|
||||||
import 'package:playmaker/classe/theme.dart'; // 👇 IMPORT DO TEMA ADICIONADO PARA USARMOS O primaryRed
|
|
||||||
import 'dart:math' as math;
|
|
||||||
|
|
||||||
class PieChartCard extends StatefulWidget {
|
class PieChartCard extends StatefulWidget {
|
||||||
final int victories;
|
final int victories;
|
||||||
@@ -10,7 +8,7 @@ class PieChartCard extends StatefulWidget {
|
|||||||
final int draws;
|
final int draws;
|
||||||
final String title;
|
final String title;
|
||||||
final String subtitle;
|
final String subtitle;
|
||||||
final Color? backgroundColor;
|
final Color backgroundColor;
|
||||||
final VoidCallback? onTap;
|
final VoidCallback? onTap;
|
||||||
final double sf;
|
final double sf;
|
||||||
|
|
||||||
@@ -22,7 +20,7 @@ class PieChartCard extends StatefulWidget {
|
|||||||
this.title = 'DESEMPENHO',
|
this.title = 'DESEMPENHO',
|
||||||
this.subtitle = 'Temporada',
|
this.subtitle = 'Temporada',
|
||||||
this.onTap,
|
this.onTap,
|
||||||
this.backgroundColor,
|
required this.backgroundColor,
|
||||||
this.sf = 1.0,
|
this.sf = 1.0,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -61,31 +59,30 @@ class _PieChartCardState extends State<PieChartCard> with SingleTickerProviderSt
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final data = PieChartData(victories: widget.victories, defeats: widget.defeats, draws: widget.draws);
|
final data = PieChartData(victories: widget.victories, defeats: widget.defeats, draws: widget.draws);
|
||||||
|
|
||||||
// 👇 BLINDAGEM DO FUNDO E DO TEXTO PARA MODO CLARO/ESCURO
|
return AnimatedBuilder(
|
||||||
final Color cardColor = widget.backgroundColor ?? Theme.of(context).cardTheme.color ?? (Theme.of(context).brightness == Brightness.dark ? const Color(0xFF1E1E1E) : Colors.white);
|
|
||||||
final Color textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
|
|
||||||
return AnimatedBuilder(
|
|
||||||
animation: _animation,
|
animation: _animation,
|
||||||
builder: (context, child) {
|
builder: (context, child) {
|
||||||
return Transform.scale(
|
return Transform.scale(
|
||||||
|
// O scale pode passar de 1.0 (efeito back), mas a opacidade NÃO
|
||||||
scale: 0.95 + (_animation.value * 0.05),
|
scale: 0.95 + (_animation.value * 0.05),
|
||||||
child: Opacity(opacity: _animation.value.clamp(0.0, 1.0), child: child),
|
child: Opacity(
|
||||||
|
// 👇 AQUI ESTÁ A FIX: Garante que fica entre 0 e 1
|
||||||
|
opacity: _animation.value.clamp(0.0, 1.0),
|
||||||
|
child: child,
|
||||||
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
child: Card(
|
child: Card(
|
||||||
margin: EdgeInsets.zero,
|
margin: EdgeInsets.zero,
|
||||||
elevation: 0, // Ajustado para não ter sombra dupla, já que o tema pode ter
|
elevation: 4,
|
||||||
clipBehavior: Clip.antiAlias,
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14)),
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(20),
|
|
||||||
side: BorderSide(color: Colors.grey.withOpacity(0.15)), // Borda suave igual ao resto da app
|
|
||||||
),
|
|
||||||
child: InkWell(
|
child: InkWell(
|
||||||
onTap: widget.onTap,
|
onTap: widget.onTap,
|
||||||
|
borderRadius: BorderRadius.circular(14),
|
||||||
child: Container(
|
child: Container(
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: cardColor, // 👇 APLICA A COR BLINDADA
|
borderRadius: BorderRadius.circular(14),
|
||||||
|
gradient: LinearGradient(begin: Alignment.topLeft, end: Alignment.bottomRight, colors: [widget.backgroundColor.withOpacity(0.9), widget.backgroundColor.withOpacity(0.7)]),
|
||||||
),
|
),
|
||||||
child: LayoutBuilder(
|
child: LayoutBuilder(
|
||||||
builder: (context, constraints) {
|
builder: (context, constraints) {
|
||||||
@@ -93,43 +90,29 @@ class _PieChartCardState extends State<PieChartCard> with SingleTickerProviderSt
|
|||||||
final double cw = constraints.maxWidth;
|
final double cw = constraints.maxWidth;
|
||||||
|
|
||||||
return Padding(
|
return Padding(
|
||||||
padding: EdgeInsets.symmetric(horizontal: cw * 0.05, vertical: ch * 0.03),
|
padding: EdgeInsets.all(cw * 0.06),
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
// --- CABEÇALHO --- (👇 MANTIDO ALINHADO À ESQUERDA)
|
// 👇 TÍTULOS UM POUCO MAIS PRESENTES
|
||||||
FittedBox(
|
FittedBox(
|
||||||
fit: BoxFit.scaleDown,
|
fit: BoxFit.scaleDown,
|
||||||
child: Text(widget.title.toUpperCase(),
|
child: Text(widget.title.toUpperCase(), style: TextStyle(fontSize: ch * 0.06, fontWeight: FontWeight.bold, color: Colors.white.withOpacity(0.9), letterSpacing: 1.0)),
|
||||||
style: TextStyle(
|
|
||||||
fontSize: ch * 0.045,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: AppTheme.primaryRed, // 👇 USANDO O TEU primaryRed
|
|
||||||
letterSpacing: 1.2
|
|
||||||
)
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
Text(widget.subtitle,
|
FittedBox(
|
||||||
style: TextStyle(
|
fit: BoxFit.scaleDown,
|
||||||
fontSize: ch * 0.055,
|
child: Text(widget.subtitle, style: TextStyle(fontSize: ch * 0.07, fontWeight: FontWeight.bold, color: Colors.white)),
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: AppTheme.backgroundLight, // 👇 USANDO O TEU backgroundLight
|
|
||||||
)
|
|
||||||
),
|
),
|
||||||
|
|
||||||
const Expanded(flex: 1, child: SizedBox()),
|
SizedBox(height: ch * 0.03),
|
||||||
|
|
||||||
// --- MIOLO (GRÁFICO MAIOR À ESQUERDA + STATS) ---
|
// MEIO (GRÁFICO + ESTATÍSTICAS)
|
||||||
Expanded(
|
Expanded(
|
||||||
flex: 9,
|
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.end, // Changed from spaceBetween to end to push stats more to the right
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
// 1. Lado Esquerdo: Donut Chart
|
Expanded(
|
||||||
// 👇 MUDANÇA AQUI: Gráfico ainda maior! cw * 0.52
|
flex: 1,
|
||||||
SizedBox(
|
|
||||||
width: cw * 0.52,
|
|
||||||
height: cw * 0.52,
|
|
||||||
child: PieChartWidget(
|
child: PieChartWidget(
|
||||||
victoryPercentage: data.victoryPercentage,
|
victoryPercentage: data.victoryPercentage,
|
||||||
defeatPercentage: data.defeatPercentage,
|
defeatPercentage: data.defeatPercentage,
|
||||||
@@ -137,57 +120,56 @@ class _PieChartCardState extends State<PieChartCard> with SingleTickerProviderSt
|
|||||||
sf: widget.sf,
|
sf: widget.sf,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
SizedBox(width: cw * 0.05),
|
||||||
SizedBox(width: cw * 0.005), // Reduzi o espaço no meio para dar lugar ao gráfico
|
|
||||||
|
|
||||||
// 2. Lado Direito: Números Dinâmicos
|
|
||||||
Expanded(
|
Expanded(
|
||||||
child: FittedBox(
|
flex: 1,
|
||||||
alignment: Alignment.centerRight, // Encosta os números à direita
|
child: Column(
|
||||||
fit: BoxFit.scaleDown,
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
child: Column(
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
mainAxisSize: MainAxisSize.min,
|
children: [
|
||||||
crossAxisAlignment: CrossAxisAlignment.end, // Alinha os números à direita para ficar arrumado
|
_buildDynStatRow("VIT", data.victories.toString(), (data.victoryPercentage * 100).toStringAsFixed(0), Colors.green, ch),
|
||||||
children: [
|
_buildDynStatRow("EMP", data.draws.toString(), (data.drawPercentage * 100).toStringAsFixed(0), Colors.yellow, ch),
|
||||||
_buildDynStatRow("VIT", data.victories.toString(), (data.victoryPercentage * 100).toStringAsFixed(0), Colors.green, textColor, ch, cw),
|
_buildDynStatRow("DER", data.defeats.toString(), (data.defeatPercentage * 100).toStringAsFixed(0), Colors.red, ch),
|
||||||
_buildDynStatRow("EMP", data.draws.toString(), (data.drawPercentage * 100).toStringAsFixed(0), Colors.amber, textColor, ch, cw),
|
_buildDynDivider(ch),
|
||||||
_buildDynStatRow("DER", data.defeats.toString(), (data.defeatPercentage * 100).toStringAsFixed(0), Colors.red, textColor, ch, cw),
|
_buildDynStatRow("TOT", data.total.toString(), "100", Colors.white, ch),
|
||||||
_buildDynDivider(cw, textColor),
|
],
|
||||||
_buildDynStatRow("TOT", data.total.toString(), "100", textColor, textColor, ch, cw),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
const Expanded(flex: 1, child: SizedBox()),
|
// 👇 RODAPÉ AJUSTADO
|
||||||
|
SizedBox(height: ch * 0.03),
|
||||||
// --- RODAPÉ: BOTÃO WIN RATE GIGANTE --- (👇 MUDANÇA AQUI: Alinhado à esquerda)
|
|
||||||
Container(
|
Container(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
padding: EdgeInsets.symmetric(vertical: ch * 0.025),
|
padding: EdgeInsets.symmetric(vertical: ch * 0.035),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: textColor.withOpacity(0.05), // 👇 Fundo adaptável
|
color: Colors.white24, // Igual ao fundo do botão detalhes
|
||||||
borderRadius: BorderRadius.circular(12),
|
borderRadius: BorderRadius.circular(ch * 0.03), // Borda arredondada
|
||||||
),
|
),
|
||||||
child: FittedBox(
|
child: Center(
|
||||||
fit: BoxFit.scaleDown,
|
child: FittedBox(
|
||||||
child: Row(
|
fit: BoxFit.scaleDown,
|
||||||
mainAxisAlignment: MainAxisAlignment.start, // 👇 MUDANÇA AQUI: Letras mais para a esquerda!
|
child: Row(
|
||||||
children: [
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
Icon(Icons.stars, color: Colors.green, size: ch * 0.075),
|
children: [
|
||||||
const SizedBox(width: 10),
|
Icon(
|
||||||
Text('WIN RATE: ${(data.victoryPercentage * 100).toStringAsFixed(1)}%',
|
data.victoryPercentage >= 0.5 ? Icons.trending_up : Icons.trending_down,
|
||||||
style: TextStyle(
|
color: Colors.green,
|
||||||
color: AppTheme.backgroundLight,
|
size: ch * 0.09
|
||||||
fontWeight: FontWeight.w900,
|
|
||||||
letterSpacing: 1.0,
|
|
||||||
fontSize: ch * 0.06
|
|
||||||
),
|
),
|
||||||
),
|
SizedBox(width: cw * 0.02),
|
||||||
],
|
Text(
|
||||||
|
'WIN RATE: ${(data.victoryPercentage * 100).toStringAsFixed(1)}%',
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: ch * 0.05,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Colors.white
|
||||||
|
)
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -201,39 +183,34 @@ class _PieChartCardState extends State<PieChartCard> with SingleTickerProviderSt
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
// 👇 PERCENTAGENS SUBIDAS LIGEIRAMENTE (0.10 e 0.045)
|
||||||
// 👇 Ajustei a linha de stats para alinhar melhor agora que os números estão encostados à direita
|
Widget _buildDynStatRow(String label, String number, String percent, Color color, double ch) {
|
||||||
Widget _buildDynStatRow(String label, String number, String percent, Color statColor, Color textColor, double ch, double cw) {
|
|
||||||
return Padding(
|
return Padding(
|
||||||
padding: EdgeInsets.symmetric(vertical: ch * 0.005),
|
padding: EdgeInsets.only(bottom: ch * 0.01),
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
crossAxisAlignment: CrossAxisAlignment.center,
|
||||||
children: [
|
children: [
|
||||||
SizedBox(
|
// Número subiu para 0.10
|
||||||
width: cw * 0.12,
|
Expanded(flex: 2, child: FittedBox(fit: BoxFit.scaleDown, alignment: Alignment.centerLeft, child: Text(number, style: TextStyle(fontSize: ch * 0.10, fontWeight: FontWeight.bold, color: color, height: 1.0)))),
|
||||||
child: Column(
|
SizedBox(width: ch * 0.02),
|
||||||
crossAxisAlignment: CrossAxisAlignment.end,
|
Expanded(
|
||||||
mainAxisSize: MainAxisSize.min,
|
flex: 3,
|
||||||
children: [
|
child: Column(crossAxisAlignment: CrossAxisAlignment.start, mainAxisSize: MainAxisSize.min, children: [
|
||||||
Text(label, style: TextStyle(fontSize: ch * 0.045, color: textColor.withOpacity(0.6), fontWeight: FontWeight.bold)), // 👇 TEXTO ADAPTÁVEL (increased from 0.035)
|
Row(children: [
|
||||||
Text('$percent%', style: TextStyle(fontSize: ch * 0.05, color: statColor, fontWeight: FontWeight.bold)), // (increased from 0.04)
|
Container(width: ch * 0.018, height: ch * 0.018, margin: EdgeInsets.only(right: ch * 0.015), decoration: BoxDecoration(color: color, shape: BoxShape.circle)),
|
||||||
],
|
// Label subiu para 0.045
|
||||||
),
|
Expanded(child: FittedBox(fit: BoxFit.scaleDown, alignment: Alignment.centerLeft, child: Text(label, style: TextStyle(fontSize: ch * 0.033, color: Colors.white.withOpacity(0.8), fontWeight: FontWeight.w600))))
|
||||||
|
]),
|
||||||
|
// Percentagem subiu para 0.05
|
||||||
|
FittedBox(fit: BoxFit.scaleDown, alignment: Alignment.centerLeft, child: Text('$percent%', style: TextStyle(fontSize: ch * 0.04, color: color, fontWeight: FontWeight.bold))),
|
||||||
|
]),
|
||||||
),
|
),
|
||||||
SizedBox(width: cw * 0.03),
|
|
||||||
Text(number, style: TextStyle(fontSize: ch * 0.15, fontWeight: FontWeight.w900, color: statColor, height: 1)), // (increased from 0.125)
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildDynDivider(double cw, Color textColor) {
|
Widget _buildDynDivider(double ch) {
|
||||||
return Container(
|
return Container(height: 0.5, color: Colors.white.withOpacity(0.1), margin: EdgeInsets.symmetric(vertical: ch * 0.01));
|
||||||
width: cw * 0.35,
|
|
||||||
height: 1.5,
|
|
||||||
color: textColor.withOpacity(0.2), // 👇 LINHA ADAPTÁVEL
|
|
||||||
margin: const EdgeInsets.symmetric(vertical: 4)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -19,9 +19,12 @@ class PieChartWidget extends StatelessWidget {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return LayoutBuilder(
|
return LayoutBuilder(
|
||||||
builder: (context, constraints) {
|
builder: (context, constraints) {
|
||||||
|
// 👇 MAGIA ANTI-DESAPARECIMENTO 👇
|
||||||
|
// Vê o espaço real. Se por algum motivo for infinito, assume 100 para não sumir.
|
||||||
final double w = constraints.maxWidth.isInfinite ? 100.0 : constraints.maxWidth;
|
final double w = constraints.maxWidth.isInfinite ? 100.0 : constraints.maxWidth;
|
||||||
final double h = constraints.maxHeight.isInfinite ? 100.0 : constraints.maxHeight;
|
final double h = constraints.maxHeight.isInfinite ? 100.0 : constraints.maxHeight;
|
||||||
|
|
||||||
|
// Pega no menor valor para garantir que o círculo não é cortado
|
||||||
final double size = math.min(w, h);
|
final double size = math.min(w, h);
|
||||||
|
|
||||||
return Center(
|
return Center(
|
||||||
@@ -29,7 +32,7 @@ class PieChartWidget extends StatelessWidget {
|
|||||||
width: size,
|
width: size,
|
||||||
height: size,
|
height: size,
|
||||||
child: CustomPaint(
|
child: CustomPaint(
|
||||||
painter: _DonutChartPainter(
|
painter: _PieChartPainter(
|
||||||
victoryPercentage: victoryPercentage,
|
victoryPercentage: victoryPercentage,
|
||||||
defeatPercentage: defeatPercentage,
|
defeatPercentage: defeatPercentage,
|
||||||
drawPercentage: drawPercentage,
|
drawPercentage: drawPercentage,
|
||||||
@@ -45,27 +48,24 @@ class PieChartWidget extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildCenterLabels(double size) {
|
Widget _buildCenterLabels(double size) {
|
||||||
final bool hasGames = victoryPercentage > 0 || defeatPercentage > 0 || drawPercentage > 0;
|
|
||||||
|
|
||||||
return Column(
|
return Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
// 👇 Casa decimal aplicada aqui!
|
'${(victoryPercentage * 100).toStringAsFixed(1)}%',
|
||||||
hasGames ? '${(victoryPercentage * 100).toStringAsFixed(1)}%' : '---',
|
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: size * (hasGames ? 0.20 : 0.15),
|
fontSize: size * 0.18, // O texto cresce ou encolhe com o círculo
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
color: hasGames ? Colors.white : Colors.white54,
|
color: Colors.white,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
SizedBox(height: size * 0.02),
|
SizedBox(height: size * 0.02),
|
||||||
Text(
|
Text(
|
||||||
hasGames ? 'Vitórias' : 'Sem Jogos',
|
'Vitórias',
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: size * 0.08,
|
fontSize: size * 0.10,
|
||||||
color: hasGames ? Colors.white70 : Colors.white38,
|
color: Colors.white.withOpacity(0.8),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -73,12 +73,12 @@ class PieChartWidget extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class _DonutChartPainter extends CustomPainter {
|
class _PieChartPainter extends CustomPainter {
|
||||||
final double victoryPercentage;
|
final double victoryPercentage;
|
||||||
final double defeatPercentage;
|
final double defeatPercentage;
|
||||||
final double drawPercentage;
|
final double drawPercentage;
|
||||||
|
|
||||||
_DonutChartPainter({
|
_PieChartPainter({
|
||||||
required this.victoryPercentage,
|
required this.victoryPercentage,
|
||||||
required this.defeatPercentage,
|
required this.defeatPercentage,
|
||||||
required this.drawPercentage,
|
required this.drawPercentage,
|
||||||
@@ -87,40 +87,59 @@ class _DonutChartPainter extends CustomPainter {
|
|||||||
@override
|
@override
|
||||||
void paint(Canvas canvas, Size size) {
|
void paint(Canvas canvas, Size size) {
|
||||||
final center = Offset(size.width / 2, size.height / 2);
|
final center = Offset(size.width / 2, size.height / 2);
|
||||||
final radius = (size.width / 2) - (size.width * 0.1);
|
// Margem de 5% para a linha de fora não ser cortada
|
||||||
final strokeWidth = size.width * 0.2;
|
final radius = (size.width / 2) - (size.width * 0.05);
|
||||||
|
|
||||||
if (victoryPercentage == 0 && defeatPercentage == 0 && drawPercentage == 0) {
|
|
||||||
final bgPaint = Paint()
|
|
||||||
..color = Colors.white.withOpacity(0.05)
|
|
||||||
..style = PaintingStyle.stroke
|
|
||||||
..strokeWidth = strokeWidth;
|
|
||||||
canvas.drawCircle(center, radius, bgPaint);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const victoryColor = Colors.green;
|
const victoryColor = Colors.green;
|
||||||
const defeatColor = Colors.red;
|
const defeatColor = Colors.red;
|
||||||
const drawColor = Colors.amber;
|
const drawColor = Colors.yellow;
|
||||||
|
const borderColor = Colors.white30;
|
||||||
|
|
||||||
double startAngle = -math.pi / 2;
|
double startAngle = -math.pi / 2;
|
||||||
|
|
||||||
void drawDonutSector(double percentage, Color color) {
|
if (victoryPercentage > 0) {
|
||||||
if (percentage <= 0) return;
|
final sweepAngle = 2 * math.pi * victoryPercentage;
|
||||||
final sweepAngle = 2 * math.pi * percentage;
|
_drawSector(canvas, center, radius, startAngle, sweepAngle, victoryColor, size.width);
|
||||||
final paint = Paint()
|
|
||||||
..color = color
|
|
||||||
..style = PaintingStyle.stroke
|
|
||||||
..strokeWidth = strokeWidth
|
|
||||||
..strokeCap = StrokeCap.butt;
|
|
||||||
|
|
||||||
canvas.drawArc(Rect.fromCircle(center: center, radius: radius), startAngle, sweepAngle, false, paint);
|
|
||||||
startAngle += sweepAngle;
|
startAngle += sweepAngle;
|
||||||
}
|
}
|
||||||
|
|
||||||
drawDonutSector(victoryPercentage, victoryColor);
|
if (drawPercentage > 0) {
|
||||||
drawDonutSector(drawPercentage, drawColor);
|
final sweepAngle = 2 * math.pi * drawPercentage;
|
||||||
drawDonutSector(defeatPercentage, defeatColor);
|
_drawSector(canvas, center, radius, startAngle, sweepAngle, drawColor, size.width);
|
||||||
|
startAngle += sweepAngle;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defeatPercentage > 0) {
|
||||||
|
final sweepAngle = 2 * math.pi * defeatPercentage;
|
||||||
|
_drawSector(canvas, center, radius, startAngle, sweepAngle, defeatColor, size.width);
|
||||||
|
}
|
||||||
|
|
||||||
|
final borderPaint = Paint()
|
||||||
|
..color = borderColor
|
||||||
|
..style = PaintingStyle.stroke
|
||||||
|
..strokeWidth = size.width * 0.02;
|
||||||
|
|
||||||
|
canvas.drawCircle(center, radius, borderPaint);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _drawSector(Canvas canvas, Offset center, double radius, double startAngle, double sweepAngle, Color color, double totalWidth) {
|
||||||
|
final paint = Paint()
|
||||||
|
..color = color
|
||||||
|
..style = PaintingStyle.fill;
|
||||||
|
|
||||||
|
canvas.drawArc(Rect.fromCircle(center: center, radius: radius), startAngle, sweepAngle, true, paint);
|
||||||
|
|
||||||
|
if (sweepAngle < 2 * math.pi) {
|
||||||
|
final linePaint = Paint()
|
||||||
|
..color = Colors.white.withOpacity(0.5)
|
||||||
|
..style = PaintingStyle.stroke
|
||||||
|
..strokeWidth = totalWidth * 0.015;
|
||||||
|
|
||||||
|
final lineX = center.dx + radius * math.cos(startAngle);
|
||||||
|
final lineY = center.dy + radius * math.sin(startAngle);
|
||||||
|
|
||||||
|
canvas.drawLine(center, Offset(lineX, lineY), linePaint);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
|||||||
@@ -1,47 +1,35 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart'; // Para as orientações
|
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
import 'package:playmaker/classe/theme.dart';
|
|
||||||
import 'pages/login.dart';
|
import 'pages/login.dart';
|
||||||
|
|
||||||
// Variável global para controlar o Tema
|
|
||||||
final ValueNotifier<ThemeMode> themeNotifier = ValueNotifier(ThemeMode.system);
|
|
||||||
|
|
||||||
void main() async {
|
void main() async {
|
||||||
// 1. Inicializa os bindings do Flutter
|
|
||||||
WidgetsFlutterBinding.ensureInitialized();
|
WidgetsFlutterBinding.ensureInitialized();
|
||||||
|
|
||||||
// 2. Inicializa o Supabase
|
|
||||||
await Supabase.initialize(
|
await Supabase.initialize(
|
||||||
url: 'https://sihwjdshexjyvsbettcd.supabase.co',
|
url: 'https://sihwjdshexjyvsbettcd.supabase.co',
|
||||||
anonKey: 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6InNpaHdqZHNoZXhqeXZzYmV0dGNkIiwicm9sZSI6ImFub24iLCJpYXQiOjE3Njg5MTQxMjgsImV4cCI6MjA4NDQ5MDEyOH0.gW3AvTJVNyE1Dqa72OTnhrUIKsndexrY3pKxMIAaAy8',
|
anonKey: 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6InNpaHdqZHNoZXhqeXZzYmV0dGNkIiwicm9sZSI6ImFub24iLCJpYXQiOjE3Njg5MTQxMjgsImV4cCI6MjA4NDQ5MDEyOH0.gW3AvTJVNyE1Dqa72OTnhrUIKsndexrY3pKxMIAaAy8', // Uma string longa
|
||||||
);
|
|
||||||
|
|
||||||
// 3. Deixa a orientação livre (Portrait) para o arranque da App
|
);
|
||||||
SystemChrome.setPreferredOrientations([
|
|
||||||
DeviceOrientation.portraitUp,
|
|
||||||
]);
|
|
||||||
|
|
||||||
runApp(const MyApp());
|
runApp(const MyApp());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class MyApp extends StatelessWidget {
|
class MyApp extends StatelessWidget {
|
||||||
const MyApp({super.key});
|
const MyApp({super.key});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return ValueListenableBuilder<ThemeMode>(
|
return MaterialApp(
|
||||||
valueListenable: themeNotifier,
|
debugShowCheckedModeBanner: false,
|
||||||
builder: (_, ThemeMode currentMode, __) {
|
title: 'PlayMaker',
|
||||||
return MaterialApp(
|
theme: ThemeData(
|
||||||
debugShowCheckedModeBanner: false,
|
colorScheme: ColorScheme.fromSeed(
|
||||||
title: 'PlayMaker',
|
seedColor: const Color(0xFFE74C3C),
|
||||||
theme: AppTheme.lightTheme,
|
),
|
||||||
darkTheme: AppTheme.darkTheme,
|
useMaterial3: true,
|
||||||
themeMode: currentMode,
|
),
|
||||||
home: const LoginPage(),
|
home: const LoginPage(),
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,71 +1,32 @@
|
|||||||
class Game {
|
class Game {
|
||||||
final String id;
|
final String id;
|
||||||
final String userId;
|
|
||||||
final String myTeam;
|
final String myTeam;
|
||||||
final String opponentTeam;
|
final String opponentTeam;
|
||||||
final String myScore;
|
final String myScore;
|
||||||
final String opponentScore;
|
final String opponentScore;
|
||||||
final String season;
|
|
||||||
final String status;
|
final String status;
|
||||||
final DateTime gameDate;
|
final String season;
|
||||||
|
|
||||||
// Novos campos que estão na tua base de dados
|
|
||||||
final int remainingSeconds;
|
|
||||||
final int myTimeouts;
|
|
||||||
final int oppTimeouts;
|
|
||||||
final int currentQuarter;
|
|
||||||
final String topPtsName;
|
|
||||||
final String topAstName;
|
|
||||||
final String topRbsName;
|
|
||||||
final String topDefName;
|
|
||||||
final String mvpName;
|
|
||||||
|
|
||||||
Game({
|
Game({
|
||||||
required this.id,
|
required this.id,
|
||||||
required this.userId,
|
|
||||||
required this.myTeam,
|
required this.myTeam,
|
||||||
required this.opponentTeam,
|
required this.opponentTeam,
|
||||||
required this.myScore,
|
required this.myScore,
|
||||||
required this.opponentScore,
|
required this.opponentScore,
|
||||||
required this.season,
|
|
||||||
required this.status,
|
required this.status,
|
||||||
required this.gameDate,
|
required this.season,
|
||||||
required this.remainingSeconds,
|
|
||||||
required this.myTimeouts,
|
|
||||||
required this.oppTimeouts,
|
|
||||||
required this.currentQuarter,
|
|
||||||
required this.topPtsName,
|
|
||||||
required this.topAstName,
|
|
||||||
required this.topRbsName,
|
|
||||||
required this.topDefName,
|
|
||||||
required this.mvpName,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// 👇 A MÁGICA ACONTECE AQUI: Lemos os dados e protegemos os NULLs
|
factory Game.fromMap(Map<String, dynamic> map) {
|
||||||
factory Game.fromMap(Map<String, dynamic> json) {
|
|
||||||
return Game(
|
return Game(
|
||||||
id: json['id']?.toString() ?? '',
|
// O "?." converte para texto com segurança, e o "?? '...'" diz o que mostrar se for nulo (vazio)
|
||||||
userId: json['user_id']?.toString() ?? '',
|
id: map['id']?.toString() ?? '',
|
||||||
myTeam: json['my_team']?.toString() ?? 'Minha Equipa',
|
myTeam: map['my_team']?.toString() ?? 'Desconhecida',
|
||||||
opponentTeam: json['opponent_team']?.toString() ?? 'Adversário',
|
opponentTeam: map['opponent_team']?.toString() ?? 'Adversário',
|
||||||
myScore: (json['my_score'] ?? 0).toString(), // Protege NULL e converte Int4 para String
|
myScore: map['my_score']?.toString() ?? '0',
|
||||||
opponentScore: (json['opponent_score'] ?? 0).toString(),
|
opponentScore: map['opponent_score']?.toString() ?? '0',
|
||||||
season: json['season']?.toString() ?? '---',
|
status: map['status']?.toString() ?? 'Terminado',
|
||||||
status: json['status']?.toString() ?? 'Decorrer',
|
season: map['season']?.toString() ?? 'Sem Época',
|
||||||
gameDate: json['game_date'] != null ? DateTime.tryParse(json['game_date']) ?? DateTime.now() : DateTime.now(),
|
|
||||||
|
|
||||||
// Proteção para os Inteiros (se for NULL, assume 0)
|
|
||||||
remainingSeconds: json['remaining_seconds'] as int? ?? 600, // 600s = 10 minutos
|
|
||||||
myTimeouts: json['my_timeouts'] as int? ?? 0,
|
|
||||||
oppTimeouts: json['opp_timeouts'] as int? ?? 0,
|
|
||||||
currentQuarter: json['current_quarter'] as int? ?? 1,
|
|
||||||
|
|
||||||
// Proteção para os Nomes (se for NULL, assume '---')
|
|
||||||
topPtsName: json['top_pts_name']?.toString() ?? '---',
|
|
||||||
topAstName: json['top_ast_name']?.toString() ?? '---',
|
|
||||||
topRbsName: json['top_rbs_name']?.toString() ?? '---',
|
|
||||||
topDefName: json['top_def_name']?.toString() ?? '---',
|
|
||||||
mvpName: json['mvp_name']?.toString() ?? '---',
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -3,43 +3,24 @@ class Person {
|
|||||||
final String teamId;
|
final String teamId;
|
||||||
final String name;
|
final String name;
|
||||||
final String type; // 'Jogador' ou 'Treinador'
|
final String type; // 'Jogador' ou 'Treinador'
|
||||||
final String? number; // O número é opcional (Treinadores não têm)
|
final String number;
|
||||||
|
|
||||||
// 👇 A NOVA PROPRIEDADE AQUI!
|
|
||||||
final String? imageUrl;
|
|
||||||
|
|
||||||
Person({
|
Person({
|
||||||
required this.id,
|
required this.id,
|
||||||
required this.teamId,
|
required this.teamId,
|
||||||
required this.name,
|
required this.name,
|
||||||
required this.type,
|
required this.type,
|
||||||
this.number,
|
required this.number,
|
||||||
this.imageUrl, // 👇 ADICIONADO AO CONSTRUTOR
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Lê os dados do Supabase e converte para a classe Person
|
// 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']?.toString() ?? '',
|
id: map['id'] ?? '',
|
||||||
teamId: map['team_id']?.toString() ?? '',
|
teamId: map['team_id'] ?? '',
|
||||||
name: map['name']?.toString() ?? 'Desconhecido',
|
name: map['name'] ?? '',
|
||||||
type: map['type']?.toString() ?? 'Jogador',
|
type: map['type'] ?? 'Jogador',
|
||||||
number: map['number']?.toString(),
|
number: map['number']?.toString() ?? '',
|
||||||
|
|
||||||
// 👇 AGORA ELE JÁ SABE LER O LINK DA IMAGEM DA TUA BASE DE DADOS!
|
|
||||||
imageUrl: map['image_url']?.toString(),
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prepara os dados para enviar para o Supabase (se necessário)
|
|
||||||
Map<String, dynamic> toMap() {
|
|
||||||
return {
|
|
||||||
'id': id,
|
|
||||||
'team_id': teamId,
|
|
||||||
'name': name,
|
|
||||||
'type': type,
|
|
||||||
'number': number,
|
|
||||||
'image_url': imageUrl, // 👇 TAMBÉM GUARDA A IMAGEM
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -4,33 +4,26 @@ class Team {
|
|||||||
final String season;
|
final String season;
|
||||||
final String imageUrl;
|
final String imageUrl;
|
||||||
final bool isFavorite;
|
final bool isFavorite;
|
||||||
final String createdAt;
|
|
||||||
final int playerCount; // 👇 NOVA VARIÁVEL AQUI
|
|
||||||
|
|
||||||
Team({
|
Team({
|
||||||
required this.id,
|
required this.id,
|
||||||
required this.name,
|
required this.name,
|
||||||
required this.season,
|
required this.season,
|
||||||
required this.imageUrl,
|
required this.imageUrl,
|
||||||
required this.isFavorite,
|
this.isFavorite = false
|
||||||
required this.createdAt,
|
|
||||||
this.playerCount = 0, // 👇 VALOR POR DEFEITO
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Mapeia o JSON que vem do Supabase (id costuma ser UUID ou String)
|
||||||
factory Team.fromMap(Map<String, dynamic> map) {
|
factory Team.fromMap(Map<String, dynamic> map) {
|
||||||
return Team(
|
return Team(
|
||||||
id: map['id']?.toString() ?? '',
|
id: map['id']?.toString() ?? '',
|
||||||
name: map['name']?.toString() ?? 'Sem Nome',
|
name: map['name'] ?? '',
|
||||||
season: map['season']?.toString() ?? '',
|
season: map['season'] ?? '',
|
||||||
imageUrl: map['image_url']?.toString() ?? '',
|
imageUrl: map['image_url'] ?? '',
|
||||||
isFavorite: map['is_favorite'] ?? false,
|
isFavorite: map['is_favorite'] ?? false,
|
||||||
createdAt: map['created_at']?.toString() ?? '',
|
|
||||||
// 👇 AGORA ELE LÊ A CONTAGEM DA TUA NOVA VIEW!
|
|
||||||
playerCount: map['player_count'] != null ? int.tryParse(map['player_count'].toString()) ?? 0 : 0,
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Map<String, dynamic> toMap() {
|
Map<String, dynamic> toMap() {
|
||||||
return {
|
return {
|
||||||
'name': name,
|
'name': name,
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,4 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/classe/theme.dart'; // 👇 IMPORT DO TEMA
|
|
||||||
import '../controllers/register_controller.dart';
|
import '../controllers/register_controller.dart';
|
||||||
import '../widgets/register_widgets.dart';
|
import '../widgets/register_widgets.dart';
|
||||||
import '../utils/size_extension.dart'; // 👇 O NOSSO SUPERPODER!
|
import '../utils/size_extension.dart'; // 👇 O NOSSO SUPERPODER!
|
||||||
@@ -23,20 +22,11 @@ class _RegisterPageState extends State<RegisterPage> {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
// 👇 BLINDADO: Adapta-se automaticamente ao Modo Claro/Escuro
|
backgroundColor: Colors.white,
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text("Criar Conta", style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold)),
|
||||||
"Criar Conta",
|
backgroundColor: Colors.white,
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 18 * context.sf,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: Theme.of(context).colorScheme.onSurface, // 👇 Adaptável ao Modo Escuro
|
|
||||||
)
|
|
||||||
),
|
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
iconTheme: IconThemeData(color: Theme.of(context).colorScheme.onSurface),
|
|
||||||
),
|
),
|
||||||
body: Center(
|
body: Center(
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
@@ -50,7 +40,7 @@ class _RegisterPageState extends State<RegisterPage> {
|
|||||||
child: Column(
|
child: Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
children: [
|
children: [
|
||||||
const RegisterHeader(),
|
const RegisterHeader(), // 🔥 Agora sim, usa o Header bonito!
|
||||||
SizedBox(height: 30 * context.sf),
|
SizedBox(height: 30 * context.sf),
|
||||||
|
|
||||||
RegisterFormFields(controller: _controller),
|
RegisterFormFields(controller: _controller),
|
||||||
|
|||||||
@@ -1,174 +1,85 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/pages/PlacarPage.dart';
|
import 'package:playmaker/pages/PlacarPage.dart';
|
||||||
import 'package:playmaker/classe/theme.dart';
|
|
||||||
import 'package:cached_network_image/cached_network_image.dart';
|
|
||||||
import '../controllers/team_controller.dart';
|
|
||||||
import '../controllers/game_controller.dart';
|
import '../controllers/game_controller.dart';
|
||||||
|
import '../controllers/team_controller.dart';
|
||||||
import '../models/game_model.dart';
|
import '../models/game_model.dart';
|
||||||
import '../utils/size_extension.dart';
|
import '../utils/size_extension.dart'; // 👇 NOVO SUPERPODER AQUI TAMBÉM!
|
||||||
import 'pdf_export_service.dart';
|
|
||||||
|
|
||||||
|
// --- CARD DE EXIBIÇÃO DO JOGO ---
|
||||||
class GameResultCard extends StatelessWidget {
|
class GameResultCard extends StatelessWidget {
|
||||||
final String gameId, myTeam, opponentTeam, myScore, opponentScore, status, season;
|
final String gameId, myTeam, opponentTeam, myScore, opponentScore, status, season;
|
||||||
final String? myTeamLogo, opponentTeamLogo;
|
final String? myTeamLogo, opponentTeamLogo;
|
||||||
final double sf;
|
|
||||||
final VoidCallback onDelete;
|
|
||||||
|
|
||||||
const GameResultCard({
|
const GameResultCard({
|
||||||
super.key, required this.gameId, required this.myTeam, required this.opponentTeam,
|
super.key, required this.gameId, required this.myTeam, required this.opponentTeam,
|
||||||
required this.myScore, required this.opponentScore, required this.status, required this.season,
|
required this.myScore, required this.opponentScore, required this.status, required this.season,
|
||||||
this.myTeamLogo, this.opponentTeamLogo, required this.sf, required this.onDelete,
|
this.myTeamLogo, this.opponentTeamLogo,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final bgColor = Theme.of(context).cardTheme.color ?? Theme.of(context).colorScheme.surface;
|
|
||||||
final textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
|
|
||||||
return Container(
|
return Container(
|
||||||
margin: EdgeInsets.only(bottom: 16 * sf),
|
margin: EdgeInsets.only(bottom: 16 * context.sf),
|
||||||
padding: EdgeInsets.all(16 * sf),
|
padding: EdgeInsets.all(16 * context.sf),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(color: Colors.white, borderRadius: BorderRadius.circular(20 * context.sf), boxShadow: [BoxShadow(color: Colors.black12, blurRadius: 10 * context.sf)]),
|
||||||
color: bgColor,
|
child: Row(
|
||||||
borderRadius: BorderRadius.circular(20 * sf),
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
boxShadow: [BoxShadow(color: Colors.black12, blurRadius: 10 * sf)],
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.1)),
|
|
||||||
),
|
|
||||||
child: Stack(
|
|
||||||
children: [
|
children: [
|
||||||
Row(
|
Expanded(child: _buildTeamInfo(context, myTeam, const Color(0xFFE74C3C), myTeamLogo)),
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
_buildScoreCenter(context, gameId),
|
||||||
children: [
|
Expanded(child: _buildTeamInfo(context, opponentTeam, Colors.black87, opponentTeamLogo)),
|
||||||
Expanded(child: _buildTeamInfo(myTeam, AppTheme.primaryRed, myTeamLogo, sf, textColor)),
|
|
||||||
_buildScoreCenter(context, gameId, sf, textColor),
|
|
||||||
Expanded(child: _buildTeamInfo(opponentTeam, Colors.grey.shade600, opponentTeamLogo, sf, textColor)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
|
|
||||||
Positioned(
|
|
||||||
top: -10 * sf,
|
|
||||||
right: -10 * sf,
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
IconButton(
|
|
||||||
icon: Icon(Icons.picture_as_pdf, color: AppTheme.primaryRed.withOpacity(0.8), size: 22 * sf),
|
|
||||||
splashRadius: 20 * sf,
|
|
||||||
tooltip: 'Gerar PDF',
|
|
||||||
onPressed: () async {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(const SnackBar(content: Text('A gerar PDF...'), duration: Duration(seconds: 1)));
|
|
||||||
await PdfExportService.generateAndPrintBoxScore(
|
|
||||||
gameId: gameId,
|
|
||||||
myTeam: myTeam,
|
|
||||||
opponentTeam: opponentTeam,
|
|
||||||
myScore: myScore,
|
|
||||||
opponentScore: opponentScore,
|
|
||||||
season: season,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
IconButton(
|
|
||||||
icon: Icon(Icons.delete_outline, color: Colors.grey.shade400, size: 22 * sf),
|
|
||||||
splashRadius: 20 * sf,
|
|
||||||
tooltip: 'Eliminar Jogo',
|
|
||||||
onPressed: () => _showDeleteConfirmation(context),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _showDeleteConfirmation(BuildContext context) {
|
Widget _buildTeamInfo(BuildContext context, String name, Color color, String? logoUrl) {
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (ctx) => AlertDialog(
|
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * sf)),
|
|
||||||
title: Text('Eliminar Jogo', style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16 * sf, color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
content: Text('Tem a certeza que deseja eliminar este jogo? Esta ação apagará todas as estatísticas associadas e não pode ser desfeita.', style: TextStyle(fontSize: 14 * sf, color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () => Navigator.pop(ctx),
|
|
||||||
child: Text('CANCELAR', style: TextStyle(color: Colors.grey, fontSize: 14 * sf))
|
|
||||||
),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(ctx);
|
|
||||||
onDelete();
|
|
||||||
},
|
|
||||||
child: Text('ELIMINAR', style: TextStyle(color: AppTheme.primaryRed, fontWeight: FontWeight.bold, fontSize: 14 * sf))
|
|
||||||
),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildTeamInfo(String name, Color color, String? logoUrl, double sf, Color textColor) {
|
|
||||||
final double avatarSize = 48 * sf;
|
|
||||||
|
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
ClipOval(
|
CircleAvatar(radius: 24 * context.sf, backgroundColor: color, backgroundImage: (logoUrl != null && logoUrl.isNotEmpty) ? NetworkImage(logoUrl) : null, child: (logoUrl == null || logoUrl.isEmpty) ? Icon(Icons.shield, color: Colors.white, size: 24 * context.sf) : null),
|
||||||
child: Container(
|
SizedBox(height: 6 * context.sf),
|
||||||
width: avatarSize,
|
Text(name, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 13 * context.sf), textAlign: TextAlign.center, overflow: TextOverflow.ellipsis, maxLines: 2),
|
||||||
height: avatarSize,
|
|
||||||
color: color.withOpacity(0.1),
|
|
||||||
child: (logoUrl != null && logoUrl.isNotEmpty)
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: logoUrl,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero,
|
|
||||||
placeholder: (context, url) => Center(child: Icon(Icons.shield, color: color, size: 24 * sf)),
|
|
||||||
errorWidget: (context, url, error) => Center(child: Icon(Icons.shield, color: color, size: 24 * sf)),
|
|
||||||
)
|
|
||||||
: Center(child: Icon(Icons.shield, color: color, size: 24 * sf)),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(height: 6 * sf),
|
|
||||||
Text(name, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 13 * sf, color: textColor), textAlign: TextAlign.center, overflow: TextOverflow.ellipsis, maxLines: 2),
|
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildScoreCenter(BuildContext context, String id, double sf, Color textColor) {
|
Widget _buildScoreCenter(BuildContext context, String id) {
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
Row(
|
Row(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
_scoreBox(myScore, AppTheme.successGreen, sf),
|
_scoreBox(context, myScore, Colors.green),
|
||||||
Text(" : ", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 22 * sf, color: textColor)),
|
Text(" : ", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 22 * context.sf)),
|
||||||
_scoreBox(opponentScore, Colors.grey, sf),
|
_scoreBox(context, opponentScore, Colors.grey),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
SizedBox(height: 10 * sf),
|
SizedBox(height: 10 * context.sf),
|
||||||
TextButton.icon(
|
TextButton.icon(
|
||||||
onPressed: () => Navigator.push(context, MaterialPageRoute(builder: (context) => PlacarPage(gameId: id, myTeam: myTeam, opponentTeam: opponentTeam))),
|
onPressed: () => Navigator.push(context, MaterialPageRoute(builder: (context) => PlacarPage(gameId: id, myTeam: myTeam, opponentTeam: opponentTeam))),
|
||||||
icon: Icon(Icons.play_circle_fill, size: 18 * sf, color: AppTheme.primaryRed),
|
icon: Icon(Icons.play_circle_fill, size: 18 * context.sf, color: const Color(0xFFE74C3C)),
|
||||||
label: Text("RETORNAR", style: TextStyle(fontSize: 11 * sf, color: AppTheme.primaryRed, fontWeight: FontWeight.bold)),
|
label: Text("RETORNAR", style: TextStyle(fontSize: 11 * context.sf, color: const Color(0xFFE74C3C), fontWeight: FontWeight.bold)),
|
||||||
style: TextButton.styleFrom(backgroundColor: AppTheme.primaryRed.withOpacity(0.1), padding: EdgeInsets.symmetric(horizontal: 14 * sf, vertical: 8 * sf), shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * sf)), visualDensity: VisualDensity.compact),
|
style: TextButton.styleFrom(backgroundColor: const Color(0xFFE74C3C).withOpacity(0.1), padding: EdgeInsets.symmetric(horizontal: 14 * context.sf, vertical: 8 * context.sf), shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)), visualDensity: VisualDensity.compact),
|
||||||
),
|
),
|
||||||
SizedBox(height: 6 * sf),
|
SizedBox(height: 6 * context.sf),
|
||||||
Text(status, style: TextStyle(fontSize: 12 * sf, color: Colors.blue, fontWeight: FontWeight.bold)),
|
Text(status, style: TextStyle(fontSize: 12 * context.sf, color: Colors.blue, fontWeight: FontWeight.bold)),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _scoreBox(String pts, Color c, double sf) => Container(
|
Widget _scoreBox(BuildContext context, String pts, Color c) => Container(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 12 * sf, vertical: 6 * sf),
|
padding: EdgeInsets.symmetric(horizontal: 12 * context.sf, vertical: 6 * context.sf),
|
||||||
decoration: BoxDecoration(color: c, borderRadius: BorderRadius.circular(8 * sf)),
|
decoration: BoxDecoration(color: c, borderRadius: BorderRadius.circular(8 * context.sf)),
|
||||||
child: Text(pts, style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 16 * sf)),
|
child: Text(pts, style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 16 * context.sf)),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// --- POPUP DE CRIAÇÃO ---
|
||||||
class CreateGameDialogManual extends StatefulWidget {
|
class CreateGameDialogManual extends StatefulWidget {
|
||||||
final TeamController teamController;
|
final TeamController teamController;
|
||||||
final GameController gameController;
|
final GameController gameController;
|
||||||
final double sf;
|
|
||||||
|
|
||||||
const CreateGameDialogManual({super.key, required this.teamController, required this.gameController, required this.sf});
|
const CreateGameDialogManual({super.key, required this.teamController, required this.gameController});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<CreateGameDialogManual> createState() => _CreateGameDialogManualState();
|
State<CreateGameDialogManual> createState() => _CreateGameDialogManualState();
|
||||||
@@ -194,29 +105,24 @@ class _CreateGameDialogManualState extends State<CreateGameDialogManual> {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)),
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * widget.sf)),
|
title: Text('Configurar Partida', style: TextStyle(fontWeight: FontWeight.bold, fontSize: 18 * context.sf)),
|
||||||
title: Text('Configurar Partida', style: TextStyle(fontWeight: FontWeight.bold, fontSize: 18 * widget.sf, color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
content: SingleChildScrollView(
|
content: SingleChildScrollView(
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
TextField(
|
TextField(controller: _seasonController, style: TextStyle(fontSize: 14 * context.sf), decoration: InputDecoration(labelText: 'Temporada', labelStyle: TextStyle(fontSize: 14 * context.sf), border: const OutlineInputBorder(), prefixIcon: Icon(Icons.calendar_today, size: 20 * context.sf))),
|
||||||
controller: _seasonController,
|
SizedBox(height: 15 * context.sf),
|
||||||
style: TextStyle(fontSize: 14 * widget.sf, color: Theme.of(context).colorScheme.onSurface),
|
|
||||||
decoration: InputDecoration(labelText: 'Temporada', labelStyle: TextStyle(fontSize: 14 * widget.sf), border: const OutlineInputBorder(), prefixIcon: Icon(Icons.calendar_today, size: 20 * widget.sf))
|
|
||||||
),
|
|
||||||
SizedBox(height: 15 * widget.sf),
|
|
||||||
_buildSearch(context, "Minha Equipa", _myTeamController),
|
_buildSearch(context, "Minha Equipa", _myTeamController),
|
||||||
Padding(padding: EdgeInsets.symmetric(vertical: 10 * widget.sf), child: Text("VS", style: TextStyle(fontWeight: FontWeight.bold, color: Colors.grey, fontSize: 16 * widget.sf))),
|
Padding(padding: EdgeInsets.symmetric(vertical: 10 * context.sf), child: Text("VS", style: TextStyle(fontWeight: FontWeight.bold, color: Colors.grey, fontSize: 16 * context.sf))),
|
||||||
_buildSearch(context, "Adversário", _opponentController),
|
_buildSearch(context, "Adversário", _opponentController),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(onPressed: () => Navigator.pop(context), child: Text('CANCELAR', style: TextStyle(fontSize: 14 * widget.sf, color: Colors.grey))),
|
TextButton(onPressed: () => Navigator.pop(context), child: Text('CANCELAR', style: TextStyle(fontSize: 14 * context.sf))),
|
||||||
ElevatedButton(
|
ElevatedButton(
|
||||||
style: ElevatedButton.styleFrom(backgroundColor: AppTheme.primaryRed, shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10 * widget.sf)), padding: EdgeInsets.symmetric(horizontal: 16 * widget.sf, vertical: 10 * widget.sf)),
|
style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFFE74C3C), shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10 * context.sf)), padding: EdgeInsets.symmetric(horizontal: 16 * context.sf, vertical: 10 * context.sf)),
|
||||||
onPressed: _isLoading ? null : () async {
|
onPressed: _isLoading ? null : () async {
|
||||||
if (_myTeamController.text.isNotEmpty && _opponentController.text.isNotEmpty) {
|
if (_myTeamController.text.isNotEmpty && _opponentController.text.isNotEmpty) {
|
||||||
setState(() => _isLoading = true);
|
setState(() => _isLoading = true);
|
||||||
@@ -228,7 +134,7 @@ class _CreateGameDialogManualState extends State<CreateGameDialogManual> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
child: _isLoading ? SizedBox(width: 20 * widget.sf, height: 20 * widget.sf, child: const CircularProgressIndicator(color: Colors.white, strokeWidth: 2)) : Text('CRIAR', style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 14 * widget.sf)),
|
child: _isLoading ? SizedBox(width: 20 * context.sf, height: 20 * context.sf, child: const CircularProgressIndicator(color: Colors.white, strokeWidth: 2)) : Text('CRIAR', style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 14 * context.sf)),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
@@ -250,10 +156,9 @@ class _CreateGameDialogManualState extends State<CreateGameDialogManual> {
|
|||||||
return Align(
|
return Align(
|
||||||
alignment: Alignment.topLeft,
|
alignment: Alignment.topLeft,
|
||||||
child: Material(
|
child: Material(
|
||||||
color: Theme.of(context).colorScheme.surface,
|
elevation: 4.0, borderRadius: BorderRadius.circular(8 * context.sf),
|
||||||
elevation: 4.0, borderRadius: BorderRadius.circular(8 * widget.sf),
|
|
||||||
child: ConstrainedBox(
|
child: ConstrainedBox(
|
||||||
constraints: BoxConstraints(maxHeight: 250 * widget.sf, maxWidth: MediaQuery.of(context).size.width * 0.7),
|
constraints: BoxConstraints(maxHeight: 250 * context.sf, maxWidth: MediaQuery.of(context).size.width * 0.7),
|
||||||
child: ListView.builder(
|
child: ListView.builder(
|
||||||
padding: EdgeInsets.zero, shrinkWrap: true, itemCount: options.length,
|
padding: EdgeInsets.zero, shrinkWrap: true, itemCount: options.length,
|
||||||
itemBuilder: (BuildContext context, int index) {
|
itemBuilder: (BuildContext context, int index) {
|
||||||
@@ -261,23 +166,8 @@ class _CreateGameDialogManualState extends State<CreateGameDialogManual> {
|
|||||||
final String name = option['name'].toString();
|
final String name = option['name'].toString();
|
||||||
final String? imageUrl = option['image_url'];
|
final String? imageUrl = option['image_url'];
|
||||||
return ListTile(
|
return ListTile(
|
||||||
leading: ClipOval(
|
leading: CircleAvatar(radius: 20 * context.sf, backgroundColor: Colors.grey.shade200, backgroundImage: (imageUrl != null && imageUrl.isNotEmpty) ? NetworkImage(imageUrl) : null, child: (imageUrl == null || imageUrl.isEmpty) ? Icon(Icons.shield, color: Colors.grey, size: 20 * context.sf) : null),
|
||||||
child: Container(
|
title: Text(name, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14 * context.sf)),
|
||||||
width: 40 * widget.sf,
|
|
||||||
height: 40 * widget.sf,
|
|
||||||
color: Colors.grey.withOpacity(0.2),
|
|
||||||
child: (imageUrl != null && imageUrl.isNotEmpty)
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: imageUrl,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero,
|
|
||||||
placeholder: (context, url) => Icon(Icons.shield, color: Colors.grey, size: 20 * widget.sf),
|
|
||||||
errorWidget: (context, url, error) => Icon(Icons.shield, color: Colors.grey, size: 20 * widget.sf),
|
|
||||||
)
|
|
||||||
: Icon(Icons.shield, color: Colors.grey, size: 20 * widget.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
title: Text(name, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14 * widget.sf, color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
onTap: () { onSelected(option); },
|
onTap: () { onSelected(option); },
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@@ -290,9 +180,8 @@ class _CreateGameDialogManualState extends State<CreateGameDialogManual> {
|
|||||||
if (txtCtrl.text.isEmpty && controller.text.isNotEmpty) txtCtrl.text = controller.text;
|
if (txtCtrl.text.isEmpty && controller.text.isNotEmpty) txtCtrl.text = controller.text;
|
||||||
txtCtrl.addListener(() { controller.text = txtCtrl.text; });
|
txtCtrl.addListener(() { controller.text = txtCtrl.text; });
|
||||||
return TextField(
|
return TextField(
|
||||||
controller: txtCtrl, focusNode: node,
|
controller: txtCtrl, focusNode: node, style: TextStyle(fontSize: 14 * context.sf),
|
||||||
style: TextStyle(fontSize: 14 * widget.sf, color: Theme.of(context).colorScheme.onSurface),
|
decoration: InputDecoration(labelText: label, labelStyle: TextStyle(fontSize: 14 * context.sf), prefixIcon: Icon(Icons.search, size: 20 * context.sf), border: const OutlineInputBorder()),
|
||||||
decoration: InputDecoration(labelText: label, labelStyle: TextStyle(fontSize: 14 * widget.sf), prefixIcon: Icon(Icons.search, size: 20 * widget.sf, color: AppTheme.primaryRed)),
|
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@@ -301,6 +190,7 @@ class _CreateGameDialogManualState extends State<CreateGameDialogManual> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// --- PÁGINA PRINCIPAL DOS JOGOS ---
|
||||||
class GamePage extends StatefulWidget {
|
class GamePage extends StatefulWidget {
|
||||||
const GamePage({super.key});
|
const GamePage({super.key});
|
||||||
|
|
||||||
@@ -319,16 +209,16 @@ class _GamePageState extends State<GamePage> {
|
|||||||
bool isFilterActive = selectedSeason != 'Todas' || selectedTeam != 'Todas';
|
bool isFilterActive = selectedSeason != 'Todas' || selectedTeam != 'Todas';
|
||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
backgroundColor: const Color(0xFFF5F7FA),
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text("Jogos", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 20 * context.sf)),
|
title: Text("Jogos", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 20 * context.sf)),
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
backgroundColor: Colors.white,
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
actions: [
|
actions: [
|
||||||
Padding(
|
Padding(
|
||||||
padding: EdgeInsets.only(right: 8.0 * context.sf),
|
padding: EdgeInsets.only(right: 8.0 * context.sf),
|
||||||
child: IconButton(
|
child: IconButton(
|
||||||
icon: Icon(isFilterActive ? Icons.filter_list_alt : Icons.filter_list, color: isFilterActive ? AppTheme.primaryRed : Theme.of(context).colorScheme.onSurface, size: 26 * context.sf),
|
icon: Icon(isFilterActive ? Icons.filter_list_alt : Icons.filter_list, color: isFilterActive ? const Color(0xFFE74C3C) : Colors.black87, size: 26 * context.sf),
|
||||||
onPressed: () => _showFilterPopup(context),
|
onPressed: () => _showFilterPopup(context),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -342,9 +232,9 @@ class _GamePageState extends State<GamePage> {
|
|||||||
stream: gameController.getFilteredGames(teamFilter: selectedTeam, seasonFilter: selectedSeason),
|
stream: gameController.getFilteredGames(teamFilter: selectedTeam, seasonFilter: selectedSeason),
|
||||||
builder: (context, gameSnapshot) {
|
builder: (context, gameSnapshot) {
|
||||||
if (gameSnapshot.connectionState == ConnectionState.waiting && teamsList.isEmpty) return const Center(child: CircularProgressIndicator());
|
if (gameSnapshot.connectionState == ConnectionState.waiting && teamsList.isEmpty) return const Center(child: CircularProgressIndicator());
|
||||||
if (gameSnapshot.hasError) return Center(child: Text("Erro: ${gameSnapshot.error}", style: TextStyle(fontSize: 14 * context.sf, color: Theme.of(context).colorScheme.onSurface)));
|
if (gameSnapshot.hasError) return Center(child: Text("Erro: ${gameSnapshot.error}", style: TextStyle(fontSize: 14 * context.sf)));
|
||||||
if (!gameSnapshot.hasData || gameSnapshot.data!.isEmpty) {
|
if (!gameSnapshot.hasData || gameSnapshot.data!.isEmpty) {
|
||||||
return Center(child: Column(mainAxisAlignment: MainAxisAlignment.center, children: [Icon(Icons.search_off, size: 48 * context.sf, color: Colors.grey.withOpacity(0.3)), SizedBox(height: 10 * context.sf), Text("Nenhum jogo encontrado.", style: TextStyle(fontSize: 14 * context.sf, color: Colors.grey))]));
|
return Center(child: Column(mainAxisAlignment: MainAxisAlignment.center, children: [Icon(Icons.search_off, size: 48 * context.sf, color: Colors.grey.shade300), SizedBox(height: 10 * context.sf), Text("Nenhum jogo encontrado.", style: TextStyle(fontSize: 14 * context.sf, color: Colors.grey.shade600))]));
|
||||||
}
|
}
|
||||||
return ListView.builder(
|
return ListView.builder(
|
||||||
padding: EdgeInsets.all(16 * context.sf),
|
padding: EdgeInsets.all(16 * context.sf),
|
||||||
@@ -357,31 +247,8 @@ class _GamePageState extends State<GamePage> {
|
|||||||
if (team['name'] == game.opponentTeam) oppLogo = team['image_url'];
|
if (team['name'] == game.opponentTeam) oppLogo = team['image_url'];
|
||||||
}
|
}
|
||||||
return GameResultCard(
|
return GameResultCard(
|
||||||
gameId: game.id,
|
gameId: game.id, myTeam: game.myTeam, opponentTeam: game.opponentTeam, myScore: game.myScore,
|
||||||
myTeam: game.myTeam,
|
opponentScore: game.opponentScore, status: game.status, season: game.season, myTeamLogo: myLogo, opponentTeamLogo: oppLogo,
|
||||||
opponentTeam: game.opponentTeam,
|
|
||||||
myScore: game.myScore,
|
|
||||||
opponentScore: game.opponentScore,
|
|
||||||
status: game.status,
|
|
||||||
season: game.season,
|
|
||||||
myTeamLogo: myLogo,
|
|
||||||
opponentTeamLogo: oppLogo,
|
|
||||||
sf: context.sf,
|
|
||||||
onDelete: () async {
|
|
||||||
bool success = await gameController.deleteGame(game.id);
|
|
||||||
if (context.mounted) {
|
|
||||||
if (success) {
|
|
||||||
setState(() {});
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
const SnackBar(content: Text('Jogo eliminado com sucesso!'), backgroundColor: Colors.green)
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
const SnackBar(content: Text('Erro ao eliminar o jogo.'), backgroundColor: Colors.red)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@@ -390,10 +257,10 @@ class _GamePageState extends State<GamePage> {
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
floatingActionButton: FloatingActionButton(
|
floatingActionButton: FloatingActionButton(
|
||||||
heroTag: 'add_game_btn',
|
heroTag: 'add_game_btn', // 👇 A MÁGICA ESTÁ AQUI TAMBÉM!
|
||||||
backgroundColor: AppTheme.primaryRed,
|
backgroundColor: const Color(0xFFE74C3C),
|
||||||
child: Icon(Icons.add, color: Colors.white, size: 24 * context.sf),
|
child: Icon(Icons.add, color: Colors.white, size: 24 * context.sf),
|
||||||
onPressed: () => showDialog(context: context, builder: (context) => CreateGameDialogManual(teamController: teamController, gameController: gameController, sf: context.sf)),
|
onPressed: () => showDialog(context: context, builder: (context) => CreateGameDialogManual(teamController: teamController, gameController: gameController)),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -407,36 +274,34 @@ class _GamePageState extends State<GamePage> {
|
|||||||
return StatefulBuilder(
|
return StatefulBuilder(
|
||||||
builder: (context, setPopupState) {
|
builder: (context, setPopupState) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)),
|
||||||
title: Row(
|
title: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Text('Filtrar Jogos', style: TextStyle(fontWeight: FontWeight.bold, fontSize: 18 * context.sf, color: Theme.of(context).colorScheme.onSurface)),
|
Text('Filtrar Jogos', style: TextStyle(fontWeight: FontWeight.bold, fontSize: 18 * context.sf)),
|
||||||
IconButton(icon: const Icon(Icons.close, color: Colors.grey), onPressed: () => Navigator.pop(context), padding: EdgeInsets.zero, constraints: const BoxConstraints())
|
IconButton(icon: const Icon(Icons.close, color: Colors.grey), onPressed: () => Navigator.pop(context), padding: EdgeInsets.zero, constraints: const BoxConstraints())
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
content: Column(
|
content: Column(
|
||||||
mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start,
|
mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Text("Temporada", style: TextStyle(fontSize: 12 * context.sf, color: Colors.grey, fontWeight: FontWeight.bold)),
|
Text("Temporada", style: TextStyle(fontSize: 12 * context.sf, color: Colors.grey.shade600, fontWeight: FontWeight.bold)),
|
||||||
SizedBox(height: 6 * context.sf),
|
SizedBox(height: 6 * context.sf),
|
||||||
Container(
|
Container(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 12 * context.sf), decoration: BoxDecoration(color: Theme.of(context).cardTheme.color, borderRadius: BorderRadius.circular(10 * context.sf), border: Border.all(color: Colors.grey.withOpacity(0.2))),
|
padding: EdgeInsets.symmetric(horizontal: 12 * context.sf), decoration: BoxDecoration(color: Colors.grey.shade100, borderRadius: BorderRadius.circular(10 * context.sf)),
|
||||||
child: DropdownButtonHideUnderline(
|
child: DropdownButtonHideUnderline(
|
||||||
child: DropdownButton<String>(
|
child: DropdownButton<String>(
|
||||||
dropdownColor: Theme.of(context).colorScheme.surface,
|
isExpanded: true, value: tempSeason, style: TextStyle(fontSize: 14 * context.sf, color: Colors.black87, fontWeight: FontWeight.bold),
|
||||||
isExpanded: true, value: tempSeason, style: TextStyle(fontSize: 14 * context.sf, color: Theme.of(context).colorScheme.onSurface, fontWeight: FontWeight.bold),
|
|
||||||
items: ['Todas', '2024/25', '2025/26'].map((String value) => DropdownMenuItem<String>(value: value, child: Text(value))).toList(),
|
items: ['Todas', '2024/25', '2025/26'].map((String value) => DropdownMenuItem<String>(value: value, child: Text(value))).toList(),
|
||||||
onChanged: (newValue) => setPopupState(() => tempSeason = newValue!),
|
onChanged: (newValue) => setPopupState(() => tempSeason = newValue!),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
SizedBox(height: 20 * context.sf),
|
SizedBox(height: 20 * context.sf),
|
||||||
Text("Equipa", style: TextStyle(fontSize: 12 * context.sf, color: Colors.grey, fontWeight: FontWeight.bold)),
|
Text("Equipa", style: TextStyle(fontSize: 12 * context.sf, color: Colors.grey.shade600, fontWeight: FontWeight.bold)),
|
||||||
SizedBox(height: 6 * context.sf),
|
SizedBox(height: 6 * context.sf),
|
||||||
Container(
|
Container(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 12 * context.sf), decoration: BoxDecoration(color: Theme.of(context).cardTheme.color, borderRadius: BorderRadius.circular(10 * context.sf), border: Border.all(color: Colors.grey.withOpacity(0.2))),
|
padding: EdgeInsets.symmetric(horizontal: 12 * context.sf), decoration: BoxDecoration(color: Colors.grey.shade100, borderRadius: BorderRadius.circular(10 * context.sf)),
|
||||||
child: StreamBuilder<List<Map<String, dynamic>>>(
|
child: StreamBuilder<List<Map<String, dynamic>>>(
|
||||||
stream: teamController.teamsStream,
|
stream: teamController.teamsStream,
|
||||||
builder: (context, snapshot) {
|
builder: (context, snapshot) {
|
||||||
@@ -445,8 +310,7 @@ class _GamePageState extends State<GamePage> {
|
|||||||
if (!teamNames.contains(tempTeam)) tempTeam = 'Todas';
|
if (!teamNames.contains(tempTeam)) tempTeam = 'Todas';
|
||||||
return DropdownButtonHideUnderline(
|
return DropdownButtonHideUnderline(
|
||||||
child: DropdownButton<String>(
|
child: DropdownButton<String>(
|
||||||
dropdownColor: Theme.of(context).colorScheme.surface,
|
isExpanded: true, value: tempTeam, style: TextStyle(fontSize: 14 * context.sf, color: Colors.black87, fontWeight: FontWeight.bold),
|
||||||
isExpanded: true, value: tempTeam, style: TextStyle(fontSize: 14 * context.sf, color: Theme.of(context).colorScheme.onSurface, fontWeight: FontWeight.bold),
|
|
||||||
items: teamNames.map((String value) => DropdownMenuItem<String>(value: value, child: Text(value, overflow: TextOverflow.ellipsis))).toList(),
|
items: teamNames.map((String value) => DropdownMenuItem<String>(value: value, child: Text(value, overflow: TextOverflow.ellipsis))).toList(),
|
||||||
onChanged: (newValue) => setPopupState(() => tempTeam = newValue!),
|
onChanged: (newValue) => setPopupState(() => tempTeam = newValue!),
|
||||||
),
|
),
|
||||||
@@ -458,7 +322,7 @@ class _GamePageState extends State<GamePage> {
|
|||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(onPressed: () { setState(() { selectedSeason = 'Todas'; selectedTeam = 'Todas'; }); Navigator.pop(context); }, child: Text('LIMPAR', style: TextStyle(fontSize: 12 * context.sf, color: Colors.grey))),
|
TextButton(onPressed: () { setState(() { selectedSeason = 'Todas'; selectedTeam = 'Todas'; }); Navigator.pop(context); }, child: Text('LIMPAR', style: TextStyle(fontSize: 12 * context.sf, color: Colors.grey))),
|
||||||
ElevatedButton(style: ElevatedButton.styleFrom(backgroundColor: AppTheme.primaryRed, shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10 * context.sf))), onPressed: () { setState(() { selectedSeason = tempSeason; selectedTeam = tempTeam; }); Navigator.pop(context); }, child: Text('APLICAR', style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 13 * context.sf))),
|
ElevatedButton(style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFFE74C3C), shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10 * context.sf))), onPressed: () { setState(() { selectedSeason = tempSeason; selectedTeam = tempTeam; }); Navigator.pop(context); }, child: Text('APLICAR', style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 13 * context.sf))),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,15 +1,13 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/classe/theme.dart';
|
import 'package:playmaker/classe/home.config.dart';
|
||||||
import 'package:playmaker/grafico%20de%20pizza/grafico.dart';
|
import 'package:playmaker/grafico%20de%20pizza/grafico.dart';
|
||||||
import 'package:playmaker/pages/gamePage.dart';
|
import 'package:playmaker/pages/gamePage.dart';
|
||||||
import 'package:playmaker/pages/teamPage.dart';
|
import 'package:playmaker/pages/teamPage.dart';
|
||||||
import 'package:playmaker/controllers/team_controller.dart';
|
import 'package:playmaker/controllers/team_controller.dart';
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
import 'package:playmaker/pages/status_page.dart';
|
import 'package:playmaker/pages/status_page.dart';
|
||||||
import 'package:cached_network_image/cached_network_image.dart';
|
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
|
||||||
import '../utils/size_extension.dart';
|
import '../utils/size_extension.dart';
|
||||||
import 'settings_screen.dart';
|
import 'package:playmaker/grafico%20de%20pizza/controllers/contollers_grafico.dart';
|
||||||
|
|
||||||
class HomeScreen extends StatefulWidget {
|
class HomeScreen extends StatefulWidget {
|
||||||
const HomeScreen({super.key});
|
const HomeScreen({super.key});
|
||||||
@@ -30,109 +28,26 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
|
|
||||||
final _supabase = Supabase.instance.client;
|
final _supabase = Supabase.instance.client;
|
||||||
|
|
||||||
String? _avatarUrl;
|
|
||||||
bool _isMemoryLoaded = false; // A variável mágica que impede o "piscar" inicial
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_loadUserAvatar();
|
|
||||||
}
|
|
||||||
|
|
||||||
// FUNÇÃO OTIMIZADA: Carrega da memória instantaneamente e atualiza em background
|
|
||||||
Future<void> _loadUserAvatar() async {
|
|
||||||
// 1. LÊ DA MEMÓRIA RÁPIDA PRIMEIRO
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final savedUrl = prefs.getString('meu_avatar_guardado');
|
|
||||||
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
if (savedUrl != null) _avatarUrl = savedUrl;
|
|
||||||
_isMemoryLoaded = true; // Avisa o ecrã que a memória já respondeu!
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. VAI AO SUPABASE VERIFICAR SE TROCASTE DE FOTO
|
|
||||||
final userId = _supabase.auth.currentUser?.id;
|
|
||||||
if (userId == null) return;
|
|
||||||
|
|
||||||
try {
|
|
||||||
final data = await _supabase
|
|
||||||
.from('profiles')
|
|
||||||
.select('avatar_url')
|
|
||||||
.eq('id', userId)
|
|
||||||
.maybeSingle();
|
|
||||||
|
|
||||||
if (mounted && data != null && data['avatar_url'] != null) {
|
|
||||||
final urlDoSupabase = data['avatar_url'];
|
|
||||||
|
|
||||||
// Se a foto na base de dados for nova, ele guarda e atualiza!
|
|
||||||
if (urlDoSupabase != savedUrl) {
|
|
||||||
await prefs.setString('meu_avatar_guardado', urlDoSupabase);
|
|
||||||
setState(() {
|
|
||||||
_avatarUrl = urlDoSupabase;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
debugPrint("Erro ao carregar avatar na Home: $e");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
// Já não precisamos calcular o sf aqui!
|
||||||
|
|
||||||
final List<Widget> pages = [
|
final List<Widget> pages = [
|
||||||
_buildHomeContent(context),
|
_buildHomeContent(context), // Passamos só o context
|
||||||
const GamePage(),
|
const GamePage(),
|
||||||
const TeamsPage(),
|
const TeamsPage(),
|
||||||
const StatusPage(),
|
const StatusPage(),
|
||||||
];
|
];
|
||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
backgroundColor: Colors.white,
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text('PlayMaker', style: TextStyle(fontSize: 20 * context.sf, fontWeight: FontWeight.bold)),
|
title: Text('PlayMaker', style: TextStyle(fontSize: 20 * context.sf)),
|
||||||
backgroundColor: AppTheme.primaryRed,
|
backgroundColor: HomeConfig.primaryColor,
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
elevation: 0,
|
leading: IconButton(
|
||||||
|
icon: Icon(Icons.person, size: 24 * context.sf),
|
||||||
leading: Padding(
|
onPressed: () {},
|
||||||
padding: EdgeInsets.all(10.0 * context.sf),
|
|
||||||
child: InkWell(
|
|
||||||
borderRadius: BorderRadius.circular(100),
|
|
||||||
onTap: () async {
|
|
||||||
await Navigator.push(
|
|
||||||
context,
|
|
||||||
MaterialPageRoute(builder: (context) => const SettingsScreen()),
|
|
||||||
);
|
|
||||||
_loadUserAvatar();
|
|
||||||
},
|
|
||||||
// SÓ MOSTRA A IMAGEM OU O BONECO DEPOIS DE LER A MEMÓRIA
|
|
||||||
child: !_isMemoryLoaded
|
|
||||||
// Nos primeiros 0.05 segs, mostra só o círculo de fundo (sem boneco)
|
|
||||||
? CircleAvatar(backgroundColor: Colors.white.withOpacity(0.2))
|
|
||||||
|
|
||||||
// Depois da memória responder:
|
|
||||||
: _avatarUrl != null && _avatarUrl!.isNotEmpty
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: _avatarUrl!,
|
|
||||||
fadeInDuration: Duration.zero, // Corta o atraso visual!
|
|
||||||
imageBuilder: (context, imageProvider) => CircleAvatar(
|
|
||||||
backgroundColor: Colors.white.withOpacity(0.2),
|
|
||||||
backgroundImage: imageProvider,
|
|
||||||
),
|
|
||||||
placeholder: (context, url) => CircleAvatar(backgroundColor: Colors.white.withOpacity(0.2)),
|
|
||||||
errorWidget: (context, url, error) => CircleAvatar(
|
|
||||||
backgroundColor: Colors.white.withOpacity(0.2),
|
|
||||||
child: Icon(Icons.person, color: Colors.white, size: 20 * context.sf),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
// Se não tiver foto nenhuma, aí sim mostra o boneco
|
|
||||||
: CircleAvatar(
|
|
||||||
backgroundColor: Colors.white.withOpacity(0.2),
|
|
||||||
child: Icon(Icons.person, color: Colors.white, size: 20 * context.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
@@ -147,6 +62,7 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||||
surfaceTintColor: Theme.of(context).colorScheme.surfaceTint,
|
surfaceTintColor: Theme.of(context).colorScheme.surfaceTint,
|
||||||
elevation: 1,
|
elevation: 1,
|
||||||
|
// O math.min não é necessário se já tens o sf. Mas podes usar context.sf
|
||||||
height: 70 * (context.sf < 1.2 ? context.sf : 1.2),
|
height: 70 * (context.sf < 1.2 ? context.sf : 1.2),
|
||||||
destinations: const [
|
destinations: const [
|
||||||
NavigationDestination(icon: Icon(Icons.home_outlined), selectedIcon: Icon(Icons.home_filled), label: 'Home'),
|
NavigationDestination(icon: Icon(Icons.home_outlined), selectedIcon: Icon(Icons.home_filled), label: 'Home'),
|
||||||
@@ -161,19 +77,13 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
void _showTeamSelector(BuildContext context) {
|
void _showTeamSelector(BuildContext context) {
|
||||||
showModalBottomSheet(
|
showModalBottomSheet(
|
||||||
context: context,
|
context: context,
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.vertical(top: Radius.circular(20 * context.sf))),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.vertical(top: Radius.circular(20 * context.sf))),
|
||||||
builder: (context) {
|
builder: (context) {
|
||||||
return StreamBuilder<List<Map<String, dynamic>>>(
|
return StreamBuilder<List<Map<String, dynamic>>>(
|
||||||
stream: _teamController.teamsStream,
|
stream: _teamController.teamsStream,
|
||||||
builder: (context, snapshot) {
|
builder: (context, snapshot) {
|
||||||
// Correção: Verifica hasData para evitar piscar tela de loading
|
if (snapshot.connectionState == ConnectionState.waiting) return const SizedBox(height: 200, child: Center(child: CircularProgressIndicator()));
|
||||||
if (!snapshot.hasData && snapshot.connectionState == ConnectionState.waiting) {
|
if (!snapshot.hasData || snapshot.data!.isEmpty) return SizedBox(height: 200 * context.sf, child: const Center(child: Text("Nenhuma equipa criada.")));
|
||||||
return const SizedBox(height: 200, child: Center(child: CircularProgressIndicator()));
|
|
||||||
}
|
|
||||||
if (!snapshot.hasData || snapshot.data!.isEmpty) {
|
|
||||||
return SizedBox(height: 200 * context.sf, child: Center(child: Text("Nenhuma equipa criada.", style: TextStyle(color: Theme.of(context).colorScheme.onSurface))));
|
|
||||||
}
|
|
||||||
|
|
||||||
final teams = snapshot.data!;
|
final teams = snapshot.data!;
|
||||||
return ListView.builder(
|
return ListView.builder(
|
||||||
@@ -182,15 +92,14 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
final team = teams[index];
|
final team = teams[index];
|
||||||
return ListTile(
|
return ListTile(
|
||||||
leading: const Icon(Icons.shield, color: AppTheme.primaryRed),
|
title: Text(team['name']),
|
||||||
title: Text(team['name'] ?? 'Sem Nome', style: TextStyle(color: Theme.of(context).colorScheme.onSurface, fontWeight: FontWeight.bold)),
|
|
||||||
onTap: () {
|
onTap: () {
|
||||||
setState(() {
|
setState(() {
|
||||||
_selectedTeamId = team['id'].toString();
|
_selectedTeamId = team['id'];
|
||||||
_selectedTeamName = team['name'] ?? 'Desconhecido';
|
_selectedTeamName = team['name'];
|
||||||
_teamWins = int.tryParse(team['wins']?.toString() ?? '0') ?? 0;
|
_teamWins = team['wins'] != null ? int.tryParse(team['wins'].toString()) ?? 0 : 0;
|
||||||
_teamLosses = int.tryParse(team['losses']?.toString() ?? '0') ?? 0;
|
_teamLosses = team['losses'] != null ? int.tryParse(team['losses'].toString()) ?? 0 : 0;
|
||||||
_teamDraws = int.tryParse(team['draws']?.toString() ?? '0') ?? 0;
|
_teamDraws = team['draws'] != null ? int.tryParse(team['draws'].toString()) ?? 0 : 0;
|
||||||
});
|
});
|
||||||
Navigator.pop(context);
|
Navigator.pop(context);
|
||||||
},
|
},
|
||||||
@@ -206,7 +115,6 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
Widget _buildHomeContent(BuildContext context) {
|
Widget _buildHomeContent(BuildContext context) {
|
||||||
final double wScreen = MediaQuery.of(context).size.width;
|
final double wScreen = MediaQuery.of(context).size.width;
|
||||||
final double cardHeight = wScreen * 0.5;
|
final double cardHeight = wScreen * 0.5;
|
||||||
final textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
|
|
||||||
return StreamBuilder<List<Map<String, dynamic>>>(
|
return StreamBuilder<List<Map<String, dynamic>>>(
|
||||||
stream: _selectedTeamId != null
|
stream: _selectedTeamId != null
|
||||||
@@ -225,20 +133,12 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
onTap: () => _showTeamSelector(context),
|
onTap: () => _showTeamSelector(context),
|
||||||
child: Container(
|
child: Container(
|
||||||
padding: EdgeInsets.all(12 * context.sf),
|
padding: EdgeInsets.all(12 * context.sf),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(color: Colors.grey.shade100, borderRadius: BorderRadius.circular(15 * context.sf), border: Border.all(color: Colors.grey.shade300)),
|
||||||
color: Theme.of(context).cardTheme.color,
|
|
||||||
borderRadius: BorderRadius.circular(15 * context.sf),
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.2))
|
|
||||||
),
|
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Row(children: [
|
Row(children: [Icon(Icons.shield, color: HomeConfig.primaryColor, size: 24 * context.sf), SizedBox(width: 10 * context.sf), Text(_selectedTeamName, style: TextStyle(fontSize: 16 * context.sf, fontWeight: FontWeight.bold))]),
|
||||||
Icon(Icons.shield, color: AppTheme.primaryRed, size: 24 * context.sf),
|
const Icon(Icons.arrow_drop_down),
|
||||||
SizedBox(width: 10 * context.sf),
|
|
||||||
Text(_selectedTeamName, style: TextStyle(fontSize: 16 * context.sf, fontWeight: FontWeight.bold, color: textColor))
|
|
||||||
]),
|
|
||||||
Icon(Icons.arrow_drop_down, color: textColor),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -249,9 +149,9 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
height: cardHeight,
|
height: cardHeight,
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
Expanded(child: _buildStatCard(context: context, title: 'Mais Pontos', playerName: leaders['pts_name'], statValue: leaders['pts_val'].toString(), statLabel: 'TOTAL', color: AppTheme.statPtsBg, isHighlighted: true)),
|
Expanded(child: _buildStatCard(context: context, title: 'Mais Pontos', playerName: leaders['pts_name'], statValue: leaders['pts_val'].toString(), statLabel: 'TOTAL', color: const Color(0xFF1565C0), isHighlighted: true)),
|
||||||
SizedBox(width: 12 * context.sf),
|
SizedBox(width: 12 * context.sf),
|
||||||
Expanded(child: _buildStatCard(context: context, title: 'Assistências', playerName: leaders['ast_name'], statValue: leaders['ast_val'].toString(), statLabel: 'TOTAL', color: AppTheme.statAstBg)),
|
Expanded(child: _buildStatCard(context: context, title: 'Assistências', playerName: leaders['ast_name'], statValue: leaders['ast_val'].toString(), statLabel: 'TOTAL', color: const Color(0xFF2E7D32))),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -261,7 +161,7 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
height: cardHeight,
|
height: cardHeight,
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
Expanded(child: _buildStatCard(context: context, title: 'Rebotes', playerName: leaders['rbs_name'], statValue: leaders['rbs_val'].toString(), statLabel: 'TOTAL', color: AppTheme.statRebBg)),
|
Expanded(child: _buildStatCard(context: context, title: 'Rebotes', playerName: leaders['rbs_name'], statValue: leaders['rbs_val'].toString(), statLabel: 'TOTAL', color: const Color(0xFF6A1B9A))),
|
||||||
SizedBox(width: 12 * context.sf),
|
SizedBox(width: 12 * context.sf),
|
||||||
Expanded(
|
Expanded(
|
||||||
child: PieChartCard(
|
child: PieChartCard(
|
||||||
@@ -270,8 +170,8 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
draws: _teamDraws,
|
draws: _teamDraws,
|
||||||
title: 'DESEMPENHO',
|
title: 'DESEMPENHO',
|
||||||
subtitle: 'Temporada',
|
subtitle: 'Temporada',
|
||||||
backgroundColor: AppTheme.statPieBg,
|
backgroundColor: const Color(0xFFC62828),
|
||||||
sf: context.sf
|
sf: context.sf // Aqui o PieChartCard ainda usa sf, então passamos
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -279,62 +179,22 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
),
|
),
|
||||||
SizedBox(height: 40 * context.sf),
|
SizedBox(height: 40 * context.sf),
|
||||||
|
|
||||||
Text('Histórico de Jogos', style: TextStyle(fontSize: 20 * context.sf, fontWeight: FontWeight.bold, color: textColor)),
|
Text('Histórico de Jogos', style: TextStyle(fontSize: 20 * context.sf, fontWeight: FontWeight.bold, color: Colors.grey[800])),
|
||||||
SizedBox(height: 16 * context.sf),
|
SizedBox(height: 16 * context.sf),
|
||||||
|
|
||||||
_selectedTeamName == "Selecionar Equipa"
|
_selectedTeamName == "Selecionar Equipa"
|
||||||
? Container(
|
? Container(
|
||||||
width: double.infinity,
|
padding: EdgeInsets.all(20 * context.sf),
|
||||||
padding: EdgeInsets.all(24.0 * context.sf),
|
alignment: Alignment.center,
|
||||||
decoration: BoxDecoration(
|
child: Text("Seleciona uma equipa no topo.", style: TextStyle(color: Colors.grey, fontSize: 14 * context.sf)),
|
||||||
color: Theme.of(context).cardTheme.color ?? Colors.white,
|
|
||||||
borderRadius: BorderRadius.circular(16 * context.sf),
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.1)),
|
|
||||||
boxShadow: [BoxShadow(color: Colors.black.withOpacity(0.04), blurRadius: 10, offset: const Offset(0, 4))],
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(18 * context.sf),
|
|
||||||
decoration: BoxDecoration(color: AppTheme.primaryRed.withOpacity(0.08), shape: BoxShape.circle),
|
|
||||||
child: Icon(Icons.shield_outlined, color: AppTheme.primaryRed, size: 42 * context.sf),
|
|
||||||
),
|
|
||||||
SizedBox(height: 20 * context.sf),
|
|
||||||
Text("Nenhuma Equipa Ativa", style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold, color: textColor)),
|
|
||||||
SizedBox(height: 8 * context.sf),
|
|
||||||
Text(
|
|
||||||
"Escolha uma equipa no seletor acima para ver as estatísticas e o histórico.",
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
style: TextStyle(fontSize: 13 * context.sf, color: Colors.grey.shade600, height: 1.4),
|
|
||||||
),
|
|
||||||
SizedBox(height: 24 * context.sf),
|
|
||||||
SizedBox(
|
|
||||||
width: double.infinity,
|
|
||||||
height: 48 * context.sf,
|
|
||||||
child: ElevatedButton.icon(
|
|
||||||
onPressed: () => _showTeamSelector(context),
|
|
||||||
style: ElevatedButton.styleFrom(
|
|
||||||
backgroundColor: AppTheme.primaryRed,
|
|
||||||
foregroundColor: Colors.white,
|
|
||||||
elevation: 0,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10 * context.sf)),
|
|
||||||
),
|
|
||||||
icon: Icon(Icons.touch_app, size: 20 * context.sf),
|
|
||||||
label: Text("Selecionar Agora", style: TextStyle(fontSize: 15 * context.sf, fontWeight: FontWeight.bold)),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
: StreamBuilder<List<Map<String, dynamic>>>(
|
: StreamBuilder<List<Map<String, dynamic>>>(
|
||||||
stream: _supabase.from('games').stream(primaryKey: ['id']).order('game_date', ascending: false),
|
stream: _supabase.from('games').stream(primaryKey: ['id'])
|
||||||
|
.order('game_date', ascending: false),
|
||||||
builder: (context, gameSnapshot) {
|
builder: (context, gameSnapshot) {
|
||||||
if (gameSnapshot.hasError) return Text("Erro: ${gameSnapshot.error}", style: const TextStyle(color: Colors.red));
|
|
||||||
|
|
||||||
// Correção: Verifica hasData em vez de ConnectionState para manter a lista na tela enquanto atualiza em plano de fundo
|
if (gameSnapshot.hasError) return Text("Erro: ${gameSnapshot.error}", style: const TextStyle(color: Colors.red));
|
||||||
if (!gameSnapshot.hasData && gameSnapshot.connectionState == ConnectionState.waiting) {
|
if (gameSnapshot.connectionState == ConnectionState.waiting) return const Center(child: CircularProgressIndicator());
|
||||||
return const Center(child: CircularProgressIndicator());
|
|
||||||
}
|
|
||||||
|
|
||||||
final todosOsJogos = gameSnapshot.data ?? [];
|
final todosOsJogos = gameSnapshot.data ?? [];
|
||||||
final gamesList = todosOsJogos.where((game) {
|
final gamesList = todosOsJogos.where((game) {
|
||||||
@@ -346,11 +206,10 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
|
|
||||||
if (gamesList.isEmpty) {
|
if (gamesList.isEmpty) {
|
||||||
return Container(
|
return Container(
|
||||||
width: double.infinity,
|
|
||||||
padding: EdgeInsets.all(20 * context.sf),
|
padding: EdgeInsets.all(20 * context.sf),
|
||||||
decoration: BoxDecoration(color: Theme.of(context).cardTheme.color, borderRadius: BorderRadius.circular(14)),
|
decoration: BoxDecoration(color: Colors.grey.shade50, borderRadius: BorderRadius.circular(14)),
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
child: const Text("Ainda não há jogos terminados.", style: TextStyle(color: Colors.grey)),
|
child: Text("Ainda não há jogos terminados para $_selectedTeamName.", style: TextStyle(color: Colors.grey)),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -358,8 +217,8 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
children: gamesList.map((game) {
|
children: gamesList.map((game) {
|
||||||
String dbMyTeam = game['my_team']?.toString() ?? '';
|
String dbMyTeam = game['my_team']?.toString() ?? '';
|
||||||
String dbOppTeam = game['opponent_team']?.toString() ?? '';
|
String dbOppTeam = game['opponent_team']?.toString() ?? '';
|
||||||
int dbMyScore = int.tryParse(game['my_score']?.toString() ?? '0') ?? 0;
|
int dbMyScore = int.tryParse(game['my_score'].toString()) ?? 0;
|
||||||
int dbOppScore = int.tryParse(game['opponent_score']?.toString() ?? '0') ?? 0;
|
int dbOppScore = int.tryParse(game['opponent_score'].toString()) ?? 0;
|
||||||
|
|
||||||
String opponent; int myScore; int oppScore;
|
String opponent; int myScore; int oppScore;
|
||||||
|
|
||||||
@@ -377,15 +236,23 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
if (myScore < oppScore) result = 'D';
|
if (myScore < oppScore) result = 'D';
|
||||||
|
|
||||||
return _buildGameHistoryCard(
|
return _buildGameHistoryCard(
|
||||||
context: context, opponent: opponent, result: result,
|
context: context, // Usamos o context para o sf
|
||||||
myScore: myScore, oppScore: oppScore, date: date,
|
opponent: opponent,
|
||||||
topPts: game['top_pts_name'] ?? '---', topAst: game['top_ast_name'] ?? '---',
|
result: result,
|
||||||
topRbs: game['top_rbs_name'] ?? '---', topDef: game['top_def_name'] ?? '---', mvp: game['mvp_name'] ?? '---',
|
myScore: myScore,
|
||||||
|
oppScore: oppScore,
|
||||||
|
date: date,
|
||||||
|
topPts: game['top_pts_name'] ?? '---',
|
||||||
|
topAst: game['top_ast_name'] ?? '---',
|
||||||
|
topRbs: game['top_rbs_name'] ?? '---',
|
||||||
|
topDef: game['top_def_name'] ?? '---',
|
||||||
|
mvp: game['mvp_name'] ?? '---',
|
||||||
);
|
);
|
||||||
}).toList(),
|
}).toList(),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|
||||||
SizedBox(height: 20 * context.sf),
|
SizedBox(height: 20 * context.sf),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@@ -398,45 +265,29 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
Map<String, dynamic> _calculateLeaders(List<Map<String, dynamic>> data) {
|
Map<String, dynamic> _calculateLeaders(List<Map<String, dynamic>> data) {
|
||||||
Map<String, int> ptsMap = {}; Map<String, int> astMap = {}; Map<String, int> rbsMap = {}; Map<String, String> namesMap = {};
|
Map<String, int> ptsMap = {}; Map<String, int> astMap = {}; Map<String, int> rbsMap = {}; Map<String, String> namesMap = {};
|
||||||
for (var row in data) {
|
for (var row in data) {
|
||||||
String pid = row['member_id']?.toString() ?? "unknown";
|
String pid = row['member_id'].toString();
|
||||||
namesMap[pid] = row['player_name']?.toString() ?? "Desconhecido";
|
namesMap[pid] = row['player_name']?.toString() ?? "Desconhecido";
|
||||||
ptsMap[pid] = (ptsMap[pid] ?? 0) + (int.tryParse(row['pts']?.toString() ?? '0') ?? 0);
|
ptsMap[pid] = (ptsMap[pid] ?? 0) + (row['pts'] as int? ?? 0);
|
||||||
astMap[pid] = (astMap[pid] ?? 0) + (int.tryParse(row['ast']?.toString() ?? '0') ?? 0);
|
astMap[pid] = (astMap[pid] ?? 0) + (row['ast'] as int? ?? 0);
|
||||||
rbsMap[pid] = (rbsMap[pid] ?? 0) + (int.tryParse(row['rbs']?.toString() ?? '0') ?? 0);
|
rbsMap[pid] = (rbsMap[pid] ?? 0) + (row['rbs'] as int? ?? 0);
|
||||||
}
|
}
|
||||||
|
if (ptsMap.isEmpty) return {'pts_name': '---', 'pts_val': 0, 'ast_name': '---', 'ast_val': 0, 'rbs_name': '---', 'rbs_val': 0};
|
||||||
if (ptsMap.isEmpty) {
|
String getBest(Map<String, int> map) { var bestId = map.entries.reduce((a, b) => a.value > b.value ? a : b).key; return namesMap[bestId]!; }
|
||||||
return {'pts_name': '---', 'pts_val': 0, 'ast_name': '---', 'ast_val': 0, 'rbs_name': '---', 'rbs_val': 0};
|
int getBestVal(Map<String, int> map) => map.values.reduce((a, b) => a > b ? a : b);
|
||||||
}
|
return {'pts_name': getBest(ptsMap), 'pts_val': getBestVal(ptsMap), 'ast_name': getBest(astMap), 'ast_val': getBestVal(astMap), 'rbs_name': getBest(rbsMap), 'rbs_val': getBestVal(rbsMap)};
|
||||||
|
|
||||||
String getBest(Map<String, int> map) {
|
|
||||||
if (map.isEmpty) return '---';
|
|
||||||
var bestId = map.entries.reduce((a, b) => a.value > b.value ? a : b).key;
|
|
||||||
return namesMap[bestId] ?? '---';
|
|
||||||
}
|
|
||||||
|
|
||||||
int getBestVal(Map<String, int> map) {
|
|
||||||
if (map.isEmpty) return 0;
|
|
||||||
return map.values.reduce((a, b) => a > b ? a : b);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
'pts_name': getBest(ptsMap), 'pts_val': getBestVal(ptsMap),
|
|
||||||
'ast_name': getBest(astMap), 'ast_val': getBestVal(astMap),
|
|
||||||
'rbs_name': getBest(rbsMap), 'rbs_val': getBestVal(rbsMap)
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildStatCard({required BuildContext context, required String title, required String playerName, required String statValue, required String statLabel, required Color color, bool isHighlighted = false}) {
|
Widget _buildStatCard({required BuildContext context, required String title, required String playerName, required String statValue, required String statLabel, required Color color, bool isHighlighted = false}) {
|
||||||
return Card(
|
return Card(
|
||||||
elevation: 4, margin: EdgeInsets.zero,
|
elevation: 4, margin: EdgeInsets.zero,
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14), side: isHighlighted ? const BorderSide(color: AppTheme.warningAmber, width: 2) : BorderSide.none),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14), side: isHighlighted ? const BorderSide(color: Colors.amber, width: 2) : BorderSide.none),
|
||||||
child: Container(
|
child: Container(
|
||||||
decoration: BoxDecoration(borderRadius: BorderRadius.circular(14), gradient: LinearGradient(begin: Alignment.topCenter, end: Alignment.bottomCenter, colors: [color.withOpacity(0.9), color])),
|
decoration: BoxDecoration(borderRadius: BorderRadius.circular(14), gradient: LinearGradient(begin: Alignment.topCenter, end: Alignment.bottomCenter, colors: [color.withOpacity(0.9), color])),
|
||||||
child: LayoutBuilder(
|
child: LayoutBuilder(
|
||||||
builder: (context, constraints) {
|
builder: (context, constraints) {
|
||||||
final double ch = constraints.maxHeight;
|
final double ch = constraints.maxHeight;
|
||||||
final double cw = constraints.maxWidth;
|
final double cw = constraints.maxWidth;
|
||||||
|
|
||||||
return Padding(
|
return Padding(
|
||||||
padding: EdgeInsets.all(cw * 0.06),
|
padding: EdgeInsets.all(cw * 0.06),
|
||||||
child: Column(
|
child: Column(
|
||||||
@@ -476,15 +327,13 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
}) {
|
}) {
|
||||||
bool isWin = result == 'V';
|
bool isWin = result == 'V';
|
||||||
bool isDraw = result == 'E';
|
bool isDraw = result == 'E';
|
||||||
Color statusColor = isWin ? AppTheme.successGreen : (isDraw ? AppTheme.warningAmber : AppTheme.oppTeamRed);
|
Color statusColor = isWin ? Colors.green : (isDraw ? Colors.yellow.shade700 : Colors.red);
|
||||||
final bgColor = Theme.of(context).cardTheme.color;
|
|
||||||
final textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
|
|
||||||
return Container(
|
return Container(
|
||||||
margin: EdgeInsets.only(bottom: 14 * context.sf),
|
margin: EdgeInsets.only(bottom: 14 * context.sf),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: bgColor, borderRadius: BorderRadius.circular(16),
|
color: Colors.white, borderRadius: BorderRadius.circular(16),
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.1)), boxShadow: [BoxShadow(color: Colors.black.withOpacity(0.04), blurRadius: 8, offset: const Offset(0, 4))],
|
border: Border.all(color: Colors.grey.shade200), boxShadow: [BoxShadow(color: Colors.black.withOpacity(0.04), blurRadius: 8, offset: const Offset(0, 4))],
|
||||||
),
|
),
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
@@ -507,16 +356,16 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
Row(
|
Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Expanded(child: Text(_selectedTeamName == "Selecionar Equipa" ? "Minha Equipa" : _selectedTeamName, style: TextStyle(fontSize: 14 * context.sf, fontWeight: FontWeight.bold, color: textColor), maxLines: 1, overflow: TextOverflow.ellipsis)),
|
Expanded(child: Text(_selectedTeamName == "Selecionar Equipa" ? "Minha Equipa" : _selectedTeamName, style: TextStyle(fontSize: 14 * context.sf, fontWeight: FontWeight.bold), maxLines: 1, overflow: TextOverflow.ellipsis)),
|
||||||
Padding(
|
Padding(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 8 * context.sf),
|
padding: EdgeInsets.symmetric(horizontal: 8 * context.sf),
|
||||||
child: Container(
|
child: Container(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 8 * context.sf, vertical: 4 * context.sf),
|
padding: EdgeInsets.symmetric(horizontal: 8 * context.sf, vertical: 4 * context.sf),
|
||||||
decoration: BoxDecoration(color: Theme.of(context).colorScheme.onSurface.withOpacity(0.05), borderRadius: BorderRadius.circular(8)),
|
decoration: BoxDecoration(color: Colors.grey.shade100, borderRadius: BorderRadius.circular(8)),
|
||||||
child: Text('$myScore - $oppScore', style: TextStyle(fontSize: 15 * context.sf, fontWeight: FontWeight.w900, letterSpacing: 1.5, color: textColor)),
|
child: Text('$myScore - $oppScore', style: TextStyle(fontSize: 15 * context.sf, fontWeight: FontWeight.w900, letterSpacing: 1.5, color: Colors.black87)),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Expanded(child: Text(opponent, style: TextStyle(fontSize: 14 * context.sf, fontWeight: FontWeight.bold, color: textColor), textAlign: TextAlign.right, maxLines: 1, overflow: TextOverflow.ellipsis)),
|
Expanded(child: Text(opponent, style: TextStyle(fontSize: 14 * context.sf, fontWeight: FontWeight.bold), textAlign: TextAlign.right, maxLines: 1, overflow: TextOverflow.ellipsis)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -525,10 +374,10 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Divider(height: 1, color: Colors.grey.withOpacity(0.1), thickness: 1.5),
|
Divider(height: 1, color: Colors.grey.shade100, thickness: 1.5),
|
||||||
Container(
|
Container(
|
||||||
width: double.infinity, padding: EdgeInsets.symmetric(horizontal: 16 * context.sf, vertical: 12 * context.sf),
|
width: double.infinity, padding: EdgeInsets.symmetric(horizontal: 16 * context.sf, vertical: 12 * context.sf),
|
||||||
decoration: BoxDecoration(color: Theme.of(context).colorScheme.surface, borderRadius: const BorderRadius.only(bottomLeft: Radius.circular(16), bottomRight: Radius.circular(16))),
|
decoration: BoxDecoration(color: Colors.grey.shade50, borderRadius: const BorderRadius.only(bottomLeft: Radius.circular(16), bottomRight: Radius.circular(16))),
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
Row(
|
Row(
|
||||||
@@ -564,13 +413,13 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
children: [
|
children: [
|
||||||
Icon(icon, size: 14 * context.sf, color: color),
|
Icon(icon, size: 14 * context.sf, color: color),
|
||||||
SizedBox(width: 4 * context.sf),
|
SizedBox(width: 4 * context.sf),
|
||||||
Text('$label: ', style: TextStyle(fontSize: 11 * context.sf, color: Colors.grey, fontWeight: FontWeight.bold)),
|
Text('$label: ', style: TextStyle(fontSize: 11 * context.sf, color: Colors.grey.shade600, fontWeight: FontWeight.bold)),
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Text(
|
child: Text(
|
||||||
value,
|
value,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: 11 * context.sf,
|
fontSize: 11 * context.sf,
|
||||||
color: isMvp ? AppTheme.warningAmber : Theme.of(context).colorScheme.onSurface,
|
color: isMvp ? Colors.amber.shade900 : Colors.black87,
|
||||||
fontWeight: FontWeight.bold
|
fontWeight: FontWeight.bold
|
||||||
),
|
),
|
||||||
maxLines: 1, overflow: TextOverflow.ellipsis
|
maxLines: 1, overflow: TextOverflow.ellipsis
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/controllers/login_controller.dart';
|
import 'package:playmaker/controllers/login_controller.dart';
|
||||||
import '../widgets/login_widgets.dart';
|
import '../widgets/login_widgets.dart';
|
||||||
import 'home.dart';
|
import 'home.dart'; // <--- IMPORTANTE: Importa a tua HomeScreen
|
||||||
import '../utils/size_extension.dart';
|
import '../utils/size_extension.dart'; // 👇 O NOSSO SUPERPODER!
|
||||||
|
|
||||||
class LoginPage extends StatefulWidget {
|
class LoginPage extends StatefulWidget {
|
||||||
const LoginPage({super.key});
|
const LoginPage({super.key});
|
||||||
@@ -23,8 +23,7 @@ class _LoginPageState extends State<LoginPage> {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
// 👇 Adaptável ao Modo Claro/Escuro do Flutter
|
backgroundColor: Colors.white,
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
|
||||||
body: SafeArea(
|
body: SafeArea(
|
||||||
child: ListenableBuilder(
|
child: ListenableBuilder(
|
||||||
listenable: controller,
|
listenable: controller,
|
||||||
@@ -33,6 +32,7 @@ class _LoginPageState extends State<LoginPage> {
|
|||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
child: Container(
|
child: Container(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
|
// Garante que o form não fica gigante num tablet
|
||||||
constraints: BoxConstraints(maxWidth: 450 * context.sf),
|
constraints: BoxConstraints(maxWidth: 450 * context.sf),
|
||||||
padding: EdgeInsets.all(32 * context.sf),
|
padding: EdgeInsets.all(32 * context.sf),
|
||||||
child: Column(
|
child: Column(
|
||||||
|
|||||||
@@ -1,374 +0,0 @@
|
|||||||
import 'dart:typed_data';
|
|
||||||
import 'package:pdf/pdf.dart';
|
|
||||||
import 'package:pdf/widgets.dart' as pw;
|
|
||||||
import 'package:printing/printing.dart';
|
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
|
||||||
|
|
||||||
class PdfExportService {
|
|
||||||
static Future<void> generateAndPrintBoxScore({
|
|
||||||
required String gameId,
|
|
||||||
required String myTeam,
|
|
||||||
required String opponentTeam,
|
|
||||||
required String myScore,
|
|
||||||
required String opponentScore,
|
|
||||||
required String season,
|
|
||||||
}) async {
|
|
||||||
final supabase = Supabase.instance.client;
|
|
||||||
|
|
||||||
final gameData = await supabase.from('games').select().eq('id', gameId).single();
|
|
||||||
|
|
||||||
final teamsData = await supabase.from('teams').select('id, name').inFilter('name', [myTeam, opponentTeam]);
|
|
||||||
String? myTeamId, oppTeamId;
|
|
||||||
for (var t in teamsData) {
|
|
||||||
if (t['name'] == myTeam) myTeamId = t['id'].toString();
|
|
||||||
if (t['name'] == opponentTeam) oppTeamId = t['id'].toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
List<dynamic> myPlayers = myTeamId != null ? await supabase.from('members').select().eq('team_id', myTeamId).eq('type', 'Jogador') : [];
|
|
||||||
List<dynamic> oppPlayers = oppTeamId != null ? await supabase.from('members').select().eq('team_id', oppTeamId).eq('type', 'Jogador') : [];
|
|
||||||
|
|
||||||
final statsData = await supabase.from('player_stats').select().eq('game_id', gameId);
|
|
||||||
Map<String, Map<String, dynamic>> statsMap = {};
|
|
||||||
for (var s in statsData) {
|
|
||||||
statsMap[s['member_id'].toString()] = s;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<List<String>> myTeamTable = _buildTeamTableData(myPlayers, statsMap);
|
|
||||||
List<List<String>> oppTeamTable = _buildTeamTableData(oppPlayers, statsMap);
|
|
||||||
|
|
||||||
final pdf = pw.Document();
|
|
||||||
|
|
||||||
pdf.addPage(
|
|
||||||
pw.Page( // 1. Trocado de MultiPage para Page
|
|
||||||
pageFormat: PdfPageFormat.a4.landscape,
|
|
||||||
margin: const pw.EdgeInsets.all(16), // Margens ligeiramente reduzidas para aproveitar o espaço
|
|
||||||
build: (pw.Context context) {
|
|
||||||
// 2. Envolvemos tudo num FittedBox
|
|
||||||
return pw.FittedBox(
|
|
||||||
fit: pw.BoxFit.scaleDown, // Reduz o tamanho apenas se não couber na página
|
|
||||||
child: pw.Container(
|
|
||||||
// Fixamos a largura do contentor à largura útil da página
|
|
||||||
width: PdfPageFormat.a4.landscape.availableWidth,
|
|
||||||
// 3. Colocamos todos os elementos dentro de uma Column
|
|
||||||
child: pw.Column(
|
|
||||||
crossAxisAlignment: pw.CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
pw.Row(
|
|
||||||
mainAxisAlignment: pw.MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
pw.Text('Relatório Estatístico', style: pw.TextStyle(fontSize: 24, fontWeight: pw.FontWeight.bold)),
|
|
||||||
pw.Column(
|
|
||||||
crossAxisAlignment: pw.CrossAxisAlignment.end,
|
|
||||||
children: [
|
|
||||||
pw.Text('$myTeam vs $opponentTeam', style: pw.TextStyle(fontSize: 16, fontWeight: pw.FontWeight.bold)),
|
|
||||||
pw.Text('Resultado: $myScore - $opponentScore', style: const pw.TextStyle(fontSize: 14)),
|
|
||||||
pw.Text('Época: $season', style: const pw.TextStyle(fontSize: 12)),
|
|
||||||
]
|
|
||||||
)
|
|
||||||
]
|
|
||||||
),
|
|
||||||
pw.SizedBox(height: 15), // Espaçamentos verticais um pouco mais otimizados
|
|
||||||
|
|
||||||
pw.Text('Equipa: $myTeam', style: pw.TextStyle(fontSize: 14, fontWeight: pw.FontWeight.bold, color: const PdfColor.fromInt(0xFFA00000))),
|
|
||||||
pw.SizedBox(height: 4),
|
|
||||||
_buildPdfTable(myTeamTable, const PdfColor.fromInt(0xFFA00000)),
|
|
||||||
|
|
||||||
pw.SizedBox(height: 15),
|
|
||||||
|
|
||||||
pw.Text('Equipa: $opponentTeam', style: pw.TextStyle(fontSize: 14, fontWeight: pw.FontWeight.bold, color: PdfColors.grey700)),
|
|
||||||
pw.SizedBox(height: 4),
|
|
||||||
_buildPdfTable(oppTeamTable, PdfColors.grey700),
|
|
||||||
|
|
||||||
pw.SizedBox(height: 15),
|
|
||||||
|
|
||||||
pw.Row(
|
|
||||||
crossAxisAlignment: pw.CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
_buildSummaryBox('Melhor Marcador', gameData['top_pts_name'] ?? '---'),
|
|
||||||
pw.SizedBox(width: 10),
|
|
||||||
_buildSummaryBox('Melhor Ressaltador', gameData['top_rbs_name'] ?? '---'),
|
|
||||||
pw.SizedBox(width: 10),
|
|
||||||
_buildSummaryBox('Melhor Passador', gameData['top_ast_name'] ?? '---'),
|
|
||||||
pw.SizedBox(width: 10),
|
|
||||||
_buildSummaryBox('MVP', gameData['mvp_name'] ?? '---'),
|
|
||||||
]
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
await Printing.layoutPdf(
|
|
||||||
onLayout: (PdfPageFormat format) async => pdf.save(),
|
|
||||||
name: 'BoxScore_${myTeam}_vs_${opponentTeam}.pdf',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static List<List<String>> _buildTeamTableData(List<dynamic> players, Map<String, Map<String, dynamic>> statsMap) {
|
|
||||||
List<List<String>> tableData = [];
|
|
||||||
int tPts = 0, tFgm = 0, tFga = 0, tFtm = 0, tFta = 0, tFls = 0;
|
|
||||||
int tOrb = 0, tDrb = 0, tTr = 0, tStl = 0, tAst = 0, tTov = 0, tBlk = 0;
|
|
||||||
int tP3m = 0, tP2m = 0, tP3a = 0, tP2a = 0;
|
|
||||||
|
|
||||||
players.sort((a, b) {
|
|
||||||
int numA = int.tryParse(a['number']?.toString() ?? '0') ?? 0;
|
|
||||||
int numB = int.tryParse(b['number']?.toString() ?? '0') ?? 0;
|
|
||||||
return numA.compareTo(numB);
|
|
||||||
});
|
|
||||||
|
|
||||||
for (var p in players) {
|
|
||||||
String id = p['id'].toString();
|
|
||||||
String name = p['name'] ?? 'Desconhecido';
|
|
||||||
String number = p['number']?.toString() ?? '-';
|
|
||||||
|
|
||||||
var stat = statsMap[id] ?? {};
|
|
||||||
|
|
||||||
int pts = stat['pts'] ?? 0;
|
|
||||||
int fgm = stat['fgm'] ?? 0;
|
|
||||||
int fga = stat['fga'] ?? 0;
|
|
||||||
int ftm = stat['ftm'] ?? 0;
|
|
||||||
int fta = stat['fta'] ?? 0;
|
|
||||||
int p2m = stat['p2m'] ?? 0;
|
|
||||||
int p2a = stat['p2a'] ?? 0;
|
|
||||||
int p3m = stat['p3m'] ?? 0;
|
|
||||||
int p3a = stat['p3a'] ?? 0;
|
|
||||||
int fls = stat['fls'] ?? 0;
|
|
||||||
int orb = stat['orb'] ?? 0;
|
|
||||||
int drb = stat['drb'] ?? 0;
|
|
||||||
int tr = orb + drb;
|
|
||||||
int stl = stat['stl'] ?? 0;
|
|
||||||
int ast = stat['ast'] ?? 0;
|
|
||||||
int tov = stat['tov'] ?? 0;
|
|
||||||
int blk = stat['blk'] ?? 0;
|
|
||||||
|
|
||||||
tPts += pts; tFgm += fgm; tFga += fga; tFtm += ftm; tFta += fta;
|
|
||||||
tFls += fls; tOrb += orb; tDrb += drb; tTr += tr; tStl += stl;
|
|
||||||
tAst += ast; tTov += tov; tBlk += blk;
|
|
||||||
tP3m += p3m; tP2m += p2m; tP3a += p3a; tP2a += p2a;
|
|
||||||
|
|
||||||
String p2Pct = p2a > 0 ? ((p2m / p2a) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
String p3Pct = p3a > 0 ? ((p3m / p3a) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
String globalPct = fga > 0 ? ((fgm / fga) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
String llPct = fta > 0 ? ((ftm / fta) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
|
|
||||||
tableData.add([
|
|
||||||
number, name, pts.toString(),
|
|
||||||
p2m.toString(), p2a.toString(), p2Pct,
|
|
||||||
p3m.toString(), p3a.toString(), p3Pct,
|
|
||||||
fgm.toString(), fga.toString(), globalPct,
|
|
||||||
ftm.toString(), fta.toString(), llPct,
|
|
||||||
fls.toString(), orb.toString(), drb.toString(), tr.toString(),
|
|
||||||
stl.toString(), ast.toString(), tov.toString(), blk.toString()
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tableData.isEmpty) {
|
|
||||||
tableData.add([
|
|
||||||
'-', 'Sem jogadores registados', '0',
|
|
||||||
'0', '0', '0%',
|
|
||||||
'0', '0', '0%',
|
|
||||||
'0', '0', '0%',
|
|
||||||
'0', '0', '0%',
|
|
||||||
'0', '0', '0', '0', '0', '0', '0', '0'
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
String tP2Pct = tP2a > 0 ? ((tP2m / tP2a) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
String tP3Pct = tP3a > 0 ? ((tP3m / tP3a) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
String tGlobalPct = tFga > 0 ? ((tFgm / tFga) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
String tLlPct = tFta > 0 ? ((tFtm / tFta) * 100).toStringAsFixed(0) + '%' : '0%';
|
|
||||||
|
|
||||||
tableData.add([
|
|
||||||
'', 'TOTAIS', tPts.toString(),
|
|
||||||
tP2m.toString(), tP2a.toString(), tP2Pct,
|
|
||||||
tP3m.toString(), tP3a.toString(), tP3Pct,
|
|
||||||
tFgm.toString(), tFga.toString(), tGlobalPct,
|
|
||||||
tFtm.toString(), tFta.toString(), tLlPct,
|
|
||||||
tFls.toString(), tOrb.toString(), tDrb.toString(), tTr.toString(),
|
|
||||||
tStl.toString(), tAst.toString(), tTov.toString(), tBlk.toString()
|
|
||||||
]);
|
|
||||||
|
|
||||||
return tableData;
|
|
||||||
}
|
|
||||||
|
|
||||||
static pw.Widget _buildPdfTable(List<List<String>> data, PdfColor headerColor) {
|
|
||||||
final headerStyle = pw.TextStyle(color: PdfColors.white, fontWeight: pw.FontWeight.bold, fontSize: 8);
|
|
||||||
final subHeaderStyle = pw.TextStyle(color: PdfColors.white, fontWeight: pw.FontWeight.bold, fontSize: 7);
|
|
||||||
final cellStyle = const pw.TextStyle(fontSize: 8);
|
|
||||||
|
|
||||||
// Agora usamos apenas 15 colunas principais na tabela.
|
|
||||||
// Os grupos (2P, 3P, etc.) são subdivididos INTERNAMENTE para evitar erros de colSpan.
|
|
||||||
return pw.Table(
|
|
||||||
border: pw.TableBorder.all(color: PdfColors.grey400, width: 0.5),
|
|
||||||
columnWidths: {
|
|
||||||
0: const pw.FlexColumnWidth(1.2), // Nº
|
|
||||||
1: const pw.FlexColumnWidth(5.0), // NOME (Maior para caber nomes como S.Gilgeous-alexander)
|
|
||||||
2: const pw.FlexColumnWidth(1.5), // PT
|
|
||||||
3: const pw.FlexColumnWidth(4.5), // 2 PONTOS (Grupo de 3)
|
|
||||||
4: const pw.FlexColumnWidth(4.5), // 3 PONTOS (Grupo de 3)
|
|
||||||
5: const pw.FlexColumnWidth(4.5), // GLOBAL (Grupo de 3)
|
|
||||||
6: const pw.FlexColumnWidth(4.5), // L. LIVRES (Grupo de 3)
|
|
||||||
7: const pw.FlexColumnWidth(1.5), // FLS
|
|
||||||
8: const pw.FlexColumnWidth(1.5), // RO
|
|
||||||
9: const pw.FlexColumnWidth(1.5), // RD
|
|
||||||
10: const pw.FlexColumnWidth(1.5), // TR
|
|
||||||
11: const pw.FlexColumnWidth(1.5), // BR
|
|
||||||
12: const pw.FlexColumnWidth(1.5), // AS
|
|
||||||
13: const pw.FlexColumnWidth(1.5), // BP
|
|
||||||
14: const pw.FlexColumnWidth(1.5), // BLK
|
|
||||||
},
|
|
||||||
children: [
|
|
||||||
// --- LINHA 1: CABEÇALHOS ---
|
|
||||||
pw.TableRow(
|
|
||||||
decoration: pw.BoxDecoration(color: headerColor),
|
|
||||||
children: [
|
|
||||||
_simpleHeader('Nº', subHeaderStyle),
|
|
||||||
_simpleHeader('NOME', subHeaderStyle, align: pw.Alignment.centerLeft),
|
|
||||||
_simpleHeader('PT', subHeaderStyle),
|
|
||||||
_groupHeader('2 PONTOS', headerStyle, subHeaderStyle),
|
|
||||||
_groupHeader('3 PONTOS', headerStyle, subHeaderStyle),
|
|
||||||
_groupHeader('GLOBAL', headerStyle, subHeaderStyle),
|
|
||||||
_groupHeader('L. LIVRES', headerStyle, subHeaderStyle),
|
|
||||||
_simpleHeader('FLS', subHeaderStyle),
|
|
||||||
_simpleHeader('RO', subHeaderStyle),
|
|
||||||
_simpleHeader('RD', subHeaderStyle),
|
|
||||||
_simpleHeader('TR', subHeaderStyle),
|
|
||||||
_simpleHeader('BR', subHeaderStyle),
|
|
||||||
_simpleHeader('AS', subHeaderStyle),
|
|
||||||
_simpleHeader('BP', subHeaderStyle),
|
|
||||||
_simpleHeader('BLK', subHeaderStyle),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
// --- LINHAS 2+: DADOS ---
|
|
||||||
...data.map((row) {
|
|
||||||
bool isTotais = row[1] == 'TOTAIS';
|
|
||||||
var rowStyle = isTotais ? pw.TextStyle(fontSize: 8, fontWeight: pw.FontWeight.bold) : cellStyle;
|
|
||||||
return pw.TableRow(
|
|
||||||
decoration: pw.BoxDecoration(
|
|
||||||
color: isTotais ? PdfColors.grey200 : PdfColors.white,
|
|
||||||
),
|
|
||||||
children: [
|
|
||||||
_simpleData(row[0], rowStyle),
|
|
||||||
_simpleData(row[1], rowStyle, align: pw.Alignment.centerLeft),
|
|
||||||
_simpleData(row[2], rowStyle),
|
|
||||||
_groupData(row[3], row[4], row[5], rowStyle), // 2P: C, T, %
|
|
||||||
_groupData(row[6], row[7], row[8], rowStyle), // 3P: C, T, %
|
|
||||||
_groupData(row[9], row[10], row[11], rowStyle), // GLOBAL: C, T, %
|
|
||||||
_groupData(row[12], row[13], row[14], rowStyle), // L. LIVRES: C, T, %
|
|
||||||
_simpleData(row[15], rowStyle),
|
|
||||||
_simpleData(row[16], rowStyle),
|
|
||||||
_simpleData(row[17], rowStyle),
|
|
||||||
_simpleData(row[18], rowStyle),
|
|
||||||
_simpleData(row[19], rowStyle),
|
|
||||||
_simpleData(row[20], rowStyle),
|
|
||||||
_simpleData(row[21], rowStyle),
|
|
||||||
_simpleData(row[22], rowStyle),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// ==== WIDGETS AUXILIARES PARA RESOLVER A ESTRUTURA DO PDF ====
|
|
||||||
|
|
||||||
// Cabeçalho simples (Colunas que não se dividem)
|
|
||||||
static pw.Widget _simpleHeader(String text, pw.TextStyle style, {pw.Alignment align = pw.Alignment.center}) {
|
|
||||||
return pw.Container(
|
|
||||||
alignment: align,
|
|
||||||
padding: const pw.EdgeInsets.symmetric(vertical: 2, horizontal: 2),
|
|
||||||
child: pw.Text(text, style: style),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Dados simples
|
|
||||||
static pw.Widget _simpleData(String text, pw.TextStyle style, {pw.Alignment align = pw.Alignment.center}) {
|
|
||||||
return pw.Container(
|
|
||||||
alignment: align,
|
|
||||||
padding: const pw.EdgeInsets.symmetric(vertical: 3, horizontal: 2),
|
|
||||||
child: pw.Text(text, style: style),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cria a divisão do Cabeçalho (O falso ColSpan que une "2 PONTOS" sobre "C | T | %")
|
|
||||||
static pw.Widget _groupHeader(String title, pw.TextStyle hStyle, pw.TextStyle sStyle) {
|
|
||||||
return pw.Column(
|
|
||||||
children: [
|
|
||||||
pw.Container(
|
|
||||||
width: double.infinity,
|
|
||||||
alignment: pw.Alignment.center,
|
|
||||||
padding: const pw.EdgeInsets.symmetric(vertical: 2),
|
|
||||||
decoration: const pw.BoxDecoration(
|
|
||||||
border: pw.Border(bottom: pw.BorderSide(color: PdfColors.white, width: 0.5)),
|
|
||||||
),
|
|
||||||
child: pw.Text(title, style: hStyle),
|
|
||||||
),
|
|
||||||
pw.Row(
|
|
||||||
children: [
|
|
||||||
pw.Expanded(child: pw.Container(alignment: pw.Alignment.center, child: pw.Text('C', style: sStyle))),
|
|
||||||
pw.Container(width: 0.5, height: 10, color: PdfColors.white), // Divisória vertical manual
|
|
||||||
pw.Expanded(child: pw.Container(alignment: pw.Alignment.center, child: pw.Text('T', style: sStyle))),
|
|
||||||
pw.Container(width: 0.5, height: 10, color: PdfColors.white), // Divisória vertical manual
|
|
||||||
pw.Expanded(child: pw.Container(alignment: pw.Alignment.center, child: pw.Text('%', style: sStyle))),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static pw.Widget _groupData(String c, String t, String pct, pw.TextStyle style) {
|
|
||||||
return pw.Row(
|
|
||||||
children: [
|
|
||||||
pw.Expanded(
|
|
||||||
child: pw.Container(
|
|
||||||
alignment: pw.Alignment.center,
|
|
||||||
padding: const pw.EdgeInsets.symmetric(vertical: 3),
|
|
||||||
child: pw.Text(c, style: style),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
pw.Container(width: 0.5, height: 12, color: PdfColors.grey400), // Divisória cinza
|
|
||||||
pw.Expanded(
|
|
||||||
child: pw.Container(
|
|
||||||
alignment: pw.Alignment.center,
|
|
||||||
padding: const pw.EdgeInsets.symmetric(vertical: 3),
|
|
||||||
child: pw.Text(t, style: style),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
pw.Container(width: 0.5, height: 12, color: PdfColors.grey400), // Divisória cinza
|
|
||||||
pw.Expanded(
|
|
||||||
child: pw.Container(
|
|
||||||
alignment: pw.Alignment.center,
|
|
||||||
padding: const pw.EdgeInsets.symmetric(vertical: 3),
|
|
||||||
child: pw.Text(pct, style: style),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
static pw.Widget _buildSummaryBox(String title, String value) {
|
|
||||||
return pw.Container(
|
|
||||||
width: 120,
|
|
||||||
decoration: pw.BoxDecoration(
|
|
||||||
border: pw.TableBorder.all(color: PdfColors.black, width: 1),
|
|
||||||
),
|
|
||||||
child: pw.Column(
|
|
||||||
children: [
|
|
||||||
pw.Container(
|
|
||||||
width: double.infinity,
|
|
||||||
padding: const pw.EdgeInsets.all(4),
|
|
||||||
color: const PdfColor.fromInt(0xFFA00000),
|
|
||||||
child: pw.Text(title, style: pw.TextStyle(color: PdfColors.white, fontSize: 9, fontWeight: pw.FontWeight.bold), textAlign: pw.TextAlign.center),
|
|
||||||
),
|
|
||||||
pw.Container(
|
|
||||||
width: double.infinity,
|
|
||||||
padding: const pw.EdgeInsets.all(6),
|
|
||||||
child: pw.Text(value, style: pw.TextStyle(fontSize: 10, fontWeight: pw.FontWeight.bold), textAlign: pw.TextAlign.center),
|
|
||||||
),
|
|
||||||
]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,385 +0,0 @@
|
|||||||
import 'dart:io';
|
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:playmaker/classe/theme.dart';
|
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
|
||||||
import 'package:image_picker/image_picker.dart';
|
|
||||||
import 'package:cached_network_image/cached_network_image.dart'; // 👇 IMPORTAÇÃO PARA CACHE
|
|
||||||
import 'package:shared_preferences/shared_preferences.dart'; // 👇 IMPORTAÇÃO PARA MEMÓRIA RÁPIDA
|
|
||||||
|
|
||||||
import '../utils/size_extension.dart';
|
|
||||||
import 'login.dart';
|
|
||||||
import '../main.dart';
|
|
||||||
|
|
||||||
class SettingsScreen extends StatefulWidget {
|
|
||||||
const SettingsScreen({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<SettingsScreen> createState() => _SettingsScreenState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _SettingsScreenState extends State<SettingsScreen> {
|
|
||||||
|
|
||||||
File? _localImageFile;
|
|
||||||
String? _uploadedImageUrl;
|
|
||||||
bool _isUploadingImage = false;
|
|
||||||
bool _isMemoryLoaded = false; // 👇 VARIÁVEL MÁGICA CONTRA O PISCAR
|
|
||||||
|
|
||||||
final supabase = Supabase.instance.client;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_loadUserAvatar();
|
|
||||||
}
|
|
||||||
|
|
||||||
// 👇 LÊ A IMAGEM DA MEMÓRIA INSTANTANEAMENTE E CONFIRMA NA BD
|
|
||||||
Future<void> _loadUserAvatar() async {
|
|
||||||
// 1. Lê da memória rápida primeiro!
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
final savedUrl = prefs.getString('meu_avatar_guardado');
|
|
||||||
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
if (savedUrl != null) _uploadedImageUrl = savedUrl;
|
|
||||||
_isMemoryLoaded = true; // Avisa que já leu a memória
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
final userId = supabase.auth.currentUser?.id;
|
|
||||||
if (userId == null) return;
|
|
||||||
|
|
||||||
try {
|
|
||||||
final data = await supabase
|
|
||||||
.from('profiles')
|
|
||||||
.select('avatar_url')
|
|
||||||
.eq('id', userId)
|
|
||||||
.maybeSingle();
|
|
||||||
|
|
||||||
if (mounted && data != null && data['avatar_url'] != null) {
|
|
||||||
final urlDoSupabase = data['avatar_url'];
|
|
||||||
|
|
||||||
// Atualiza a memória se a foto na base de dados for diferente
|
|
||||||
if (urlDoSupabase != savedUrl) {
|
|
||||||
await prefs.setString('meu_avatar_guardado', urlDoSupabase);
|
|
||||||
setState(() {
|
|
||||||
_uploadedImageUrl = urlDoSupabase;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
print("Erro ao carregar avatar: $e");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _handleImageChange() async {
|
|
||||||
final ImagePicker picker = ImagePicker();
|
|
||||||
|
|
||||||
final XFile? pickedFile = await picker.pickImage(source: ImageSource.gallery);
|
|
||||||
if (pickedFile == null || !mounted) return;
|
|
||||||
|
|
||||||
try {
|
|
||||||
setState(() {
|
|
||||||
_localImageFile = File(pickedFile.path);
|
|
||||||
_isUploadingImage = true;
|
|
||||||
});
|
|
||||||
|
|
||||||
final userId = supabase.auth.currentUser?.id;
|
|
||||||
if (userId == null) throw Exception("Utilizador não autenticado.");
|
|
||||||
|
|
||||||
final String storagePath = '$userId/profile_picture.png';
|
|
||||||
|
|
||||||
await supabase.storage.from('avatars').upload(
|
|
||||||
storagePath,
|
|
||||||
_localImageFile!,
|
|
||||||
fileOptions: const FileOptions(cacheControl: '3600', upsert: true)
|
|
||||||
);
|
|
||||||
|
|
||||||
final String publicUrl = supabase.storage.from('avatars').getPublicUrl(storagePath);
|
|
||||||
|
|
||||||
await supabase
|
|
||||||
.from('profiles')
|
|
||||||
.upsert({
|
|
||||||
'id': userId,
|
|
||||||
'avatar_url': publicUrl
|
|
||||||
});
|
|
||||||
|
|
||||||
// 👇 MÁGICA: GUARDA LOGO O NOVO URL NA MEMÓRIA PARA A HOME SABER!
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
await prefs.setString('meu_avatar_guardado', publicUrl);
|
|
||||||
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
_uploadedImageUrl = publicUrl;
|
|
||||||
_isUploadingImage = false;
|
|
||||||
_localImageFile = null;
|
|
||||||
});
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
const SnackBar(content: Text("Foto atualizada!"), backgroundColor: Colors.green)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
_isUploadingImage = false;
|
|
||||||
_localImageFile = null;
|
|
||||||
});
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(content: Text("Erro: $e"), backgroundColor: AppTheme.primaryRed)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final Color primaryRed = AppTheme.primaryRed;
|
|
||||||
final Color bgColor = Theme.of(context).scaffoldBackgroundColor;
|
|
||||||
final Color cardColor = Theme.of(context).cardTheme.color ?? Theme.of(context).colorScheme.surface;
|
|
||||||
final Color textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
final Color textLightColor = textColor.withOpacity(0.6);
|
|
||||||
|
|
||||||
bool isDark = Theme.of(context).brightness == Brightness.dark;
|
|
||||||
|
|
||||||
return Scaffold(
|
|
||||||
backgroundColor: bgColor,
|
|
||||||
appBar: AppBar(
|
|
||||||
backgroundColor: primaryRed,
|
|
||||||
foregroundColor: Colors.white,
|
|
||||||
elevation: 0,
|
|
||||||
centerTitle: true,
|
|
||||||
title: Text(
|
|
||||||
"Perfil e Definições",
|
|
||||||
style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.w600),
|
|
||||||
),
|
|
||||||
leading: IconButton(
|
|
||||||
icon: const Icon(Icons.arrow_back),
|
|
||||||
onPressed: () => Navigator.pop(context),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
body: SingleChildScrollView(
|
|
||||||
padding: EdgeInsets.symmetric(horizontal: 16.0 * context.sf, vertical: 24.0 * context.sf),
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.all(20 * context.sf),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: cardColor,
|
|
||||||
borderRadius: BorderRadius.circular(16 * context.sf),
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.1)),
|
|
||||||
boxShadow: [
|
|
||||||
BoxShadow(color: Colors.black.withOpacity(0.04), blurRadius: 10, offset: const Offset(0, 4)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
_buildTappableProfileAvatar(context, primaryRed),
|
|
||||||
SizedBox(width: 16 * context.sf),
|
|
||||||
Expanded(
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"Treinador",
|
|
||||||
style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold, color: textColor),
|
|
||||||
),
|
|
||||||
SizedBox(height: 4 * context.sf),
|
|
||||||
Text(
|
|
||||||
supabase.auth.currentUser?.email ?? "sem@email.com",
|
|
||||||
style: TextStyle(color: textLightColor, fontSize: 14 * context.sf),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
SizedBox(height: 32 * context.sf),
|
|
||||||
|
|
||||||
Padding(
|
|
||||||
padding: EdgeInsets.only(left: 4 * context.sf, bottom: 12 * context.sf),
|
|
||||||
child: Text(
|
|
||||||
"Definições",
|
|
||||||
style: TextStyle(color: textLightColor, fontSize: 14 * context.sf, fontWeight: FontWeight.bold),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: cardColor,
|
|
||||||
borderRadius: BorderRadius.circular(16 * context.sf),
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.1)),
|
|
||||||
boxShadow: [
|
|
||||||
BoxShadow(color: Colors.black.withOpacity(0.04), blurRadius: 10, offset: const Offset(0, 4)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: ListTile(
|
|
||||||
contentPadding: EdgeInsets.symmetric(horizontal: 20 * context.sf, vertical: 8 * context.sf),
|
|
||||||
leading: Icon(isDark ? Icons.dark_mode : Icons.light_mode, color: primaryRed, size: 28 * context.sf),
|
|
||||||
title: Text(
|
|
||||||
"Modo Escuro",
|
|
||||||
style: TextStyle(fontWeight: FontWeight.bold, color: textColor, fontSize: 16 * context.sf),
|
|
||||||
),
|
|
||||||
subtitle: Text(
|
|
||||||
"Altera as cores da aplicação",
|
|
||||||
style: TextStyle(color: textLightColor, fontSize: 13 * context.sf),
|
|
||||||
),
|
|
||||||
trailing: Switch(
|
|
||||||
value: isDark,
|
|
||||||
activeColor: primaryRed,
|
|
||||||
onChanged: (bool value) {
|
|
||||||
themeNotifier.value = value ? ThemeMode.dark : ThemeMode.light;
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
SizedBox(height: 32 * context.sf),
|
|
||||||
|
|
||||||
Padding(
|
|
||||||
padding: EdgeInsets.only(left: 4 * context.sf, bottom: 12 * context.sf),
|
|
||||||
child: Text(
|
|
||||||
"Conta",
|
|
||||||
style: TextStyle(color: textLightColor, fontSize: 14 * context.sf, fontWeight: FontWeight.bold),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: cardColor,
|
|
||||||
borderRadius: BorderRadius.circular(16 * context.sf),
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.1)),
|
|
||||||
boxShadow: [
|
|
||||||
BoxShadow(color: Colors.black.withOpacity(0.04), blurRadius: 10, offset: const Offset(0, 4)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
child: ListTile(
|
|
||||||
contentPadding: EdgeInsets.symmetric(horizontal: 20 * context.sf, vertical: 4 * context.sf),
|
|
||||||
leading: Icon(Icons.logout_outlined, color: primaryRed, size: 26 * context.sf),
|
|
||||||
title: Text(
|
|
||||||
"Terminar Sessão",
|
|
||||||
style: TextStyle(color: primaryRed, fontWeight: FontWeight.bold, fontSize: 15 * context.sf),
|
|
||||||
),
|
|
||||||
onTap: () => _confirmLogout(context),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
SizedBox(height: 50 * context.sf),
|
|
||||||
|
|
||||||
Center(
|
|
||||||
child: Text(
|
|
||||||
"PlayMaker v1.0.0",
|
|
||||||
style: TextStyle(color: textLightColor.withOpacity(0.7), fontSize: 13 * context.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(height: 20 * context.sf),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 👇 AVATAR OTIMIZADO: SEM LAG, COM CACHE E MEMÓRIA
|
|
||||||
Widget _buildTappableProfileAvatar(BuildContext context, Color primaryRed) {
|
|
||||||
return GestureDetector(
|
|
||||||
onTap: () {
|
|
||||||
_handleImageChange();
|
|
||||||
},
|
|
||||||
child: Stack(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
width: 72 * context.sf,
|
|
||||||
height: 72 * context.sf,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: primaryRed.withOpacity(0.1),
|
|
||||||
shape: BoxShape.circle,
|
|
||||||
),
|
|
||||||
child: ClipOval(
|
|
||||||
child: _isUploadingImage && _localImageFile != null
|
|
||||||
// 1. Mostrar imagem local (galeria) ENQUANTO está a fazer upload
|
|
||||||
? Image.file(_localImageFile!, fit: BoxFit.cover)
|
|
||||||
|
|
||||||
// 2. Antes da memória carregar, fica só o fundo (evita piscar)
|
|
||||||
: !_isMemoryLoaded
|
|
||||||
? const SizedBox()
|
|
||||||
|
|
||||||
// 3. Depois da memória carregar, se houver URL, desenha com Cache!
|
|
||||||
: _uploadedImageUrl != null && _uploadedImageUrl!.isNotEmpty
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: _uploadedImageUrl!,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero, // Fica instantâneo!
|
|
||||||
placeholder: (context, url) => const SizedBox(),
|
|
||||||
errorWidget: (context, url, error) => Icon(Icons.person, color: primaryRed, size: 36 * context.sf),
|
|
||||||
)
|
|
||||||
|
|
||||||
// 4. Se não houver URL, mete o boneco
|
|
||||||
: Icon(Icons.person, color: primaryRed, size: 36 * context.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
// ÍCONE DE LÁPIS
|
|
||||||
Positioned(
|
|
||||||
bottom: 0,
|
|
||||||
right: 0,
|
|
||||||
child: Container(
|
|
||||||
padding: EdgeInsets.all(6 * context.sf),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: Theme.of(context).colorScheme.surface,
|
|
||||||
shape: BoxShape.circle,
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.2)),
|
|
||||||
),
|
|
||||||
child: Icon(Icons.edit_outlined, color: primaryRed, size: 16 * context.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
|
|
||||||
// LOADING OVERLAY (Enquanto faz o upload)
|
|
||||||
if (_isUploadingImage)
|
|
||||||
Positioned.fill(
|
|
||||||
child: Container(
|
|
||||||
decoration: BoxDecoration(color: Colors.black.withOpacity(0.4), shape: BoxShape.circle),
|
|
||||||
child: const Padding(
|
|
||||||
padding: EdgeInsets.all(16.0),
|
|
||||||
child: CircularProgressIndicator(color: Colors.white, strokeWidth: 3),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void _confirmLogout(BuildContext context) {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (ctx) => AlertDialog(
|
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(16 * context.sf)),
|
|
||||||
title: Text("Terminar Sessão", style: TextStyle(color: Theme.of(context).colorScheme.onSurface, fontWeight: FontWeight.bold)),
|
|
||||||
content: Text("Tens a certeza que queres sair da conta?", style: TextStyle(color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
actions: [
|
|
||||||
TextButton(onPressed: () => Navigator.pop(ctx), child: const Text("Cancelar", style: TextStyle(color: Colors.grey))),
|
|
||||||
ElevatedButton(
|
|
||||||
style: ElevatedButton.styleFrom(backgroundColor: AppTheme.primaryRed, shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8))),
|
|
||||||
onPressed: () async {
|
|
||||||
// Limpa a memória do Avatar ao sair para não aparecer na conta de outra pessoa!
|
|
||||||
final prefs = await SharedPreferences.getInstance();
|
|
||||||
await prefs.remove('meu_avatar_guardado');
|
|
||||||
|
|
||||||
await Supabase.instance.client.auth.signOut();
|
|
||||||
if (ctx.mounted) {
|
|
||||||
Navigator.of(ctx).pushAndRemoveUntil(
|
|
||||||
MaterialPageRoute(builder: (context) => const LoginPage()),
|
|
||||||
(Route<dynamic> route) => false,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: const Text("Sair", style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold))
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +1,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
import 'package:playmaker/classe/theme.dart';
|
|
||||||
import 'package:cached_network_image/cached_network_image.dart'; // 👇 A MAGIA DO CACHE
|
|
||||||
import '../controllers/team_controller.dart';
|
import '../controllers/team_controller.dart';
|
||||||
import '../utils/size_extension.dart';
|
import '../utils/size_extension.dart'; // 👇 A MAGIA DO SF!
|
||||||
|
|
||||||
class StatusPage extends StatefulWidget {
|
class StatusPage extends StatefulWidget {
|
||||||
const StatusPage({super.key});
|
const StatusPage({super.key});
|
||||||
@@ -23,9 +21,6 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final bgColor = Theme.of(context).cardTheme.color ?? Colors.white;
|
|
||||||
final textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
|
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
Padding(
|
Padding(
|
||||||
@@ -35,20 +30,20 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
child: Container(
|
child: Container(
|
||||||
padding: EdgeInsets.all(12 * context.sf),
|
padding: EdgeInsets.all(12 * context.sf),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: bgColor,
|
color: Colors.white,
|
||||||
borderRadius: BorderRadius.circular(15 * context.sf),
|
borderRadius: BorderRadius.circular(15 * context.sf),
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.2)),
|
border: Border.all(color: Colors.grey.shade300),
|
||||||
boxShadow: [BoxShadow(color: Colors.black.withOpacity(0.05), blurRadius: 5)]
|
boxShadow: [BoxShadow(color: Colors.black.withOpacity(0.05), blurRadius: 5)]
|
||||||
),
|
),
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Row(children: [
|
Row(children: [
|
||||||
Icon(Icons.shield, color: AppTheme.primaryRed, size: 24 * context.sf),
|
Icon(Icons.shield, color: const Color(0xFFE74C3C), size: 24 * context.sf),
|
||||||
SizedBox(width: 10 * context.sf),
|
SizedBox(width: 10 * context.sf),
|
||||||
Text(_selectedTeamName, style: TextStyle(fontSize: 16 * context.sf, fontWeight: FontWeight.bold, color: textColor))
|
Text(_selectedTeamName, style: TextStyle(fontSize: 16 * context.sf, fontWeight: FontWeight.bold))
|
||||||
]),
|
]),
|
||||||
Icon(Icons.arrow_drop_down, color: textColor),
|
const Icon(Icons.arrow_drop_down),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -68,7 +63,7 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
stream: _supabase.from('members').stream(primaryKey: ['id']).eq('team_id', _selectedTeamId!),
|
stream: _supabase.from('members').stream(primaryKey: ['id']).eq('team_id', _selectedTeamId!),
|
||||||
builder: (context, membersSnapshot) {
|
builder: (context, membersSnapshot) {
|
||||||
if (statsSnapshot.connectionState == ConnectionState.waiting || gamesSnapshot.connectionState == ConnectionState.waiting || membersSnapshot.connectionState == ConnectionState.waiting) {
|
if (statsSnapshot.connectionState == ConnectionState.waiting || gamesSnapshot.connectionState == ConnectionState.waiting || membersSnapshot.connectionState == ConnectionState.waiting) {
|
||||||
return const Center(child: CircularProgressIndicator(color: AppTheme.primaryRed));
|
return const Center(child: CircularProgressIndicator(color: Color(0xFFE74C3C)));
|
||||||
}
|
}
|
||||||
|
|
||||||
final membersData = membersSnapshot.data ?? [];
|
final membersData = membersSnapshot.data ?? [];
|
||||||
@@ -87,7 +82,7 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
return _isAscending ? valA.compareTo(valB) : valB.compareTo(valA);
|
return _isAscending ? valA.compareTo(valB) : valB.compareTo(valA);
|
||||||
});
|
});
|
||||||
|
|
||||||
return _buildStatsGrid(context, playerTotals, teamTotals, bgColor, textColor);
|
return _buildStatsGrid(context, playerTotals, teamTotals);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -99,17 +94,17 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 👇 AGORA GUARDA TAMBÉM O IMAGE_URL DO MEMBRO PARA MOSTRAR NA TABELA
|
|
||||||
List<Map<String, dynamic>> _aggregateStats(List<dynamic> stats, List<dynamic> games, List<dynamic> members) {
|
List<Map<String, dynamic>> _aggregateStats(List<dynamic> stats, List<dynamic> games, List<dynamic> members) {
|
||||||
Map<String, Map<String, dynamic>> aggregated = {};
|
Map<String, Map<String, dynamic>> aggregated = {};
|
||||||
|
|
||||||
for (var member in members) {
|
for (var member in members) {
|
||||||
String name = member['name']?.toString() ?? "Desconhecido";
|
String name = member['name']?.toString() ?? "Desconhecido";
|
||||||
String? imageUrl = member['image_url']?.toString(); // 👈 CAPTURA A IMAGEM AQUI
|
aggregated[name] = {'name': name, 'j': 0, 'pts': 0, 'ast': 0, 'rbs': 0, 'stl': 0, 'blk': 0, 'mvp': 0, 'def': 0};
|
||||||
aggregated[name] = {'name': name, 'image_url': imageUrl, 'j': 0, 'pts': 0, 'ast': 0, 'rbs': 0, 'stl': 0, 'blk': 0, 'mvp': 0, 'def': 0};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var row in stats) {
|
for (var row in stats) {
|
||||||
String name = row['player_name']?.toString() ?? "Desconhecido";
|
String name = row['player_name']?.toString() ?? "Desconhecido";
|
||||||
if (!aggregated.containsKey(name)) aggregated[name] = {'name': name, 'image_url': null, 'j': 0, 'pts': 0, 'ast': 0, 'rbs': 0, 'stl': 0, 'blk': 0, 'mvp': 0, 'def': 0};
|
if (!aggregated.containsKey(name)) aggregated[name] = {'name': name, 'j': 0, 'pts': 0, 'ast': 0, 'rbs': 0, 'stl': 0, 'blk': 0, 'mvp': 0, 'def': 0};
|
||||||
|
|
||||||
aggregated[name]!['j'] += 1;
|
aggregated[name]!['j'] += 1;
|
||||||
aggregated[name]!['pts'] += (row['pts'] ?? 0);
|
aggregated[name]!['pts'] += (row['pts'] ?? 0);
|
||||||
@@ -118,6 +113,7 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
aggregated[name]!['stl'] += (row['stl'] ?? 0);
|
aggregated[name]!['stl'] += (row['stl'] ?? 0);
|
||||||
aggregated[name]!['blk'] += (row['blk'] ?? 0);
|
aggregated[name]!['blk'] += (row['blk'] ?? 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var game in games) {
|
for (var game in games) {
|
||||||
String? mvp = game['mvp_name'];
|
String? mvp = game['mvp_name'];
|
||||||
String? defRaw = game['top_def_name'];
|
String? defRaw = game['top_def_name'];
|
||||||
@@ -135,80 +131,55 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
for (var p in players) {
|
for (var p in players) {
|
||||||
tPts += (p['pts'] as int); tAst += (p['ast'] as int); tRbs += (p['rbs'] as int); tStl += (p['stl'] as int); tBlk += (p['blk'] as int); tMvp += (p['mvp'] as int); tDef += (p['def'] as int);
|
tPts += (p['pts'] as int); tAst += (p['ast'] as int); tRbs += (p['rbs'] as int); tStl += (p['stl'] as int); tBlk += (p['blk'] as int); tMvp += (p['mvp'] as int); tDef += (p['def'] as int);
|
||||||
}
|
}
|
||||||
return {'name': 'TOTAL EQUIPA', 'image_url': null, 'j': teamGames, 'pts': tPts, 'ast': tAst, 'rbs': tRbs, 'stl': tStl, 'blk': tBlk, 'mvp': tMvp, 'def': tDef};
|
return {'name': 'TOTAL EQUIPA', 'j': teamGames, 'pts': tPts, 'ast': tAst, 'rbs': tRbs, 'stl': tStl, 'blk': tBlk, 'mvp': tMvp, 'def': tDef};
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildStatsGrid(BuildContext context, List<Map<String, dynamic>> players, Map<String, dynamic> teamTotals, Color bgColor, Color textColor) {
|
Widget _buildStatsGrid(BuildContext context, List<Map<String, dynamic>> players, Map<String, dynamic> teamTotals) {
|
||||||
return Container(
|
return Container(
|
||||||
color: Colors.transparent,
|
color: Colors.white,
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
scrollDirection: Axis.vertical,
|
scrollDirection: Axis.vertical,
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
scrollDirection: Axis.horizontal,
|
scrollDirection: Axis.horizontal,
|
||||||
child: DataTable(
|
child: DataTable(
|
||||||
columnSpacing: 25 * context.sf,
|
columnSpacing: 25 * context.sf,
|
||||||
headingRowColor: WidgetStateProperty.all(Theme.of(context).colorScheme.surface),
|
headingRowColor: MaterialStateProperty.all(Colors.grey.shade100),
|
||||||
dataRowMaxHeight: 60 * context.sf,
|
dataRowHeight: 60 * context.sf,
|
||||||
dataRowMinHeight: 60 * context.sf,
|
|
||||||
columns: [
|
columns: [
|
||||||
DataColumn(label: Text('JOGADOR', style: TextStyle(color: textColor))),
|
DataColumn(label: const Text('JOGADOR')),
|
||||||
_buildSortableColumn(context, 'J', 'j', textColor),
|
_buildSortableColumn(context, 'J', 'j'),
|
||||||
_buildSortableColumn(context, 'PTS', 'pts', textColor),
|
_buildSortableColumn(context, 'PTS', 'pts'),
|
||||||
_buildSortableColumn(context, 'AST', 'ast', textColor),
|
_buildSortableColumn(context, 'AST', 'ast'),
|
||||||
_buildSortableColumn(context, 'RBS', 'rbs', textColor),
|
_buildSortableColumn(context, 'RBS', 'rbs'),
|
||||||
_buildSortableColumn(context, 'STL', 'stl', textColor),
|
_buildSortableColumn(context, 'STL', 'stl'),
|
||||||
_buildSortableColumn(context, 'BLK', 'blk', textColor),
|
_buildSortableColumn(context, 'BLK', 'blk'),
|
||||||
_buildSortableColumn(context, 'DEF 🛡️', 'def', textColor),
|
_buildSortableColumn(context, 'DEF 🛡️', 'def'),
|
||||||
_buildSortableColumn(context, 'MVP 🏆', 'mvp', textColor),
|
_buildSortableColumn(context, 'MVP 🏆', 'mvp'),
|
||||||
],
|
],
|
||||||
rows: [
|
rows: [
|
||||||
...players.map((player) => DataRow(cells: [
|
...players.map((player) => DataRow(cells: [
|
||||||
DataCell(
|
DataCell(Row(children: [CircleAvatar(radius: 15 * context.sf, backgroundColor: Colors.grey.shade200, child: Icon(Icons.person, size: 18 * context.sf)), SizedBox(width: 10 * context.sf), Text(player['name'], style: TextStyle(fontWeight: FontWeight.bold, fontSize: 13 * context.sf))])),
|
||||||
Row(
|
DataCell(Center(child: Text(player['j'].toString()))),
|
||||||
children: [
|
_buildStatCell(context, player['pts'], isHighlight: true),
|
||||||
// 👇 FOTO DO JOGADOR NA TABELA (COM CACHE!) 👇
|
_buildStatCell(context, player['ast']),
|
||||||
ClipOval(
|
_buildStatCell(context, player['rbs']),
|
||||||
child: Container(
|
_buildStatCell(context, player['stl']),
|
||||||
width: 30 * context.sf,
|
_buildStatCell(context, player['blk']),
|
||||||
height: 30 * context.sf,
|
_buildStatCell(context, player['def'], isBlue: true),
|
||||||
color: Colors.grey.withOpacity(0.2),
|
_buildStatCell(context, player['mvp'], isGold: true),
|
||||||
child: (player['image_url'] != null && player['image_url'].toString().isNotEmpty)
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: player['image_url'],
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero,
|
|
||||||
placeholder: (context, url) => Icon(Icons.person, size: 18 * context.sf, color: Colors.grey),
|
|
||||||
errorWidget: (context, url, error) => Icon(Icons.person, size: 18 * context.sf, color: Colors.grey),
|
|
||||||
)
|
|
||||||
: Icon(Icons.person, size: 18 * context.sf, color: Colors.grey),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(width: 10 * context.sf),
|
|
||||||
Text(player['name'], style: TextStyle(fontWeight: FontWeight.bold, fontSize: 13 * context.sf, color: textColor))
|
|
||||||
]
|
|
||||||
)
|
|
||||||
),
|
|
||||||
DataCell(Center(child: Text(player['j'].toString(), style: TextStyle(color: textColor)))),
|
|
||||||
_buildStatCell(context, player['pts'], textColor, isHighlight: true),
|
|
||||||
_buildStatCell(context, player['ast'], textColor),
|
|
||||||
_buildStatCell(context, player['rbs'], textColor),
|
|
||||||
_buildStatCell(context, player['stl'], textColor),
|
|
||||||
_buildStatCell(context, player['blk'], textColor),
|
|
||||||
_buildStatCell(context, player['def'], textColor, isBlue: true),
|
|
||||||
_buildStatCell(context, player['mvp'], textColor, isGold: true),
|
|
||||||
])),
|
])),
|
||||||
DataRow(
|
DataRow(
|
||||||
color: WidgetStateProperty.all(Theme.of(context).colorScheme.surface.withOpacity(0.5)),
|
color: MaterialStateProperty.all(Colors.grey.shade50),
|
||||||
cells: [
|
cells: [
|
||||||
DataCell(Text('TOTAL EQUIPA', style: TextStyle(fontWeight: FontWeight.w900, color: textColor, fontSize: 12 * context.sf))),
|
DataCell(Text('TOTAL EQUIPA', style: TextStyle(fontWeight: FontWeight.w900, color: Colors.black, fontSize: 12 * context.sf))),
|
||||||
DataCell(Center(child: Text(teamTotals['j'].toString(), style: TextStyle(fontWeight: FontWeight.bold, color: textColor)))),
|
DataCell(Center(child: Text(teamTotals['j'].toString(), style: const TextStyle(fontWeight: FontWeight.bold)))),
|
||||||
_buildStatCell(context, teamTotals['pts'], textColor, isHighlight: true),
|
_buildStatCell(context, teamTotals['pts'], isHighlight: true),
|
||||||
_buildStatCell(context, teamTotals['ast'], textColor),
|
_buildStatCell(context, teamTotals['ast']),
|
||||||
_buildStatCell(context, teamTotals['rbs'], textColor),
|
_buildStatCell(context, teamTotals['rbs']),
|
||||||
_buildStatCell(context, teamTotals['stl'], textColor),
|
_buildStatCell(context, teamTotals['stl']),
|
||||||
_buildStatCell(context, teamTotals['blk'], textColor),
|
_buildStatCell(context, teamTotals['blk']),
|
||||||
_buildStatCell(context, teamTotals['def'], textColor, isBlue: true),
|
_buildStatCell(context, teamTotals['def'], isBlue: true),
|
||||||
_buildStatCell(context, teamTotals['mvp'], textColor, isGold: true),
|
_buildStatCell(context, teamTotals['mvp'], isGold: true),
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
@@ -218,37 +189,37 @@ class _StatusPageState extends State<StatusPage> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
DataColumn _buildSortableColumn(BuildContext context, String title, String sortKey, Color textColor) {
|
DataColumn _buildSortableColumn(BuildContext context, String title, String sortKey) {
|
||||||
return DataColumn(label: InkWell(
|
return DataColumn(label: InkWell(
|
||||||
onTap: () => setState(() {
|
onTap: () => setState(() {
|
||||||
if (_sortColumn == sortKey) _isAscending = !_isAscending;
|
if (_sortColumn == sortKey) _isAscending = !_isAscending;
|
||||||
else { _sortColumn = sortKey; _isAscending = false; }
|
else { _sortColumn = sortKey; _isAscending = false; }
|
||||||
}),
|
}),
|
||||||
child: Row(children: [
|
child: Row(children: [
|
||||||
Text(title, style: TextStyle(fontSize: 12 * context.sf, fontWeight: FontWeight.bold, color: textColor)),
|
Text(title, style: TextStyle(fontSize: 12 * context.sf, fontWeight: FontWeight.bold)),
|
||||||
if (_sortColumn == sortKey) Icon(_isAscending ? Icons.arrow_drop_up : Icons.arrow_drop_down, size: 18 * context.sf, color: AppTheme.primaryRed),
|
if (_sortColumn == sortKey) Icon(_isAscending ? Icons.arrow_drop_up : Icons.arrow_drop_down, size: 18 * context.sf, color: const Color(0xFFE74C3C)),
|
||||||
]),
|
]),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
DataCell _buildStatCell(BuildContext context, int value, Color textColor, {bool isHighlight = false, bool isGold = false, bool isBlue = false}) {
|
DataCell _buildStatCell(BuildContext context, int value, {bool isHighlight = false, bool isGold = false, bool isBlue = false}) {
|
||||||
return DataCell(Center(child: Container(
|
return DataCell(Center(child: Container(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 8 * context.sf, vertical: 4 * context.sf),
|
padding: EdgeInsets.symmetric(horizontal: 8 * context.sf, vertical: 4 * context.sf),
|
||||||
decoration: BoxDecoration(color: isGold && value > 0 ? Colors.amber.withOpacity(0.2) : (isBlue && value > 0 ? Colors.blue.withOpacity(0.1) : Colors.transparent), borderRadius: BorderRadius.circular(6)),
|
decoration: BoxDecoration(color: isGold && value > 0 ? Colors.amber.withOpacity(0.2) : (isBlue && value > 0 ? Colors.blue.withOpacity(0.1) : Colors.transparent), borderRadius: BorderRadius.circular(6)),
|
||||||
child: Text(value == 0 ? "-" : value.toString(), style: TextStyle(
|
child: Text(value == 0 ? "-" : value.toString(), style: TextStyle(
|
||||||
fontWeight: (isHighlight || isGold || isBlue) ? FontWeight.w900 : FontWeight.w600,
|
fontWeight: (isHighlight || isGold || isBlue) ? FontWeight.w900 : FontWeight.w600,
|
||||||
fontSize: 14 * context.sf, color: isGold && value > 0 ? Colors.orange.shade900 : (isBlue && value > 0 ? Colors.blue.shade800 : (isHighlight ? AppTheme.successGreen : textColor))
|
fontSize: 14 * context.sf, color: isGold && value > 0 ? Colors.orange.shade900 : (isBlue && value > 0 ? Colors.blue.shade800 : (isHighlight ? Colors.green.shade700 : Colors.black87))
|
||||||
)),
|
)),
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
void _showTeamSelector(BuildContext context) {
|
void _showTeamSelector(BuildContext context) {
|
||||||
showModalBottomSheet(context: context, backgroundColor: Theme.of(context).colorScheme.surface, builder: (context) => StreamBuilder<List<Map<String, dynamic>>>(
|
showModalBottomSheet(context: context, builder: (context) => StreamBuilder<List<Map<String, dynamic>>>(
|
||||||
stream: _teamController.teamsStream,
|
stream: _teamController.teamsStream,
|
||||||
builder: (context, snapshot) {
|
builder: (context, snapshot) {
|
||||||
final teams = snapshot.data ?? [];
|
final teams = snapshot.data ?? [];
|
||||||
return ListView.builder(itemCount: teams.length, itemBuilder: (context, i) => ListTile(
|
return ListView.builder(itemCount: teams.length, itemBuilder: (context, i) => ListTile(
|
||||||
title: Text(teams[i]['name'], style: TextStyle(color: Theme.of(context).colorScheme.onSurface)),
|
title: Text(teams[i]['name']),
|
||||||
onTap: () { setState(() { _selectedTeamId = teams[i]['id']; _selectedTeamName = teams[i]['name']; }); Navigator.pop(context); },
|
onTap: () { setState(() { _selectedTeamId = teams[i]['id']; _selectedTeamName = teams[i]['name']; }); Navigator.pop(context); },
|
||||||
));
|
));
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,13 +1,8 @@
|
|||||||
import 'dart:io';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:image_picker/image_picker.dart';
|
|
||||||
import 'package:image_cropper/image_cropper.dart';
|
|
||||||
import 'package:cached_network_image/cached_network_image.dart'; // 👇 A MAGIA DO CACHE AQUI
|
|
||||||
import 'package:playmaker/screens/team_stats_page.dart';
|
import 'package:playmaker/screens/team_stats_page.dart';
|
||||||
import 'package:playmaker/classe/theme.dart';
|
|
||||||
import '../controllers/team_controller.dart';
|
import '../controllers/team_controller.dart';
|
||||||
import '../models/team_model.dart';
|
import '../models/team_model.dart';
|
||||||
import '../utils/size_extension.dart';
|
import '../utils/size_extension.dart'; // 👇 IMPORTANTE: O TEU NOVO SUPERPODER
|
||||||
|
|
||||||
class TeamsPage extends StatefulWidget {
|
class TeamsPage extends StatefulWidget {
|
||||||
const TeamsPage({super.key});
|
const TeamsPage({super.key});
|
||||||
@@ -37,14 +32,14 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
return StatefulBuilder(
|
return StatefulBuilder(
|
||||||
builder: (context, setModalState) {
|
builder: (context, setModalState) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
backgroundColor: const Color(0xFF2C3E50),
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)),
|
||||||
title: Row(
|
title: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Text("Filtros de pesquisa", style: TextStyle(color: Theme.of(context).colorScheme.onSurface, fontSize: 18 * context.sf, fontWeight: FontWeight.bold)),
|
Text("Filtros de pesquisa", style: TextStyle(color: Colors.white, fontSize: 18 * context.sf, fontWeight: FontWeight.bold)),
|
||||||
IconButton(
|
IconButton(
|
||||||
icon: Icon(Icons.close, color: Colors.grey, size: 20 * context.sf),
|
icon: Icon(Icons.close, color: Colors.white, size: 20 * context.sf),
|
||||||
onPressed: () => Navigator.pop(context),
|
onPressed: () => Navigator.pop(context),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
@@ -52,7 +47,7 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
content: Column(
|
content: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
Divider(color: Colors.grey.withOpacity(0.2)),
|
const Divider(color: Colors.white24),
|
||||||
SizedBox(height: 16 * context.sf),
|
SizedBox(height: 16 * context.sf),
|
||||||
Row(
|
Row(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
@@ -87,7 +82,7 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () => Navigator.pop(context),
|
onPressed: () => Navigator.pop(context),
|
||||||
child: Text("CONCLUÍDO", style: TextStyle(color: AppTheme.primaryRed, fontWeight: FontWeight.bold, fontSize: 14 * context.sf)),
|
child: Text("CONCLUÍDO", style: TextStyle(color: const Color(0xFFE74C3C), fontWeight: FontWeight.bold, fontSize: 14 * context.sf)),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
@@ -112,7 +107,7 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
child: Text(
|
child: Text(
|
||||||
opt,
|
opt,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
color: isSelected ? AppTheme.primaryRed : Theme.of(context).colorScheme.onSurface.withOpacity(0.7),
|
color: isSelected ? const Color(0xFFE74C3C) : Colors.white70,
|
||||||
fontWeight: isSelected ? FontWeight.bold : FontWeight.normal,
|
fontWeight: isSelected ? FontWeight.bold : FontWeight.normal,
|
||||||
fontSize: 14 * context.sf,
|
fontSize: 14 * context.sf,
|
||||||
),
|
),
|
||||||
@@ -126,15 +121,16 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
// 🔥 OLHA QUE LIMPEZA: Já não precisamos de calcular nada aqui!
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
backgroundColor: const Color(0xFFF5F7FA),
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text("Minhas Equipas", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 20 * context.sf)),
|
title: Text("Minhas Equipas", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 20 * context.sf)),
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
backgroundColor: const Color(0xFFF5F7FA),
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
actions: [
|
actions: [
|
||||||
IconButton(
|
IconButton(
|
||||||
icon: Icon(Icons.filter_list, color: AppTheme.primaryRed, size: 24 * context.sf),
|
icon: Icon(Icons.filter_list, color: const Color(0xFFE74C3C), size: 24 * context.sf),
|
||||||
onPressed: () => _showFilterDialog(context),
|
onPressed: () => _showFilterDialog(context),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -146,8 +142,8 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
floatingActionButton: FloatingActionButton(
|
floatingActionButton: FloatingActionButton(
|
||||||
heroTag: 'add_team_btn',
|
heroTag: 'add_team_btn', // 👇 A MÁGICA ESTÁ AQUI!
|
||||||
backgroundColor: AppTheme.primaryRed,
|
backgroundColor: const Color(0xFFE74C3C),
|
||||||
child: Icon(Icons.add, color: Colors.white, size: 24 * context.sf),
|
child: Icon(Icons.add, color: Colors.white, size: 24 * context.sf),
|
||||||
onPressed: () => _showCreateDialog(context),
|
onPressed: () => _showCreateDialog(context),
|
||||||
),
|
),
|
||||||
@@ -160,13 +156,13 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
child: TextField(
|
child: TextField(
|
||||||
controller: _searchController,
|
controller: _searchController,
|
||||||
onChanged: (v) => setState(() => _searchQuery = v.toLowerCase()),
|
onChanged: (v) => setState(() => _searchQuery = v.toLowerCase()),
|
||||||
style: TextStyle(fontSize: 16 * context.sf, color: Theme.of(context).colorScheme.onSurface),
|
style: TextStyle(fontSize: 16 * context.sf),
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
hintText: 'Pesquisar equipa...',
|
hintText: 'Pesquisar equipa...',
|
||||||
hintStyle: TextStyle(fontSize: 16 * context.sf, color: Colors.grey),
|
hintStyle: TextStyle(fontSize: 16 * context.sf),
|
||||||
prefixIcon: Icon(Icons.search, color: AppTheme.primaryRed, size: 22 * context.sf),
|
prefixIcon: Icon(Icons.search, color: const Color(0xFFE74C3C), size: 22 * context.sf),
|
||||||
filled: true,
|
filled: true,
|
||||||
fillColor: Theme.of(context).colorScheme.surface,
|
fillColor: Colors.white,
|
||||||
border: OutlineInputBorder(borderRadius: BorderRadius.circular(15 * context.sf), borderSide: BorderSide.none),
|
border: OutlineInputBorder(borderRadius: BorderRadius.circular(15 * context.sf), borderSide: BorderSide.none),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -174,11 +170,11 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildTeamsList() {
|
Widget _buildTeamsList() {
|
||||||
return FutureBuilder<List<Map<String, dynamic>>>(
|
return StreamBuilder<List<Map<String, dynamic>>>(
|
||||||
future: controller.getTeamsWithStats(),
|
stream: controller.teamsStream,
|
||||||
builder: (context, snapshot) {
|
builder: (context, snapshot) {
|
||||||
if (snapshot.connectionState == ConnectionState.waiting) return Center(child: CircularProgressIndicator(color: AppTheme.primaryRed));
|
if (snapshot.connectionState == ConnectionState.waiting) return const Center(child: CircularProgressIndicator());
|
||||||
if (!snapshot.hasData || snapshot.data!.isEmpty) return Center(child: Text("Nenhuma equipa encontrada.", style: TextStyle(fontSize: 16 * context.sf, color: Theme.of(context).colorScheme.onSurface)));
|
if (!snapshot.hasData || snapshot.data!.isEmpty) return Center(child: Text("Nenhuma equipa encontrada.", style: TextStyle(fontSize: 16 * context.sf)));
|
||||||
|
|
||||||
var data = List<Map<String, dynamic>>.from(snapshot.data!);
|
var data = List<Map<String, dynamic>>.from(snapshot.data!);
|
||||||
|
|
||||||
@@ -194,45 +190,27 @@ class _TeamsPageState extends State<TeamsPage> {
|
|||||||
else return (b['created_at'] ?? '').toString().compareTo((a['created_at'] ?? '').toString());
|
else return (b['created_at'] ?? '').toString().compareTo((a['created_at'] ?? '').toString());
|
||||||
});
|
});
|
||||||
|
|
||||||
return RefreshIndicator(
|
return ListView.builder(
|
||||||
color: AppTheme.primaryRed,
|
padding: EdgeInsets.symmetric(horizontal: 16 * context.sf),
|
||||||
onRefresh: () async => setState(() {}),
|
itemCount: data.length,
|
||||||
child: ListView.builder(
|
itemBuilder: (context, index) {
|
||||||
padding: EdgeInsets.symmetric(horizontal: 16 * context.sf),
|
final team = Team.fromMap(data[index]);
|
||||||
itemCount: data.length,
|
return GestureDetector(
|
||||||
itemBuilder: (context, index) {
|
onTap: () => Navigator.push(context, MaterialPageRoute(builder: (context) => TeamStatsPage(team: team))),
|
||||||
final team = Team.fromMap(data[index]);
|
child: TeamCard(
|
||||||
return GestureDetector(
|
team: team,
|
||||||
onTap: () => Navigator.push(context, MaterialPageRoute(builder: (context) => TeamStatsPage(team: team))).then((_) => setState(() {})),
|
controller: controller,
|
||||||
child: TeamCard(
|
onFavoriteTap: () => controller.toggleFavorite(team.id, team.isFavorite),
|
||||||
team: team,
|
),
|
||||||
controller: controller,
|
);
|
||||||
onFavoriteTap: () async {
|
},
|
||||||
await controller.toggleFavorite(team.id, team.isFavorite);
|
|
||||||
setState(() {});
|
|
||||||
},
|
|
||||||
onDelete: () => setState(() {}),
|
|
||||||
sf: context.sf,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _showCreateDialog(BuildContext context) {
|
void _showCreateDialog(BuildContext context) {
|
||||||
showDialog(
|
showDialog(context: context, builder: (context) => CreateTeamDialog(onConfirm: (name, season, imageUrl) => controller.createTeam(name, season, imageUrl)));
|
||||||
context: context,
|
|
||||||
builder: (context) => CreateTeamDialog(
|
|
||||||
sf: context.sf,
|
|
||||||
onConfirm: (name, season, imageFile) async {
|
|
||||||
await controller.createTeam(name, season, imageFile);
|
|
||||||
setState(() {});
|
|
||||||
}
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -241,160 +219,83 @@ class TeamCard extends StatelessWidget {
|
|||||||
final Team team;
|
final Team team;
|
||||||
final TeamController controller;
|
final TeamController controller;
|
||||||
final VoidCallback onFavoriteTap;
|
final VoidCallback onFavoriteTap;
|
||||||
final VoidCallback onDelete;
|
|
||||||
final double sf;
|
|
||||||
|
|
||||||
const TeamCard({
|
const TeamCard({super.key, required this.team, required this.controller, required this.onFavoriteTap});
|
||||||
super.key,
|
|
||||||
required this.team,
|
|
||||||
required this.controller,
|
|
||||||
required this.onFavoriteTap,
|
|
||||||
required this.onDelete,
|
|
||||||
required this.sf,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final bgColor = Theme.of(context).cardTheme.color ?? Theme.of(context).colorScheme.surface;
|
return Card(
|
||||||
final textColor = Theme.of(context).colorScheme.onSurface;
|
color: Colors.white, elevation: 3, margin: EdgeInsets.only(bottom: 12 * context.sf),
|
||||||
final double avatarSize = 56 * sf; // 2 * radius (28)
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * context.sf)),
|
||||||
|
child: ListTile(
|
||||||
return Container(
|
contentPadding: EdgeInsets.symmetric(horizontal: 16 * context.sf, vertical: 8 * context.sf),
|
||||||
margin: EdgeInsets.only(bottom: 12 * sf),
|
leading: Stack(
|
||||||
decoration: BoxDecoration(
|
clipBehavior: Clip.none,
|
||||||
color: bgColor,
|
children: [
|
||||||
borderRadius: BorderRadius.circular(15 * sf),
|
CircleAvatar(
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.15)),
|
radius: 28 * context.sf, backgroundColor: Colors.grey[200],
|
||||||
boxShadow: [BoxShadow(color: Colors.black.withOpacity(0.05), blurRadius: 10 * sf)]
|
backgroundImage: (team.imageUrl.isNotEmpty && team.imageUrl.startsWith('http')) ? NetworkImage(team.imageUrl) : null,
|
||||||
),
|
child: (team.imageUrl.isEmpty || !team.imageUrl.startsWith('http')) ? Text(team.imageUrl.isEmpty ? "🏀" : team.imageUrl, style: TextStyle(fontSize: 24 * context.sf)) : null,
|
||||||
child: Material(
|
|
||||||
color: Colors.transparent,
|
|
||||||
borderRadius: BorderRadius.circular(15 * sf),
|
|
||||||
child: ListTile(
|
|
||||||
contentPadding: EdgeInsets.symmetric(horizontal: 16 * sf, vertical: 8 * sf),
|
|
||||||
leading: Stack(
|
|
||||||
clipBehavior: Clip.none,
|
|
||||||
children: [
|
|
||||||
// 👇 AVATAR DA EQUIPA OTIMIZADO COM CACHE 👇
|
|
||||||
ClipOval(
|
|
||||||
child: Container(
|
|
||||||
width: avatarSize,
|
|
||||||
height: avatarSize,
|
|
||||||
color: Colors.grey.withOpacity(0.2),
|
|
||||||
child: (team.imageUrl.isNotEmpty && team.imageUrl.startsWith('http'))
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: team.imageUrl,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero, // Fica instantâneo no scroll
|
|
||||||
placeholder: (context, url) => const SizedBox(), // Fica só o fundo cinza
|
|
||||||
errorWidget: (context, url, error) => Center(
|
|
||||||
child: Text("🏀", style: TextStyle(fontSize: 24 * sf)),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: Center(
|
|
||||||
child: Text(
|
|
||||||
team.imageUrl.isEmpty ? "🏀" : team.imageUrl,
|
|
||||||
style: TextStyle(fontSize: 24 * sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
left: -15 * sf,
|
|
||||||
top: -10 * sf,
|
|
||||||
child: IconButton(
|
|
||||||
icon: Icon(
|
|
||||||
team.isFavorite ? Icons.star : Icons.star_border,
|
|
||||||
color: team.isFavorite ? AppTheme.warningAmber : Theme.of(context).colorScheme.onSurface.withOpacity(0.2),
|
|
||||||
size: 28 * sf,
|
|
||||||
shadows: [Shadow(color: Colors.black.withOpacity(team.isFavorite ? 0.3 : 0.1), blurRadius: 4 * sf)],
|
|
||||||
),
|
|
||||||
onPressed: onFavoriteTap,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
title: Text(
|
|
||||||
team.name,
|
|
||||||
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16 * sf, color: textColor),
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
),
|
|
||||||
subtitle: Padding(
|
|
||||||
padding: EdgeInsets.only(top: 6.0 * sf),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
Icon(Icons.groups_outlined, size: 16 * sf, color: Colors.grey),
|
|
||||||
SizedBox(width: 4 * sf),
|
|
||||||
Text(
|
|
||||||
"${team.playerCount} Jogs.",
|
|
||||||
style: TextStyle(
|
|
||||||
color: team.playerCount > 0 ? AppTheme.successGreen : AppTheme.warningAmber,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 13 * sf,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(width: 8 * sf),
|
|
||||||
Expanded(
|
|
||||||
child: Text("| ${team.season}", style: TextStyle(color: Colors.grey, fontSize: 13 * sf), overflow: TextOverflow.ellipsis),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
),
|
||||||
),
|
Positioned(
|
||||||
trailing: Row(
|
left: -15 * context.sf, top: -10 * context.sf,
|
||||||
mainAxisSize: MainAxisSize.min,
|
child: IconButton(
|
||||||
|
icon: Icon(team.isFavorite ? Icons.star : Icons.star_border, color: team.isFavorite ? Colors.amber : Colors.black.withOpacity(0.1), size: 28 * context.sf, shadows: [Shadow(color: Colors.black.withOpacity(team.isFavorite ? 0.3 : 0.1), blurRadius: 4 * context.sf)]),
|
||||||
|
onPressed: onFavoriteTap,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
title: Text(team.name, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16 * context.sf), overflow: TextOverflow.ellipsis),
|
||||||
|
subtitle: Padding(
|
||||||
|
padding: EdgeInsets.only(top: 6.0 * context.sf),
|
||||||
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
IconButton(
|
Icon(Icons.groups_outlined, size: 16 * context.sf, color: Colors.grey),
|
||||||
tooltip: 'Ver Estatísticas',
|
SizedBox(width: 4 * context.sf),
|
||||||
icon: Icon(Icons.bar_chart_rounded, color: Colors.blue, size: 24 * sf),
|
StreamBuilder<int>(
|
||||||
onPressed: () => Navigator.push(context, MaterialPageRoute(builder: (context) => TeamStatsPage(team: team))).then((_) => onDelete()),
|
stream: controller.getPlayerCountStream(team.id),
|
||||||
),
|
initialData: 0,
|
||||||
IconButton(
|
builder: (context, snapshot) {
|
||||||
tooltip: 'Eliminar Equipa',
|
final count = snapshot.data ?? 0;
|
||||||
icon: Icon(Icons.delete_outline, color: AppTheme.primaryRed, size: 24 * sf),
|
return Text("$count Jogs.", style: TextStyle(color: count > 0 ? Colors.green[700] : Colors.orange, fontWeight: FontWeight.bold, fontSize: 13 * context.sf));
|
||||||
onPressed: () => _confirmDelete(context, sf, bgColor, textColor),
|
},
|
||||||
),
|
),
|
||||||
|
SizedBox(width: 8 * context.sf),
|
||||||
|
Expanded(child: Text("| ${team.season}", style: TextStyle(color: Colors.grey, fontSize: 13 * context.sf), overflow: TextOverflow.ellipsis)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
trailing: Row(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
IconButton(tooltip: 'Ver Estatísticas', icon: Icon(Icons.bar_chart_rounded, color: Colors.blue, size: 24 * context.sf), onPressed: () => Navigator.push(context, MaterialPageRoute(builder: (context) => TeamStatsPage(team: team)))),
|
||||||
|
IconButton(tooltip: 'Eliminar Equipa', icon: Icon(Icons.delete_outline, color: const Color(0xFFE74C3C), size: 24 * context.sf), onPressed: () => _confirmDelete(context)),
|
||||||
|
],
|
||||||
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _confirmDelete(BuildContext context, double sf, Color cardColor, Color textColor) {
|
void _confirmDelete(BuildContext context) {
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (ctx) => AlertDialog(
|
builder: (context) => AlertDialog(
|
||||||
backgroundColor: cardColor,
|
title: Text('Eliminar Equipa?', style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold)),
|
||||||
surfaceTintColor: Colors.transparent,
|
content: Text('Tens a certeza que queres eliminar "${team.name}"?', style: TextStyle(fontSize: 14 * context.sf)),
|
||||||
title: Text('Eliminar Equipa?', style: TextStyle(fontSize: 18 * sf, fontWeight: FontWeight.bold, color: textColor)),
|
|
||||||
content: Text('Tens a certeza que queres eliminar "${team.name}"?', style: TextStyle(fontSize: 14 * sf, color: textColor)),
|
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(onPressed: () => Navigator.pop(context), child: Text('Cancelar', style: TextStyle(fontSize: 14 * context.sf))),
|
||||||
onPressed: () => Navigator.pop(ctx),
|
TextButton(onPressed: () { controller.deleteTeam(team.id); Navigator.pop(context); }, child: Text('Eliminar', style: TextStyle(color: Colors.red, fontSize: 14 * context.sf))),
|
||||||
child: Text('Cancelar', style: TextStyle(fontSize: 14 * sf, color: Colors.grey)),
|
|
||||||
),
|
|
||||||
TextButton(
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.pop(ctx);
|
|
||||||
onDelete();
|
|
||||||
controller.deleteTeam(team.id).catchError((e) {
|
|
||||||
if (context.mounted) ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('Erro ao eliminar: $e'), backgroundColor: Colors.red));
|
|
||||||
});
|
|
||||||
},
|
|
||||||
child: Text('Eliminar', style: TextStyle(color: AppTheme.primaryRed, fontSize: 14 * sf)),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- DIALOG DE CRIAÇÃO (COM CROPPER E ESCUDO) ---
|
// --- DIALOG DE CRIAÇÃO ---
|
||||||
class CreateTeamDialog extends StatefulWidget {
|
class CreateTeamDialog extends StatefulWidget {
|
||||||
final Function(String name, String season, File? imageFile) onConfirm;
|
final Function(String name, String season, String imageUrl) onConfirm;
|
||||||
final double sf;
|
const CreateTeamDialog({super.key, required this.onConfirm});
|
||||||
|
|
||||||
const CreateTeamDialog({super.key, required this.onConfirm, required this.sf});
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<CreateTeamDialog> createState() => _CreateTeamDialogState();
|
State<CreateTeamDialog> createState() => _CreateTeamDialogState();
|
||||||
@@ -402,112 +303,37 @@ class CreateTeamDialog extends StatefulWidget {
|
|||||||
|
|
||||||
class _CreateTeamDialogState extends State<CreateTeamDialog> {
|
class _CreateTeamDialogState extends State<CreateTeamDialog> {
|
||||||
final TextEditingController _nameController = TextEditingController();
|
final TextEditingController _nameController = TextEditingController();
|
||||||
|
final TextEditingController _imageController = TextEditingController();
|
||||||
String _selectedSeason = '2024/25';
|
String _selectedSeason = '2024/25';
|
||||||
|
|
||||||
File? _selectedImage;
|
|
||||||
bool _isLoading = false;
|
|
||||||
bool _isPickerActive = false;
|
|
||||||
|
|
||||||
Future<void> _pickImage() async {
|
|
||||||
if (_isPickerActive) return;
|
|
||||||
setState(() => _isPickerActive = true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
final ImagePicker picker = ImagePicker();
|
|
||||||
final XFile? pickedFile = await picker.pickImage(source: ImageSource.gallery);
|
|
||||||
|
|
||||||
if (pickedFile != null) {
|
|
||||||
CroppedFile? croppedFile = await ImageCropper().cropImage(
|
|
||||||
sourcePath: pickedFile.path,
|
|
||||||
aspectRatio: const CropAspectRatio(ratioX: 1, ratioY: 1),
|
|
||||||
uiSettings: [
|
|
||||||
AndroidUiSettings(
|
|
||||||
toolbarTitle: 'Recortar Logo',
|
|
||||||
toolbarColor: AppTheme.primaryRed,
|
|
||||||
toolbarWidgetColor: Colors.white,
|
|
||||||
initAspectRatio: CropAspectRatioPreset.square,
|
|
||||||
lockAspectRatio: true,
|
|
||||||
hideBottomControls: true,
|
|
||||||
),
|
|
||||||
IOSUiSettings(title: 'Recortar Logo', aspectRatioLockEnabled: true, resetButtonHidden: true),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
|
|
||||||
if (croppedFile != null && mounted) {
|
|
||||||
setState(() {
|
|
||||||
_selectedImage = File(croppedFile.path);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (mounted) setState(() => _isPickerActive = false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * context.sf)),
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * widget.sf)),
|
title: Text('Nova Equipa', style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold)),
|
||||||
title: Text('Nova Equipa', style: TextStyle(fontSize: 18 * widget.sf, fontWeight: FontWeight.bold, color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
content: SingleChildScrollView(
|
content: SingleChildScrollView(
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
GestureDetector(
|
TextField(controller: _nameController, style: TextStyle(fontSize: 14 * context.sf), decoration: InputDecoration(labelText: 'Nome da Equipa', labelStyle: TextStyle(fontSize: 14 * context.sf)), textCapitalization: TextCapitalization.words),
|
||||||
onTap: _pickImage,
|
SizedBox(height: 15 * context.sf),
|
||||||
child: Stack(
|
|
||||||
children: [
|
|
||||||
CircleAvatar(
|
|
||||||
radius: 40 * widget.sf,
|
|
||||||
backgroundColor: Theme.of(context).colorScheme.onSurface.withOpacity(0.05),
|
|
||||||
backgroundImage: _selectedImage != null ? FileImage(_selectedImage!) : null,
|
|
||||||
child: _selectedImage == null
|
|
||||||
? Icon(Icons.add_photo_alternate_outlined, size: 30 * widget.sf, color: Colors.grey)
|
|
||||||
: null,
|
|
||||||
),
|
|
||||||
if (_selectedImage == null)
|
|
||||||
Positioned(
|
|
||||||
bottom: 0, right: 0,
|
|
||||||
child: Container(
|
|
||||||
padding: EdgeInsets.all(4 * widget.sf),
|
|
||||||
decoration: const BoxDecoration(color: AppTheme.primaryRed, shape: BoxShape.circle),
|
|
||||||
child: Icon(Icons.add, color: Colors.white, size: 16 * widget.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(height: 10 * widget.sf),
|
|
||||||
Text("Logótipo (Opcional)", style: TextStyle(fontSize: 12 * widget.sf, color: Colors.grey)),
|
|
||||||
SizedBox(height: 20 * widget.sf),
|
|
||||||
|
|
||||||
TextField(controller: _nameController, style: TextStyle(fontSize: 14 * widget.sf, color: Theme.of(context).colorScheme.onSurface), decoration: InputDecoration(labelText: 'Nome da Equipa', labelStyle: TextStyle(fontSize: 14 * widget.sf)), textCapitalization: TextCapitalization.words),
|
|
||||||
SizedBox(height: 15 * widget.sf),
|
|
||||||
DropdownButtonFormField<String>(
|
DropdownButtonFormField<String>(
|
||||||
dropdownColor: Theme.of(context).colorScheme.surface,
|
value: _selectedSeason, decoration: InputDecoration(labelText: 'Temporada', labelStyle: TextStyle(fontSize: 14 * context.sf)),
|
||||||
value: _selectedSeason, decoration: InputDecoration(labelText: 'Temporada', labelStyle: TextStyle(fontSize: 14 * widget.sf)),
|
style: TextStyle(fontSize: 14 * context.sf, color: Colors.black87),
|
||||||
style: TextStyle(fontSize: 14 * widget.sf, color: Theme.of(context).colorScheme.onSurface),
|
|
||||||
items: ['2023/24', '2024/25', '2025/26'].map((s) => DropdownMenuItem(value: s, child: Text(s))).toList(),
|
items: ['2023/24', '2024/25', '2025/26'].map((s) => DropdownMenuItem(value: s, child: Text(s))).toList(),
|
||||||
onChanged: (val) => setState(() => _selectedSeason = val!),
|
onChanged: (val) => setState(() => _selectedSeason = val!),
|
||||||
),
|
),
|
||||||
|
SizedBox(height: 15 * context.sf),
|
||||||
|
TextField(controller: _imageController, style: TextStyle(fontSize: 14 * context.sf), decoration: InputDecoration(labelText: 'URL Imagem ou Emoji', labelStyle: TextStyle(fontSize: 14 * context.sf), hintText: 'Ex: 🏀 ou https://...', hintStyle: TextStyle(fontSize: 14 * context.sf))),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(onPressed: () => Navigator.pop(context), child: Text('Cancelar', style: TextStyle(fontSize: 14 * widget.sf, color: Colors.grey))),
|
TextButton(onPressed: () => Navigator.pop(context), child: Text('Cancelar', style: TextStyle(fontSize: 14 * context.sf))),
|
||||||
ElevatedButton(
|
ElevatedButton(
|
||||||
style: ElevatedButton.styleFrom(backgroundColor: AppTheme.primaryRed, padding: EdgeInsets.symmetric(horizontal: 16 * widget.sf, vertical: 10 * widget.sf)),
|
style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFFE74C3C), padding: EdgeInsets.symmetric(horizontal: 16 * context.sf, vertical: 10 * context.sf)),
|
||||||
onPressed: _isLoading ? null : () async {
|
onPressed: () { if (_nameController.text.trim().isNotEmpty) { widget.onConfirm(_nameController.text.trim(), _selectedSeason, _imageController.text.trim()); Navigator.pop(context); } },
|
||||||
if (_nameController.text.trim().isNotEmpty) {
|
child: Text('Criar', style: TextStyle(color: Colors.white, fontSize: 14 * context.sf)),
|
||||||
setState(() => _isLoading = true);
|
|
||||||
await widget.onConfirm(_nameController.text.trim(), _selectedSeason, _selectedImage);
|
|
||||||
if (context.mounted) Navigator.pop(context);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: _isLoading
|
|
||||||
? SizedBox(width: 16 * widget.sf, height: 16 * widget.sf, child: const CircularProgressIndicator(color: Colors.white, strokeWidth: 2))
|
|
||||||
: Text('Criar', style: TextStyle(color: Colors.white, fontSize: 14 * widget.sf)),
|
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,103 +1,53 @@
|
|||||||
import 'dart:io';
|
import 'dart:async';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:image_picker/image_picker.dart';
|
|
||||||
import 'package:image_cropper/image_cropper.dart';
|
|
||||||
import 'package:shimmer/shimmer.dart';
|
|
||||||
import 'package:cached_network_image/cached_network_image.dart'; // 👇 MAGIA DO CACHE AQUI
|
|
||||||
import 'package:supabase_flutter/supabase_flutter.dart';
|
import 'package:supabase_flutter/supabase_flutter.dart';
|
||||||
import 'package:playmaker/classe/theme.dart';
|
|
||||||
import '../models/team_model.dart';
|
import '../models/team_model.dart';
|
||||||
import '../models/person_model.dart';
|
import '../models/person_model.dart';
|
||||||
import '../utils/size_extension.dart';
|
|
||||||
|
|
||||||
// ==========================================
|
// --- CABEÇALHO ---
|
||||||
// 1. CABEÇALHO (AGORA COM CACHE DE IMAGEM INSTANTÂNEO)
|
|
||||||
// ==========================================
|
|
||||||
class StatsHeader extends StatelessWidget {
|
class StatsHeader extends StatelessWidget {
|
||||||
final Team team;
|
final Team team;
|
||||||
final String? currentImageUrl;
|
|
||||||
final VoidCallback onEditPhoto;
|
|
||||||
final bool isUploading;
|
|
||||||
|
|
||||||
const StatsHeader({
|
const StatsHeader({super.key, required this.team});
|
||||||
super.key,
|
|
||||||
required this.team,
|
|
||||||
required this.currentImageUrl,
|
|
||||||
required this.onEditPhoto,
|
|
||||||
required this.isUploading,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Container(
|
return Container(
|
||||||
padding: EdgeInsets.only(top: 50 * context.sf, left: 20 * context.sf, right: 20 * context.sf, bottom: 20 * context.sf),
|
padding: const EdgeInsets.only(top: 50, left: 20, right: 20, bottom: 20),
|
||||||
decoration: BoxDecoration(
|
decoration: const BoxDecoration(
|
||||||
color: AppTheme.primaryRed,
|
color: Color(0xFF2C3E50),
|
||||||
borderRadius: BorderRadius.only(
|
borderRadius: BorderRadius.only(bottomLeft: Radius.circular(30), bottomRight: Radius.circular(30)),
|
||||||
bottomLeft: Radius.circular(30 * context.sf),
|
|
||||||
bottomRight: Radius.circular(30 * context.sf)
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
IconButton(
|
IconButton(
|
||||||
icon: Icon(Icons.arrow_back, color: Colors.white, size: 24 * context.sf),
|
icon: const Icon(Icons.arrow_back, color: Colors.white),
|
||||||
onPressed: () => Navigator.pop(context)
|
onPressed: () => Navigator.pop(context),
|
||||||
),
|
),
|
||||||
SizedBox(width: 10 * context.sf),
|
const SizedBox(width: 10),
|
||||||
|
|
||||||
GestureDetector(
|
// IMAGEM OU EMOJI DA EQUIPA AQUI!
|
||||||
onTap: onEditPhoto,
|
CircleAvatar(
|
||||||
child: Stack(
|
radius: 24,
|
||||||
alignment: Alignment.center,
|
backgroundColor: Colors.white24,
|
||||||
children: [
|
backgroundImage: (team.imageUrl.isNotEmpty && team.imageUrl.startsWith('http'))
|
||||||
// 👇 AVATAR DA EQUIPA SEM LAG 👇
|
? NetworkImage(team.imageUrl)
|
||||||
ClipOval(
|
: null,
|
||||||
child: Container(
|
child: (team.imageUrl.isEmpty || !team.imageUrl.startsWith('http'))
|
||||||
width: 56 * context.sf,
|
? Text(
|
||||||
height: 56 * context.sf,
|
team.imageUrl.isEmpty ? "🛡️" : team.imageUrl,
|
||||||
color: Colors.white24,
|
style: const TextStyle(fontSize: 20),
|
||||||
child: (currentImageUrl != null && currentImageUrl!.isNotEmpty && currentImageUrl!.startsWith('http'))
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: currentImageUrl!,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero, // Corta o atraso
|
|
||||||
placeholder: (context, url) => Center(child: Text("🛡️", style: TextStyle(fontSize: 24 * context.sf))),
|
|
||||||
errorWidget: (context, url, error) => Center(child: Text("🛡️", style: TextStyle(fontSize: 24 * context.sf))),
|
|
||||||
)
|
|
||||||
: Center(
|
|
||||||
child: Text(
|
|
||||||
(currentImageUrl != null && currentImageUrl!.isNotEmpty) ? currentImageUrl! : "🛡️",
|
|
||||||
style: TextStyle(fontSize: 24 * context.sf)
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
bottom: 0, right: 0,
|
|
||||||
child: Container(
|
|
||||||
padding: EdgeInsets.all(4 * context.sf),
|
|
||||||
decoration: const BoxDecoration(color: Colors.white, shape: BoxShape.circle),
|
|
||||||
child: Icon(Icons.edit, color: AppTheme.primaryRed, size: 12 * context.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (isUploading)
|
|
||||||
Container(
|
|
||||||
width: 56 * context.sf, height: 56 * context.sf,
|
|
||||||
decoration: const BoxDecoration(color: Colors.black45, shape: BoxShape.circle),
|
|
||||||
child: const Padding(padding: EdgeInsets.all(12.0), child: CircularProgressIndicator(color: Colors.white, strokeWidth: 2)),
|
|
||||||
)
|
)
|
||||||
],
|
: null,
|
||||||
),
|
|
||||||
),
|
),
|
||||||
|
|
||||||
SizedBox(width: 15 * context.sf),
|
const SizedBox(width: 15),
|
||||||
Expanded(
|
Expanded( // Expanded evita overflow se o nome for muito longo
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Text(team.name, style: TextStyle(color: Colors.white, fontSize: 20 * context.sf, fontWeight: FontWeight.bold), overflow: TextOverflow.ellipsis),
|
Text(team.name, style: const TextStyle(color: Colors.white, fontSize: 20, fontWeight: FontWeight.bold), overflow: TextOverflow.ellipsis),
|
||||||
Text(team.season, style: TextStyle(color: Colors.white70, fontSize: 14 * context.sf)),
|
Text(team.season, style: const TextStyle(color: Colors.white70, fontSize: 14)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -110,28 +60,25 @@ class StatsHeader extends StatelessWidget {
|
|||||||
// --- CARD DE RESUMO ---
|
// --- CARD DE RESUMO ---
|
||||||
class StatsSummaryCard extends StatelessWidget {
|
class StatsSummaryCard extends StatelessWidget {
|
||||||
final int total;
|
final int total;
|
||||||
|
|
||||||
const StatsSummaryCard({super.key, required this.total});
|
const StatsSummaryCard({super.key, required this.total});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final Color bgColor = Theme.of(context).brightness == Brightness.dark ? const Color(0xFF1E1E1E) : Colors.white;
|
|
||||||
return Card(
|
return Card(
|
||||||
elevation: 4,
|
elevation: 4,
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20)),
|
||||||
child: Container(
|
child: Container(
|
||||||
padding: EdgeInsets.all(20 * context.sf),
|
padding: const EdgeInsets.all(20),
|
||||||
decoration: BoxDecoration(color: bgColor, borderRadius: BorderRadius.circular(20 * context.sf), border: Border.all(color: Colors.grey.withOpacity(0.15))),
|
decoration: BoxDecoration(
|
||||||
|
borderRadius: BorderRadius.circular(20),
|
||||||
|
gradient: LinearGradient(colors: [Colors.blue.shade700, Colors.blue.shade400]),
|
||||||
|
),
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Row(
|
const Text("Total de Membros", style: TextStyle(color: Colors.white, fontSize: 16)),
|
||||||
children: [
|
Text("$total", style: const TextStyle(color: Colors.white, fontSize: 28, fontWeight: FontWeight.bold)),
|
||||||
Icon(Icons.groups, color: AppTheme.primaryRed, size: 28 * context.sf),
|
|
||||||
SizedBox(width: 10 * context.sf),
|
|
||||||
Text("Total de Membros", style: TextStyle(color: Theme.of(context).colorScheme.onSurface, fontSize: 16 * context.sf, fontWeight: FontWeight.w600)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Text("$total", style: TextStyle(color: Theme.of(context).colorScheme.onSurface, fontSize: 28 * context.sf, fontWeight: FontWeight.bold)),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -142,6 +89,7 @@ class StatsSummaryCard extends StatelessWidget {
|
|||||||
// --- TÍTULO DE SECÇÃO ---
|
// --- TÍTULO DE SECÇÃO ---
|
||||||
class StatsSectionTitle extends StatelessWidget {
|
class StatsSectionTitle extends StatelessWidget {
|
||||||
final String title;
|
final String title;
|
||||||
|
|
||||||
const StatsSectionTitle({super.key, required this.title});
|
const StatsSectionTitle({super.key, required this.title});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -149,119 +97,63 @@ class StatsSectionTitle extends StatelessWidget {
|
|||||||
return Column(
|
return Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Text(title, style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold, color: Theme.of(context).colorScheme.onSurface)),
|
Text(title, style: const TextStyle(fontSize: 18, fontWeight: FontWeight.bold, color: Color(0xFF2C3E50))),
|
||||||
Divider(color: Colors.grey.withOpacity(0.2)),
|
const Divider(),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- CARD DA PESSOA (FOTO SEM LAG) ---
|
// --- CARD DA PESSOA (JOGADOR/TREINADOR) ---
|
||||||
class PersonCard extends StatelessWidget {
|
class PersonCard extends StatelessWidget {
|
||||||
final Person person;
|
final Person person;
|
||||||
final bool isCoach;
|
final bool isCoach;
|
||||||
final VoidCallback onEdit;
|
final VoidCallback onEdit;
|
||||||
final VoidCallback onDelete;
|
final VoidCallback onDelete;
|
||||||
|
|
||||||
const PersonCard({super.key, required this.person, required this.isCoach, required this.onEdit, required this.onDelete});
|
const PersonCard({
|
||||||
|
super.key,
|
||||||
|
required this.person,
|
||||||
|
required this.isCoach,
|
||||||
|
required this.onEdit,
|
||||||
|
required this.onDelete,
|
||||||
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final Color defaultBg = Theme.of(context).brightness == Brightness.dark ? const Color(0xFF1E1E1E) : Colors.white;
|
|
||||||
final Color coachBg = Theme.of(context).brightness == Brightness.dark ? AppTheme.warningAmber.withOpacity(0.1) : const Color(0xFFFFF9C4);
|
|
||||||
final String? pImage = person.imageUrl;
|
|
||||||
final Color iconColor = isCoach ? Colors.white : AppTheme.primaryRed;
|
|
||||||
|
|
||||||
return Card(
|
return Card(
|
||||||
margin: EdgeInsets.only(top: 12 * context.sf),
|
margin: const EdgeInsets.only(top: 12),
|
||||||
elevation: 2,
|
elevation: 2,
|
||||||
color: isCoach ? coachBg : defaultBg,
|
color: isCoach ? const Color(0xFFFFF9C4) : Colors.white,
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * context.sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15)),
|
||||||
child: Padding(
|
child: ListTile(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 16 * context.sf, vertical: 12 * context.sf),
|
leading: isCoach
|
||||||
child: Row(
|
? const CircleAvatar(backgroundColor: Colors.orange, child: Icon(Icons.person, color: Colors.white))
|
||||||
|
: Container(
|
||||||
|
width: 45,
|
||||||
|
height: 45,
|
||||||
|
alignment: Alignment.center,
|
||||||
|
decoration: BoxDecoration(color: Colors.blue.withOpacity(0.1), borderRadius: BorderRadius.circular(10)),
|
||||||
|
child: Text(person.number ?? "J", style: const TextStyle(color: Colors.blue, fontWeight: FontWeight.bold, fontSize: 16)),
|
||||||
|
),
|
||||||
|
title: Text(person.name, style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
|
||||||
|
// --- CANTO DIREITO (Trailing) ---
|
||||||
|
trailing: Row(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
// 👇 FOTO DO JOGADOR/TREINADOR INSTANTÂNEA 👇
|
// IMAGEM DA EQUIPA NO CARD DO JOGADOR
|
||||||
ClipOval(
|
|
||||||
child: Container(
|
|
||||||
width: 44 * context.sf,
|
|
||||||
height: 44 * context.sf,
|
|
||||||
color: isCoach ? AppTheme.warningAmber : AppTheme.primaryRed.withOpacity(0.1),
|
|
||||||
child: (pImage != null && pImage.isNotEmpty)
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: pImage,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero,
|
|
||||||
placeholder: (context, url) => Icon(Icons.person, color: iconColor, size: 24 * context.sf),
|
|
||||||
errorWidget: (context, url, error) => Icon(Icons.person, color: iconColor, size: 24 * context.sf),
|
|
||||||
)
|
|
||||||
: Icon(Icons.person, color: iconColor, size: 24 * context.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(width: 12 * context.sf),
|
|
||||||
Expanded(
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
if (!isCoach && person.number != null && person.number!.isNotEmpty) ...[
|
|
||||||
Container(
|
|
||||||
padding: EdgeInsets.symmetric(horizontal: 8 * context.sf, vertical: 4 * context.sf),
|
|
||||||
decoration: BoxDecoration(color: AppTheme.primaryRed.withOpacity(0.1), borderRadius: BorderRadius.circular(6 * context.sf)),
|
|
||||||
child: Text(person.number!, style: TextStyle(color: AppTheme.primaryRed, fontWeight: FontWeight.bold, fontSize: 14 * context.sf)),
|
|
||||||
),
|
|
||||||
SizedBox(width: 10 * context.sf),
|
|
||||||
],
|
|
||||||
Expanded(
|
|
||||||
child: Text(person.name, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16 * context.sf, color: Theme.of(context).colorScheme.onSurface), overflow: TextOverflow.ellipsis)
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Row(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
IconButton(icon: Icon(Icons.edit_outlined, color: Colors.blue, size: 22 * context.sf), onPressed: onEdit, padding: EdgeInsets.zero, constraints: const BoxConstraints()),
|
|
||||||
SizedBox(width: 16 * context.sf),
|
|
||||||
IconButton(icon: Icon(Icons.delete_outline, color: AppTheme.primaryRed, size: 22 * context.sf), onPressed: onDelete, padding: EdgeInsets.zero, constraints: const BoxConstraints()),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ==========================================
|
const SizedBox(width: 5), // Espaço
|
||||||
// WIDGET NOVO: SKELETON LOADING (SHIMMER)
|
|
||||||
// ==========================================
|
|
||||||
class SkeletonLoadingStats extends StatelessWidget {
|
|
||||||
const SkeletonLoadingStats({super.key});
|
|
||||||
|
|
||||||
@override
|
IconButton(
|
||||||
Widget build(BuildContext context) {
|
icon: const Icon(Icons.edit_outlined, color: Colors.blue),
|
||||||
final isDark = Theme.of(context).brightness == Brightness.dark;
|
onPressed: onEdit,
|
||||||
final baseColor = isDark ? Colors.grey[800]! : Colors.grey[300]!;
|
),
|
||||||
final highlightColor = isDark ? Colors.grey[700]! : Colors.grey[100]!;
|
IconButton(
|
||||||
|
icon: const Icon(Icons.delete_outline, color: Colors.red),
|
||||||
return Shimmer.fromColors(
|
onPressed: onDelete,
|
||||||
baseColor: baseColor,
|
),
|
||||||
highlightColor: highlightColor,
|
|
||||||
child: SingleChildScrollView(
|
|
||||||
padding: EdgeInsets.all(16.0 * context.sf),
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Container(height: 80 * context.sf, width: double.infinity, decoration: BoxDecoration(color: Colors.white, borderRadius: BorderRadius.circular(20 * context.sf))),
|
|
||||||
SizedBox(height: 30 * context.sf),
|
|
||||||
Container(height: 20 * context.sf, width: 150 * context.sf, color: Colors.white),
|
|
||||||
SizedBox(height: 10 * context.sf),
|
|
||||||
for (int i = 0; i < 3; i++) ...[
|
|
||||||
Container(
|
|
||||||
height: 60 * context.sf, width: double.infinity,
|
|
||||||
margin: EdgeInsets.only(top: 12 * context.sf),
|
|
||||||
decoration: BoxDecoration(color: Colors.white, borderRadius: BorderRadius.circular(15 * context.sf)),
|
|
||||||
),
|
|
||||||
]
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -272,8 +164,10 @@ class SkeletonLoadingStats extends StatelessWidget {
|
|||||||
// ==========================================
|
// ==========================================
|
||||||
// 2. PÁGINA PRINCIPAL
|
// 2. PÁGINA PRINCIPAL
|
||||||
// ==========================================
|
// ==========================================
|
||||||
|
|
||||||
class TeamStatsPage extends StatefulWidget {
|
class TeamStatsPage extends StatefulWidget {
|
||||||
final Team team;
|
final Team team;
|
||||||
|
|
||||||
const TeamStatsPage({super.key, required this.team});
|
const TeamStatsPage({super.key, required this.team});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -283,103 +177,71 @@ class TeamStatsPage extends StatefulWidget {
|
|||||||
class _TeamStatsPageState extends State<TeamStatsPage> {
|
class _TeamStatsPageState extends State<TeamStatsPage> {
|
||||||
final StatsController _controller = StatsController();
|
final StatsController _controller = StatsController();
|
||||||
|
|
||||||
late String _teamImageUrl;
|
|
||||||
bool _isUploadingTeamPhoto = false;
|
|
||||||
bool _isPickerActive = false;
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
_teamImageUrl = widget.team.imageUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _updateTeamPhoto() async {
|
|
||||||
if (_isPickerActive) return;
|
|
||||||
setState(() => _isPickerActive = true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
final File? croppedFile = await _controller.pickAndCropImage(context);
|
|
||||||
if (croppedFile == null) return;
|
|
||||||
|
|
||||||
setState(() => _isUploadingTeamPhoto = true);
|
|
||||||
|
|
||||||
final fileName = 'team_${widget.team.id}_${DateTime.now().millisecondsSinceEpoch}.png';
|
|
||||||
final supabase = Supabase.instance.client;
|
|
||||||
|
|
||||||
await supabase.storage.from('avatars').upload(fileName, croppedFile, fileOptions: const FileOptions(upsert: true));
|
|
||||||
final publicUrl = supabase.storage.from('avatars').getPublicUrl(fileName);
|
|
||||||
|
|
||||||
await supabase.from('teams').update({'image_url': publicUrl}).eq('id', widget.team.id);
|
|
||||||
|
|
||||||
if (_teamImageUrl.isNotEmpty && _teamImageUrl.startsWith('http')) {
|
|
||||||
final oldPath = _controller.extractPathFromUrl(_teamImageUrl, 'avatars');
|
|
||||||
if (oldPath != null) await supabase.storage.from('avatars').remove([oldPath]);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mounted) setState(() => _teamImageUrl = publicUrl);
|
|
||||||
} catch (e) {
|
|
||||||
if (mounted) ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text("Erro: $e"), backgroundColor: AppTheme.primaryRed));
|
|
||||||
} finally {
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {
|
|
||||||
_isUploadingTeamPhoto = false;
|
|
||||||
_isPickerActive = false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
backgroundColor: const Color(0xFFF5F7FA),
|
||||||
body: Column(
|
body: Column(
|
||||||
children: [
|
children: [
|
||||||
StatsHeader(team: widget.team, currentImageUrl: _teamImageUrl, onEditPhoto: _updateTeamPhoto, isUploading: _isUploadingTeamPhoto),
|
// Cabeçalho
|
||||||
|
StatsHeader(team: widget.team),
|
||||||
|
|
||||||
Expanded(
|
Expanded(
|
||||||
child: StreamBuilder<List<Person>>(
|
child: StreamBuilder<List<Person>>(
|
||||||
stream: _controller.getMembers(widget.team.id),
|
stream: _controller.getMembers(widget.team.id),
|
||||||
builder: (context, snapshot) {
|
builder: (context, snapshot) {
|
||||||
if (snapshot.connectionState == ConnectionState.waiting) {
|
if (snapshot.connectionState == ConnectionState.waiting) {
|
||||||
return const SkeletonLoadingStats();
|
return const Center(child: CircularProgressIndicator());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (snapshot.hasError) return Center(child: Text("Erro ao carregar: ${snapshot.error}", style: TextStyle(color: Theme.of(context).colorScheme.onSurface)));
|
if (snapshot.hasError) {
|
||||||
|
return Center(child: Text("Erro ao carregar: ${snapshot.error}"));
|
||||||
|
}
|
||||||
|
|
||||||
final members = snapshot.data ?? [];
|
final members = snapshot.data ?? [];
|
||||||
|
|
||||||
final coaches = members.where((m) => m.type == 'Treinador').toList()..sort((a, b) => a.name.compareTo(b.name));
|
final coaches = members.where((m) => m.type == 'Treinador').toList();
|
||||||
final players = members.where((m) => m.type == 'Jogador').toList()..sort((a, b) {
|
final players = members.where((m) => m.type == 'Jogador').toList();
|
||||||
int numA = int.tryParse(a.number ?? '999') ?? 999;
|
|
||||||
int numB = int.tryParse(b.number ?? '999') ?? 999;
|
|
||||||
return numA.compareTo(numB);
|
|
||||||
});
|
|
||||||
|
|
||||||
return RefreshIndicator(
|
return RefreshIndicator(
|
||||||
color: AppTheme.primaryRed,
|
|
||||||
onRefresh: () async => setState(() {}),
|
onRefresh: () async => setState(() {}),
|
||||||
child: SingleChildScrollView(
|
child: SingleChildScrollView(
|
||||||
physics: const AlwaysScrollableScrollPhysics(),
|
physics: const AlwaysScrollableScrollPhysics(),
|
||||||
padding: EdgeInsets.all(16.0 * context.sf),
|
padding: const EdgeInsets.all(16.0),
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
StatsSummaryCard(total: members.length),
|
StatsSummaryCard(total: members.length),
|
||||||
SizedBox(height: 30 * context.sf),
|
const SizedBox(height: 30),
|
||||||
|
|
||||||
|
// TREINADORES
|
||||||
if (coaches.isNotEmpty) ...[
|
if (coaches.isNotEmpty) ...[
|
||||||
const StatsSectionTitle(title: "Treinadores"),
|
const StatsSectionTitle(title: "Treinadores"),
|
||||||
...coaches.map((c) => PersonCard(person: c, isCoach: true, onEdit: () => _controller.showEditPersonDialog(context, widget.team.id, c), onDelete: () => _confirmDelete(context, c))),
|
...coaches.map((c) => PersonCard(
|
||||||
SizedBox(height: 30 * context.sf),
|
person: c,
|
||||||
|
isCoach: true,
|
||||||
|
|
||||||
|
onEdit: () => _controller.showEditPersonDialog(context, widget.team.id, c),
|
||||||
|
onDelete: () => _confirmDelete(context, c),
|
||||||
|
)),
|
||||||
|
const SizedBox(height: 30),
|
||||||
],
|
],
|
||||||
|
|
||||||
|
// JOGADORES
|
||||||
const StatsSectionTitle(title: "Jogadores"),
|
const StatsSectionTitle(title: "Jogadores"),
|
||||||
if (players.isEmpty)
|
if (players.isEmpty)
|
||||||
Padding(padding: EdgeInsets.only(top: 20 * context.sf), child: Text("Nenhum jogador nesta equipa.", style: TextStyle(color: Colors.grey, fontSize: 16 * context.sf)))
|
const Padding(
|
||||||
|
padding: EdgeInsets.only(top: 20),
|
||||||
|
child: Text("Nenhum jogador nesta equipa.", style: TextStyle(color: Colors.grey, fontSize: 16)),
|
||||||
|
)
|
||||||
else
|
else
|
||||||
...players.map((p) => PersonCard(person: p, isCoach: false, onEdit: () => _controller.showEditPersonDialog(context, widget.team.id, p), onDelete: () => _confirmDelete(context, p))),
|
...players.map((p) => PersonCard(
|
||||||
SizedBox(height: 80 * context.sf),
|
person: p,
|
||||||
|
isCoach: false,
|
||||||
|
onEdit: () => _controller.showEditPersonDialog(context, widget.team.id, p),
|
||||||
|
onDelete: () => _confirmDelete(context, p),
|
||||||
|
)),
|
||||||
|
const SizedBox(height: 80),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -392,102 +254,63 @@ class _TeamStatsPageState extends State<TeamStatsPage> {
|
|||||||
floatingActionButton: FloatingActionButton(
|
floatingActionButton: FloatingActionButton(
|
||||||
heroTag: 'fab_team_${widget.team.id}',
|
heroTag: 'fab_team_${widget.team.id}',
|
||||||
onPressed: () => _controller.showAddPersonDialog(context, widget.team.id),
|
onPressed: () => _controller.showAddPersonDialog(context, widget.team.id),
|
||||||
backgroundColor: AppTheme.successGreen,
|
backgroundColor: const Color(0xFF00C853),
|
||||||
child: Icon(Icons.add, color: Colors.white, size: 24 * context.sf),
|
child: const Icon(Icons.add, color: Colors.white),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _confirmDelete(BuildContext context, Person person) {
|
void _confirmDelete(BuildContext context, Person person) {
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (ctx) => AlertDialog(
|
builder: (ctx) => AlertDialog(
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
title: const Text("Eliminar Membro?"),
|
||||||
title: Text("Eliminar Membro?", style: TextStyle(color: Theme.of(context).colorScheme.onSurface)),
|
content: Text("Tens a certeza que queres remover ${person.name}?"),
|
||||||
content: Text("Tens a certeza que queres remover ${person.name}?", style: TextStyle(color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(onPressed: () => Navigator.pop(ctx), child: const Text("Cancelar", style: TextStyle(color: Colors.grey))),
|
TextButton(onPressed: () => Navigator.pop(ctx), child: const Text("Cancelar")),
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () {
|
onPressed: () async {
|
||||||
Navigator.pop(ctx);
|
await _controller.deletePerson(person.id);
|
||||||
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text("A remover ${person.name}..."), duration: const Duration(seconds: 1)));
|
if (ctx.mounted) Navigator.pop(ctx);
|
||||||
|
|
||||||
_controller.deletePerson(person).catchError((e) {
|
|
||||||
if (context.mounted) ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text("Erro: $e"), backgroundColor: AppTheme.primaryRed));
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
child: const Text("Eliminar", style: TextStyle(color: AppTheme.primaryRed)),
|
child: const Text("Eliminar", style: TextStyle(color: Colors.red)),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ==========================================
|
// ==========================================
|
||||||
// 3. CONTROLLER
|
// 3. CONTROLLER
|
||||||
// ==========================================
|
// ==========================================
|
||||||
|
|
||||||
class StatsController {
|
class StatsController {
|
||||||
final _supabase = Supabase.instance.client;
|
final _supabase = Supabase.instance.client;
|
||||||
|
|
||||||
Stream<List<Person>> getMembers(String teamId) {
|
Stream<List<Person>> getMembers(String teamId) {
|
||||||
return _supabase.from('members').stream(primaryKey: ['id']).eq('team_id', teamId).map((data) => data.map((json) => Person.fromMap(json)).toList());
|
return _supabase
|
||||||
|
.from('members')
|
||||||
|
.stream(primaryKey: ['id'])
|
||||||
|
.eq('team_id', teamId)
|
||||||
|
.order('name', ascending: true)
|
||||||
|
.map((data) => data.map((json) => Person.fromMap(json)).toList());
|
||||||
}
|
}
|
||||||
|
|
||||||
String? extractPathFromUrl(String url, String bucket) {
|
Future<void> deletePerson(String personId) async {
|
||||||
if (url.isEmpty) return null;
|
|
||||||
final parts = url.split('/$bucket/');
|
|
||||||
if (parts.length > 1) return parts.last;
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> deletePerson(Person person) async {
|
|
||||||
try {
|
try {
|
||||||
await _supabase.from('members').delete().eq('id', person.id);
|
await _supabase.from('members').delete().eq('id', personId);
|
||||||
|
|
||||||
if (person.imageUrl != null && person.imageUrl!.isNotEmpty) {
|
|
||||||
final path = extractPathFromUrl(person.imageUrl!, 'avatars');
|
|
||||||
if (path != null) await _supabase.storage.from('avatars').remove([path]);
|
|
||||||
}
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
debugPrint("Erro ao eliminar: $e");
|
debugPrint("Erro ao eliminar: $e");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void showAddPersonDialog(BuildContext context, String teamId) { _showForm(context, teamId: teamId); }
|
void showAddPersonDialog(BuildContext context, String teamId) {
|
||||||
void showEditPersonDialog(BuildContext context, String teamId, Person person) { _showForm(context, teamId: teamId, person: person); }
|
_showForm(context, teamId: teamId);
|
||||||
|
}
|
||||||
|
|
||||||
Future<File?> pickAndCropImage(BuildContext context) async {
|
void showEditPersonDialog(BuildContext context, String teamId, Person person) {
|
||||||
final picker = ImagePicker();
|
_showForm(context, teamId: teamId, person: person);
|
||||||
final pickedFile = await picker.pickImage(source: ImageSource.gallery);
|
|
||||||
|
|
||||||
if (pickedFile == null) return null;
|
|
||||||
|
|
||||||
CroppedFile? croppedFile = await ImageCropper().cropImage(
|
|
||||||
sourcePath: pickedFile.path,
|
|
||||||
aspectRatio: const CropAspectRatio(ratioX: 1, ratioY: 1),
|
|
||||||
uiSettings: [
|
|
||||||
AndroidUiSettings(
|
|
||||||
toolbarTitle: 'Recortar Foto',
|
|
||||||
toolbarColor: AppTheme.primaryRed,
|
|
||||||
toolbarWidgetColor: Colors.white,
|
|
||||||
initAspectRatio: CropAspectRatioPreset.square,
|
|
||||||
lockAspectRatio: true,
|
|
||||||
hideBottomControls: true,
|
|
||||||
),
|
|
||||||
IOSUiSettings(
|
|
||||||
title: 'Recortar Foto',
|
|
||||||
aspectRatioLockEnabled: true,
|
|
||||||
resetButtonHidden: true,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
|
|
||||||
if (croppedFile != null) {
|
|
||||||
return File(croppedFile.path);
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void _showForm(BuildContext context, {required String teamId, Person? person}) {
|
void _showForm(BuildContext context, {required String teamId, Person? person}) {
|
||||||
@@ -496,101 +319,37 @@ class StatsController {
|
|||||||
final numCtrl = TextEditingController(text: person?.number ?? '');
|
final numCtrl = TextEditingController(text: person?.number ?? '');
|
||||||
String selectedType = person?.type ?? 'Jogador';
|
String selectedType = person?.type ?? 'Jogador';
|
||||||
|
|
||||||
File? selectedImage;
|
|
||||||
bool isUploading = false;
|
|
||||||
bool isPickerActive = false;
|
|
||||||
String? currentImageUrl = isEdit ? person.imageUrl : null;
|
|
||||||
|
|
||||||
String? nameError;
|
|
||||||
String? numError;
|
|
||||||
|
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (ctx) => StatefulBuilder(
|
builder: (ctx) => StatefulBuilder(
|
||||||
builder: (ctx, setState) => AlertDialog(
|
builder: (ctx, setState) => AlertDialog(
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15)),
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * context.sf)),
|
title: Text(isEdit ? "Editar Membro" : "Novo Membro"),
|
||||||
title: Text(isEdit ? "Editar Membro" : "Novo Membro", style: TextStyle(color: Theme.of(context).colorScheme.onSurface)),
|
|
||||||
content: SingleChildScrollView(
|
content: SingleChildScrollView(
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
GestureDetector(
|
|
||||||
onTap: () async {
|
|
||||||
if (isPickerActive) return;
|
|
||||||
setState(() => isPickerActive = true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
final File? croppedFile = await pickAndCropImage(context);
|
|
||||||
if (croppedFile != null) {
|
|
||||||
setState(() => selectedImage = croppedFile);
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
setState(() => isPickerActive = false);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
child: Stack(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
children: [
|
|
||||||
// 👇 PREVIEW DA FOTO NO POPUP SEM LAG 👇
|
|
||||||
ClipOval(
|
|
||||||
child: Container(
|
|
||||||
width: 80 * context.sf,
|
|
||||||
height: 80 * context.sf,
|
|
||||||
color: Theme.of(context).colorScheme.onSurface.withOpacity(0.05),
|
|
||||||
child: selectedImage != null
|
|
||||||
? Image.file(selectedImage!, fit: BoxFit.cover)
|
|
||||||
: (currentImageUrl != null && currentImageUrl!.isNotEmpty)
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: currentImageUrl!,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
fadeInDuration: Duration.zero,
|
|
||||||
placeholder: (context, url) => Icon(Icons.add_a_photo, size: 30 * context.sf, color: Colors.grey),
|
|
||||||
errorWidget: (context, url, error) => Icon(Icons.add_a_photo, size: 30 * context.sf, color: Colors.grey),
|
|
||||||
)
|
|
||||||
: Icon(Icons.add_a_photo, size: 30 * context.sf, color: Colors.grey),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
bottom: 0, right: 0,
|
|
||||||
child: Container(
|
|
||||||
padding: EdgeInsets.all(6 * context.sf),
|
|
||||||
decoration: BoxDecoration(color: AppTheme.primaryRed, shape: BoxShape.circle, border: Border.all(color: Colors.white, width: 2)),
|
|
||||||
child: Icon(Icons.edit, color: Colors.white, size: 14 * context.sf),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
SizedBox(height: 20 * context.sf),
|
|
||||||
|
|
||||||
TextField(
|
TextField(
|
||||||
controller: nameCtrl,
|
controller: nameCtrl,
|
||||||
style: TextStyle(color: Theme.of(context).colorScheme.onSurface),
|
decoration: const InputDecoration(labelText: "Nome Completo"),
|
||||||
decoration: InputDecoration(
|
|
||||||
labelText: "Nome Completo",
|
|
||||||
errorText: nameError,
|
|
||||||
),
|
|
||||||
textCapitalization: TextCapitalization.words,
|
textCapitalization: TextCapitalization.words,
|
||||||
),
|
),
|
||||||
SizedBox(height: 15 * context.sf),
|
const SizedBox(height: 15),
|
||||||
DropdownButtonFormField<String>(
|
DropdownButtonFormField<String>(
|
||||||
value: selectedType,
|
value: selectedType,
|
||||||
dropdownColor: Theme.of(context).colorScheme.surface,
|
|
||||||
style: TextStyle(color: Theme.of(context).colorScheme.onSurface, fontSize: 16 * context.sf),
|
|
||||||
decoration: const InputDecoration(labelText: "Função"),
|
decoration: const InputDecoration(labelText: "Função"),
|
||||||
items: ["Jogador", "Treinador"].map((e) => DropdownMenuItem(value: e, child: Text(e))).toList(),
|
items: ["Jogador", "Treinador"]
|
||||||
onChanged: (v) { if (v != null) setState(() => selectedType = v); },
|
.map((e) => DropdownMenuItem(value: e, child: Text(e)))
|
||||||
|
.toList(),
|
||||||
|
onChanged: (v) {
|
||||||
|
if (v != null) setState(() => selectedType = v);
|
||||||
|
},
|
||||||
),
|
),
|
||||||
if (selectedType == "Jogador") ...[
|
if (selectedType == "Jogador") ...[
|
||||||
SizedBox(height: 15 * context.sf),
|
const SizedBox(height: 15),
|
||||||
TextField(
|
TextField(
|
||||||
controller: numCtrl,
|
controller: numCtrl,
|
||||||
style: TextStyle(color: Theme.of(context).colorScheme.onSurface),
|
decoration: const InputDecoration(labelText: "Número da Camisola"),
|
||||||
decoration: InputDecoration(
|
|
||||||
labelText: "Número da Camisola",
|
|
||||||
errorText: numError,
|
|
||||||
),
|
|
||||||
keyboardType: TextInputType.number,
|
keyboardType: TextInputType.number,
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
@@ -598,45 +357,28 @@ class StatsController {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(onPressed: () => Navigator.pop(ctx), child: const Text("Cancelar", style: TextStyle(color: Colors.grey))),
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(ctx),
|
||||||
|
child: const Text("Cancelar")
|
||||||
|
),
|
||||||
ElevatedButton(
|
ElevatedButton(
|
||||||
style: ElevatedButton.styleFrom(backgroundColor: AppTheme.successGreen, foregroundColor: Colors.white, shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8 * context.sf))),
|
style: ElevatedButton.styleFrom(
|
||||||
onPressed: isUploading ? null : () async {
|
backgroundColor: const Color(0xFF00C853),
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8))
|
||||||
|
),
|
||||||
|
onPressed: () async {
|
||||||
|
if (nameCtrl.text.trim().isEmpty) return;
|
||||||
|
|
||||||
setState(() {
|
String? numeroFinal = (selectedType == "Treinador")
|
||||||
nameError = null;
|
? null
|
||||||
numError = null;
|
: (numCtrl.text.trim().isEmpty ? null : numCtrl.text.trim());
|
||||||
});
|
|
||||||
|
|
||||||
if (nameCtrl.text.trim().isEmpty) {
|
|
||||||
setState(() => nameError = "O nome é obrigatório");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setState(() => isUploading = true);
|
|
||||||
|
|
||||||
String? numeroFinal = (selectedType == "Treinador") ? null : (numCtrl.text.trim().isEmpty ? null : numCtrl.text.trim());
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
String? finalImageUrl = currentImageUrl;
|
|
||||||
|
|
||||||
if (selectedImage != null) {
|
|
||||||
final fileName = 'person_${DateTime.now().millisecondsSinceEpoch}.png';
|
|
||||||
await _supabase.storage.from('avatars').upload(fileName, selectedImage!, fileOptions: const FileOptions(upsert: true));
|
|
||||||
finalImageUrl = _supabase.storage.from('avatars').getPublicUrl(fileName);
|
|
||||||
|
|
||||||
if (currentImageUrl != null && currentImageUrl!.isNotEmpty) {
|
|
||||||
final oldPath = extractPathFromUrl(currentImageUrl!, 'avatars');
|
|
||||||
if (oldPath != null) await _supabase.storage.from('avatars').remove([oldPath]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isEdit) {
|
if (isEdit) {
|
||||||
await _supabase.from('members').update({
|
await _supabase.from('members').update({
|
||||||
'name': nameCtrl.text.trim(),
|
'name': nameCtrl.text.trim(),
|
||||||
'type': selectedType,
|
'type': selectedType,
|
||||||
'number': numeroFinal,
|
'number': numeroFinal,
|
||||||
'image_url': finalImageUrl,
|
|
||||||
}).eq('id', person.id);
|
}).eq('id', person.id);
|
||||||
} else {
|
} else {
|
||||||
await _supabase.from('members').insert({
|
await _supabase.from('members').insert({
|
||||||
@@ -644,24 +386,23 @@ class StatsController {
|
|||||||
'name': nameCtrl.text.trim(),
|
'name': nameCtrl.text.trim(),
|
||||||
'type': selectedType,
|
'type': selectedType,
|
||||||
'number': numeroFinal,
|
'number': numeroFinal,
|
||||||
'image_url': finalImageUrl,
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (ctx.mounted) Navigator.pop(ctx);
|
if (ctx.mounted) Navigator.pop(ctx);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
setState(() {
|
debugPrint("Erro Supabase: $e");
|
||||||
isUploading = false;
|
if (ctx.mounted) {
|
||||||
if (e is PostgrestException && e.code == '23505') {
|
String errorMsg = "Erro ao guardar: $e";
|
||||||
numError = "Este número já está em uso!";
|
if (e.toString().contains('unique')) {
|
||||||
} else if (e.toString().toLowerCase().contains('unique') || e.toString().toLowerCase().contains('duplicate')) {
|
errorMsg = "Já existe um membro com este numero na equipa.";
|
||||||
numError = "Este número já está em uso!";
|
|
||||||
} else {
|
|
||||||
nameError = "Erro ao guardar. Tente novamente.";
|
|
||||||
}
|
}
|
||||||
});
|
ScaffoldMessenger.of(ctx).showSnackBar(
|
||||||
|
SnackBar(content: Text(errorMsg), backgroundColor: Colors.red)
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
child: isUploading ? SizedBox(width: 16 * context.sf, height: 16 * context.sf, child: const CircularProgressIndicator(color: Colors.white, strokeWidth: 2)) : const Text("Guardar"),
|
child: const Text("Guardar", style: TextStyle(color: Colors.white)),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -1,31 +1,15 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'dart:math' as math;
|
import 'dart:math' as math;
|
||||||
|
|
||||||
|
// Esta extensão adiciona o superpoder "sf" ao BuildContext
|
||||||
extension SizeExtension on BuildContext {
|
extension SizeExtension on BuildContext {
|
||||||
|
|
||||||
double get sf {
|
double get sf {
|
||||||
final Size size = MediaQuery.of(this).size;
|
final double wScreen = MediaQuery.of(this).size.width;
|
||||||
|
final double hScreen = MediaQuery.of(this).size.height;
|
||||||
|
|
||||||
// 1. Definimos os valores base do design (geralmente feitos no Figma/Adobe XD)
|
// Calcula e devolve a escala na hora!
|
||||||
const double baseWidth = 375;
|
return math.min(wScreen, hScreen) / 400;
|
||||||
const double baseHeight = 812;
|
|
||||||
|
|
||||||
// 2. Calculamos o rácio de largura e altura
|
|
||||||
double scaleW = size.width / baseWidth;
|
|
||||||
double scaleH = size.height / baseHeight;
|
|
||||||
|
|
||||||
// 3. Usamos a média ou o menor valor para manter a proporção
|
|
||||||
// O 'min' evita que o texto estique demasiado se o ecrã for muito alto ou largo
|
|
||||||
double scale = math.min(scaleW, scaleH);
|
|
||||||
|
|
||||||
// 4. Segurança (Clamping): Não deixa as coisas ficarem minúsculas
|
|
||||||
// nem exageradamente grandes em tablets.
|
|
||||||
return scale.clamp(0.8, 1.4);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Atalhos úteis para facilitar o código
|
|
||||||
double get screenWidth => MediaQuery.of(this).size.width;
|
|
||||||
double get screenHeight => MediaQuery.of(this).size.height;
|
|
||||||
|
|
||||||
// Verifica se é Tablet (opcional)
|
|
||||||
bool get isTablet => screenWidth > 600;
|
|
||||||
}
|
}
|
||||||
@@ -1,83 +1,104 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/pages/PlacarPage.dart';
|
import 'package:playmaker/pages/PlacarPage.dart';
|
||||||
import 'package:playmaker/classe/theme.dart'; // 👇 IMPORT DO TEMA!
|
|
||||||
import '../controllers/team_controller.dart';
|
import '../controllers/team_controller.dart';
|
||||||
import '../controllers/game_controller.dart';
|
import '../controllers/game_controller.dart';
|
||||||
|
|
||||||
|
// --- CARD DE EXIBIÇÃO DO JOGO ---
|
||||||
class GameResultCard extends StatelessWidget {
|
class GameResultCard extends StatelessWidget {
|
||||||
final String gameId, myTeam, opponentTeam, myScore, opponentScore, status, season;
|
final String gameId;
|
||||||
final String? myTeamLogo, opponentTeamLogo;
|
final String myTeam, opponentTeam, myScore, opponentScore, status, season;
|
||||||
final double sf;
|
final String? myTeamLogo;
|
||||||
|
final String? opponentTeamLogo;
|
||||||
|
final double sf; // NOVA VARIÁVEL DE ESCALA
|
||||||
|
|
||||||
const GameResultCard({
|
const GameResultCard({
|
||||||
super.key, required this.gameId, required this.myTeam, required this.opponentTeam,
|
super.key,
|
||||||
required this.myScore, required this.opponentScore, required this.status, required this.season,
|
required this.gameId,
|
||||||
this.myTeamLogo, this.opponentTeamLogo, required this.sf,
|
required this.myTeam,
|
||||||
|
required this.opponentTeam,
|
||||||
|
required this.myScore,
|
||||||
|
required this.opponentScore,
|
||||||
|
required this.status,
|
||||||
|
required this.season,
|
||||||
|
this.myTeamLogo,
|
||||||
|
this.opponentTeamLogo,
|
||||||
|
required this.sf, // OBRIGATÓRIO RECEBER A ESCALA
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
// 👇 Puxa as cores de fundo dependendo do Modo (Claro/Escuro)
|
|
||||||
final bgColor = Theme.of(context).colorScheme.surface;
|
|
||||||
final textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
|
|
||||||
return Container(
|
return Container(
|
||||||
margin: EdgeInsets.only(bottom: 16 * sf),
|
margin: EdgeInsets.only(bottom: 16 * sf),
|
||||||
padding: EdgeInsets.all(16 * sf),
|
padding: EdgeInsets.all(16 * sf),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: bgColor, // Usa a cor do tema
|
color: Colors.white,
|
||||||
borderRadius: BorderRadius.circular(20 * sf),
|
borderRadius: BorderRadius.circular(20 * sf),
|
||||||
boxShadow: [BoxShadow(color: Colors.black12, blurRadius: 10 * sf)],
|
boxShadow: [BoxShadow(color: Colors.black12, blurRadius: 10 * sf)],
|
||||||
),
|
),
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Expanded(child: _buildTeamInfo(myTeam, AppTheme.primaryRed, myTeamLogo, sf, textColor)), // Usa o primaryRed
|
Expanded(child: _buildTeamInfo(myTeam, const Color(0xFFE74C3C), myTeamLogo, sf)),
|
||||||
_buildScoreCenter(context, gameId, sf),
|
_buildScoreCenter(context, gameId, sf),
|
||||||
Expanded(child: _buildTeamInfo(opponentTeam, textColor, opponentTeamLogo, sf, textColor)),
|
Expanded(child: _buildTeamInfo(opponentTeam, Colors.black87, opponentTeamLogo, sf)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildTeamInfo(String name, Color color, String? logoUrl, double sf, Color textColor) {
|
Widget _buildTeamInfo(String name, Color color, String? logoUrl, double sf) {
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
CircleAvatar(
|
CircleAvatar(
|
||||||
radius: 24 * sf,
|
radius: 24 * sf, // Ajuste do tamanho do logo
|
||||||
backgroundColor: color,
|
backgroundColor: color,
|
||||||
backgroundImage: (logoUrl != null && logoUrl.isNotEmpty) ? NetworkImage(logoUrl) : null,
|
backgroundImage: (logoUrl != null && logoUrl.isNotEmpty)
|
||||||
child: (logoUrl == null || logoUrl.isEmpty) ? Icon(Icons.shield, color: Colors.white, size: 24 * sf) : null,
|
? NetworkImage(logoUrl)
|
||||||
|
: null,
|
||||||
|
child: (logoUrl == null || logoUrl.isEmpty)
|
||||||
|
? Icon(Icons.shield, color: Colors.white, size: 24 * sf)
|
||||||
|
: null,
|
||||||
),
|
),
|
||||||
SizedBox(height: 6 * sf),
|
SizedBox(height: 6 * sf),
|
||||||
Text(name,
|
Text(name,
|
||||||
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 13 * sf, color: textColor), // Adapta à noite/dia
|
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 13 * sf),
|
||||||
textAlign: TextAlign.center, overflow: TextOverflow.ellipsis, maxLines: 2,
|
textAlign: TextAlign.center,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
maxLines: 2, // Permite 2 linhas para nomes compridos não cortarem
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildScoreCenter(BuildContext context, String id, double sf) {
|
Widget _buildScoreCenter(BuildContext context, String id, double sf) {
|
||||||
final textColor = Theme.of(context).colorScheme.onSurface;
|
|
||||||
|
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
Row(
|
Row(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: [
|
children: [
|
||||||
_scoreBox(myScore, AppTheme.successGreen, sf), // Verde do tema
|
_scoreBox(myScore, Colors.green, sf),
|
||||||
Text(" : ", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 22 * sf, color: textColor)),
|
Text(" : ", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 22 * sf)),
|
||||||
_scoreBox(opponentScore, Colors.grey, sf),
|
_scoreBox(opponentScore, Colors.grey, sf),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
SizedBox(height: 10 * sf),
|
SizedBox(height: 10 * sf),
|
||||||
TextButton.icon(
|
TextButton.icon(
|
||||||
onPressed: () => Navigator.push(context, MaterialPageRoute(builder: (context) => PlacarPage(gameId: id, myTeam: myTeam, opponentTeam: opponentTeam))),
|
onPressed: () {
|
||||||
icon: Icon(Icons.play_circle_fill, size: 18 * sf, color: AppTheme.primaryRed),
|
Navigator.push(
|
||||||
label: Text("RETORNAR", style: TextStyle(fontSize: 11 * sf, color: AppTheme.primaryRed, fontWeight: FontWeight.bold)),
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => PlacarPage(
|
||||||
|
gameId: id,
|
||||||
|
myTeam: myTeam,
|
||||||
|
opponentTeam: opponentTeam,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
icon: Icon(Icons.play_circle_fill, size: 18 * sf, color: const Color(0xFFE74C3C)),
|
||||||
|
label: Text("RETORNAR", style: TextStyle(fontSize: 11 * sf, color: const Color(0xFFE74C3C), fontWeight: FontWeight.bold)),
|
||||||
style: TextButton.styleFrom(
|
style: TextButton.styleFrom(
|
||||||
backgroundColor: AppTheme.primaryRed.withOpacity(0.1),
|
backgroundColor: const Color(0xFFE74C3C).withOpacity(0.1),
|
||||||
padding: EdgeInsets.symmetric(horizontal: 14 * sf, vertical: 8 * sf),
|
padding: EdgeInsets.symmetric(horizontal: 14 * sf, vertical: 8 * sf),
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * sf)),
|
||||||
visualDensity: VisualDensity.compact,
|
visualDensity: VisualDensity.compact,
|
||||||
@@ -95,3 +116,203 @@ class GameResultCard extends StatelessWidget {
|
|||||||
child: Text(pts, style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 16 * sf)),
|
child: Text(pts, style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 16 * sf)),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// --- POPUP DE CRIAÇÃO ---
|
||||||
|
class CreateGameDialogManual extends StatefulWidget {
|
||||||
|
final TeamController teamController;
|
||||||
|
final GameController gameController;
|
||||||
|
final double sf; // NOVA VARIÁVEL DE ESCALA
|
||||||
|
|
||||||
|
const CreateGameDialogManual({
|
||||||
|
super.key,
|
||||||
|
required this.teamController,
|
||||||
|
required this.gameController,
|
||||||
|
required this.sf,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<CreateGameDialogManual> createState() => _CreateGameDialogManualState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CreateGameDialogManualState extends State<CreateGameDialogManual> {
|
||||||
|
late TextEditingController _seasonController;
|
||||||
|
final TextEditingController _myTeamController = TextEditingController();
|
||||||
|
final TextEditingController _opponentController = TextEditingController();
|
||||||
|
|
||||||
|
bool _isLoading = false;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_seasonController = TextEditingController(text: _calculateSeason());
|
||||||
|
}
|
||||||
|
|
||||||
|
String _calculateSeason() {
|
||||||
|
final now = DateTime.now();
|
||||||
|
return now.month >= 7 ? "${now.year}/${(now.year + 1).toString().substring(2)}" : "${now.year - 1}/${now.year.toString().substring(2)}";
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * widget.sf)),
|
||||||
|
title: Text('Configurar Partida', style: TextStyle(fontWeight: FontWeight.bold, fontSize: 18 * widget.sf)),
|
||||||
|
content: SingleChildScrollView(
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
TextField(
|
||||||
|
controller: _seasonController,
|
||||||
|
style: TextStyle(fontSize: 14 * widget.sf),
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: 'Temporada',
|
||||||
|
labelStyle: TextStyle(fontSize: 14 * widget.sf),
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
prefixIcon: Icon(Icons.calendar_today, size: 20 * widget.sf)
|
||||||
|
),
|
||||||
|
),
|
||||||
|
SizedBox(height: 15 * widget.sf),
|
||||||
|
|
||||||
|
_buildSearch(label: "Minha Equipa", controller: _myTeamController, sf: widget.sf),
|
||||||
|
|
||||||
|
Padding(
|
||||||
|
padding: EdgeInsets.symmetric(vertical: 10 * widget.sf),
|
||||||
|
child: Text("VS", style: TextStyle(fontWeight: FontWeight.bold, color: Colors.grey, fontSize: 16 * widget.sf))
|
||||||
|
),
|
||||||
|
|
||||||
|
_buildSearch(label: "Adversário", controller: _opponentController, sf: widget.sf),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context),
|
||||||
|
child: Text('CANCELAR', style: TextStyle(fontSize: 14 * widget.sf))
|
||||||
|
),
|
||||||
|
|
||||||
|
ElevatedButton(
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: const Color(0xFFE74C3C),
|
||||||
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10 * widget.sf)),
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: 16 * widget.sf, vertical: 10 * widget.sf)
|
||||||
|
),
|
||||||
|
onPressed: _isLoading ? null : () async {
|
||||||
|
if (_myTeamController.text.isNotEmpty && _opponentController.text.isNotEmpty) {
|
||||||
|
setState(() => _isLoading = true);
|
||||||
|
|
||||||
|
String? newGameId = await widget.gameController.createGame(
|
||||||
|
_myTeamController.text,
|
||||||
|
_opponentController.text,
|
||||||
|
_seasonController.text,
|
||||||
|
);
|
||||||
|
|
||||||
|
setState(() => _isLoading = false);
|
||||||
|
|
||||||
|
if (newGameId != null && context.mounted) {
|
||||||
|
Navigator.pop(context);
|
||||||
|
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => PlacarPage(
|
||||||
|
gameId: newGameId,
|
||||||
|
myTeam: _myTeamController.text,
|
||||||
|
opponentTeam: _opponentController.text,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: _isLoading
|
||||||
|
? SizedBox(width: 20 * widget.sf, height: 20 * widget.sf, child: const CircularProgressIndicator(color: Colors.white, strokeWidth: 2))
|
||||||
|
: Text('CRIAR', style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 14 * widget.sf)),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildSearch({required String label, required TextEditingController controller, required double sf}) {
|
||||||
|
return StreamBuilder<List<Map<String, dynamic>>>(
|
||||||
|
stream: widget.teamController.teamsStream,
|
||||||
|
builder: (context, snapshot) {
|
||||||
|
List<Map<String, dynamic>> teamList = snapshot.hasData ? snapshot.data! : [];
|
||||||
|
|
||||||
|
return Autocomplete<Map<String, dynamic>>(
|
||||||
|
displayStringForOption: (Map<String, dynamic> option) => option['name'].toString(),
|
||||||
|
|
||||||
|
optionsBuilder: (TextEditingValue val) {
|
||||||
|
if (val.text.isEmpty) return const Iterable<Map<String, dynamic>>.empty();
|
||||||
|
return teamList.where((t) =>
|
||||||
|
t['name'].toString().toLowerCase().contains(val.text.toLowerCase()));
|
||||||
|
},
|
||||||
|
|
||||||
|
onSelected: (Map<String, dynamic> selection) {
|
||||||
|
controller.text = selection['name'].toString();
|
||||||
|
},
|
||||||
|
|
||||||
|
optionsViewBuilder: (context, onSelected, options) {
|
||||||
|
return Align(
|
||||||
|
alignment: Alignment.topLeft,
|
||||||
|
child: Material(
|
||||||
|
elevation: 4.0,
|
||||||
|
borderRadius: BorderRadius.circular(8 * sf),
|
||||||
|
child: ConstrainedBox(
|
||||||
|
constraints: BoxConstraints(maxHeight: 250 * sf, maxWidth: MediaQuery.of(context).size.width * 0.7),
|
||||||
|
child: ListView.builder(
|
||||||
|
padding: EdgeInsets.zero,
|
||||||
|
shrinkWrap: true,
|
||||||
|
itemCount: options.length,
|
||||||
|
itemBuilder: (BuildContext context, int index) {
|
||||||
|
final option = options.elementAt(index);
|
||||||
|
final String name = option['name'].toString();
|
||||||
|
final String? imageUrl = option['image_url'];
|
||||||
|
|
||||||
|
return ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
radius: 20 * sf,
|
||||||
|
backgroundColor: Colors.grey.shade200,
|
||||||
|
backgroundImage: (imageUrl != null && imageUrl.isNotEmpty)
|
||||||
|
? NetworkImage(imageUrl)
|
||||||
|
: null,
|
||||||
|
child: (imageUrl == null || imageUrl.isEmpty)
|
||||||
|
? Icon(Icons.shield, color: Colors.grey, size: 20 * sf)
|
||||||
|
: null,
|
||||||
|
),
|
||||||
|
title: Text(name, style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14 * sf)),
|
||||||
|
onTap: () {
|
||||||
|
onSelected(option);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
fieldViewBuilder: (ctx, txtCtrl, node, submit) {
|
||||||
|
if (txtCtrl.text.isEmpty && controller.text.isNotEmpty) {
|
||||||
|
txtCtrl.text = controller.text;
|
||||||
|
}
|
||||||
|
txtCtrl.addListener(() {
|
||||||
|
controller.text = txtCtrl.text;
|
||||||
|
});
|
||||||
|
|
||||||
|
return TextField(
|
||||||
|
controller: txtCtrl,
|
||||||
|
focusNode: node,
|
||||||
|
style: TextStyle(fontSize: 14 * sf),
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: label,
|
||||||
|
labelStyle: TextStyle(fontSize: 14 * sf),
|
||||||
|
prefixIcon: Icon(Icons.search, size: 20 * sf),
|
||||||
|
border: const OutlineInputBorder()
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/classe/home.config.dart';
|
|
||||||
|
|
||||||
class StatCard extends StatelessWidget {
|
class StatCard extends StatelessWidget {
|
||||||
final String title;
|
final String title;
|
||||||
@@ -11,6 +10,11 @@ class StatCard extends StatelessWidget {
|
|||||||
final bool isHighlighted;
|
final bool isHighlighted;
|
||||||
final VoidCallback? onTap;
|
final VoidCallback? onTap;
|
||||||
|
|
||||||
|
// Variáveis novas para que o tamanho não fique preso à HomeConfig
|
||||||
|
final double sf;
|
||||||
|
final double cardWidth;
|
||||||
|
final double cardHeight;
|
||||||
|
|
||||||
const StatCard({
|
const StatCard({
|
||||||
super.key,
|
super.key,
|
||||||
required this.title,
|
required this.title,
|
||||||
@@ -21,27 +25,30 @@ class StatCard extends StatelessWidget {
|
|||||||
required this.icon,
|
required this.icon,
|
||||||
this.isHighlighted = false,
|
this.isHighlighted = false,
|
||||||
this.onTap,
|
this.onTap,
|
||||||
|
this.sf = 1.0, // Default 1.0 para não dar erro se não passares o valor
|
||||||
|
required this.cardWidth,
|
||||||
|
required this.cardHeight,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return SizedBox(
|
return SizedBox(
|
||||||
width: HomeConfig.cardwidthPadding,
|
width: cardWidth,
|
||||||
height: HomeConfig.cardheightPadding,
|
height: cardHeight,
|
||||||
child: Card(
|
child: Card(
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
shape: RoundedRectangleBorder(
|
shape: RoundedRectangleBorder(
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20 * sf),
|
||||||
side: isHighlighted
|
side: isHighlighted
|
||||||
? const BorderSide(color: Colors.amber, width: 2)
|
? BorderSide(color: Colors.amber, width: 2 * sf)
|
||||||
: BorderSide.none,
|
: BorderSide.none,
|
||||||
),
|
),
|
||||||
child: InkWell(
|
child: InkWell(
|
||||||
onTap: onTap,
|
onTap: onTap,
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20 * sf),
|
||||||
child: Container(
|
child: Container(
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20 * sf),
|
||||||
gradient: LinearGradient(
|
gradient: LinearGradient(
|
||||||
begin: Alignment.topCenter,
|
begin: Alignment.topCenter,
|
||||||
end: Alignment.bottomCenter,
|
end: Alignment.bottomCenter,
|
||||||
@@ -52,13 +59,14 @@ class StatCard extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.all(20.0),
|
padding: EdgeInsets.all(16.0 * sf),
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
// Cabeçalho
|
// Cabeçalho
|
||||||
Row(
|
Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Column(
|
child: Column(
|
||||||
@@ -66,12 +74,12 @@ class StatCard extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
title.toUpperCase(),
|
title.toUpperCase(),
|
||||||
style: HomeConfig.titleStyle,
|
style: TextStyle(fontSize: 11 * sf, fontWeight: FontWeight.bold, color: Colors.white70),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 5),
|
SizedBox(height: 2 * sf),
|
||||||
Text(
|
Text(
|
||||||
playerName,
|
playerName,
|
||||||
style: HomeConfig.playerNameStyle,
|
style: TextStyle(fontSize: 14 * sf, fontWeight: FontWeight.bold, color: Colors.white),
|
||||||
maxLines: 1,
|
maxLines: 1,
|
||||||
overflow: TextOverflow.ellipsis,
|
overflow: TextOverflow.ellipsis,
|
||||||
),
|
),
|
||||||
@@ -80,50 +88,75 @@ class StatCard extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
if (isHighlighted)
|
if (isHighlighted)
|
||||||
Container(
|
Container(
|
||||||
padding: const EdgeInsets.all(8),
|
padding: EdgeInsets.all(6 * sf),
|
||||||
decoration: const BoxDecoration(
|
decoration: const BoxDecoration(
|
||||||
color: Colors.amber,
|
color: Colors.amber,
|
||||||
shape: BoxShape.circle,
|
shape: BoxShape.circle,
|
||||||
),
|
),
|
||||||
child: const Icon(Icons.star, size: 20, color: Colors.white),
|
child: Icon(
|
||||||
|
Icons.star,
|
||||||
|
size: 16 * sf,
|
||||||
|
color: Colors.white,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
const SizedBox(height: 10),
|
|
||||||
|
SizedBox(height: 8 * sf),
|
||||||
|
|
||||||
// Ícone
|
// Ícone
|
||||||
Container(
|
Container(
|
||||||
width: 60,
|
width: 45 * sf,
|
||||||
height: 60,
|
height: 45 * sf,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white.withOpacity(0.2),
|
color: Colors.white.withOpacity(0.2),
|
||||||
shape: BoxShape.circle,
|
shape: BoxShape.circle,
|
||||||
),
|
),
|
||||||
child: Icon(icon, size: 30, color: Colors.white),
|
child: Icon(
|
||||||
|
icon,
|
||||||
|
size: 24 * sf,
|
||||||
|
color: Colors.white,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
|
|
||||||
// Estatística
|
// Estatística
|
||||||
Center(
|
Center(
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
Text(statValue, style: HomeConfig.statValueStyle),
|
Text(
|
||||||
const SizedBox(height: 5),
|
statValue,
|
||||||
Text(statLabel.toUpperCase(), style: HomeConfig.statLabelStyle),
|
style: TextStyle(fontSize: 34 * sf, fontWeight: FontWeight.bold, color: Colors.white),
|
||||||
|
),
|
||||||
|
SizedBox(height: 2 * sf),
|
||||||
|
Text(
|
||||||
|
statLabel.toUpperCase(),
|
||||||
|
style: TextStyle(fontSize: 12 * sf, color: Colors.white70),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
|
|
||||||
// Botão
|
// Botão
|
||||||
Container(
|
Container(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
padding: const EdgeInsets.symmetric(vertical: 12),
|
padding: EdgeInsets.symmetric(vertical: 8 * sf),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: Colors.white.withOpacity(0.2),
|
color: Colors.white.withOpacity(0.2),
|
||||||
borderRadius: BorderRadius.circular(15),
|
borderRadius: BorderRadius.circular(10 * sf),
|
||||||
),
|
),
|
||||||
child: const Center(
|
child: Center(
|
||||||
child: Text(
|
child: Text(
|
||||||
'VER DETALHES',
|
'VER DETALHES',
|
||||||
style: TextStyle(color: Colors.white, fontWeight: FontWeight.bold, fontSize: 14, letterSpacing: 1),
|
style: TextStyle(
|
||||||
|
color: Colors.white,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
fontSize: 11 * sf,
|
||||||
|
letterSpacing: 1,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -144,7 +177,7 @@ class SportGrid extends StatelessWidget {
|
|||||||
const SportGrid({
|
const SportGrid({
|
||||||
super.key,
|
super.key,
|
||||||
required this.children,
|
required this.children,
|
||||||
this.spacing = HomeConfig.cardSpacing,
|
this.spacing = 20.0, // Valor padrão se não for passado nada
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -153,6 +186,7 @@ class SportGrid extends StatelessWidget {
|
|||||||
|
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
|
// Primeira linha
|
||||||
if (children.length >= 2)
|
if (children.length >= 2)
|
||||||
Padding(
|
Padding(
|
||||||
padding: EdgeInsets.only(bottom: spacing),
|
padding: EdgeInsets.only(bottom: spacing),
|
||||||
@@ -165,6 +199,8 @@ class SportGrid extends StatelessWidget {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// Segunda linha
|
||||||
if (children.length >= 4)
|
if (children.length >= 4)
|
||||||
Row(
|
Row(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/controllers/login_controller.dart';
|
import 'package:playmaker/controllers/login_controller.dart';
|
||||||
import 'package:playmaker/pages/RegisterPage.dart';
|
import 'package:playmaker/pages/RegisterPage.dart';
|
||||||
import 'package:playmaker/classe/theme.dart'; // 👇 IMPORT DO TEMA
|
import '../utils/size_extension.dart'; // 👇 O NOSSO SUPERPODER!
|
||||||
import '../utils/size_extension.dart';
|
|
||||||
|
|
||||||
class BasketTrackHeader extends StatelessWidget {
|
class BasketTrackHeader extends StatelessWidget {
|
||||||
const BasketTrackHeader({super.key});
|
const BasketTrackHeader({super.key});
|
||||||
@@ -11,49 +10,32 @@ class BasketTrackHeader extends StatelessWidget {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
// Usamos um Stack para controlar a sobreposição exata
|
SizedBox(
|
||||||
Stack(
|
width: 200 * context.sf, // Ajusta o tamanho da imagem suavemente
|
||||||
alignment: Alignment.center,
|
height: 200 * context.sf,
|
||||||
children: [
|
child: Image.asset(
|
||||||
// 1. A Imagem (Aumentada para 320)
|
'assets/playmaker-logos.png',
|
||||||
SizedBox(
|
fit: BoxFit.contain,
|
||||||
width: 320 * context.sf,
|
),
|
||||||
height: 350 * context.sf,
|
),
|
||||||
child: Image.asset(
|
Text(
|
||||||
'assets/playmaker-logos.png',
|
'BasketTrack',
|
||||||
fit: BoxFit.contain,
|
style: TextStyle(
|
||||||
),
|
fontSize: 36 * context.sf,
|
||||||
),
|
fontWeight: FontWeight.bold,
|
||||||
// 2. O Texto "subido" para dentro da área da imagem
|
color: Colors.grey[900],
|
||||||
Positioned(
|
),
|
||||||
bottom: 5 * context.sf, // Ajusta este valor para aproximar/afastar do centro da logo
|
),
|
||||||
child: Column(
|
SizedBox(height: 6 * context.sf),
|
||||||
children: [
|
Text(
|
||||||
Text(
|
'Gere as tuas equipas e estatísticas',
|
||||||
'BasketTrack',
|
style: TextStyle(
|
||||||
style: TextStyle(
|
fontSize: 16 * context.sf,
|
||||||
fontSize: 36 * context.sf,
|
color: Colors.grey[600],
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.w500,
|
||||||
color: Theme.of(context).colorScheme.onSurface,
|
),
|
||||||
),
|
textAlign: TextAlign.center,
|
||||||
),
|
|
||||||
SizedBox(height: 4 * context.sf),
|
|
||||||
Text(
|
|
||||||
'Gere as tuas equipas e estatísticas',
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 16 * context.sf,
|
|
||||||
color: Colors.grey,
|
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
),
|
||||||
// Espaço extra para não bater nos campos de login logo a seguir
|
|
||||||
SizedBox(height: 10 * context.sf),
|
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -70,17 +52,13 @@ class LoginFormFields extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
TextField(
|
TextField(
|
||||||
controller: controller.emailController,
|
controller: controller.emailController,
|
||||||
style: TextStyle(fontSize: 15 * context.sf, color: Theme.of(context).colorScheme.onSurface),
|
style: TextStyle(fontSize: 15 * context.sf),
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'E-mail',
|
labelText: 'E-mail',
|
||||||
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
||||||
prefixIcon: Icon(Icons.email_outlined, size: 22 * context.sf, color: AppTheme.primaryRed), // 👇 Cor do tema
|
prefixIcon: Icon(Icons.email_outlined, size: 22 * context.sf),
|
||||||
errorText: controller.emailError,
|
errorText: controller.emailError,
|
||||||
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12 * context.sf),
|
|
||||||
borderSide: BorderSide(color: AppTheme.primaryRed, width: 2), // 👇 Cor do tema ao focar
|
|
||||||
),
|
|
||||||
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
||||||
),
|
),
|
||||||
keyboardType: TextInputType.emailAddress,
|
keyboardType: TextInputType.emailAddress,
|
||||||
@@ -89,21 +67,16 @@ class LoginFormFields extends StatelessWidget {
|
|||||||
TextField(
|
TextField(
|
||||||
controller: controller.passwordController,
|
controller: controller.passwordController,
|
||||||
obscureText: controller.obscurePassword,
|
obscureText: controller.obscurePassword,
|
||||||
style: TextStyle(fontSize: 15 * context.sf, color: Theme.of(context).colorScheme.onSurface),
|
style: TextStyle(fontSize: 15 * context.sf),
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'Palavra-passe',
|
labelText: 'Palavra-passe',
|
||||||
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
||||||
prefixIcon: Icon(Icons.lock_outlined, size: 22 * context.sf, color: AppTheme.primaryRed), // 👇 Cor do tema
|
prefixIcon: Icon(Icons.lock_outlined, size: 22 * context.sf),
|
||||||
errorText: controller.passwordError,
|
errorText: controller.passwordError,
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12 * context.sf),
|
|
||||||
borderSide: BorderSide(color: AppTheme.primaryRed, width: 2), // 👇 Cor do tema ao focar
|
|
||||||
),
|
|
||||||
suffixIcon: IconButton(
|
suffixIcon: IconButton(
|
||||||
icon: Icon(
|
icon: Icon(
|
||||||
controller.obscurePassword ? Icons.visibility_outlined : Icons.visibility_off_outlined,
|
controller.obscurePassword ? Icons.visibility_outlined : Icons.visibility_off_outlined,
|
||||||
size: 22 * context.sf,
|
size: 22 * context.sf
|
||||||
color: Colors.grey,
|
|
||||||
),
|
),
|
||||||
onPressed: controller.togglePasswordVisibility,
|
onPressed: controller.togglePasswordVisibility,
|
||||||
),
|
),
|
||||||
@@ -133,7 +106,7 @@ class LoginButton extends StatelessWidget {
|
|||||||
if (success) onLoginSuccess();
|
if (success) onLoginSuccess();
|
||||||
},
|
},
|
||||||
style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
backgroundColor: AppTheme.primaryRed, // 👇 Usando a cor do tema
|
backgroundColor: const Color(0xFFE74C3C),
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14 * context.sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14 * context.sf)),
|
||||||
elevation: 3,
|
elevation: 3,
|
||||||
@@ -162,8 +135,8 @@ class CreateAccountButton extends StatelessWidget {
|
|||||||
Navigator.push(context, MaterialPageRoute(builder: (context) => const RegisterPage()));
|
Navigator.push(context, MaterialPageRoute(builder: (context) => const RegisterPage()));
|
||||||
},
|
},
|
||||||
style: OutlinedButton.styleFrom(
|
style: OutlinedButton.styleFrom(
|
||||||
foregroundColor: AppTheme.primaryRed, // 👇 Usando a cor do tema
|
foregroundColor: const Color(0xFFE74C3C),
|
||||||
side: BorderSide(color: AppTheme.primaryRed, width: 2 * context.sf), // 👇 Usando a cor do tema
|
side: BorderSide(color: const Color(0xFFE74C3C), width: 2 * context.sf),
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14 * context.sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14 * context.sf)),
|
||||||
),
|
),
|
||||||
child: Text('Criar Conta', style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold)),
|
child: Text('Criar Conta', style: TextStyle(fontSize: 18 * context.sf, fontWeight: FontWeight.bold)),
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,5 +1,4 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/classe/theme.dart'; // 👇 IMPORT DO TEMA
|
|
||||||
import '../controllers/register_controller.dart';
|
import '../controllers/register_controller.dart';
|
||||||
import '../utils/size_extension.dart'; // 👇 O NOSSO SUPERPODER!
|
import '../utils/size_extension.dart'; // 👇 O NOSSO SUPERPODER!
|
||||||
|
|
||||||
@@ -10,20 +9,16 @@ class RegisterHeader extends StatelessWidget {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
Icon(Icons.person_add_outlined, size: 100 * context.sf, color: AppTheme.primaryRed), // 👇 Cor do tema
|
Icon(Icons.person_add_outlined, size: 100 * context.sf, color: const Color(0xFFE74C3C)),
|
||||||
SizedBox(height: 10 * context.sf),
|
SizedBox(height: 10 * context.sf),
|
||||||
Text(
|
Text(
|
||||||
'Nova Conta',
|
'Nova Conta',
|
||||||
style: TextStyle(
|
style: TextStyle(fontSize: 36 * context.sf, fontWeight: FontWeight.bold, color: Colors.grey[900]),
|
||||||
fontSize: 36 * context.sf,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
color: Theme.of(context).colorScheme.onSurface, // 👇 Adaptável ao Modo Escuro
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
SizedBox(height: 5 * context.sf),
|
SizedBox(height: 5 * context.sf),
|
||||||
Text(
|
Text(
|
||||||
'Cria o teu perfil no BasketTrack',
|
'Cria o teu perfil no BasketTrack',
|
||||||
style: TextStyle(fontSize: 16 * context.sf, color: Colors.grey, fontWeight: FontWeight.w500),
|
style: TextStyle(fontSize: 16 * context.sf, color: Colors.grey[600], fontWeight: FontWeight.w500),
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -50,16 +45,12 @@ class _RegisterFormFieldsState extends State<RegisterFormFields> {
|
|||||||
children: [
|
children: [
|
||||||
TextFormField(
|
TextFormField(
|
||||||
controller: widget.controller.nameController,
|
controller: widget.controller.nameController,
|
||||||
style: TextStyle(fontSize: 15 * context.sf, color: Theme.of(context).colorScheme.onSurface),
|
style: TextStyle(fontSize: 15 * context.sf),
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'Nome Completo',
|
labelText: 'Nome Completo',
|
||||||
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
||||||
prefixIcon: Icon(Icons.person_outline, size: 22 * context.sf, color: AppTheme.primaryRed), // 👇 Cor do tema
|
prefixIcon: Icon(Icons.person_outline, size: 22 * context.sf),
|
||||||
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12 * context.sf),
|
|
||||||
borderSide: BorderSide(color: AppTheme.primaryRed, width: 2), // 👇 Destaque ao focar
|
|
||||||
),
|
|
||||||
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -68,16 +59,12 @@ class _RegisterFormFieldsState extends State<RegisterFormFields> {
|
|||||||
TextFormField(
|
TextFormField(
|
||||||
controller: widget.controller.emailController,
|
controller: widget.controller.emailController,
|
||||||
validator: widget.controller.validateEmail,
|
validator: widget.controller.validateEmail,
|
||||||
style: TextStyle(fontSize: 15 * context.sf, color: Theme.of(context).colorScheme.onSurface),
|
style: TextStyle(fontSize: 15 * context.sf),
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'E-mail',
|
labelText: 'E-mail',
|
||||||
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
||||||
prefixIcon: Icon(Icons.email_outlined, size: 22 * context.sf, color: AppTheme.primaryRed),
|
prefixIcon: Icon(Icons.email_outlined, size: 22 * context.sf),
|
||||||
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12 * context.sf),
|
|
||||||
borderSide: BorderSide(color: AppTheme.primaryRed, width: 2),
|
|
||||||
),
|
|
||||||
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
||||||
),
|
),
|
||||||
keyboardType: TextInputType.emailAddress,
|
keyboardType: TextInputType.emailAddress,
|
||||||
@@ -88,17 +75,13 @@ class _RegisterFormFieldsState extends State<RegisterFormFields> {
|
|||||||
controller: widget.controller.passwordController,
|
controller: widget.controller.passwordController,
|
||||||
obscureText: _obscurePassword,
|
obscureText: _obscurePassword,
|
||||||
validator: widget.controller.validatePassword,
|
validator: widget.controller.validatePassword,
|
||||||
style: TextStyle(fontSize: 15 * context.sf, color: Theme.of(context).colorScheme.onSurface),
|
style: TextStyle(fontSize: 15 * context.sf),
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'Palavra-passe',
|
labelText: 'Palavra-passe',
|
||||||
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
||||||
prefixIcon: Icon(Icons.lock_outlined, size: 22 * context.sf, color: AppTheme.primaryRed),
|
prefixIcon: Icon(Icons.lock_outlined, size: 22 * context.sf),
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12 * context.sf),
|
|
||||||
borderSide: BorderSide(color: AppTheme.primaryRed, width: 2),
|
|
||||||
),
|
|
||||||
suffixIcon: IconButton(
|
suffixIcon: IconButton(
|
||||||
icon: Icon(_obscurePassword ? Icons.visibility_outlined : Icons.visibility_off_outlined, size: 22 * context.sf, color: Colors.grey),
|
icon: Icon(_obscurePassword ? Icons.visibility_outlined : Icons.visibility_off_outlined, size: 22 * context.sf),
|
||||||
onPressed: () => setState(() => _obscurePassword = !_obscurePassword),
|
onPressed: () => setState(() => _obscurePassword = !_obscurePassword),
|
||||||
),
|
),
|
||||||
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
||||||
@@ -111,15 +94,11 @@ class _RegisterFormFieldsState extends State<RegisterFormFields> {
|
|||||||
controller: widget.controller.confirmPasswordController,
|
controller: widget.controller.confirmPasswordController,
|
||||||
obscureText: _obscurePassword,
|
obscureText: _obscurePassword,
|
||||||
validator: widget.controller.validateConfirmPassword,
|
validator: widget.controller.validateConfirmPassword,
|
||||||
style: TextStyle(fontSize: 15 * context.sf, color: Theme.of(context).colorScheme.onSurface),
|
style: TextStyle(fontSize: 15 * context.sf),
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: 'Confirmar Palavra-passe',
|
labelText: 'Confirmar Palavra-passe',
|
||||||
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
labelStyle: TextStyle(fontSize: 15 * context.sf),
|
||||||
prefixIcon: Icon(Icons.lock_clock_outlined, size: 22 * context.sf, color: AppTheme.primaryRed),
|
prefixIcon: Icon(Icons.lock_clock_outlined, size: 22 * context.sf),
|
||||||
focusedBorder: OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.circular(12 * context.sf),
|
|
||||||
borderSide: BorderSide(color: AppTheme.primaryRed, width: 2),
|
|
||||||
),
|
|
||||||
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
border: OutlineInputBorder(borderRadius: BorderRadius.circular(12 * context.sf)),
|
||||||
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
contentPadding: EdgeInsets.symmetric(vertical: 18 * context.sf, horizontal: 16 * context.sf),
|
||||||
),
|
),
|
||||||
@@ -142,7 +121,7 @@ class RegisterButton extends StatelessWidget {
|
|||||||
child: ElevatedButton(
|
child: ElevatedButton(
|
||||||
onPressed: controller.isLoading ? null : () => controller.signUp(context),
|
onPressed: controller.isLoading ? null : () => controller.signUp(context),
|
||||||
style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
backgroundColor: AppTheme.primaryRed, // 👇 Cor do tema
|
backgroundColor: const Color(0xFFE74C3C),
|
||||||
foregroundColor: Colors.white,
|
foregroundColor: Colors.white,
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14 * context.sf)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(14 * context.sf)),
|
||||||
elevation: 3,
|
elevation: 3,
|
||||||
|
|||||||
@@ -118,7 +118,8 @@ class PersonCard extends StatelessWidget {
|
|||||||
height: 45,
|
height: 45,
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
decoration: BoxDecoration(color: Colors.blue.withOpacity(0.1), borderRadius: BorderRadius.circular(10)),
|
decoration: BoxDecoration(color: Colors.blue.withOpacity(0.1), borderRadius: BorderRadius.circular(10)),
|
||||||
child: Text(person.number ?? "J", style: const TextStyle(color: Colors.blue, fontWeight: FontWeight.bold, fontSize: 16)), ),
|
child: Text(person.number, style: const TextStyle(color: Colors.blue, fontWeight: FontWeight.bold, fontSize: 16)),
|
||||||
|
),
|
||||||
title: Text(person.name, style: const TextStyle(fontWeight: FontWeight.bold)),
|
title: Text(person.name, style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
trailing: Row(
|
trailing: Row(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
|||||||
@@ -1,67 +1,158 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:playmaker/classe/theme.dart'; // 👇 IMPORT DO TEMA
|
import 'package:playmaker/screens/team_stats_page.dart';
|
||||||
import '../models/team_model.dart';
|
import '../models/team_model.dart';
|
||||||
import '../models/person_model.dart';
|
import '../controllers/team_controller.dart';
|
||||||
import '../utils/size_extension.dart'; // 👇 O NOSSO SUPERPODER!
|
|
||||||
|
|
||||||
// --- CABEÇALHO ---
|
class TeamCard extends StatelessWidget {
|
||||||
class StatsHeader extends StatelessWidget {
|
|
||||||
final Team team;
|
final Team team;
|
||||||
|
final TeamController controller;
|
||||||
|
final VoidCallback onFavoriteTap;
|
||||||
|
final double sf; // <-- Variável de escala
|
||||||
|
|
||||||
const StatsHeader({super.key, required this.team});
|
const TeamCard({
|
||||||
|
super.key,
|
||||||
|
required this.team,
|
||||||
|
required this.controller,
|
||||||
|
required this.onFavoriteTap,
|
||||||
|
required this.sf,
|
||||||
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Container(
|
return Card(
|
||||||
padding: EdgeInsets.only(
|
color: Colors.white,
|
||||||
top: 50 * context.sf,
|
elevation: 3,
|
||||||
left: 20 * context.sf,
|
margin: EdgeInsets.only(bottom: 12 * sf),
|
||||||
right: 20 * context.sf,
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * sf)),
|
||||||
bottom: 20 * context.sf
|
child: ListTile(
|
||||||
),
|
contentPadding: EdgeInsets.symmetric(horizontal: 16 * sf, vertical: 8 * sf),
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: AppTheme.primaryRed, // 👇 Usando a cor do teu tema!
|
// --- 1. IMAGEM + FAVORITO ---
|
||||||
borderRadius: BorderRadius.only(
|
leading: Stack(
|
||||||
bottomLeft: Radius.circular(30 * context.sf),
|
clipBehavior: Clip.none,
|
||||||
bottomRight: Radius.circular(30 * context.sf)
|
children: [
|
||||||
|
CircleAvatar(
|
||||||
|
radius: 28 * sf,
|
||||||
|
backgroundColor: Colors.grey[200],
|
||||||
|
backgroundImage: (team.imageUrl.isNotEmpty && team.imageUrl.startsWith('http'))
|
||||||
|
? NetworkImage(team.imageUrl)
|
||||||
|
: null,
|
||||||
|
child: (team.imageUrl.isEmpty || !team.imageUrl.startsWith('http'))
|
||||||
|
? Text(
|
||||||
|
team.imageUrl.isEmpty ? "🏀" : team.imageUrl,
|
||||||
|
style: TextStyle(fontSize: 24 * sf),
|
||||||
|
)
|
||||||
|
: null,
|
||||||
|
),
|
||||||
|
Positioned(
|
||||||
|
left: -15 * sf,
|
||||||
|
top: -10 * sf,
|
||||||
|
child: IconButton(
|
||||||
|
icon: Icon(
|
||||||
|
team.isFavorite ? Icons.star : Icons.star_border,
|
||||||
|
color: team.isFavorite ? Colors.amber : Colors.black.withOpacity(0.1),
|
||||||
|
size: 28 * sf,
|
||||||
|
shadows: [
|
||||||
|
Shadow(
|
||||||
|
color: Colors.black.withOpacity(team.isFavorite ? 0.3 : 0.1),
|
||||||
|
blurRadius: 4 * sf,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
onPressed: onFavoriteTap,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
),
|
|
||||||
child: Row(
|
// --- 2. TÍTULO ---
|
||||||
children: [
|
title: Text(
|
||||||
IconButton(
|
team.name,
|
||||||
icon: Icon(Icons.arrow_back, color: Colors.white, size: 24 * context.sf),
|
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16 * sf),
|
||||||
onPressed: () => Navigator.pop(context),
|
overflow: TextOverflow.ellipsis, // Previne overflows em nomes longos
|
||||||
),
|
),
|
||||||
SizedBox(width: 10 * context.sf),
|
|
||||||
CircleAvatar(
|
// --- 3. SUBTÍTULO (Contagem + Época em TEMPO REAL) ---
|
||||||
radius: 24 * context.sf,
|
subtitle: Padding(
|
||||||
backgroundColor: Colors.white24,
|
padding: EdgeInsets.only(top: 6.0 * sf),
|
||||||
backgroundImage: (team.imageUrl.isNotEmpty && team.imageUrl.startsWith('http'))
|
child: Row(
|
||||||
? NetworkImage(team.imageUrl)
|
children: [
|
||||||
: null,
|
Icon(Icons.groups_outlined, size: 16 * sf, color: Colors.grey),
|
||||||
child: (team.imageUrl.isEmpty || !team.imageUrl.startsWith('http'))
|
SizedBox(width: 4 * sf),
|
||||||
? Text(
|
|
||||||
team.imageUrl.isEmpty ? "🛡️" : team.imageUrl,
|
// 👇 A CORREÇÃO ESTÁ AQUI: StreamBuilder em vez de FutureBuilder 👇
|
||||||
style: TextStyle(fontSize: 20 * context.sf),
|
StreamBuilder<int>(
|
||||||
)
|
stream: controller.getPlayerCountStream(team.id),
|
||||||
: null,
|
initialData: 0,
|
||||||
),
|
builder: (context, snapshot) {
|
||||||
SizedBox(width: 15 * context.sf),
|
final count = snapshot.data ?? 0;
|
||||||
Expanded(
|
return Text(
|
||||||
child: Column(
|
"$count Jogs.", // Abreviado para poupar espaço
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
style: TextStyle(
|
||||||
children: [
|
color: count > 0 ? Colors.green[700] : Colors.orange,
|
||||||
Text(
|
fontWeight: FontWeight.bold,
|
||||||
team.name,
|
fontSize: 13 * sf,
|
||||||
style: TextStyle(color: Colors.white, fontSize: 20 * context.sf, fontWeight: FontWeight.bold),
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
|
||||||
|
SizedBox(width: 8 * sf),
|
||||||
|
Expanded( // Garante que a temporada se adapta se faltar espaço
|
||||||
|
child: Text(
|
||||||
|
"| ${team.season}",
|
||||||
|
style: TextStyle(color: Colors.grey, fontSize: 13 * sf),
|
||||||
overflow: TextOverflow.ellipsis,
|
overflow: TextOverflow.ellipsis,
|
||||||
),
|
),
|
||||||
Text(
|
),
|
||||||
team.season,
|
],
|
||||||
style: TextStyle(color: Colors.white70, fontSize: 14 * context.sf)
|
),
|
||||||
),
|
),
|
||||||
],
|
|
||||||
|
// --- 4. BOTÕES (Estatísticas e Apagar) ---
|
||||||
|
trailing: Row(
|
||||||
|
mainAxisSize: MainAxisSize.min, // <-- ISTO RESOLVE O OVERFLOW DAS RISCAS AMARELAS
|
||||||
|
children: [
|
||||||
|
IconButton(
|
||||||
|
tooltip: 'Ver Estatísticas',
|
||||||
|
icon: Icon(Icons.bar_chart_rounded, color: Colors.blue, size: 24 * sf),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => TeamStatsPage(team: team),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
),
|
),
|
||||||
|
IconButton(
|
||||||
|
tooltip: 'Eliminar Equipa',
|
||||||
|
icon: Icon(Icons.delete_outline, color: const Color(0xFFE74C3C), size: 24 * sf),
|
||||||
|
onPressed: () => _confirmDelete(context),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _confirmDelete(BuildContext context) {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (context) => AlertDialog(
|
||||||
|
title: Text('Eliminar Equipa?', style: TextStyle(fontSize: 18 * sf, fontWeight: FontWeight.bold)),
|
||||||
|
content: Text('Tens a certeza que queres eliminar "${team.name}"?', style: TextStyle(fontSize: 14 * sf)),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context),
|
||||||
|
child: Text('Cancelar', style: TextStyle(fontSize: 14 * sf)),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () {
|
||||||
|
controller.deleteTeam(team.id);
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
child: Text('Eliminar', style: TextStyle(color: Colors.red, fontSize: 14 * sf)),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@@ -69,164 +160,90 @@ class StatsHeader extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- CARD DE RESUMO ---
|
// --- DIALOG DE CRIAÇÃO ---
|
||||||
class StatsSummaryCard extends StatelessWidget {
|
class CreateTeamDialog extends StatefulWidget {
|
||||||
final int total;
|
final Function(String name, String season, String imageUrl) onConfirm;
|
||||||
|
final double sf; // Recebe a escala
|
||||||
|
|
||||||
const StatsSummaryCard({super.key, required this.total});
|
const CreateTeamDialog({super.key, required this.onConfirm, required this.sf});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<CreateTeamDialog> createState() => _CreateTeamDialogState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CreateTeamDialogState extends State<CreateTeamDialog> {
|
||||||
|
final TextEditingController _nameController = TextEditingController();
|
||||||
|
final TextEditingController _imageController = TextEditingController();
|
||||||
|
String _selectedSeason = '2024/25';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
// 👇 Adaptável ao Modo Escuro
|
return AlertDialog(
|
||||||
final cardColor = Theme.of(context).brightness == Brightness.dark
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * widget.sf)),
|
||||||
? const Color(0xFF1E1E1E)
|
title: Text('Nova Equipa', style: TextStyle(fontSize: 18 * widget.sf, fontWeight: FontWeight.bold)),
|
||||||
: Colors.white;
|
content: SingleChildScrollView(
|
||||||
|
child: Column(
|
||||||
return Card(
|
mainAxisSize: MainAxisSize.min,
|
||||||
elevation: 4,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20 * context.sf)),
|
|
||||||
child: Container(
|
|
||||||
padding: EdgeInsets.all(20 * context.sf),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: cardColor,
|
|
||||||
borderRadius: BorderRadius.circular(20 * context.sf),
|
|
||||||
border: Border.all(color: Colors.grey.withOpacity(0.15)),
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
children: [
|
||||||
Row(
|
TextField(
|
||||||
children: [
|
controller: _nameController,
|
||||||
Icon(Icons.groups, color: AppTheme.primaryRed, size: 28 * context.sf), // 👇 Cor do tema
|
style: TextStyle(fontSize: 14 * widget.sf),
|
||||||
SizedBox(width: 10 * context.sf),
|
decoration: InputDecoration(
|
||||||
Text(
|
labelText: 'Nome da Equipa',
|
||||||
"Total de Membros",
|
labelStyle: TextStyle(fontSize: 14 * widget.sf)
|
||||||
style: TextStyle(
|
),
|
||||||
color: Theme.of(context).colorScheme.onSurface, // 👇 Adaptável
|
textCapitalization: TextCapitalization.words,
|
||||||
fontSize: 16 * context.sf,
|
|
||||||
fontWeight: FontWeight.w600
|
|
||||||
)
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
),
|
||||||
Text(
|
SizedBox(height: 15 * widget.sf),
|
||||||
"$total",
|
DropdownButtonFormField<String>(
|
||||||
style: TextStyle(
|
value: _selectedSeason,
|
||||||
color: Theme.of(context).colorScheme.onSurface, // 👇 Adaptável
|
decoration: InputDecoration(
|
||||||
fontSize: 28 * context.sf,
|
labelText: 'Temporada',
|
||||||
fontWeight: FontWeight.bold
|
labelStyle: TextStyle(fontSize: 14 * widget.sf)
|
||||||
)
|
),
|
||||||
|
style: TextStyle(fontSize: 14 * widget.sf, color: Colors.black87),
|
||||||
|
items: ['2023/24', '2024/25', '2025/26']
|
||||||
|
.map((s) => DropdownMenuItem(value: s, child: Text(s)))
|
||||||
|
.toList(),
|
||||||
|
onChanged: (val) => setState(() => _selectedSeason = val!),
|
||||||
|
),
|
||||||
|
SizedBox(height: 15 * widget.sf),
|
||||||
|
TextField(
|
||||||
|
controller: _imageController,
|
||||||
|
style: TextStyle(fontSize: 14 * widget.sf),
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: 'URL Imagem ou Emoji',
|
||||||
|
labelStyle: TextStyle(fontSize: 14 * widget.sf),
|
||||||
|
hintText: 'Ex: 🏀 ou https://...',
|
||||||
|
hintStyle: TextStyle(fontSize: 14 * widget.sf)
|
||||||
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
actions: [
|
||||||
}
|
TextButton(
|
||||||
}
|
onPressed: () => Navigator.pop(context),
|
||||||
|
child: Text('Cancelar', style: TextStyle(fontSize: 14 * widget.sf))
|
||||||
// --- TÍTULO DE SECÇÃO ---
|
),
|
||||||
class StatsSectionTitle extends StatelessWidget {
|
ElevatedButton(
|
||||||
final String title;
|
style: ElevatedButton.styleFrom(
|
||||||
|
backgroundColor: const Color(0xFFE74C3C),
|
||||||
const StatsSectionTitle({super.key, required this.title});
|
padding: EdgeInsets.symmetric(horizontal: 16 * widget.sf, vertical: 10 * widget.sf)
|
||||||
|
),
|
||||||
@override
|
onPressed: () {
|
||||||
Widget build(BuildContext context) {
|
if (_nameController.text.trim().isNotEmpty) {
|
||||||
return Column(
|
widget.onConfirm(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
_nameController.text.trim(),
|
||||||
children: [
|
_selectedSeason,
|
||||||
Text(
|
_imageController.text.trim(),
|
||||||
title,
|
);
|
||||||
style: TextStyle(
|
Navigator.pop(context);
|
||||||
fontSize: 18 * context.sf,
|
}
|
||||||
fontWeight: FontWeight.bold,
|
},
|
||||||
color: Theme.of(context).colorScheme.onSurface // 👇 Adaptável
|
child: Text('Criar', style: TextStyle(color: Colors.white, fontSize: 14 * widget.sf)),
|
||||||
)
|
|
||||||
),
|
),
|
||||||
Divider(color: Colors.grey.withOpacity(0.3)),
|
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- CARD DA PESSOA (JOGADOR/TREINADOR) ---
|
|
||||||
class PersonCard extends StatelessWidget {
|
|
||||||
final Person person;
|
|
||||||
final bool isCoach;
|
|
||||||
final VoidCallback onEdit;
|
|
||||||
final VoidCallback onDelete;
|
|
||||||
|
|
||||||
const PersonCard({
|
|
||||||
super.key,
|
|
||||||
required this.person,
|
|
||||||
required this.isCoach,
|
|
||||||
required this.onEdit,
|
|
||||||
required this.onDelete,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
// 👇 Cores adaptáveis para o Card
|
|
||||||
final defaultBg = Theme.of(context).brightness == Brightness.dark
|
|
||||||
? const Color(0xFF1E1E1E)
|
|
||||||
: Colors.white;
|
|
||||||
|
|
||||||
final coachBg = Theme.of(context).brightness == Brightness.dark
|
|
||||||
? AppTheme.warningAmber.withOpacity(0.1) // Amarelo escuro se for modo noturno
|
|
||||||
: const Color(0xFFFFF9C4); // Amarelo claro original
|
|
||||||
|
|
||||||
return Card(
|
|
||||||
margin: EdgeInsets.only(top: 12 * context.sf),
|
|
||||||
elevation: 2,
|
|
||||||
color: isCoach ? coachBg : defaultBg,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15 * context.sf)),
|
|
||||||
child: ListTile(
|
|
||||||
contentPadding: EdgeInsets.symmetric(horizontal: 16 * context.sf, vertical: 4 * context.sf),
|
|
||||||
leading: isCoach
|
|
||||||
? CircleAvatar(
|
|
||||||
radius: 22 * context.sf,
|
|
||||||
backgroundColor: AppTheme.warningAmber, // 👇 Cor do tema
|
|
||||||
child: Icon(Icons.person, color: Colors.white, size: 24 * context.sf)
|
|
||||||
)
|
|
||||||
: Container(
|
|
||||||
width: 45 * context.sf,
|
|
||||||
height: 45 * context.sf,
|
|
||||||
alignment: Alignment.center,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: AppTheme.primaryRed.withOpacity(0.1), // 👇 Cor do tema
|
|
||||||
borderRadius: BorderRadius.circular(10 * context.sf)
|
|
||||||
),
|
|
||||||
child: Text(
|
|
||||||
person.number ?? "J",
|
|
||||||
style: TextStyle(
|
|
||||||
color: AppTheme.primaryRed, // 👇 Cor do tema
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 16 * context.sf
|
|
||||||
)
|
|
||||||
),
|
|
||||||
),
|
|
||||||
title: Text(
|
|
||||||
person.name,
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 16 * context.sf,
|
|
||||||
color: Theme.of(context).colorScheme.onSurface, // 👇 Adaptável
|
|
||||||
)
|
|
||||||
),
|
|
||||||
trailing: Row(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
IconButton(
|
|
||||||
icon: Icon(Icons.edit_outlined, color: Colors.blue, size: 22 * context.sf),
|
|
||||||
onPressed: onEdit,
|
|
||||||
),
|
|
||||||
IconButton(
|
|
||||||
icon: Icon(Icons.delete_outline, color: AppTheme.primaryRed, size: 22 * context.sf), // 👇 Cor do tema
|
|
||||||
onPressed: onDelete,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,228 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'dart:math' as math;
|
|
||||||
|
|
||||||
class ZoneMapDialog extends StatelessWidget {
|
|
||||||
final String playerName;
|
|
||||||
final bool isMake;
|
|
||||||
final bool is3PointAction; // 👇 AGORA O POP-UP SABE O QUE ARRASTASTE!
|
|
||||||
final Function(String zone, int points, double relativeX, double relativeY) onZoneSelected;
|
|
||||||
|
|
||||||
const ZoneMapDialog({
|
|
||||||
super.key,
|
|
||||||
required this.playerName,
|
|
||||||
required this.isMake,
|
|
||||||
required this.is3PointAction,
|
|
||||||
required this.onZoneSelected,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final Color headerColor = const Color(0xFFE88F15);
|
|
||||||
final Color yellowBackground = const Color(0xFFDFAB00);
|
|
||||||
|
|
||||||
final double screenHeight = MediaQuery.of(context).size.height;
|
|
||||||
final double dialogHeight = screenHeight * 0.95;
|
|
||||||
final double dialogWidth = dialogHeight * 1.0;
|
|
||||||
|
|
||||||
return Dialog(
|
|
||||||
backgroundColor: yellowBackground,
|
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(12)),
|
|
||||||
clipBehavior: Clip.antiAlias,
|
|
||||||
insetPadding: const EdgeInsets.all(10),
|
|
||||||
child: SizedBox(
|
|
||||||
height: dialogHeight,
|
|
||||||
width: dialogWidth,
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
height: 40,
|
|
||||||
color: headerColor,
|
|
||||||
width: double.infinity,
|
|
||||||
child: Stack(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
isMake ? "Lançamento de $playerName (Marcou)" : "Lançamento de $playerName (Falhou)",
|
|
||||||
style: const TextStyle(color: Colors.white, fontSize: 14, fontWeight: FontWeight.bold),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
right: 8,
|
|
||||||
child: InkWell(
|
|
||||||
onTap: () => Navigator.pop(context),
|
|
||||||
child: Container(
|
|
||||||
padding: const EdgeInsets.all(4),
|
|
||||||
decoration: const BoxDecoration(color: Colors.white, shape: BoxShape.circle),
|
|
||||||
child: Icon(Icons.close, color: headerColor, size: 16),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Expanded(
|
|
||||||
child: LayoutBuilder(
|
|
||||||
builder: (context, constraints) {
|
|
||||||
return GestureDetector(
|
|
||||||
onTapUp: (details) => _calculateAndReturnZone(context, details.localPosition, constraints.biggest),
|
|
||||||
child: CustomPaint(
|
|
||||||
size: Size(constraints.maxWidth, constraints.maxHeight),
|
|
||||||
painter: DebugPainter(is3PointAction: is3PointAction), // 👇 Passa a info para o desenhador
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void _calculateAndReturnZone(BuildContext context, Offset tap, Size size) {
|
|
||||||
final double w = size.width;
|
|
||||||
final double h = size.height;
|
|
||||||
final double x = tap.dx;
|
|
||||||
final double y = tap.dy;
|
|
||||||
final double basketX = w / 2;
|
|
||||||
|
|
||||||
final double margin = w * 0.10;
|
|
||||||
final double length = h * 0.35;
|
|
||||||
final double larguraDoArco = (w / 2) - margin;
|
|
||||||
final double alturaDoArco = larguraDoArco * 0.30;
|
|
||||||
final double totalArcoHeight = alturaDoArco * 4;
|
|
||||||
|
|
||||||
String zone = "";
|
|
||||||
int pts = 2;
|
|
||||||
|
|
||||||
// 1. SABER SE CLICOU NA ZONA DE 3 OU DE 2
|
|
||||||
bool is3 = false;
|
|
||||||
if (y < length) {
|
|
||||||
if (x < margin || x > w - margin) is3 = true;
|
|
||||||
} else {
|
|
||||||
double dx = x - basketX;
|
|
||||||
double dy = y - length;
|
|
||||||
double ellipse = (dx * dx) / (larguraDoArco * larguraDoArco) + (dy * dy) / (math.pow(totalArcoHeight / 2, 2));
|
|
||||||
if (ellipse > 1.0) is3 = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 👇 MAGIA AQUI: BLOQUEIA O CLIQUE NA ZONA ESCURA! 👇
|
|
||||||
if (is3PointAction && !is3) return; // Arrastou 3pts mas clicou na de 2pts -> IGNORA
|
|
||||||
if (!is3PointAction && is3) return; // Arrastou 2pts mas clicou na de 3pts -> IGNORA
|
|
||||||
|
|
||||||
double angle = math.atan2(y - length, x - basketX);
|
|
||||||
|
|
||||||
if (is3) {
|
|
||||||
pts = 3;
|
|
||||||
if (y < length) {
|
|
||||||
zone = (x < w / 2) ? "Canto Esquerdo (3pt)" : "Canto Direito (3pt)";
|
|
||||||
} else if (angle > 2.35) {
|
|
||||||
zone = "Ala Esquerda (3pt)";
|
|
||||||
} else if (angle < 0.78) {
|
|
||||||
zone = "Ala Direita (3pt)";
|
|
||||||
} else {
|
|
||||||
zone = "Topo (3pt)";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
pts = 2;
|
|
||||||
final double pW = w * 0.28;
|
|
||||||
final double pH = h * 0.38;
|
|
||||||
if (x > basketX - pW / 2 && x < basketX + pW / 2 && y < pH) {
|
|
||||||
zone = "Garrafão";
|
|
||||||
} else {
|
|
||||||
if (y < length) {
|
|
||||||
zone = (x < w / 2) ? "Meia Distância (Canto Esq)" : "Meia Distância (Canto Dir)";
|
|
||||||
} else if (angle > 2.35) {
|
|
||||||
zone = "Meia Distância (Esq)";
|
|
||||||
} else if (angle < 0.78) {
|
|
||||||
zone = "Meia Distância (Dir)";
|
|
||||||
} else {
|
|
||||||
zone = "Meia Distância (Centro)";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 👇 A MUDANÇA ESTÁ AQUI! Passamos os dados e deixamos quem chamou decidir como fechar!
|
|
||||||
onZoneSelected(zone, pts, x / w, y / h);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class DebugPainter extends CustomPainter {
|
|
||||||
final bool is3PointAction;
|
|
||||||
|
|
||||||
DebugPainter({required this.is3PointAction});
|
|
||||||
|
|
||||||
@override
|
|
||||||
void paint(Canvas canvas, Size size) {
|
|
||||||
final double w = size.width;
|
|
||||||
final double h = size.height;
|
|
||||||
final double basketX = w / 2;
|
|
||||||
|
|
||||||
final Paint whiteStroke = Paint()..color = Colors.white..style = PaintingStyle.stroke..strokeWidth = 2.0;
|
|
||||||
final Paint blackStroke = Paint()..color = Colors.black87..style = PaintingStyle.stroke..strokeWidth = 2.0;
|
|
||||||
|
|
||||||
final double margin = w * 0.10;
|
|
||||||
final double length = h * 0.35;
|
|
||||||
final double larguraDoArco = (w / 2) - margin;
|
|
||||||
final double alturaDoArco = larguraDoArco * 0.30;
|
|
||||||
final double totalArcoHeight = alturaDoArco * 4;
|
|
||||||
|
|
||||||
// DESENHA O CAMPO
|
|
||||||
canvas.drawLine(Offset(margin, 0), Offset(margin, length), whiteStroke);
|
|
||||||
canvas.drawLine(Offset(w - margin, 0), Offset(w - margin, length), whiteStroke);
|
|
||||||
canvas.drawLine(Offset(0, length), Offset(margin, length), whiteStroke);
|
|
||||||
canvas.drawLine(Offset(w - margin, length), Offset(w, length), whiteStroke);
|
|
||||||
canvas.drawArc(Rect.fromCenter(center: Offset(basketX, length), width: larguraDoArco * 2, height: totalArcoHeight), 0, math.pi, false, whiteStroke);
|
|
||||||
|
|
||||||
double sXL = basketX + (larguraDoArco * math.cos(math.pi * 0.75));
|
|
||||||
double sYL = length + ((totalArcoHeight / 2) * math.sin(math.pi * 0.75));
|
|
||||||
double sXR = basketX + (larguraDoArco * math.cos(math.pi * 0.25));
|
|
||||||
double sYR = length + ((totalArcoHeight / 2) * math.sin(math.pi * 0.25));
|
|
||||||
|
|
||||||
canvas.drawLine(Offset(sXL, sYL), Offset(0, h * 0.85), whiteStroke);
|
|
||||||
canvas.drawLine(Offset(sXR, sYR), Offset(w, h * 0.85), whiteStroke);
|
|
||||||
|
|
||||||
final double pW = w * 0.28;
|
|
||||||
final double pH = h * 0.38;
|
|
||||||
canvas.drawRect(Rect.fromLTWH(basketX - pW / 2, 0, pW, pH), blackStroke);
|
|
||||||
|
|
||||||
final double ftR = pW / 2;
|
|
||||||
canvas.drawArc(Rect.fromCircle(center: Offset(basketX, pH), radius: ftR), 0, math.pi, false, blackStroke);
|
|
||||||
for (int i = 0; i < 10; i++) {
|
|
||||||
canvas.drawArc(Rect.fromCircle(center: Offset(basketX, pH), radius: ftR), math.pi + (i * 2 * (math.pi / 20)), math.pi / 20, false, blackStroke);
|
|
||||||
}
|
|
||||||
|
|
||||||
canvas.drawLine(Offset(basketX - pW / 2, pH), Offset(sXL, sYL), blackStroke);
|
|
||||||
canvas.drawLine(Offset(basketX + pW / 2, pH), Offset(sXR, sYR), blackStroke);
|
|
||||||
|
|
||||||
canvas.drawArc(Rect.fromCircle(center: Offset(basketX, h), radius: w * 0.12), math.pi, math.pi, false, blackStroke);
|
|
||||||
canvas.drawCircle(Offset(basketX, h * 0.12), w * 0.02, blackStroke);
|
|
||||||
canvas.drawLine(Offset(basketX - w * 0.08, h * 0.12 - 5), Offset(basketX + w * 0.08, h * 0.12 - 5), blackStroke);
|
|
||||||
|
|
||||||
// ==========================================
|
|
||||||
// 👇 EFEITO DE ESCURECIMENTO (SHADOW) 👇
|
|
||||||
// ==========================================
|
|
||||||
final Paint shadowPaint = Paint()..color = Colors.black.withOpacity(0.75); // 75% escuro!
|
|
||||||
|
|
||||||
// Cria o molde da área de 2 pontos
|
|
||||||
Path path2pt = Path();
|
|
||||||
path2pt.moveTo(margin, 0);
|
|
||||||
path2pt.lineTo(margin, length);
|
|
||||||
// Faz o arco curvo da linha de 3 pontos
|
|
||||||
path2pt.arcTo(Rect.fromCenter(center: Offset(basketX, length), width: larguraDoArco * 2, height: totalArcoHeight), math.pi, -math.pi, false);
|
|
||||||
path2pt.lineTo(w - margin, 0);
|
|
||||||
path2pt.close();
|
|
||||||
|
|
||||||
if (is3PointAction) {
|
|
||||||
// Arrastou 3 Pontos -> Escurece a Zona de 2!
|
|
||||||
canvas.drawPath(path2pt, shadowPaint);
|
|
||||||
} else {
|
|
||||||
// Arrastou 2 Pontos -> Escurece a Zona de 3!
|
|
||||||
Path fullScreen = Path()..addRect(Rect.fromLTWH(0, 0, w, h));
|
|
||||||
Path path3pt = Path.combine(PathOperation.difference, fullScreen, path2pt);
|
|
||||||
canvas.drawPath(path3pt, shadowPaint);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
bool shouldRepaint(covariant CustomPainter oldDelegate) => true;
|
|
||||||
}
|
|
||||||
@@ -6,21 +6,13 @@
|
|||||||
|
|
||||||
#include "generated_plugin_registrant.h"
|
#include "generated_plugin_registrant.h"
|
||||||
|
|
||||||
#include <file_selector_linux/file_selector_plugin.h>
|
|
||||||
#include <gtk/gtk_plugin.h>
|
#include <gtk/gtk_plugin.h>
|
||||||
#include <printing/printing_plugin.h>
|
|
||||||
#include <url_launcher_linux/url_launcher_plugin.h>
|
#include <url_launcher_linux/url_launcher_plugin.h>
|
||||||
|
|
||||||
void fl_register_plugins(FlPluginRegistry* registry) {
|
void fl_register_plugins(FlPluginRegistry* registry) {
|
||||||
g_autoptr(FlPluginRegistrar) file_selector_linux_registrar =
|
|
||||||
fl_plugin_registry_get_registrar_for_plugin(registry, "FileSelectorPlugin");
|
|
||||||
file_selector_plugin_register_with_registrar(file_selector_linux_registrar);
|
|
||||||
g_autoptr(FlPluginRegistrar) gtk_registrar =
|
g_autoptr(FlPluginRegistrar) gtk_registrar =
|
||||||
fl_plugin_registry_get_registrar_for_plugin(registry, "GtkPlugin");
|
fl_plugin_registry_get_registrar_for_plugin(registry, "GtkPlugin");
|
||||||
gtk_plugin_register_with_registrar(gtk_registrar);
|
gtk_plugin_register_with_registrar(gtk_registrar);
|
||||||
g_autoptr(FlPluginRegistrar) printing_registrar =
|
|
||||||
fl_plugin_registry_get_registrar_for_plugin(registry, "PrintingPlugin");
|
|
||||||
printing_plugin_register_with_registrar(printing_registrar);
|
|
||||||
g_autoptr(FlPluginRegistrar) url_launcher_linux_registrar =
|
g_autoptr(FlPluginRegistrar) url_launcher_linux_registrar =
|
||||||
fl_plugin_registry_get_registrar_for_plugin(registry, "UrlLauncherPlugin");
|
fl_plugin_registry_get_registrar_for_plugin(registry, "UrlLauncherPlugin");
|
||||||
url_launcher_plugin_register_with_registrar(url_launcher_linux_registrar);
|
url_launcher_plugin_register_with_registrar(url_launcher_linux_registrar);
|
||||||
|
|||||||
@@ -3,9 +3,7 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
list(APPEND FLUTTER_PLUGIN_LIST
|
list(APPEND FLUTTER_PLUGIN_LIST
|
||||||
file_selector_linux
|
|
||||||
gtk
|
gtk
|
||||||
printing
|
|
||||||
url_launcher_linux
|
url_launcher_linux
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -6,19 +6,13 @@ import FlutterMacOS
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
import app_links
|
import app_links
|
||||||
import file_selector_macos
|
|
||||||
import path_provider_foundation
|
import path_provider_foundation
|
||||||
import printing
|
|
||||||
import shared_preferences_foundation
|
import shared_preferences_foundation
|
||||||
import sqflite_darwin
|
|
||||||
import url_launcher_macos
|
import url_launcher_macos
|
||||||
|
|
||||||
func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
|
func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
|
||||||
AppLinksMacosPlugin.register(with: registry.registrar(forPlugin: "AppLinksMacosPlugin"))
|
AppLinksMacosPlugin.register(with: registry.registrar(forPlugin: "AppLinksMacosPlugin"))
|
||||||
FileSelectorPlugin.register(with: registry.registrar(forPlugin: "FileSelectorPlugin"))
|
|
||||||
PathProviderPlugin.register(with: registry.registrar(forPlugin: "PathProviderPlugin"))
|
PathProviderPlugin.register(with: registry.registrar(forPlugin: "PathProviderPlugin"))
|
||||||
PrintingPlugin.register(with: registry.registrar(forPlugin: "PrintingPlugin"))
|
|
||||||
SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin"))
|
SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin"))
|
||||||
SqflitePlugin.register(with: registry.registrar(forPlugin: "SqflitePlugin"))
|
|
||||||
UrlLauncherPlugin.register(with: registry.registrar(forPlugin: "UrlLauncherPlugin"))
|
UrlLauncherPlugin.register(with: registry.registrar(forPlugin: "UrlLauncherPlugin"))
|
||||||
}
|
}
|
||||||
|
|||||||
354
pubspec.lock
354
pubspec.lock
@@ -41,14 +41,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.0.4"
|
version: "1.0.4"
|
||||||
archive:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: archive
|
|
||||||
sha256: a96e8b390886ee8abb49b7bd3ac8df6f451c621619f52a26e815fdcf568959ff
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "4.0.9"
|
|
||||||
async:
|
async:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -57,22 +49,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.13.0"
|
version: "2.13.0"
|
||||||
barcode:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: barcode
|
|
||||||
sha256: "7b6729c37e3b7f34233e2318d866e8c48ddb46c1f7ad01ff7bb2a8de1da2b9f4"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.2.9"
|
|
||||||
bidi:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: bidi
|
|
||||||
sha256: "77f475165e94b261745cf1032c751e2032b8ed92ccb2bf5716036db79320637d"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.0.13"
|
|
||||||
boolean_selector:
|
boolean_selector:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -81,38 +57,14 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.1.2"
|
version: "2.1.2"
|
||||||
cached_network_image:
|
|
||||||
dependency: "direct main"
|
|
||||||
description:
|
|
||||||
name: cached_network_image
|
|
||||||
sha256: "7c1183e361e5c8b0a0f21a28401eecdbde252441106a9816400dd4c2b2424916"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "3.4.1"
|
|
||||||
cached_network_image_platform_interface:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: cached_network_image_platform_interface
|
|
||||||
sha256: "35814b016e37fbdc91f7ae18c8caf49ba5c88501813f73ce8a07027a395e2829"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "4.1.1"
|
|
||||||
cached_network_image_web:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: cached_network_image_web
|
|
||||||
sha256: "980842f4e8e2535b8dbd3d5ca0b1f0ba66bf61d14cc3a17a9b4788a3685ba062"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "1.3.1"
|
|
||||||
characters:
|
characters:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: characters
|
name: characters
|
||||||
sha256: f71061c654a3380576a52b451dd5532377954cf9dbd272a78fc8479606670803
|
sha256: faf38497bda5ead2a8c7615f4f7939df04333478bf32e4173fcb06d428b5716b
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.4.0"
|
version: "1.4.1"
|
||||||
clock:
|
clock:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -137,14 +89,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.1.2"
|
version: "3.1.2"
|
||||||
cross_file:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: cross_file
|
|
||||||
sha256: "28bb3ae56f117b5aec029d702a90f57d285cd975c3c5c281eaca38dbc47c5937"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.3.5+2"
|
|
||||||
crypto:
|
crypto:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -201,38 +145,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "7.0.1"
|
version: "7.0.1"
|
||||||
file_selector_linux:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: file_selector_linux
|
|
||||||
sha256: "2567f398e06ac72dcf2e98a0c95df2a9edd03c2c2e0cacd4780f20cdf56263a0"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.9.4"
|
|
||||||
file_selector_macos:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: file_selector_macos
|
|
||||||
sha256: "5e0bbe9c312416f1787a68259ea1505b52f258c587f12920422671807c4d618a"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.9.5"
|
|
||||||
file_selector_platform_interface:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: file_selector_platform_interface
|
|
||||||
sha256: "35e0bd61ebcdb91a3505813b055b09b79dfdc7d0aee9c09a7ba59ae4bb13dc85"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.7.0"
|
|
||||||
file_selector_windows:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: file_selector_windows
|
|
||||||
sha256: "62197474ae75893a62df75939c777763d39c2bc5f73ce5b88497208bc269abfd"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.9.3+5"
|
|
||||||
fixnum:
|
fixnum:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -246,14 +158,6 @@ packages:
|
|||||||
description: flutter
|
description: flutter
|
||||||
source: sdk
|
source: sdk
|
||||||
version: "0.0.0"
|
version: "0.0.0"
|
||||||
flutter_cache_manager:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: flutter_cache_manager
|
|
||||||
sha256: "400b6592f16a4409a7f2bb929a9a7e38c72cceb8ffb99ee57bbf2cb2cecf8386"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "3.4.1"
|
|
||||||
flutter_lints:
|
flutter_lints:
|
||||||
dependency: "direct dev"
|
dependency: "direct dev"
|
||||||
description:
|
description:
|
||||||
@@ -262,14 +166,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "5.0.0"
|
version: "5.0.0"
|
||||||
flutter_plugin_android_lifecycle:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: flutter_plugin_android_lifecycle
|
|
||||||
sha256: ee8068e0e1cd16c4a82714119918efdeed33b3ba7772c54b5d094ab53f9b7fd1
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.0.33"
|
|
||||||
flutter_test:
|
flutter_test:
|
||||||
dependency: "direct dev"
|
dependency: "direct dev"
|
||||||
description: flutter
|
description: flutter
|
||||||
@@ -320,102 +216,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "4.1.2"
|
version: "4.1.2"
|
||||||
image:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image
|
|
||||||
sha256: f9881ff4998044947ec38d098bc7c8316ae1186fa786eddffdb867b9bc94dfce
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "4.8.0"
|
|
||||||
image_cropper:
|
|
||||||
dependency: "direct main"
|
|
||||||
description:
|
|
||||||
name: image_cropper
|
|
||||||
sha256: "46c8f9aae51c8350b2a2982462f85a129e77b04675d35b09db5499437d7a996b"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "11.0.0"
|
|
||||||
image_cropper_for_web:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_cropper_for_web
|
|
||||||
sha256: e09749714bc24c4e3b31fbafa2e5b7229b0ff23e8b14d4ba44bd723b77611a0f
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "7.0.0"
|
|
||||||
image_cropper_platform_interface:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_cropper_platform_interface
|
|
||||||
sha256: "886a30ec199362cdcc2fbb053b8e53347fbfb9dbbdaa94f9ff85622609f5e7ff"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "8.0.0"
|
|
||||||
image_picker:
|
|
||||||
dependency: "direct main"
|
|
||||||
description:
|
|
||||||
name: image_picker
|
|
||||||
sha256: "784210112be18ea55f69d7076e2c656a4e24949fa9e76429fe53af0c0f4fa320"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "1.2.1"
|
|
||||||
image_picker_android:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_picker_android
|
|
||||||
sha256: eda9b91b7e266d9041084a42d605a74937d996b87083395c5e47835916a86156
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.8.13+14"
|
|
||||||
image_picker_for_web:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_picker_for_web
|
|
||||||
sha256: "66257a3191ab360d23a55c8241c91a6e329d31e94efa7be9cf7a212e65850214"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "3.1.1"
|
|
||||||
image_picker_ios:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_picker_ios
|
|
||||||
sha256: b9c4a438a9ff4f60808c9cf0039b93a42bb6c2211ef6ebb647394b2b3fa84588
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.8.13+6"
|
|
||||||
image_picker_linux:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_picker_linux
|
|
||||||
sha256: "1f81c5f2046b9ab724f85523e4af65be1d47b038160a8c8deed909762c308ed4"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.2.2"
|
|
||||||
image_picker_macos:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_picker_macos
|
|
||||||
sha256: "86f0f15a309de7e1a552c12df9ce5b59fe927e71385329355aec4776c6a8ec91"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.2.2+1"
|
|
||||||
image_picker_platform_interface:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_picker_platform_interface
|
|
||||||
sha256: "567e056716333a1647c64bb6bd873cff7622233a5c3f694be28a583d4715690c"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.11.1"
|
|
||||||
image_picker_windows:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: image_picker_windows
|
|
||||||
sha256: d248c86554a72b5495a31c56f060cf73a41c7ff541689327b1a7dbccc33adfae
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "0.2.2"
|
|
||||||
jwt_decode:
|
jwt_decode:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -468,18 +268,18 @@ packages:
|
|||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: matcher
|
name: matcher
|
||||||
sha256: dc58c723c3c24bf8d3e2d3ad3f2f9d7bd9cf43ec6feaa64181775e60190153f2
|
sha256: "12956d0ad8390bbcc63ca2e1469c0619946ccb52809807067a7020d57e647aa6"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "0.12.17"
|
version: "0.12.18"
|
||||||
material_color_utilities:
|
material_color_utilities:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: material_color_utilities
|
name: material_color_utilities
|
||||||
sha256: f7142bb1154231d7ea5f96bc7bde4bda2a0945d2806bb11670e30b850d56bdec
|
sha256: "9c337007e82b1889149c82ed242ed1cb24a66044e30979c44912381e9be4c48b"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "0.11.1"
|
version: "0.13.0"
|
||||||
meta:
|
meta:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -504,14 +304,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.0.0"
|
version: "1.0.0"
|
||||||
octo_image:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: octo_image
|
|
||||||
sha256: "34faa6639a78c7e3cbe79be6f9f96535867e879748ade7d17c9b1ae7536293bd"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.1.0"
|
|
||||||
path:
|
path:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -520,14 +312,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.9.1"
|
version: "1.9.1"
|
||||||
path_parsing:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: path_parsing
|
|
||||||
sha256: "883402936929eac138ee0a45da5b0f2c80f89913e6dc3bf77eb65b84b409c6ca"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "1.1.0"
|
|
||||||
path_provider:
|
path_provider:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -576,30 +360,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.3.0"
|
version: "2.3.0"
|
||||||
pdf:
|
|
||||||
dependency: "direct main"
|
|
||||||
description:
|
|
||||||
name: pdf
|
|
||||||
sha256: e47a275b267873d5944ad5f5ff0dcc7ac2e36c02b3046a0ffac9b72fd362c44b
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "3.12.0"
|
|
||||||
pdf_widget_wrapper:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: pdf_widget_wrapper
|
|
||||||
sha256: c930860d987213a3d58c7ec3b7ecf8085c3897f773e8dc23da9cae60a5d6d0f5
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "1.0.4"
|
|
||||||
petitparser:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: petitparser
|
|
||||||
sha256: "91bd59303e9f769f108f8df05e371341b15d59e995e6806aefab827b58336675"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "7.0.2"
|
|
||||||
platform:
|
platform:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -624,14 +384,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "4.0.0"
|
version: "4.0.0"
|
||||||
posix:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: posix
|
|
||||||
sha256: "185ef7606574f789b40f289c233efa52e96dead518aed988e040a10737febb07"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "6.5.0"
|
|
||||||
postgrest:
|
postgrest:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -640,14 +392,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.6.0"
|
version: "2.6.0"
|
||||||
printing:
|
|
||||||
dependency: "direct main"
|
|
||||||
description:
|
|
||||||
name: printing
|
|
||||||
sha256: "689170c9ddb1bda85826466ba80378aa8993486d3c959a71cd7d2d80cb606692"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "5.14.3"
|
|
||||||
provider:
|
provider:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@@ -656,14 +400,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "6.1.5+1"
|
version: "6.1.5+1"
|
||||||
qr:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: qr
|
|
||||||
sha256: "5a1d2586170e172b8a8c8470bbbffd5eb0cd38a66c0d77155ea138d3af3a4445"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "3.0.2"
|
|
||||||
realtime_client:
|
realtime_client:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -689,7 +425,7 @@ packages:
|
|||||||
source: hosted
|
source: hosted
|
||||||
version: "0.28.0"
|
version: "0.28.0"
|
||||||
shared_preferences:
|
shared_preferences:
|
||||||
dependency: "direct main"
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: shared_preferences
|
name: shared_preferences
|
||||||
sha256: "2939ae520c9024cb197fc20dee269cd8cdbf564c8b5746374ec6cacdc5169e64"
|
sha256: "2939ae520c9024cb197fc20dee269cd8cdbf564c8b5746374ec6cacdc5169e64"
|
||||||
@@ -744,14 +480,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.4.1"
|
version: "2.4.1"
|
||||||
shimmer:
|
|
||||||
dependency: "direct main"
|
|
||||||
description:
|
|
||||||
name: shimmer
|
|
||||||
sha256: "5f88c883a22e9f9f299e5ba0e4f7e6054857224976a5d9f839d4ebdc94a14ac9"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "3.0.0"
|
|
||||||
sky_engine:
|
sky_engine:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description: flutter
|
description: flutter
|
||||||
@@ -765,46 +493,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.10.1"
|
version: "1.10.1"
|
||||||
sqflite:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: sqflite
|
|
||||||
sha256: e2297b1da52f127bc7a3da11439985d9b536f75070f3325e62ada69a5c585d03
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.4.2"
|
|
||||||
sqflite_android:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: sqflite_android
|
|
||||||
sha256: "881e28efdcc9950fd8e9bb42713dcf1103e62a2e7168f23c9338d82db13dec40"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.4.2+3"
|
|
||||||
sqflite_common:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: sqflite_common
|
|
||||||
sha256: "6ef422a4525ecc601db6c0a2233ff448c731307906e92cabc9ba292afaae16a6"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.5.6"
|
|
||||||
sqflite_darwin:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: sqflite_darwin
|
|
||||||
sha256: "279832e5cde3fe99e8571879498c9211f3ca6391b0d818df4e17d9fff5c6ccb3"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.4.2"
|
|
||||||
sqflite_platform_interface:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: sqflite_platform_interface
|
|
||||||
sha256: "8dd4515c7bdcae0a785b0062859336de775e8c65db81ae33dd5445f35be61920"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "2.4.0"
|
|
||||||
stack_trace:
|
stack_trace:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -853,14 +541,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.12.0"
|
version: "2.12.0"
|
||||||
synchronized:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: synchronized
|
|
||||||
sha256: c254ade258ec8282947a0acbbc90b9575b4f19673533ee46f2f6e9b3aeefd7c0
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "3.4.0"
|
|
||||||
term_glyph:
|
term_glyph:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -873,10 +553,10 @@ packages:
|
|||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: test_api
|
name: test_api
|
||||||
sha256: ab2726c1a94d3176a45960b6234466ec367179b87dd74f1611adb1f3b5fb9d55
|
sha256: "93167629bfc610f71560ab9312acdda4959de4df6fac7492c89ff0d3886f6636"
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "0.7.7"
|
version: "0.7.9"
|
||||||
typed_data:
|
typed_data:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -949,14 +629,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.1.5"
|
version: "3.1.5"
|
||||||
uuid:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: uuid
|
|
||||||
sha256: "1fef9e8e11e2991bb773070d4656b7bd5d850967a2456cfc83cf47925ba79489"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "4.5.3"
|
|
||||||
vector_math:
|
vector_math:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -1005,14 +677,6 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.1.0"
|
version: "1.1.0"
|
||||||
xml:
|
|
||||||
dependency: transitive
|
|
||||||
description:
|
|
||||||
name: xml
|
|
||||||
sha256: "971043b3a0d3da28727e40ed3e0b5d18b742fa5a68665cca88e74b7876d5e025"
|
|
||||||
url: "https://pub.dev"
|
|
||||||
source: hosted
|
|
||||||
version: "6.6.1"
|
|
||||||
yet_another_json_isolate:
|
yet_another_json_isolate:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
|||||||
@@ -36,13 +36,6 @@ dependencies:
|
|||||||
cupertino_icons: ^1.0.8
|
cupertino_icons: ^1.0.8
|
||||||
provider: ^6.1.5+1
|
provider: ^6.1.5+1
|
||||||
supabase_flutter: ^2.12.0
|
supabase_flutter: ^2.12.0
|
||||||
image_picker: ^1.2.1
|
|
||||||
image_cropper: ^11.0.0
|
|
||||||
shimmer: ^3.0.0
|
|
||||||
cached_network_image: ^3.4.1
|
|
||||||
shared_preferences: ^2.5.4
|
|
||||||
printing: ^5.14.3
|
|
||||||
pdf: ^3.12.0
|
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
flutter_test:
|
flutter_test:
|
||||||
|
|||||||
@@ -7,17 +7,11 @@
|
|||||||
#include "generated_plugin_registrant.h"
|
#include "generated_plugin_registrant.h"
|
||||||
|
|
||||||
#include <app_links/app_links_plugin_c_api.h>
|
#include <app_links/app_links_plugin_c_api.h>
|
||||||
#include <file_selector_windows/file_selector_windows.h>
|
|
||||||
#include <printing/printing_plugin.h>
|
|
||||||
#include <url_launcher_windows/url_launcher_windows.h>
|
#include <url_launcher_windows/url_launcher_windows.h>
|
||||||
|
|
||||||
void RegisterPlugins(flutter::PluginRegistry* registry) {
|
void RegisterPlugins(flutter::PluginRegistry* registry) {
|
||||||
AppLinksPluginCApiRegisterWithRegistrar(
|
AppLinksPluginCApiRegisterWithRegistrar(
|
||||||
registry->GetRegistrarForPlugin("AppLinksPluginCApi"));
|
registry->GetRegistrarForPlugin("AppLinksPluginCApi"));
|
||||||
FileSelectorWindowsRegisterWithRegistrar(
|
|
||||||
registry->GetRegistrarForPlugin("FileSelectorWindows"));
|
|
||||||
PrintingPluginRegisterWithRegistrar(
|
|
||||||
registry->GetRegistrarForPlugin("PrintingPlugin"));
|
|
||||||
UrlLauncherWindowsRegisterWithRegistrar(
|
UrlLauncherWindowsRegisterWithRegistrar(
|
||||||
registry->GetRegistrarForPlugin("UrlLauncherWindows"));
|
registry->GetRegistrarForPlugin("UrlLauncherWindows"));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,8 +4,6 @@
|
|||||||
|
|
||||||
list(APPEND FLUTTER_PLUGIN_LIST
|
list(APPEND FLUTTER_PLUGIN_LIST
|
||||||
app_links
|
app_links
|
||||||
file_selector_windows
|
|
||||||
printing
|
|
||||||
url_launcher_windows
|
url_launcher_windows
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user