using UnityEngine;
using UnityEngine.SceneManagement;
#if CINEMACHINE_POST_PROCESSING_V2
using System.Collections.Generic;
using UnityEngine.Rendering.PostProcessing;
#endif
namespace Cinemachine.PostFX
{
#if !CINEMACHINE_POST_PROCESSING_V2
// Workaround for Unity scripting bug
///
/// This behaviour is a liaison between Cinemachine with the Post-Processing v2 module. You must
/// have the Post-Processing V2 stack package installed in order to use this behaviour.
///
/// As a component on the Virtual Camera, it holds
/// a Post-Processing Profile asset that will be applied to the Unity camera whenever
/// the Virtual camera is live. It also has the optional functionality of animating
/// the Focus Distance and DepthOfField properties of the Camera State, and
/// applying them to the current Post-Processing profile, provided that profile has a
/// DepthOfField effect that is enabled.
///
[SaveDuringPlay]
[AddComponentMenu("")] // Hide in menu
public class CinemachinePostProcessing : CinemachineExtension
{
/// Apply PostProcessing effects
/// The virtual camera being processed
/// The current pipeline stage
/// The current virtual camera state
/// The current applicable deltaTime
protected override void PostPipelineStageCallback(
CinemachineVirtualCameraBase vcam,
CinemachineCore.Stage stage, ref CameraState state, float deltaTime) {}
}
#else
///
/// This behaviour is a liaison between Cinemachine with the Post-Processing v2 module. You must
/// have the Post-Processing V2 stack package installed in order to use this behaviour.
///
/// As a component on the Virtual Camera, it holds
/// a Post-Processing Profile asset that will be applied to the Unity camera whenever
/// the Virtual camera is live. It also has the optional functionality of animating
/// the Focus Distance and DepthOfField properties of the Camera State, and
/// applying them to the current Post-Processing profile, provided that profile has a
/// DepthOfField effect that is enabled.
///
[DocumentationSorting(DocumentationSortingAttribute.Level.UserRef)]
[ExecuteAlways]
[AddComponentMenu("")] // Hide in menu
[SaveDuringPlay]
[DisallowMultipleComponent]
[HelpURL(Documentation.BaseURL + "manual/CinemachinePostProcessing.html")]
public class CinemachinePostProcessing : CinemachineExtension
{
///
/// This is the priority for the vcam's PostProcessing volumes. It's set to a high
/// number in order to ensure that it overrides other volumes for the active vcam.
/// You can change this value if necessary to work with other systems.
///
static public float s_VolumePriority = 1000f;
/// This is obsolete, please use m_FocusTracking
[HideInInspector]
public bool m_FocusTracksTarget;
/// The reference object for focus tracking
public enum FocusTrackingMode
{
/// No focus tracking
None,
/// Focus offset is relative to the LookAt target
LookAtTarget,
/// Focus offset is relative to the Follow target
FollowTarget,
/// Focus offset is relative to the Custom target set here
CustomTarget,
/// Focus offset is relative to the camera
Camera
};
/// If the profile has the appropriate overrides, will set the base focus
/// distance to be the distance from the selected target to the camera.
/// The Focus Offset field will then modify that distance
[Tooltip("If the profile has the appropriate overrides, will set the base focus "
+ "distance to be the distance from the selected target to the camera."
+ "The Focus Offset field will then modify that distance.")]
public FocusTrackingMode m_FocusTracking;
/// The target to use if Focus Tracks Target is set to Custom Target
[Tooltip("The target to use if Focus Tracks Target is set to Custom Target")]
public Transform m_FocusTarget;
/// Offset from target distance, to be used with Focus Tracks Target.
/// Offsets the sharpest point away from the location of the focus target
[Tooltip("Offset from target distance, to be used with Focus Tracks Target. "
+ "Offsets the sharpest point away from the location of the focus target.")]
public float m_FocusOffset;
///
/// This Post-Processing profile will be applied whenever this virtual camera is live
///
[Tooltip("This Post-Processing profile will be applied whenever this virtual camera is live")]
public PostProcessProfile m_Profile;
class VcamExtraState
{
public PostProcessProfile mProfileCopy;
public void CreateProfileCopy(PostProcessProfile source)
{
DestroyProfileCopy();
PostProcessProfile profile = ScriptableObject.CreateInstance();
if (source != null)
{
foreach (var item in source.settings)
{
var itemCopy = Instantiate(item);
profile.settings.Add(itemCopy);
}
}
mProfileCopy = profile;
}
public void DestroyProfileCopy()
{
if (mProfileCopy != null)
RuntimeUtility.DestroyObject(mProfileCopy);
mProfileCopy = null;
}
}
/// True if the profile is enabled and nontrivial
public bool IsValid { get { return m_Profile != null && m_Profile.settings.Count > 0; } }
/// Called by the editor when the shared asset has been edited
public void InvalidateCachedProfile()
{
var list = GetAllExtraStates();
for (int i = 0; i < list.Count; ++i)
list[i].DestroyProfileCopy();
}
protected override void OnEnable()
{
base.OnEnable();
// Map legacy m_FocusTracksTarget to focus mode
if (m_FocusTracksTarget)
{
m_FocusTracking = VirtualCamera.LookAt != null
? FocusTrackingMode.LookAtTarget : FocusTrackingMode.Camera;
}
m_FocusTracksTarget = false;
}
protected override void OnDestroy()
{
InvalidateCachedProfile();
base.OnDestroy();
}
/// Apply PostProcessing effects
/// The virtual camera being processed
/// The current pipeline stage
/// The current virtual camera state
/// The current applicable deltaTime
protected override void PostPipelineStageCallback(
CinemachineVirtualCameraBase vcam,
CinemachineCore.Stage stage, ref CameraState state, float deltaTime)
{
// Set the focus after the camera has been fully positioned.
if (stage == CinemachineCore.Stage.Finalize)
{
var extra = GetExtraState(vcam);
if (!IsValid)
extra.DestroyProfileCopy();
else
{
var profile = m_Profile;
// Handle Follow Focus
if (m_FocusTracking == FocusTrackingMode.None)
extra.DestroyProfileCopy();
else
{
if (extra.mProfileCopy == null)
extra.CreateProfileCopy(m_Profile);
profile = extra.mProfileCopy;
DepthOfField dof;
if (profile.TryGetSettings(out dof))
{
float focusDistance = m_FocusOffset;
if (m_FocusTracking == FocusTrackingMode.LookAtTarget)
focusDistance += (state.FinalPosition - state.ReferenceLookAt).magnitude;
else
{
Transform focusTarget = null;
switch (m_FocusTracking)
{
default: break;
case FocusTrackingMode.FollowTarget: focusTarget = VirtualCamera.Follow; break;
case FocusTrackingMode.CustomTarget: focusTarget = m_FocusTarget; break;
}
if (focusTarget != null)
focusDistance += (state.FinalPosition - focusTarget.position).magnitude;
}
#if UNITY_2022_2_OR_NEWER
state.Lens.FocusDistance =
#endif
dof.focusDistance.value = Mathf.Max(0, focusDistance);
}
}
// Apply the post-processing
state.AddCustomBlendable(new CameraState.CustomBlendable(profile, 1));
}
}
}
static void OnCameraCut(CinemachineBrain brain)
{
// Debug.Log("Camera cut event");
PostProcessLayer postFX = GetPPLayer(brain);
if (postFX != null)
postFX.ResetHistory();
}
static void ApplyPostFX(CinemachineBrain brain)
{
PostProcessLayer ppLayer = GetPPLayer(brain);
if (ppLayer == null || !ppLayer.enabled || ppLayer.volumeLayer == 0)
return;
CameraState state = brain.CurrentCameraState;
int numBlendables = state.NumCustomBlendables;
List volumes = GetDynamicBrainVolumes(brain, ppLayer, numBlendables);
for (int i = 0; i < volumes.Count; ++i)
{
volumes[i].weight = 0;
volumes[i].sharedProfile = null;
volumes[i].profile = null;
}
PostProcessVolume firstVolume = null;
int numPPblendables = 0;
for (int i = 0; i < numBlendables; ++i)
{
var b = state.GetCustomBlendable(i);
var profile = b.m_Custom as PostProcessProfile;
if (!(profile == null)) // in case it was deleted
{
PostProcessVolume v = volumes[i];
if (firstVolume == null)
firstVolume = v;
v.sharedProfile = profile;
v.isGlobal = true;
v.priority = s_VolumePriority - (numBlendables - i) - 1;
v.weight = b.m_Weight;
++numPPblendables;
}
#if true // set this to true to force first weight to 1
// If more than one volume, then set the frst one's weight to 1
if (numPPblendables > 1)
firstVolume.weight = 1;
#endif
}
}
static string sVolumeOwnerName = "__CMVolumes";
static List sVolumes = new List();
static List GetDynamicBrainVolumes(
CinemachineBrain brain, PostProcessLayer ppLayer, int minVolumes)
{
// Locate the camera's child object that holds our dynamic volumes
GameObject volumeOwner = null;
Transform t = brain.transform;
int numChildren = t.childCount;
sVolumes.Clear();
for (int i = 0; volumeOwner == null && i < numChildren; ++i)
{
GameObject child = t.GetChild(i).gameObject;
if (child.hideFlags == HideFlags.HideAndDontSave)
{
child.GetComponents(sVolumes);
if (sVolumes.Count > 0)
volumeOwner = child;
}
}
if (minVolumes > 0)
{
if (volumeOwner == null)
{
volumeOwner = new GameObject(sVolumeOwnerName);
volumeOwner.hideFlags = HideFlags.HideAndDontSave;
volumeOwner.transform.parent = t;
}
// Update the volume's layer so it will be seen
int mask = ppLayer.volumeLayer.value;
for (int i = 0; i < 32; ++i)
{
if ((mask & (1 << i)) != 0)
{
volumeOwner.layer = i;
break;
}
}
while (sVolumes.Count < minVolumes)
sVolumes.Add(volumeOwner.gameObject.AddComponent());
}
return sVolumes;
}
static Dictionary mBrainToLayer
= new Dictionary();
static PostProcessLayer GetPPLayer(CinemachineBrain brain)
{
bool found = mBrainToLayer.TryGetValue(brain, out PostProcessLayer layer);
if (layer != null)
return layer; // layer is valid and in our lookup
// If the layer in the lookup table is a deleted object, we must remove
// the brain's callback for it
if (found && !ReferenceEquals(layer, null))
{
// layer is a deleted object
brain.m_CameraCutEvent.RemoveListener(OnCameraCut);
mBrainToLayer.Remove(brain);
layer = null;
found = false;
}
// Brain is not in our lookup - add it.
#if UNITY_2019_2_OR_NEWER
brain.TryGetComponent(out layer);
if (layer != null)
{
brain.m_CameraCutEvent.AddListener(OnCameraCut); // valid layer
mBrainToLayer[brain] = layer;
}
#else
// In order to avoid calling GetComponent() every frame in the case
// where there is legitimately no layer on the brain, we will add
// null to the lookup table if no layer is present.
if (!found)
{
layer = brain.GetComponent();
if (layer != null)
brain.m_CameraCutEvent.AddListener(OnCameraCut); // valid layer
// Exception: never add null in the case where user adds a layer while
// in the editor. If we were to add null in this case, then the new
// layer would not be detected. We are willing to live with
// calling GetComponent() every frame while in edit mode.
if (Application.isPlaying || layer != null)
mBrainToLayer[brain] = layer;
}
#endif
return layer;
}
static void CleanupLookupTable()
{
var iter = mBrainToLayer.GetEnumerator();
while (iter.MoveNext())
{
var brain = iter.Current.Key;
if (brain != null)
brain.m_CameraCutEvent.RemoveListener(OnCameraCut);
}
mBrainToLayer.Clear();
}
#if UNITY_EDITOR
[UnityEditor.InitializeOnLoad]
class EditorInitialize
{
static EditorInitialize()
{
UnityEditor.EditorApplication.playModeStateChanged += (pmsc) => CleanupLookupTable();
InitializeModule();
}
}
#endif
[RuntimeInitializeOnLoadMethod]
static void InitializeModule()
{
// After the brain pushes the state to the camera, hook in to the PostFX
CinemachineCore.CameraUpdatedEvent.RemoveListener(ApplyPostFX);
CinemachineCore.CameraUpdatedEvent.AddListener(ApplyPostFX);
// Clean up our resources
SceneManager.sceneUnloaded += (scene) => CleanupLookupTable();
}
}
#endif
}