using UnityEngine; using Cinemachine.Utility; using System.Collections.Generic; namespace Cinemachine { /// /// The output of the Cinemachine engine for a specific virtual camera. The information /// in this struct can be blended, and provides what is needed to calculate an /// appropriate camera position, orientation, and lens setting. /// /// Raw values are what the Cinemachine behaviours generate. The correction channel /// holds perturbations to the raw values - e.g. noise or smoothing, or obstacle /// avoidance corrections. Coirrections are not considered when making time-based /// calculations such as damping. /// /// The Final position and orientation is the comination of the raw values and /// their corrections. /// public struct CameraState { /// /// Camera Lens Settings. /// public LensSettings Lens; /// /// Which way is up. World space unit vector. Must have a length of 1. /// public Vector3 ReferenceUp; /// /// The world space focus point of the camera. What the camera wants to look at. /// There is a special constant define to represent "nothing". Be careful to /// check for that (or check the HasLookAt property). /// public Vector3 ReferenceLookAt; /// /// Returns true if this state has a valid ReferenceLookAt value. /// #pragma warning disable 1718 // comparison made to same variable public bool HasLookAt => ReferenceLookAt == ReferenceLookAt; // will be false if NaN /// /// This constant represents "no point in space" or "no direction". /// public static Vector3 kNoPoint = new Vector3(float.NaN, float.NaN, float.NaN); /// /// Raw (un-corrected) world space position of this camera /// public Vector3 RawPosition; /// /// Raw (un-corrected) world space orientation of this camera /// public Quaternion RawOrientation; /// This is a way for the Body component to bypass aim damping, /// useful for when the body needs to rotate its point of view, but does not /// want interference from the aim damping. The value is the camera /// rotation, in Euler degrees. public Vector3 PositionDampingBypass; /// /// Subjective estimation of how "good" the shot is. /// Larger values mean better quality. Default is 1. /// public float ShotQuality; /// /// Position correction. This will be added to the raw position. /// This value doesn't get fed back into the system when calculating the next frame. /// Can be noise, or smoothing, or both, or something else. /// public Vector3 PositionCorrection; /// /// Orientation correction. This will be added to the raw orientation. /// This value doesn't get fed back into the system when calculating the next frame. /// Can be noise, or smoothing, or both, or something else. /// public Quaternion OrientationCorrection; /// /// Position with correction applied. /// public Vector3 CorrectedPosition => RawPosition + PositionCorrection; /// /// Orientation with correction applied. /// public Quaternion CorrectedOrientation => RawOrientation * OrientationCorrection; /// /// Position with correction applied. This is what the final camera gets. /// public Vector3 FinalPosition => RawPosition + PositionCorrection; /// /// Orientation with correction and dutch applied. This is what the final camera gets. /// public Quaternion FinalOrientation { get { if (Mathf.Abs(Lens.Dutch) > UnityVectorExtensions.Epsilon) return CorrectedOrientation * Quaternion.AngleAxis(Lens.Dutch, Vector3.forward); return CorrectedOrientation; } } /// /// These hints can be or'ed toether to influence how blending is done, and how state /// is applied to the camera /// public enum BlendHintValue { /// Normal state blending Nothing = 0, /// This state does not affect the camera position NoPosition = 1, /// This state does not affect the camera rotation NoOrientation = 2, /// Combination of NoPosition and NoOrientation NoTransform = NoPosition | NoOrientation, /// Spherical blend about the LookAt target (if any) SphericalPositionBlend = 4, /// Cylindrical blend about the LookAt target (if any) CylindricalPositionBlend = 8, /// Radial blend when the LookAt target changes(if any) RadialAimBlend = 16, /// Ignore the LookAt target and just slerp the orientation IgnoreLookAtTarget = 32, /// This state does not affect the lens NoLens = 64, } /// /// These hints can be or'ed toether to influence how blending is done, and how state /// is applied to the camera /// public BlendHintValue BlendHint; /// /// State with default values /// public static CameraState Default { get { CameraState state = new CameraState(); state.Lens = LensSettings.Default; state.ReferenceUp = Vector3.up; state.ReferenceLookAt = kNoPoint; state.RawPosition = Vector3.zero; state.RawOrientation = Quaternion.identity; state.ShotQuality = 1; state.PositionCorrection = Vector3.zero; state.OrientationCorrection = Quaternion.identity; state.PositionDampingBypass = Vector3.zero; state.BlendHint = BlendHintValue.Nothing; return state; } } /// Opaque structure represent extra blendable stuff and its weight. /// The base system ignores this data - it is intended for extension modules public struct CustomBlendable { /// The custom stuff that the extension module will consider public Object m_Custom; /// The weight of the custom stuff. Must be 0...1 public float m_Weight; /// Constructor with specific values /// The custom stuff that the extension module will consider /// The weight of the custom stuff. Must be 0...1 public CustomBlendable(Object custom, float weight) { m_Custom = custom; m_Weight = weight; } }; // This is to avoid excessive GC allocs CustomBlendable mCustom0; CustomBlendable mCustom1; CustomBlendable mCustom2; CustomBlendable mCustom3; List m_CustomOverflow; /// The number of custom blendables that will be applied to the camera. /// The base system manages but otherwise ignores this data - it is intended for /// extension modules public int NumCustomBlendables { get; private set; } /// Get a custom blendable that will be applied to the camera. /// The base system manages but otherwise ignores this data - it is intended for /// extension modules /// Which one to get. Must be in range [0...NumCustomBlendables) /// The custom blendable at the specified index. public CustomBlendable GetCustomBlendable(int index) { switch (index) { case 0: return mCustom0; case 1: return mCustom1; case 2: return mCustom2; case 3: return mCustom3; default: { index -= 4; if (m_CustomOverflow != null && index < m_CustomOverflow.Count) return m_CustomOverflow[index]; return new CustomBlendable(null, 0); } } } int FindCustomBlendable(Object custom) { if (mCustom0.m_Custom == custom) return 0; if (mCustom1.m_Custom == custom) return 1; if (mCustom2.m_Custom == custom) return 2; if (mCustom3.m_Custom == custom) return 3; if (m_CustomOverflow != null) { for (int i = 0; i < m_CustomOverflow.Count; ++i) if (m_CustomOverflow[i].m_Custom == custom) return i + 4; } return -1; } /// Add a custom blendable to the pot for eventual application to the camera. /// The base system manages but otherwise ignores this data - it is intended for /// extension modules /// The custom blendable to add. If b.m_Custom is the same as an /// already-added custom blendable, then they will be merged and the weights combined. public void AddCustomBlendable(CustomBlendable b) { // Attempt to merge common blendables to avoid growth int index = FindCustomBlendable(b.m_Custom); if (index >= 0) b.m_Weight += GetCustomBlendable(index).m_Weight; else index = NumCustomBlendables++; switch (index) { case 0: mCustom0 = b; break; case 1: mCustom1 = b; break; case 2: mCustom2 = b; break; case 3: mCustom3 = b; break; default: { index -= 4; if (m_CustomOverflow == null) m_CustomOverflow = new List(); if (index < m_CustomOverflow.Count) m_CustomOverflow[index] = b; else m_CustomOverflow.Add(b); break; } } } /// Intelligently blend the contents of two states. /// The first state, corresponding to t=0 /// The second state, corresponding to t=1 /// How much to interpolate. Internally clamped to 0..1 /// Linearly interpolated CameraState public static CameraState Lerp(CameraState stateA, CameraState stateB, float t) { t = Mathf.Clamp01(t); float adjustedT = t; CameraState state = new CameraState(); // Combine the blend hints intelligently if (((stateA.BlendHint & stateB.BlendHint) & BlendHintValue.NoPosition) != 0) state.BlendHint |= BlendHintValue.NoPosition; if (((stateA.BlendHint & stateB.BlendHint) & BlendHintValue.NoOrientation) != 0) state.BlendHint |= BlendHintValue.NoOrientation; if (((stateA.BlendHint & stateB.BlendHint) & BlendHintValue.NoLens) != 0) state.BlendHint |= BlendHintValue.NoLens; if (((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.SphericalPositionBlend) != 0) state.BlendHint |= BlendHintValue.SphericalPositionBlend; if (((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.CylindricalPositionBlend) != 0) state.BlendHint |= BlendHintValue.CylindricalPositionBlend; if (((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.NoLens) == 0) state.Lens = LensSettings.Lerp(stateA.Lens, stateB.Lens, t); else if (((stateA.BlendHint & stateB.BlendHint) & BlendHintValue.NoLens) == 0) { if ((stateA.BlendHint & BlendHintValue.NoLens) != 0) state.Lens = stateB.Lens; else state.Lens = stateA.Lens; } state.ReferenceUp = Vector3.Slerp(stateA.ReferenceUp, stateB.ReferenceUp, t); state.ShotQuality = Mathf.Lerp(stateA.ShotQuality, stateB.ShotQuality, t); state.PositionCorrection = ApplyPosBlendHint( stateA.PositionCorrection, stateA.BlendHint, stateB.PositionCorrection, stateB.BlendHint, state.PositionCorrection, Vector3.Lerp(stateA.PositionCorrection, stateB.PositionCorrection, t)); state.OrientationCorrection = ApplyRotBlendHint( stateA.OrientationCorrection, stateA.BlendHint, stateB.OrientationCorrection, stateB.BlendHint, state.OrientationCorrection, Quaternion.Slerp(stateA.OrientationCorrection, stateB.OrientationCorrection, t)); // LookAt target if (!stateA.HasLookAt || !stateB.HasLookAt) state.ReferenceLookAt = kNoPoint; else { // Re-interpolate FOV to preserve target composition, if possible float fovA = stateA.Lens.FieldOfView; float fovB = stateB.Lens.FieldOfView; if (((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.NoLens) == 0 && !state.Lens.Orthographic && !Mathf.Approximately(fovA, fovB)) { LensSettings lens = state.Lens; lens.FieldOfView = InterpolateFOV( fovA, fovB, Mathf.Max((stateA.ReferenceLookAt - stateA.CorrectedPosition).magnitude, stateA.Lens.NearClipPlane), Mathf.Max((stateB.ReferenceLookAt - stateB.CorrectedPosition).magnitude, stateB.Lens.NearClipPlane), t); state.Lens = lens; // Make sure we preserve the screen composition through FOV changes adjustedT = Mathf.Abs((lens.FieldOfView - fovA) / (fovB - fovA)); } // Linear interpolation of lookAt target point state.ReferenceLookAt = Vector3.Lerp( stateA.ReferenceLookAt, stateB.ReferenceLookAt, adjustedT); } // Raw position state.RawPosition = ApplyPosBlendHint( stateA.RawPosition, stateA.BlendHint, stateB.RawPosition, stateB.BlendHint, state.RawPosition, state.InterpolatePosition( stateA.RawPosition, stateA.ReferenceLookAt, stateB.RawPosition, stateB.ReferenceLookAt, t)); // Interpolate the LookAt in Screen Space if requested if (state.HasLookAt && ((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.RadialAimBlend) != 0) { state.ReferenceLookAt = state.RawPosition + Vector3.Slerp( stateA.ReferenceLookAt - state.RawPosition, stateB.ReferenceLookAt - state.RawPosition, adjustedT); } // Clever orientation interpolation Quaternion newOrient = state.RawOrientation; if (((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.NoOrientation) == 0) { Vector3 dirTarget = Vector3.zero; if (state.HasLookAt)//&& ((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.RadialAimBlend) == 0) { // If orientations are different, use LookAt to blend them float angle = Quaternion.Angle(stateA.RawOrientation, stateB.RawOrientation); if (angle > UnityVectorExtensions.Epsilon) dirTarget = state.ReferenceLookAt - state.CorrectedPosition; } if (dirTarget.AlmostZero() || ((stateA.BlendHint | stateB.BlendHint) & BlendHintValue.IgnoreLookAtTarget) != 0) { // Don't know what we're looking at - can only slerp newOrient = Quaternion.Slerp(stateA.RawOrientation, stateB.RawOrientation, t); } else { // Rotate while preserving our lookAt target var up = state.ReferenceUp; dirTarget.Normalize(); if (Vector3.Cross(dirTarget, up).AlmostZero()) { // Looking up or down at the pole newOrient = Quaternion.Slerp(stateA.RawOrientation, stateB.RawOrientation, t); up = newOrient * Vector3.up; } // Blend the desired offsets from center newOrient = Quaternion.LookRotation(dirTarget, up); var deltaA = -stateA.RawOrientation.GetCameraRotationToTarget( stateA.ReferenceLookAt - stateA.CorrectedPosition, up); var deltaB = -stateB.RawOrientation.GetCameraRotationToTarget( stateB.ReferenceLookAt - stateB.CorrectedPosition, up); newOrient = newOrient.ApplyCameraRotation(Vector2.Lerp(deltaA, deltaB, adjustedT), up); } } state.RawOrientation = ApplyRotBlendHint( stateA.RawOrientation, stateA.BlendHint, stateB.RawOrientation, stateB.BlendHint, state.RawOrientation, newOrient); // Accumulate the custom blendables and apply the weights for (int i = 0; i < stateA.NumCustomBlendables; ++i) { CustomBlendable b = stateA.GetCustomBlendable(i); b.m_Weight *= (1-t); if (b.m_Weight > 0) state.AddCustomBlendable(b); } for (int i = 0; i < stateB.NumCustomBlendables; ++i) { CustomBlendable b = stateB.GetCustomBlendable(i); b.m_Weight *= t; if (b.m_Weight > 0) state.AddCustomBlendable(b); } return state; } static float InterpolateFOV(float fovA, float fovB, float dA, float dB, float t) { // We interpolate shot height float hA = dA * 2f * Mathf.Tan(fovA * Mathf.Deg2Rad / 2f); float hB = dB * 2f * Mathf.Tan(fovB * Mathf.Deg2Rad / 2f); float h = Mathf.Lerp(hA, hB, t); float fov = 179f; float d = Mathf.Lerp(dA, dB, t); if (d > UnityVectorExtensions.Epsilon) fov = 2f * Mathf.Atan(h / (2 * d)) * Mathf.Rad2Deg; return Mathf.Clamp(fov, Mathf.Min(fovA, fovB), Mathf.Max(fovA, fovB)); } static Vector3 ApplyPosBlendHint( Vector3 posA, BlendHintValue hintA, Vector3 posB, BlendHintValue hintB, Vector3 original, Vector3 blended) { if (((hintA | hintB) & BlendHintValue.NoPosition) == 0) return blended; if (((hintA & hintB) & BlendHintValue.NoPosition) != 0) return original; if ((hintA & BlendHintValue.NoPosition) != 0) return posB; return posA; } static Quaternion ApplyRotBlendHint( Quaternion rotA, BlendHintValue hintA, Quaternion rotB, BlendHintValue hintB, Quaternion original, Quaternion blended) { if (((hintA | hintB) & BlendHintValue.NoOrientation) == 0) return blended; if (((hintA & hintB) & BlendHintValue.NoOrientation) != 0) return original; if ((hintA & BlendHintValue.NoOrientation) != 0) return rotB; return rotA; } Vector3 InterpolatePosition( Vector3 posA, Vector3 pivotA, Vector3 posB, Vector3 pivotB, float t) { if (pivotA == pivotA && pivotB == pivotB) // check for NaN { if ((BlendHint & BlendHintValue.CylindricalPositionBlend) != 0) { // Cylindrical interpolation about pivot var a = Vector3.ProjectOnPlane(posA - pivotA, ReferenceUp); var b = Vector3.ProjectOnPlane(posB - pivotB, ReferenceUp); var c = Vector3.Slerp(a, b, t); posA = (posA - a) + c; posB = (posB - b) + c; } else if ((BlendHint & BlendHintValue.SphericalPositionBlend) != 0) { // Spherical interpolation about pivot var c = Vector3.Slerp(posA - pivotA, posB - pivotB, t); posA = pivotA + c; posB = pivotB + c; } } return Vector3.Lerp(posA, posB, t); } } }