using SanAndreasUnity.Behaviours.Vehicles;
using SanAndreasUnity.Importing.Animation;
using SanAndreasUnity.Utilities;
using System.Collections.Generic;
using System.Linq;
using UnityEngine;
using SanAndreasUnity.Net;

namespace SanAndreasUnity.Behaviours
{
	[DefaultExecutionOrder(-50)]
    [RequireComponent(typeof(Ped))]
    public class PlayerController : MonoBehaviour
    {
        
		public static PlayerController Instance { get { return Ped.Instance != null ? Ped.Instance.GetComponent<PlayerController>() : null; } }

		private Ped m_ped;

		public static bool _showVel = true;

        // Alpha speedometer
        private const float velTimer = 1 / 4f;

		private float m_velCounter = velTimer;

		private Vector3 m_lastPos = Vector3.zero,
			m_deltaPos = Vector3.zero;

        private Vector2 _mouseAbsolute;
        private Vector2 _smoothMouse = Vector2.zero;
        
        
        public Vector2 CursorSensitivity = new Vector2(2f, 2f);

        public Vector2 clampInDegrees = new Vector2(90, 60);

		public float EnterVehicleRadius { get { return m_ped.EnterVehicleRadius; } }

        public Vector2 smoothing = new Vector2(10, 10);
		[SerializeField] private bool m_doSmooth = true;

		[SerializeField] private bool m_smoothMovement = false;

        public float CurVelocity { get { return m_deltaPos.magnitude * 3.6f / velTimer; } }



        private void Awake()
        {
            m_ped = GetComponent<Ped>();

        }


        private void OnGUI()
        {
			if (!m_ped.IsControlledByLocalPlayer)
				return;

			if (!Loader.HasLoaded)
				return;
			
            
            // show that we are in flying state
			if (m_ped.CurrentState is Peds.States.FlyState)
            {
                int height = 25;
                GUILayout.BeginArea(new Rect(Screen.width - 140, Screen.height - height, 140, height));
				GUILayout.Label("Flying-mode enabled!");
				GUILayout.EndArea();
            }

            if (_showVel)
                GUI.Label(GUIUtils.GetCornerRect(ScreenCorner.TopLeft, 100, 25, new Vector2(5, 5)), string.Format("{0:0.0} km/h", m_deltaPos.magnitude * 3.6f / velTimer), new GUIStyle("label") { alignment = TextAnchor.MiddleCenter });

			// show current ped state
			GUI.Label (GUIUtils.GetCornerRect(ScreenCorner.BottomLeft, 250, 50), string.Format("Current ped state: {0}", m_ped.CurrentState != null ? m_ped.CurrentState.GetType().Name : "none") );
            
        }

        private void FixedUpdate()
        {
            m_velCounter -= Time.deltaTime;
            if (m_velCounter <= 0)
            {
                Vector3 t = new Vector3(transform.position.x, 0, transform.position.z);

                m_deltaPos = t - m_lastPos;
                m_lastPos = t;

                m_velCounter = velTimer;
            }
        }

        private void Update()
        {
			if (!m_ped.IsControlledByLocalPlayer)
				return;
			
			// FIXME: this should not be here
            if (Input.GetKeyDown(KeyCode.F9))
                _showVel = !_showVel;

            if (!Loader.HasLoaded)
                return;


			// reset ped input
			m_ped.ResetMovementInput ();
			m_ped.MouseMoveInput = Vector2.zero;
			m_ped.MouseScrollInput = Vector2.zero;
			

			if (!GameManager.CanPlayerReadInput()) return;


			// states must be read before events, otherwise callback functions for events will not have access
			// to states (they will always be unpressed/reset, because we did a reset above)
			this.ReadStates ();
			this.ReadEvents ();


        }

		void ReadStates()
		{

			this.ReadCameraInput ();

			m_ped.MouseScrollInput = Input.mouseScrollDelta;


			m_ped.IsAimOn = Input.GetButton ("RightClick");
			m_ped.IsFireOn = Input.GetButton ("LeftClick");

			m_ped.IsJumpOn = Input.GetButton ("Jump");


			Vector3 inputMove = Vector3.zero;
			if (m_smoothMovement)
				inputMove = new Vector3 (Input.GetAxis ("Horizontal"), 0f, Input.GetAxis ("Vertical"));
			else
				inputMove = new Vector3 (Input.GetAxisRaw ("Horizontal"), 0f, Input.GetAxisRaw ("Vertical"));

			if (inputMove.sqrMagnitude > 0f)
			{
				inputMove.Normalize();

				if (Input.GetButton ("Walk"))
					m_ped.IsWalkOn = true;
				else if (Input.GetButton ("Sprint"))
					m_ped.IsSprintOn = true;
				else
					m_ped.IsRunOn = true;

			}

			if (m_ped.Camera != null)
				m_ped.Movement = m_ped.Camera.transform.TransformVector (inputMove).normalized;
			else
				m_ped.Movement = inputMove.normalized;

			if (m_ped.Movement.sqrMagnitude > float.Epsilon) {
				// only assign heading if there is any movement - we don't want the heading to be zero vector
				m_ped.Heading = m_ped.Movement;
			}

		}

		void ReadEvents()
		{

			if (Input.GetButtonDown ("LeftClick"))
				m_ped.OnFireButtonPressed ();

			if (Input.GetButtonDown ("RightClick"))
				m_ped.OnAimButtonPressed ();

			if (Input.GetKeyDown (KeyCode.Q))
				m_ped.OnPreviousWeaponButtonPressed();
			else if (Input.GetKeyDown (KeyCode.E))
				m_ped.OnNextWeaponButtonPressed();

			if (Input.GetButtonDown("Use"))
				m_ped.OnSubmitPressed ();

			if (Input.GetButtonDown("Jump"))
				m_ped.OnJumpButtonPressed ();

			if (Input.GetKeyDown(KeyCode.C))
				m_ped.OnCrouchButtonPressed ();

			if (Input.GetKeyDown (KeyCode.T))
				m_ped.OnFlyButtonPressed();

			if (Input.GetKeyDown (KeyCode.R))
				m_ped.OnFlyThroughButtonPressed();

		}

		private void ReadCameraInput ()
		{

			if (GameManager.CanPlayerReadInput())
			{
				// rotate camera

				// FIXME: Camera rotation should be done by ped's current state. We should only assign mouse move input.

				var mouseDelta = new Vector2(Input.GetAxisRaw("Mouse X"), Input.GetAxisRaw("Mouse Y"));
                var rightAnalogDelta = new Vector2(Input.GetAxisRaw("Joystick X"), Input.GetAxisRaw("Joystick Y"));

				Vector2 totalMouseDelta = mouseDelta + rightAnalogDelta;

				totalMouseDelta = Vector2.Scale (totalMouseDelta, this.CursorSensitivity);

				m_ped.MouseMoveInput = totalMouseDelta;


				if (m_doSmooth)
				{
					
					_smoothMouse.x = Mathf.Lerp (_smoothMouse.x, totalMouseDelta.x, 1f / smoothing.x);
					_smoothMouse.y = Mathf.Lerp (_smoothMouse.y, totalMouseDelta.y, 1f / smoothing.y);

					_mouseAbsolute += _smoothMouse;
				}
				else
				{
					_mouseAbsolute += totalMouseDelta;
				}


				if (clampInDegrees.y > 0)
					_mouseAbsolute.y = Mathf.Clamp(_mouseAbsolute.y, -clampInDegrees.y, clampInDegrees.y);
				
			}


		}


        private void OnDrawGizmosSelected()
        {
			if (null == m_ped)
				return;

            Gizmos.color = Color.white;

			// draw enter vehicle radius
            Gizmos.DrawWireSphere(transform.position, EnterVehicleRadius);

			// find closest vehicle in entering range

            var vehicles = FindObjectsOfType<Vehicle>()
                .Where(x => Vector3.Distance(transform.position, x.FindClosestSeatTransform(transform.position).position) < EnterVehicleRadius)
                .OrderBy(x => Vector3.Distance(transform.position, x.FindClosestSeatTransform(transform.position).position)).ToArray();

            foreach (var vehicle in vehicles)
            {
				// draw all seats
                foreach (var seat in vehicle.Seats)
                {
                    Gizmos.color = Color.red;
                    Gizmos.DrawWireSphere(seat.Parent.position, 0.1f);
                }

				// draw closest seat

                var closestSeat = vehicle.FindClosestSeat(transform.position);

                if (closestSeat != Vehicle.SeatAlignment.None)
                {
                    var closestSeatTransform = vehicle.GetSeatTransform(closestSeat);

                    Gizmos.color = Color.green;
                    Gizmos.DrawWireSphere(closestSeatTransform.position, 0.1f);
                    Gizmos.DrawLine(transform.position, closestSeatTransform.position);
                }

                break;
            }
        }

    }
}