using System;
using System.Collections.Generic;
using Unity.BossRoom.Gameplay.UI;
using TMPro;
using Unity.BossRoom.ConnectionManagement;
using Unity.Multiplayer.Samples.Utilities;
using Unity.Netcode;
using UnityEngine;
using VContainer;
using Avatar = Unity.BossRoom.Gameplay.Configuration.Avatar;
using System.Linq;
namespace Unity.BossRoom.Gameplay.GameState
{
///
/// Client specialization of the Character Select game state. Mainly controls the UI during character-select.
///
[RequireComponent(typeof(NetcodeHooks))]
public class ClientCharSelectState : GameStateBehaviour
{
///
/// Reference to the scene's state object so that UI can access state
///
public static ClientCharSelectState Instance { get; private set; }
[SerializeField]
NetcodeHooks m_NetcodeHooks;
public override GameState ActiveState { get { return GameState.CharSelect; } }
[SerializeField]
NetworkCharSelection m_NetworkCharSelection;
[SerializeField]
[Tooltip("This is triggered when the player chooses a character")]
string m_AnimationTriggerOnCharSelect = "BeginRevive";
[SerializeField]
[Tooltip("This is triggered when the player presses the \"Ready\" button")]
string m_AnimationTriggerOnCharChosen = "BeginRevive";
[Header("Lobby Seats")]
[SerializeField]
[Tooltip("Collection of 8 portrait-boxes, one for each potential lobby member")]
List m_PlayerSeats;
[System.Serializable]
public class ColorAndIndicator
{
public Sprite Indicator;
public Color Color;
}
[Tooltip("Representational information for each player")]
public ColorAndIndicator[] m_IdentifiersForEachPlayerNumber;
[SerializeField]
[Tooltip("Text element containing player count which updates as players connect")]
TextMeshProUGUI m_NumPlayersText;
[SerializeField]
[Tooltip("Text element for the Ready button")]
TextMeshProUGUI m_ReadyButtonText;
[Header("UI Elements for different lobby modes")]
[SerializeField]
[Tooltip("UI elements to turn on when the player hasn't chosen their seat yet. Turned off otherwise!")]
List m_UIElementsForNoSeatChosen;
[SerializeField]
[Tooltip("UI elements to turn on when the player has locked in their seat choice (and is now waiting for other players to do the same). Turned off otherwise!")]
List m_UIElementsForSeatChosen;
[SerializeField]
[Tooltip("UI elements to turn on when the lobby is closed (and game is about to start). Turned off otherwise!")]
List m_UIElementsForLobbyEnding;
[SerializeField]
[Tooltip("UI elements to turn on when there's been a fatal error (and the client cannot proceed). Turned off otherwise!")]
List m_UIElementsForFatalError;
[Header("Misc")]
[SerializeField]
[Tooltip("The controller for the class-info box")]
UICharSelectClassInfoBox m_ClassInfoBox;
[SerializeField]
Transform m_CharacterGraphicsParent;
int m_LastSeatSelected = -1;
bool m_HasLocalPlayerLockedIn = false;
GameObject m_CurrentCharacterGraphics;
Animator m_CurrentCharacterGraphicsAnimator;
Dictionary m_SpawnedCharacterGraphics = new Dictionary();
///
/// Conceptual modes or stages that the lobby can be in. We don't actually
/// bother to keep track of what LobbyMode we're in at any given time; it's just
/// an abstraction that makes it easier to configure which UI elements should
/// be enabled/disabled in each stage of the lobby.
///
enum LobbyMode
{
ChooseSeat, // "Choose your seat!" stage
SeatChosen, // "Waiting for other players!" stage
LobbyEnding, // "Get ready! Game is starting!" stage
FatalError, // "Fatal Error" stage
}
Dictionary> m_LobbyUIElementsByMode;
[Inject]
ConnectionManager m_ConnectionManager;
protected override void Awake()
{
base.Awake();
Instance = this;
m_NetcodeHooks.OnNetworkSpawnHook += OnNetworkSpawn;
m_NetcodeHooks.OnNetworkDespawnHook += OnNetworkDespawn;
m_LobbyUIElementsByMode = new Dictionary>()
{
{ LobbyMode.ChooseSeat, m_UIElementsForNoSeatChosen },
{ LobbyMode.SeatChosen, m_UIElementsForSeatChosen },
{ LobbyMode.LobbyEnding, m_UIElementsForLobbyEnding },
{ LobbyMode.FatalError, m_UIElementsForFatalError },
};
}
protected override void OnDestroy()
{
if (Instance == this)
{
Instance = null;
}
base.OnDestroy();
}
protected override void Start()
{
base.Start();
for (int i = 0; i < m_PlayerSeats.Count; ++i)
{
m_PlayerSeats[i].Initialize(i);
}
ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
UpdateCharacterSelection(NetworkCharSelection.SeatState.Inactive);
}
void OnNetworkDespawn()
{
if (m_NetworkCharSelection)
{
m_NetworkCharSelection.IsLobbyClosed.OnValueChanged -= OnLobbyClosedChanged;
m_NetworkCharSelection.LobbyPlayers.OnListChanged -= OnLobbyPlayerStateChanged;
}
}
void OnNetworkSpawn()
{
if (!NetworkManager.Singleton.IsClient)
{
enabled = false;
}
else
{
m_NetworkCharSelection.IsLobbyClosed.OnValueChanged += OnLobbyClosedChanged;
m_NetworkCharSelection.LobbyPlayers.OnListChanged += OnLobbyPlayerStateChanged;
}
}
///
/// Called when our PlayerNumber (e.g. P1, P2, etc.) has been assigned by the server
///
///
void OnAssignedPlayerNumber(int playerNum)
{
m_ClassInfoBox.OnSetPlayerNumber(playerNum);
}
void UpdatePlayerCount()
{
int count = m_NetworkCharSelection.LobbyPlayers.Count;
var pstr = (count > 1) ? "players" : "player";
m_NumPlayersText.text = "" + count + " " + pstr + " connected";
}
///
/// Called by the server when any of the seats in the lobby have changed. (Including ours!)
///
void OnLobbyPlayerStateChanged(NetworkListEvent changeEvent)
{
UpdateSeats();
UpdatePlayerCount();
// now let's find our local player in the list and update the character/info box appropriately
int localPlayerIdx = -1;
for (int i = 0; i < m_NetworkCharSelection.LobbyPlayers.Count; ++i)
{
if (m_NetworkCharSelection.LobbyPlayers[i].ClientId == NetworkManager.Singleton.LocalClientId)
{
localPlayerIdx = i;
break;
}
}
if (localPlayerIdx == -1)
{
// we aren't currently participating in the lobby!
// this can happen for various reasons, such as the lobby being full and us not getting a seat.
UpdateCharacterSelection(NetworkCharSelection.SeatState.Inactive);
}
else if (m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].SeatState == NetworkCharSelection.SeatState.Inactive)
{
// we haven't chosen a seat yet (or were kicked out of our seat by someone else)
UpdateCharacterSelection(NetworkCharSelection.SeatState.Inactive);
// make sure our player num is properly set in Lobby UI
OnAssignedPlayerNumber(m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].PlayerNumber);
}
else
{
// we have a seat! Note that if our seat is LockedIn, this function will also switch the lobby mode
UpdateCharacterSelection(m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].SeatState, m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].SeatIdx);
}
}
///
/// Internal utility that sets the character-graphics and class-info box based on
/// our chosen seat. It also triggers a LobbyMode change when it notices that our seat-state
/// is LockedIn.
///
/// Our current seat state
/// Which seat we're sitting in, or -1 if SeatState is Inactive
void UpdateCharacterSelection(NetworkCharSelection.SeatState state, int seatIdx = -1)
{
if (seatIdx < -1 || seatIdx >= m_NetworkCharSelection.AvatarConfiguration.Length)
{
Debug.LogError($"Invalid seat index: {seatIdx}. Must be between 0 and {m_NetworkCharSelection.AvatarConfiguration.Length - 1}.");
return; // Prevent out-of-bounds access.
}
bool isNewSeat = m_LastSeatSelected != seatIdx;
m_LastSeatSelected = seatIdx;
if (state == NetworkCharSelection.SeatState.Inactive)
{
// Deactivate current character graphics when unselecting a seat
if (m_CurrentCharacterGraphics)
{
m_CurrentCharacterGraphics.SetActive(false);
}
m_ClassInfoBox.ConfigureForNoSelection();
m_HasLocalPlayerLockedIn = false; // Reset lock-in status
ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
return;
}
if (seatIdx == -1)
{
Debug.LogWarning("Seat index is -1 for an active state. This should not happen.");
return;
}
// Change character preview when selecting a new seat
if (isNewSeat)
{
var selectedCharacterGraphics = GetCharacterGraphics(m_NetworkCharSelection.AvatarConfiguration[seatIdx]);
if (m_CurrentCharacterGraphics)
{
m_CurrentCharacterGraphics.SetActive(false); // Deactivate previous graphics
}
selectedCharacterGraphics.SetActive(true); // Activate new character graphics
m_CurrentCharacterGraphics = selectedCharacterGraphics;
m_CurrentCharacterGraphicsAnimator = m_CurrentCharacterGraphics.GetComponent();
m_ClassInfoBox.ConfigureForClass(m_NetworkCharSelection.AvatarConfiguration[seatIdx].CharacterClass);
}
// Handle lock-in and active state changes
if (state == NetworkCharSelection.SeatState.LockedIn && !m_HasLocalPlayerLockedIn)
{
// Local player locks in their choice
ConfigureUIForLobbyMode(m_NetworkCharSelection.IsLobbyClosed.Value ? LobbyMode.LobbyEnding : LobbyMode.SeatChosen);
m_HasLocalPlayerLockedIn = true;
// m_CurrentCharacterGraphicsAnimator.SetTrigger(m_AnimationTriggerOnCharChosen); // Optional animation trigger
}
else if (state == NetworkCharSelection.SeatState.Active && m_HasLocalPlayerLockedIn)
{
// Reset if locked-in choice was unselected
ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
m_ClassInfoBox.SetLockedIn(false);
m_HasLocalPlayerLockedIn = false;
}
else if (state == NetworkCharSelection.SeatState.Active && isNewSeat)
{
// Handle animation trigger when actively selecting a new seat
// m_CurrentCharacterGraphicsAnimator.SetTrigger(m_AnimationTriggerOnCharSelect); // Optional animation trigger
}
}
///
/// Internal utility that sets the graphics for the eight lobby-seats (based on their current networked state)
///
void UpdateSeats()
{
// Players can hop between seats -- and can even SHARE seats -- while they're choosing a class.
// Once they have chosen their class (by "locking in" their seat), other players in that seat are kicked out.
// But until a seat is locked in, we need to display each seat as being used by the latest player to choose it.
// So we go through all players and figure out who should visually be shown as sitting in that seat.
NetworkCharSelection.LobbyPlayerState[] curSeats = new NetworkCharSelection.LobbyPlayerState[m_PlayerSeats.Count];
foreach (NetworkCharSelection.LobbyPlayerState playerState in m_NetworkCharSelection.LobbyPlayers)
{
if (playerState.SeatIdx == -1 || playerState.SeatState == NetworkCharSelection.SeatState.Inactive)
continue; // this player isn't seated at all!
if (curSeats[playerState.SeatIdx].SeatState == NetworkCharSelection.SeatState.Inactive
|| (curSeats[playerState.SeatIdx].SeatState == NetworkCharSelection.SeatState.Active && curSeats[playerState.SeatIdx].LastChangeTime < playerState.LastChangeTime))
{
// this is the best candidate to be displayed in this seat (so far)
curSeats[playerState.SeatIdx] = playerState;
}
}
// now actually update the seats in the UI
for (int i = 0; i < m_PlayerSeats.Count; ++i)
{
m_PlayerSeats[i].SetState(curSeats[i].SeatState, curSeats[i].PlayerNumber, curSeats[i].PlayerName);
}
}
///
/// Called by the server when the lobby closes (because all players are seated and locked in)
///
void OnLobbyClosedChanged(bool wasLobbyClosed, bool isLobbyClosed)
{
if (isLobbyClosed)
{
ConfigureUIForLobbyMode(LobbyMode.LobbyEnding);
}
else
{
if (m_LastSeatSelected == -1)
{
ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
}
else
{
ConfigureUIForLobbyMode(LobbyMode.SeatChosen);
m_ClassInfoBox.ConfigureForClass(m_NetworkCharSelection.AvatarConfiguration[m_LastSeatSelected].CharacterClass);
}
}
}
///
/// Turns on the UI elements for a specified "lobby mode", and turns off UI elements for all other modes.
/// It can also disable/enable the lobby seats and the "Ready" button if they are inappropriate for the
/// given mode.
///
void ConfigureUIForLobbyMode(LobbyMode mode)
{
// first the easy bit: turn off all the inappropriate ui elements, and turn the appropriate ones on!
foreach (var list in m_LobbyUIElementsByMode.Values)
{
foreach (var uiElement in list)
{
uiElement.SetActive(false);
}
}
foreach (var uiElement in m_LobbyUIElementsByMode[mode])
{
uiElement.SetActive(true);
}
// that finishes the easy bit. Next, each lobby mode might also need to configure the lobby seats and class-info box.
bool isSeatsDisabledInThisMode = false;
switch (mode)
{
case LobbyMode.ChooseSeat:
if (m_LastSeatSelected == -1)
{
if (m_CurrentCharacterGraphics)
{
m_CurrentCharacterGraphics.gameObject.SetActive(false);
}
m_ClassInfoBox.ConfigureForNoSelection();
}
m_ReadyButtonText.text = "READY!";
break;
case LobbyMode.SeatChosen:
isSeatsDisabledInThisMode = true;
m_ClassInfoBox.SetLockedIn(true);
m_ReadyButtonText.text = "UNREADY";
break;
case LobbyMode.FatalError:
isSeatsDisabledInThisMode = true;
m_ClassInfoBox.ConfigureForNoSelection();
break;
case LobbyMode.LobbyEnding:
isSeatsDisabledInThisMode = true;
m_ClassInfoBox.ConfigureForNoSelection();
break;
}
// go through all our seats and enable or disable buttons
foreach (var seat in m_PlayerSeats)
{
// disable interaction if seat is already locked or all seats disabled
seat.SetDisableInteraction(seat.IsLocked() || isSeatsDisabledInThisMode);
}
}
///
/// Called directly by UI elements!
///
///
public void OnPlayerClickedSeat(int seatIdx)
{
if (m_NetworkCharSelection.IsSpawned)
{
m_NetworkCharSelection.ServerChangeSeatRpc(NetworkManager.Singleton.LocalClientId, seatIdx, false);
}
}
///
/// Called directly by UI elements!
///
public void OnPlayerClickedReady()
{
if (m_NetworkCharSelection.IsSpawned)
{
// request to lock in or unlock if already locked in
m_NetworkCharSelection.ServerChangeSeatRpc(NetworkManager.Singleton.LocalClientId, m_LastSeatSelected, !m_HasLocalPlayerLockedIn);
}
}
GameObject GetCharacterGraphics(Avatar avatar)
{
if (!m_SpawnedCharacterGraphics.TryGetValue(avatar.Guid, out GameObject characterGraphics))
{
characterGraphics = Instantiate(avatar.GraphicsCharacterSelect, m_CharacterGraphicsParent);
m_SpawnedCharacterGraphics.Add(avatar.Guid, characterGraphics);
}
return characterGraphics;
}
}
}
//using System;
//using System.Collections.Generic;
//using Unity.BossRoom.Gameplay.UI;
//using TMPro;
//using Unity.BossRoom.ConnectionManagement;
//using Unity.Multiplayer.Samples.Utilities;
//using Unity.Netcode;
//using UnityEngine;
//using VContainer;
//using Avatar = Unity.BossRoom.Gameplay.Configuration.Avatar;
//using System.Linq;
//namespace Unity.BossRoom.Gameplay.GameState
//{
// ///
// /// Client specialization of the Character Select game state. Mainly controls the UI during character-select.
// ///
// [RequireComponent(typeof(NetcodeHooks))]
// public class ClientCharSelectState : GameStateBehaviour
// {
// ///
// /// Reference to the scene's state object so that UI can access state
// ///
// public static ClientCharSelectState Instance { get; private set; }
// [SerializeField]
// NetcodeHooks m_NetcodeHooks;
// public override GameState ActiveState { get { return GameState.CharSelect; } }
// [SerializeField]
// NetworkCharSelection m_NetworkCharSelection;
// [SerializeField]
// [Tooltip("This is triggered when the player chooses a character")]
// string m_AnimationTriggerOnCharSelect = "BeginRevive";
// [SerializeField]
// [Tooltip("This is triggered when the player presses the \"Ready\" button")]
// string m_AnimationTriggerOnCharChosen = "BeginRevive";
// [Header("Lobby Seats")]
// [SerializeField]
// [Tooltip("Collection of 8 portrait-boxes, one for each potential lobby member")]
// List m_PlayerSeats;
// [System.Serializable]
// public class ColorAndIndicator
// {
// public Sprite Indicator;
// public Color Color;
// }
// [Tooltip("Representational information for each player")]
// public ColorAndIndicator[] m_IdentifiersForEachPlayerNumber;
// [SerializeField]
// [Tooltip("Text element containing player count which updates as players connect")]
// TextMeshProUGUI m_NumPlayersText;
// [SerializeField]
// [Tooltip("Text element for the Ready button")]
// TextMeshProUGUI m_ReadyButtonText;
// [Header("UI Elements for different lobby modes")]
// [SerializeField]
// [Tooltip("UI elements to turn on when the player hasn't chosen their seat yet. Turned off otherwise!")]
// List m_UIElementsForNoSeatChosen;
// [SerializeField]
// [Tooltip("UI elements to turn on when the player has locked in their seat choice (and is now waiting for other players to do the same). Turned off otherwise!")]
// List m_UIElementsForSeatChosen;
// [SerializeField]
// [Tooltip("UI elements to turn on when the lobby is closed (and game is about to start). Turned off otherwise!")]
// List m_UIElementsForLobbyEnding;
// [SerializeField]
// [Tooltip("UI elements to turn on when there's been a fatal error (and the client cannot proceed). Turned off otherwise!")]
// List m_UIElementsForFatalError;
// [Header("Misc")]
// [SerializeField]
// [Tooltip("The controller for the class-info box")]
// UICharSelectClassInfoBox m_ClassInfoBox;
// [SerializeField]
// Transform m_CharacterGraphicsParent;
// int m_LastSeatSelected = -1;
// bool m_HasLocalPlayerLockedIn = false;
// GameObject m_CurrentCharacterGraphics;
// Animator m_CurrentCharacterGraphicsAnimator;
// Dictionary m_SpawnedCharacterGraphics = new Dictionary();
// ///
// /// Conceptual modes or stages that the lobby can be in. We don't actually
// /// bother to keep track of what LobbyMode we're in at any given time; it's just
// /// an abstraction that makes it easier to configure which UI elements should
// /// be enabled/disabled in each stage of the lobby.
// ///
// enum LobbyMode
// {
// ChooseSeat, // "Choose your seat!" stage
// SeatChosen, // "Waiting for other players!" stage
// LobbyEnding, // "Get ready! Game is starting!" stage
// FatalError, // "Fatal Error" stage
// }
// Dictionary> m_LobbyUIElementsByMode;
// [Inject]
// ConnectionManager m_ConnectionManager;
// protected override void Awake()
// {
// base.Awake();
// Instance = this;
// m_NetcodeHooks.OnNetworkSpawnHook += OnNetworkSpawn;
// m_NetcodeHooks.OnNetworkDespawnHook += OnNetworkDespawn;
// m_LobbyUIElementsByMode = new Dictionary>()
// {
// { LobbyMode.ChooseSeat, m_UIElementsForNoSeatChosen },
// { LobbyMode.SeatChosen, m_UIElementsForSeatChosen },
// { LobbyMode.LobbyEnding, m_UIElementsForLobbyEnding },
// { LobbyMode.FatalError, m_UIElementsForFatalError },
// };
// }
// protected override void OnDestroy()
// {
// if (Instance == this)
// {
// Instance = null;
// }
// base.OnDestroy();
// }
// protected override void Start()
// {
// base.Start();
// for (int i = 0; i < m_PlayerSeats.Count; ++i)
// {
// m_PlayerSeats[i].Initialize(i);
// }
// ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
// UpdateCharacterSelection(NetworkCharSelection.SeatState.Inactive);
// }
// void OnNetworkDespawn()
// {
// if (m_NetworkCharSelection)
// {
// m_NetworkCharSelection.IsLobbyClosed.OnValueChanged -= OnLobbyClosedChanged;
// m_NetworkCharSelection.LobbyPlayers.OnListChanged -= OnLobbyPlayerStateChanged;
// }
// }
// void OnNetworkSpawn()
// {
// if (!NetworkManager.Singleton.IsClient)
// {
// enabled = false;
// }
// else
// {
// m_NetworkCharSelection.IsLobbyClosed.OnValueChanged += OnLobbyClosedChanged;
// m_NetworkCharSelection.LobbyPlayers.OnListChanged += OnLobbyPlayerStateChanged;
// }
// }
// ///
// /// Called when our PlayerNumber (e.g. P1, P2, etc.) has been assigned by the server
// ///
// ///
// void OnAssignedPlayerNumber(int playerNum)
// {
// m_ClassInfoBox.OnSetPlayerNumber(playerNum);
// }
// void UpdatePlayerCount()
// {
// int count = m_NetworkCharSelection.LobbyPlayers.Count;
// var pstr = (count > 1) ? "players" : "player";
// m_NumPlayersText.text = "" + count + " " + pstr + " connected";
// }
// ///
// /// Called by the server when any of the seats in the lobby have changed. (Including ours!)
// ///
// void OnLobbyPlayerStateChanged(NetworkListEvent changeEvent)
// {
// UpdateSeats();
// UpdatePlayerCount();
// // now let's find our local player in the list and update the character/info box appropriately
// int localPlayerIdx = -1;
// for (int i = 0; i < m_NetworkCharSelection.LobbyPlayers.Count; ++i)
// {
// if (m_NetworkCharSelection.LobbyPlayers[i].ClientId == NetworkManager.Singleton.LocalClientId)
// {
// localPlayerIdx = i;
// break;
// }
// }
// if (localPlayerIdx == -1)
// {
// // we aren't currently participating in the lobby!
// // this can happen for various reasons, such as the lobby being full and us not getting a seat.
// UpdateCharacterSelection(NetworkCharSelection.SeatState.Inactive);
// }
// else if (m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].SeatState == NetworkCharSelection.SeatState.Inactive)
// {
// // we haven't chosen a seat yet (or were kicked out of our seat by someone else)
// UpdateCharacterSelection(NetworkCharSelection.SeatState.Inactive);
// // make sure our player num is properly set in Lobby UI
// OnAssignedPlayerNumber(m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].PlayerNumber);
// }
// else
// {
// // we have a seat! Note that if our seat is LockedIn, this function will also switch the lobby mode
// UpdateCharacterSelection(m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].SeatState, m_NetworkCharSelection.LobbyPlayers[localPlayerIdx].SeatIdx);
// }
// }
// ///
// /// Internal utility that sets the character-graphics and class-info box based on
// /// our chosen seat. It also triggers a LobbyMode change when it notices that our seat-state
// /// is LockedIn.
// ///
// /// Our current seat state
// /// Which seat we're sitting in, or -1 if SeatState is Inactive
// void UpdateCharacterSelection(NetworkCharSelection.SeatState state, int seatIdx = -1)
// {
// if (seatIdx < -1 || seatIdx >= m_NetworkCharSelection.AvatarConfiguration.Length)
// {
// Debug.LogError($"Invalid seat index: {seatIdx}. Must be between 0 and {m_NetworkCharSelection.AvatarConfiguration.Length - 1}.");
// return; // Prevent out-of-bounds access.
// }
// bool isNewSeat = m_LastSeatSelected != seatIdx;
// m_LastSeatSelected = seatIdx;
// if (state == NetworkCharSelection.SeatState.Inactive)
// {
// // Deactivate current character graphics when unselecting a seat
// if (m_CurrentCharacterGraphics)
// {
// m_CurrentCharacterGraphics.SetActive(false);
// }
// m_ClassInfoBox.ConfigureForNoSelection();
// m_HasLocalPlayerLockedIn = false; // Reset lock-in status
// ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
// return;
// }
// if (seatIdx == -1)
// {
// Debug.LogWarning("Seat index is -1 for an active state. This should not happen.");
// return;
// }
// // Change character preview when selecting a new seat
// if (isNewSeat)
// {
// var selectedCharacterGraphics = GetCharacterGraphics(m_NetworkCharSelection.AvatarConfiguration[seatIdx]);
// if (m_CurrentCharacterGraphics)
// {
// m_CurrentCharacterGraphics.SetActive(false); // Deactivate previous graphics
// }
// selectedCharacterGraphics.SetActive(true); // Activate new character graphics
// m_CurrentCharacterGraphics = selectedCharacterGraphics;
// m_CurrentCharacterGraphicsAnimator = m_CurrentCharacterGraphics.GetComponent();
// m_ClassInfoBox.ConfigureForClass(m_NetworkCharSelection.AvatarConfiguration[seatIdx].CharacterClass);
// }
// // Handle lock-in and active state changes
// if (state == NetworkCharSelection.SeatState.LockedIn && !m_HasLocalPlayerLockedIn)
// {
// // Local player locks in their choice
// ConfigureUIForLobbyMode(m_NetworkCharSelection.IsLobbyClosed.Value ? LobbyMode.LobbyEnding : LobbyMode.SeatChosen);
// m_HasLocalPlayerLockedIn = true;
// // m_CurrentCharacterGraphicsAnimator.SetTrigger(m_AnimationTriggerOnCharChosen); // Optional animation trigger
// }
// else if (state == NetworkCharSelection.SeatState.Active && m_HasLocalPlayerLockedIn)
// {
// // Reset if locked-in choice was unselected
// ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
// m_ClassInfoBox.SetLockedIn(false);
// m_HasLocalPlayerLockedIn = false;
// }
// else if (state == NetworkCharSelection.SeatState.Active && isNewSeat)
// {
// // Handle animation trigger when actively selecting a new seat
// // m_CurrentCharacterGraphicsAnimator.SetTrigger(m_AnimationTriggerOnCharSelect); // Optional animation trigger
// }
// }
// ///
// /// Internal utility that sets the graphics for the eight lobby-seats (based on their current networked state)
// ///
// void UpdateSeats()
// {
// // Players can hop between seats -- and can even SHARE seats -- while they're choosing a class.
// // Once they have chosen their class (by "locking in" their seat), other players in that seat are kicked out.
// // But until a seat is locked in, we need to display each seat as being used by the latest player to choose it.
// // So we go through all players and figure out who should visually be shown as sitting in that seat.
// NetworkCharSelection.LobbyPlayerState[] curSeats = new NetworkCharSelection.LobbyPlayerState[m_PlayerSeats.Count];
// foreach (NetworkCharSelection.LobbyPlayerState playerState in m_NetworkCharSelection.LobbyPlayers)
// {
// if (playerState.SeatIdx == -1 || playerState.SeatState == NetworkCharSelection.SeatState.Inactive)
// continue; // this player isn't seated at all!
// if (curSeats[playerState.SeatIdx].SeatState == NetworkCharSelection.SeatState.Inactive
// || (curSeats[playerState.SeatIdx].SeatState == NetworkCharSelection.SeatState.Active && curSeats[playerState.SeatIdx].LastChangeTime < playerState.LastChangeTime))
// {
// // this is the best candidate to be displayed in this seat (so far)
// curSeats[playerState.SeatIdx] = playerState;
// }
// }
// // now actually update the seats in the UI
// for (int i = 0; i < m_PlayerSeats.Count; ++i)
// {
// m_PlayerSeats[i].SetState(curSeats[i].SeatState, curSeats[i].PlayerNumber, curSeats[i].PlayerName);
// }
// }
// ///
// /// Called by the server when the lobby closes (because all players are seated and locked in)
// ///
// void OnLobbyClosedChanged(bool wasLobbyClosed, bool isLobbyClosed)
// {
// if (isLobbyClosed)
// {
// ConfigureUIForLobbyMode(LobbyMode.LobbyEnding);
// }
// else
// {
// if (m_LastSeatSelected == -1)
// {
// ConfigureUIForLobbyMode(LobbyMode.ChooseSeat);
// }
// else
// {
// ConfigureUIForLobbyMode(LobbyMode.SeatChosen);
// m_ClassInfoBox.ConfigureForClass(m_NetworkCharSelection.AvatarConfiguration[m_LastSeatSelected].CharacterClass);
// }
// }
// }
// ///
// /// Turns on the UI elements for a specified "lobby mode", and turns off UI elements for all other modes.
// /// It can also disable/enable the lobby seats and the "Ready" button if they are inappropriate for the
// /// given mode.
// ///
// void ConfigureUIForLobbyMode(LobbyMode mode)
// {
// // first the easy bit: turn off all the inappropriate ui elements, and turn the appropriate ones on!
// foreach (var list in m_LobbyUIElementsByMode.Values)
// {
// foreach (var uiElement in list)
// {
// uiElement.SetActive(false);
// }
// }
// foreach (var uiElement in m_LobbyUIElementsByMode[mode])
// {
// uiElement.SetActive(true);
// }
// // that finishes the easy bit. Next, each lobby mode might also need to configure the lobby seats and class-info box.
// bool isSeatsDisabledInThisMode = false;
// switch (mode)
// {
// case LobbyMode.ChooseSeat:
// if (m_LastSeatSelected == -1)
// {
// if (m_CurrentCharacterGraphics)
// {
// m_CurrentCharacterGraphics.gameObject.SetActive(false);
// }
// m_ClassInfoBox.ConfigureForNoSelection();
// }
// m_ReadyButtonText.text = "READY!";
// break;
// case LobbyMode.SeatChosen:
// isSeatsDisabledInThisMode = true;
// m_ClassInfoBox.SetLockedIn(true);
// m_ReadyButtonText.text = "UNREADY";
// break;
// case LobbyMode.FatalError:
// isSeatsDisabledInThisMode = true;
// m_ClassInfoBox.ConfigureForNoSelection();
// break;
// case LobbyMode.LobbyEnding:
// isSeatsDisabledInThisMode = true;
// m_ClassInfoBox.ConfigureForNoSelection();
// break;
// }
// // go through all our seats and enable or disable buttons
// foreach (var seat in m_PlayerSeats)
// {
// // disable interaction if seat is already locked or all seats disabled
// seat.SetDisableInteraction(seat.IsLocked() || isSeatsDisabledInThisMode);
// }
// }
// ///
// /// Called directly by UI elements!
// ///
// ///
// public void OnPlayerClickedSeat(int seatIdx)
// {
// if (m_NetworkCharSelection.IsSpawned)
// {
// m_NetworkCharSelection.ServerChangeSeatRpc(NetworkManager.Singleton.LocalClientId, seatIdx, false);
// }
// }
// ///
// /// Called directly by UI elements!
// ///
// public void OnPlayerClickedReady()
// {
// if (m_NetworkCharSelection.IsSpawned)
// {
// // request to lock in or unlock if already locked in
// m_NetworkCharSelection.ServerChangeSeatRpc(NetworkManager.Singleton.LocalClientId, m_LastSeatSelected, !m_HasLocalPlayerLockedIn);
// }
// }
// GameObject GetCharacterGraphics(Avatar avatar)
// {
// if (!m_SpawnedCharacterGraphics.TryGetValue(avatar.Guid, out GameObject characterGraphics))
// {
// characterGraphics = Instantiate(avatar.GraphicsCharacterSelect, m_CharacterGraphicsParent);
// m_SpawnedCharacterGraphics.Add(avatar.Guid, characterGraphics);
// }
// return characterGraphics;
// }
// }
//}