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.

284 lines
11 KiB
C#

using UnityEngine;
using System.Collections;
namespace RootMotion.FinalIK {
/// <summary>
/// Handles smooth aim target switching, weight blending, target interpolation and root rotation.
/// </summary>
public class AimController : MonoBehaviour {
[Tooltip("Reference to the AimIK component.")]
/// <summary>
/// Reference to the AimIK component.
/// </summary>
public AimIK ik;
[Tooltip("Master weight of the IK solver.")]
/// <summary>
/// Master weight of the IK solver.
/// </summary>
[Range(0f, 1f)] public float weight = 1f;
[Header("Target Smoothing")]
[Tooltip("The target to aim at. Do not use the Target transform that is assigned to AimIK. Set to null if you wish to stop aiming.")]
/// <summary>
/// The target to aim at. Do not use the Target transform that is assigned to AimIK. Set to null if you wish to stop aiming.
/// </summary>
public Transform target;
[Tooltip("The time it takes to switch targets.")]
/// <summary>
/// The time it takes to switch targets.
/// </summary>
public float targetSwitchSmoothTime = 0.3f;
[Tooltip("The time it takes to blend in/out of AimIK weight.")]
/// <summary>
/// The time it takes to blend in/out of AimIK weight.
/// </summary>
public float weightSmoothTime = 0.3f;
[Header("Turning Towards The Target")]
[Tooltip("Enables smooth turning towards the target according to the parameters under this header.")]
/// <summary>
/// Enables smooth turning towards the target according to the parameters under this header.
/// </summary>
public bool smoothTurnTowardsTarget = true;
[Tooltip("Speed of turning towards the target using Vector3.RotateTowards.")]
/// <summary>
/// Speed of turning towards the target using Vector3.RotateTowards.
/// </summary>
public float maxRadiansDelta = 3f;
[Tooltip("Speed of moving towards the target using Vector3.RotateTowards.")]
/// <summary>
/// Speed of moving towards the target using Vector3.RotateTowards.
/// </summary>
public float maxMagnitudeDelta = 3f;
[Tooltip("Speed of slerping towards the target.")]
/// <summary>
/// Speed of slerping towards the target.
/// </summary>
public float slerpSpeed = 3f;
[Tooltip("Smoothing time for turning towards the yaw and pitch of the target using Mathf.SmoothDampAngle. Value of 0 means smooth damping is disabled." )]
/// <summary>
/// Smoothing time for turning towards the yaw and pitch of the target using Mathf.SmoothDampAngle. Value of 0 means smooth damping is disabled.
/// </summary>
public float smoothDampTime = 0f;
[Tooltip("The position of the pivot that the aim target is rotated around relative to the root of the character.")]
/// <summary>
/// The position of the pivot that the aim target is rotated around relative to the root of the character.
/// </summary>
public Vector3 pivotOffsetFromRoot = Vector3.up;
[Tooltip("Minimum distance of aiming from the first bone. Keeps the solver from failing if the target is too close.")]
/// <summary>
/// Minimum distance of aiming from the first bone. Keeps the solver from failing if the target is too close.
/// </summary>
public float minDistance = 1f;
[Tooltip("Offset applied to the target in world space. Convenient for scripting aiming inaccuracy.")]
/// <summary>
/// Offset applied to the target in world space. Convenient for scripting aiming inaccuracy.
/// </summary>
public Vector3 offset;
[Header("RootRotation")]
[Tooltip("Character root will be rotate around the Y axis to keep root forward within this angle from the aiming direction.")]
/// <summary>
///Character root will be rotate around the Y axis to keep root forward within this angle from the aiming direction.
/// </summary>
[Range(0f, 180f)] public float maxRootAngle = 45f;
[Tooltip("If enabled, aligns the root forward to target direction after 'Max Root Angle' has been exceeded.")]
/// <summary>
/// If enabled, aligns the root forward to target direction after 'Max Root Angle' has been exceeded.
/// </summary>
public bool turnToTarget;
[Tooltip("The time of turning towards the target direction if 'Max Root Angle has been exceeded and 'Turn To Target' is enabled.")]
/// <summary>
/// The time of turning towards the target direction if 'Max Root Angle has been exceeded and 'Turn To Target' is enabled.
/// </summary>
public float turnToTargetTime = 0.2f;
[Header("Mode")]
[Tooltip("If true, AimIK will consider whatever the current direction of the weapon to be the forward aiming direction and work additively on top of that. This enables you to use recoil and reloading animations seamlessly with AimIK. Adjust the Vector3 value below if the weapon is not aiming perfectly forward in the aiming animation clip.")]
/// <summary>
/// If true, AimIK will consider whatever the current direction of the weapon to be the forward aiming direction and work additively on top of that. This enables you to use recoil and reloading animations seamlessly with AimIK. Adjust the Vector3 value below if the weapon is not aiming perfectly forward in the aiming animation clip.
/// </summary>
public bool useAnimatedAimDirection;
[Tooltip("The direction of the animated weapon aiming in character space. Tweak this value to adjust the aiming. 'Use Animated Aim Direction' must be enabled for this property to work.")]
/// <summary>
/// The direction of the animated weapon aiming in character space. Tweak this value to adjust the aiming. 'Use Animated Aim Direction' must be enabled for this property to work.
/// </summary>
public Vector3 animatedAimDirection = Vector3.forward;
private Transform lastTarget;
private float switchWeight, switchWeightV;
private float weightV;
private Vector3 lastPosition;
private Vector3 dir;
private bool lastSmoothTowardsTarget;
private bool turningToTarget;
private float turnToTargetMlp = 1f;
private float turnToTargetMlpV;
void Start() {
lastPosition = ik.solver.IKPosition;
dir = ik.solver.IKPosition - pivot;
ik.solver.target = null;
}
void LateUpdate () {
// If target has changed...
if (target != lastTarget) {
if (lastTarget == null && target != null && ik.solver.IKPositionWeight <= 0f)
{
lastPosition = target.position;
dir = target.position - pivot;
ik.solver.IKPosition = target.position + offset;
}
else
{
lastPosition = ik.solver.IKPosition;
dir = ik.solver.IKPosition - pivot;
}
switchWeight = 0f;
lastTarget = target;
}
// Smooth weight
float targetWeight = target != null ? weight : 0f;
ik.solver.IKPositionWeight = Mathf.SmoothDamp(ik.solver.IKPositionWeight, targetWeight, ref weightV, weightSmoothTime);
if (ik.solver.IKPositionWeight >= 0.999f && targetWeight > ik.solver.IKPositionWeight) ik.solver.IKPositionWeight = 1f;
if (ik.solver.IKPositionWeight <= 0.001f && targetWeight < ik.solver.IKPositionWeight) ik.solver.IKPositionWeight = 0f;
if (ik.solver.IKPositionWeight <= 0f) return;
// Smooth target switching
switchWeight = Mathf.SmoothDamp(switchWeight, 1f, ref switchWeightV, targetSwitchSmoothTime);
if (switchWeight >= 0.999f) switchWeight = 1f;
if (target != null) {
ik.solver.IKPosition = Vector3.Lerp(lastPosition, target.position + offset, switchWeight);
}
// Smooth turn towards target
if (smoothTurnTowardsTarget != lastSmoothTowardsTarget) {
dir = ik.solver.IKPosition - pivot;
lastSmoothTowardsTarget = smoothTurnTowardsTarget;
}
if (smoothTurnTowardsTarget) {
Vector3 targetDir = ik.solver.IKPosition - pivot;
// Slerp
if (slerpSpeed > 0f) dir = Vector3.Slerp(dir, targetDir, Time.deltaTime * slerpSpeed);
// RotateTowards
if (maxRadiansDelta > 0 || maxMagnitudeDelta > 0f) dir = Vector3.RotateTowards(dir, targetDir, Time.deltaTime * maxRadiansDelta, maxMagnitudeDelta);
// SmoothDamp
if (smoothDampTime > 0f)
{
float yaw = V3Tools.GetYaw(dir);
float targetYaw = V3Tools.GetYaw(targetDir);
float y = Mathf.SmoothDampAngle(yaw, targetYaw, ref yawV, smoothDampTime);
float pitch = V3Tools.GetPitch(dir);
float targetPitch = V3Tools.GetPitch(targetDir);
float p = Mathf.SmoothDampAngle(pitch, targetPitch, ref pitchV, smoothDampTime);
float dirMag = Mathf.SmoothDamp(dir.magnitude, targetDir.magnitude, ref dirMagV, smoothDampTime);
dir = Quaternion.Euler(p, y, 0f) * Vector3.forward * dirMag;
}
ik.solver.IKPosition = pivot + dir;
}
// Min distance from the pivot
ApplyMinDistance();
// Root rotation
RootRotation();
// Offset mode
if (useAnimatedAimDirection) {
ik.solver.axis = ik.solver.transform.InverseTransformVector(ik.transform.rotation * animatedAimDirection);
}
}
private float yawV, pitchV, dirMagV;
// Pivot of rotating the aiming direction.
private Vector3 pivot {
get {
return ik.transform.position + ik.transform.rotation * pivotOffsetFromRoot;
}
}
// Make sure aiming target is not too close (might make the solver instable when the target is closer to the first bone than the last bone is).
void ApplyMinDistance() {
Vector3 aimFrom = pivot;
Vector3 direction = (ik.solver.IKPosition - aimFrom);
direction = direction.normalized * Mathf.Max(direction.magnitude, minDistance);
ik.solver.IKPosition = aimFrom + direction;
}
// Character root will be rotate around the Y axis to keep root forward within this angle from the aiming direction.
private void RootRotation() {
float max = Mathf.Lerp(180f, maxRootAngle * turnToTargetMlp, ik.solver.IKPositionWeight);
if (max < 180f) {
Vector3 faceDirLocal = Quaternion.Inverse(ik.transform.rotation) * (ik.solver.IKPosition - pivot);
float angle = Mathf.Atan2(faceDirLocal.x, faceDirLocal.z) * Mathf.Rad2Deg;
float rotation = 0f;
if (angle > max) {
rotation = angle - max;
if (!turningToTarget && turnToTarget) StartCoroutine(TurnToTarget());
}
if (angle < -max) {
rotation = angle + max;
if (!turningToTarget && turnToTarget) StartCoroutine(TurnToTarget());
}
ik.transform.rotation = Quaternion.AngleAxis(rotation, ik.transform.up) * ik.transform.rotation;
}
}
// Aligns the root forward to target direction after "Max Root Angle" has been exceeded.
private IEnumerator TurnToTarget()
{
turningToTarget = true;
while (turnToTargetMlp > 0f)
{
turnToTargetMlp = Mathf.SmoothDamp(turnToTargetMlp, 0f, ref turnToTargetMlpV, turnToTargetTime);
if (turnToTargetMlp < 0.01f) turnToTargetMlp = 0f;
yield return null;
}
turnToTargetMlp = 1f;
turningToTarget = false;
}
}
}