using System.Collections;
using System.Collections.Generic;
using UnityEngine;
namespace Autohand {
    [System.Serializable]
    public struct HandPoseData{
        public Vector3 handOffset;
        /// DEPRECATED -> USE LOCAL_QUATERNION_OFFSET INSTEAD
        public Vector3 rotationOffset;
        public Quaternion localQuaternionOffset;
        public Vector3[] posePositions;
        public Quaternion[] poseRotations;
        
        /// Creates a new pose using the current hand relative to a given grabbable
        public HandPoseData(Hand hand, Grabbable grabbable) {
            posePositions = new Vector3[0];
            poseRotations = new Quaternion[0];
            handOffset = new Vector3();
            rotationOffset = Vector3.zero;
            localQuaternionOffset = Quaternion.identity;
            SavePose(hand, grabbable.transform);
        }
        /// Creates a new pose using the current hand relative to a given grabbable
        public HandPoseData(Hand hand, Transform point) {
            posePositions = new Vector3[0];
            poseRotations = new Quaternion[0];
            handOffset = new Vector3();
            rotationOffset = Vector3.zero;
            localQuaternionOffset = Quaternion.identity;
            SavePose(hand, point);
        }
        
        /// Creates a new pose using the current hand shape
        public HandPoseData(Hand hand) {
            posePositions = new Vector3[0];
            poseRotations = new Quaternion[0];
            handOffset = new Vector3();
            rotationOffset = Vector3.zero;
            localQuaternionOffset = Quaternion.identity;
            SavePose(hand, null);
        }
        /// Creates a new pose using the current hand shape
        public HandPoseData(HandPoseData data) {
            posePositions = new Vector3[data.posePositions.Length];
            data.posePositions.CopyTo(posePositions, 0);
            poseRotations = new Quaternion[data.poseRotations.Length];
            data.poseRotations.CopyTo(poseRotations, 0);
            handOffset = data.handOffset;
            rotationOffset = data.rotationOffset;
            localQuaternionOffset = data.localQuaternionOffset;
        }
        public void SavePose(Hand hand, Transform relativeTo) {
            var posePositionsList = new List();
            var poseRotationsList = new List();
            
            if(relativeTo != null){
                var tempContainer = AutoHandExtensions.transformRuler;
                tempContainer.localScale = relativeTo.lossyScale;
                tempContainer.transform.position = relativeTo.position;
                tempContainer.transform.rotation = relativeTo.rotation;
                var handMatch = AutoHandExtensions.transformRulerChild;
                handMatch.transform.position = hand.transform.position;
                handMatch.transform.rotation = hand.transform.rotation;
                handOffset = handMatch.localPosition;
                localQuaternionOffset = handMatch.localRotation;
#if UNITY_EDITOR
                if(Application.isEditor && !Application.isPlaying)
                    GameObject.DestroyImmediate(tempContainer.gameObject);
#endif
            }
            else {
                handOffset = hand.transform.localPosition;
                localQuaternionOffset = hand.transform.localRotation;
            }
            rotationOffset = Vector3.zero;
            foreach(var finger in hand.fingers) {
                AssignChildrenPose(finger.transform);
            }
            void AssignChildrenPose(Transform obj) {
                AddPoint(obj.localPosition, obj.localRotation);
                for(int j = 0; j < obj.childCount; j++) {
                    AssignChildrenPose(obj.GetChild(j));
                }
            }
            void AddPoint(Vector3 pos, Quaternion rot) {
                posePositionsList.Add(pos);
                poseRotationsList.Add(rot);
            }
            
            posePositions = new Vector3[posePositionsList.Count];
            poseRotations = new Quaternion[posePositionsList.Count];
            for(int i = 0; i < posePositionsList.Count; i++) {
                posePositions[i] = posePositionsList[i];
                poseRotations[i] = poseRotationsList[i];
            }
        }
        public Quaternion GetRotationOffset(){
            if (rotationOffset != Vector3.zero)
                localQuaternionOffset = Quaternion.Euler(rotationOffset);
            return localQuaternionOffset;
        }
        public void SetPose(Hand hand, Transform relativeTo = null) {
            //This might prevent static poses from breaking from the update
            if(rotationOffset != Vector3.zero)
                localQuaternionOffset = Quaternion.Euler(rotationOffset);
            if(relativeTo != null && relativeTo != hand.transform) {
                var tempContainer = AutoHandExtensions.transformRuler;
                tempContainer.localScale = relativeTo.lossyScale;
                tempContainer.position = relativeTo.position;
                tempContainer.rotation = relativeTo.rotation;
                var handMatch = AutoHandExtensions.transformRulerChild;
                handMatch.localPosition = handOffset;
                handMatch.localRotation = localQuaternionOffset;
                hand.transform.position = handMatch.position;
                hand.transform.rotation = handMatch.rotation;
                hand.body.position = hand.transform.position;
                hand.body.rotation = hand.transform.rotation;
#if UNITY_EDITOR
                if(Application.isEditor && !Application.isPlaying)
                    GameObject.DestroyImmediate(tempContainer.gameObject);
#endif
            }
            int i = -1;
            void AssignChildrenPose(Transform obj, HandPoseData pose) {
                i++;
                obj.localPosition = pose.posePositions[i];
                obj.localRotation = pose.poseRotations[i];
                for(int j = 0; j < obj.childCount; j++) {
                    AssignChildrenPose(obj.GetChild(j), pose);
                }
            }
            if(posePositions != null)
                foreach(var finger in hand.fingers)
                    AssignChildrenPose(finger.transform, this);
        }
        /// Sets the finger pose without changing the hands position
        public void SetFingerPose(Hand hand, Transform relativeTo = null) {
            
            int i = -1;
            void AssignChildrenPose(Transform obj, HandPoseData pose) {
                i++;
                obj.localPosition = pose.posePositions[i];
                obj.localRotation = pose.poseRotations[i];
                for(int j = 0; j < obj.childCount; j++) {
                    AssignChildrenPose(obj.GetChild(j), pose);
                }
            }
            if(posePositions != null)
                foreach(var finger in hand.fingers)
                    AssignChildrenPose(finger.transform, this);
        }
        /// Sets the position without setting the finger pose
        public void SetPosition(Hand hand, Transform relativeTo = null) {
            //This might prevent static poses from breaking from the update
            if(rotationOffset != Vector3.zero)
                localQuaternionOffset = Quaternion.Euler(rotationOffset);
            if(relativeTo != null && relativeTo != hand.transform) {
                var tempContainer = AutoHandExtensions.transformRuler;
                tempContainer.localScale = relativeTo.lossyScale;
                tempContainer.position = relativeTo.position;
                tempContainer.rotation = relativeTo.rotation;
                var handMatch = AutoHandExtensions.transformRulerChild;
                handMatch.localPosition = handOffset;
                handMatch.localRotation = localQuaternionOffset;
                hand.transform.position = handMatch.position;
                hand.transform.rotation = handMatch.rotation;
                hand.body.position = hand.transform.position;
                hand.body.rotation = hand.transform.rotation;
#if UNITY_EDITOR
                if(Application.isEditor && !Application.isPlaying)
                    GameObject.DestroyImmediate(tempContainer.gameObject);
#endif
            }
        }
        public static HandPoseData LerpPose(HandPoseData from, HandPoseData to, float point) {
            var lerpPose = new HandPoseData();
            lerpPose.handOffset = Vector3.Lerp(from.handOffset, to.handOffset, point);
            lerpPose.localQuaternionOffset = Quaternion.Lerp(from.localQuaternionOffset, to.localQuaternionOffset, point);
            lerpPose.posePositions = new Vector3[from.posePositions.Length];
            lerpPose.poseRotations = new Quaternion[from.poseRotations.Length];
            for(int i = 0; i < from.posePositions.Length; i++) {
                lerpPose.posePositions[i] = Vector3.Lerp(from.posePositions[i], to.posePositions[i], point);
                lerpPose.poseRotations[i] = Quaternion.Lerp(from.poseRotations[i], to.poseRotations[i], point);
            }
            return lerpPose;
        }
        public static void LerpPose(ref HandPoseData lerpPose, HandPoseData from, HandPoseData to, float point) {
            lerpPose.handOffset = Vector3.Lerp(from.handOffset, to.handOffset, point);
            lerpPose.localQuaternionOffset = Quaternion.Lerp(from.localQuaternionOffset, to.localQuaternionOffset, point);
            lerpPose.posePositions = new Vector3[from.posePositions.Length];
            lerpPose.poseRotations = new Quaternion[from.poseRotations.Length];
            for(int i = 0; i < from.posePositions.Length; i++) {
                lerpPose.posePositions[i] = Vector3.Lerp(from.posePositions[i], to.posePositions[i], point);
                lerpPose.poseRotations[i] = Quaternion.Lerp(from.poseRotations[i], to.poseRotations[i], point);
            }
        }
    }
}