feat: Implement multiplayer game session management
- Add models for managing player assignments and game session initialization in `inicio_partida_multijugador.dart`. - Create a multiplayer room state management system in `sala_multijugador.dart`, including user registration, selection, and session validation. - Develop a UI screen for displaying player words sequentially in `pantalla_palabras_cliente.dart`. - Implement unit tests for the multiplayer session management and player assignment logic in `inicio_partida_multijugador_test.dart` and `sala_multijugador_test.dart`.
This commit is contained in:
@@ -2,6 +2,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:farolero/l10n/generated/app_localizations.dart';
|
||||
import 'package:provider/provider.dart';
|
||||
import '../estado/estado_juego.dart';
|
||||
import '../modelos/inicio_partida_multijugador.dart';
|
||||
import '../modelos/palabra.dart';
|
||||
import '../modelos/partida.dart';
|
||||
import '../modelos/usuario.dart';
|
||||
@@ -153,15 +154,21 @@ class _PantallaCrearPartidaState extends State<PantallaCrearPartida> {
|
||||
onIniciar: () {
|
||||
// Cuando el host toca "Iniciar" con suficientes jugadores
|
||||
final estado = context.read<EstadoJuego>();
|
||||
final sala = nearby.estadoSala;
|
||||
if (sala == null) return;
|
||||
final validacion = sala.iniciarPartida();
|
||||
if (!validacion.exitoso) {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(
|
||||
content: Text(
|
||||
'No se puede iniciar: ${validacion.codigo ?? "sala inválida"}',
|
||||
),
|
||||
),
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
// Set host local player first (required for host-included game)
|
||||
estado.setHostJugador(nombre.trim());
|
||||
|
||||
final jugadoresMulti = [
|
||||
nombre.trim(),
|
||||
...nearby.jugadores.map((j) => j.nombre),
|
||||
];
|
||||
estado.crearPartida(
|
||||
estado.crearPartidaDesdeSala(
|
||||
config: ConfigPartida(
|
||||
modoMultimovil: true,
|
||||
categoria: _categoria,
|
||||
@@ -169,24 +176,41 @@ class _PantallaCrearPartidaState extends State<PantallaCrearPartida> {
|
||||
pistaImpostor: _pistaImpostor,
|
||||
tiempoDebateSegundos: _tiempoDebate,
|
||||
),
|
||||
nombresJugadores: jugadoresMulti,
|
||||
sala: sala,
|
||||
);
|
||||
|
||||
// Enviar palabras a cada jugador via Nearby
|
||||
final partida = estado.partida!;
|
||||
final impostores = <String, bool>{};
|
||||
for (int i = 0; i < nearby.jugadores.length; i++) {
|
||||
final jugadorNearby = nearby.jugadores[i];
|
||||
// El jugador [0] es el host, los de nearby son [1..n]
|
||||
final jugadorPartida = partida.jugadores[i + 1];
|
||||
impostores[jugadorNearby.endpointId] =
|
||||
jugadorPartida.esImpostor;
|
||||
}
|
||||
final asignaciones = partida.jugadores.map((jugador) {
|
||||
final usuarioSala = sala.usuarios[jugador.id];
|
||||
final clientId = usuarioSala?.clienteIdSeleccionado;
|
||||
final cliente = clientId == null ? null : sala.clientes[clientId];
|
||||
return AsignacionJugador(
|
||||
jugadorId: jugador.id,
|
||||
nombre: jugador.nombre,
|
||||
clientId: clientId ?? sala.hostClientId,
|
||||
endpointId: cliente?.endpointId,
|
||||
);
|
||||
}).toList();
|
||||
final impostores = {
|
||||
for (final jugador in partida.jugadores)
|
||||
jugador.id: jugador.esImpostor,
|
||||
};
|
||||
final jugadoresTodos = partida.jugadores
|
||||
.map(
|
||||
(jugador) => {
|
||||
'id': jugador.id,
|
||||
'nombre': jugador.nombre,
|
||||
'eliminado': jugador.eliminado,
|
||||
},
|
||||
)
|
||||
.toList();
|
||||
|
||||
nearby.enviarInicioPartida(
|
||||
nearby.enviarInicioPartidaMulti(
|
||||
asignaciones: asignaciones,
|
||||
palabraSecreta: partida.palabraSecreta,
|
||||
categoria: _categoria,
|
||||
impostores: impostores,
|
||||
impostoresPorJugadorId: impostores,
|
||||
jugadoresTodos: jugadoresTodos,
|
||||
);
|
||||
|
||||
Navigator.pushReplacement(
|
||||
|
||||
@@ -3,9 +3,12 @@ import 'package:flutter/material.dart';
|
||||
import 'package:farolero/l10n/generated/app_localizations.dart';
|
||||
import 'package:provider/provider.dart';
|
||||
import '../estado/estado_juego.dart';
|
||||
import '../modelos/inicio_partida_multijugador.dart';
|
||||
import '../modelos/partida.dart';
|
||||
import '../servicios/servicio_nearby.dart';
|
||||
import '../tema/tema_app.dart';
|
||||
import 'pantalla_votacion_cliente.dart';
|
||||
import 'pantalla_palabras_cliente.dart';
|
||||
|
||||
class PantallaGestorHost extends StatefulWidget {
|
||||
final VoidCallback onPartidaFin;
|
||||
@@ -51,8 +54,11 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
setState(() => _clientesListos[endpointId] = true);
|
||||
} else if (mensaje.tipo == TipoMensaje.voto) {
|
||||
final votanteId = mensaje.datos['votanteId'] as String?;
|
||||
final votoId = mensaje.datos['votoporId'] as String?;
|
||||
final votoId =
|
||||
mensaje.datos['votadoId'] as String? ??
|
||||
mensaje.datos['votoporId'] as String?;
|
||||
if (votanteId != null && votoId != null) {
|
||||
context.read<EstadoJuego>().registrarVoto(votanteId, votoId);
|
||||
setState(() => _votosRecibidos[votanteId] = votoId);
|
||||
}
|
||||
}
|
||||
@@ -116,9 +122,8 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
);
|
||||
}
|
||||
|
||||
final numJugadores = partida.jugadores.length + 1;
|
||||
final todosListos = _clientesListos.length >= numJugadores - 1;
|
||||
final todosVotaron = _votosRecibidos.length >= numJugadores - 1;
|
||||
final todosListos = _clientesListos.length >= nearby.jugadores.length;
|
||||
final todosVotaron = estado.todosHanVotado();
|
||||
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
@@ -294,14 +299,45 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
|
||||
void _mostrarPalabraHost(BuildContext context) {
|
||||
final estado = context.read<EstadoJuego>();
|
||||
final sala = context.read<ServicioNearby>().estadoSala;
|
||||
final partida = estado.partida;
|
||||
if (partida == null) return;
|
||||
if (partida == null || sala == null) return;
|
||||
|
||||
// Buscar el jugador host local
|
||||
final hostLocal = partida.jugadores.firstWhere(
|
||||
(j) => j.nombre == context.read<ServicioNearby>().miNombre,
|
||||
orElse: () => partida.jugadores.first,
|
||||
);
|
||||
final jugadoresHost = sala
|
||||
.usuariosPorCliente(sala.hostClientId)
|
||||
.where((usuario) => partida.jugadores.any((j) => j.id == usuario.id))
|
||||
.map((usuario) {
|
||||
final jugador = partida.jugadores.firstWhere(
|
||||
(j) => j.id == usuario.id,
|
||||
);
|
||||
return JugadorInicioPartida(
|
||||
jugadorId: jugador.id,
|
||||
nombre: jugador.nombre,
|
||||
esImpostor: jugador.esImpostor,
|
||||
palabra: jugador.palabra,
|
||||
);
|
||||
})
|
||||
.toList();
|
||||
|
||||
if (jugadoresHost.length > 1) {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (_) => PantallaPalabrasCliente(
|
||||
jugadores: jugadoresHost,
|
||||
pistaCategoria: partida.config.pistaImpostor
|
||||
? partida.categoriaReal
|
||||
: null,
|
||||
onTodosVistos: () => Navigator.of(context).pop(),
|
||||
),
|
||||
),
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
final hostLocal = jugadoresHost.isNotEmpty
|
||||
? partida.jugadores.firstWhere((j) => j.id == jugadoresHost.first.jugadorId)
|
||||
: partida.jugadores.first;
|
||||
|
||||
Navigator.push(
|
||||
context,
|
||||
@@ -393,6 +429,12 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
bool todosVotaron,
|
||||
ServicioNearby nearby,
|
||||
) {
|
||||
final estado = context.watch<EstadoJuego>();
|
||||
final partida = estado.partida!;
|
||||
final totalVotos = partida.jugadoresActivos.length;
|
||||
final votosEmitidos = estado.votos.length;
|
||||
final progreso = totalVotos == 0 ? 0.0 : votosEmitidos / totalVotos;
|
||||
|
||||
return Card(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16),
|
||||
@@ -410,19 +452,12 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
),
|
||||
child: Column(
|
||||
children: [
|
||||
Text(
|
||||
l10n.votesProgress(
|
||||
_votosRecibidos.length,
|
||||
nearby.jugadores.length + 1,
|
||||
),
|
||||
),
|
||||
Text(l10n.votesProgress(votosEmitidos, totalVotos)),
|
||||
const SizedBox(height: 8),
|
||||
ClipRRect(
|
||||
borderRadius: BorderRadius.circular(4),
|
||||
child: LinearProgressIndicator(
|
||||
value:
|
||||
_votosRecibidos.length /
|
||||
(nearby.jugadores.length + 1),
|
||||
value: progreso.clamp(0.0, 1.0).toDouble(),
|
||||
backgroundColor: TemaApp.colorSuperficie,
|
||||
valueColor: const AlwaysStoppedAnimation(
|
||||
TemaApp.colorAcento,
|
||||
@@ -433,6 +468,19 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
],
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
SizedBox(
|
||||
width: double.infinity,
|
||||
child: OutlinedButton.icon(
|
||||
onPressed: _hostYaVoto(context) ? null : () => _abrirVotacionHost(context),
|
||||
icon: const Icon(Icons.how_to_vote),
|
||||
label: Text(
|
||||
_hostYaVoto(context)
|
||||
? 'Votos del host registrados'
|
||||
: 'Votar por los jugadores de este m?vil',
|
||||
),
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
Text(
|
||||
l10n.playersVoted,
|
||||
@@ -441,15 +489,11 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
const SizedBox(height: 8),
|
||||
Expanded(
|
||||
child: ListView.builder(
|
||||
itemCount: nearby.jugadores.length + 1,
|
||||
itemCount: partida.jugadoresActivos.length,
|
||||
itemBuilder: (context, index) {
|
||||
final esHost = index == 0;
|
||||
final nombre = esHost
|
||||
? (nearby.miNombre ?? 'Host')
|
||||
: nearby.jugadores[index - 1].nombre;
|
||||
final haVotado =
|
||||
esHost || _votosRecibidos.containsKey(nombre);
|
||||
return _buildJugadorTile(nombre, esHost, haVotado);
|
||||
final jugador = partida.jugadoresActivos[index];
|
||||
final haVotado = estado.votos.containsKey(jugador.id);
|
||||
return _buildJugadorTile(jugador.nombre, false, haVotado);
|
||||
},
|
||||
),
|
||||
),
|
||||
@@ -478,6 +522,51 @@ class _PantallaGestorHostState extends State<PantallaGestorHost> {
|
||||
);
|
||||
}
|
||||
|
||||
bool _hostYaVoto(BuildContext context) {
|
||||
final estado = context.read<EstadoJuego>();
|
||||
final sala = context.read<ServicioNearby>().estadoSala;
|
||||
if (sala == null || estado.partida == null) return false;
|
||||
final hostIds = sala.usuariosPorCliente(sala.hostClientId).map((u) => u.id);
|
||||
return hostIds.every((id) => estado.votos.containsKey(id));
|
||||
}
|
||||
|
||||
void _abrirVotacionHost(BuildContext context) {
|
||||
final estado = context.read<EstadoJuego>();
|
||||
final sala = context.read<ServicioNearby>().estadoSala;
|
||||
final partida = estado.partida;
|
||||
if (sala == null || partida == null) return;
|
||||
|
||||
final jugadoresHost = sala.usuariosPorCliente(sala.hostClientId)
|
||||
.where((usuario) => partida.jugadoresActivos.any((j) => j.id == usuario.id))
|
||||
.map(
|
||||
(usuario) => JugadorInicioPartida(
|
||||
jugadorId: usuario.id,
|
||||
nombre: usuario.nombre,
|
||||
esImpostor: partida.jugadores.firstWhere((j) => j.id == usuario.id).esImpostor,
|
||||
palabra: partida.jugadores.firstWhere((j) => j.id == usuario.id).palabra,
|
||||
),
|
||||
)
|
||||
.toList();
|
||||
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (_) => PantallaVotacionCliente(
|
||||
jugadores: partida.jugadoresActivos,
|
||||
jugadoresControlados: jugadoresHost,
|
||||
onVotos: (votos) {
|
||||
for (final entry in votos.entries) {
|
||||
estado.registrarVoto(entry.key, entry.value);
|
||||
_votosRecibidos[entry.key] = entry.value;
|
||||
}
|
||||
if (mounted) setState(() {});
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildJugadorTile(String nombre, bool esHost, bool listo) {
|
||||
return Container(
|
||||
margin: const EdgeInsets.only(bottom: 8),
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:provider/provider.dart';
|
||||
import 'package:qr_flutter/qr_flutter.dart';
|
||||
import 'package:farolero/l10n/generated/app_localizations.dart';
|
||||
@@ -6,7 +6,7 @@ import '../modelos/usuario.dart';
|
||||
import '../servicios/servicio_nearby.dart';
|
||||
import '../tema/tema_app.dart';
|
||||
|
||||
/// Pantalla de lobby del host: muestra QR y lista de jugadores conectados
|
||||
/// Lobby del host. El host es autoridad de sala y también cliente local.
|
||||
class PantallaLobbyHost extends StatefulWidget {
|
||||
final String nombreSala;
|
||||
final VoidCallback onIniciar;
|
||||
@@ -23,14 +23,16 @@ class PantallaLobbyHost extends StatefulWidget {
|
||||
|
||||
class _PantallaLobbyHostState extends State<PantallaLobbyHost> {
|
||||
bool _iniciando = false;
|
||||
String? _perfilSeleccionado;
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final l10n = AppLocalizations.of(context)!;
|
||||
final nearby = context.watch<ServicioNearby>();
|
||||
final jugadores = nearby.jugadores;
|
||||
final totalJugadores = jugadores.length + 1; // +1 host
|
||||
final sala = nearby.estadoSala;
|
||||
final usuarios = nearby.usuarios;
|
||||
final seleccionados = usuarios.where((u) => u.estaSeleccionado).length;
|
||||
final validacionInicio = sala?.validarInicio();
|
||||
final puedeIniciar = validacionInicio?.exitoso ?? false;
|
||||
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
@@ -47,7 +49,6 @@ class _PantallaLobbyHostState extends State<PantallaLobbyHost> {
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: Column(
|
||||
children: [
|
||||
// QR Code
|
||||
Container(
|
||||
padding: const EdgeInsets.all(16),
|
||||
decoration: BoxDecoration(
|
||||
@@ -57,174 +58,66 @@ class _PantallaLobbyHostState extends State<PantallaLobbyHost> {
|
||||
child: QrImageView(
|
||||
data: nearby.generarDatosQR(widget.nombreSala),
|
||||
version: QrVersions.auto,
|
||||
size: 180,
|
||||
size: 160,
|
||||
backgroundColor: Colors.white,
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
Text(
|
||||
l10n.scanToJoin,
|
||||
style: Theme.of(context).textTheme.bodyMedium,
|
||||
),
|
||||
const SizedBox(height: 24),
|
||||
|
||||
// Selección de perfil
|
||||
Card(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
l10n.selectYourProfile,
|
||||
style: Theme.of(context).textTheme.titleMedium,
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
DropdownButtonFormField<String>(
|
||||
initialValue: _perfilSeleccionado,
|
||||
decoration: InputDecoration(
|
||||
prefixIcon: const Icon(Icons.person),
|
||||
hintText: l10n.selectProfile,
|
||||
border: const OutlineInputBorder(),
|
||||
),
|
||||
items: [
|
||||
// Opción para crear nuevo usuario
|
||||
DropdownMenuItem<String>(
|
||||
value: '_new_',
|
||||
child: Row(
|
||||
children: [
|
||||
const Icon(Icons.add, size: 18),
|
||||
const SizedBox(width: 8),
|
||||
Text(l10n.createNewUser),
|
||||
],
|
||||
),
|
||||
),
|
||||
// Usuarios existentes
|
||||
...nearby.usuarios.map((usuario) {
|
||||
return DropdownMenuItem<String>(
|
||||
value: usuario.nombre,
|
||||
child: Row(
|
||||
children: [
|
||||
Text(usuario.avatar ?? '👤'),
|
||||
const SizedBox(width: 8),
|
||||
Text(usuario.nombre),
|
||||
],
|
||||
),
|
||||
);
|
||||
}),
|
||||
],
|
||||
onChanged: (valor) {
|
||||
if (valor == '_new_') {
|
||||
_crearNuevoUsuario(context);
|
||||
} else {
|
||||
setState(() => _perfilSeleccionado = valor);
|
||||
}
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
Text(l10n.scanToJoin),
|
||||
const SizedBox(height: 16),
|
||||
|
||||
// Lista de jugadores
|
||||
_buildResumenSala(context, seleccionados, nearby.jugadores.length),
|
||||
const SizedBox(height: 12),
|
||||
Expanded(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Row(
|
||||
child: Card(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(12),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
l10n.connectedPlayers,
|
||||
style: Theme.of(context).textTheme.titleLarge,
|
||||
),
|
||||
const Spacer(),
|
||||
Container(
|
||||
padding: const EdgeInsets.symmetric(
|
||||
horizontal: 12,
|
||||
vertical: 4,
|
||||
),
|
||||
decoration: BoxDecoration(
|
||||
color: totalJugadores >= 3
|
||||
? TemaApp.colorVerde.withValues(alpha: 0.2)
|
||||
: TemaApp.colorNaranja.withValues(alpha: 0.2),
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
),
|
||||
child: Text(
|
||||
'$totalJugadores',
|
||||
style: TextStyle(
|
||||
color: totalJugadores >= 3
|
||||
? TemaApp.colorVerde
|
||||
: TemaApp.colorNaranja,
|
||||
fontWeight: FontWeight.bold,
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: Text(
|
||||
'Usuarios de la partida',
|
||||
style: Theme.of(context).textTheme.titleLarge,
|
||||
),
|
||||
),
|
||||
),
|
||||
IconButton.filledTonal(
|
||||
onPressed: () => _crearNuevoUsuario(context),
|
||||
icon: const Icon(Icons.person_add),
|
||||
),
|
||||
],
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
Expanded(
|
||||
child: usuarios.isEmpty
|
||||
? Center(child: Text(l10n.waitingForPlayers))
|
||||
: ListView.builder(
|
||||
itemCount: usuarios.length,
|
||||
itemBuilder: (context, index) =>
|
||||
_buildUsuarioTile(context, usuarios[index]),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
|
||||
// Host (yo)
|
||||
_buildJugadorTile(
|
||||
nombre: nearby.miNombre ?? 'Host',
|
||||
esHost: true,
|
||||
),
|
||||
|
||||
// Jugadores conectados
|
||||
Expanded(
|
||||
child: jugadores.isEmpty
|
||||
? Center(
|
||||
child: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
const Text(
|
||||
'📱',
|
||||
style: TextStyle(fontSize: 48),
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
Text(
|
||||
l10n.waitingForPlayers,
|
||||
style: Theme.of(context).textTheme.bodyLarge,
|
||||
),
|
||||
],
|
||||
),
|
||||
)
|
||||
: ListView.builder(
|
||||
itemCount: jugadores.length,
|
||||
itemBuilder: (context, index) {
|
||||
final j = jugadores[index];
|
||||
return _buildJugadorTile(nombre: j.nombre);
|
||||
},
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
|
||||
// Botón iniciar
|
||||
if (totalJugadores < 3)
|
||||
Text(
|
||||
l10n.needMorePlayers(3 - totalJugadores),
|
||||
style: Theme.of(
|
||||
context,
|
||||
).textTheme.bodyMedium?.copyWith(color: TemaApp.colorNaranja),
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
if (_perfilSeleccionado == null)
|
||||
if (!puedeIniciar)
|
||||
Text(
|
||||
l10n.selectProfile,
|
||||
style: Theme.of(
|
||||
context,
|
||||
).textTheme.bodyMedium?.copyWith(color: TemaApp.colorNaranja),
|
||||
_mensajeValidacion(validacionInicio?.codigo),
|
||||
style: Theme.of(context)
|
||||
.textTheme
|
||||
.bodyMedium
|
||||
?.copyWith(color: TemaApp.colorNaranja),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
const SizedBox(height: 12),
|
||||
SizedBox(
|
||||
width: double.infinity,
|
||||
child: ElevatedButton.icon(
|
||||
onPressed:
|
||||
totalJugadores >= 3 &&
|
||||
_perfilSeleccionado != null &&
|
||||
!_iniciando
|
||||
onPressed: puedeIniciar && !_iniciando
|
||||
? () {
|
||||
setState(() => _iniciando = true);
|
||||
widget.onIniciar();
|
||||
@@ -240,45 +133,129 @@ class _PantallaLobbyHostState extends State<PantallaLobbyHost> {
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildJugadorTile({required String nombre, bool esHost = false}) {
|
||||
Widget _buildResumenSala(
|
||||
BuildContext context,
|
||||
int seleccionados,
|
||||
int clientesRemotos,
|
||||
) {
|
||||
return Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: _buildStat(
|
||||
context,
|
||||
icon: Icons.groups,
|
||||
label: 'Jugadores seleccionados',
|
||||
value: '$seleccionados',
|
||||
ok: seleccionados >= 3,
|
||||
),
|
||||
),
|
||||
const SizedBox(width: 8),
|
||||
Expanded(
|
||||
child: _buildStat(
|
||||
context,
|
||||
icon: Icons.devices,
|
||||
label: 'Móviles conectados',
|
||||
value: '${clientesRemotos + 1}',
|
||||
ok: true,
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildStat(
|
||||
BuildContext context, {
|
||||
required IconData icon,
|
||||
required String label,
|
||||
required String value,
|
||||
required bool ok,
|
||||
}) {
|
||||
final color = ok ? TemaApp.colorVerde : TemaApp.colorNaranja;
|
||||
return Container(
|
||||
margin: const EdgeInsets.only(bottom: 8),
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
|
||||
padding: const EdgeInsets.all(12),
|
||||
decoration: BoxDecoration(
|
||||
color: TemaApp.colorTarjeta,
|
||||
color: color.withValues(alpha: 0.18),
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
border: esHost
|
||||
? Border.all(color: TemaApp.colorAcento.withValues(alpha: 0.5))
|
||||
: null,
|
||||
),
|
||||
child: Row(
|
||||
children: [
|
||||
Text(esHost ? '👑' : '🎭', style: const TextStyle(fontSize: 20)),
|
||||
const SizedBox(width: 12),
|
||||
Icon(icon, color: color),
|
||||
const SizedBox(width: 8),
|
||||
Expanded(
|
||||
child: Text(nombre, style: Theme.of(context).textTheme.titleMedium),
|
||||
child: Text(label, style: Theme.of(context).textTheme.bodySmall),
|
||||
),
|
||||
if (esHost)
|
||||
Container(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 2),
|
||||
decoration: BoxDecoration(
|
||||
color: TemaApp.colorAcento.withValues(alpha: 0.2),
|
||||
borderRadius: BorderRadius.circular(8),
|
||||
),
|
||||
child: const Text(
|
||||
'HOST',
|
||||
style: TextStyle(
|
||||
color: TemaApp.colorAcento,
|
||||
fontSize: 10,
|
||||
fontWeight: FontWeight.bold,
|
||||
),
|
||||
),
|
||||
Text(
|
||||
value,
|
||||
style: TextStyle(color: color, fontWeight: FontWeight.bold),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildUsuarioTile(BuildContext context, Usuario usuario) {
|
||||
final nearby = context.read<ServicioNearby>();
|
||||
final miClientId = nearby.miClientId;
|
||||
final seleccionadoPorMi = usuario.clienteIdSeleccionado == miClientId;
|
||||
final seleccionadoPorOtro =
|
||||
usuario.estaSeleccionado && usuario.clienteIdSeleccionado != miClientId;
|
||||
|
||||
return ListTile(
|
||||
leading: CircleAvatar(
|
||||
backgroundColor: seleccionadoPorMi
|
||||
? TemaApp.colorVerde
|
||||
: seleccionadoPorOtro
|
||||
? TemaApp.colorNaranja
|
||||
: TemaApp.colorTarjeta,
|
||||
child: Text(usuario.avatar ?? '👤'),
|
||||
),
|
||||
title: Text(usuario.nombre),
|
||||
subtitle: Text(
|
||||
seleccionadoPorMi
|
||||
? 'Seleccionado por este móvil'
|
||||
: seleccionadoPorOtro
|
||||
? 'Seleccionado por otro cliente'
|
||||
: 'Disponible',
|
||||
),
|
||||
trailing: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
if (seleccionadoPorMi)
|
||||
IconButton(
|
||||
tooltip: 'Liberar',
|
||||
icon: const Icon(Icons.close),
|
||||
onPressed: () => nearby.liberarUsuarioSala(usuario.id),
|
||||
)
|
||||
else if (!seleccionadoPorOtro)
|
||||
IconButton(
|
||||
tooltip: 'Seleccionar',
|
||||
icon: const Icon(Icons.check_circle_outline),
|
||||
onPressed: () => nearby.seleccionarUsuarioSala(usuario.id),
|
||||
),
|
||||
if (!usuario.estaSeleccionado)
|
||||
IconButton(
|
||||
tooltip: 'Eliminar',
|
||||
icon: const Icon(Icons.delete_outline, color: TemaApp.colorAcento),
|
||||
onPressed: () => nearby.eliminarUsuarioSala(usuario.id),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
String _mensajeValidacion(String? codigo) {
|
||||
switch (codigo) {
|
||||
case 'faltan_jugadores':
|
||||
return 'Seleccioná al menos 3 usuarios para iniciar.';
|
||||
case 'host_sin_usuario':
|
||||
return 'El móvil servidor debe seleccionar al menos un usuario.';
|
||||
case 'sala_cerrada':
|
||||
return 'La sala ya no está en lobby.';
|
||||
default:
|
||||
return 'Completá la selección de usuarios para iniciar.';
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _crearNuevoUsuario(BuildContext context) async {
|
||||
final l10n = AppLocalizations.of(context)!;
|
||||
final controller = TextEditingController();
|
||||
@@ -312,12 +289,7 @@ class _PantallaLobbyHostState extends State<PantallaLobbyHost> {
|
||||
);
|
||||
|
||||
if (nombre != null && nombre.trim().isNotEmpty) {
|
||||
final nuevoUsuario = Usuario(
|
||||
id: DateTime.now().millisecondsSinceEpoch.toString(),
|
||||
nombre: nombre.trim(),
|
||||
);
|
||||
nearby.agregarUsuario(nuevoUsuario);
|
||||
setState(() => _perfilSeleccionado = nombre.trim());
|
||||
await nearby.crearUsuarioSala(nombre.trim(), seleccionar: true);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
128
lib/pantallas/pantalla_palabras_cliente.dart
Normal file
128
lib/pantallas/pantalla_palabras_cliente.dart
Normal file
@@ -0,0 +1,128 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:farolero/l10n/generated/app_localizations.dart';
|
||||
import 'package:farolero/modelos/inicio_partida_multijugador.dart';
|
||||
import 'package:farolero/tema/tema_app.dart';
|
||||
|
||||
/// Reveal secuencial para clientes que manejan uno o varios jugadores.
|
||||
class PantallaPalabrasCliente extends StatefulWidget {
|
||||
final List<JugadorInicioPartida> jugadores;
|
||||
final String? pistaCategoria;
|
||||
final VoidCallback onTodosVistos;
|
||||
|
||||
const PantallaPalabrasCliente({
|
||||
super.key,
|
||||
required this.jugadores,
|
||||
this.pistaCategoria,
|
||||
required this.onTodosVistos,
|
||||
});
|
||||
|
||||
@override
|
||||
State<PantallaPalabrasCliente> createState() => _PantallaPalabrasClienteState();
|
||||
}
|
||||
|
||||
class _PantallaPalabrasClienteState extends State<PantallaPalabrasCliente> {
|
||||
int _indice = 0;
|
||||
bool _visible = false;
|
||||
|
||||
JugadorInicioPartida get _actual => widget.jugadores[_indice];
|
||||
bool get _esUltimo => _indice == widget.jugadores.length - 1;
|
||||
|
||||
void _continuar() {
|
||||
if (_esUltimo) {
|
||||
widget.onTodosVistos();
|
||||
return;
|
||||
}
|
||||
setState(() {
|
||||
_indice++;
|
||||
_visible = false;
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final l10n = AppLocalizations.of(context)!;
|
||||
final actual = _actual;
|
||||
|
||||
return Scaffold(
|
||||
backgroundColor: TemaApp.colorFondo,
|
||||
body: SafeArea(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(24),
|
||||
child: Column(
|
||||
children: [
|
||||
Text(
|
||||
'Jugador ${_indice + 1} de ${widget.jugadores.length}',
|
||||
style: Theme.of(context).textTheme.titleMedium,
|
||||
),
|
||||
const Spacer(),
|
||||
Text(
|
||||
actual.nombre,
|
||||
style: Theme.of(context).textTheme.headlineMedium,
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
const SizedBox(height: 24),
|
||||
GestureDetector(
|
||||
onTap: () => setState(() => _visible = !_visible),
|
||||
child: AnimatedContainer(
|
||||
duration: const Duration(milliseconds: 250),
|
||||
width: double.infinity,
|
||||
padding: const EdgeInsets.symmetric(
|
||||
vertical: 48,
|
||||
horizontal: 24,
|
||||
),
|
||||
decoration: BoxDecoration(
|
||||
color: _visible ? TemaApp.colorAcento : TemaApp.colorTarjeta,
|
||||
borderRadius: BorderRadius.circular(24),
|
||||
),
|
||||
child: Column(
|
||||
children: [
|
||||
Icon(
|
||||
_visible ? Icons.visibility : Icons.visibility_off,
|
||||
color: _visible ? Colors.white : TemaApp.colorTextoSecundario,
|
||||
size: 32,
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
Text(
|
||||
_visible
|
||||
? (actual.esImpostor
|
||||
? l10n.youAreImpostor
|
||||
: actual.palabra ?? '')
|
||||
: '???',
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(
|
||||
fontSize: 32,
|
||||
fontWeight: FontWeight.bold,
|
||||
color: _visible
|
||||
? Colors.white
|
||||
: TemaApp.colorTextoSecundario,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
if (_visible && actual.esImpostor && widget.pistaCategoria != null) ...[
|
||||
const SizedBox(height: 12),
|
||||
Text(
|
||||
l10n.clueIs(widget.pistaCategoria!),
|
||||
style: const TextStyle(color: TemaApp.colorNaranja),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
],
|
||||
const Spacer(),
|
||||
SizedBox(
|
||||
width: double.infinity,
|
||||
height: 56,
|
||||
child: ElevatedButton.icon(
|
||||
onPressed: _continuar,
|
||||
icon: Icon(_esUltimo ? Icons.check : Icons.arrow_forward),
|
||||
label: Text(_esUltimo ? l10n.iveSeenIt : 'Siguiente jugador'),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -3,11 +3,13 @@ import 'package:mobile_scanner/mobile_scanner.dart';
|
||||
import 'package:provider/provider.dart';
|
||||
import 'package:farolero/l10n/generated/app_localizations.dart';
|
||||
import '../modelos/jugador.dart';
|
||||
import '../modelos/inicio_partida_multijugador.dart';
|
||||
import '../modelos/usuario.dart';
|
||||
import '../servicios/servicio_nearby.dart';
|
||||
import '../servicios/servicio_permisos.dart';
|
||||
import '../tema/tema_app.dart';
|
||||
import 'pantalla_palabra_cliente.dart';
|
||||
import 'pantalla_palabras_cliente.dart';
|
||||
import 'pantalla_debate_cliente.dart';
|
||||
import 'pantalla_votacion_cliente.dart';
|
||||
|
||||
@@ -36,6 +38,7 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
||||
bool _esImpostor = false;
|
||||
String? _pistaCategoria;
|
||||
final List<Jugador> _jugadores = [];
|
||||
final List<JugadorInicioPartida> _jugadoresControlados = [];
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
@@ -51,13 +54,38 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
||||
nearby.onMensaje((endpointId, mensaje) {
|
||||
if (mensaje.tipo == TipoMensaje.partidaInicio) {
|
||||
// El host ha iniciado la partida — nos ha enviado nuestra palabra
|
||||
final jugadoresData = mensaje.datos['jugadores'] as List<dynamic>?;
|
||||
final jugadoresTodosData =
|
||||
mensaje.datos['jugadoresTodos'] as List<dynamic>?;
|
||||
setState(() {
|
||||
_palabraRecibida = mensaje.datos['palabra'] as String?;
|
||||
_esImpostor = mensaje.datos['esImpostor'] as bool? ?? false;
|
||||
_jugadoresControlados
|
||||
..clear()
|
||||
..addAll(
|
||||
(jugadoresData ?? []).map(
|
||||
(json) => JugadorInicioPartida.fromJson(
|
||||
json as Map<String, dynamic>,
|
||||
),
|
||||
),
|
||||
);
|
||||
_jugadores
|
||||
..clear()
|
||||
..addAll(
|
||||
(jugadoresTodosData ?? []).map(
|
||||
(json) => Jugador.fromJson(json as Map<String, dynamic>),
|
||||
),
|
||||
);
|
||||
if (_jugadoresControlados.isNotEmpty) {
|
||||
final primero = _jugadoresControlados.first;
|
||||
_palabraRecibida = primero.palabra;
|
||||
_esImpostor = primero.esImpostor;
|
||||
} else {
|
||||
_palabraRecibida = mensaje.datos['palabra'] as String?;
|
||||
_esImpostor = mensaje.datos['esImpostor'] as bool? ?? false;
|
||||
}
|
||||
_pistaCategoria = mensaje.datos['categoria'] as String?;
|
||||
});
|
||||
// Navegar a pantalla de palabra del cliente
|
||||
if (mounted && _palabraRecibida != null) {
|
||||
if (mounted && (_jugadoresControlados.isNotEmpty || _palabraRecibida != null)) {
|
||||
_navegarAPalabra();
|
||||
}
|
||||
} else if (mensaje.tipo == TipoMensaje.fase) {
|
||||
@@ -71,6 +99,28 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
||||
}
|
||||
|
||||
void _navegarAPalabra() {
|
||||
if (_jugadoresControlados.isNotEmpty) {
|
||||
Navigator.of(context).push(
|
||||
MaterialPageRoute(
|
||||
builder: (_) => PantallaPalabrasCliente(
|
||||
jugadores: List.unmodifiable(_jugadoresControlados),
|
||||
pistaCategoria: _pistaCategoria,
|
||||
onTodosVistos: () {
|
||||
final nearby = context.read<ServicioNearby>();
|
||||
if (nearby.hostEndpointId != null) {
|
||||
nearby.enviarMensaje(
|
||||
nearby.hostEndpointId!,
|
||||
MensajeP2P(tipo: TipoMensaje.listo, datos: {}),
|
||||
);
|
||||
}
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
),
|
||||
),
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
Navigator.of(context).push(
|
||||
MaterialPageRoute(
|
||||
builder: (_) => PantallaPalabraCliente(
|
||||
@@ -121,16 +171,23 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
||||
MaterialPageRoute(
|
||||
builder: (_) => PantallaVotacionCliente(
|
||||
jugadores: _jugadores,
|
||||
onVoto: (votoporId) {
|
||||
jugadoresControlados: List.unmodifiable(_jugadoresControlados),
|
||||
onVotos: (votos) {
|
||||
final nearby = context.read<ServicioNearby>();
|
||||
if (nearby.hostEndpointId != null) {
|
||||
nearby.enviarMensaje(
|
||||
nearby.hostEndpointId!,
|
||||
MensajeP2P(
|
||||
tipo: TipoMensaje.voto,
|
||||
datos: {'votoporId': votoporId},
|
||||
),
|
||||
);
|
||||
for (final entry in votos.entries) {
|
||||
nearby.enviarMensaje(
|
||||
nearby.hostEndpointId!,
|
||||
MensajeP2P(
|
||||
tipo: TipoMensaje.voto,
|
||||
datos: {
|
||||
'votanteId': entry.key,
|
||||
'votadoId': entry.value,
|
||||
'votoporId': entry.value,
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
@@ -605,19 +662,7 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
||||
),
|
||||
const Divider(),
|
||||
// Usuarios existentes
|
||||
...usuarios.map(
|
||||
(usuario) => ListTile(
|
||||
leading: Text(
|
||||
usuario.avatar ?? '👤',
|
||||
style: const TextStyle(fontSize: 24),
|
||||
),
|
||||
title: Text(usuario.nombre),
|
||||
onTap: () {
|
||||
// Seleccionar usuario - enviar al host
|
||||
_enviarUsuarioAlHost(usuario);
|
||||
},
|
||||
),
|
||||
),
|
||||
...usuarios.map(_buildUsuarioSalaTile),
|
||||
],
|
||||
),
|
||||
),
|
||||
@@ -671,34 +716,57 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
||||
);
|
||||
|
||||
if (nombre != null && nombre.trim().isNotEmpty) {
|
||||
final nuevoUsuario = Usuario(
|
||||
id: DateTime.now().millisecondsSinceEpoch.toString(),
|
||||
nombre: nombre.trim(),
|
||||
);
|
||||
// Agregar localmente
|
||||
nearby.agregarUsuario(nuevoUsuario);
|
||||
// Enviar al host
|
||||
_enviarUsuarioAlHost(nuevoUsuario);
|
||||
await nearby.crearUsuarioSala(nombre.trim(), seleccionar: true);
|
||||
}
|
||||
}
|
||||
|
||||
/// Envía el usuario seleccionado/creado al host
|
||||
/// Env?a el usuario seleccionado/creado al host
|
||||
void _enviarUsuarioAlHost(Usuario usuario) {
|
||||
final nearby = context.read<ServicioNearby>();
|
||||
if (nearby.hostEndpointId != null) {
|
||||
nearby.enviarMensaje(
|
||||
nearby.hostEndpointId!,
|
||||
MensajeP2P(
|
||||
tipo: TipoMensaje.usuarioNuevo,
|
||||
datos: {'usuario': usuario.toJson()},
|
||||
),
|
||||
);
|
||||
ScaffoldMessenger.of(
|
||||
context,
|
||||
).showSnackBar(SnackBar(content: Text('${usuario.nombre} seleccionado')));
|
||||
}
|
||||
nearby.seleccionarUsuarioSala(usuario.id);
|
||||
ScaffoldMessenger.of(
|
||||
context,
|
||||
).showSnackBar(SnackBar(content: Text('${usuario.nombre} seleccionado')));
|
||||
}
|
||||
|
||||
Widget _buildUsuarioSalaTile(Usuario usuario) {
|
||||
final nearby = context.read<ServicioNearby>();
|
||||
final miClientId = nearby.miClientId;
|
||||
final seleccionadoPorMi = usuario.clienteIdSeleccionado == miClientId;
|
||||
final seleccionadoPorOtro =
|
||||
usuario.estaSeleccionado && usuario.clienteIdSeleccionado != miClientId;
|
||||
|
||||
return ListTile(
|
||||
leading: Text(
|
||||
usuario.avatar ?? '??',
|
||||
style: const TextStyle(fontSize: 24),
|
||||
),
|
||||
title: Text(usuario.nombre),
|
||||
subtitle: Text(
|
||||
seleccionadoPorMi
|
||||
? 'Seleccionado por este m?vil'
|
||||
: seleccionadoPorOtro
|
||||
? 'No disponible'
|
||||
: 'Disponible',
|
||||
),
|
||||
trailing: seleccionadoPorMi
|
||||
? IconButton(
|
||||
icon: const Icon(Icons.close),
|
||||
onPressed: () => nearby.liberarUsuarioSala(usuario.id),
|
||||
)
|
||||
: null,
|
||||
enabled: !seleccionadoPorOtro,
|
||||
onTap: seleccionadoPorOtro
|
||||
? null
|
||||
: () {
|
||||
if (seleccionadoPorMi) {
|
||||
nearby.liberarUsuarioSala(usuario.id);
|
||||
} else {
|
||||
_enviarUsuarioAlHost(usuario);
|
||||
}
|
||||
},
|
||||
);
|
||||
}
|
||||
// ==================== HELPERS ====================
|
||||
|
||||
Widget _buildError(String msg) {
|
||||
|
||||
@@ -1,18 +1,22 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:farolero/l10n/generated/app_localizations.dart';
|
||||
import 'package:farolero/modelos/inicio_partida_multijugador.dart';
|
||||
import 'package:farolero/modelos/jugador.dart';
|
||||
import 'package:farolero/tema/tema_app.dart';
|
||||
|
||||
/// Pantalla de votación para el cliente (multidispositivo).
|
||||
/// El cliente recibe fase=votacion y ve esta pantalla para elegir a quién votar.
|
||||
/// Pantalla de votación para cliente multidispositivo.
|
||||
/// Un cliente puede manejar uno o varios jugadores, por eso se recoge un voto
|
||||
/// por cada jugador controlado activo.
|
||||
class PantallaVotacionCliente extends StatefulWidget {
|
||||
final List<Jugador> jugadores;
|
||||
final Function(String votoporId) onVoto;
|
||||
final List<JugadorInicioPartida> jugadoresControlados;
|
||||
final Function(Map<String, String> votos) onVotos;
|
||||
|
||||
const PantallaVotacionCliente({
|
||||
super.key,
|
||||
required this.jugadores,
|
||||
required this.onVoto,
|
||||
this.jugadoresControlados = const [],
|
||||
required this.onVotos,
|
||||
});
|
||||
|
||||
@override
|
||||
@@ -20,7 +24,14 @@ class PantallaVotacionCliente extends StatefulWidget {
|
||||
}
|
||||
|
||||
class _PantallaVotacionClienteState extends State<PantallaVotacionCliente> {
|
||||
String? _votoSeleccionado;
|
||||
final Map<String, String> _votosPorVotante = {};
|
||||
|
||||
List<JugadorInicioPartida> get _votantes => widget.jugadoresControlados;
|
||||
bool get _modoMultiVotante => _votantes.length > 1;
|
||||
bool get _votacionCompleta {
|
||||
if (_votantes.isEmpty) return _votosPorVotante.containsKey('_legacy');
|
||||
return _votantes.every((votante) => _votosPorVotante[votante.jugadorId] != null);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
@@ -45,56 +56,31 @@ class _PantallaVotacionClienteState extends State<PantallaVotacionCliente> {
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
Text(
|
||||
l10n.selectOnePlayer,
|
||||
_modoMultiVotante
|
||||
? 'Emití un voto por cada jugador que manejás.'
|
||||
: l10n.selectOnePlayer,
|
||||
style: TextStyle(color: TemaApp.colorTextoSecundario),
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
Expanded(
|
||||
child: ListView.builder(
|
||||
itemCount: widget.jugadores.length,
|
||||
itemBuilder: (context, index) {
|
||||
final jugador = widget.jugadores[index];
|
||||
final selected = _votoSeleccionado == jugador.id;
|
||||
return Card(
|
||||
color: selected
|
||||
? TemaApp.colorAcento.withValues(alpha: 0.3)
|
||||
: TemaApp.colorTarjeta,
|
||||
margin: const EdgeInsets.only(bottom: 8),
|
||||
child: ListTile(
|
||||
leading: CircleAvatar(
|
||||
backgroundColor: selected
|
||||
? TemaApp.colorAcento
|
||||
: TemaApp.colorAcento.withValues(alpha: 0.3),
|
||||
child: Text(
|
||||
'${index + 1}',
|
||||
style: TextStyle(
|
||||
color: selected
|
||||
? Colors.white
|
||||
: TemaApp.colorTexto,
|
||||
),
|
||||
),
|
||||
),
|
||||
title: Text(jugador.nombre),
|
||||
trailing: selected
|
||||
? const Icon(Icons.check_circle,
|
||||
color: TemaApp.colorAcento)
|
||||
: null,
|
||||
onTap: () {
|
||||
setState(() => _votoSeleccionado = jugador.id);
|
||||
child: _votantes.isEmpty
|
||||
? _buildSelectorLegacy()
|
||||
: ListView.builder(
|
||||
itemCount: _votantes.length,
|
||||
itemBuilder: (context, index) {
|
||||
final votante = _votantes[index];
|
||||
return _buildSelectorParaVotante(context, votante);
|
||||
},
|
||||
),
|
||||
);
|
||||
},
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 16),
|
||||
SizedBox(
|
||||
width: double.infinity,
|
||||
height: 56,
|
||||
child: ElevatedButton.icon(
|
||||
onPressed: _votoSeleccionado == null
|
||||
? null
|
||||
: () => widget.onVoto(_votoSeleccionado!),
|
||||
onPressed: _votacionCompleta
|
||||
? () => widget.onVotos(Map.unmodifiable(_votosPorVotante))
|
||||
: null,
|
||||
icon: const Icon(Icons.how_to_vote),
|
||||
label: Text(l10n.votar),
|
||||
style: ElevatedButton.styleFrom(
|
||||
@@ -109,4 +95,87 @@ class _PantallaVotacionClienteState extends State<PantallaVotacionCliente> {
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildSelectorLegacy() {
|
||||
return ListView.builder(
|
||||
itemCount: widget.jugadores.length,
|
||||
itemBuilder: (context, index) {
|
||||
final jugador = widget.jugadores[index];
|
||||
final selected = _votosPorVotante['_legacy'] == jugador.id;
|
||||
return _buildJugadorVotable(
|
||||
jugador: jugador,
|
||||
index: index,
|
||||
selected: selected,
|
||||
onTap: () => setState(() => _votosPorVotante['_legacy'] = jugador.id),
|
||||
);
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildSelectorParaVotante(
|
||||
BuildContext context,
|
||||
JugadorInicioPartida votante,
|
||||
) {
|
||||
return Card(
|
||||
color: TemaApp.colorSuperficie,
|
||||
margin: const EdgeInsets.only(bottom: 16),
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(12),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
'Voto de ${votante.nombre}',
|
||||
style: Theme.of(context).textTheme.titleMedium,
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
...widget.jugadores.asMap().entries.map((entry) {
|
||||
final jugador = entry.value;
|
||||
final selected = _votosPorVotante[votante.jugadorId] == jugador.id;
|
||||
return _buildJugadorVotable(
|
||||
jugador: jugador,
|
||||
index: entry.key,
|
||||
selected: selected,
|
||||
onTap: () => setState(
|
||||
() => _votosPorVotante[votante.jugadorId] = jugador.id,
|
||||
),
|
||||
);
|
||||
}),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildJugadorVotable({
|
||||
required Jugador jugador,
|
||||
required int index,
|
||||
required bool selected,
|
||||
required VoidCallback onTap,
|
||||
}) {
|
||||
return Card(
|
||||
color: selected
|
||||
? TemaApp.colorAcento.withValues(alpha: 0.3)
|
||||
: TemaApp.colorTarjeta,
|
||||
margin: const EdgeInsets.only(bottom: 8),
|
||||
child: ListTile(
|
||||
leading: CircleAvatar(
|
||||
backgroundColor: selected
|
||||
? TemaApp.colorAcento
|
||||
: TemaApp.colorAcento.withValues(alpha: 0.3),
|
||||
child: Text(
|
||||
'${index + 1}',
|
||||
style: TextStyle(
|
||||
color: selected ? Colors.white : TemaApp.colorTexto,
|
||||
),
|
||||
),
|
||||
),
|
||||
title: Text(jugador.nombre),
|
||||
trailing: selected
|
||||
? const Icon(Icons.check_circle, color: TemaApp.colorAcento)
|
||||
: null,
|
||||
onTap: onTap,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user