Car/Assets/VolumetricLightBeam/Scripts/Config.cs

868 lines
34 KiB
C#
Raw Normal View History

#if UNITY_EDITOR
//#define PROFILE_INSTANCE_LOADING
using UnityEditor;
#endif
using UnityEngine;
using UnityEngine.Serialization;
#if VLB_URP
using UnityEngine.Rendering.Universal;
#endif
namespace VLB
{
[HelpURL(Consts.Help.UrlConfig)]
public class Config : ScriptableObject
{
public const string ClassName = "Config";
public const string kAssetName = "VLBConfigOverride";
public const string kAssetNameExt = ".asset";
/// <summary>
/// Override the layer on which the procedural geometry is created or not
/// </summary>
public bool geometryOverrideLayer = Consts.Config.GeometryOverrideLayerDefault;
/// <summary>
/// The layer the procedural geometry gameObject is in (only if geometryOverrideLayer is enabled)
/// </summary>
public int geometryLayerID = Consts.Config.GeometryLayerIDDefault;
/// <summary>
/// The tag applied on the procedural geometry gameObject
/// </summary>
public string geometryTag = Consts.Config.GeometryTagDefault;
/// <summary>
/// Determine in which order beams are rendered compared to other objects.
/// This way for example transparent objects are rendered after opaque objects, and so on.
/// </summary>
public int geometryRenderQueue = (int)Consts.Config.GeometryRenderQueueDefault;
public int geometryRenderQueueHD = (int)Consts.Config.HD.GeometryRenderQueueDefault;
/// <summary>
/// Select the Render Pipeline (Built-In or SRP) in use.
/// </summary>
public RenderPipeline renderPipeline
{
get { return m_RenderPipeline; }
set
{
#if UNITY_EDITOR
m_RenderPipeline = value;
#else
Debug.LogError("Modifying the RenderPipeline in standalone builds is not permitted");
#endif
}
}
[FormerlySerializedAs("renderPipeline"), FormerlySerializedAs("_RenderPipeline")]
[SerializeField] RenderPipeline m_RenderPipeline = Consts.Config.GeometryRenderPipelineDefault;
/// <summary>
/// MultiPass: Use the 2 pass shader. Will generate 2 drawcalls per beam.
/// SinglePass: Use the 1 pass shader. Will generate 1 drawcall per beam.
/// GPUInstancing: Dynamically batch multiple beams to combine and reduce draw calls (Feature only supported in Unity 5.6 or above). More info: https://docs.unity3d.com/Manual/GPUInstancing.html
/// SRPBatcher: Use the SRP Batcher to automatically batch multiple beams and reduce draw calls. Only available when using SRP.
/// </summary>
public RenderingMode renderingMode
{
get { return m_RenderingMode; }
set
{
#if UNITY_EDITOR
m_RenderingMode = value;
#else
Debug.LogError("Modifying the RenderingMode in standalone builds is not permitted");
#endif
}
}
[FormerlySerializedAs("renderingMode"), FormerlySerializedAs("_RenderingMode")]
[SerializeField] RenderingMode m_RenderingMode = Consts.Config.GeometryRenderingModeDefault;
public bool IsSRPBatcherSupported()
{
// The SRP Batcher Rendering Mode is only compatible when using a SRP
if (renderPipeline == RenderPipeline.BuiltIn) return false;
// SRP Batcher only works with URP and HDRP
var rp = SRPHelper.projectRenderPipeline;
return rp == RenderPipeline.URP || rp == RenderPipeline.HDRP;
}
/// <summary>
/// Actual Rendering Mode used on the current platform
/// </summary>
public RenderingMode GetActualRenderingMode(ShaderMode shaderMode)
{
if (renderingMode == RenderingMode.SRPBatcher && !IsSRPBatcherSupported()) return RenderingMode.Default;
// Using a Scriptable Render Pipeline with 'Multi-Pass' Rendering Mode is not supported
if (renderPipeline != RenderPipeline.BuiltIn && renderingMode == RenderingMode.MultiPass) return RenderingMode.Default;
// HD beams require single pass shaders
if (shaderMode == ShaderMode.HD && renderingMode == RenderingMode.MultiPass) return RenderingMode.Default;
return renderingMode;
}
/// <summary>
/// Depending on the actual Rendering Mode used, returns true if the single pass shader will be used, false otherwise.
/// </summary>
public bool SD_useSinglePassShader { get { return GetActualRenderingMode(ShaderMode.SD) != RenderingMode.MultiPass; } }
public bool SD_requiresDoubleSidedMesh { get { return SD_useSinglePassShader; } }
/// <summary>
/// Main shader applied to the cone beam geometry
/// </summary>
public Shader GetBeamShader(ShaderMode mode)
{
#if UNITY_EDITOR
var shader = GetBeamShaderInternal(mode);
if (shader == null)
RefreshShader(mode, RefreshShaderFlags.All);
return shader;
#else
return GetBeamShaderInternal(mode);
#endif
}
ref Shader GetBeamShaderInternal(ShaderMode mode)
{
if(mode == ShaderMode.SD) return ref _BeamShader;
else return ref _BeamShaderHD;
}
int GetRenderQueueInternal(ShaderMode mode)
{
if (mode == ShaderMode.SD) return geometryRenderQueue;
else return geometryRenderQueueHD;
}
public Material NewMaterialTransient(ShaderMode mode, bool gpuInstanced)
{
var material = MaterialManager.NewMaterialPersistent(GetBeamShader(mode), gpuInstanced);
if (material)
{
material.hideFlags = Consts.Internal.ProceduralObjectsHideFlags;
material.renderQueue = GetRenderQueueInternal(mode);
}
return material;
}
/// <summary>
/// Depending on the quality of your screen, you might see some artifacts with high contrast visual (like a white beam over a black background).
/// These is a very common problem known as color banding.
/// To help with this issue, the plugin offers a Dithering factor: it smooths the banding by introducing a subtle pattern of noise.
/// </summary>
public float ditheringFactor = Consts.Config.DitheringFactor;
/// <summary>
/// Contribution of the attached spotlight temperature to the final beam color.
/// Only useful when:
/// - The beams is attached to a Unity spotlight.
/// - The beams color is linked to the Unity Light color.
/// - The Unity light uses 'color temperature mode' and is specified with 'Filter' and 'Temperature' properties.
/// </summary>
public bool useLightColorTemperature = Consts.Config.UseLightColorTemperatureDefault;
/// <summary>
/// Number of Sides of the shared cone mesh
/// </summary>
public int sharedMeshSides = Consts.Config.SharedMeshSidesDefault;
/// <summary>
/// Number of Segments of the shared cone mesh
/// </summary>
public int sharedMeshSegments = Consts.Config.SharedMeshSegmentsDefault;
/// <summary>
/// Distance from the camera the beam will fade (for HD beams only, for SD beams, this option can be configured per beam)
/// 0 = hard intersection
/// Higher values produce soft intersection when the camera is near the cone triangles.
/// </summary>
public float hdBeamsCameraBlendingDistance = Consts.Config.HD.CameraBlendingDistance;
/// <summary>
/// When using URP, specify a custom Renderer index used by the depth cameras for the 'Dynamic Occlusion (Depth Buffer)' with SD beams and 'Volumetric Shadow' for HD Beams features.
/// The 'Renderer list' is editable in the URP asset.
/// We recommend to specify a custom index referencing the URP default 'ForwardRenderer' when you are using a custom renderer that doesn't support writing to depth render texture.
/// This is the case if you encounter errors like: 'RenderTexture.Create failed: colorFormat & depthStencilFormat cannot both be none.'
/// Set -1 to disable this feature.
/// </summary>
public int urpDepthCameraScriptableRendererIndex = -1;
public void SetURPScriptableRendererIndexToDepthCamera(Camera camera)
{
#if VLB_URP
if (urpDepthCameraScriptableRendererIndex < 0)
return;
Debug.Assert(camera);
var cameraData = camera.GetUniversalAdditionalCameraData();
if (cameraData)
{
cameraData.SetRenderer(urpDepthCameraScriptableRendererIndex);
}
#endif
}
/// <summary>
/// Global 3D Noise texture scaling: higher scale make the noise more visible, but potentially less realistic.
/// </summary>
[Range(Consts.Beam.NoiseScaleMin, Consts.Beam.NoiseScaleMax)]
public float globalNoiseScale = Consts.Beam.NoiseScaleDefault;
/// <summary>
/// Global World Space direction and speed of the noise scrolling, simulating the fog/smoke movement
/// </summary>
public Vector3 globalNoiseVelocity = Consts.Beam.NoiseVelocityDefault;
/// <summary>
/// Tag used to retrieve the camera used to compute the fade out factor on beams
/// </summary>
public string fadeOutCameraTag = Consts.Config.FadeOutCameraTagDefault;
public Transform fadeOutCameraTransform
{
get
{
if (m_CachedFadeOutCamera == null)
{
ForceUpdateFadeOutCamera();
}
return m_CachedFadeOutCamera;
}
}
/// <summary>
/// Call this function if you want to manually change the fadeOutCameraTag property at runtime
/// </summary>
public void ForceUpdateFadeOutCamera()
{
var gao = GameObject.FindGameObjectWithTag(fadeOutCameraTag);
if (gao)
m_CachedFadeOutCamera = gao.transform;
}
/// <summary>
/// 3D Texture storing noise data.
/// </summary>
[HighlightNull]
public Texture3D noiseTexture3D = null;
/// <summary>
/// ParticleSystem prefab instantiated for the Volumetric Dust Particles feature (Unity 5.5 or above)
/// </summary>
[HighlightNull]
public ParticleSystem dustParticlesPrefab = null;
/// <summary>
/// Noise texture for dithering feature
/// </summary>
[HighlightNull]
public Texture2D ditheringNoiseTexture = null;
[HighlightNull]
public Texture2D jitteringNoiseTexture = null;
/// <summary>
/// Off: do not support having a gradient as color.
/// High Only: support gradient color only for devices with Shader Level = 35 or higher.
/// High and Low: support gradient color for all devices.
/// </summary>
public FeatureEnabledColorGradient featureEnabledColorGradient = Consts.Config.FeatureEnabledColorGradientDefault;
/// <summary>
/// Support 'Soft Intersection with Opaque Geometry' feature or not.
/// </summary>
public bool featureEnabledDepthBlend = Consts.Config.FeatureEnabledDefault;
/// <summary>
/// Support 'Noise 3D' feature or not.
/// </summary>
public bool featureEnabledNoise3D = Consts.Config.FeatureEnabledDefault;
/// <summary>
/// Support 'Dynamic Occlusion' features or not.
/// </summary>
public bool featureEnabledDynamicOcclusion = Consts.Config.FeatureEnabledDefault;
/// <summary>
/// Support 'Mesh Skewing' feature or not.
/// </summary>
public bool featureEnabledMeshSkewing = Consts.Config.FeatureEnabledDefault;
/// <summary>
/// Support 'Shader Accuracy' property set to 'High' or not.
/// </summary>
public bool featureEnabledShaderAccuracyHigh = Consts.Config.FeatureEnabledDefault;
/// <summary>
/// Support 'Shadow' features or not.
/// </summary>
public bool featureEnabledShadow = true;
/// <summary>
/// Support 'Cookie' feature or not.
/// </summary>
public bool featureEnabledCookie = true;
/// RAYMARCHING BEGIN
[SerializeField] RaymarchingQuality[] m_RaymarchingQualities = null;
[SerializeField] int m_DefaultRaymarchingQualityUniqueID = 0;
public int defaultRaymarchingQualityUniqueID => m_DefaultRaymarchingQualityUniqueID;
public RaymarchingQuality GetRaymarchingQualityForIndex(int index)
{
Debug.Assert(index >= 0);
Debug.Assert(m_RaymarchingQualities != null);
Debug.Assert(index < m_RaymarchingQualities.Length);
return m_RaymarchingQualities[index];
}
public RaymarchingQuality GetRaymarchingQualityForUniqueID(int id)
{
int index = GetRaymarchingQualityIndexForUniqueID(id);
if (index >= 0)
return GetRaymarchingQualityForIndex(index);
return null;
}
public int GetRaymarchingQualityIndexForUniqueID(int id)
{
for (int i = 0; i < m_RaymarchingQualities.Length; ++i)
{
var qual = m_RaymarchingQualities[i];
if (qual != null && qual.uniqueID == id)
return i;
}
Debug.LogErrorFormat("Failed to find RaymarchingQualityIndex for Unique ID {0}", id);
return -1;
}
public bool IsRaymarchingQualityUniqueIDValid(int id) { return GetRaymarchingQualityIndexForUniqueID(id) >= 0; }
#if UNITY_EDITOR
public void AddRaymarchingQuality(RaymarchingQuality qual)
{
ArrayUtility.Add(ref m_RaymarchingQualities, qual);
}
public void RemoveRaymarchingQualityAtIndex(int index)
{
Debug.Assert(index >= 0);
Debug.Assert(index < m_RaymarchingQualities.Length);
ArrayUtility.RemoveAt(ref m_RaymarchingQualities, index);
}
#endif
public int raymarchingQualitiesCount { get { return Mathf.Max(1, m_RaymarchingQualities != null ? m_RaymarchingQualities.Length : 1); } }
void CreateDefaultRaymarchingQualityPreset(bool onlyIfNeeded)
{
if (m_RaymarchingQualities == null || m_RaymarchingQualities.Length == 0 || !onlyIfNeeded)
{
m_RaymarchingQualities = new RaymarchingQuality[3];
// set forced unique ID for default qualities to keep finding them even when deleting Config instance
m_RaymarchingQualities[0] = RaymarchingQuality.New("Fast", 1, 5);
m_RaymarchingQualities[1] = RaymarchingQuality.New("Balanced", 2, 10);
m_RaymarchingQualities[2] = RaymarchingQuality.New("High", 3, 20);
m_DefaultRaymarchingQualityUniqueID = m_RaymarchingQualities[1].uniqueID;
}
}
/// RAYMARCHING END
public bool isHDRPExposureWeightSupported
{
get
{
#if UNITY_2021_1_OR_NEWER
return renderPipeline == RenderPipeline.HDRP;
#else
return false; // GetCurrentExposureMultiplier is accessible but doesn't return a proper value in Unity 2020 for some reasons
#endif
}
}
// INTERNAL
#pragma warning disable 0414
[SerializeField] int pluginVersion = -1;
[SerializeField] Material _DummyMaterial = null;
[SerializeField] Material _DummyMaterialHD = null;
[SerializeField] Shader _BeamShader = null;
[SerializeField] Shader _BeamShaderHD = null;
#pragma warning restore 0414
Transform m_CachedFadeOutCamera = null;
public bool hasRenderPipelineMismatch { get { return (SRPHelper.projectRenderPipeline == RenderPipeline.BuiltIn) != (m_RenderPipeline == RenderPipeline.BuiltIn); } }
[RuntimeInitializeOnLoadMethod]
static void OnStartup()
{
Instance.m_CachedFadeOutCamera = null;
Instance.RefreshGlobalShaderProperties();
#if UNITY_EDITOR
Instance.RefreshShaders(RefreshShaderFlags.All);
#endif
if (Instance.hasRenderPipelineMismatch)
Debug.LogError("It looks like the 'Render Pipeline' is not correctly set in the config. Please make sure to select the proper value depending on your pipeline in use.", Instance);
}
#if UNITY_EDITOR
[InitializeOnLoadMethod]
static void OnProjectLoadedInEditor()
{
// Code executed on Unity Editor startup
// use the static variable and NOT the Instance property to prevent from creating a Config instance right away when you unpack the plugin,
// otherwise other assets (noise texture...) might not be loaded and references can be broken
if (ms_Instance)
ms_Instance.SetScriptingDefineSymbolsForCurrentRenderPipeline();
}
public void SetScriptingDefineSymbolsForCurrentRenderPipeline()
{
SRPHelper.SetScriptingDefineSymbolsForRenderPipeline(renderPipeline);
}
#endif
public void Reset()
{
geometryOverrideLayer = Consts.Config.GeometryOverrideLayerDefault;
geometryLayerID = Consts.Config.GeometryLayerIDDefault;
geometryTag = Consts.Config.GeometryTagDefault;
geometryRenderQueue = (int)Consts.Config.GeometryRenderQueueDefault;
geometryRenderQueueHD = (int)Consts.Config.HD.GeometryRenderQueueDefault;
sharedMeshSides = Consts.Config.SharedMeshSidesDefault;
sharedMeshSegments = Consts.Config.SharedMeshSegmentsDefault;
globalNoiseScale = Consts.Beam.NoiseScaleDefault;
globalNoiseVelocity = Consts.Beam.NoiseVelocityDefault;
renderPipeline = Consts.Config.GeometryRenderPipelineDefault;
renderingMode = Consts.Config.GeometryRenderingModeDefault;
ditheringFactor = Consts.Config.DitheringFactor;
useLightColorTemperature = Consts.Config.UseLightColorTemperatureDefault;
fadeOutCameraTag = Consts.Config.FadeOutCameraTagDefault;
featureEnabledColorGradient = Consts.Config.FeatureEnabledColorGradientDefault;
featureEnabledDepthBlend = Consts.Config.FeatureEnabledDefault;
featureEnabledNoise3D = Consts.Config.FeatureEnabledDefault;
featureEnabledDynamicOcclusion = Consts.Config.FeatureEnabledDefault;
featureEnabledMeshSkewing = Consts.Config.FeatureEnabledDefault;
featureEnabledShaderAccuracyHigh = Consts.Config.FeatureEnabledDefault;
hdBeamsCameraBlendingDistance = Consts.Config.HD.CameraBlendingDistance;
urpDepthCameraScriptableRendererIndex = -1;
CreateDefaultRaymarchingQualityPreset(onlyIfNeeded: false);
ResetInternalData();
#if UNITY_EDITOR
GlobalMeshSD.Destroy();
Utils._EditorSetAllMeshesDirty();
#endif
}
void RefreshGlobalShaderProperties()
{
Shader.SetGlobalFloat(ShaderProperties.GlobalUsesReversedZBuffer, SystemInfo.usesReversedZBuffer ? 1.0f : 0.0f);
Shader.SetGlobalFloat(ShaderProperties.GlobalDitheringFactor, ditheringFactor);
Shader.SetGlobalTexture(ShaderProperties.GlobalDitheringNoiseTex, ditheringNoiseTexture);
Shader.SetGlobalFloat(ShaderProperties.HD.GlobalCameraBlendingDistance, hdBeamsCameraBlendingDistance);
Shader.SetGlobalTexture(ShaderProperties.HD.GlobalJitteringNoiseTex, jitteringNoiseTexture);
}
#if UNITY_EDITOR
public void _EditorSetRenderingModeAndRefreshShader(RenderingMode mode)
{
renderingMode = mode;
RefreshShaders(RefreshShaderFlags.All);
}
void OnValidate()
{
sharedMeshSides = Mathf.Clamp(sharedMeshSides, Consts.Config.SharedMeshSidesMin, Consts.Config.SharedMeshSidesMax);
sharedMeshSegments = Mathf.Clamp(sharedMeshSegments, Consts.Config.SharedMeshSegmentsMin, Consts.Config.SharedMeshSegmentsMax);
ditheringFactor = Mathf.Clamp01(ditheringFactor);
hdBeamsCameraBlendingDistance = Mathf.Max(hdBeamsCameraBlendingDistance, 0f);
}
void AutoSelectRenderPipeline()
{
var newPipeline = SRPHelper.projectRenderPipeline;
if (newPipeline != renderPipeline)
{
renderPipeline = newPipeline;
EditorUtility.SetDirty(this); // make sure to save this property change
RefreshShaders(RefreshShaderFlags.All);
SetScriptingDefineSymbolsForCurrentRenderPipeline();
}
}
public static void EditorSelectInstance()
{
Selection.activeObject = Instance; // this will create the instance if it doesn't exist
if (Selection.activeObject == null)
Debug.LogError("Cannot find any Config resource");
}
ref Material GetDummyMaterial(ShaderMode shaderMode)
{
if (shaderMode == ShaderMode.SD) return ref _DummyMaterial;
else return ref _DummyMaterialHD;
}
[System.Flags]
public enum RefreshShaderFlags
{
Reference = 1 << 1,
Dummy = 1 << 2,
All = Reference | Dummy,
}
public void RefreshShaders(RefreshShaderFlags flags)
{
foreach (ShaderMode shaderMode in System.Enum.GetValues(typeof(ShaderMode)))
RefreshShader(shaderMode, flags);
}
public void RefreshShader(ShaderMode shaderMode, RefreshShaderFlags flags)
{
ref Shader shader = ref GetBeamShaderInternal(shaderMode);
if (flags.HasFlag(RefreshShaderFlags.Reference))
{
var prevShader = shader;
var configProps = new ShaderGenerator.ConfigProps
{
renderPipeline = m_RenderPipeline,
renderingMode = GetActualRenderingMode(shaderMode),
dithering = ditheringFactor > 0.0f,
noise3D = featureEnabledNoise3D,
colorGradient = featureEnabledColorGradient,
depthBlend = featureEnabledDepthBlend,
dynamicOcclusion = featureEnabledDynamicOcclusion,
meshSkewing = featureEnabledMeshSkewing,
shaderAccuracyHigh = featureEnabledShaderAccuracyHigh,
cookie = featureEnabledCookie,
shadow = featureEnabledShadow,
raymarchingQualities = m_RaymarchingQualities
};
shader = ShaderGenerator.Generate(shaderMode, configProps);
if (shader != prevShader)
{
EditorUtility.SetDirty(this);
}
}
if (flags.HasFlag(RefreshShaderFlags.Dummy) && shader != null)
{
bool gpuInstanced = GetActualRenderingMode(shaderMode) == RenderingMode.GPUInstancing;
ref var dummyMat = ref GetDummyMaterial(shaderMode);
dummyMat = DummyMaterial.Create(shaderMode, shader, gpuInstanced);
}
if (GetDummyMaterial(shaderMode) == null)
{
Debug.LogErrorFormat(this, "No dummy material referenced to VLB config for ShaderMode {0}, please try to reset this asset.", shaderMode);
}
RefreshGlobalShaderProperties();
}
static void DeleteAsset<T>(ref T assetObject) where T : UnityEngine.Object
{
if (assetObject)
{
var path = UnityEditor.AssetDatabase.GetAssetPath(assetObject);
AssetDatabase.DeleteAsset(path);
assetObject = null;
}
}
public static void CleanGeneratedAssets()
{
var instance = Instance;
if (instance)
{
DeleteAsset(ref instance._DummyMaterial);
DeleteAsset(ref instance._DummyMaterialHD);
DeleteAsset(ref instance._BeamShader);
DeleteAsset(ref instance._BeamShaderHD);
DeleteAsset(ref instance);
}
}
#endif // UNITY_EDITOR
public void ResetInternalData()
{
noiseTexture3D = Resources.Load("Noise3D_64x64x64") as Texture3D;
dustParticlesPrefab = Resources.Load("DustParticles", typeof(ParticleSystem)) as ParticleSystem;
ditheringNoiseTexture = Resources.Load("VLBDitheringNoise", typeof(Texture2D)) as Texture2D;
jitteringNoiseTexture = Resources.Load("VLBBlueNoise", typeof(Texture2D)) as Texture2D;
#if UNITY_EDITOR
RefreshShaders(RefreshShaderFlags.All);
#endif
}
public ParticleSystem NewVolumetricDustParticles()
{
if (!dustParticlesPrefab)
{
if (Application.isPlaying)
{
Debug.LogError("Failed to instantiate VolumetricDustParticles prefab.");
}
return null;
}
var instance = Instantiate(dustParticlesPrefab);
instance.useAutoRandomSeed = false;
instance.name = "Dust Particles";
instance.gameObject.hideFlags = Consts.Internal.ProceduralObjectsHideFlags;
instance.gameObject.SetActive(true);
return instance;
}
void OnEnable()
{
CreateDefaultRaymarchingQualityPreset(onlyIfNeeded:true);
HandleBackwardCompatibility(pluginVersion, Version.Current);
pluginVersion = Version.Current;
}
void HandleBackwardCompatibility(int serializedVersion, int newVersion)
{
#if UNITY_EDITOR
if (serializedVersion == -1) return; // freshly new spawned config: nothing to do
if (serializedVersion == newVersion) return; // same version: nothing to do
if (serializedVersion < 1830)
{
AutoSelectRenderPipeline();
}
if (serializedVersion < 1950)
{
ResetInternalData(); // retrieve Noise3D texture converted from binary data to texture 3D asset in 1950
EditorUtility.SetDirty(this); // make sure to save this property change
}
if (serializedVersion < 1980)
{
useLightColorTemperature = false; // light temperature support introduced in version 1980
EditorUtility.SetDirty(this); // make sure to save this property change
}
if (serializedVersion < 20000)
{
ResetInternalData(); // retrieve Jittering Noise texture introduced in 20000
EditorUtility.SetDirty(this); // make sure to save this property change
}
if (serializedVersion < 20002)
{
SetScriptingDefineSymbolsForCurrentRenderPipeline(); // new Scripting Define Symbols introduced in 20002
}
if (newVersion > serializedVersion)
{
// Import to keep, we have to regenerate the shader each time the plugin is updated
RefreshShaders(RefreshShaderFlags.All);
}
#endif
}
// Singleton management
static Config ms_Instance = null;
public static Config Instance { get { return GetInstance(true); } }
#if UNITY_EDITOR && VLB_DEBUG
public struct Guard : System.IDisposable {
public Guard(bool assert) {
if (m_IsAccessing && assert) Debug.LogError("Circular loop in Config.Instance");
m_IsAccessing = true;
}
public void Dispose() { m_IsAccessing = false; }
static bool m_IsAccessing = false;
}
#endif // UNITY_EDITOR && VLB_DEBUG
#if UNITY_EDITOR
static bool ms_ShouldInvalidateCache = false;
public Config()
{
ms_ShouldInvalidateCache = true; // new instance detected, force the cache to be refreshed
}
#endif
static Config LoadAssetInternal(string assetName)
{
#if PROFILE_INSTANCE_LOADING
var startTime = EditorApplication.timeSinceStartup;
#endif
var instance = Resources.Load<Config>(assetName);
#if PROFILE_INSTANCE_LOADING
var totalTime = EditorApplication.timeSinceStartup - startTime;
Debug.Log($"Loading {assetName} in {(int)(totalTime*1000)} ms");
#endif
return instance;
}
private static Config GetInstance(bool assertIfNotFound)
{
#if UNITY_EDITOR && VLB_DEBUG
using (new Guard(true))
#endif // UNITY_EDITOR && VLB_DEBUG
{
bool updateInstance = ms_Instance == null;
#if UNITY_EDITOR
updateInstance |= ms_ShouldInvalidateCache; // Force instance reloading when detecting Config asset changes
#endif
if (updateInstance)
{
#if UNITY_EDITOR
if (ms_IsCreatingInstance)
{
Debug.LogError(string.Format("Trying to access Config.Instance while creating it. Breaking before infinite loop."));
return null;
}
#endif // UNITY_EDITOR
// Try to load the instance
{
var newInstance = LoadAssetInternal(kAssetName + PlatformHelper.GetCurrentPlatformSuffix());
if (newInstance == null) newInstance = LoadAssetInternal(kAssetName);
#if UNITY_EDITOR
if (newInstance && newInstance != ms_Instance)
{
ms_Instance = newInstance;
newInstance.RefreshGlobalShaderProperties(); // make sure noise textures are properly loaded as soon as the editor is started
}
ms_ShouldInvalidateCache = false;
#endif // UNITY_EDITOR
ms_Instance = newInstance;
}
if (ms_Instance == null)
{
#if UNITY_EDITOR
ms_IsCreatingInstance = true;
ms_Instance = CreateInstanceAsset();
ms_IsCreatingInstance = false;
ms_Instance.AutoSelectRenderPipeline();
ms_Instance.SetScriptingDefineSymbolsForCurrentRenderPipeline(); // force set define symbols the first time we create the instance
if (Application.isPlaying)
ms_Instance.Reset(); // Reset is not automatically when instancing a ScriptableObject when in playmode
#endif // UNITY_EDITOR
Debug.Assert(!(assertIfNotFound && ms_Instance == null), string.Format("Can't find any resource of type '{0}'. Make sure you have a ScriptableObject of this type in a 'Resources' folder.", typeof(Config)));
}
}
}
return ms_Instance;
}
#if UNITY_EDITOR
static bool ms_IsCreatingInstance = false;
public bool IsCurrentlyUsedInstance() { return Instance == this; }
public bool HasValidAssetName()
{
if (name.IndexOf(kAssetName) != 0)
return false;
return PlatformHelper.IsValidPlatformSuffix(GetAssetSuffix());
}
public string GetAssetSuffix()
{
var fullname = name;
var strToFind = kAssetName;
if (fullname.IndexOf(strToFind) == 0) return fullname.Substring(strToFind.Length);
else return "";
}
static void CreateFolderAndAsset(Object obj, string folderParent, string folderResources, string assetName)
{
if (!AssetDatabase.IsValidFolder(string.Format("{0}/{1}", folderParent, folderResources)))
AssetDatabase.CreateFolder(folderParent, folderResources);
CreateAsset(obj, string.Format("{0}/{1}/{2}", folderParent, folderResources, assetName));
}
public static void CreateAsset(Object obj, string fullPath)
{
AssetDatabase.CreateAsset(obj, fullPath);
AssetDatabase.SaveAssets();
}
static Config CreateInstanceAsset()
{
var asset = CreateInstance<Config>();
Debug.Assert(asset != null);
CreateFolderAndAsset(asset, "Assets", "Resources", kAssetName + kAssetNameExt);
return asset;
}
public string GetDebugInfo()
{
#if UNITY_2021_2_OR_NEWER
string scriptingDefineSymbols = PlayerSettings.GetScriptingDefineSymbols(UnityEditor.Build.NamedBuildTarget.FromBuildTargetGroup(EditorUserBuildSettings.selectedBuildTargetGroup));
#else
string scriptingDefineSymbols = PlayerSettings.GetScriptingDefineSymbolsForGroup(EditorUserBuildSettings.selectedBuildTargetGroup);
#endif
return "Unity version: " + Application.unityVersion
+ "\nVLB version: " + Version.Current
+ "\nPlatform: " + Application.platform
+ "\nOS: " + SystemInfo.operatingSystem
+ "\nShader Level: " + SystemInfo.graphicsShaderLevel
+ "\nGraphics API: " + SystemInfo.graphicsDeviceType
+ "\nUses Reversed ZBuffer: " + SystemInfo.usesReversedZBuffer
+ "\nScripting Define Symbols: " + scriptingDefineSymbols
+ "\nRender Pipeline Asset: " + (UnityEngine.Rendering.GraphicsSettings.defaultRenderPipeline != null ? UnityEngine.Rendering.GraphicsSettings.defaultRenderPipeline.ToString() : "none")
+ "\nRender Pipeline Enum: " + SRPHelper.projectRenderPipeline
+ "\nRender Pipeline Selected: " + renderPipeline
+ "\nRender Pipeline Symbol: " + SRPHelper.renderPipelineScriptingDefineSymbolAsString
+ "\nRendering Mode SD: " + GetActualRenderingMode(ShaderMode.SD)
+ "\nRendering Mode HD: " + GetActualRenderingMode(ShaderMode.HD)
+ "\nRendering Path: " + (Camera.main != null ? Camera.main.actualRenderingPath.ToString() : "no main camera")
+ "\nColor Space: " + QualitySettings.activeColorSpace
;
}
#endif // UNITY_EDITOR
}
}