using System; using UnityEngine; using UnityEngine.Experimental.Rendering; namespace UnityEditor { internal class LightweightHairGUI : LightweightShaderGUI { public enum WorkflowMode { Specular = 0, Metallic } public enum SmoothnessMapChannel { SpecularMetallicAlpha, AlbedoAlpha, } private static class Styles { public static GUIContent twoSidedText = new GUIContent("Two Sided", "Render front and back faces"); public static GUIContent alphaClipText = new GUIContent("Alpha Clip", "Enable Alpha Clip"); public static GUIContent albedoText = new GUIContent("Albedo", "Albedo (RGB) and Transparency (A)"); public static GUIContent clipThresholdText = new GUIContent("Clip Threshold", "Threshold for alpha clip"); public static GUIContent specularMapText = new GUIContent("Specular", "Specular (RGB) and Smoothness (A)"); public static GUIContent metallicMapText = new GUIContent("Metallic", "Metallic (R) and Smoothness (A)"); public static GUIContent smoothnessText = new GUIContent("Smoothness", "Smoothness value"); public static GUIContent smoothnessScaleText = new GUIContent("Smoothness", "Smoothness scale factor"); public static GUIContent smoothnessMapChannelText = new GUIContent("Source", "Smoothness texture and channel"); public static GUIContent highlightsText = new GUIContent("Specular Highlights", "Specular Highlights"); public static GUIContent reflectionsText = new GUIContent("Reflections", "Glossy Reflections"); public static GUIContent normalMapText = new GUIContent("Normal Map", "Normal Map"); public static GUIContent occlusionText = new GUIContent("Occlusion", "Occlusion (G)"); public static GUIContent emissionText = new GUIContent("Color", "Emission (RGB)"); public static GUIContent bumpScaleNotSupported = new GUIContent("Bump scale is not supported on mobile platforms"); public static GUIContent fixNow = new GUIContent("Fix now"); //Hair Properties public static GUIContent hairAreaText = new GUIContent("Hair", ""); public static GUIContent hairShadows = new GUIContent("Shadows", "TODO"); public static GUIContent hairMapText = new GUIContent("Hair Map", ""); public static GUIContent specularShift0Text = new GUIContent("Specular Shift 0", ""); public static GUIContent specularShift1Text = new GUIContent("Specular Shift 1", ""); public static GUIContent specularTint0 = new GUIContent("Specular Tint 0", ""); public static GUIContent specularTint1 = new GUIContent("Specular Tint 1", ""); public static string primaryMapsText = "Main Maps"; public static string secondaryMapsText = "Secondary Maps"; public static string forwardText = "Forward Rendering Options"; public static string workflowModeText = "Workflow Mode"; public static string surfaceType = "Surface Type"; public static string blendingMode = "Blending Mode"; public static string advancedText = "Advanced Options"; public static readonly string[] workflowNames = Enum.GetNames(typeof(WorkflowMode)); public static readonly string[] surfaceNames = Enum.GetNames(typeof(SurfaceType)); public static readonly string[] blendNames = Enum.GetNames(typeof(BlendMode)); public static readonly string[] metallicSmoothnessChannelNames = {"Metallic Alpha", "Albedo Alpha"}; public static readonly string[] specularSmoothnessChannelNames = {"Specular Alpha", "Albedo Alpha"}; } private MaterialProperty workflowMode; private MaterialProperty surfaceType; private MaterialProperty blendMode; private MaterialProperty culling; private MaterialProperty alphaClip; private MaterialProperty albedoColor; private MaterialProperty albedoMap; private MaterialProperty alphaThreshold; private MaterialProperty smoothness; private MaterialProperty smoothnessScale; private MaterialProperty smoothnessMapChannel; private MaterialProperty metallic; private MaterialProperty specColor; private MaterialProperty metallicGlossMap; private MaterialProperty specGlossMap; private MaterialProperty highlights; private MaterialProperty reflections; private MaterialProperty bumpScale; private MaterialProperty bumpMap; private MaterialProperty occlusionStrength; private MaterialProperty occlusionMap; private MaterialProperty emissionColorForRendering; private MaterialProperty emissionMap; //Hair Properties private MaterialProperty recieveShadows; private MaterialProperty hairMap; private MaterialProperty specularShift0; private MaterialProperty specularShift1; private MaterialProperty specularTint0; private MaterialProperty specularTint1; public override void FindProperties(MaterialProperty[] properties) { workflowMode = FindProperty("_WorkflowMode", properties); surfaceType = FindProperty("_Surface", properties); blendMode = FindProperty("_Blend", properties); culling = FindProperty("_Cull", properties); alphaClip = FindProperty("_AlphaClip", properties); albedoColor = FindProperty("_Color", properties); albedoMap = FindProperty("_MainTex", properties); alphaThreshold = FindProperty("_Cutoff", properties); smoothness = FindProperty("_Glossiness", properties); smoothnessScale = FindProperty("_GlossMapScale", properties, false); smoothnessMapChannel = FindProperty("_SmoothnessTextureChannel", properties, false); metallic = FindProperty("_Metallic", properties); specColor = FindProperty("_SpecColor", properties); metallicGlossMap = FindProperty("_MetallicGlossMap", properties); specGlossMap = FindProperty("_SpecGlossMap", properties); highlights = FindProperty("_SpecularHighlights", properties); reflections = FindProperty("_GlossyReflections", properties); bumpScale = FindProperty("_BumpScale", properties); bumpMap = FindProperty("_BumpMap", properties); occlusionStrength = FindProperty("_OcclusionStrength", properties); occlusionMap = FindProperty("_OcclusionMap", properties); emissionColorForRendering = FindProperty("_EmissionColor", properties); emissionMap = FindProperty("_EmissionMap", properties); //Hair Properties recieveShadows = FindProperty("_RecieveShadows", properties); hairMap = FindProperty("_HairMap", properties); specularShift0 = FindProperty("_SpecularShift0", properties); specularShift1 = FindProperty("_SpecularShift1", properties); specularTint0 = FindProperty("_SpecularTint0", properties); specularTint1 = FindProperty("_SpecularTint1", properties); } public override void MaterialChanged(Material material) { material.shaderKeywords = null; SetupMaterialBlendMode(material); SetMaterialKeywords(material); } public override void ShaderPropertiesGUI(Material material) { // Use default labelWidth EditorGUIUtility.labelWidth = 0f; // Detect any changes to the material EditorGUI.BeginChangeCheck(); { DoPopup(Styles.workflowModeText, workflowMode, Styles.workflowNames); DoPopup(Styles.surfaceType, surfaceType, Styles.surfaceNames); if ((SurfaceType)material.GetFloat("_Surface") == SurfaceType.Transparent) DoPopup(Styles.blendingMode, blendMode, Styles.blendNames); EditorGUI.BeginChangeCheck(); bool twoSidedEnabled = EditorGUILayout.Toggle(Styles.twoSidedText, culling.floatValue == 0); if (EditorGUI.EndChangeCheck()) culling.floatValue = twoSidedEnabled ? 0 : 2; EditorGUI.BeginChangeCheck(); bool alphaClipEnabled = EditorGUILayout.Toggle(Styles.alphaClipText, alphaClip.floatValue == 1); if (EditorGUI.EndChangeCheck()) alphaClip.floatValue = alphaClipEnabled ? 1 : 0; // Primary properties GUILayout.Label(Styles.primaryMapsText, EditorStyles.boldLabel); DoAlbedoArea(material); DoMetallicSpecularArea(); DoNormalArea(); m_MaterialEditor.TexturePropertySingleLine(Styles.occlusionText, occlusionMap, occlusionMap.textureValue != null ? occlusionStrength : null); DoEmissionArea(material); EditorGUI.BeginChangeCheck(); m_MaterialEditor.TextureScaleOffsetProperty(albedoMap); if (EditorGUI.EndChangeCheck()) emissionMap.textureScaleAndOffset = albedoMap.textureScaleAndOffset; // Apply the main texture scale and offset to the emission texture as well, for Enlighten's sake EditorGUILayout.Space(); m_MaterialEditor.ShaderProperty(highlights, Styles.highlightsText); m_MaterialEditor.ShaderProperty(reflections, Styles.reflectionsText); EditorGUILayout.Space(); DoHairArea(); } if (EditorGUI.EndChangeCheck()) { foreach (var obj in blendMode.targets) MaterialChanged((Material)obj); } EditorGUILayout.Space(); // NB renderqueue editor is not shown on purpose: we want to override it based on blend mode GUILayout.Label(Styles.advancedText, EditorStyles.boldLabel); m_MaterialEditor.EnableInstancingField(); m_MaterialEditor.DoubleSidedGIField(); } public override void AssignNewShaderToMaterial(Material material, Shader oldShader, Shader newShader) { // _Emission property is lost after assigning Standard shader to the material // thus transfer it before assigning the new shader if (material.HasProperty("_Emission")) { material.SetColor("_EmissionColor", material.GetColor("_Emission")); } base.AssignNewShaderToMaterial(material, oldShader, newShader); if (oldShader == null || !oldShader.name.Contains("Legacy Shaders/")) { SetupMaterialBlendMode(material); return; } SurfaceType surfaceType = SurfaceType.Opaque; BlendMode blendMode = BlendMode.Alpha; if (oldShader.name.Contains("/Transparent/Cutout/")) { surfaceType = SurfaceType.Opaque; material.SetFloat("_AlphaClip", 1); } else if (oldShader.name.Contains("/Transparent/")) { // NOTE: legacy shaders did not provide physically based transparency // therefore Fade mode surfaceType = SurfaceType.Transparent; blendMode = BlendMode.Alpha; } material.SetFloat("_Surface", (float)surfaceType); material.SetFloat("_Blend", (float)blendMode); if (oldShader.name.Equals("Standard (Specular setup)")) { material.SetFloat("_WorkflowMode", (float)WorkflowMode.Specular); Texture texture = material.GetTexture("_SpecGlossMap"); if (texture != null) material.SetTexture("_MetallicSpecGlossMap", texture); } else { material.SetFloat("_WorkflowMode", (float)WorkflowMode.Metallic); Texture texture = material.GetTexture("_MetallicGlossMap"); if (texture != null) material.SetTexture("_MetallicSpecGlossMap", texture); } MaterialChanged(material); } void DoAlbedoArea(Material material) { m_MaterialEditor.TexturePropertySingleLine(Styles.albedoText, albedoMap, albedoColor); if (material.GetFloat("_AlphaClip") == 1) { m_MaterialEditor.ShaderProperty(alphaThreshold, Styles.clipThresholdText.text, MaterialEditor.kMiniTextureFieldLabelIndentLevel + 1); } } void DoNormalArea() { m_MaterialEditor.TexturePropertySingleLine(Styles.normalMapText, bumpMap, bumpMap.textureValue != null ? bumpScale : null); if (bumpScale.floatValue != 1 && UnityEditorInternal.InternalEditorUtility.IsMobilePlatform(EditorUserBuildSettings.activeBuildTarget)) if (m_MaterialEditor.HelpBoxWithButton(Styles.bumpScaleNotSupported, Styles.fixNow)) bumpScale.floatValue = 1; } void DoEmissionArea(Material material) { // Emission for GI? if (m_MaterialEditor.EmissionEnabledProperty()) { bool hadEmissionTexture = emissionMap.textureValue != null; // Texture and HDR color controls m_MaterialEditor.TexturePropertyWithHDRColor(Styles.emissionText, emissionMap, emissionColorForRendering, false); // If texture was assigned and color was black set color to white float brightness = emissionColorForRendering.colorValue.maxColorComponent; if (emissionMap.textureValue != null && !hadEmissionTexture && brightness <= 0f) emissionColorForRendering.colorValue = Color.white; // LW does not support RealtimeEmissive. We set it to bake emissive and handle the emissive is black right. material.globalIlluminationFlags = MaterialGlobalIlluminationFlags.BakedEmissive; if (brightness <= 0f) material.globalIlluminationFlags |= MaterialGlobalIlluminationFlags.EmissiveIsBlack; } } void DoMetallicSpecularArea() { string[] metallicSpecSmoothnessChannelName; bool hasGlossMap = false; if ((WorkflowMode)workflowMode.floatValue == WorkflowMode.Metallic) { hasGlossMap = metallicGlossMap.textureValue != null; metallicSpecSmoothnessChannelName = Styles.metallicSmoothnessChannelNames; m_MaterialEditor.TexturePropertySingleLine(Styles.metallicMapText, metallicGlossMap, hasGlossMap ? null : metallic); } else { hasGlossMap = specGlossMap.textureValue != null; metallicSpecSmoothnessChannelName = Styles.specularSmoothnessChannelNames; m_MaterialEditor.TexturePropertySingleLine(Styles.specularMapText, specGlossMap, hasGlossMap ? null : specColor); } bool showSmoothnessScale = hasGlossMap; if (smoothnessMapChannel != null) { int smoothnessChannel = (int)smoothnessMapChannel.floatValue; if (smoothnessChannel == (int)SmoothnessMapChannel.AlbedoAlpha) showSmoothnessScale = true; } int indentation = 2; // align with labels of texture properties m_MaterialEditor.ShaderProperty(showSmoothnessScale ? smoothnessScale : smoothness, showSmoothnessScale ? Styles.smoothnessScaleText : Styles.smoothnessText, indentation); int prevIndentLevel = EditorGUI.indentLevel; EditorGUI.indentLevel = 3; if (smoothnessMapChannel != null) DoPopup(Styles.smoothnessMapChannelText.text, smoothnessMapChannel, metallicSpecSmoothnessChannelName); EditorGUI.indentLevel = prevIndentLevel; } void DoHairArea() { GUILayout.Label(Styles.hairAreaText, EditorStyles.boldLabel); EditorGUI.BeginChangeCheck(); bool shadows = EditorGUILayout.Toggle(Styles.hairShadows, recieveShadows.floatValue == 1); if(EditorGUI.EndChangeCheck()) recieveShadows.floatValue = shadows ? 1 : 0; m_MaterialEditor.TexturePropertySingleLine(Styles.hairMapText, hairMap); m_MaterialEditor.ShaderProperty(specularShift0, Styles.specularShift0Text); m_MaterialEditor.ShaderProperty(specularShift1, Styles.specularShift1Text); m_MaterialEditor.ShaderProperty(specularTint0, Styles.specularTint0); m_MaterialEditor.ShaderProperty(specularTint1, Styles.specularTint1); } static SmoothnessMapChannel GetSmoothnessMapChannel(Material material) { int ch = (int)material.GetFloat("_SmoothnessTextureChannel"); if (ch == (int)SmoothnessMapChannel.AlbedoAlpha) return SmoothnessMapChannel.AlbedoAlpha; return SmoothnessMapChannel.SpecularMetallicAlpha; } static void SetMaterialKeywords(Material material) { // Note: keywords must be based on Material value not on MaterialProperty due to multi-edit & material animation // (MaterialProperty value might come from renderer material property block) bool isSpecularWorkFlow = (WorkflowMode)material.GetFloat("_WorkflowMode") == WorkflowMode.Specular; bool hasGlossMap = false; if (isSpecularWorkFlow) hasGlossMap = material.GetTexture("_SpecGlossMap"); else hasGlossMap = material.GetTexture("_MetallicGlossMap"); CoreUtils.SetKeyword(material, "_SPECULAR_SETUP", isSpecularWorkFlow); CoreUtils.SetKeyword(material, "_METALLICSPECGLOSSMAP", hasGlossMap); CoreUtils.SetKeyword(material, "_SPECGLOSSMAP", hasGlossMap && isSpecularWorkFlow); CoreUtils.SetKeyword(material, "_METALLICGLOSSMAP", hasGlossMap && !isSpecularWorkFlow); CoreUtils.SetKeyword(material, "_NORMALMAP", material.GetTexture("_BumpMap")); CoreUtils.SetKeyword(material, "_SPECULARHIGHLIGHTS_OFF", material.GetFloat("_SpecularHighlights") == 0.0f); CoreUtils.SetKeyword(material, "_GLOSSYREFLECTIONS_OFF", material.GetFloat("_GlossyReflections") == 0.0f); CoreUtils.SetKeyword(material, "_OCCLUSIONMAP", material.GetTexture("_OcclusionMap")); CoreUtils.SetKeyword(material, "_PARALLAXMAP", material.GetTexture("_ParallaxMap")); CoreUtils.SetKeyword(material, "_DETAIL_MULX2", material.GetTexture("_DetailAlbedoMap") || material.GetTexture("_DetailNormalMap")); //Hair Keywords CoreUtils.SetKeyword(material, "_HAIR_RECEIVE_SHADOWS", material.GetFloat("_RecieveShadows") == 1); CoreUtils.SetKeyword(material, "_TWO_SIDED", material.GetFloat("_Cull") == 0); // A material's GI flag internally keeps track of whether emission is enabled at all, it's enabled but has no effect // or is enabled and may be modified at runtime. This state depends on the values of the current flag and emissive color. // The fixup routine makes sure that the material is in the correct state if/when changes are made to the mode or color. MaterialEditor.FixupEmissiveFlag(material); bool shouldEmissionBeEnabled = (material.globalIlluminationFlags & MaterialGlobalIlluminationFlags.EmissiveIsBlack) == 0; CoreUtils.SetKeyword(material, "_EMISSION", shouldEmissionBeEnabled); if (material.HasProperty("_SmoothnessTextureChannel")) { CoreUtils.SetKeyword(material, "_SMOOTHNESS_TEXTURE_ALBEDO_CHANNEL_A", GetSmoothnessMapChannel(material) == SmoothnessMapChannel.AlbedoAlpha); } } } }