首页 > 其他分享 >【SHADER系列】(一)UGUI毛玻璃效果

【SHADER系列】(一)UGUI毛玻璃效果

时间:2022-11-23 16:00:45浏览次数:75  
标签:GrabTexture sum vertex SHADER GrabPixel float4 UGUI uvgrab 毛玻璃

旧版内建渲染管线下的毛玻璃模糊效果:

实现原理很简单,就是利用Unity的GrabPass来抓取当前屏幕渲染的图像,然后进行一个Blur高斯模糊算法。为了效果更好,可以分别进行横竖模糊,只不过会变成4个DC而已。

使用也很简单,新建一个材质,赋予这个着色器,并给UGUI的Image设置该材质即可。通过调节Color的黑白值来设置模糊的程度。

GrabPass {}
Pass1 { 横向模糊 }
GrabPass {}
Pass2 { 纵向模糊 }
Shader "Custom/BackBlur"
{
    Properties
    {
        [PerRendererData] _MainTex("Sprite Texture", 2D) = "white" {}
        _Color("Main Color", Color) = (1,1,1,1)
        _Size("Size", Range(0, 20)) = 1
    }
    Category{

        // We must be transparent, so other objects are drawn before this one.  
        Tags {
            "Queue" = "Transparent"
            "IgnoreProjector" = "True"
            "RenderType" = "Transparent"
            "PreviewType" = "Plane"
            "CanUseSpriteAtlas" = "True"
        }


        SubShader {

            ZWrite Off
            // Horizontal blur  
            GrabPass {
                Tags { "LightMode" = "Always" }
            }
            Pass {
                Tags { "LightMode" = "Always" }

                Name "BackBlurHor"
                CGPROGRAM
                #pragma vertex vert  
                #pragma fragment frag  
                #pragma fragmentoption ARB_precision_hint_fastest  
                #include "UnityCG.cginc"  

                struct appdata_t {
                    float4 vertex : POSITION;
                    float2 texcoord : TEXCOORD0;
                    float4 color    : COLOR;
                };

                struct v2f {
                    float4 vertex : POSITION;
                    float4 uvgrab : TEXCOORD0;
                    float4 color    : COLOR;
                };

                v2f vert(appdata_t v) {
                    v2f o;
                    o.vertex = UnityObjectToClipPos(v.vertex);
                    #if UNITY_UV_STARTS_AT_TOP  
                    float scale = -1.0;
                    #else  
                    float scale = 1.0;
                    #endif  
                    o.uvgrab.xy = (float2(o.vertex.x, o.vertex.y*scale) + o.vertex.w) * 0.5;
                    o.uvgrab.zw = o.vertex.zw;

                    o.color = v.color;
                    return o;
                }

                sampler2D _GrabTexture;
                float4 _GrabTexture_TexelSize;
                sampler2D _MainTex;
                float4 _MainTex_TexelSize;
                float _Size;
                uniform float4 _Color;

                half4 GrabPixel(v2f i, float weight, float kernelx) {
                    if (i.uvgrab.x == 0 && i.uvgrab.y == 0) {
                        kernelx = 0;
                    }
                    return tex2Dproj(_GrabTexture, UNITY_PROJ_COORD(float4(i.uvgrab.x + _GrabTexture_TexelSize.x*kernelx*_Size, i.uvgrab.y, i.uvgrab.z, i.uvgrab.w))) * weight;
                }
                half4 frag(v2f i) : COLOR {
                    half4 sum = half4(0,0,0,0);
                    // #define GRABPIXEL(weight, kernelx) tex2Dproj(_GrabTexture, UNITY_PROJ_COORD(float4(i.uvgrab.x + _GrabTexture_TexelSize.x * kernelx*_Size, i.uvgrab.y, i.uvgrab.z, i.uvgrab.w))) * weight

                    sum += GrabPixel(i, 0.05, -4.0);
                    sum += GrabPixel(i, 0.09, -3.0);
                    sum += GrabPixel(i, 0.12, -2.0);
                    sum += GrabPixel(i, 0.15, -1.0);
                    sum += GrabPixel(i, 0.18,  0.0);
                    sum += GrabPixel(i, 0.15, +1.0);
                    sum += GrabPixel(i, 0.12, +2.0);
                    sum += GrabPixel(i, 0.09, +3.0);
                    sum += GrabPixel(i, 0.05, +4.0);

                    float4 col5 = tex2Dproj(_GrabTexture, UNITY_PROJ_COORD(i.uvgrab));
                    float decayFactor = 1.0f;
                    if (i.uvgrab.x == 0 && i.uvgrab.y == 0) {
                        decayFactor = 0;
                    }
                    sum = lerp(col5, sum, decayFactor) * i.color * _Color;

                    return sum;
                }
                ENDCG
            }
            // Vertical blur  
            GrabPass {
                Tags { "LightMode" = "Always" }
            }
            Pass {
                Tags { "LightMode" = "Always" }

                Name "BackBlurVer"
                CGPROGRAM
                #pragma vertex vert  
                #pragma fragment frag  
                #pragma fragmentoption ARB_precision_hint_fastest  
                #include "UnityCG.cginc"  

                struct appdata_t {
                    float4 vertex : POSITION;
                    float2 texcoord: TEXCOORD0;
                    float4 color    : COLOR;
                };

                struct v2f {
                    float4 vertex : POSITION;
                    float4 uvgrab : TEXCOORD0;
                    float4 color    : COLOR;
                };

                v2f vert(appdata_t v) {
                    v2f o;
                    o.vertex = UnityObjectToClipPos(v.vertex);
                    #if UNITY_UV_STARTS_AT_TOP  
                    float scale = -1.0;
                    #else  
                    float scale = 1.0;
                    #endif  
                    o.uvgrab.xy = (float2(o.vertex.x, o.vertex.y*scale) + o.vertex.w) * 0.5;
                    o.uvgrab.zw = o.vertex.zw;

                    o.color = v.color;
                    return o;
                }

                sampler2D _GrabTexture;
                float4 _GrabTexture_TexelSize;
                float _Size;
                uniform float4 _Color;

                half4 GrabPixel(v2f i, float weight, float kernely) {
                    if (i.uvgrab.x == 0 && i.uvgrab.y == 0) {
                        kernely = 0;
                    }
                    return tex2Dproj(_GrabTexture, UNITY_PROJ_COORD(float4(i.uvgrab.x, i.uvgrab.y + _GrabTexture_TexelSize.y*kernely*_Size, i.uvgrab.z, i.uvgrab.w))) * weight;
                }

                half4 frag(v2f i) : COLOR {
                    half4 sum = half4(0,0,0,0);
                    // #define GRABPIXEL(weight,kernely) tex2Dproj( _GrabTexture, UNITY_PROJ_COORD(float4(i.uvgrab.x, i.uvgrab.y + _GrabTexture_TexelSize.y * kernely*_Size, i.uvgrab.z, i.uvgrab.w))) * weight  

                    sum += GrabPixel(i, 0.05, -4.0);
                    sum += GrabPixel(i, 0.09, -3.0);
                    sum += GrabPixel(i, 0.12, -2.0);
                    sum += GrabPixel(i, 0.15, -1.0);
                    sum += GrabPixel(i, 0.18,  0.0);
                    sum += GrabPixel(i, 0.15, +1.0);
                    sum += GrabPixel(i, 0.12, +2.0);
                    sum += GrabPixel(i, 0.09, +3.0);
                    sum += GrabPixel(i, 0.05, +4.0);

                    float4 col5 = tex2Dproj(_GrabTexture, UNITY_PROJ_COORD(i.uvgrab));
                    float decayFactor = 1.0f;
                    if (i.uvgrab.x == 0 && i.uvgrab.y == 0) {
                        decayFactor = 0;
                    }
                    sum = lerp(col5, sum, decayFactor) * i.color * _Color;

                    return sum;
                }
                ENDCG
            }
        }
    }
}

 

 

URP渲染管线下的毛玻璃:

URP默认不支持多PASS渲染,也不支持GrabPass,所以以上方式失效。

至于怎么在URP下实现类似GrabPass的,可以参考:Unity urp环境下实现默认管线的GrabPass功能 - 知乎 (zhihu.com), 【Unity3D】URP下的GrabPass方案 - 简书 (jianshu.com)

实现方案参考:Unity URP UI背景模糊效果 - 哔哩哔哩 (bilibili.com)

 

标签:GrabTexture,sum,vertex,SHADER,GrabPixel,float4,UGUI,uvgrab,毛玻璃
From: https://www.cnblogs.com/jeason1997/p/16918580.html

相关文章

  • ShaderGraph普通水面
    效果:正常水面波动,水下折射。主要思路是在法线贴图基础上修改uv插值形成波动效果,根据噪音得到场景颜色与场景深度节点等运算进行lerp混合得到折射效果      ......
  • ShaderGraph溶解
    思路首先给片元着色器添加alphaclip,片元着色器会出现一个AlphaClipThreshold节点,这个节点能够设置透明度剪裁阈值,要添加一个噪声节点,SimpleNoise输出到Aplha,现在......
  • Unity Shader 7 CG 语法
    1Cg语法基础如C++、C#和Java等高级语言一样,Cg语言也有自己的数据类型和关键字。掌握和理解这些关键字是写好Cg程序的基础。1.2、Cg的数据类型与关键字基本数据类型:Cg支......
  • Unity Shader 6 Surface Shaders
    UnityShader6SurfaceShaders就是顶点Shader和片段Sahder的封装。SurfaceShaders无pass通道Tags描述渲染类型LOD层次细节#CGPROGRAM表示CG代码块开始#ENDC......
  • Unity Shader 6 Fixed Function Shader 固定管线 材质
    固定管线材质使用语法定义材质属性_MainTex("MainTexture",2D)="White"{}设置纹理setTexture[_MainTex]设置纹理参数texture*Primary混合定义的纹理属......
  • Unity Shader 5 Fixed Function Shader 固定管线Shader
    是相对于可编程的shader而言的。所谓固定管线,是说芯片上一组电路已经固定实现了特定的运算功能,程序能做的只是提供场景数据以及微调运算功能的参数。它适用于所有的显卡,常......
  • ShaderGraph轮廓光
    思路先给纹理,法线,金属贴图,再创建一个自发光贴图和一个菲涅尔效应,将菲涅尔效应与一个颜色属性相乘,得到的结果再与自发光贴图相加,再将结果输出到发光点。     ......
  • ShaderGraph之基础光照
    在ShaderGraph中如果想要自定义一个纹理输入,需要添加SampleTexture2D节点,输出到片元着色器的BaseColor节点。还需要自定义一个Texture2D属性,用于外部修改纹理。新建材......
  • Unity Shader 4 ShaderLab
    通用基础的ShaderUnlit不发光的纹理VertexLit顶点光照Diffuse漫反射Normalmapped法线贴图,增加一个或更多纹理和几个着色器结构Specular镜面高光,模拟......
  • Unity Shader 2 编程语言
    Shader包括了HLSL基于DirectX的HightLevelShadingLanguageGLSL基于OpenGL的OpenGLShadingLanguageCGNVIDIA的CForGraphic简称CGUnity中提供了surfac......