using System; using System.Collections.Generic; using UnityEditor; using UnityEngine.Serialization; namespace UnityEngine.Rendering.Universal { /// /// Universal Render Pipeline's Global Settings. /// Global settings are unique per Render Pipeline type. In URP, Global Settings contain: /// - light layer names /// [URPHelpURL("urp-global-settings")] partial class UniversalRenderPipelineGlobalSettings : RenderPipelineGlobalSettings, ISerializationCallbackReceiver { #region Version system #pragma warning disable CS0414 [SerializeField] int k_AssetVersion = 3; #pragma warning restore CS0414 public void OnBeforeSerialize() { } public void OnAfterDeserialize() { #if UNITY_EDITOR if (k_AssetVersion != 3) { EditorApplication.delayCall += () => UpgradeAsset(this.GetInstanceID()); } #endif } #if UNITY_EDITOR static void UpgradeAsset(int assetInstanceID) { UniversalRenderPipelineGlobalSettings asset = EditorUtility.InstanceIDToObject(assetInstanceID) as UniversalRenderPipelineGlobalSettings; if (asset.k_AssetVersion < 2) { #pragma warning disable 618 // Obsolete warning // Renamed supportRuntimeDebugDisplay => stripDebugVariants, which results in inverted logic asset.m_StripDebugVariants = !asset.supportRuntimeDebugDisplay; asset.k_AssetVersion = 2; #pragma warning restore 618 // Obsolete warning // For old test projects lets keep post processing stripping enabled, as huge chance they did not used runtime profile creating #if UNITY_INCLUDE_TESTS asset.m_StripUnusedPostProcessingVariants = true; #endif } if (asset.k_AssetVersion < 3) { int index = 0; asset.m_RenderingLayerNames = new string[8]; #pragma warning disable 618 // Obsolete warning asset.m_RenderingLayerNames[index++] = asset.lightLayerName0; asset.m_RenderingLayerNames[index++] = asset.lightLayerName1; asset.m_RenderingLayerNames[index++] = asset.lightLayerName2; asset.m_RenderingLayerNames[index++] = asset.lightLayerName3; asset.m_RenderingLayerNames[index++] = asset.lightLayerName4; asset.m_RenderingLayerNames[index++] = asset.lightLayerName5; asset.m_RenderingLayerNames[index++] = asset.lightLayerName6; asset.m_RenderingLayerNames[index++] = asset.lightLayerName7; #pragma warning restore 618 // Obsolete warning asset.k_AssetVersion = 3; asset.UpdateRenderingLayerNames(); } EditorUtility.SetDirty(asset); } #endif #endregion private static UniversalRenderPipelineGlobalSettings cachedInstance = null; /// /// Active URP Global Settings asset. If the value is null then no UniversalRenderPipelineGlobalSettings has been registered to the Graphics Settings with the UniversalRenderPipeline. /// public static UniversalRenderPipelineGlobalSettings instance { get { #if !UNITY_EDITOR // The URP Global Settings could have been changed by script, undo/redo (case 1342987), or file update - file versioning, let us make sure we display the correct one // In a Player, we do not need to worry about those changes as we only support loading one if (cachedInstance == null) #endif cachedInstance = GraphicsSettings.GetSettingsForRenderPipeline() as UniversalRenderPipelineGlobalSettings; return cachedInstance; } } static internal void UpdateGraphicsSettings(UniversalRenderPipelineGlobalSettings newSettings) { if (newSettings == cachedInstance) return; if (newSettings != null) GraphicsSettings.RegisterRenderPipelineSettings(newSettings as RenderPipelineGlobalSettings); else GraphicsSettings.UnregisterRenderPipelineSettings(); cachedInstance = newSettings; } /// Default name when creating an URP Global Settings asset. public static readonly string defaultAssetName = "UniversalRenderPipelineGlobalSettings"; #if UNITY_EDITOR //Making sure there is at least one UniversalRenderPipelineGlobalSettings instance in the project internal static UniversalRenderPipelineGlobalSettings Ensure(string folderPath = "", bool canCreateNewAsset = true) { if (UniversalRenderPipelineGlobalSettings.instance) return UniversalRenderPipelineGlobalSettings.instance; UniversalRenderPipelineGlobalSettings assetCreated = null; string path = $"Assets/{folderPath}/{defaultAssetName}.asset"; assetCreated = AssetDatabase.LoadAssetAtPath(path); if (assetCreated == null) { var guidGlobalSettingsAssets = AssetDatabase.FindAssets("t:UniversalRenderPipelineGlobalSettings"); //If we could not find the asset at the default path, find the first one if (guidGlobalSettingsAssets.Length > 0) { var curGUID = guidGlobalSettingsAssets[0]; path = AssetDatabase.GUIDToAssetPath(curGUID); assetCreated = AssetDatabase.LoadAssetAtPath(path); } else if (canCreateNewAsset)// or create one altogether { if (!AssetDatabase.IsValidFolder("Assets/" + folderPath)) AssetDatabase.CreateFolder("Assets", folderPath); assetCreated = Create(path); // TODO: Reenable after next urp template is published //Debug.LogWarning("No URP Global Settings Asset is assigned. One will be created for you. If you want to modify it, go to Project Settings > Graphics > URP Settings."); } else { Debug.LogError("If you are building a Player, make sure to save an URP Global Settings asset by opening the project in the Editor first."); return null; } } Debug.Assert(assetCreated, "Could not create URP's Global Settings - URP may not work correctly - Open Project Settings > Graphics > URP Settings for additional help."); UpdateGraphicsSettings(assetCreated); return UniversalRenderPipelineGlobalSettings.instance; } internal static UniversalRenderPipelineGlobalSettings Create(string path, UniversalRenderPipelineGlobalSettings src = null) { UniversalRenderPipelineGlobalSettings assetCreated = null; // make sure the asset does not already exists assetCreated = AssetDatabase.LoadAssetAtPath(path); if (assetCreated == null) { assetCreated = ScriptableObject.CreateInstance(); if (assetCreated != null) { assetCreated.name = System.IO.Path.GetFileName(path); } AssetDatabase.CreateAsset(assetCreated, path); Debug.Assert(assetCreated); } if (assetCreated) { if (src != null) { System.Array.Copy(src.m_RenderingLayerNames, assetCreated.m_RenderingLayerNames, src.m_RenderingLayerNames.Length); } AssetDatabase.SaveAssets(); AssetDatabase.Refresh(); } return assetCreated; } #endif void Reset() { UpdateRenderingLayerNames(); } [SerializeField] string[] m_RenderingLayerNames = new string[] { "Default" }; string[] renderingLayerNames { get { if (m_RenderingLayerNames == null) UpdateRenderingLayerNames(); return m_RenderingLayerNames; } } [System.NonSerialized] string[] m_PrefixedRenderingLayerNames; string[] prefixedRenderingLayerNames { get { if (m_PrefixedRenderingLayerNames == null) UpdateRenderingLayerNames(); return m_PrefixedRenderingLayerNames; } } /// Names used for display of rendering layer masks. public string[] renderingLayerMaskNames => renderingLayerNames; /// Names used for display of rendering layer masks with a prefix. public string[] prefixedRenderingLayerMaskNames => prefixedRenderingLayerNames; [SerializeField] uint m_ValidRenderingLayers; /// Valid rendering layers that can be used by graphics. public uint validRenderingLayers => m_ValidRenderingLayers; /// Regenerate Rendering Layer names and their prefixed versions. internal void UpdateRenderingLayerNames() { // Update prefixed if (m_PrefixedRenderingLayerNames == null) m_PrefixedRenderingLayerNames = new string[32]; for (int i = 0; i < m_PrefixedRenderingLayerNames.Length; ++i) { uint renderingLayer = (uint)(1 << i); m_ValidRenderingLayers = i < m_RenderingLayerNames.Length ? (m_ValidRenderingLayers | renderingLayer) : (m_ValidRenderingLayers & ~renderingLayer); m_PrefixedRenderingLayerNames[i] = i < m_RenderingLayerNames.Length ? m_RenderingLayerNames[i] : $"Unused Layer {i}"; } // Update decals DecalProjector.UpdateAllDecalProperties(); } /// /// Names used for display of light layers with Layer's index as prefix. /// For example: "0: Light Layer Default" /// [Obsolete("This is obsolete, please use prefixedRenderingLayerMaskNames instead.", false)] public string[] prefixedLightLayerNames => new string[0]; #region Light Layer Names [3D] /// Name for light layer 0. [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string lightLayerName0; /// Name for light layer 1. [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string lightLayerName1; /// Name for light layer 2. [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string lightLayerName2; /// Name for light layer 3. [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string lightLayerName3; /// Name for light layer 4. [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string lightLayerName4; /// Name for light layer 5. [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string lightLayerName5; /// Name for light layer 6. [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string lightLayerName6; /// Name for light layer 7. [Obsolete("This is obsolete, please use renderingLayerNames instead.", false)] public string lightLayerName7; /// /// Names used for display of light layers. /// [Obsolete("This is obsolete, please use renderingLayerMaskNames instead.", false)] public string[] lightLayerNames => new string[0]; internal void ResetRenderingLayerNames() { m_RenderingLayerNames = new string[] { "Default"}; } #endregion #region Misc Settings [SerializeField] bool m_StripDebugVariants = true; [SerializeField] bool m_StripUnusedPostProcessingVariants = false; [SerializeField] bool m_StripUnusedVariants = true; [SerializeField] bool m_StripUnusedLODCrossFadeVariants = true; [SerializeField] bool m_StripScreenCoordOverrideVariants = true; /// /// Controls whether debug display shaders for Rendering Debugger are available in Player builds. /// [Obsolete("Please use stripRuntimeDebugShaders instead.", false)] public bool supportRuntimeDebugDisplay = false; /// /// Controls whether debug display shaders for Rendering Debugger are available in Player builds. /// public bool stripDebugVariants { get => m_StripDebugVariants; set { m_StripDebugVariants = value; } } /// /// Controls whether strips automatically post processing shader variants based on components. /// It strips based on VolumeProfiles in project and not scenes that actually uses it. /// public bool stripUnusedPostProcessingVariants { get => m_StripUnusedPostProcessingVariants; set { m_StripUnusedPostProcessingVariants = value; } } /// /// Controls whether strip off variants if the feature is enabled. /// public bool stripUnusedVariants { get => m_StripUnusedVariants; set { m_StripUnusedVariants = value; } } /// /// If this property is true, Unity strips the LOD variants if the LOD cross-fade feature (UniversalRenderingPipelineAsset.enableLODCrossFade) is disabled. /// [Obsolete("No longer used as Shader Prefiltering automatically strips out unused LOD Crossfade variants. Please use the LOD Crossfade setting in the URP Asset to disable the feature if not used.", false)] public bool stripUnusedLODCrossFadeVariants { get => m_StripUnusedLODCrossFadeVariants; set { m_StripUnusedLODCrossFadeVariants = value; } } /// /// Controls whether Screen Coordinates Override shader variants are automatically stripped. /// public bool stripScreenCoordOverrideVariants { get => m_StripScreenCoordOverrideVariants; set => m_StripScreenCoordOverrideVariants = value; } #endregion } }