using System; using System.Linq; using System.Collections.Generic; using UnityEngine; using UnityEngine.Rendering; using UnityEngine.Rendering.Universal; using UnityEngine.UIElements; using UnityEditor.ShaderGraph; using UnityEditor.ShaderGraph.Internal; using UnityEditor.UIElements; using UnityEditor.ShaderGraph.Serialization; using UnityEditor.ShaderGraph.Legacy; #if HAS_VFX_GRAPH using UnityEditor.VFX; #endif namespace UnityEditor.Rendering.Universal.ShaderGraph { /// /// Options for the material type. /// public enum MaterialType { /// /// Use this for URP lit. /// Lit, /// /// Use this for URP unlit. /// Unlit, /// /// Use this for sprite lit. /// SpriteLit, /// /// Use this for Sprite unlit. /// SpriteUnlit, } /// /// Workflow modes for the shader. /// public enum WorkflowMode { /// /// Use this for specular workflow. /// Specular, /// /// Use this for metallic workflow. /// Metallic, } enum SurfaceType { Opaque, Transparent, } enum ZWriteControl { Auto = 0, ForceEnabled = 1, ForceDisabled = 2 } enum ZTestMode // the values here match UnityEngine.Rendering.CompareFunction { Disabled = 0, Never = 1, Less = 2, Equal = 3, LEqual = 4, // default for most rendering Greater = 5, NotEqual = 6, GEqual = 7, Always = 8, } enum AlphaMode { Alpha, Premultiply, Additive, Multiply, } internal enum RenderFace { Front = 2, // = CullMode.Back -- render front face only Back = 1, // = CullMode.Front -- render back face only Both = 0 // = CullMode.Off -- render both faces } sealed class UniversalTarget : Target, IHasMetadata, ILegacyTarget, IMaySupportVFX #if HAS_VFX_GRAPH , IRequireVFXContext #endif { public override int latestVersion => 1; // Constants static readonly GUID kSourceCodeGuid = new GUID("8c72f47fdde33b14a9340e325ce56f4d"); // UniversalTarget.cs public const string kPipelineTag = "UniversalPipeline"; public const string kComplexLitMaterialTypeTag = "\"UniversalMaterialType\" = \"ComplexLit\""; public const string kLitMaterialTypeTag = "\"UniversalMaterialType\" = \"Lit\""; public const string kUnlitMaterialTypeTag = "\"UniversalMaterialType\" = \"Unlit\""; public static readonly string[] kSharedTemplateDirectories = GenerationUtils.GetDefaultSharedTemplateDirectories().Union(new string[] { "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Templates" #if HAS_VFX_GRAPH , "Packages/com.unity.visualeffectgraph/Editor/ShaderGraph/Templates" #endif }).ToArray(); public const string kUberTemplatePath = "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Templates/ShaderPass.template"; // SubTarget List m_SubTargets; List m_SubTargetNames; int activeSubTargetIndex => m_SubTargets.IndexOf(m_ActiveSubTarget); // Subtarget Data [SerializeField] List> m_Datas = new List>(); // View PopupField m_SubTargetField; TextField m_CustomGUIField; #if HAS_VFX_GRAPH Toggle m_SupportVFXToggle; #endif [SerializeField] JsonData m_ActiveSubTarget; // when checked, allows the material to control ALL surface settings (uber shader style) [SerializeField] bool m_AllowMaterialOverride = false; [SerializeField] SurfaceType m_SurfaceType = SurfaceType.Opaque; [SerializeField] ZTestMode m_ZTestMode = ZTestMode.LEqual; [SerializeField] ZWriteControl m_ZWriteControl = ZWriteControl.Auto; [SerializeField] AlphaMode m_AlphaMode = AlphaMode.Alpha; [SerializeField] RenderFace m_RenderFace = RenderFace.Front; [SerializeField] bool m_AlphaClip = false; [SerializeField] bool m_CastShadows = true; [SerializeField] bool m_ReceiveShadows = true; [SerializeField] bool m_SupportsLODCrossFade = false; [SerializeField] string m_CustomEditorGUI; [SerializeField] bool m_SupportVFX; internal override bool ignoreCustomInterpolators => false; internal override int padCustomInterpolatorLimit => 4; internal override bool prefersSpritePreview => activeSubTarget is UniversalSpriteUnlitSubTarget or UniversalSpriteLitSubTarget or UniversalSpriteCustomLitSubTarget; public UniversalTarget() { displayName = "Universal"; m_SubTargets = TargetUtils.GetSubTargets(this); m_SubTargetNames = m_SubTargets.Select(x => x.displayName).ToList(); TargetUtils.ProcessSubTargetList(ref m_ActiveSubTarget, ref m_SubTargets); ProcessSubTargetDatas(m_ActiveSubTarget.value); } public string renderType { get { if (surfaceType == SurfaceType.Transparent) return $"{RenderType.Transparent}"; else return $"{RenderType.Opaque}"; } } // this sets up the default renderQueue -- but it can be overridden by ResetMaterialKeywords() public string renderQueue { get { if (surfaceType == SurfaceType.Transparent) return $"{UnityEditor.ShaderGraph.RenderQueue.Transparent}"; else if (alphaClip) return $"{UnityEditor.ShaderGraph.RenderQueue.AlphaTest}"; else return $"{UnityEditor.ShaderGraph.RenderQueue.Geometry}"; } } public string disableBatching { get { if (supportsLodCrossFade) return $"{UnityEditor.ShaderGraph.DisableBatching.LODFading}"; else return $"{UnityEditor.ShaderGraph.DisableBatching.False}"; } } public SubTarget activeSubTarget { get => m_ActiveSubTarget.value; set => m_ActiveSubTarget = value; } public bool allowMaterialOverride { get => m_AllowMaterialOverride; set => m_AllowMaterialOverride = value; } public SurfaceType surfaceType { get => m_SurfaceType; set => m_SurfaceType = value; } public ZWriteControl zWriteControl { get => m_ZWriteControl; set => m_ZWriteControl = value; } public ZTestMode zTestMode { get => m_ZTestMode; set => m_ZTestMode = value; } public AlphaMode alphaMode { get => m_AlphaMode; set => m_AlphaMode = value; } public RenderFace renderFace { get => m_RenderFace; set => m_RenderFace = value; } public bool alphaClip { get => m_AlphaClip; set => m_AlphaClip = value; } public bool castShadows { get => m_CastShadows; set => m_CastShadows = value; } public bool receiveShadows { get => m_ReceiveShadows; set => m_ReceiveShadows = value; } public bool supportsLodCrossFade { get => m_SupportsLODCrossFade; set => m_SupportsLODCrossFade = value; } public string customEditorGUI { get => m_CustomEditorGUI; set => m_CustomEditorGUI = value; } // generally used to know if we need to build a depth pass public bool mayWriteDepth { get { if (allowMaterialOverride) { // material may or may not choose to write depth... we should create the depth pass return true; } else { switch (zWriteControl) { case ZWriteControl.Auto: return (surfaceType == SurfaceType.Opaque); case ZWriteControl.ForceDisabled: return false; default: return true; } } } } public override bool IsActive() { bool isUniversalRenderPipeline = GraphicsSettings.currentRenderPipeline is UniversalRenderPipelineAsset; return isUniversalRenderPipeline && activeSubTarget.IsActive(); } public override bool IsNodeAllowedByTarget(Type nodeType) { SRPFilterAttribute srpFilter = NodeClassCache.GetAttributeOnNodeType(nodeType); bool worksWithThisSrp = srpFilter == null || srpFilter.srpTypes.Contains(typeof(UniversalRenderPipeline)); SubTargetFilterAttribute subTargetFilter = NodeClassCache.GetAttributeOnNodeType(nodeType); bool worksWithThisSubTarget = subTargetFilter == null || subTargetFilter.subTargetTypes.Contains(activeSubTarget.GetType()); if (activeSubTarget.IsActive()) worksWithThisSubTarget &= activeSubTarget.IsNodeAllowedBySubTarget(nodeType); return worksWithThisSrp && worksWithThisSubTarget && base.IsNodeAllowedByTarget(nodeType); } public override void Setup(ref TargetSetupContext context) { // Setup the Target context.AddAssetDependency(kSourceCodeGuid, AssetCollection.Flags.SourceDependency); // Override EditorGUI (replaces the URP material editor by a custom one) if (!string.IsNullOrEmpty(m_CustomEditorGUI)) context.AddCustomEditorForRenderPipeline(m_CustomEditorGUI, typeof(UniversalRenderPipelineAsset)); // Setup the active SubTarget TargetUtils.ProcessSubTargetList(ref m_ActiveSubTarget, ref m_SubTargets); m_ActiveSubTarget.value.target = this; ProcessSubTargetDatas(m_ActiveSubTarget.value); m_ActiveSubTarget.value.Setup(ref context); } public override void OnAfterMultiDeserialize(string json) { TargetUtils.ProcessSubTargetList(ref m_ActiveSubTarget, ref m_SubTargets); m_ActiveSubTarget.value.target = this; // OnAfterMultiDeserialize order is not guaranteed to be hierarchical (target->subtarget). // Update active subTarget (only, since the target is shared and non-active subTargets could override active settings) // after Target has been deserialized and target <-> subtarget references are intact. m_ActiveSubTarget.value.OnAfterParentTargetDeserialized(); } public override void GetFields(ref TargetFieldContext context) { var descs = context.blocks.Select(x => x.descriptor); // Core fields context.AddField(Fields.GraphVertex, descs.Contains(BlockFields.VertexDescription.Position) || descs.Contains(BlockFields.VertexDescription.Normal) || descs.Contains(BlockFields.VertexDescription.Tangent)); context.AddField(Fields.GraphPixel); // SubTarget fields m_ActiveSubTarget.value.GetFields(ref context); } public override void GetActiveBlocks(ref TargetActiveBlockContext context) { // Core blocks if (!(m_ActiveSubTarget.value is UnityEditor.Rendering.Fullscreen.ShaderGraph.FullscreenSubTarget)) { context.AddBlock(BlockFields.VertexDescription.Position); context.AddBlock(BlockFields.VertexDescription.Normal); context.AddBlock(BlockFields.VertexDescription.Tangent); context.AddBlock(BlockFields.SurfaceDescription.BaseColor); } // SubTarget blocks m_ActiveSubTarget.value.GetActiveBlocks(ref context); } public override void ProcessPreviewMaterial(Material material) { m_ActiveSubTarget.value.ProcessPreviewMaterial(material); } public override object saveContext => m_ActiveSubTarget.value?.saveContext; public override void CollectShaderProperties(PropertyCollector collector, GenerationMode generationMode) { base.CollectShaderProperties(collector, generationMode); activeSubTarget.CollectShaderProperties(collector, generationMode); collector.AddShaderProperty(LightmappingShaderProperties.kLightmapsArray); collector.AddShaderProperty(LightmappingShaderProperties.kLightmapsIndirectionArray); collector.AddShaderProperty(LightmappingShaderProperties.kShadowMasksArray); // SubTarget blocks m_ActiveSubTarget.value.CollectShaderProperties(collector, generationMode); } public override void GetPropertiesGUI(ref TargetPropertyGUIContext context, Action onChange, Action registerUndo) { // Core properties m_SubTargetField = new PopupField(m_SubTargetNames, activeSubTargetIndex); context.AddProperty("Material", m_SubTargetField, (evt) => { if (Equals(activeSubTargetIndex, m_SubTargetField.index)) return; registerUndo("Change Material"); m_ActiveSubTarget = m_SubTargets[m_SubTargetField.index]; ProcessSubTargetDatas(m_ActiveSubTarget.value); onChange(); }); // SubTarget properties m_ActiveSubTarget.value.GetPropertiesGUI(ref context, onChange, registerUndo); // Custom Editor GUI // Requires FocusOutEvent m_CustomGUIField = new TextField("") { value = customEditorGUI }; m_CustomGUIField.RegisterCallback(s => { if (Equals(customEditorGUI, m_CustomGUIField.value)) return; registerUndo("Change Custom Editor GUI"); customEditorGUI = m_CustomGUIField.value; onChange(); }); context.AddProperty("Custom Editor GUI", m_CustomGUIField, (evt) => { }); #if HAS_VFX_GRAPH if (VFXViewPreference.generateOutputContextWithShaderGraph) { // VFX Support if (!(m_ActiveSubTarget.value is UniversalSubTarget)) context.AddHelpBox(MessageType.Info, $"The {m_ActiveSubTarget.value.displayName} target does not support VFX Graph."); else { m_SupportVFXToggle = new Toggle("") { value = m_SupportVFX }; context.AddProperty("Support VFX Graph", m_SupportVFXToggle, (evt) => { m_SupportVFX = m_SupportVFXToggle.value; }); } } #endif } // this is a copy of ZTestMode, but hides the "Disabled" option, which is invalid enum ZTestModeForUI { Never = 1, Less = 2, Equal = 3, LEqual = 4, // default for most rendering Greater = 5, NotEqual = 6, GEqual = 7, Always = 8, }; public void AddDefaultMaterialOverrideGUI(ref TargetPropertyGUIContext context, Action onChange, Action registerUndo) { // At some point we may want to convert this to be a per-property control // or Unify the UX with the upcoming "lock" feature of the Material Variant properties context.AddProperty("Allow Material Override", new Toggle() { value = allowMaterialOverride }, (evt) => { if (Equals(allowMaterialOverride, evt.newValue)) return; registerUndo("Change Allow Material Override"); allowMaterialOverride = evt.newValue; onChange(); }); } public void AddDefaultSurfacePropertiesGUI(ref TargetPropertyGUIContext context, Action onChange, Action registerUndo, bool showReceiveShadows) { context.AddProperty("Surface Type", new EnumField(SurfaceType.Opaque) { value = surfaceType }, (evt) => { if (Equals(surfaceType, evt.newValue)) return; registerUndo("Change Surface"); surfaceType = (SurfaceType)evt.newValue; onChange(); }); context.AddProperty("Blending Mode", new EnumField(AlphaMode.Alpha) { value = alphaMode }, surfaceType == SurfaceType.Transparent, (evt) => { if (Equals(alphaMode, evt.newValue)) return; registerUndo("Change Blend"); alphaMode = (AlphaMode)evt.newValue; onChange(); }); context.AddProperty("Render Face", new EnumField(RenderFace.Front) { value = renderFace }, (evt) => { if (Equals(renderFace, evt.newValue)) return; registerUndo("Change Render Face"); renderFace = (RenderFace)evt.newValue; onChange(); }); context.AddProperty("Depth Write", new EnumField(ZWriteControl.Auto) { value = zWriteControl }, (evt) => { if (Equals(zWriteControl, evt.newValue)) return; registerUndo("Change Depth Write Control"); zWriteControl = (ZWriteControl)evt.newValue; onChange(); }); context.AddProperty("Depth Test", new EnumField(ZTestModeForUI.LEqual) { value = (ZTestModeForUI)zTestMode }, (evt) => { if (Equals(zTestMode, evt.newValue)) return; registerUndo("Change Depth Test"); zTestMode = (ZTestMode)evt.newValue; onChange(); }); context.AddProperty("Alpha Clipping", "Avoid using when Alpha and AlphaThreshold are constant for the entire material as enabling in this case could introduce visual artifacts and will add an unnecessary performance cost when used with MSAA (due to AlphaToMask)", 0, new Toggle() { value = alphaClip }, (evt) => { if (Equals(alphaClip, evt.newValue)) return; registerUndo("Change Alpha Clip"); alphaClip = evt.newValue; onChange(); }); context.AddProperty("Cast Shadows", new Toggle() { value = castShadows }, (evt) => { if (Equals(castShadows, evt.newValue)) return; registerUndo("Change Cast Shadows"); castShadows = evt.newValue; onChange(); }); if (showReceiveShadows) context.AddProperty("Receive Shadows", new Toggle() { value = receiveShadows }, (evt) => { if (Equals(receiveShadows, evt.newValue)) return; registerUndo("Change Receive Shadows"); receiveShadows = evt.newValue; onChange(); }); context.AddProperty("Supports LOD Cross Fade", new Toggle() { value = supportsLodCrossFade }, (evt) => { if (Equals(supportsLodCrossFade, evt.newValue)) return; registerUndo("Change Supports LOD Cross Fade"); supportsLodCrossFade = evt.newValue; onChange(); }); } public bool TrySetActiveSubTarget(Type subTargetType) { if (!subTargetType.IsSubclassOf(typeof(SubTarget))) return false; foreach (var subTarget in m_SubTargets) { if (subTarget.GetType().Equals(subTargetType)) { m_ActiveSubTarget = subTarget; ProcessSubTargetDatas(m_ActiveSubTarget); return true; } } return false; } void ProcessSubTargetDatas(SubTarget subTarget) { var typeCollection = TypeCache.GetTypesDerivedFrom(); foreach (var type in typeCollection) { if (type.IsGenericType) continue; // Data requirement interfaces need generic type arguments // Therefore we need to use reflections to call the method var methodInfo = typeof(UniversalTarget).GetMethod(nameof(SetDataOnSubTarget)); var genericMethodInfo = methodInfo.MakeGenericMethod(type); genericMethodInfo.Invoke(this, new object[] { subTarget }); } } void ClearUnusedData() { for (int i = 0; i < m_Datas.Count; i++) { var data = m_Datas[i]; var type = data.value.GetType(); // Data requirement interfaces need generic type arguments // Therefore we need to use reflections to call the method var methodInfo = typeof(UniversalTarget).GetMethod(nameof(ValidateDataForSubTarget)); var genericMethodInfo = methodInfo.MakeGenericMethod(type); genericMethodInfo.Invoke(this, new object[] { m_ActiveSubTarget.value, data.value }); } } public void SetDataOnSubTarget(SubTarget subTarget) where T : JsonObject { if (!(subTarget is IRequiresData requiresData)) return; // Ensure data object exists in list var data = m_Datas.SelectValue().FirstOrDefault(x => x.GetType().Equals(typeof(T))) as T; if (data == null) { data = Activator.CreateInstance(typeof(T)) as T; m_Datas.Add(data); } // Apply data object to SubTarget requiresData.data = data; } public void ValidateDataForSubTarget(SubTarget subTarget, T data) where T : JsonObject { if (!(subTarget is IRequiresData requiresData)) { m_Datas.Remove(data); } } public override void OnBeforeSerialize() { ClearUnusedData(); } public bool TryUpgradeFromMasterNode(IMasterNode1 masterNode, out Dictionary blockMap) { void UpgradeAlphaClip() { var clipThresholdId = 8; var node = masterNode as AbstractMaterialNode; var clipThresholdSlot = node.FindSlot(clipThresholdId); if (clipThresholdSlot == null) return; clipThresholdSlot.owner = node; if (clipThresholdSlot.isConnected || clipThresholdSlot.value > 0.0f) { m_AlphaClip = true; } } // Upgrade Target allowMaterialOverride = false; switch (masterNode) { case PBRMasterNode1 pbrMasterNode: m_SurfaceType = (SurfaceType)pbrMasterNode.m_SurfaceType; m_AlphaMode = (AlphaMode)pbrMasterNode.m_AlphaMode; m_RenderFace = pbrMasterNode.m_TwoSided ? RenderFace.Both : RenderFace.Front; UpgradeAlphaClip(); m_CustomEditorGUI = pbrMasterNode.m_OverrideEnabled ? pbrMasterNode.m_ShaderGUIOverride : ""; break; case UnlitMasterNode1 unlitMasterNode: m_SurfaceType = (SurfaceType)unlitMasterNode.m_SurfaceType; m_AlphaMode = (AlphaMode)unlitMasterNode.m_AlphaMode; m_RenderFace = unlitMasterNode.m_TwoSided ? RenderFace.Both : RenderFace.Front; UpgradeAlphaClip(); m_CustomEditorGUI = unlitMasterNode.m_OverrideEnabled ? unlitMasterNode.m_ShaderGUIOverride : ""; break; case SpriteLitMasterNode1 spriteLitMasterNode: m_CustomEditorGUI = spriteLitMasterNode.m_OverrideEnabled ? spriteLitMasterNode.m_ShaderGUIOverride : ""; break; case SpriteUnlitMasterNode1 spriteUnlitMasterNode: m_CustomEditorGUI = spriteUnlitMasterNode.m_OverrideEnabled ? spriteUnlitMasterNode.m_ShaderGUIOverride : ""; break; } // Upgrade SubTarget foreach (var subTarget in m_SubTargets) { if (!(subTarget is ILegacyTarget legacySubTarget)) continue; if (legacySubTarget.TryUpgradeFromMasterNode(masterNode, out blockMap)) { m_ActiveSubTarget = subTarget; return true; } } blockMap = null; return false; } public override bool WorksWithSRP(RenderPipelineAsset scriptableRenderPipeline) { return scriptableRenderPipeline?.GetType() == typeof(UniversalRenderPipelineAsset); } #if HAS_VFX_GRAPH public void ConfigureContextData(VFXContext context, VFXContextCompiledData data) { if (!(m_ActiveSubTarget.value is IRequireVFXContext vfxSubtarget)) return; vfxSubtarget.ConfigureContextData(context, data); } #endif public bool CanSupportVFX() { if (m_ActiveSubTarget.value == null) return false; if (m_ActiveSubTarget.value is UniversalUnlitSubTarget) return true; if (m_ActiveSubTarget.value is UniversalLitSubTarget) return true; if (m_ActiveSubTarget.value is UniversalSpriteLitSubTarget) return true; if (m_ActiveSubTarget.value is UniversalSpriteUnlitSubTarget) return true; if (m_ActiveSubTarget.value is UniversalSpriteCustomLitSubTarget) return true; //It excludes: // - UniversalDecalSubTarget return false; } public bool SupportsVFX() => CanSupportVFX() && m_SupportVFX; [Serializable] class UniversalTargetLegacySerialization { [SerializeField] public bool m_TwoSided = false; } public override void OnAfterDeserialize(string json) { base.OnAfterDeserialize(json); if (this.sgVersion < latestVersion) { if (this.sgVersion == 0) { // deserialize the old settings to upgrade var oldSettings = JsonUtility.FromJson(json); this.m_RenderFace = oldSettings.m_TwoSided ? RenderFace.Both : RenderFace.Front; } ChangeVersion(latestVersion); } } #region Metadata string IHasMetadata.identifier { get { // defer to subtarget if (m_ActiveSubTarget.value is IHasMetadata subTargetHasMetaData) return subTargetHasMetaData.identifier; return null; } } ScriptableObject IHasMetadata.GetMetadataObject(GraphDataReadOnly graph) { // defer to subtarget if (m_ActiveSubTarget.value is IHasMetadata subTargetHasMetaData) return subTargetHasMetaData.GetMetadataObject(graph); return null; } #endregion } #region Passes static class CorePasses { /// /// Automatically enables Alpha-To-Coverage in the provided opaque pass targets using alpha clipping /// /// The pass to modify /// The target to query internal static void AddAlphaToMaskControlToPass(ref PassDescriptor pass, UniversalTarget target) { if (target.allowMaterialOverride) { // When material overrides are allowed, we have to rely on the _AlphaToMask material property since we can't be // sure of the surface type and alpha clip state based on the target alone. pass.renderStates.Add(RenderState.AlphaToMask("[_AlphaToMask]")); } else if (target.alphaClip && (target.surfaceType == SurfaceType.Opaque)) { pass.renderStates.Add(RenderState.AlphaToMask("On")); } } internal static void AddAlphaClipControlToPass(ref PassDescriptor pass, UniversalTarget target) { if (target.allowMaterialOverride) pass.keywords.Add(CoreKeywordDescriptors.AlphaTestOn); else if (target.alphaClip) pass.defines.Add(CoreKeywordDescriptors.AlphaTestOn, 1); } internal static void AddLODCrossFadeControlToPass(ref PassDescriptor pass, UniversalTarget target) { if (target.supportsLodCrossFade) { pass.includes.Add(CoreIncludes.LODCrossFade); pass.keywords.Add(CoreKeywordDescriptors.LODFadeCrossFade); pass.defines.Add(CoreKeywordDescriptors.UseUnityCrossFade, 1); } } internal static void AddTargetSurfaceControlsToPass(ref PassDescriptor pass, UniversalTarget target, bool blendModePreserveSpecular = false) { // the surface settings can either be material controlled or target controlled if (target.allowMaterialOverride) { // setup material control of via keyword pass.keywords.Add(CoreKeywordDescriptors.SurfaceTypeTransparent); pass.keywords.Add(CoreKeywordDescriptors.AlphaPremultiplyOn); pass.keywords.Add(CoreKeywordDescriptors.AlphaModulateOn); } else { // setup target control via define if (target.surfaceType == SurfaceType.Transparent) { pass.defines.Add(CoreKeywordDescriptors.SurfaceTypeTransparent, 1); // alpha premultiply in shader only needed when alpha is different for diffuse & specular if ((target.alphaMode == AlphaMode.Alpha || target.alphaMode == AlphaMode.Additive) && blendModePreserveSpecular) pass.defines.Add(CoreKeywordDescriptors.AlphaPremultiplyOn, 1); else if (target.alphaMode == AlphaMode.Multiply) pass.defines.Add(CoreKeywordDescriptors.AlphaModulateOn, 1); } } AddAlphaClipControlToPass(ref pass, target); } // used by lit/unlit subtargets public static PassDescriptor DepthOnly(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "DepthOnly", referenceName = "SHADERPASS_DEPTHONLY", lightMode = "DepthOnly", useInPreview = true, // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentAlphaOnly, // Fields structs = CoreStructCollections.Default, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.DepthOnly(target), pragmas = CorePragmas.Instanced, defines = new DefineCollection(), keywords = new KeywordCollection(), includes = new IncludeCollection { CoreIncludes.DepthOnly }, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); AddLODCrossFadeControlToPass(ref result, target); return result; } // used by lit/unlit subtargets public static PassDescriptor DepthNormal(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "DepthNormals", referenceName = "SHADERPASS_DEPTHNORMALS", lightMode = "DepthNormals", useInPreview = true, // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentDepthNormals, // Fields structs = CoreStructCollections.Default, requiredFields = CoreRequiredFields.DepthNormals, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.DepthNormalsOnly(target), pragmas = CorePragmas.Instanced, defines = new DefineCollection(), keywords = new KeywordCollection(), includes = new IncludeCollection { CoreIncludes.DepthNormalsOnly }, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); AddLODCrossFadeControlToPass(ref result, target); return result; } // used by lit/unlit subtargets public static PassDescriptor DepthNormalOnly(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "DepthNormalsOnly", referenceName = "SHADERPASS_DEPTHNORMALSONLY", lightMode = "DepthNormalsOnly", useInPreview = true, // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentDepthNormals, // Fields structs = CoreStructCollections.Default, requiredFields = CoreRequiredFields.DepthNormals, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.DepthNormalsOnly(target), pragmas = CorePragmas.Instanced, defines = new DefineCollection(), keywords = new KeywordCollection { CoreKeywordDescriptors.GBufferNormalsOct }, includes = new IncludeCollection { CoreIncludes.DepthNormalsOnly }, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); AddLODCrossFadeControlToPass(ref result, target); return result; } // used by lit/unlit targets public static PassDescriptor ShadowCaster(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "ShadowCaster", referenceName = "SHADERPASS_SHADOWCASTER", lightMode = "ShadowCaster", // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentAlphaOnly, // Fields structs = CoreStructCollections.Default, requiredFields = CoreRequiredFields.ShadowCaster, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.ShadowCaster(target), pragmas = CorePragmas.Instanced, defines = new DefineCollection(), keywords = new KeywordCollection { CoreKeywords.ShadowCaster }, includes = new IncludeCollection { CoreIncludes.ShadowCaster }, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); AddLODCrossFadeControlToPass(ref result, target); return result; } public static PassDescriptor SceneSelection(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "SceneSelectionPass", referenceName = "SHADERPASS_DEPTHONLY", lightMode = "SceneSelectionPass", useInPreview = false, // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentAlphaOnly, // Fields structs = CoreStructCollections.Default, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.SceneSelection(target), pragmas = CorePragmas.Instanced, defines = new DefineCollection { CoreDefines.SceneSelection, { CoreKeywordDescriptors.AlphaClipThreshold, 1 } }, keywords = new KeywordCollection(), includes = CoreIncludes.SceneSelection, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); return result; } public static PassDescriptor ScenePicking(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "ScenePickingPass", referenceName = "SHADERPASS_DEPTHONLY", lightMode = "Picking", useInPreview = false, // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentAlphaOnly, // Fields structs = CoreStructCollections.Default, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.ScenePicking(target), pragmas = CorePragmas.Instanced, defines = new DefineCollection { CoreDefines.ScenePicking, { CoreKeywordDescriptors.AlphaClipThreshold, 1 } }, keywords = new KeywordCollection(), includes = CoreIncludes.ScenePicking, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); return result; } public static PassDescriptor _2DSceneSelection(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "SceneSelectionPass", referenceName = "SHADERPASS_DEPTHONLY", lightMode = "SceneSelectionPass", useInPreview = false, // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentAlphaOnly, // Fields structs = CoreStructCollections.Default, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.SceneSelection(target), pragmas = CorePragmas._2DDefault, defines = new DefineCollection { CoreDefines.SceneSelection, { CoreKeywordDescriptors.AlphaClipThreshold, 0 } }, keywords = new KeywordCollection(), includes = CoreIncludes.ScenePicking, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); return result; } public static PassDescriptor _2DScenePicking(UniversalTarget target) { var result = new PassDescriptor() { // Definition displayName = "ScenePickingPass", referenceName = "SHADERPASS_DEPTHONLY", lightMode = "Picking", useInPreview = false, // Template passTemplatePath = UniversalTarget.kUberTemplatePath, sharedTemplateDirectories = UniversalTarget.kSharedTemplateDirectories, // Port Mask validVertexBlocks = CoreBlockMasks.Vertex, validPixelBlocks = CoreBlockMasks.FragmentAlphaOnly, // Fields structs = CoreStructCollections.Default, fieldDependencies = CoreFieldDependencies.Default, // Conditional State renderStates = CoreRenderStates.ScenePicking(target), pragmas = CorePragmas._2DDefault, defines = new DefineCollection { CoreDefines.ScenePicking, { CoreKeywordDescriptors.AlphaClipThreshold, 0 } }, keywords = new KeywordCollection(), includes = CoreIncludes.SceneSelection, // Custom Interpolator Support customInterpolators = CoreCustomInterpDescriptors.Common }; AddAlphaClipControlToPass(ref result, target); return result; } } #endregion #region PortMasks class CoreBlockMasks { public static readonly BlockFieldDescriptor[] Vertex = new BlockFieldDescriptor[] { BlockFields.VertexDescription.Position, BlockFields.VertexDescription.Normal, BlockFields.VertexDescription.Tangent, }; public static readonly BlockFieldDescriptor[] FragmentAlphaOnly = new BlockFieldDescriptor[] { BlockFields.SurfaceDescription.Alpha, BlockFields.SurfaceDescription.AlphaClipThreshold, }; public static readonly BlockFieldDescriptor[] FragmentColorAlpha = new BlockFieldDescriptor[] { BlockFields.SurfaceDescription.BaseColor, BlockFields.SurfaceDescription.Alpha, BlockFields.SurfaceDescription.AlphaClipThreshold, }; public static readonly BlockFieldDescriptor[] FragmentDepthNormals = new BlockFieldDescriptor[] { BlockFields.SurfaceDescription.NormalOS, BlockFields.SurfaceDescription.NormalTS, BlockFields.SurfaceDescription.NormalWS, BlockFields.SurfaceDescription.Alpha, BlockFields.SurfaceDescription.AlphaClipThreshold, }; } #endregion #region StructCollections static class CoreStructCollections { public static readonly StructCollection Default = new StructCollection { { Structs.Attributes }, { UniversalStructs.Varyings }, { Structs.SurfaceDescriptionInputs }, { Structs.VertexDescriptionInputs }, }; } #endregion #region RequiredFields static class CoreRequiredFields { public static readonly FieldCollection ShadowCaster = new FieldCollection() { StructFields.Varyings.normalWS, }; public static readonly FieldCollection DepthNormals = new FieldCollection() { StructFields.Attributes.uv1, // needed for meta vertex position StructFields.Varyings.normalWS, StructFields.Varyings.tangentWS, // needed for vertex lighting }; } #endregion #region FieldDependencies static class CoreFieldDependencies { public static readonly DependencyCollection Default = new DependencyCollection() { { FieldDependencies.Default }, new FieldDependency(UniversalStructFields.Varyings.stereoTargetEyeIndexAsRTArrayIdx, StructFields.Attributes.instanceID), new FieldDependency(UniversalStructFields.Varyings.stereoTargetEyeIndexAsBlendIdx0, StructFields.Attributes.instanceID), }; } #endregion #region RenderStates static class CoreRenderStates { public static class Uniforms { public static readonly string srcBlend = "[" + Property.SrcBlend + "]"; public static readonly string dstBlend = "[" + Property.DstBlend + "]"; public static readonly string cullMode = "[" + Property.CullMode + "]"; public static readonly string zWrite = "[" + Property.ZWrite + "]"; public static readonly string zTest = "[" + Property.ZTest + "]"; } // used by sprite targets, NOT used by lit/unlit anymore public static readonly RenderStateCollection Default = new RenderStateCollection { { RenderState.ZTest(ZTest.LEqual) }, { RenderState.ZWrite(ZWrite.On), new FieldCondition(UniversalFields.SurfaceOpaque, true) }, { RenderState.ZWrite(ZWrite.Off), new FieldCondition(UniversalFields.SurfaceTransparent, true) }, { RenderState.Cull(Cull.Back), new FieldCondition(Fields.DoubleSided, false) }, { RenderState.Cull(Cull.Off), new FieldCondition(Fields.DoubleSided, true) }, { RenderState.Blend(Blend.One, Blend.Zero), new FieldCondition(UniversalFields.SurfaceOpaque, true) }, { RenderState.Blend(Blend.SrcAlpha, Blend.OneMinusSrcAlpha, Blend.One, Blend.OneMinusSrcAlpha), new FieldCondition(Fields.BlendAlpha, true) }, { RenderState.Blend(Blend.One, Blend.OneMinusSrcAlpha, Blend.One, Blend.OneMinusSrcAlpha), new FieldCondition(UniversalFields.BlendPremultiply, true) }, { RenderState.Blend(Blend.SrcAlpha, Blend.One, Blend.One, Blend.One), new FieldCondition(UniversalFields.BlendAdd, true) }, { RenderState.Blend(Blend.DstColor, Blend.Zero), new FieldCondition(UniversalFields.BlendMultiply, true) }, }; public static Cull RenderFaceToCull(RenderFace renderFace) { switch (renderFace) { case RenderFace.Back: return Cull.Front; case RenderFace.Front: return Cull.Back; case RenderFace.Both: return Cull.Off; } return Cull.Back; } // used by lit/unlit subtargets public static RenderStateCollection UberSwitchedRenderState(UniversalTarget target, bool blendModePreserveSpecular = false) { if (target.allowMaterialOverride) { return new RenderStateCollection { RenderState.ZTest(Uniforms.zTest), RenderState.ZWrite(Uniforms.zWrite), RenderState.Cull(Uniforms.cullMode), RenderState.Blend(Uniforms.srcBlend, Uniforms.dstBlend), }; } else { var result = new RenderStateCollection(); result.Add(RenderState.ZTest(target.zTestMode.ToString())); if (target.zWriteControl == ZWriteControl.Auto) { if (target.surfaceType == SurfaceType.Opaque) result.Add(RenderState.ZWrite(ZWrite.On)); else result.Add(RenderState.ZWrite(ZWrite.Off)); } else if (target.zWriteControl == ZWriteControl.ForceEnabled) result.Add(RenderState.ZWrite(ZWrite.On)); else result.Add(RenderState.ZWrite(ZWrite.Off)); result.Add(RenderState.Cull(RenderFaceToCull(target.renderFace))); if (target.surfaceType == SurfaceType.Opaque) { result.Add(RenderState.Blend(Blend.One, Blend.Zero)); } else { // Lift alpha multiply from ROP to shader in preserve spec for different diffuse and specular blends. Blend blendSrcRGB = blendModePreserveSpecular ? Blend.One : Blend.SrcAlpha; switch (target.alphaMode) { case AlphaMode.Alpha: result.Add(RenderState.Blend(blendSrcRGB, Blend.OneMinusSrcAlpha, Blend.One, Blend.OneMinusSrcAlpha)); break; case AlphaMode.Premultiply: result.Add(RenderState.Blend(Blend.One, Blend.OneMinusSrcAlpha, Blend.One, Blend.OneMinusSrcAlpha)); break; case AlphaMode.Additive: result.Add(RenderState.Blend(blendSrcRGB, Blend.One, Blend.One, Blend.One)); break; case AlphaMode.Multiply: result.Add(RenderState.Blend(Blend.DstColor, Blend.Zero, Blend.Zero, Blend.One)); // Multiply RGB only, keep A break; } } return result; } } // used by lit target ONLY public static readonly RenderStateCollection Meta = new RenderStateCollection { { RenderState.Cull(Cull.Off) }, }; public static RenderStateDescriptor UberSwitchedCullRenderState(UniversalTarget target) { if (target.allowMaterialOverride) return RenderState.Cull(Uniforms.cullMode); else return RenderState.Cull(RenderFaceToCull(target.renderFace)); } // used by lit/unlit targets public static RenderStateCollection ShadowCaster(UniversalTarget target) { var result = new RenderStateCollection { { RenderState.ZTest(ZTest.LEqual) }, { RenderState.ZWrite(ZWrite.On) }, { UberSwitchedCullRenderState(target) }, { RenderState.ColorMask("ColorMask 0") }, }; return result; } // used by lit/unlit targets public static RenderStateCollection DepthOnly(UniversalTarget target) { var result = new RenderStateCollection { { RenderState.ZTest(ZTest.LEqual) }, { RenderState.ZWrite(ZWrite.On) }, { UberSwitchedCullRenderState(target) }, { RenderState.ColorMask("ColorMask R") }, }; return result; } // used by lit target ONLY public static RenderStateCollection DepthNormalsOnly(UniversalTarget target) { var result = new RenderStateCollection { { RenderState.ZTest(ZTest.LEqual) }, { RenderState.ZWrite(ZWrite.On) }, { UberSwitchedCullRenderState(target) } }; return result; } // Used by all targets public static RenderStateCollection SceneSelection(UniversalTarget target) { var result = new RenderStateCollection { { RenderState.Cull(Cull.Off) }, }; return result; } public static RenderStateCollection ScenePicking(UniversalTarget target) { var result = new RenderStateCollection { { UberSwitchedCullRenderState(target) } }; return result; } } #endregion #region Pragmas static class CorePragmas { public static readonly PragmaCollection Default = new PragmaCollection { { Pragma.Target(ShaderModel.Target20) }, { Pragma.Vertex("vert") }, { Pragma.Fragment("frag") }, }; public static readonly PragmaCollection Instanced = new PragmaCollection { { Pragma.Target(ShaderModel.Target20) }, { Pragma.MultiCompileInstancing }, { Pragma.Vertex("vert") }, { Pragma.Fragment("frag") }, }; public static readonly PragmaCollection Forward = new PragmaCollection { { Pragma.Target(ShaderModel.Target20) }, { Pragma.MultiCompileInstancing }, { Pragma.MultiCompileFog }, { Pragma.InstancingOptions(InstancingOptions.RenderingLayer) }, { Pragma.Vertex("vert") }, { Pragma.Fragment("frag") }, }; public static readonly PragmaCollection _2DDefault = new PragmaCollection { { Pragma.Target(ShaderModel.Target20) }, { Pragma.ExcludeRenderers(new[] { Platform.D3D9 }) }, { Pragma.Vertex("vert") }, { Pragma.Fragment("frag") }, }; public static readonly PragmaCollection GBuffer = new PragmaCollection { { Pragma.Target(ShaderModel.Target45) }, { Pragma.ExcludeRenderers(new[] { Platform.GLES, Platform.GLES3, Platform.GLCore }) }, { Pragma.MultiCompileInstancing }, { Pragma.MultiCompileFog }, { Pragma.InstancingOptions(InstancingOptions.RenderingLayer) }, { Pragma.Vertex("vert") }, { Pragma.Fragment("frag") }, }; } #endregion #region Includes static class CoreIncludes { const string kColor = "Packages/com.unity.render-pipelines.core/ShaderLibrary/Color.hlsl"; const string kTexture = "Packages/com.unity.render-pipelines.core/ShaderLibrary/Texture.hlsl"; const string kCore = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl"; const string kInput = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Input.hlsl"; const string kLighting = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Lighting.hlsl"; const string kGraphFunctions = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/ShaderGraphFunctions.hlsl"; const string kVaryings = "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Includes/Varyings.hlsl"; const string kShaderPass = "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Includes/ShaderPass.hlsl"; const string kDepthOnlyPass = "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Includes/DepthOnlyPass.hlsl"; const string kDepthNormalsOnlyPass = "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Includes/DepthNormalsOnlyPass.hlsl"; const string kShadowCasterPass = "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Includes/ShadowCasterPass.hlsl"; const string kTextureStack = "Packages/com.unity.render-pipelines.core/ShaderLibrary/TextureStack.hlsl"; const string kDBuffer = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/DBuffer.hlsl"; const string kSelectionPickingPass = "Packages/com.unity.render-pipelines.universal/Editor/ShaderGraph/Includes/SelectionPickingPass.hlsl"; const string kLODCrossFade = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/LODCrossFade.hlsl"; const string kFoveatedRenderingKeywords = "Packages/com.unity.render-pipelines.core/ShaderLibrary/FoveatedRenderingKeywords.hlsl"; const string kFoveatedRendering = "Packages/com.unity.render-pipelines.core/ShaderLibrary/FoveatedRendering.hlsl"; // Files that are included with #include_with_pragmas const string kDOTS = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/DOTS.hlsl"; const string kRenderingLayers = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/RenderingLayers.hlsl"; const string kProbeVolumes = "Packages/com.unity.render-pipelines.universal/ShaderLibrary/ProbeVolumeVariants.hlsl"; public static readonly IncludeCollection CorePregraph = new IncludeCollection { { kColor, IncludeLocation.Pregraph }, { kTexture, IncludeLocation.Pregraph }, { kCore, IncludeLocation.Pregraph }, { kLighting, IncludeLocation.Pregraph }, { kInput, IncludeLocation.Pregraph }, { kTextureStack, IncludeLocation.Pregraph }, // TODO: put this on a conditional { kFoveatedRenderingKeywords, IncludeLocation.Pregraph, true }, { kFoveatedRendering, IncludeLocation.Pregraph }, }; public static readonly IncludeCollection DOTSPregraph = new IncludeCollection { { kDOTS, IncludeLocation.Pregraph, true }, }; public static readonly IncludeCollection WriteRenderLayersPregraph = new IncludeCollection { { kRenderingLayers, IncludeLocation.Pregraph, true }, }; public static readonly IncludeCollection ShaderGraphPregraph = new IncludeCollection { { kGraphFunctions, IncludeLocation.Pregraph }, }; public static readonly IncludeCollection CorePostgraph = new IncludeCollection { { kShaderPass, IncludeLocation.Pregraph }, { kVaryings, IncludeLocation.Postgraph }, }; public static readonly IncludeCollection DepthOnly = new IncludeCollection { // Pre-graph { DOTSPregraph }, { CorePregraph }, { ShaderGraphPregraph }, // Post-graph { CorePostgraph }, { kDepthOnlyPass, IncludeLocation.Postgraph }, }; public static readonly IncludeCollection DepthNormalsOnly = new IncludeCollection { // Pre-graph { DOTSPregraph }, { WriteRenderLayersPregraph }, { CorePregraph }, { ShaderGraphPregraph }, // Post-graph { CorePostgraph }, { kDepthNormalsOnlyPass, IncludeLocation.Postgraph }, }; public static readonly IncludeCollection ShadowCaster = new IncludeCollection { // Pre-graph { DOTSPregraph }, { CorePregraph }, { ShaderGraphPregraph }, // Post-graph { CorePostgraph }, { kShadowCasterPass, IncludeLocation.Postgraph }, }; public static readonly IncludeCollection DBufferPregraph = new IncludeCollection { { kDBuffer, IncludeLocation.Pregraph }, }; public static readonly IncludeCollection SceneSelection = new IncludeCollection { // Pre-graph { CorePregraph }, { ShaderGraphPregraph }, { DOTSPregraph }, // Post-graph { CorePostgraph }, { kSelectionPickingPass, IncludeLocation.Postgraph }, }; public static readonly IncludeCollection ScenePicking = new IncludeCollection { // Pre-graph { CorePregraph }, { ShaderGraphPregraph }, { DOTSPregraph }, // Post-graph { CorePostgraph }, { kSelectionPickingPass, IncludeLocation.Postgraph }, }; public static readonly IncludeCollection LODCrossFade = new IncludeCollection { { kLODCrossFade, IncludeLocation.Pregraph } }; } #endregion #region Defines static class CoreDefines { public static readonly DefineCollection UseLegacySpriteBlocks = new DefineCollection { { CoreKeywordDescriptors.UseLegacySpriteBlocks, 1, new FieldCondition(CoreFields.UseLegacySpriteBlocks, true) }, }; public static readonly DefineCollection UseFragmentFog = new DefineCollection() { {CoreKeywordDescriptors.UseFragmentFog, 1}, }; public static readonly DefineCollection SceneSelection = new DefineCollection { { CoreKeywordDescriptors.SceneSelectionPass, 1 }, }; public static readonly DefineCollection ScenePicking = new DefineCollection { { CoreKeywordDescriptors.ScenePickingPass, 1 }, }; } #endregion #region KeywordDescriptors static class CoreKeywordDescriptors { public static readonly KeywordDescriptor StaticLightmap = new KeywordDescriptor() { displayName = "Static Lightmap", referenceName = "LIGHTMAP_ON", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor DynamicLightmap = new KeywordDescriptor() { displayName = "Dynamic Lightmap", referenceName = "DYNAMICLIGHTMAP_ON", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor DirectionalLightmapCombined = new KeywordDescriptor() { displayName = "Directional Lightmap Combined", referenceName = "DIRLIGHTMAP_COMBINED", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor SampleGI = new KeywordDescriptor() { displayName = "Sample GI", referenceName = "_SAMPLE_GI", type = KeywordType.Boolean, definition = KeywordDefinition.ShaderFeature, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor AlphaTestOn = new KeywordDescriptor() { displayName = ShaderKeywordStrings._ALPHATEST_ON, referenceName = ShaderKeywordStrings._ALPHATEST_ON, type = KeywordType.Boolean, definition = KeywordDefinition.ShaderFeature, scope = KeywordScope.Local, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor SurfaceTypeTransparent = new KeywordDescriptor() { displayName = ShaderKeywordStrings._SURFACE_TYPE_TRANSPARENT, referenceName = ShaderKeywordStrings._SURFACE_TYPE_TRANSPARENT, type = KeywordType.Boolean, definition = KeywordDefinition.ShaderFeature, scope = KeywordScope.Global, // needs to match HDRP stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor AlphaPremultiplyOn = new KeywordDescriptor() { displayName = ShaderKeywordStrings._ALPHAPREMULTIPLY_ON, referenceName = ShaderKeywordStrings._ALPHAPREMULTIPLY_ON, type = KeywordType.Boolean, definition = KeywordDefinition.ShaderFeature, scope = KeywordScope.Local, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor AlphaModulateOn = new KeywordDescriptor() { displayName = ShaderKeywordStrings._ALPHAMODULATE_ON, referenceName = ShaderKeywordStrings._ALPHAMODULATE_ON, type = KeywordType.Boolean, definition = KeywordDefinition.ShaderFeature, scope = KeywordScope.Local, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor EvaluateSh = new KeywordDescriptor() { displayName = "Evaluate SH", referenceName = "EVALUATE_SH", type = KeywordType.Enum, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, entries = new KeywordEntry[] { new KeywordEntry() { displayName = "Off", referenceName = "" }, new KeywordEntry() { displayName = "Evaluate SH Mixed", referenceName = "MIXED" }, new KeywordEntry() { displayName = "Evaluate SH Vertex", referenceName = "VERTEX" }, } }; public static readonly KeywordDescriptor MainLightShadows = new KeywordDescriptor() { displayName = "Main Light Shadows", referenceName = "", type = KeywordType.Enum, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, entries = new KeywordEntry[] { new KeywordEntry() { displayName = "Off", referenceName = "" }, new KeywordEntry() { displayName = "No Cascade", referenceName = "MAIN_LIGHT_SHADOWS" }, new KeywordEntry() { displayName = "Cascade", referenceName = "MAIN_LIGHT_SHADOWS_CASCADE" }, new KeywordEntry() { displayName = "Screen", referenceName = "MAIN_LIGHT_SHADOWS_SCREEN" }, } }; public static readonly KeywordDescriptor CastingPunctualLightShadow = new KeywordDescriptor() { displayName = "Casting Punctual Light Shadow", referenceName = "_CASTING_PUNCTUAL_LIGHT_SHADOW", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Vertex, }; public static readonly KeywordDescriptor AdditionalLights = new KeywordDescriptor() { displayName = "Additional Lights", referenceName = "", type = KeywordType.Enum, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, entries = new KeywordEntry[] { new KeywordEntry() { displayName = "Off", referenceName = "" }, new KeywordEntry() { displayName = "Vertex", referenceName = "ADDITIONAL_LIGHTS_VERTEX" }, new KeywordEntry() { displayName = "Fragment", referenceName = "ADDITIONAL_LIGHTS" }, } }; public static readonly KeywordDescriptor AdditionalLightShadows = new KeywordDescriptor() { displayName = "Additional Light Shadows", referenceName = "_ADDITIONAL_LIGHT_SHADOWS", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor ReflectionProbeBlending = new KeywordDescriptor() { displayName = "Reflection Probe Blending", referenceName = "_REFLECTION_PROBE_BLENDING", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor ReflectionProbeBoxProjection = new KeywordDescriptor() { displayName = "Reflection Probe Box Projection", referenceName = "_REFLECTION_PROBE_BOX_PROJECTION", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor ShadowsSoft = new KeywordDescriptor() { displayName = "Soft Shadows", referenceName = "", type = KeywordType.Enum, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, entries = new KeywordEntry[] { new KeywordEntry() { displayName = "Off", referenceName = "" }, new KeywordEntry() { displayName = "Soft Shadows Per Light", referenceName = "SHADOWS_SOFT" }, new KeywordEntry() { displayName = "Soft Shadows Low", referenceName = "SHADOWS_SOFT_LOW" }, new KeywordEntry() { displayName = "Soft Shadows Medium", referenceName = "SHADOWS_SOFT_MEDIUM" }, new KeywordEntry() { displayName = "Soft Shadows High", referenceName = "SHADOWS_SOFT_HIGH" }, } }; public static readonly KeywordDescriptor MixedLightingSubtractive = new KeywordDescriptor() { displayName = "Mixed Lighting Subtractive", referenceName = "_MIXED_LIGHTING_SUBTRACTIVE", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor LightmapShadowMixing = new KeywordDescriptor() { displayName = "Lightmap Shadow Mixing", referenceName = "LIGHTMAP_SHADOW_MIXING", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor ShadowsShadowmask = new KeywordDescriptor() { displayName = "Shadows Shadowmask", referenceName = "SHADOWS_SHADOWMASK", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor LightLayers = new KeywordDescriptor() { displayName = "Light Layers", referenceName = "_LIGHT_LAYERS", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor RenderPassEnabled = new KeywordDescriptor() { displayName = "Render Pass Enabled", referenceName = "_RENDER_PASS_ENABLED", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor ShapeLightType0 = new KeywordDescriptor() { displayName = "Shape Light Type 0", referenceName = "USE_SHAPE_LIGHT_TYPE_0", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor ShapeLightType1 = new KeywordDescriptor() { displayName = "Shape Light Type 1", referenceName = "USE_SHAPE_LIGHT_TYPE_1", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor ShapeLightType2 = new KeywordDescriptor() { displayName = "Shape Light Type 2", referenceName = "USE_SHAPE_LIGHT_TYPE_2", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor ShapeLightType3 = new KeywordDescriptor() { displayName = "Shape Light Type 3", referenceName = "USE_SHAPE_LIGHT_TYPE_3", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor UseLegacySpriteBlocks = new KeywordDescriptor() { displayName = "UseLegacySpriteBlocks", referenceName = "USELEGACYSPRITEBLOCKS", type = KeywordType.Boolean, }; public static readonly KeywordDescriptor UseFragmentFog = new KeywordDescriptor() { displayName = "UseFragmentFog", referenceName = "_FOG_FRAGMENT", type = KeywordType.Boolean, }; public static readonly KeywordDescriptor GBufferNormalsOct = new KeywordDescriptor() { displayName = "GBuffer normal octahedron encoding", referenceName = "_GBUFFER_NORMALS_OCT", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor DBuffer = new KeywordDescriptor() { displayName = "Decals", referenceName = "", type = KeywordType.Enum, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, entries = new KeywordEntry[] { new KeywordEntry() { displayName = "Off", referenceName = "" }, new KeywordEntry() { displayName = "DBuffer Mrt1", referenceName = "DBUFFER_MRT1" }, new KeywordEntry() { displayName = "DBuffer Mrt2", referenceName = "DBUFFER_MRT2" }, new KeywordEntry() { displayName = "DBuffer Mrt3", referenceName = "DBUFFER_MRT3" }, }, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor DebugDisplay = new KeywordDescriptor() { displayName = "Debug Display", referenceName = "DEBUG_DISPLAY", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor SceneSelectionPass = new KeywordDescriptor() { displayName = "Scene Selection Pass", referenceName = "SCENESELECTIONPASS", type = KeywordType.Boolean, }; public static readonly KeywordDescriptor ScenePickingPass = new KeywordDescriptor() { displayName = "Scene Picking Pass", referenceName = "SCENEPICKINGPASS", type = KeywordType.Boolean, }; public static readonly KeywordDescriptor AlphaClipThreshold = new KeywordDescriptor() { displayName = "AlphaClipThreshold", referenceName = "ALPHA_CLIP_THRESHOLD", type = KeywordType.Boolean, definition = KeywordDefinition.Predefined, }; public static readonly KeywordDescriptor LightCookies = new KeywordDescriptor() { displayName = "Light Cookies", referenceName = "_LIGHT_COOKIES", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor ForwardPlus = new KeywordDescriptor() { displayName = "Forward+", referenceName = "_FORWARD_PLUS", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor EditorVisualization = new KeywordDescriptor() { displayName = "Editor Visualization", referenceName = "EDITOR_VISUALIZATION", type = KeywordType.Boolean, definition = KeywordDefinition.ShaderFeature, scope = KeywordScope.Global, }; public static readonly KeywordDescriptor LODFadeCrossFade = new KeywordDescriptor() { displayName = ShaderKeywordStrings.LOD_FADE_CROSSFADE, referenceName = ShaderKeywordStrings.LOD_FADE_CROSSFADE, type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, // Note: SpeedTree shaders used to have their own PS-based Crossfade, // as well as a VS-based smooth LOD transition effect. // These shaders need the LOD_FADE_CROSSFADE keyword in the VS // to skip the VS-based effect. scope = KeywordScope.Global }; public static readonly KeywordDescriptor UseUnityCrossFade = new KeywordDescriptor() { displayName = ShaderKeywordStrings.USE_UNITY_CROSSFADE, referenceName = ShaderKeywordStrings.USE_UNITY_CROSSFADE, type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; public static readonly KeywordDescriptor ScreenSpaceAmbientOcclusion = new KeywordDescriptor() { displayName = "Screen Space Ambient Occlusion", referenceName = "_SCREEN_SPACE_OCCLUSION", type = KeywordType.Boolean, definition = KeywordDefinition.MultiCompile, scope = KeywordScope.Global, stages = KeywordShaderStage.Fragment, }; } #endregion #region Keywords static class CoreKeywords { public static readonly KeywordCollection ShadowCaster = new KeywordCollection { { CoreKeywordDescriptors.CastingPunctualLightShadow }, }; } #endregion #region FieldDescriptors static class CoreFields { public static readonly FieldDescriptor UseLegacySpriteBlocks = new FieldDescriptor("Universal", "UseLegacySpriteBlocks", "UNIVERSAL_USELEGACYSPRITEBLOCKS"); } #endregion #region CustomInterpolators static class CoreCustomInterpDescriptors { public static readonly CustomInterpSubGen.Collection Common = new CustomInterpSubGen.Collection { // Custom interpolators are not explicitly defined in the SurfaceDescriptionInputs template. // This entry point will let us generate a block of pass-through assignments for each field. CustomInterpSubGen.Descriptor.MakeBlock(CustomInterpSubGen.Splice.k_spliceCopyToSDI, "output", "input"), // sgci_PassThroughFunc is called from BuildVaryings in Varyings.hlsl to copy custom interpolators from vertex descriptions. // this entry point allows for the function to be defined before it is used. CustomInterpSubGen.Descriptor.MakeFunc(CustomInterpSubGen.Splice.k_splicePreSurface, "CustomInterpolatorPassThroughFunc", "Varyings", "VertexDescription", "CUSTOMINTERPOLATOR_VARYPASSTHROUGH_FUNC", "FEATURES_GRAPH_VERTEX") }; } #endregion }