/* * 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 } }