using System.Collections.Generic; using UnityEngine.Rendering.Universal; using UnityEngine.Rendering; namespace UnityEngine.Experimental.Rendering.Universal { /// /// The queue type for the objects to render. /// public enum RenderQueueType { /// /// Use this for opaque objects. /// Opaque, /// /// Use this for transparent objects. /// Transparent, } /// /// The class for the render objects renderer feature. /// [ExcludeFromPreset] [Tooltip("Render Objects simplifies the injection of additional render passes by exposing a selection of commonly used settings.")] [URPHelpURL("renderer-features/renderer-feature-render-objects")] public class RenderObjects : ScriptableRendererFeature { /// /// Settings class used for the render objects renderer feature. /// [System.Serializable] public class RenderObjectsSettings { /// /// The profiler tag used with the pass. /// public string passTag = "RenderObjectsFeature"; /// /// Controls when the render pass executes. /// public RenderPassEvent Event = RenderPassEvent.AfterRenderingOpaques; /// /// The filter settings for the pass. /// public FilterSettings filterSettings = new FilterSettings(); /// /// The override material to use. /// public Material overrideMaterial = null; /// /// The pass index to use with the override material. /// public int overrideMaterialPassIndex = 0; /// /// The override shader to use. /// public Shader overrideShader = null; /// /// The pass index to use with the override shader. /// public int overrideShaderPassIndex = 0; /// /// Options to select which type of override mode should be used. /// public enum OverrideMaterialMode { /// /// Use this to not override. /// None, /// /// Use this to use an override material. /// Material, /// /// Use this to use an override shader. /// Shader }; /// /// The selected override mode. /// public OverrideMaterialMode overrideMode = OverrideMaterialMode.Material; //default to Material as this was previously the only option /// /// Sets whether it should override depth or not. /// public bool overrideDepthState = false; /// /// The depth comparison function to use. /// public CompareFunction depthCompareFunction = CompareFunction.LessEqual; /// /// Sets whether it should write to depth or not. /// public bool enableWrite = true; /// /// The stencil settings to use. /// public StencilStateData stencilSettings = new StencilStateData(); /// /// The camera settings to use. /// public CustomCameraSettings cameraSettings = new CustomCameraSettings(); } /// /// The filter settings used. /// [System.Serializable] public class FilterSettings { // TODO: expose opaque, transparent, all ranges as drop down /// /// The queue type for the objects to render. /// public RenderQueueType RenderQueueType; /// /// The layer mask to use. /// public LayerMask LayerMask; /// /// The passes to render. /// public string[] PassNames; /// /// The constructor for the filter settings. /// public FilterSettings() { RenderQueueType = RenderQueueType.Opaque; LayerMask = 0; } } /// /// The settings for custom cameras values. /// [System.Serializable] public class CustomCameraSettings { /// /// Used to mark whether camera values should be changed or not. /// public bool overrideCamera = false; /// /// Should the values be reverted after rendering the objects? /// public bool restoreCamera = true; /// /// Changes the camera offset. /// public Vector4 offset; /// /// Changes the camera field of view. /// public float cameraFieldOfView = 60.0f; } /// /// The settings used for the Render Objects renderer feature. /// public RenderObjectsSettings settings = new RenderObjectsSettings(); RenderObjectsPass renderObjectsPass; /// public override void Create() { FilterSettings filter = settings.filterSettings; // Render Objects pass doesn't support events before rendering prepasses. // The camera is not setup before this point and all rendering is monoscopic. // Events before BeforeRenderingPrepasses should be used for input texture passes (shadow map, LUT, etc) that doesn't depend on the camera. // These events are filtering in the UI, but we still should prevent users from changing it from code or // by changing the serialized data. if (settings.Event < RenderPassEvent.BeforeRenderingPrePasses) settings.Event = RenderPassEvent.BeforeRenderingPrePasses; renderObjectsPass = new RenderObjectsPass(settings.passTag, settings.Event, filter.PassNames, filter.RenderQueueType, filter.LayerMask, settings.cameraSettings); switch (settings.overrideMode) { case RenderObjectsSettings.OverrideMaterialMode.None: renderObjectsPass.overrideMaterial = null; renderObjectsPass.overrideShader = null; break; case RenderObjectsSettings.OverrideMaterialMode.Material: renderObjectsPass.overrideMaterial = settings.overrideMaterial; renderObjectsPass.overrideMaterialPassIndex = settings.overrideMaterialPassIndex; renderObjectsPass.overrideShader = null; break; case RenderObjectsSettings.OverrideMaterialMode.Shader: renderObjectsPass.overrideMaterial = null; renderObjectsPass.overrideShader = settings.overrideShader; renderObjectsPass.overrideShaderPassIndex = settings.overrideShaderPassIndex; break; } if (settings.overrideDepthState) renderObjectsPass.SetDetphState(settings.enableWrite, settings.depthCompareFunction); if (settings.stencilSettings.overrideStencilState) renderObjectsPass.SetStencilState(settings.stencilSettings.stencilReference, settings.stencilSettings.stencilCompareFunction, settings.stencilSettings.passOperation, settings.stencilSettings.failOperation, settings.stencilSettings.zFailOperation); } /// public override void AddRenderPasses(ScriptableRenderer renderer, ref RenderingData renderingData) { if (renderingData.cameraData.cameraType == CameraType.Preview || UniversalRenderer.IsOffscreenDepthTexture(in renderingData.cameraData)) return; renderer.EnqueuePass(renderObjectsPass); } internal override bool SupportsNativeRenderPass() { return true; } } }