using System;
using UnityEngine.Scripting.APIUpdating;
using UnityEngine.Serialization;
namespace UnityEngine.U2D.IK
{
///
/// Class for storing data for a 2D IK Chain.
///
[MovedFrom("UnityEngine.Experimental.U2D.IK")]
[Serializable]
public class IKChain2D
{
[SerializeField]
[FormerlySerializedAs("m_Target")]
Transform m_EffectorTransform;
[SerializeField]
[FormerlySerializedAs("m_Effector")]
Transform m_TargetTransform;
[SerializeField]
int m_TransformCount;
[SerializeField]
Transform[] m_Transforms;
[SerializeField]
Quaternion[] m_DefaultLocalRotations;
[SerializeField]
Quaternion[] m_StoredLocalRotations;
///
/// Lengths of IK Chain.
///
/// Array of lengths in the IK Chain.
protected float[] m_Lengths;
///
/// Get and set the transform used as the IK Effector.
///
public Transform effector
{
get => m_EffectorTransform;
set => m_EffectorTransform = value;
}
///
/// Get and set the transform used as the IK Target.
///
public Transform target
{
get => m_TargetTransform;
set => m_TargetTransform = value;
}
///
/// Get the transforms that are used in the IK Chain.
///
public Transform[] transforms => m_Transforms;
///
/// Get the root transform for the IK Chain.
///
public Transform rootTransform
{
get
{
if (m_Transforms != null && transformCount > 0 && m_Transforms.Length == transformCount)
return m_Transforms[0];
return null;
}
}
Transform lastTransform
{
get
{
if (m_Transforms != null && transformCount > 0 && m_Transforms.Length == transformCount)
return m_Transforms[transformCount - 1];
return null;
}
}
///
/// Get and Set the number of transforms in the IK Chain.
///
public int transformCount
{
get => m_TransformCount;
set => m_TransformCount = Mathf.Max(0, value);
}
///
/// Returns true if the IK Chain is valid. False otherwise.
///
public bool isValid => Validate();
///
/// Gets the length of the IK Chain.
///
public float[] lengths
{
get
{
if (isValid)
{
PrepareLengths();
return m_Lengths;
}
return null;
}
}
bool Validate()
{
if (effector == null)
return false;
if (transformCount == 0)
return false;
if (m_Transforms == null || m_Transforms.Length != transformCount)
return false;
if (m_DefaultLocalRotations == null || m_DefaultLocalRotations.Length != transformCount)
return false;
if (m_StoredLocalRotations == null || m_StoredLocalRotations.Length != transformCount)
return false;
if (rootTransform == null)
return false;
if (lastTransform != effector)
return false;
return !target || !IKUtility.IsDescendentOf(target, rootTransform);
}
///
/// Initialize the IK Chain.
///
public void Initialize()
{
if (effector == null || transformCount == 0 || IKUtility.GetAncestorCount(effector) < transformCount - 1)
return;
m_Transforms = new Transform[transformCount];
m_DefaultLocalRotations = new Quaternion[transformCount];
m_StoredLocalRotations = new Quaternion[transformCount];
var currentTransform = effector;
var index = transformCount - 1;
while (currentTransform && index >= 0)
{
m_Transforms[index] = currentTransform;
m_DefaultLocalRotations[index] = currentTransform.localRotation;
currentTransform = currentTransform.parent;
--index;
}
}
void PrepareLengths()
{
var currentTransform = effector;
var index = transformCount - 1;
if (m_Lengths == null || m_Lengths.Length != transformCount - 1)
m_Lengths = new float[transformCount - 1];
while (currentTransform && index >= 0)
{
if (currentTransform.parent && index > 0)
m_Lengths[index - 1] = (currentTransform.position - currentTransform.parent.position).magnitude;
currentTransform = currentTransform.parent;
--index;
}
}
///
/// Restores the IK Chain to it's default pose.
///
/// True to constrain the target rotation. False otherwise.
public void RestoreDefaultPose(bool targetRotationIsConstrained)
{
var count = targetRotationIsConstrained ? transformCount : transformCount - 1;
for (var i = 0; i < count; ++i)
m_Transforms[i].localRotation = m_DefaultLocalRotations[i];
}
///
/// Explicitly stores the local rotation.
///
public void StoreLocalRotations()
{
for (var i = 0; i < m_Transforms.Length; ++i)
m_StoredLocalRotations[i] = m_Transforms[i].localRotation;
}
///
/// Blend between Forward Kinematics and Inverse Kinematics.
///
/// Weight for blend
/// True to constrain target rotation. False otherwise.
public void BlendFkToIk(float finalWeight, bool targetRotationIsConstrained)
{
var count = targetRotationIsConstrained ? transformCount : transformCount - 1;
for (var i = 0; i < count; ++i)
m_Transforms[i].localRotation = Quaternion.Slerp(m_StoredLocalRotations[i], m_Transforms[i].localRotation, finalWeight);
}
}
}