using UnityEngine; namespace Cinemachine { /// /// An abstract representation of a mutator acting on a Cinemachine Virtual Camera /// [DocumentationSorting(DocumentationSortingAttribute.Level.API)] public abstract class CinemachineComponentBase : MonoBehaviour { /// Useful constant for very small floats protected const float Epsilon = Utility.UnityVectorExtensions.Epsilon; /// Get the associated CinemachineVirtualCameraBase public CinemachineVirtualCameraBase VirtualCamera { get { if (m_vcamOwner == null) m_vcamOwner = GetComponent(); if (m_vcamOwner == null && transform.parent != null) m_vcamOwner = transform.parent.GetComponent(); return m_vcamOwner; } } CinemachineVirtualCameraBase m_vcamOwner; /// Returns the owner vcam's Follow target. public Transform FollowTarget { get { CinemachineVirtualCameraBase vcam = VirtualCamera; return vcam == null ? null : vcam.ResolveFollow(vcam.Follow); } } /// Returns the owner vcam's LookAt target. public Transform LookAtTarget { get { CinemachineVirtualCameraBase vcam = VirtualCamera; return vcam == null ? null : vcam.ResolveLookAt(vcam.LookAt); } } /// Get Follow target as ICinemachineTargetGroup, or null if target is not a group public ICinemachineTargetGroup AbstractFollowTargetGroup => VirtualCamera.AbstractFollowTargetGroup; /// Get Follow target as CinemachineTargetGroup, or null if target is not a CinemachineTargetGroup public CinemachineTargetGroup FollowTargetGroup => AbstractFollowTargetGroup as CinemachineTargetGroup; /// Get the position of the Follow target. Special handling: If the Follow target is /// a VirtualCamera, returns the vcam State's position, not the transform's position public Vector3 FollowTargetPosition { get { var vcam = VirtualCamera.FollowTargetAsVcam; if (vcam != null) return vcam.State.FinalPosition; Transform target = FollowTarget; if (target != null) return TargetPositionCache.GetTargetPosition(target); return Vector3.zero; } } /// Get the rotation of the Follow target. Special handling: If the Follow target is /// a VirtualCamera, returns the vcam State's rotation, not the transform's rotation public Quaternion FollowTargetRotation { get { var vcam = VirtualCamera.FollowTargetAsVcam; if (vcam != null) return vcam.State.FinalOrientation; Transform target = FollowTarget; if (target != null) return TargetPositionCache.GetTargetRotation(target); return Quaternion.identity; } } /// Get LookAt target as ICinemachineTargetGroup, or null if target is not a group public ICinemachineTargetGroup AbstractLookAtTargetGroup => VirtualCamera.AbstractLookAtTargetGroup; /// Get LookAt target as CinemachineTargetGroup, or null if target is not a CinemachineTargetGroup public CinemachineTargetGroup LookAtTargetGroup => AbstractLookAtTargetGroup as CinemachineTargetGroup; /// Get the position of the LookAt target. Special handling: If the LookAt target is /// a VirtualCamera, returns the vcam State's position, not the transform's position public Vector3 LookAtTargetPosition { get { var vcam = VirtualCamera.LookAtTargetAsVcam; if (vcam != null) return vcam.State.FinalPosition; Transform target = LookAtTarget; if (target != null) return TargetPositionCache.GetTargetPosition(target); return Vector3.zero; } } /// Get the rotation of the LookAt target. Special handling: If the LookAt target is /// a VirtualCamera, returns the vcam State's rotation, not the transform's rotation public Quaternion LookAtTargetRotation { get { var vcam = VirtualCamera.LookAtTargetAsVcam; if (vcam != null) return vcam.State.FinalOrientation; Transform target = LookAtTarget; if (target != null) return TargetPositionCache.GetTargetRotation(target); return Quaternion.identity; } } /// Returns the owner vcam's CameraState. public CameraState VcamState { get { CinemachineVirtualCameraBase vcam = VirtualCamera; return vcam == null ? CameraState.Default : vcam.State; } } /// Returns true if this object is enabled and set up to produce results. public abstract bool IsValid { get; } /// Override this to do such things as offset the RefereceLookAt. /// Base class implementation does nothing. /// Input state that must be mutated /// Current effective deltaTime public virtual void PrePipelineMutateCameraState(ref CameraState curState, float deltaTime) {} /// What part of the pipeline this fits into public abstract CinemachineCore.Stage Stage { get; } /// Special for Body Stage compoments that want to be applied after Aim /// stage because they use the aim as inout for the procedural placement public virtual bool BodyAppliesAfterAim { get { return false; } } /// Mutates the camera state. This state will later be applied to the camera. /// Input state that must be mutated /// Delta time for time-based effects (ignore if less than 0) public abstract void MutateCameraState(ref CameraState curState, float deltaTime); /// Notification that this virtual camera is going live. /// Base class implementation does nothing. /// 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) /// Transition settings for this vcam /// True if the vcam should do an internal update as a result of this call public virtual bool OnTransitionFromCamera( ICinemachineCamera fromCam, Vector3 worldUp, float deltaTime, ref CinemachineVirtualCameraBase.TransitionParams transitionParams) { return false; } /// This is called to notify the component that a target got warped, /// so that the component can update its internal state to make the camera /// also warp seamlessy. Base class implementation does nothing. /// The object that was warped /// The amount the target's position changed public virtual void OnTargetObjectWarped(Transform target, Vector3 positionDelta) {} /// /// Force the virtual camera to assume a given position and orientation. /// Procedural placement then takes over. /// Base class implementation does nothing. /// Worldspace pposition to take /// Worldspace orientation to take public virtual void ForceCameraPosition(Vector3 pos, Quaternion rot) {} /// /// Report maximum damping time needed for this component. /// Only used in editor for timeline scrubbing. /// /// Highest damping setting in this component public virtual float GetMaxDampTime() { return 0; } /// Components that require user input should implement this and return true. public virtual bool RequiresUserInput => false; } }