feat: discovery automático + QR como fallback en PantallaUnirse
- Discovery: busca hosts cercanos automáticamente y los muestra en lista - Cada host aparece como tile tocable con nombre de la sala - QR fallback: botón 'Escanear QR' debajo de la lista - ServicioNearby: hostsEncontrados map, pararBusqueda(), no auto-connect - Flujo: nombre → buscar → lista de salas (o QR) → conectar → espera - l10n: searchGames, searchingGames, noGamesFound, orScanQR (es/en)
This commit is contained in:
@@ -249,5 +249,11 @@
|
|||||||
"scanQR": "Scan QR",
|
"scanQR": "Scan QR",
|
||||||
"scanHostQR": "Point at the host's QR code",
|
"scanHostQR": "Point at the host's QR code",
|
||||||
"connectedWaiting": "Connected!",
|
"connectedWaiting": "Connected!",
|
||||||
"waitingForHost": "Waiting for the host to start the game..."
|
"waitingForHost": "Waiting for the host to start the game...",
|
||||||
|
"enterNameToSearch": "Enter your name to search for nearby games",
|
||||||
|
"searchGames": "Search games",
|
||||||
|
"searchingGames": "Searching for nearby games...",
|
||||||
|
"noGamesFound": "No games found",
|
||||||
|
"noGamesFoundHint": "Make sure the host has the room open and you are nearby",
|
||||||
|
"orScanQR": "Not showing up? Scan the host's QR code"
|
||||||
}
|
}
|
||||||
@@ -249,5 +249,11 @@
|
|||||||
"scanQR": "Escanear QR",
|
"scanQR": "Escanear QR",
|
||||||
"scanHostQR": "Apunta al QR del host",
|
"scanHostQR": "Apunta al QR del host",
|
||||||
"connectedWaiting": "¡Conectado!",
|
"connectedWaiting": "¡Conectado!",
|
||||||
"waitingForHost": "Esperando a que el host inicie la partida..."
|
"waitingForHost": "Esperando a que el host inicie la partida...",
|
||||||
|
"enterNameToSearch": "Escribe tu nombre para buscar partidas cercanas",
|
||||||
|
"searchGames": "Buscar partidas",
|
||||||
|
"searchingGames": "Buscando partidas cercanas...",
|
||||||
|
"noGamesFound": "No se encontraron partidas",
|
||||||
|
"noGamesFoundHint": "Asegúrate de que el host tiene la sala abierta y estáis cerca",
|
||||||
|
"orScanQR": "¿No aparece? Escanea el QR del host"
|
||||||
}
|
}
|
||||||
@@ -1070,6 +1070,42 @@ abstract class AppLocalizations {
|
|||||||
/// In es, this message translates to:
|
/// In es, this message translates to:
|
||||||
/// **'Esperando a que el host inicie la partida...'**
|
/// **'Esperando a que el host inicie la partida...'**
|
||||||
String get waitingForHost;
|
String get waitingForHost;
|
||||||
|
|
||||||
|
/// No description provided for @enterNameToSearch.
|
||||||
|
///
|
||||||
|
/// In es, this message translates to:
|
||||||
|
/// **'Escribe tu nombre para buscar partidas cercanas'**
|
||||||
|
String get enterNameToSearch;
|
||||||
|
|
||||||
|
/// No description provided for @searchGames.
|
||||||
|
///
|
||||||
|
/// In es, this message translates to:
|
||||||
|
/// **'Buscar partidas'**
|
||||||
|
String get searchGames;
|
||||||
|
|
||||||
|
/// No description provided for @searchingGames.
|
||||||
|
///
|
||||||
|
/// In es, this message translates to:
|
||||||
|
/// **'Buscando partidas cercanas...'**
|
||||||
|
String get searchingGames;
|
||||||
|
|
||||||
|
/// No description provided for @noGamesFound.
|
||||||
|
///
|
||||||
|
/// In es, this message translates to:
|
||||||
|
/// **'No se encontraron partidas'**
|
||||||
|
String get noGamesFound;
|
||||||
|
|
||||||
|
/// No description provided for @noGamesFoundHint.
|
||||||
|
///
|
||||||
|
/// In es, this message translates to:
|
||||||
|
/// **'Asegúrate de que el host tiene la sala abierta y estáis cerca'**
|
||||||
|
String get noGamesFoundHint;
|
||||||
|
|
||||||
|
/// No description provided for @orScanQR.
|
||||||
|
///
|
||||||
|
/// In es, this message translates to:
|
||||||
|
/// **'¿No aparece? Escanea el QR del host'**
|
||||||
|
String get orScanQR;
|
||||||
}
|
}
|
||||||
|
|
||||||
class _AppLocalizationsDelegate
|
class _AppLocalizationsDelegate
|
||||||
|
|||||||
@@ -511,4 +511,24 @@ class AppLocalizationsAr extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -514,4 +514,24 @@ class AppLocalizationsCa extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -517,4 +517,24 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -512,4 +512,23 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Waiting for the host to start the game...';
|
String get waitingForHost => 'Waiting for the host to start the game...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch => 'Enter your name to search for nearby games';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Search games';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Searching for nearby games...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No games found';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Make sure the host has the room open and you are nearby';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => 'Not showing up? Scan the host\'s QR code';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -513,4 +513,24 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -516,4 +516,24 @@ class AppLocalizationsEu extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -514,4 +514,24 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -513,4 +513,24 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -514,4 +514,24 @@ class AppLocalizationsIt extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -511,4 +511,24 @@ class AppLocalizationsJa extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -511,4 +511,24 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -514,4 +514,24 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -514,4 +514,24 @@ class AppLocalizationsPl extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -515,4 +515,24 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -514,4 +514,24 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -513,4 +513,24 @@ class AppLocalizationsTr extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -510,6 +510,26 @@ class AppLocalizationsZh extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
String get waitingForHost => 'Esperando a que el host inicie la partida...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get enterNameToSearch =>
|
||||||
|
'Escribe tu nombre para buscar partidas cercanas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchGames => 'Buscar partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get searchingGames => 'Buscando partidas cercanas...';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFound => 'No se encontraron partidas';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get noGamesFoundHint =>
|
||||||
|
'Asegúrate de que el host tiene la sala abierta y estáis cerca';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get orScanQR => '¿No aparece? Escanea el QR del host';
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The translations for Chinese, as used in Taiwan (`zh_TW`).
|
/// The translations for Chinese, as used in Taiwan (`zh_TW`).
|
||||||
|
|||||||
@@ -5,7 +5,8 @@ import 'package:farolero/l10n/generated/app_localizations.dart';
|
|||||||
import '../servicios/servicio_nearby.dart';
|
import '../servicios/servicio_nearby.dart';
|
||||||
import '../tema/tema_app.dart';
|
import '../tema/tema_app.dart';
|
||||||
|
|
||||||
/// Pantalla para unirse a una partida multidispositivo
|
/// Pantalla para unirse a una partida multidispositivo.
|
||||||
|
/// Flujo: nombre → discovery automático (lista de salas) → fallback QR
|
||||||
class PantallaUnirse extends StatefulWidget {
|
class PantallaUnirse extends StatefulWidget {
|
||||||
const PantallaUnirse({super.key});
|
const PantallaUnirse({super.key});
|
||||||
|
|
||||||
@@ -16,10 +17,13 @@ class PantallaUnirse extends StatefulWidget {
|
|||||||
class _PantallaUnirseState extends State<PantallaUnirse> {
|
class _PantallaUnirseState extends State<PantallaUnirse> {
|
||||||
final _nombreController = TextEditingController();
|
final _nombreController = TextEditingController();
|
||||||
final _formKey = GlobalKey<FormState>();
|
final _formKey = GlobalKey<FormState>();
|
||||||
bool _escaneando = false;
|
|
||||||
|
// Estados de la pantalla
|
||||||
|
bool _buscando = false;
|
||||||
|
bool _escaneandoQR = false;
|
||||||
bool _conectando = false;
|
bool _conectando = false;
|
||||||
String? _error;
|
String? _error;
|
||||||
String? _salaEncontrada;
|
String? _salaSeleccionada;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void dispose() {
|
void dispose() {
|
||||||
@@ -27,10 +31,51 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
|||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _iniciarEscaneo() async {
|
/// Paso 1: validar nombre e iniciar discovery
|
||||||
|
Future<void> _iniciarBusqueda() async {
|
||||||
if (!_formKey.currentState!.validate()) return;
|
if (!_formKey.currentState!.validate()) return;
|
||||||
|
|
||||||
|
final nearby = context.read<ServicioNearby>();
|
||||||
|
final ok = await nearby.buscarHosts(_nombreController.text.trim());
|
||||||
|
|
||||||
|
if (ok) {
|
||||||
|
setState(() {
|
||||||
|
_buscando = true;
|
||||||
|
_error = null;
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setState(() {
|
||||||
|
_error = 'No se pudo iniciar la búsqueda. Verifica Bluetooth y ubicación.';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Conectar a un host de la lista
|
||||||
|
Future<void> _conectarAHost(String endpointId, String nombreHost) async {
|
||||||
setState(() {
|
setState(() {
|
||||||
_escaneando = true;
|
_conectando = true;
|
||||||
|
_salaSeleccionada = nombreHost;
|
||||||
|
});
|
||||||
|
|
||||||
|
final nearby = context.read<ServicioNearby>();
|
||||||
|
// Parar discovery antes de conectar
|
||||||
|
await nearby.pararBusqueda();
|
||||||
|
final ok = await nearby.conectarAHost(endpointId, _nombreController.text.trim());
|
||||||
|
|
||||||
|
if (!ok && mounted) {
|
||||||
|
setState(() {
|
||||||
|
_conectando = false;
|
||||||
|
_error = 'No se pudo conectar a $nombreHost';
|
||||||
|
});
|
||||||
|
// Reiniciar búsqueda
|
||||||
|
_iniciarBusqueda();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Fallback: escanear QR
|
||||||
|
void _abrirEscaner() {
|
||||||
|
setState(() {
|
||||||
|
_escaneandoQR = true;
|
||||||
_error = null;
|
_error = null;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -45,20 +90,15 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
|||||||
final datos = ServicioNearby.parsearQR(valor);
|
final datos = ServicioNearby.parsearQR(valor);
|
||||||
if (datos != null) {
|
if (datos != null) {
|
||||||
setState(() {
|
setState(() {
|
||||||
_escaneando = false;
|
_escaneandoQR = false;
|
||||||
_conectando = true;
|
_conectando = true;
|
||||||
_salaEncontrada = datos['sala'] as String? ?? 'Sala';
|
_salaSeleccionada = datos['host'] as String? ?? datos['sala'] as String? ?? 'Sala';
|
||||||
});
|
});
|
||||||
|
|
||||||
// Iniciar búsqueda de hosts via Nearby
|
// Iniciar búsqueda para que Nearby encuentre al host
|
||||||
final nearby = context.read<ServicioNearby>();
|
final nearby = context.read<ServicioNearby>();
|
||||||
final ok = await nearby.buscarHosts(_nombreController.text.trim());
|
if (!nearby.buscando) {
|
||||||
|
await nearby.buscarHosts(_nombreController.text.trim());
|
||||||
if (!ok && mounted) {
|
|
||||||
setState(() {
|
|
||||||
_conectando = false;
|
|
||||||
_error = 'No se pudo iniciar la búsqueda. Verifica Bluetooth y ubicación.';
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -70,98 +110,194 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
|||||||
final l10n = AppLocalizations.of(context)!;
|
final l10n = AppLocalizations.of(context)!;
|
||||||
final nearby = context.watch<ServicioNearby>();
|
final nearby = context.watch<ServicioNearby>();
|
||||||
|
|
||||||
// Si estamos conectados, mostrar pantalla de espera
|
// Si estamos conectados → pantalla de espera
|
||||||
if (nearby.conectado && !nearby.esHost) {
|
if (nearby.conectado && !nearby.esHost) {
|
||||||
return _buildPantallaEspera(context, l10n, nearby);
|
return _buildPantallaEspera(context, l10n);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Si escaneando QR
|
||||||
|
if (_escaneandoQR) {
|
||||||
|
return _buildEscaner(context, l10n);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Si buscando hosts o conectando
|
||||||
|
if (_buscando || _conectando) {
|
||||||
|
return _buildDiscovery(context, l10n, nearby);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Formulario nombre
|
||||||
|
return _buildFormularioNombre(context, l10n);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ==================== PASO 1: NOMBRE ====================
|
||||||
|
|
||||||
|
Widget _buildFormularioNombre(BuildContext context, AppLocalizations l10n) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(l10n.joinGameTitle),
|
||||||
|
),
|
||||||
|
body: Padding(
|
||||||
|
padding: const EdgeInsets.all(32),
|
||||||
|
child: Form(
|
||||||
|
key: _formKey,
|
||||||
|
child: Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
const Text('📱', style: TextStyle(fontSize: 64)),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
Text(
|
||||||
|
l10n.joinGameTitle,
|
||||||
|
style: Theme.of(context).textTheme.headlineMedium,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
Text(
|
||||||
|
l10n.enterNameToSearch,
|
||||||
|
style: Theme.of(context).textTheme.bodyLarge,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 32),
|
||||||
|
TextFormField(
|
||||||
|
controller: _nombreController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: l10n.yourName,
|
||||||
|
prefixIcon: const Icon(Icons.person),
|
||||||
|
),
|
||||||
|
validator: (v) {
|
||||||
|
if (v == null || v.trim().isEmpty) return l10n.nameRequired;
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
textCapitalization: TextCapitalization.words,
|
||||||
|
onFieldSubmitted: (_) => _iniciarBusqueda(),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
SizedBox(
|
||||||
|
width: double.infinity,
|
||||||
|
child: ElevatedButton.icon(
|
||||||
|
onPressed: _iniciarBusqueda,
|
||||||
|
icon: const Icon(Icons.search),
|
||||||
|
label: Text(l10n.searchGames),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (_error != null) ...[
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
_buildError(_error!),
|
||||||
|
],
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ==================== PASO 2: DISCOVERY ====================
|
||||||
|
|
||||||
|
Widget _buildDiscovery(BuildContext context, AppLocalizations l10n, ServicioNearby nearby) {
|
||||||
|
final hosts = nearby.hostsEncontrados;
|
||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(l10n.joinGameTitle),
|
title: Text(l10n.joinGameTitle),
|
||||||
leading: IconButton(
|
leading: IconButton(
|
||||||
icon: const Icon(Icons.close),
|
icon: const Icon(Icons.arrow_back),
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
await nearby.desconectar();
|
await nearby.pararBusqueda();
|
||||||
if (context.mounted) Navigator.pop(context);
|
setState(() {
|
||||||
|
_buscando = false;
|
||||||
|
_conectando = false;
|
||||||
|
});
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
body: _escaneando
|
body: Padding(
|
||||||
? _buildEscaner(context, l10n)
|
padding: const EdgeInsets.all(24),
|
||||||
: _buildFormulario(context, l10n),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildFormulario(BuildContext context, AppLocalizations l10n) {
|
|
||||||
return Padding(
|
|
||||||
padding: const EdgeInsets.all(32),
|
|
||||||
child: Form(
|
|
||||||
key: _formKey,
|
|
||||||
child: Column(
|
child: Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
children: [
|
||||||
const Text('📱', style: TextStyle(fontSize: 64)),
|
// Estado
|
||||||
const SizedBox(height: 24),
|
|
||||||
Text(
|
|
||||||
l10n.joinGameTitle,
|
|
||||||
style: Theme.of(context).textTheme.headlineMedium,
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
Text(
|
|
||||||
l10n.enterNameAndScan,
|
|
||||||
style: Theme.of(context).textTheme.bodyLarge,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
),
|
|
||||||
const SizedBox(height: 32),
|
|
||||||
|
|
||||||
// Campo nombre
|
|
||||||
TextFormField(
|
|
||||||
controller: _nombreController,
|
|
||||||
decoration: InputDecoration(
|
|
||||||
labelText: l10n.yourName,
|
|
||||||
prefixIcon: const Icon(Icons.person),
|
|
||||||
),
|
|
||||||
validator: (v) {
|
|
||||||
if (v == null || v.trim().isEmpty) {
|
|
||||||
return l10n.nameRequired;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
},
|
|
||||||
textCapitalization: TextCapitalization.words,
|
|
||||||
),
|
|
||||||
const SizedBox(height: 24),
|
|
||||||
|
|
||||||
if (_conectando) ...[
|
if (_conectando) ...[
|
||||||
const CircularProgressIndicator(color: TemaApp.colorAcento),
|
const CircularProgressIndicator(color: TemaApp.colorAcento),
|
||||||
const SizedBox(height: 12),
|
const SizedBox(height: 12),
|
||||||
Text(
|
Text(
|
||||||
'${l10n.connectingTo} ${_salaEncontrada ?? ""}...',
|
'${l10n.connectingTo} ${_salaSeleccionada ?? ""}...',
|
||||||
style: Theme.of(context).textTheme.bodyMedium,
|
style: Theme.of(context).textTheme.bodyLarge,
|
||||||
),
|
),
|
||||||
|
const SizedBox(height: 24),
|
||||||
] else ...[
|
] else ...[
|
||||||
// Botón escanear QR
|
// Buscando
|
||||||
SizedBox(
|
Row(
|
||||||
width: double.infinity,
|
children: [
|
||||||
child: ElevatedButton.icon(
|
const SizedBox(
|
||||||
onPressed: _iniciarEscaneo,
|
width: 20, height: 20,
|
||||||
icon: const Icon(Icons.qr_code_scanner),
|
child: CircularProgressIndicator(
|
||||||
label: Text(l10n.scanQR),
|
strokeWidth: 2,
|
||||||
),
|
color: TemaApp.colorNaranja,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 12),
|
||||||
|
Text(
|
||||||
|
l10n.searchingGames,
|
||||||
|
style: Theme.of(context).textTheme.titleMedium,
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
|
const SizedBox(height: 24),
|
||||||
],
|
],
|
||||||
|
|
||||||
|
// Lista de hosts encontrados
|
||||||
|
Expanded(
|
||||||
|
child: hosts.isEmpty && !_conectando
|
||||||
|
? Center(
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
const Text('📡', style: TextStyle(fontSize: 48)),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
Text(
|
||||||
|
l10n.noGamesFound,
|
||||||
|
style: Theme.of(context).textTheme.bodyLarge,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
Text(
|
||||||
|
l10n.noGamesFoundHint,
|
||||||
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
||||||
|
color: Colors.grey,
|
||||||
|
),
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: ListView.builder(
|
||||||
|
itemCount: hosts.length,
|
||||||
|
itemBuilder: (context, index) {
|
||||||
|
final entry = hosts.entries.elementAt(index);
|
||||||
|
return _buildHostTile(entry.key, entry.value);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
if (_error != null) ...[
|
if (_error != null) ...[
|
||||||
const SizedBox(height: 16),
|
_buildError(_error!),
|
||||||
Container(
|
const SizedBox(height: 12),
|
||||||
padding: const EdgeInsets.all(12),
|
],
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: TemaApp.colorAcento.withValues(alpha: 0.1),
|
// Fallback: escanear QR
|
||||||
borderRadius: BorderRadius.circular(12),
|
if (!_conectando) ...[
|
||||||
|
const Divider(),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
Text(
|
||||||
|
l10n.orScanQR,
|
||||||
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
||||||
|
color: Colors.grey,
|
||||||
),
|
),
|
||||||
child: Text(
|
),
|
||||||
_error!,
|
const SizedBox(height: 8),
|
||||||
style: const TextStyle(color: TemaApp.colorAcento),
|
SizedBox(
|
||||||
textAlign: TextAlign.center,
|
width: double.infinity,
|
||||||
|
child: OutlinedButton.icon(
|
||||||
|
onPressed: _abrirEscaner,
|
||||||
|
icon: const Icon(Icons.qr_code_scanner),
|
||||||
|
label: Text(l10n.scanQR),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -171,68 +307,108 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildEscaner(BuildContext context, AppLocalizations l10n) {
|
Widget _buildHostTile(String endpointId, String nombre) {
|
||||||
return Stack(
|
return Container(
|
||||||
children: [
|
margin: const EdgeInsets.only(bottom: 8),
|
||||||
MobileScanner(
|
child: Material(
|
||||||
onDetect: _onQRDetectado,
|
color: TemaApp.colorTarjeta,
|
||||||
),
|
borderRadius: BorderRadius.circular(12),
|
||||||
// Overlay
|
child: InkWell(
|
||||||
Positioned(
|
borderRadius: BorderRadius.circular(12),
|
||||||
bottom: 0,
|
onTap: _conectando ? null : () => _conectarAHost(endpointId, nombre),
|
||||||
left: 0,
|
child: Padding(
|
||||||
right: 0,
|
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 16),
|
||||||
child: Container(
|
child: Row(
|
||||||
padding: const EdgeInsets.all(24),
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
gradient: LinearGradient(
|
|
||||||
begin: Alignment.topCenter,
|
|
||||||
end: Alignment.bottomCenter,
|
|
||||||
colors: [
|
|
||||||
Colors.transparent,
|
|
||||||
Colors.black.withValues(alpha: 0.8),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
children: [
|
||||||
Text(
|
const Text('🎭', style: TextStyle(fontSize: 28)),
|
||||||
l10n.scanHostQR,
|
const SizedBox(width: 16),
|
||||||
style: Theme.of(context).textTheme.titleLarge?.copyWith(
|
Expanded(
|
||||||
color: Colors.white,
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
nombre,
|
||||||
|
style: Theme.of(context).textTheme.titleMedium,
|
||||||
|
),
|
||||||
|
Text(
|
||||||
|
'Toca para unirte',
|
||||||
|
style: Theme.of(context).textTheme.bodySmall?.copyWith(
|
||||||
|
color: Colors.grey,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 16),
|
const Icon(Icons.arrow_forward_ios, size: 16, color: Colors.grey),
|
||||||
OutlinedButton(
|
|
||||||
onPressed: () => setState(() => _escaneando = false),
|
|
||||||
style: OutlinedButton.styleFrom(
|
|
||||||
foregroundColor: Colors.white,
|
|
||||||
side: const BorderSide(color: Colors.white),
|
|
||||||
),
|
|
||||||
child: Text(l10n.cancel),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildPantallaEspera(
|
// ==================== ESCÁNER QR ====================
|
||||||
BuildContext context,
|
|
||||||
AppLocalizations l10n,
|
Widget _buildEscaner(BuildContext context, AppLocalizations l10n) {
|
||||||
ServicioNearby nearby,
|
|
||||||
) {
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(_salaEncontrada ?? l10n.joinGameTitle),
|
title: Text(l10n.scanQR),
|
||||||
|
leading: IconButton(
|
||||||
|
icon: const Icon(Icons.arrow_back),
|
||||||
|
onPressed: () => setState(() => _escaneandoQR = false),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
body: Stack(
|
||||||
|
children: [
|
||||||
|
MobileScanner(onDetect: _onQRDetectado),
|
||||||
|
Positioned(
|
||||||
|
bottom: 0,
|
||||||
|
left: 0,
|
||||||
|
right: 0,
|
||||||
|
child: Container(
|
||||||
|
padding: const EdgeInsets.all(24),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
gradient: LinearGradient(
|
||||||
|
begin: Alignment.topCenter,
|
||||||
|
end: Alignment.bottomCenter,
|
||||||
|
colors: [
|
||||||
|
Colors.transparent,
|
||||||
|
Colors.black.withValues(alpha: 0.8),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
child: Text(
|
||||||
|
l10n.scanHostQR,
|
||||||
|
style: Theme.of(context).textTheme.titleLarge?.copyWith(
|
||||||
|
color: Colors.white,
|
||||||
|
),
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ==================== ESPERA ====================
|
||||||
|
|
||||||
|
Widget _buildPantallaEspera(BuildContext context, AppLocalizations l10n) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(_salaSeleccionada ?? l10n.joinGameTitle),
|
||||||
leading: IconButton(
|
leading: IconButton(
|
||||||
icon: const Icon(Icons.close),
|
icon: const Icon(Icons.close),
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
|
final nearby = context.read<ServicioNearby>();
|
||||||
await nearby.desconectar();
|
await nearby.desconectar();
|
||||||
if (context.mounted) Navigator.pop(context);
|
if (context.mounted) {
|
||||||
|
setState(() {
|
||||||
|
_buscando = false;
|
||||||
|
_conectando = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -267,4 +443,21 @@ class _PantallaUnirseState extends State<PantallaUnirse> {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ==================== HELPERS ====================
|
||||||
|
|
||||||
|
Widget _buildError(String msg) {
|
||||||
|
return Container(
|
||||||
|
padding: const EdgeInsets.all(12),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: TemaApp.colorAcento.withValues(alpha: 0.1),
|
||||||
|
borderRadius: BorderRadius.circular(12),
|
||||||
|
),
|
||||||
|
child: Text(
|
||||||
|
msg,
|
||||||
|
style: const TextStyle(color: TemaApp.colorAcento),
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -71,6 +71,9 @@ class ServicioNearby extends ChangeNotifier {
|
|||||||
final Map<String, JugadorConectado> _jugadores = {};
|
final Map<String, JugadorConectado> _jugadores = {};
|
||||||
final List<OnMensajeCallback> _listeners = [];
|
final List<OnMensajeCallback> _listeners = [];
|
||||||
|
|
||||||
|
// Hosts descubiertos (para discovery automático)
|
||||||
|
final Map<String, String> _hostsEncontrados = {}; // endpointId -> nombre
|
||||||
|
|
||||||
// Estado para clientes
|
// Estado para clientes
|
||||||
String? _palabraRecibida;
|
String? _palabraRecibida;
|
||||||
bool? _soyImpostor;
|
bool? _soyImpostor;
|
||||||
@@ -92,6 +95,7 @@ class ServicioNearby extends ChangeNotifier {
|
|||||||
|
|
||||||
List<JugadorConectado> get jugadores => _jugadores.values.toList();
|
List<JugadorConectado> get jugadores => _jugadores.values.toList();
|
||||||
int get numJugadoresConectados => _jugadores.length;
|
int get numJugadoresConectados => _jugadores.length;
|
||||||
|
Map<String, String> get hostsEncontrados => Map.unmodifiable(_hostsEncontrados);
|
||||||
|
|
||||||
/// Registra un listener de mensajes
|
/// Registra un listener de mensajes
|
||||||
void onMensaje(OnMensajeCallback callback) {
|
void onMensaje(OnMensajeCallback callback) {
|
||||||
@@ -239,12 +243,26 @@ class ServicioNearby extends ChangeNotifier {
|
|||||||
|
|
||||||
void _onEndpointEncontrado(String endpointId, String endpointName, String serviceId) {
|
void _onEndpointEncontrado(String endpointId, String endpointName, String serviceId) {
|
||||||
debugPrint('Host encontrado: $endpointName ($endpointId)');
|
debugPrint('Host encontrado: $endpointName ($endpointId)');
|
||||||
// Auto-conectar al primer host encontrado
|
_hostsEncontrados[endpointId] = endpointName;
|
||||||
conectarAHost(endpointId, _miNombre ?? 'Jugador');
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _onEndpointPerdido(String? endpointId) {
|
void _onEndpointPerdido(String? endpointId) {
|
||||||
debugPrint('Endpoint perdido: $endpointId');
|
debugPrint('Endpoint perdido: $endpointId');
|
||||||
|
if (endpointId != null) {
|
||||||
|
_hostsEncontrados.remove(endpointId);
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Para el discovery sin desconectar
|
||||||
|
Future<void> pararBusqueda() async {
|
||||||
|
try {
|
||||||
|
await Nearby().stopDiscovery();
|
||||||
|
} catch (_) {}
|
||||||
|
_buscando = false;
|
||||||
|
_hostsEncontrados.clear();
|
||||||
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _onPayloadRecibido(String endpointId, Payload payload) {
|
void _onPayloadRecibido(String endpointId, Payload payload) {
|
||||||
@@ -443,6 +461,7 @@ class ServicioNearby extends ChangeNotifier {
|
|||||||
_faseActual = null;
|
_faseActual = null;
|
||||||
_datosPartida = null;
|
_datosPartida = null;
|
||||||
_jugadores.clear();
|
_jugadores.clear();
|
||||||
|
_hostsEncontrados.clear();
|
||||||
notifyListeners();
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user