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
{
/// <summary>
/// Client specialization of the Character Select game state. Mainly controls the UI during character-select.
/// </summary>
[RequireComponent(typeof(NetcodeHooks))]
public class ClientCharSelectState : GameStateBehaviour
{
/// <summary>
/// Reference to the scene's state object so that UI can access state
/// </summary>
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 < UICharSelectPlayerSeat > 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 < GameObject > 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 < GameObject > m_UIElementsForSeatChosen ;
[SerializeField]
[Tooltip("UI elements to turn on when the lobby is closed (and game is about to start). Turned off otherwise!")]
List < GameObject > 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 < GameObject > 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 < Guid , GameObject > m_SpawnedCharacterGraphics = new Dictionary < Guid , GameObject > ( ) ;
/// <summary>
/// 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.
/// </summary>
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 < LobbyMode , List < GameObject > > 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 , List < GameObject > > ( )
{
{ 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 ;
}
}
/// <summary>
/// Called when our PlayerNumber (e.g. P1, P2, etc.) has been assigned by the server
/// </summary>
/// <param name="playerNum"></param>
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 = "<b>" + count + "</b> " + pstr + " connected" ;
}
/// <summary>
/// Called by the server when any of the seats in the lobby have changed. (Including ours!)
/// </summary>
void OnLobbyPlayerStateChanged ( NetworkListEvent < NetworkCharSelection . LobbyPlayerState > 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 ) ;
}
}
/// <summary>
/// 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.
/// </summary>
/// <param name="state">Our current seat state</param>
/// <param name="seatIdx">Which seat we're sitting in, or -1 if SeatState is Inactive</param>
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 < Animator > ( ) ;
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
}
}
/// <summary>
/// Internal utility that sets the graphics for the eight lobby-seats (based on their current networked state)
/// </summary>
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 ) ;
}
}
/// <summary>
/// Called by the server when the lobby closes (because all players are seated and locked in)
/// </summary>
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 ) ;
}
}
}
/// <summary>
/// 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.
/// </summary>
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 ) ;
}
}
/// <summary>
/// Called directly by UI elements!
/// </summary>
/// <param name="seatIdx"></param>
public void OnPlayerClickedSeat ( int seatIdx )
{
if ( m_NetworkCharSelection . IsSpawned )
{
m_NetworkCharSelection . ServerChangeSeatRpc ( NetworkManager . Singleton . LocalClientId , seatIdx , false ) ;
}
}
/// <summary>
/// Called directly by UI elements!
/// </summary>
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 ;
}
}
}