/*
* author: mark joshwel
* date: 11/8/2024
* description: game manager singleton for a single source of truth state management
*/
using System;
using UnityEngine;
using UnityEngine.InputSystem;
///
/// singleton class for managing the game state as a single source of truth
///
public class GameManager : MonoBehaviour
{
///
/// enum for available menus in the game, for use with ShowMenu()
///
public enum DisplayState
{
Game,
ScreenMainMenu,
ScreenOptionsMenu,
OverlayPauseMenu,
OverlayCompleteUnderTimeMenu,
OverlayFailedOverTimeMenu,
UnassociatedState
}
///
/// singleton pattern: define instance field for accessing the singleton elsewhere
///
public static GameManager Instance;
///
/// property to store the heads-up display game object
///
[SerializeField] private GameObject headsUpDisplay;
///
/// the current state of the game
///
private DisplayState _state = DisplayState.UnassociatedState;
///
/// property to check if the game is paused based on the current DisplayState
///
public bool Paused => _state != DisplayState.Game;
///
/// function to set doesn't destroy on load and checks for multiple instances
///
/// thrown if a Heads-Up Display game object isn't set in the properties
private void Awake()
{
// check if instance hasn't been set yet
if (Instance == null)
{
// set this instance as the singleton instance
Instance = this;
// don't destroy this instance on a scene load
DontDestroyOnLoad(gameObject);
Debug.Log("GameManager: Awake as singleton instance");
}
// check if instance is already set and it's not this instance
else if (Instance != null && Instance != this)
{
Debug.Log("GameManager: Awake as non-singleton instance, destroying self");
// destroy the new instance if it's not the singleton instance
Destroy(gameObject);
}
// check if the heads-up display is set
if (headsUpDisplay == null)
throw new NullReferenceException("GameManager: heads-up display is not set in game manager properties");
}
///
/// called when the game starts, sets state to the main menu
///
// /// generic exception it couldn't verify a safe state when starting the game
private void Start()
{
// set to the main menu state
SetDisplayState(DisplayState.ScreenMainMenu);
// pause the game
PauseGameHelper(DisplayState.ScreenMainMenu);
}
///
/// helper function to hide any menu that is currently showing
///
private void HideMenuHelper()
{
// get all child menus in the "Menus" parent object
foreach (var menu in GameObject.FindGameObjectsWithTag("Interfaces"))
foreach (Transform menuChild in menu.transform)
{
// disable the menu if it's currently active
if (!menuChild.gameObject.activeSelf) continue;
Debug.Log($"GameManager.HideMenuHelper: hiding menu '{menuChild}'");
menuChild.gameObject.SetActive(false);
}
}
///
/// helper function for SetDisplayState() to pause the game,
/// called before the incoming game state is set
///
/// the to-be-set state of the game
private void PauseGameHelper(DisplayState incomingState)
{
// if we're transitioning from a state of gameplay to a state of non-gameplay,
// then we should pause the game
if (_state == DisplayState.Game && incomingState != DisplayState.Game)
{
Debug.Log("GameManager.PauseGameHelper: actually pausing game");
Time.timeScale = 0f;
Cursor.lockState = CursorLockMode.None;
Cursor.visible = true;
}
// or if the incoming state is the main menu, we should probably free the cursor
else if (incomingState == DisplayState.ScreenMainMenu)
{
Debug.Log("GameManager.PauseGameHelper: freeing cursor for main menu");
Cursor.lockState = CursorLockMode.None;
Cursor.visible = true;
Debug.Log(GameObject.FindGameObjectWithTag("Player"));
var playerInput = GameObject.Find("PlayerArmature")?.GetComponent();
Debug.Log($"playerController={playerInput}");
if (playerInput != null)
{
Debug.Log("GameManager.PauseGameHelper: sentencing player input/control to a ball and chain boowomp");
playerInput.enabled = false;
}
}
// if the current and incoming states are the same, then we shouldn't do anything,
// so we return early
if (_state == incomingState)
{
Debug.Log("GameManager.PauseGameHelper: states are the same, returning early");
return;
}
// hide any menu that is currently showing
HideMenuHelper();
// get all child interfaces in the "Interfaces" parent object
foreach (var interfaces in GameObject.FindGameObjectsWithTag("Interfaces"))
foreach (Transform menu in interfaces.transform)
{
// show the menu based on the incoming state
// get the associated state of the menu
var possibleMenuObject = menu.gameObject.GetComponent();
if (possibleMenuObject == null) continue;
// guard clause if the menu isn't what we're looking for
if (possibleMenuObject.associatedState != incomingState)
continue;
// if the associated state is the same as the incoming state, then show the menu
Debug.Log($"GameManager.PauseGameHelper: showing menu for {incomingState}");
menu.gameObject.SetActive(true);
}
}
///
/// helper function for SetDisplayState() to resume the game,
/// called before the incoming game state is set
///
/// the to-be-set state of the game
private void ResumeGameHelper(DisplayState incomingState)
{
// if we're NOT transitioning from a state of non-gameplay to a state of gameplay,
// (which means currently we are in a state of gameplay),
// then we shouldn't do anything, because the game is already running,
// so we return early
if (_state == DisplayState.Game || incomingState != DisplayState.Game)
{
Debug.Log(
"GameManager.ResumeGameHelper: returning prematurely as" +
$" _state={_state} and incomingState={incomingState}");
return;
}
// else, we should resume the game
Debug.Log("GameManager.ResumeGameHelper: resuming game");
Time.timeScale = 1f;
Cursor.lockState = CursorLockMode.Locked;
Cursor.visible = false;
// hide any menu that is currently showing
HideMenuHelper();
// free the character controller
var playerInput = GameObject.Find("PlayerArmature")?.GetComponent();
if (playerInput == null) return;
Debug.Log("GameManager.ResumeGameHelper: enabling player input/control");
playerInput.enabled = true;
}
///
/// function to show a menu based on the enum passed,
/// and any other necessary actions
///
/// the game menu to show
public void SetDisplayState(DisplayState displayState)
{
// check if the game is paused or not
if (displayState is DisplayState.Game or DisplayState.UnassociatedState)
{
Debug.Log($"GameManager.SetDisplayState({displayState}): pre-resume helper");
ResumeGameHelper(displayState);
}
else
{
Debug.Log($"GameManager.SetDisplayState({displayState}): pre-pause helper");
PauseGameHelper(displayState);
}
Debug.Log($"GameManager.SetDisplayState({displayState}): post-pause/resume helper");
// set the state of the game to the incoming state
_state = displayState;
Debug.Log($"GameManager.SetDisplayState({displayState}): state is now {displayState}");
// if we're transitioning into gameplay or into the main menu,
// we'll need a post-step to enable the correct camera
if (displayState is not (DisplayState.Game or DisplayState.ScreenMainMenu))
{
Debug.Log($"GameManager.SetDisplayState({displayState}): skipping post-step");
return;
}
// disabling (1): disable the 'Menu Camera' camera
var menuCameraObject = GameObject.Find("Menu Camera")?.GetComponent();
if (menuCameraObject != null)
{
Debug.Log("GameManager.SetDisplayState: disabling 'Menu Camera' camera");
menuCameraObject.enabled = false;
}
// disabling (2): get all MainCamera-tagged objects and disable them
foreach (var mainCameraObject in GameObject.FindGameObjectsWithTag("MainCamera"))
{
Debug.Log($"GameManager.SetDisplayState: disabling 'MainCamera' camera {mainCameraObject}");
// find the camera component
var potentialCamera = mainCameraObject.GetComponent();
// if the object doesn't have a camera component, skip it
if (potentialCamera == null) continue;
// disable the camera
potentialCamera.enabled = false;
}
// declare a target camera to enable
GameObject targetCameraObject;
// switch on the state to enable the correct camera
// could be an if statement, but unity optimizes switch statements anyway
switch (displayState)
{
// if we're transitioning to the main menu state,
// change the camera to the main menu camera named very specifically such
case DisplayState.ScreenMainMenu:
Debug.Log("GameManager.SetDisplayState: targeting 'Menu Camera' camera");
targetCameraObject = GameObject.Find("Menu Camera");
break;
// else, we should enable the players' main camera
default:
Debug.Log("GameManager.SetDisplayState: targeting 'MainCamera' camera");
targetCameraObject = GameObject.Find("MainCamera");
break;
}
// find the camera component and enable it
var targetCamera = targetCameraObject?.GetComponent();
if (targetCamera == null) return;
targetCamera.enabled = true;
Debug.Log("GameManager.SetDisplayState: enabled target camera");
}
///
/// wrapper function to quit the game
/// in case of any cleanup needed
///
public void Quit()
{
// quit the application
Application.Quit();
}
///
/// resets game state and starts a new game, will call SetDisplayState()
///
public void NewGame()
{
// set to game state
SetDisplayState(DisplayState.Game);
// TODO
}
}