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
}
}