using UnityEngine;
namespace Mirror
{
    /// 
    /// This component works in conjunction with the NetworkRoomManager to make up the multiplayer room system.
    /// The RoomPrefab object of the NetworkRoomManager must have this component on it. This component holds basic room player data required for the room to function. Game specific data for room players can be put in other components on the RoomPrefab or in scripts derived from NetworkRoomPlayer.
    /// 
    [DisallowMultipleComponent]
    [AddComponentMenu("Network/NetworkRoomPlayer")]
    [HelpURL("https://mirror-networking.gitbook.io/docs/components/network-room-player")]
    public class NetworkRoomPlayer : NetworkBehaviour
    {
        /// 
        /// This flag controls whether the default UI is shown for the room player.
        /// As this UI is rendered using the old GUI system, it is only recommended for testing purposes.
        /// 
        [Tooltip("This flag controls whether the default UI is shown for the room player")]
        public bool showRoomGUI = true;
        [Header("Diagnostics")]
        /// 
        /// Diagnostic flag indicating whether this player is ready for the game to begin.
        /// Invoke CmdChangeReadyState method on the client to set this flag.
        /// When all players are ready to begin, the game will start. This should not be set directly, CmdChangeReadyState should be called on the client to set it on the server.
        /// 
        [Tooltip("Diagnostic flag indicating whether this player is ready for the game to begin")]
        [SyncVar(hook = nameof(ReadyStateChanged))]
        public bool readyToBegin;
        /// 
        /// Diagnostic index of the player, e.g. Player1, Player2, etc.
        /// 
        [Tooltip("Diagnostic index of the player, e.g. Player1, Player2, etc.")]
        [SyncVar(hook = nameof(IndexChanged))]
        public int index;
        #region Unity Callbacks
        /// 
        /// Do not use Start - Override OnStartHost / OnStartClient instead!
        /// 
        public void Start()
        {
            if (NetworkManager.singleton is NetworkRoomManager room)
            {
                // NetworkRoomPlayer object must be set to DontDestroyOnLoad along with NetworkRoomManager
                // in server and all clients, otherwise it will be respawned in the game scene which would
                // have undesirable effects.
                if (room.dontDestroyOnLoad)
                    DontDestroyOnLoad(gameObject);
                room.roomSlots.Add(this);
                if (NetworkServer.active)
                    room.RecalculateRoomPlayerIndices();
                if (NetworkClient.active)
                    room.CallOnClientEnterRoom();
            }
            else Debug.LogError("RoomPlayer could not find a NetworkRoomManager. The RoomPlayer requires a NetworkRoomManager object to function. Make sure that there is one in the scene.");
        }
        public virtual void OnDisable()
        {
            if (NetworkClient.active && NetworkManager.singleton is NetworkRoomManager room)
            {
                // only need to call this on client as server removes it before object is destroyed
                room.roomSlots.Remove(this);
                room.CallOnClientExitRoom();
            }
        }
        #endregion
        #region Commands
        [Command]
        public void CmdChangeReadyState(bool readyState)
        {
            readyToBegin = readyState;
            NetworkRoomManager room = NetworkManager.singleton as NetworkRoomManager;
            if (room != null)
            {
                room.ReadyStatusChanged();
            }
        }
        #endregion
        #region SyncVar Hooks
        /// 
        /// This is a hook that is invoked on clients when the index changes.
        /// 
        /// The old index value
        /// The new index value
        public virtual void IndexChanged(int oldIndex, int newIndex) {}
        /// 
        /// This is a hook that is invoked on clients when a RoomPlayer switches between ready or not ready.
        /// This function is called when the a client player calls CmdChangeReadyState.
        /// 
        /// New Ready State
        public virtual void ReadyStateChanged(bool oldReadyState, bool newReadyState) {}
        #endregion
        #region Room Client Virtuals
        /// 
        /// This is a hook that is invoked on clients for all room player objects when entering the room.
        /// Note: isLocalPlayer is not guaranteed to be set until OnStartLocalPlayer is called.
        /// 
        public virtual void OnClientEnterRoom() {}
        /// 
        /// This is a hook that is invoked on clients for all room player objects when exiting the room.
        /// 
        public virtual void OnClientExitRoom() {}
        #endregion
        #region Optional UI
        /// 
        /// Render a UI for the room. Override to provide your own UI
        /// 
        public virtual void OnGUI()
        {
            if (!showRoomGUI)
                return;
            NetworkRoomManager room = NetworkManager.singleton as NetworkRoomManager;
            if (room)
            {
                if (!room.showRoomGUI)
                    return;
                if (!NetworkManager.IsSceneActive(room.RoomScene))
                    return;
                DrawPlayerReadyState();
                DrawPlayerReadyButton();
            }
        }
        void DrawPlayerReadyState()
        {
            GUILayout.BeginArea(new Rect(20f + (index * 100), 200f, 90f, 130f));
            GUILayout.Label($"Player [{index + 1}]");
            if (readyToBegin)
                GUILayout.Label("Ready");
            else
                GUILayout.Label("Not Ready");
            if (((isServer && index > 0) || isServerOnly) && GUILayout.Button("REMOVE"))
            {
                // This button only shows on the Host for all players other than the Host
                // Host and Players can't remove themselves (stop the client instead)
                // Host can kick a Player this way.
                GetComponent().connectionToClient.Disconnect();
            }
            GUILayout.EndArea();
        }
        void DrawPlayerReadyButton()
        {
            if (NetworkClient.active && isLocalPlayer)
            {
                GUILayout.BeginArea(new Rect(20f, 300f, 120f, 20f));
                if (readyToBegin)
                {
                    if (GUILayout.Button("Cancel"))
                        CmdChangeReadyState(false);
                }
                else
                {
                    if (GUILayout.Button("Ready"))
                        CmdChangeReadyState(true);
                }
                GUILayout.EndArea();
            }
        }
        #endregion
    }
}