using System; namespace UnityEngine.Rendering.PostProcessing { /// /// The base abstract class for all parameter override types. /// /// public abstract class ParameterOverride { /// /// The override state of this parameter. /// public bool overrideState; internal abstract void Interp(ParameterOverride from, ParameterOverride to, float t); /// /// Returns the computed hash code for this parameter. /// /// A computed hash code public abstract int GetHash(); /// /// Casts and returns the value stored in this parameter. /// /// The type to cast to /// The value stored in this parameter public T GetValue() { return ((ParameterOverride)this).value; } /// /// This method is called right after the parent has /// been initialized. This is used in case you need to access fields or properties that /// can't be accessed in the constructor of a /// (ParameterOverride objects are generally declared and initialized in a /// ). /// /// protected internal virtual void OnEnable() { } /// /// This method is called right before the parent /// gets de-initialized. /// /// protected internal virtual void OnDisable() { } internal abstract void SetValue(ParameterOverride parameter); } /// /// The base typed class for all parameter override types. /// /// The type of value to store in this ParameterOverride /// /// Due to limitations with the serialization system in Unity you shouldn't use this class /// directly. Use one of the pre-flatten types (like or make your /// own by extending this class. /// /// /// This sample code shows how to make a custom parameter holding a float. /// /// [Serializable] /// public sealed class FloatParameter : ParameterOverride<float> /// { /// public override void Interp(float from, float to, float t) /// { /// value = from + (to - from) * t; /// } /// } /// /// [Serializable] public class ParameterOverride : ParameterOverride { /// /// The value stored in this parameter. /// public T value; /// /// Creates a ParameterOverride with a default and /// set to false. /// public ParameterOverride() : this(default(T), false) { } /// /// Creates a ParameterOverride with a given value and /// set to false. /// /// The value to set this parameter to public ParameterOverride(T value) : this(value, false) { } /// /// Creates a ParameterOverride with a given value and override state. /// /// The value to set this parameter to /// The override state for this value public ParameterOverride(T value, bool overrideState) { this.value = value; this.overrideState = overrideState; } internal override void Interp(ParameterOverride from, ParameterOverride to, float t) { // Note: this isn't completely safe but it'll do fine Interp(from.GetValue(), to.GetValue(), t); } /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public virtual void Interp(T from, T to, float t) { // Returns `to` if `dt > 0` by default so we don't have to write overrides for bools and // enumerations. value = t > 0f ? to : from; } /// /// Sets the value for this parameter to and mark the override state /// to true. /// /// public void Override(T x) { overrideState = true; value = x; } internal override void SetValue(ParameterOverride parameter) { value = parameter.GetValue(); } /// /// Returns the computed hash code for this parameter. /// /// A computed hash code public override int GetHash() { unchecked { int hash = 17; hash = hash * 23 + overrideState.GetHashCode(); hash = hash * 23 + value.GetHashCode(); return hash; } } /// /// Implicit conversion between and its value type. /// /// The parameter to implicitly cast /// A value of type . public static implicit operator T(ParameterOverride prop) { return prop.value; } } // Bypassing the limited unity serialization system... /// /// A that holds a float value. /// /// /// The interpolation method for this parameter is the same as . /// [Serializable] public sealed class FloatParameter : ParameterOverride { /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(float from, float to, float t) { value = from + (to - from) * t; } } /// /// A that holds a int value. /// /// /// The interpolation method for this parameter is the same as /// casted to int. /// [Serializable] public sealed class IntParameter : ParameterOverride { /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(int from, int to, float t) { // Int snapping interpolation. Don't use this for enums as they don't necessarily have // contiguous values. Use the default interpolator instead (same as bool). value = (int)(from + (to - from) * t); } } /// /// A that holds a bool value. /// [Serializable] public sealed class BoolParameter : ParameterOverride {} /// /// A that holds a value. /// /// /// The interpolation method for this parameter is the same as /// for each channel. /// [Serializable] public sealed class ColorParameter : ParameterOverride { /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(Color from, Color to, float t) { // Lerping color values is a sensitive subject... We looked into lerping colors using // HSV and LCH but they have some downsides that make them not work correctly in all // situations, so we stick with RGB lerping for now, at least its behavior is // predictable despite looking desaturated when `t ~= 0.5` and it's faster anyway. value.r = from.r + (to.r - from.r) * t; value.g = from.g + (to.g - from.g) * t; value.b = from.b + (to.b - from.b) * t; value.a = from.a + (to.a - from.a) * t; } /// /// Implicit conversion between and a . /// /// The parameter to implicitly cast /// A Vector4. public static implicit operator Vector4(ColorParameter prop) { return prop.value; } } /// /// A that holds a value. /// /// /// The interpolation method for this parameter is the same as /// for each axis. /// [Serializable] public sealed class Vector2Parameter : ParameterOverride { /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(Vector2 from, Vector2 to, float t) { value.x = from.x + (to.x - from.x) * t; value.y = from.y + (to.y - from.y) * t; } /// /// Implicit conversion between and a . /// /// The parameter to implicitly cast /// A Vector3. public static implicit operator Vector3(Vector2Parameter prop) { return prop.value; } /// /// Implicit conversion between and a . /// /// The parameter to implicitly cast /// A Vector4. public static implicit operator Vector4(Vector2Parameter prop) { return prop.value; } } /// /// A that holds a value. /// /// /// The interpolation method for this parameter is the same as /// for each axis. /// [Serializable] public sealed class Vector3Parameter : ParameterOverride { /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(Vector3 from, Vector3 to, float t) { value.x = from.x + (to.x - from.x) * t; value.y = from.y + (to.y - from.y) * t; value.z = from.z + (to.z - from.z) * t; } /// /// Implicit conversion between and a . /// /// The parameter to implicitly cast /// A Vector2. public static implicit operator Vector2(Vector3Parameter prop) { return prop.value; } /// /// Implicit conversion between and a . /// /// The parameter to implicitly cast /// A Vector4. public static implicit operator Vector4(Vector3Parameter prop) { return prop.value; } } /// /// A that holds a value. /// /// /// The interpolation method for this parameter is the same as /// for each axis. /// [Serializable] public sealed class Vector4Parameter : ParameterOverride { /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(Vector4 from, Vector4 to, float t) { value.x = from.x + (to.x - from.x) * t; value.y = from.y + (to.y - from.y) * t; value.z = from.z + (to.z - from.z) * t; value.w = from.w + (to.w - from.w) * t; } /// /// Implicit conversion between and a . /// /// The parameter to implicitly cast /// A Vector2. public static implicit operator Vector2(Vector4Parameter prop) { return prop.value; } /// /// Implicit conversion between and a . /// /// The parameter to implicitly cast /// A Vector3. public static implicit operator Vector3(Vector4Parameter prop) { return prop.value; } } /// /// A that holds a value. /// /// /// The interpolation method for this parameter is the same as /// for each point on the curve. /// [Serializable] public sealed class SplineParameter : ParameterOverride { /// /// This method is called right after the parent has /// been initialized. This is used in case you need to access fields or properties that /// can't be accessed in the constructor of a /// (ParameterOverride objects are generally declared and initialized in a /// ). /// /// protected internal override void OnEnable() { if (value != null) value.Cache(int.MinValue); } internal override void SetValue(ParameterOverride parameter) { base.SetValue(parameter); if (value != null) value.Cache(Time.renderedFrameCount); } /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(Spline from, Spline to, float t) { if (from == null || to == null) { base.Interp(from, to, t); return; } int frameCount = Time.renderedFrameCount; from.Cache(frameCount); to.Cache(frameCount); for (int i = 0; i < Spline.k_Precision; i++) { float a = from.cachedData[i]; float b = to.cachedData[i]; value.cachedData[i] = a + (b - a) * t; } } } /// /// A set of default textures to use as default values for . /// public enum TextureParameterDefault { /// /// No texture, or null. /// None, /// /// A black texture. /// Black, /// /// A white texture. /// White, /// /// A transparent texture. /// Transparent, /// /// A 2D lookup table in strip format with width = height * height. /// Lut2D } /// /// A that holds a value. /// /// /// Texture interpolation is done using a classic linear interpolation method. /// [Serializable] public sealed class TextureParameter : ParameterOverride { /// The default state & type for the texture. public TextureParameterDefault defaultState = TextureParameterDefault.Black; /// /// Interpolates between two values given an interpolation factor . /// /// The value to interpolate from /// The value to interpolate to /// An interpolation factor (generally in range [0,1]) /// /// By default this method does a "snap" interpolation, meaning it will return the value /// if is higher than 0, /// otherwise. /// public override void Interp(Texture from, Texture to, float t) { // Both are null, do nothing if (from == null && to == null) { value = null; return; } // Both aren't null we're ready to blend if (from != null && to != null) { value = TextureLerper.instance.Lerp(from, to, t); return; } // One of them is null, blend to/from a default value is applicable { if (defaultState == TextureParameterDefault.Lut2D) { int size = from != null ? from.height : to.height; Texture defaultTexture = RuntimeUtilities.GetLutStrip(size); if (from == null) from = defaultTexture; if (to == null) to = defaultTexture; } Color tgtColor; switch (defaultState) { case TextureParameterDefault.Black: tgtColor = Color.black; break; case TextureParameterDefault.White: tgtColor = Color.white; break; case TextureParameterDefault.Transparent: tgtColor = Color.clear; break; case TextureParameterDefault.Lut2D: { // Find the current lut size int size = from != null ? from.height : to.height; Texture defaultTexture = RuntimeUtilities.GetLutStrip(size); if (from == null) from = defaultTexture; if (to == null) to = defaultTexture; // Fail safe in case the lut size is incorrect if (from.width != to.width || from.height != to.height) { value = null; return; } value = TextureLerper.instance.Lerp(from, to, t); // All done, return return; } default: // defaultState is none, so just interpolate the base and return base.Interp(from, to, t); return; } // If we made it this far, tgtColor contains the color we'll be lerping into (or out of) if (from == null) { // color -> texture lerp, invert ratio value = TextureLerper.instance.Lerp(to, tgtColor, 1f - t); } else { value = TextureLerper.instance.Lerp(from, tgtColor, t); } } } } }