using UnityEngine;
using System.Collections;
namespace RootMotion.FinalIK {
///
/// Base class for all FBBIK effector positionOffset modifiers. Works with animatePhysics, safe delegates, offset limits.
///
public abstract class OffsetModifier: MonoBehaviour {
///
/// Limiting effector position offsets
///
[System.Serializable]
public class OffsetLimits {
[Tooltip("The effector type (this is just an enum)")]
public FullBodyBipedEffector effector;
[Tooltip("Spring force, if zero then this is a hard limit, if not, offset can exceed the limit.")]
public float spring = 0f;
[Tooltip("Which axes to limit the offset on?")]
public bool x, y, z;
[Tooltip("The limits")]
public float minX, maxX, minY, maxY, minZ, maxZ;
// Apply the limit to the effector
public void Apply(IKEffector e, Quaternion rootRotation) {
Vector3 offset = Quaternion.Inverse(rootRotation) * e.positionOffset;
if (spring <= 0f) {
// Hard limits
if (x) offset.x = Mathf.Clamp(offset.x, minX, maxX);
if (y) offset.y = Mathf.Clamp(offset.y, minY, maxY);
if (z) offset.z = Mathf.Clamp(offset.z, minZ, maxZ);
} else {
// Soft limits
if (x) offset.x = SpringAxis(offset.x, minX, maxX);
if (y) offset.y = SpringAxis(offset.y, minY, maxY);
if (z) offset.z = SpringAxis(offset.z, minZ, maxZ);
}
// Apply to the effector
e.positionOffset = rootRotation * offset;
}
// Just math for limiting floats
private float SpringAxis(float value, float min, float max) {
if (value > min && value < max) return value;
if (value < min) return Spring(value, min, true);
return Spring(value, max, false);
}
// Spring math
private float Spring(float value, float limit, bool negative) {
float illegal = value - limit;
float s = illegal * spring;
if (negative) return value + Mathf.Clamp(-s, 0, -illegal);
return value - Mathf.Clamp(s, 0, illegal);
}
}
[Tooltip("The master weight")]
public float weight = 1f;
[Tooltip("Reference to the FBBIK component")]
public FullBodyBipedIK ik;
// not using Time.deltaTime or Time.fixedDeltaTime here, because we don't know if animatePhysics is true or not on the character, so we have to keep track of time ourselves.
protected float deltaTime { get { return Time.time - lastTime; }}
protected abstract void OnModifyOffset();
protected float lastTime;
protected virtual void Start() {
StartCoroutine(Initiate());
}
private IEnumerator Initiate() {
while (ik == null) yield return null;
// You can use just LateUpdate, but note that it doesn't work when you have animatePhysics turned on for the character.
ik.solver.OnPreUpdate += ModifyOffset;
lastTime = Time.time;
}
// The main function that checks for all conditions and calls OnModifyOffset if they are met
private void ModifyOffset() {
if (!enabled) return;
if (weight <= 0f) return;
if (ik == null) return;
weight = Mathf.Clamp(weight, 0f, 1f);
if (deltaTime <= 0f) return;
OnModifyOffset();
lastTime = Time.time;
}
protected void ApplyLimits(OffsetLimits[] limits) {
// Apply the OffsetLimits
foreach (OffsetLimits limit in limits) {
limit.Apply(ik.solver.GetEffector(limit.effector), transform.rotation);
}
}
// Remove the delegate when destroyed
protected virtual void OnDestroy() {
if (ik != null) ik.solver.OnPreUpdate -= ModifyOffset;
}
}
}