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.
HighGroundRoyaleNetcode/Assets/Scripts/Gameplay/UI/UIStateDisplayHandler.cs

229 lines
6.6 KiB
C#

2 months ago
using System.Collections;
using Unity.BossRoom.Gameplay.GameplayObjects;
using Unity.BossRoom.Gameplay.GameplayObjects.Character;
using Unity.BossRoom.Infrastructure;
using Unity.BossRoom.Utils;
using Unity.Netcode;
using UnityEngine;
using UnityEngine.Assertions;
namespace Unity.BossRoom.Gameplay.UI
{
/// <summary>
/// Class designed to only run on a client. Add this to a world-space prefab to display health or name on UI.
/// </summary>
/// <remarks>
/// Execution order is explicitly set such that it this class executes its LateUpdate after any Cinemachine
/// LateUpdate calls, which may alter the final position of the game camera.
/// </remarks>
[DefaultExecutionOrder(300)]
public class UIStateDisplayHandler : NetworkBehaviour
{
[SerializeField]
bool m_DisplayHealth;
[SerializeField]
bool m_DisplayName;
[SerializeField]
UIStateDisplay m_UIStatePrefab;
// spawned in world (only one instance of this)
UIStateDisplay m_UIState;
RectTransform m_UIStateRectTransform;
bool m_UIStateActive;
[SerializeField]
NetworkHealthState m_NetworkHealthState;
[SerializeField]
NetworkNameState m_NetworkNameState;
ServerCharacter m_ServerCharacter;
ClientPlayerAvatarNetworkAnimator m_ClientPlayerAvatarNetworkAnimator;
NetworkAvatarGuidState m_NetworkAvatarGuidState;
[SerializeField]
IntVariable m_BaseHP;
[Tooltip("UI object(s) will appear positioned at this transforms position.")]
[SerializeField]
Transform m_TransformToTrack;
Camera m_Camera;
Transform m_CanvasTransform;
// as soon as any HP goes to 0, we wait this long before removing health bar UI object
const float k_DurationSeconds = 2f;
[Tooltip("World space vertical offset for positioning.")]
[SerializeField]
float m_VerticalWorldOffset;
[Tooltip("Screen space vertical offset for positioning.")]
[SerializeField]
float m_VerticalScreenOffset;
Vector3 m_VerticalOffset;
// used to compute world position based on target and offsets
Vector3 m_WorldPos;
void Awake()
{
m_ServerCharacter = GetComponent<ServerCharacter>();
}
public override void OnNetworkSpawn()
{
if (!NetworkManager.Singleton.IsClient)
{
enabled = false;
return;
}
var cameraGameObject = GameObject.FindWithTag("MainCamera");
if (cameraGameObject)
{
m_Camera = cameraGameObject.GetComponent<Camera>();
}
Assert.IsNotNull(m_Camera);
var canvasGameObject = GameObject.FindWithTag("GameCanvas");
if (canvasGameObject)
{
m_CanvasTransform = canvasGameObject.transform;
}
Assert.IsNotNull(m_CanvasTransform);
Assert.IsTrue(m_DisplayHealth || m_DisplayName, "Neither display fields are toggled on!");
if (m_DisplayHealth)
{
Assert.IsNotNull(m_NetworkHealthState, "A NetworkHealthState component needs to be attached!");
}
m_VerticalOffset = new Vector3(0f, m_VerticalScreenOffset, 0f);
// if PC, find our graphics transform and update health through callbacks, if displayed
if (TryGetComponent(out m_ClientPlayerAvatarNetworkAnimator) && TryGetComponent(out m_NetworkAvatarGuidState))
{
m_BaseHP = m_NetworkAvatarGuidState.RegisteredAvatar.CharacterClass.BaseHP;
m_TransformToTrack = m_ClientPlayerAvatarNetworkAnimator.Animator.transform;
if (m_DisplayHealth)
{
m_NetworkHealthState.HitPointsReplenished += DisplayUIHealth;
m_NetworkHealthState.HitPointsDepleted += RemoveUIHealth;
}
}
if (m_DisplayName)
{
DisplayUIName();
}
if (m_DisplayHealth)
{
DisplayUIHealth();
}
}
void OnDisable()
{
if (!m_DisplayHealth)
{
return;
}
if (m_NetworkHealthState != null)
{
m_NetworkHealthState.HitPointsReplenished -= DisplayUIHealth;
m_NetworkHealthState.HitPointsDepleted -= RemoveUIHealth;
}
}
void DisplayUIName()
{
if (m_NetworkNameState == null)
{
return;
}
if (m_UIState == null)
{
SpawnUIState();
}
m_UIState.DisplayName(m_NetworkNameState.Name);
m_UIStateActive = true;
}
void DisplayUIHealth()
{
if (m_NetworkHealthState == null)
{
return;
}
if (m_UIState == null)
{
SpawnUIState();
}
m_UIState.DisplayHealth(m_NetworkHealthState.HitPoints, m_BaseHP.Value);
m_UIStateActive = true;
}
void SpawnUIState()
{
m_UIState = Instantiate(m_UIStatePrefab, m_CanvasTransform);
// make in world UI state draw under other UI elements
m_UIState.transform.SetAsFirstSibling();
m_UIStateRectTransform = m_UIState.GetComponent<RectTransform>();
}
void RemoveUIHealth()
{
StartCoroutine(WaitToHideHealthBar());
}
IEnumerator WaitToHideHealthBar()
{
yield return new WaitForSeconds(k_DurationSeconds);
m_UIState.HideHealth();
}
/// <remarks>
/// Moving UI objects on LateUpdate ensures that the game camera is at its final position pre-render.
/// </remarks>
void LateUpdate()
{
if (m_UIStateActive && m_TransformToTrack)
{
// set world position with world offset added
m_WorldPos.Set(m_TransformToTrack.position.x,
m_TransformToTrack.position.y + m_VerticalWorldOffset,
m_TransformToTrack.position.z);
m_UIStateRectTransform.position = m_Camera.WorldToScreenPoint(m_WorldPos) + m_VerticalOffset;
}
}
public override void OnDestroy()
{
base.OnDestroy();
if (m_UIState != null)
{
Destroy(m_UIState.gameObject);
}
}
}
}