using System; using System.Collections.Generic; using System.Reflection; #if UNITY_EDITOR using System.Linq; using UnityEditor; using ShaderKeywordFilter = UnityEditor.ShaderKeywordFilter; #endif namespace UnityEngine.Rendering.Universal { /// /// Class ScriptableRendererData contains resources for a ScriptableRenderer. /// /// public abstract class ScriptableRendererData : ScriptableObject { internal bool isInvalidated { get; set; } /// /// Class contains references to shader resources used by Rendering Debugger. /// [Serializable, ReloadGroup] public sealed class DebugShaderResources { /// /// Debug shader used to output interpolated vertex attributes. /// [Reload("Shaders/Debug/DebugReplacement.shader")] public Shader debugReplacementPS; /// /// Debug shader used to output HDR Chromacity mapping. /// [Reload("Shaders/Debug/HDRDebugView.shader")] public Shader hdrDebugViewPS; } /// /// Container for shader resources used by Rendering Debugger. /// public DebugShaderResources debugShaders; /// /// Creates the instance of the ScriptableRenderer. /// /// The instance of ScriptableRenderer protected abstract ScriptableRenderer Create(); [SerializeField] internal List m_RendererFeatures = new List(10); [SerializeField] internal List m_RendererFeatureMap = new List(10); [SerializeField] bool m_UseNativeRenderPass = false; /// /// List of additional render pass features for this renderer. /// public List rendererFeatures { get => m_RendererFeatures; } /// /// Use SetDirty when changing seeings in the ScriptableRendererData. /// It will rebuild the render passes with the new data. /// public new void SetDirty() { isInvalidated = true; } internal ScriptableRenderer InternalCreateRenderer() { isInvalidated = false; return Create(); } /// /// Editor-only function that Unity calls when the script is loaded or a value changes in the Inspector. /// protected virtual void OnValidate() { SetDirty(); #if UNITY_EDITOR // Only validate ScriptableRendererFeatures when all scripts have finished compiling (to avoid false-negatives // when ScriptableRendererFeatures haven't been compiled before this check). if (!EditorApplication.isCompiling && m_RendererFeatures.Contains(null)) ValidateRendererFeatures(); #endif } /// /// This function is called when the object becomes enabled and active. /// protected virtual void OnEnable() { SetDirty(); } /// /// Specifies whether the renderer should use Native Render Pass. /// public bool useNativeRenderPass { get => m_UseNativeRenderPass; set { SetDirty(); m_UseNativeRenderPass = value; } } /// /// Returns true if contains renderer feature with specified type. /// /// Renderer Feature type. /// internal bool TryGetRendererFeature(out T rendererFeature) where T : ScriptableRendererFeature { foreach (var target in rendererFeatures) { if (target.GetType() == typeof(T)) { rendererFeature = target as T; return true; } } rendererFeature = null; return false; } #if UNITY_EDITOR internal virtual Material GetDefaultMaterial(DefaultMaterialType materialType) { return null; } internal virtual Shader GetDefaultShader() { return null; } internal bool ValidateRendererFeatures() { // Get all Subassets var subassets = AssetDatabase.LoadAllAssetsAtPath(AssetDatabase.GetAssetPath(this)); var linkedIds = new List(); var loadedAssets = new Dictionary(); var mapValid = m_RendererFeatureMap != null && m_RendererFeatureMap?.Count == m_RendererFeatures?.Count; var debugOutput = $"{name}\nValid Sub-assets:\n"; // Collect valid, compiled sub-assets foreach (var asset in subassets) { if (asset == null || !asset.GetType().IsSubclassOf(typeof(ScriptableRendererFeature))) continue; AssetDatabase.TryGetGUIDAndLocalFileIdentifier(asset, out var guid, out long localId); loadedAssets.Add(localId, asset); debugOutput += $"-{asset.name}\n--localId={localId}\n"; } // Collect assets that are connected to the list for (var i = 0; i < m_RendererFeatures?.Count; i++) { if (!m_RendererFeatures[i]) continue; if (AssetDatabase.TryGetGUIDAndLocalFileIdentifier(m_RendererFeatures[i], out var guid, out long localId)) { linkedIds.Add(localId); } } var mapDebug = mapValid ? "Linking" : "Map missing, will attempt to re-map"; debugOutput += $"Feature List Status({mapDebug}):\n"; // Try fix missing references for (var i = 0; i < m_RendererFeatures?.Count; i++) { if (m_RendererFeatures[i] == null) { if (mapValid && m_RendererFeatureMap[i] != 0) { var localId = m_RendererFeatureMap[i]; loadedAssets.TryGetValue(localId, out var asset); m_RendererFeatures[i] = (ScriptableRendererFeature)asset; } else { m_RendererFeatures[i] = (ScriptableRendererFeature)GetUnusedAsset(ref linkedIds, ref loadedAssets); } } debugOutput += m_RendererFeatures[i] != null ? $"-{i}:Linked\n" : $"-{i}:Missing\n"; } UpdateMap(); if (!m_RendererFeatures.Contains(null)) return true; Debug.LogError($"{name} is missing RendererFeatures\nThis could be due to missing scripts or compile error.", this); return false; } internal bool DuplicateFeatureCheck(Type type) { Attribute isSingleFeature = type.GetCustomAttribute(typeof(DisallowMultipleRendererFeature)); if (isSingleFeature == null) return false; if (m_RendererFeatures == null) return false; for (int i = 0; i < m_RendererFeatures.Count; i++) { ScriptableRendererFeature feature = m_RendererFeatures[i]; if (feature == null) continue; if (feature.GetType() == type) return true; } return false; } private static object GetUnusedAsset(ref List usedIds, ref Dictionary assets) { foreach (var asset in assets) { var alreadyLinked = usedIds.Any(used => asset.Key == used); if (alreadyLinked) continue; usedIds.Add(asset.Key); return asset.Value; } return null; } private void UpdateMap() { if (m_RendererFeatureMap.Count != m_RendererFeatures.Count) { m_RendererFeatureMap.Clear(); m_RendererFeatureMap.AddRange(new long[m_RendererFeatures.Count]); } for (int i = 0; i < rendererFeatures.Count; i++) { if (m_RendererFeatures[i] == null) continue; if (!AssetDatabase.TryGetGUIDAndLocalFileIdentifier(m_RendererFeatures[i], out var guid, out long localId)) continue; m_RendererFeatureMap[i] = localId; } } #endif } }