You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

233 lines
7.9 KiB
C#

6 months ago
using UnityEngine;
using System.Collections;
namespace RootMotion.FinalIK {
/// <summary>
/// %Constraint used for fixing bend direction of 3-segment node chains in a node based %IK solver.
/// </summary>
[System.Serializable]
public class IKConstraintBend {
#region Main Interface
/// <summary>
/// The first bone.
/// </summary>
public Transform bone1;
/// <summary>
/// The second (bend) bone.
/// </summary>
public Transform bone2;
/// <summary>
/// The third bone.
/// </summary>
public Transform bone3;
/// <summary>
/// The bend goal Transform.
/// </summary>
public Transform bendGoal;
/// <summary>
/// The bend direction.
/// </summary>
public Vector3 direction = Vector3.right;
/// <summary>
/// The bend rotation offset.
/// </summary>
public Quaternion rotationOffset;
/// <summary>
/// The weight. If weight is 1, will override effector rotation and the joint will be rotated at the direction. This enables for direct manipulation of the bend direction independent of effector rotation.
/// </summary>
[Range(0f, 1f)]
public float weight = 0f;
/// <summary>
/// Determines whether this IKConstraintBend is valid.
/// </summary>
public bool IsValid(IKSolverFullBody solver, Warning.Logger logger) {
if (bone1 == null || bone2 == null || bone3 == null) {
if (logger != null) logger("Bend Constraint contains a null reference.");
return false;
}
if (solver.GetPoint(bone1) == null) {
if (logger != null) logger("Bend Constraint is referencing to a bone '" + bone1.name + "' that does not excist in the Node Chain.");
return false;
}
if (solver.GetPoint(bone2) == null) {
if (logger != null) logger("Bend Constraint is referencing to a bone '" + bone2.name + "' that does not excist in the Node Chain.");
return false;
}
if (solver.GetPoint(bone3) == null) {
if (logger != null) logger("Bend Constraint is referencing to a bone '" + bone3.name + "' that does not excist in the Node Chain.");
return false;
}
return true;
}
#endregion Main Interface
public Vector3 defaultLocalDirection, defaultChildDirection;
[System.NonSerializedAttribute] public float clampF = 0.505f;
//private IKSolver.Node node1, node2, node3;
private int chainIndex1;
private int nodeIndex1;
private int chainIndex2;
private int nodeIndex2;
private int chainIndex3;
private int nodeIndex3;
public bool initiated { get; private set; }
private bool limbOrientationsSet;
public IKConstraintBend() {}
public IKConstraintBend(Transform bone1, Transform bone2, Transform bone3) {
SetBones(bone1, bone2, bone3);
}
public void SetBones(Transform bone1, Transform bone2, Transform bone3) {
this.bone1 = bone1;
this.bone2 = bone2;
this.bone3 = bone3;
}
/*
* Initiate the constraint and set defaults
* */
public void Initiate(IKSolverFullBody solver) {
solver.GetChainAndNodeIndexes(bone1, out chainIndex1, out nodeIndex1);
solver.GetChainAndNodeIndexes(bone2, out chainIndex2, out nodeIndex2);
solver.GetChainAndNodeIndexes(bone3, out chainIndex3, out nodeIndex3);
// Find the default bend direction orthogonal to the chain direction
direction = OrthoToBone1(solver, OrthoToLimb(solver, bone2.position - bone1.position));
if (!limbOrientationsSet) {
// Default bend direction relative to the first node
defaultLocalDirection = Quaternion.Inverse(bone1.rotation) * direction;
// Default plane normal
Vector3 defaultNormal = Vector3.Cross((bone3.position - bone1.position).normalized, direction);
// Default plane normal relative to the third node
defaultChildDirection = Quaternion.Inverse(bone3.rotation) * defaultNormal;
}
initiated = true;
}
/*
* Make the limb bend towards the specified local directions of the bones
* */
public void SetLimbOrientation(Vector3 upper, Vector3 lower, Vector3 last) {
if (upper == Vector3.zero) Debug.LogError("Attempting to set limb orientation to Vector3.zero axis");
if (lower == Vector3.zero) Debug.LogError("Attempting to set limb orientation to Vector3.zero axis");
if (last == Vector3.zero) Debug.LogError("Attempting to set limb orientation to Vector3.zero axis");
// Default bend direction relative to the first node
defaultLocalDirection = upper.normalized;
defaultChildDirection = last.normalized;
limbOrientationsSet = true;
}
/*
* Limits the bending joint of the limb to 90 degrees from the default 90 degrees of bend direction
* */
public void LimitBend(float solverWeight, float positionWeight) {
if (!initiated) return;
Vector3 normalDirection = bone1.rotation * -defaultLocalDirection;
Vector3 axis2 = bone3.position - bone2.position;
// Clamp the direction from knee/elbow to foot/hand to valid range (90 degrees from right-angledly bent limb)
bool changed = false;
Vector3 clampedAxis2 = V3Tools.ClampDirection(axis2, normalDirection, clampF * solverWeight, 0, out changed);
Quaternion bone3Rotation = bone3.rotation;
if (changed) {
Quaternion f = Quaternion.FromToRotation(axis2, clampedAxis2);
bone2.rotation = f * bone2.rotation;
}
// Rotating bend direction to normal when the limb is stretched out
if (positionWeight > 0f) {
Vector3 normal = bone2.position - bone1.position;
Vector3 tangent = bone3.position - bone2.position;
Vector3.OrthoNormalize(ref normal, ref tangent);
Quaternion q = Quaternion.FromToRotation(tangent, normalDirection);
bone2.rotation = Quaternion.Lerp(bone2.rotation, q * bone2.rotation, positionWeight * solverWeight);
}
if (changed || positionWeight > 0f) bone3.rotation = bone3Rotation;
}
/*
* Computes the direction from the first node to the second node
* */
public Vector3 GetDir(IKSolverFullBody solver) {
if (!initiated) return Vector3.zero;
float w = weight * solver.IKPositionWeight;
// Apply the bend goal
if (bendGoal != null) {
Vector3 b = bendGoal.position - solver.GetNode(chainIndex1, nodeIndex1).solverPosition;
if (b != Vector3.zero) direction = b;
}
if (w >= 1f) return direction.normalized;
Vector3 solverDirection = solver.GetNode(chainIndex3, nodeIndex3).solverPosition - solver.GetNode(chainIndex1, nodeIndex1).solverPosition;
// Get rotation from animated limb direction to solver limb direction
Quaternion f = Quaternion.FromToRotation(bone3.position - bone1.position, solverDirection);
// Rotate the default bend direction by f
Vector3 dir = f * (bone2.position - bone1.position);
// Effector rotation
if (solver.GetNode(chainIndex3, nodeIndex3).effectorRotationWeight > 0f) {
// Bend direction according to the effector rotation
Vector3 effectorDirection = -Vector3.Cross(solverDirection, solver.GetNode(chainIndex3, nodeIndex3).solverRotation * defaultChildDirection);
dir = Vector3.Lerp(dir, effectorDirection, solver.GetNode(chainIndex3, nodeIndex3).effectorRotationWeight);
}
// Rotation Offset
if (rotationOffset != Quaternion.identity) {
Quaternion toOrtho = Quaternion.FromToRotation(rotationOffset * solverDirection, solverDirection);
dir = toOrtho * rotationOffset * dir;
}
if (w <= 0f) return dir;
return Vector3.Lerp(dir, direction.normalized, w);
}
/*
* Ortho-Normalize a vector to the chain direction
* */
private Vector3 OrthoToLimb(IKSolverFullBody solver, Vector3 tangent) {
Vector3 normal = solver.GetNode(chainIndex3, nodeIndex3).solverPosition - solver.GetNode(chainIndex1, nodeIndex1).solverPosition;
Vector3.OrthoNormalize(ref normal, ref tangent);
return tangent;
}
/*
* Ortho-Normalize a vector to the first bone direction
* */
private Vector3 OrthoToBone1(IKSolverFullBody solver, Vector3 tangent) {
Vector3 normal = solver.GetNode(chainIndex2, nodeIndex2).solverPosition - solver.GetNode(chainIndex1, nodeIndex1).solverPosition;
Vector3.OrthoNormalize(ref normal, ref tangent);
return tangent;
}
}
}