#if CINEMACHINE_EXPERIMENTAL_VCAM
using UnityEngine;
using Cinemachine.Utility;
using System;
namespace Cinemachine
{
///
///
/// NOTE: THIS CLASS IS EXPERIMENTAL, AND NOT FOR PUBLIC USE
///
/// Lighter-weight version of the CinemachineFreeLook, with extra radial axis.
///
/// A Cinemachine Camera geared towards a 3rd person camera experience.
/// The camera orbits around its subject with three separate camera rigs defining
/// rings around the target. Each rig has its own radius, height offset, composer,
/// and lens settings.
/// Depending on the camera's position along the spline connecting these three rigs,
/// these settings are interpolated to give the final camera position and state.
///
[DocumentationSorting(DocumentationSortingAttribute.Level.UserRef)]
[DisallowMultipleComponent]
[ExecuteAlways]
[AddComponentMenu("Cinemachine/CinemachineNewFreeLook")]
[SaveDuringPlay]
public class CinemachineNewFreeLook : CinemachineNewVirtualCamera
{
/// The Vertical axis. Value is 0..1. Chooses how to blend the child rigs
[Tooltip("The Vertical axis. Value is 0..1. 0.5 is the middle rig. Chooses how to blend the child rigs")]
[AxisStateProperty]
public AxisState m_VerticalAxis = new AxisState(0, 1, false, true, 2f, 0.2f, 0.1f, "Mouse Y", false);
[Tooltip("The Radial axis. Value is the base radius of the orbits")]
[AxisStateProperty]
public AxisState m_RadialAxis = new AxisState(1, 1, false, false, 100, 0f, 0f, "Mouse ScrollWheel", false);
/// Defines the height and radius for an orbit
[Serializable]
public struct Orbit
{
/// Height relative to target
public float m_Height;
/// Radius of orbit
public float m_Radius;
}
/// Order is Top, Middle, Bottom
public Orbit[] m_Orbits = new Orbit[3];
///
[Tooltip("Controls how taut is the line that connects the rigs' orbits, which determines final placement on the Y axis")]
[Range(0f, 1f)]
public float m_SplineCurvature;
/// Identifiers for accessing override settings for top and bottom rigs
public enum RigID { Top, Bottom };
/// Override settings for top and bottom rigs
[Serializable]
public class Rig : ISerializationCallbackReceiver
{
public bool m_CustomLens;
public LensSettings m_Lens;
public bool m_CustomBody;
public TransposerSettings m_Body;
public bool m_CustomAim;
public ComposerSettings m_Aim;
public bool m_CustomNoise;
public PerlinNoiseSettings m_Noise;
public void Validate()
{
if (m_Lens.FieldOfView == 0)
m_Lens = LensSettings.Default;
m_Lens.Validate();
}
/// Blendable settings for Transposer Transposer
[Serializable] public class TransposerSettings
{
[Range(0f, 20f)] public float m_XDamping;
[Range(0f, 20f)] public float m_YDamping;
[Range(0f, 20f)] public float m_ZDamping;
[Range(0f, 20f)] public float m_PitchDamping;
[Range(0f, 20f)] public float m_YawDamping;
[Range(0f, 20f)] public float m_RollDamping;
internal void Lerp(CinemachineTransposer o, float t)
{
o.m_XDamping = Mathf.Lerp(o.m_XDamping, m_XDamping, t);
o.m_YDamping = Mathf.Lerp(o.m_YDamping, m_YDamping, t);
o.m_ZDamping = Mathf.Lerp(o.m_ZDamping, m_ZDamping, t);
o.m_PitchDamping = Mathf.Lerp(o.m_PitchDamping, m_PitchDamping, t);
o.m_YawDamping = Mathf.Lerp(o.m_YawDamping, m_YawDamping, t);
o.m_RollDamping = Mathf.Lerp(o.m_RollDamping, m_RollDamping, t);
}
internal void PullFrom(CinemachineTransposer o)
{
m_XDamping = o.m_XDamping;
m_YDamping = o.m_YDamping;
m_ZDamping = o.m_ZDamping;
m_PitchDamping = o.m_PitchDamping;
m_YawDamping = o.m_YawDamping;
m_RollDamping = o.m_RollDamping;
}
internal void PushTo(CinemachineTransposer o)
{
o.m_XDamping = m_XDamping;
o.m_YDamping = m_YDamping;
o.m_ZDamping = m_ZDamping;
o.m_PitchDamping =m_PitchDamping;
o.m_YawDamping = m_YawDamping;
o.m_RollDamping = m_RollDamping;
}
}
/// Blendable settings for Composer
[Serializable] public class ComposerSettings
{
public Vector3 m_LookAtOffset;
[Space]
[Range(0f, 20)] public float m_HorizontalDamping;
[Range(0f, 20)] public float m_VerticalDamping;
[Space]
[Range(0f, 1f)] public float m_ScreenX;
[Range(0f, 1f)] public float m_ScreenY;
[Range(0f, 1f)] public float m_DeadZoneWidth;
[Range(0f, 1f)] public float m_DeadZoneHeight;
[Range(0f, 2f)] public float m_SoftZoneWidth;
[Range(0f, 2f)] public float m_SoftZoneHeight;
[Range(-0.5f, 0.5f)] public float m_BiasX;
[Range(-0.5f, 0.5f)] public float m_BiasY;
internal void Lerp(CinemachineComposer c, float t)
{
c.m_TrackedObjectOffset = Vector3.Lerp(c.m_TrackedObjectOffset, m_LookAtOffset, t);
c.m_HorizontalDamping = Mathf.Lerp(c.m_HorizontalDamping, m_HorizontalDamping, t);
c.m_VerticalDamping = Mathf.Lerp(c.m_VerticalDamping, m_VerticalDamping, t);
c.m_ScreenX = Mathf.Lerp(c.m_ScreenX, m_ScreenX, t);
c.m_ScreenY = Mathf.Lerp(c.m_ScreenY, m_ScreenY, t);
c.m_DeadZoneWidth = Mathf.Lerp(c.m_DeadZoneWidth, m_DeadZoneWidth, t);
c.m_DeadZoneHeight = Mathf.Lerp(c.m_DeadZoneHeight, m_DeadZoneHeight, t);
c.m_SoftZoneWidth = Mathf.Lerp(c.m_SoftZoneWidth, m_SoftZoneWidth, t);
c.m_SoftZoneHeight = Mathf.Lerp(c.m_SoftZoneHeight, m_SoftZoneHeight, t);
c.m_BiasX = Mathf.Lerp(c.m_BiasX, m_BiasX, t);
c.m_BiasY = Mathf.Lerp(c.m_BiasY, m_BiasY, t);
}
internal void PullFrom(CinemachineComposer c)
{
m_LookAtOffset = c.m_TrackedObjectOffset;
m_HorizontalDamping = c.m_HorizontalDamping;
m_VerticalDamping = c.m_VerticalDamping;
m_ScreenX = c.m_ScreenX;
m_ScreenY = c.m_ScreenY;
m_DeadZoneWidth = c.m_DeadZoneWidth;
m_DeadZoneHeight = c.m_DeadZoneHeight;
m_SoftZoneWidth = c.m_SoftZoneWidth;
m_SoftZoneHeight = c.m_SoftZoneHeight;
m_BiasX = c.m_BiasX;
m_BiasY = c.m_BiasY;
}
internal void PushTo(CinemachineComposer c)
{
c.m_TrackedObjectOffset = m_LookAtOffset;
c.m_HorizontalDamping = m_HorizontalDamping;
c.m_VerticalDamping = m_VerticalDamping;
c.m_ScreenX = m_ScreenX;
c.m_ScreenY = m_ScreenY;
c.m_DeadZoneWidth = m_DeadZoneWidth;
c.m_DeadZoneHeight = m_DeadZoneHeight;
c.m_SoftZoneWidth = m_SoftZoneWidth;
c.m_SoftZoneHeight = m_SoftZoneHeight;
c.m_BiasX = m_BiasX;
c.m_BiasY = m_BiasY;
}
}
/// Blendable settings for CinemachineBasicMultiChannelPerlin
[Serializable] public class PerlinNoiseSettings
{
public float m_AmplitudeGain;
public float m_FrequencyGain;
internal void Lerp(CinemachineBasicMultiChannelPerlin p, float t)
{
p.m_AmplitudeGain = Mathf.Lerp(p.m_AmplitudeGain, m_AmplitudeGain, t);
p.m_FrequencyGain =Mathf.Lerp(p.m_FrequencyGain, m_FrequencyGain, t);
}
internal void PullFrom(CinemachineBasicMultiChannelPerlin p)
{
m_AmplitudeGain = p.m_AmplitudeGain;
m_FrequencyGain = p.m_FrequencyGain;
}
internal void PushTo(CinemachineBasicMultiChannelPerlin p)
{
p.m_AmplitudeGain = m_AmplitudeGain;
p.m_FrequencyGain = m_FrequencyGain;
}
}
// This prevents the sensor size from dirtying the scene in the event of aspect ratio change
void ISerializationCallbackReceiver.OnBeforeSerialize()
{
if (!m_Lens.IsPhysicalCamera)
m_Lens.SensorSize = Vector2.one;
}
void ISerializationCallbackReceiver.OnAfterDeserialize() {}
}
[SerializeField]
internal Rig[] m_Rigs = new Rig[2] { new Rig(), new Rig() };
/// Accessor for rig override settings
public Rig RigSettings(RigID rig) { return m_Rigs[(int)rig]; }
/// Easy access to the transposer (may be null)
CinemachineTransposer Transposer
{
get { return ComponentCache[(int)CinemachineCore.Stage.Body] as CinemachineTransposer; }
}
/// Enforce bounds for fields, when changed in inspector.
protected override void OnValidate()
{
base.OnValidate();
for (int i = 0; i < m_Rigs.Length; ++i)
m_Rigs[i].Validate();
}
private void Awake()
{
m_VerticalAxis.HasRecentering = true;
m_RadialAxis.HasRecentering = false;
}
/// Updates the child rig cache
protected override void OnEnable()
{
base.OnEnable();
UpdateInputAxisProvider();
}
///
/// API for the inspector. Internal use only
///
public void UpdateInputAxisProvider()
{
m_VerticalAxis.SetInputAxisProvider(0, null);
m_RadialAxis.SetInputAxisProvider(1, null);
var provider = GetInputAxisProvider();
if (provider != null)
{
m_VerticalAxis.SetInputAxisProvider(1, provider);
m_RadialAxis.SetInputAxisProvider(2, provider);
}
}
void Reset()
{
DestroyComponents();
#if UNITY_EDITOR
var orbital = UnityEditor.Undo.AddComponent(gameObject);
UnityEditor.Undo.AddComponent(gameObject);
#else
var orbital = gameObject.AddComponent();
gameObject.AddComponent();
#endif
orbital.HideOffsetInInspector = true;
orbital.m_BindingMode = CinemachineTransposer.BindingMode.SimpleFollowWithWorldUp;
InvalidateComponentCache();
m_Rigs = new Rig[2] { new Rig(), new Rig() };
// Default orbits
m_Orbits = new Orbit[3];
m_Orbits[0].m_Height = 10; m_Orbits[0].m_Radius = 4;
m_Orbits[1].m_Height = 2.5f; m_Orbits[1].m_Radius = 8;
m_Orbits[2].m_Height = -0.5f; m_Orbits[2].m_Radius = 5;
m_SplineCurvature = 0.5f;
}
///
/// Force the virtual camera to assume a given position and orientation.
/// Procedural placement then takes over
///
/// Worldspace pposition to take
/// Worldspace orientation to take
public override void ForceCameraPosition(Vector3 pos, Quaternion rot)
{
base.ForceCameraPosition(pos, rot);
m_VerticalAxis.Value = GetYAxisClosestValue(pos, State.ReferenceUp);
}
/// If we are transitioning from another FreeLook, grab the axis values from it.
/// The camera being deactivated. May be null.
/// Default world Up, set by the CinemachineBrain
/// Delta time for time-based effects (ignore if less than or equal to 0)
public override void OnTransitionFromCamera(
ICinemachineCamera fromCam, Vector3 worldUp, float deltaTime)
{
base.OnTransitionFromCamera(fromCam, worldUp, deltaTime);
InvokeOnTransitionInExtensions(fromCam, worldUp, deltaTime);
m_VerticalAxis.m_Recentering.DoRecentering(ref m_VerticalAxis, -1, 0.5f);
m_VerticalAxis.m_Recentering.CancelRecentering();
if (fromCam != null && m_Transitions.m_InheritPosition
&& !CinemachineCore.Instance.IsLiveInBlend(this))
{
// Note: horizontal axis already taken care of by base class
var cameraPos = fromCam.State.RawPosition;
// Special handling for FreeLook: get an undamped outgoing position
if (fromCam is CinemachineNewFreeLook)
{
var orbital = (fromCam as CinemachineNewFreeLook).Transposer;
if (orbital != null)
cameraPos = orbital.GetTargetCameraPosition(worldUp);
}
ForceCameraPosition(cameraPos, fromCam.State.FinalOrientation);
}
}
float GetYAxisClosestValue(Vector3 cameraPos, Vector3 up)
{
if (Follow != null)
{
// Rotate the camera pos to the back
Quaternion q = Quaternion.FromToRotation(up, Vector3.up);
Vector3 dir = q * (cameraPos - Follow.position);
Vector3 flatDir = dir; flatDir.y = 0;
if (!flatDir.AlmostZero())
{
float angle = Vector3.SignedAngle(flatDir, Vector3.back, Vector3.up);
dir = Quaternion.AngleAxis(angle, Vector3.up) * dir;
}
dir.x = 0;
// Sample the spline in a few places, find the 2 closest, and lerp
int i0 = 0, i1 = 0;
float a0 = 0, a1 = 0;
const int NumSamples = 13;
float step = 1f / (NumSamples-1);
for (int i = 0; i < NumSamples; ++i)
{
float a = Vector3.SignedAngle(
dir, GetLocalPositionForCameraFromInput(i * step), Vector3.right);
if (i == 0)
a0 = a1 = a;
else
{
if (Mathf.Abs(a) < Mathf.Abs(a0))
{
a1 = a0;
i1 = i0;
a0 = a;
i0 = i;
}
else if (Mathf.Abs(a) < Mathf.Abs(a1))
{
a1 = a;
i1 = i;
}
}
}
if (Mathf.Sign(a0) == Mathf.Sign(a1))
return i0 * step;
float t = Mathf.Abs(a0) / (Mathf.Abs(a0) + Mathf.Abs(a1));
return Mathf.Lerp(i0 * step, i1 * step, t);
}
return m_VerticalAxis.Value; // stay conservative
}
/// Internal use only. Called by CinemachineCore at designated update time
/// so the vcam can position itself and track its targets. All 3 child rigs are updated,
/// and a blend calculated, depending on the value of the Y axis.
/// Default world Up, set by the CinemachineBrain
/// Delta time for time-based effects (ignore if less than 0)
override public void InternalUpdateCameraState(Vector3 worldUp, float deltaTime)
{
UpdateTargetCache();
FollowTargetAttachment = 1;
LookAtTargetAttachment = 1;
// Initialize the camera state, in case the game object got moved in the editor
m_State = PullStateFromVirtualCamera(worldUp, ref m_Lens);
m_Rigs[(int)RigID.Top].m_Lens.SnapshotCameraReadOnlyProperties(ref m_Lens);
m_Rigs[(int)RigID.Bottom].m_Lens.SnapshotCameraReadOnlyProperties(ref m_Lens);
// Update our axes
bool activeCam = PreviousStateIsValid || CinemachineCore.Instance.IsLive(this);
if (!activeCam || deltaTime < 0)
m_VerticalAxis.m_Recentering.DoRecentering(ref m_VerticalAxis, -1, 0.5f);
else
{
if (m_VerticalAxis.Update(deltaTime))
m_VerticalAxis.m_Recentering.CancelRecentering();
m_RadialAxis.Update(deltaTime);
m_VerticalAxis.m_Recentering.DoRecentering(ref m_VerticalAxis, deltaTime, 0.5f);
}
// Blend the components
if (mBlender == null)
mBlender = new ComponentBlender(this);
mBlender.Blend(GetVerticalAxisValue());
// Blend the lens
if (m_Rigs[mBlender.OtherRig].m_CustomLens)
m_State.Lens = LensSettings.Lerp(
m_State.Lens, m_Rigs[mBlender.OtherRig].m_Lens, mBlender.BlendAmount);
// Do our stuff
SetReferenceLookAtTargetInState(ref m_State);
InvokeComponentPipeline(ref m_State, worldUp, deltaTime);
ApplyPositionBlendMethod(ref m_State, m_Transitions.m_BlendHint);
// Restore the components
mBlender.Restore();
// Push the raw position back to the game object's transform, so it
// moves along with the camera.
if (Follow != null)
transform.position = State.RawPosition;
if (LookAt != null)
transform.rotation = State.RawOrientation;
// Signal that it's all done
InvokePostPipelineStageCallback(this, CinemachineCore.Stage.Finalize, ref m_State, deltaTime);
PreviousStateIsValid = true;
}
ComponentBlender mBlender;
protected override void OnComponentCacheUpdated()
{
var transposer = Transposer;
if (transposer != null)
{
transposer.HideOffsetInInspector = true;
transposer.m_FollowOffset = new Vector3(
0, m_Orbits[1].m_Height, -m_Orbits[1].m_Radius);
}
}
private float GetVerticalAxisValue()
{
float range = m_VerticalAxis.m_MaxValue - m_VerticalAxis.m_MinValue;
return (range > UnityVectorExtensions.Epsilon) ? m_VerticalAxis.Value / range : 0.5f;
}
///
/// Returns the local position of the camera along the spline used to connect the
/// three camera rigs. Does not take into account the current heading of the
/// camera (or its target)
///
/// The t-value for the camera on its spline. Internally clamped to
/// the value [0,1]
/// The local offset (back + up) of the camera WRT its target based on the
/// supplied t-value
public Vector3 GetLocalPositionForCameraFromInput(float t)
{
UpdateCachedSpline();
int n = 1;
if (t > 0.5f)
{
t -= 0.5f;
n = 2;
}
Vector3 pos = SplineHelpers.Bezier3(
t * 2f, m_CachedKnots[n], m_CachedCtrl1[n], m_CachedCtrl2[n], m_CachedKnots[n+1]);
pos *= Mathf.Max(0, m_RadialAxis.Value);
return pos;
}
Vector2[] m_CachedOrbits;
float m_CachedTension;
Vector4[] m_CachedKnots;
Vector4[] m_CachedCtrl1;
Vector4[] m_CachedCtrl2;
void UpdateCachedSpline()
{
bool cacheIsValid = (m_CachedOrbits != null && m_CachedTension == m_SplineCurvature);
for (int i = 0; i < 3 && cacheIsValid; ++i)
cacheIsValid = (m_CachedOrbits[i].y == m_Orbits[i].m_Height
&& m_CachedOrbits[i].x == m_Orbits[i].m_Radius);
if (!cacheIsValid)
{
float t = m_SplineCurvature;
m_CachedKnots = new Vector4[5];
m_CachedCtrl1 = new Vector4[5];
m_CachedCtrl2 = new Vector4[5];
m_CachedKnots[1] = new Vector4(0, m_Orbits[2].m_Height, -m_Orbits[2].m_Radius, 0);
m_CachedKnots[2] = new Vector4(0, m_Orbits[1].m_Height, -m_Orbits[1].m_Radius, 0);
m_CachedKnots[3] = new Vector4(0, m_Orbits[0].m_Height, -m_Orbits[0].m_Radius, 0);
m_CachedKnots[0] = Vector4.Lerp(m_CachedKnots[0], Vector4.zero, t);
m_CachedKnots[4] = Vector4.Lerp(m_CachedKnots[3], Vector4.zero, t);
SplineHelpers.ComputeSmoothControlPoints(
ref m_CachedKnots, ref m_CachedCtrl1, ref m_CachedCtrl2);
m_CachedOrbits = new Vector2[3];
for (int i = 0; i < 3; ++i)
m_CachedOrbits[i] = new Vector2(m_Orbits[i].m_Radius, m_Orbits[i].m_Height);
m_CachedTension = m_SplineCurvature;
}
}
// Crazy damn thing for blending components at the source level
internal class ComponentBlender
{
Rig.TransposerSettings orbitalSaved = new Rig.TransposerSettings();
Rig.ComposerSettings composerSaved = new Rig.ComposerSettings();
Rig.PerlinNoiseSettings noiseSaved = new Rig.PerlinNoiseSettings();
public int OtherRig;
public float BlendAmount;
CinemachineNewFreeLook mFreeLook;
public ComponentBlender(CinemachineNewFreeLook freeLook) { mFreeLook = freeLook; }
public void Blend(float y)
{
if (y < 0.5f)
{
BlendAmount = 1 - (y * 2);
OtherRig = (int)RigID.Bottom;
}
else
{
BlendAmount = (y - 0.5f) * 2f;
OtherRig = (int)RigID.Top;
}
var orbital = mFreeLook.Transposer;
if (orbital != null && mFreeLook.m_Rigs[OtherRig].m_CustomBody)
{
orbitalSaved.PullFrom(orbital);
mFreeLook.m_Rigs[OtherRig].m_Body.Lerp(orbital, BlendAmount);
}
if (orbital != null)
orbital.m_FollowOffset = mFreeLook.GetLocalPositionForCameraFromInput(y);
var components = mFreeLook.ComponentCache;
var composer = components[(int)CinemachineCore.Stage.Aim] as CinemachineComposer;
if (composer != null && mFreeLook.m_Rigs[OtherRig].m_CustomAim)
{
composerSaved.PullFrom(composer);
mFreeLook.m_Rigs[OtherRig].m_Aim.Lerp(composer, BlendAmount);
}
var noise = components[(int)CinemachineCore.Stage.Noise] as CinemachineBasicMultiChannelPerlin;
if (noise != null && mFreeLook.m_Rigs[OtherRig].m_CustomNoise)
{
noiseSaved.PullFrom(noise);
mFreeLook.m_Rigs[OtherRig].m_Noise.Lerp(noise, BlendAmount);
}
}
public void Restore()
{
var orbital = mFreeLook.Transposer;
if (orbital != null && mFreeLook.m_Rigs[OtherRig].m_CustomBody)
orbitalSaved.PushTo(orbital);
if (orbital != null)
orbital.m_FollowOffset = new Vector3(
0, mFreeLook.m_Orbits[1].m_Height, -mFreeLook.m_Orbits[1].m_Radius);
var components = mFreeLook.ComponentCache;
var composer = components[(int)CinemachineCore.Stage.Aim] as CinemachineComposer;
if (composer != null && mFreeLook.m_Rigs[OtherRig].m_CustomAim)
composerSaved.PushTo(composer);
var noise = components[(int)CinemachineCore.Stage.Noise] as CinemachineBasicMultiChannelPerlin;
if (noise != null && mFreeLook.m_Rigs[OtherRig].m_CustomNoise)
noiseSaved.PushTo(noise);
}
}
}
}
#endif