#if !UNITY_2019_3_OR_NEWER #define CINEMACHINE_PHYSICS #endif using UnityEngine; using Cinemachine.Utility; namespace Cinemachine { /// /// Third-person follower, with complex pivoting: horizontal about the origin, /// vertical about the shoulder. /// [AddComponentMenu("")] // Don't display in add component menu [SaveDuringPlay] public class Cinemachine3rdPersonFollow : CinemachineComponentBase { /// How responsively the camera tracks the target. Each axis (camera-local) /// can have its own setting. Value is the approximate time it takes the camera /// to catch up to the target's new position. Smaller values give a more rigid /// effect, larger values give a squishier one. [Tooltip("How responsively the camera tracks the target. Each axis (camera-local) " + "can have its own setting. Value is the approximate time it takes the camera " + "to catch up to the target's new position. Smaller values give a more " + "rigid effect, larger values give a squishier one")] public Vector3 Damping; /// Position of the shoulder pivot relative to the Follow target origin. /// This offset is in target-local space. [Header("Rig")] [Tooltip("Position of the shoulder pivot relative to the Follow target origin. " + "This offset is in target-local space")] public Vector3 ShoulderOffset; /// Vertical offset of the hand in relation to the shoulder. /// Arm length will affect the follow target's screen position /// when the camera rotates vertically. [Tooltip("Vertical offset of the hand in relation to the shoulder. " + "Arm length will affect the follow target's screen position when " + "the camera rotates vertically")] public float VerticalArmLength; /// Specifies which shoulder (left, right, or in-between) the camera is on. [Tooltip("Specifies which shoulder (left, right, or in-between) the camera is on")] [Range(0, 1)] public float CameraSide; /// How far behind the hand the camera will be placed. [Tooltip("How far behind the hand the camera will be placed")] public float CameraDistance; #if CINEMACHINE_PHYSICS /// Camera will avoid obstacles on these layers. [Header("Obstacles")] [Tooltip("Camera will avoid obstacles on these layers")] public LayerMask CameraCollisionFilter; /// /// Obstacles with this tag will be ignored. It is a good idea /// to set this field to the target's tag /// [TagField] [Tooltip("Obstacles with this tag will be ignored. " + "It is a good idea to set this field to the target's tag")] public string IgnoreTag = string.Empty; /// /// Specifies how close the camera can get to obstacles /// [Tooltip("Specifies how close the camera can get to obstacles")] [Range(0, 1)] public float CameraRadius; /// /// How gradually the camera moves to correct for occlusions. /// Higher numbers will move the camera more gradually. /// [Range(0, 10)] [Tooltip("How gradually the camera moves to correct for occlusions. " + "Higher numbers will move the camera more gradually.")] public float DampingIntoCollision; /// /// How gradually the camera returns to its normal position after having been corrected by the built-in /// collision resolution system. Higher numbers will move the camera more gradually back to normal. /// [Range(0, 10)] [Tooltip("How gradually the camera returns to its normal position after having been corrected by the built-in " + "collision resolution system. Higher numbers will move the camera more gradually back to normal.")] public float DampingFromCollision; #endif // State info Vector3 m_PreviousFollowTargetPosition; Vector3 m_DampingCorrection; // this is in local rig space #if CINEMACHINE_PHYSICS float m_CamPosCollisionCorrection; #endif void OnValidate() { CameraSide = Mathf.Clamp(CameraSide, -1.0f, 1.0f); Damping.x = Mathf.Max(0, Damping.x); Damping.y = Mathf.Max(0, Damping.y); Damping.z = Mathf.Max(0, Damping.z); #if CINEMACHINE_PHYSICS CameraRadius = Mathf.Max(0.001f, CameraRadius); DampingIntoCollision = Mathf.Max(0, DampingIntoCollision); DampingFromCollision = Mathf.Max(0, DampingFromCollision); #endif } void Reset() { ShoulderOffset = new Vector3(0.5f, -0.4f, 0.0f); VerticalArmLength = 0.4f; CameraSide = 1.0f; CameraDistance = 2.0f; Damping = new Vector3(0.1f, 0.5f, 0.3f); #if CINEMACHINE_PHYSICS CameraCollisionFilter = 0; CameraRadius = 0.2f; DampingIntoCollision = 0; DampingFromCollision = 2f; #endif } #if CINEMACHINE_PHYSICS void OnDestroy() { RuntimeUtility.DestroyScratchCollider(); } #endif /// True if component is enabled and has a Follow target defined public override bool IsValid => enabled && FollowTarget != null; /// Get the Cinemachine Pipeline stage that this component implements. /// Always returns the Aim stage public override CinemachineCore.Stage Stage { get { return CinemachineCore.Stage.Body; } } #if CINEMACHINE_PHYSICS /// /// Report maximum damping time needed for this component. /// /// Highest damping setting in this component public override float GetMaxDampTime() { return Mathf.Max( Mathf.Max(DampingIntoCollision, DampingFromCollision), Mathf.Max(Damping.x, Mathf.Max(Damping.y, Damping.z))); } #endif /// Orients the camera to match the Follow target's orientation /// The current camera state /// Elapsed time since last frame, for damping calculations. /// If negative, previous state is reset. public override void MutateCameraState(ref CameraState curState, float deltaTime) { if (IsValid) { if (!VirtualCamera.PreviousStateIsValid) deltaTime = -1; PositionCamera(ref curState, deltaTime); } } /// This is called to notify the us that a target got warped, /// so that we can update its internal state to make the camera /// also warp seamlessy. /// The object that was warped /// The amount the target's position changed public override void OnTargetObjectWarped(Transform target, Vector3 positionDelta) { base.OnTargetObjectWarped(target, positionDelta); if (target == FollowTarget) { m_PreviousFollowTargetPosition += positionDelta; } } void PositionCamera(ref CameraState curState, float deltaTime) { var up = curState.ReferenceUp; var targetPos = FollowTargetPosition; var targetRot = FollowTargetRotation; var targetForward = targetRot * Vector3.forward; var heading = GetHeading(targetRot, up); if (deltaTime < 0) { // No damping - reset damping state info m_DampingCorrection = Vector3.zero; #if CINEMACHINE_PHYSICS m_CamPosCollisionCorrection = 0; #endif } else { // Damping correction is applied to the shoulder offset - stretching the rig m_DampingCorrection += Quaternion.Inverse(heading) * (m_PreviousFollowTargetPosition - targetPos); m_DampingCorrection -= VirtualCamera.DetachedFollowTargetDamp(m_DampingCorrection, Damping, deltaTime); } m_PreviousFollowTargetPosition = targetPos; var root = targetPos; GetRawRigPositions(root, targetRot, heading, out _, out Vector3 hand); // Place the camera at the correct distance from the hand var camPos = hand - (targetForward * (CameraDistance - m_DampingCorrection.z)); #if CINEMACHINE_PHYSICS // Check if hand is colliding with something, if yes, then move the hand // closer to the player. The radius is slightly enlarged, to avoid problems // next to walls float dummy = 0; var collidedHand = ResolveCollisions(root, hand, -1, CameraRadius * 1.05f, ref dummy); camPos = ResolveCollisions( collidedHand, camPos, deltaTime, CameraRadius, ref m_CamPosCollisionCorrection); #endif // Set state curState.RawPosition = camPos; curState.RawOrientation = targetRot; // not necessary, but left in to avoid breaking scenes that depend on this } /// /// Internal use only. Public for the inspector gizmo /// /// Root of the rig. /// Shoulder of the rig. /// Hand of the rig. public void GetRigPositions(out Vector3 root, out Vector3 shoulder, out Vector3 hand) { var up = VirtualCamera.State.ReferenceUp; var targetRot = FollowTargetRotation; var heading = GetHeading(targetRot, up); root = m_PreviousFollowTargetPosition; GetRawRigPositions(root, targetRot, heading, out shoulder, out hand); #if CINEMACHINE_PHYSICS float dummy = 0; hand = ResolveCollisions(root, hand, -1, CameraRadius * 1.05f, ref dummy); #endif } internal static Quaternion GetHeading(Quaternion targetRot, Vector3 up) { var targetForward = targetRot * Vector3.forward; var planeForward = Vector3.Cross(up, Vector3.Cross(targetForward.ProjectOntoPlane(up), up)); if (planeForward.AlmostZero()) planeForward = Vector3.Cross(targetRot * Vector3.right, up); return Quaternion.LookRotation(planeForward, up); } void GetRawRigPositions( Vector3 root, Quaternion targetRot, Quaternion heading, out Vector3 shoulder, out Vector3 hand) { var shoulderOffset = ShoulderOffset; shoulderOffset.x = Mathf.Lerp(-shoulderOffset.x, shoulderOffset.x, CameraSide); shoulderOffset.x += m_DampingCorrection.x; shoulderOffset.y += m_DampingCorrection.y; shoulder = root + heading * shoulderOffset; hand = shoulder + targetRot * new Vector3(0, VerticalArmLength, 0); } #if CINEMACHINE_PHYSICS Vector3 ResolveCollisions( Vector3 root, Vector3 tip, float deltaTime, float cameraRadius, ref float collisionCorrection) { if (CameraCollisionFilter.value == 0) return tip; var dir = tip - root; var len = dir.magnitude; if (len < Epsilon) return tip; dir /= len; var result = tip; float desiredCorrection = 0; if (RuntimeUtility.SphereCastIgnoreTag( root, cameraRadius, dir, out RaycastHit hitInfo, len, CameraCollisionFilter, IgnoreTag)) { var desiredResult = hitInfo.point + hitInfo.normal * cameraRadius; desiredCorrection = (desiredResult - tip).magnitude; } collisionCorrection += deltaTime < 0 ? desiredCorrection - collisionCorrection : Damper.Damp( desiredCorrection - collisionCorrection, desiredCorrection > collisionCorrection ? DampingIntoCollision : DampingFromCollision, deltaTime); // Apply the correction if (collisionCorrection > Epsilon) result -= dir * collisionCorrection; return result; } #endif } }