You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
459 lines
19 KiB
C#
459 lines
19 KiB
C#
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;
|
|
}
|
|
|
|
}
|
|
}
|