// Amplify Shader Editor - Visual Shader Editing Tool // Copyright (c) Amplify Creations, Lda using UnityEngine; using UnityEditor; using System; namespace AmplifyShaderEditor { [Serializable] [NodeAttributes( "Dither", "Camera And Screen", "Generates a dithering pattern" )] public sealed class DitheringNode : ParentNode { private const string InputTypeStr = "Pattern"; private const string CustomScreenPosStr = "screenPosition"; private string m_functionHeader = "Dither4x4Bayer( {0}, {1} )"; private string m_functionBody = string.Empty; [SerializeField] private int m_selectedPatternInt = 0; [SerializeField] private bool m_customScreenPos = false; private readonly string[] PatternsFuncStr = { "4x4Bayer", "8x8Bayer", "NoiseTex" }; private readonly string[] PatternsStr = { "4x4 Bayer", "8x8 Bayer", "Noise Texture" }; private UpperLeftWidgetHelper m_upperLeftWidget = new UpperLeftWidgetHelper(); private InputPort m_texPort; private InputPort m_ssPort; protected override void CommonInit( int uniqueId ) { base.CommonInit( uniqueId ); AddInputPort( WirePortDataType.FLOAT, false, Constants.EmptyPortValue ); AddInputPort( WirePortDataType.SAMPLER2D, false, "Pattern"); m_inputPorts[ 1 ].CreatePortRestrictions( WirePortDataType.SAMPLER2D ); m_texPort = m_inputPorts[ 1 ]; AddInputPort( WirePortDataType.FLOAT4, false, "Screen Position" ); AddInputPort( WirePortDataType.SAMPLERSTATE, false, "SS" ); m_inputPorts[ 3 ].CreatePortRestrictions( WirePortDataType.SAMPLERSTATE ); m_ssPort = m_inputPorts[ 3 ]; AddOutputPort( WirePortDataType.FLOAT, Constants.EmptyPortValue ); m_textLabelWidth = 110; m_autoWrapProperties = true; m_hasLeftDropdown = true; SetAdditonalTitleText( string.Format( Constants.SubTitleTypeFormatStr, PatternsStr[ m_selectedPatternInt ] ) ); UpdatePorts(); } public override void Destroy() { base.Destroy(); m_upperLeftWidget = null; m_texPort = null; m_ssPort = null; } public override void AfterCommonInit() { base.AfterCommonInit(); if( PaddingTitleLeft == 0 ) { PaddingTitleLeft = Constants.PropertyPickerWidth + Constants.IconsLeftRightMargin; if( PaddingTitleRight == 0 ) PaddingTitleRight = Constants.PropertyPickerWidth + Constants.IconsLeftRightMargin; } } public override void OnConnectedOutputNodeChanges( int outputPortId, int otherNodeId, int otherPortId, string name, WirePortDataType type ) { base.OnConnectedOutputNodeChanges( outputPortId, otherNodeId, otherPortId, name, type ); if( !m_texPort.CheckValidType( type ) ) { m_texPort.FullDeleteConnections(); UIUtils.ShowMessage( UniqueId, "Dithering node only accepts SAMPLER2D input type.\nTexture Object connected changed to " + type + ", connection was lost, please review and update accordingly.", MessageSeverity.Warning ); } } public override void Draw( DrawInfo drawInfo ) { base.Draw( drawInfo ); EditorGUI.BeginChangeCheck(); m_selectedPatternInt = m_upperLeftWidget.DrawWidget( this, m_selectedPatternInt, PatternsStr ); if( EditorGUI.EndChangeCheck() ) { UpdatePorts(); } } public override void DrawProperties() { base.DrawProperties(); EditorGUI.BeginChangeCheck(); m_selectedPatternInt = EditorGUILayoutPopup( "Pattern", m_selectedPatternInt, PatternsStr ); if ( EditorGUI.EndChangeCheck() ) { UpdatePorts(); } EditorGUI.BeginChangeCheck(); m_customScreenPos = EditorGUILayoutToggle( "Screen Position", m_customScreenPos ); if( EditorGUI.EndChangeCheck() ) { UpdatePorts(); } } private void UpdatePorts() { m_texPort.Visible = ( m_selectedPatternInt == 2 ); m_ssPort.Visible = ( m_selectedPatternInt == 2 ); m_inputPorts[ 2 ].Visible = m_customScreenPos; m_sizeIsDirty = true; } private void GeneratePattern( ref MasterNodeDataCollector dataCollector ) { SetAdditonalTitleText( string.Format( Constants.SubTitleTypeFormatStr, PatternsStr[ m_selectedPatternInt ] ) ); switch ( m_selectedPatternInt ) { default: case 0: { m_functionBody = string.Empty; m_functionHeader = "Dither" + PatternsFuncStr[ m_selectedPatternInt ] + "( {0}, {1} )"; IOUtils.AddFunctionHeader( ref m_functionBody, "inline float Dither" + PatternsFuncStr[ m_selectedPatternInt ] + "( int x, int y )" ); IOUtils.AddFunctionLine( ref m_functionBody, "const float dither[ 16 ] = {" ); IOUtils.AddFunctionLine( ref m_functionBody, " 1, 9, 3, 11," ); IOUtils.AddFunctionLine( ref m_functionBody, " 13, 5, 15, 7," ); IOUtils.AddFunctionLine( ref m_functionBody, " 4, 12, 2, 10," ); IOUtils.AddFunctionLine( ref m_functionBody, " 16, 8, 14, 6 };" ); IOUtils.AddFunctionLine( ref m_functionBody, "int r = y * 4 + x;" ); IOUtils.AddFunctionLine( ref m_functionBody, "return dither[r] / 16; // same # of instructions as pre-dividing due to compiler magic" ); IOUtils.CloseFunctionBody( ref m_functionBody ); } break; case 1: { m_functionBody = string.Empty; m_functionHeader = "Dither" + PatternsFuncStr[ m_selectedPatternInt ] + "( {0}, {1} )"; IOUtils.AddFunctionHeader( ref m_functionBody, "inline float Dither" + PatternsFuncStr[ m_selectedPatternInt ] + "( int x, int y )" ); IOUtils.AddFunctionLine( ref m_functionBody, "const float dither[ 64 ] = {" ); IOUtils.AddFunctionLine( ref m_functionBody, " 1, 49, 13, 61, 4, 52, 16, 64," ); IOUtils.AddFunctionLine( ref m_functionBody, " 33, 17, 45, 29, 36, 20, 48, 32," ); IOUtils.AddFunctionLine( ref m_functionBody, " 9, 57, 5, 53, 12, 60, 8, 56," ); IOUtils.AddFunctionLine( ref m_functionBody, " 41, 25, 37, 21, 44, 28, 40, 24," ); IOUtils.AddFunctionLine( ref m_functionBody, " 3, 51, 15, 63, 2, 50, 14, 62," ); IOUtils.AddFunctionLine( ref m_functionBody, " 35, 19, 47, 31, 34, 18, 46, 30," ); IOUtils.AddFunctionLine( ref m_functionBody, " 11, 59, 7, 55, 10, 58, 6, 54," ); IOUtils.AddFunctionLine( ref m_functionBody, " 43, 27, 39, 23, 42, 26, 38, 22};" ); IOUtils.AddFunctionLine( ref m_functionBody, "int r = y * 8 + x;" ); IOUtils.AddFunctionLine( ref m_functionBody, "return dither[r] / 64; // same # of instructions as pre-dividing due to compiler magic" ); IOUtils.CloseFunctionBody( ref m_functionBody ); } break; case 2: { ParentGraph outsideGraph = UIUtils.CurrentWindow.OutsideGraph; m_functionBody = string.Empty; m_functionHeader = "Dither" + PatternsFuncStr[ m_selectedPatternInt ] + "({0}, {1}, {2})"; IOUtils.AddFunctionHeader( ref m_functionBody, "inline float Dither" + PatternsFuncStr[ m_selectedPatternInt ] + "( float4 screenPos, " + GeneratorUtils.GetPropertyDeclaraction( "noiseTexture", TextureType.Texture2D, ", " ) + GeneratorUtils.GetSamplerDeclaraction( "samplernoiseTexture", TextureType.Texture2D, ", " ) + "float4 noiseTexelSize )" ); string samplingCall = GeneratorUtils.GenerateSamplingCall( ref dataCollector, WirePortDataType.SAMPLER2D, "noiseTexture", "samplernoiseTexture", "screenPos.xy * _ScreenParams.xy * noiseTexelSize.xy", MipType.MipLevel, "0" ); IOUtils.AddFunctionLine( ref m_functionBody, "float dither = "+ samplingCall + ".g;" ); IOUtils.AddFunctionLine( ref m_functionBody, "float ditherRate = noiseTexelSize.x * noiseTexelSize.y;" ); IOUtils.AddFunctionLine( ref m_functionBody, "dither = ( 1 - ditherRate ) * dither + ditherRate;" ); IOUtils.AddFunctionLine( ref m_functionBody, "return dither;" ); IOUtils.CloseFunctionBody( ref m_functionBody ); } break; } } public override void PropagateNodeData( NodeData nodeData, ref MasterNodeDataCollector dataCollector ) { base.PropagateNodeData( nodeData, ref dataCollector ); dataCollector.UsingCustomScreenPos = true; } public override string GenerateShaderForOutput( int outputId, ref MasterNodeDataCollector dataCollector, bool ignoreLocalVar ) { if ( m_outputPorts[ 0 ].IsLocalValue( dataCollector.PortCategory ) ) return m_outputPorts[ 0 ].LocalValue( dataCollector.PortCategory ); GeneratePattern( ref dataCollector ); if( !( dataCollector.IsTemplate && dataCollector.IsSRP ) ) dataCollector.AddToIncludes( UniqueId, Constants.UnityShaderVariables ); string varName = string.Empty; bool isFragment = dataCollector.IsFragmentCategory; if( m_customScreenPos && m_inputPorts[ 2 ].IsConnected ) { varName = "ditherCustomScreenPos" + OutputId; string customScreenPosVal = m_inputPorts[ 2 ].GeneratePortInstructions( ref dataCollector ); dataCollector.AddLocalVariable( UniqueId, CurrentPrecisionType, WirePortDataType.FLOAT4, varName, customScreenPosVal ); } else { if( dataCollector.TesselationActive && isFragment ) { varName = GeneratorUtils.GenerateClipPositionOnFrag( ref dataCollector, UniqueId, CurrentPrecisionType ); } else { if( dataCollector.IsTemplate ) { varName = dataCollector.TemplateDataCollectorInstance.GetScreenPosNormalized( CurrentPrecisionType ); } else { varName = GeneratorUtils.GenerateScreenPositionNormalized( ref dataCollector, UniqueId, CurrentPrecisionType, !dataCollector.UsingCustomScreenPos ); } } } string surfInstruction = varName + ".xy * _ScreenParams.xy"; m_showErrorMessage = false; string functionResult = ""; string noiseTex = string.Empty; switch ( m_selectedPatternInt ) { default: case 0: dataCollector.AddLocalVariable( UniqueId, CurrentPrecisionType, WirePortDataType.FLOAT2, "clipScreen" + OutputId, surfInstruction ); functionResult = dataCollector.AddFunctions( m_functionHeader, m_functionBody, "fmod(" + "clipScreen" + OutputId + ".x, 4)", "fmod(" + "clipScreen" + OutputId + ".y, 4)" ); break; case 1: dataCollector.AddLocalVariable( UniqueId, CurrentPrecisionType, WirePortDataType.FLOAT2, "clipScreen" + OutputId, surfInstruction ); functionResult = dataCollector.AddFunctions( m_functionHeader, m_functionBody, "fmod(" + "clipScreen" + OutputId + ".x, 8)", "fmod(" + "clipScreen" + OutputId + ".y, 8)" ); break; case 2: { if( !m_texPort.IsConnected ) { m_showErrorMessage = true; m_errorMessageTypeIsError = NodeMessageType.Warning; m_errorMessageTooltip = "Please connect a texture object to the Pattern input port to generate a proper dithered pattern"; return "0"; } else { ParentGraph outsideGraph = UIUtils.CurrentWindow.OutsideGraph; noiseTex = m_texPort.GeneratePortInstructions( ref dataCollector ); //GeneratePattern( ref dataCollector ); dataCollector.AddToUniforms( UniqueId, "float4 " + noiseTex + "_TexelSize;", dataCollector.IsSRP ); #if UNITY_2018_1_OR_NEWER if( outsideGraph.SamplingMacros ) #else if( outsideGraph.SamplingMacros && !outsideGraph.IsStandardSurface ) #endif { string sampler = string.Empty; if( m_ssPort.IsConnected ) { sampler = m_ssPort.GeneratePortInstructions( ref dataCollector ); } else { sampler = GeneratorUtils.GenerateSamplerState( ref dataCollector, UniqueId, noiseTex ); } //if( outsideGraph.IsSRP ) // functionResult = dataCollector.AddFunctions( m_functionHeader, m_functionBody, varName, noiseTex + ", " + sampler, noiseTex + "_TexelSize" ); //else functionResult = dataCollector.AddFunctions( m_functionHeader, m_functionBody, varName, noiseTex + ", " + sampler, noiseTex + "_TexelSize" ); } else { functionResult = dataCollector.AddFunctions( m_functionHeader, m_functionBody, varName, noiseTex, noiseTex + "_TexelSize" ); } } } break; } dataCollector.AddLocalVariable( UniqueId, CurrentPrecisionType, WirePortDataType.FLOAT, "dither" + OutputId, functionResult ); if( m_inputPorts[ 0 ].IsConnected ) { string driver = m_inputPorts[ 0 ].GeneratePortInstructions( ref dataCollector ); dataCollector.AddLocalVariable( UniqueId, "dither" + OutputId+" = step( dither"+ OutputId + ", "+ driver + " );" ); } //RegisterLocalVariable( 0, functionResult, ref dataCollector, "dither" + OutputId ); m_outputPorts[ 0 ].SetLocalValue( "dither" + OutputId, dataCollector.PortCategory ); return m_outputPorts[ 0 ].LocalValue( dataCollector.PortCategory ); } public override void ReadFromString( ref string[] nodeParams ) { base.ReadFromString( ref nodeParams ); m_selectedPatternInt = Convert.ToInt32( GetCurrentParam( ref nodeParams ) ); if( UIUtils.CurrentShaderVersion() > 15404 ) { m_customScreenPos = Convert.ToBoolean( GetCurrentParam( ref nodeParams ) ); } UpdatePorts(); } public override void WriteToString( ref string nodeInfo, ref string connectionsInfo ) { base.WriteToString( ref nodeInfo, ref connectionsInfo ); IOUtils.AddFieldValueToString( ref nodeInfo, m_selectedPatternInt ); IOUtils.AddFieldValueToString( ref nodeInfo, m_customScreenPos ); } } }