当前位置:网站首页>【TA-霜狼_may-《百人计划》】先行部分 手搓视差体积云

【TA-霜狼_may-《百人计划》】先行部分 手搓视差体积云

2022-08-03 02:07:00 zczplus

【TA-霜狼_may-《百人计划》】先行部分 手搓体积云

最终效果:
请添加图片描述

对代码部分进行了部分修改,并添加了详细注释:

Shader "Custom/100 learning/MyCloudShader"
{
    
    Properties
    {
    
        _Color ("Color", Color) = (1,1,1,1)
        _MainTex ("MainTex", 2D) = "white" {
    }
        _Alpha("Alpha", Range(0,1)) = 0.5
        _HeightOffset("HeightOffset", Range(0,1)) = 0.15
        _StepLayer("StepLayer", Range(2,64)) = 16
    }
    SubShader
    {
    
        Tags{
    
            "IgnoreProjector" = "True"
            "Queue" = "Transparent-50"
            "RenderType" = "Transparent"

        }

        Pass{
    
            Name "FORWARD"
            Tags{
    
                "LightMode" = "ForwardBase"
            }
            Blend SrcAlpha OneMinusSrcAlpha
            Cull Off
            
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag

            #define UNITY_PASS_FORWARDBASE
            #include "UnityCG.cginc"
            #include "AutoLight.cginc"
            #include "Lighting.cginc"

            #pragma multi_compile_fwdbase
            #pragma target 3.0

            sampler2D _MainTex;
            float4 _MainTex_ST;
            half _HeightOffset;
            half _HeightAmount;
            half4 _Color;
            half _Alpha;
            half _StepLayer;

            struct v2f{
    
                float4 pos : SV_POSITION;
                float2 uv : TEXCOORD0;
                float3 normalDir : TEXCOORD1;
                float3 viewDir : TEXCOORD2;
                float4 posWorld : TEXCOORD3;
                float2 uv2 : TEXCOORD4;
                float4 color : TEXCOORD5;
                UNITY_FOG_COORDS(7)
            };

            v2f vert(appdata_full v){
    
                v2f o;
                o.pos = UnityObjectToClipPos(v.vertex);
                // 去除了原有的Frac函数,frac会导致纹理跳变,将纹理的重复方式修改为mirror即可
                o.uv = TRANSFORM_TEX(v.texcoord, _MainTex) + float2(_Time.y * 0.02, 0);
                o.uv2 = v.texcoord;

                o.posWorld = mul(unity_ObjectToWorld, v.vertex);
                o.normalDir = UnityObjectToWorldNormal(v.normal);

                TANGENT_SPACE_ROTATION;

                o.viewDir = mul(rotation, ObjSpaceViewDir(v.vertex));
                o.color = v.color;
                UNITY_TRANSFER_FOG(o, o.pos);
                return o;
            }

            float4 frag(v2f i) : COLOR{
    
                // 根据视角计算偏移,模拟出假的3D效果,SPM优化而来的POM
                float3 viewDir = normalize(i.viewDir);
                viewDir.xy *= _HeightOffset;

                // 添加偏移值,减少狗牙感
                viewDir.z += 0.4;
                // uv用来计算陡峭视差映射需要用到z坐标
                float3 uv = float3(i.uv, 0);
                // uv2静止不需要z坐标
                float2 uv2 = i.uv2;

                // 静止贴图用于提供a分量
                float4 MainTex = tex2D(_MainTex, uv2);

                // 使用viewDir.xy 除以viewDir.z 可以得到uv的所需偏移方向,并平分成x层
                // 分两段理解,先除以z则使整个贴图的竖直方向范围为1
                // 再除以_StepLayer对竖直方向进行分层
                float3 minOffset = viewDir / (viewDir.z * _StepLayer);
                
                // 两段noise叠加,产生新的noise
                float finiNoise = tex2D(_MainTex, uv.xy).r * MainTex.r;
                float3 prev_uv = uv;

                // 根据SPM的描述,循环体中也应该写prev_uv = uv才对,每次都对prev_uv进行更新
                // 否则prev_uv中存放的位置是初始位置,当距离较远时可能误差会过大
                while(finiNoise > uv.z){
    
                    prev_uv = uv;
                    uv += minOffset;
                    finiNoise = tex2Dlod(_MainTex, float4(uv.xy, 0, 0)).r * MainTex.r;
                }

                // 选取每层选用的UV进行加权计算位置 从而得到最终的uv点
                // 下面这三句非常精髓:
                float d1 = finiNoise -uv.z;
                float d2 = finiNoise - prev_uv.z;
                float w = d1 / (d1 - d2 + 0.0000001);
                // d1 - d2 = prev_uv.z - uv.z = minOffset.z= 1/_StepLayer
                // 经过测试发现这种实现方式会有严重的锯齿,弃用
                // w = d1 / (1/_StepLayer + 0.0000001);
                uv = lerp(uv, prev_uv, w);
                half4 resultColor = tex2D(_MainTex, uv.xy) * MainTex;

                // 透明度相关的一些小track
                half rangeClt = MainTex.a * resultColor.r + _Alpha * 0.75;
                half Alpha = abs(smoothstep(rangeClt, _Alpha, 1.0));
                Alpha = Alpha*Alpha*Alpha*Alpha*Alpha;

                // return MainTex;
                return half4(resultColor.rgb * _Color.rgb, Alpha);
            }
            ENDCG
        }
    }
    FallBack Off
}

要点

  1. 要想有好的效果,使用噪声图最好整体偏亮,并且在亮部依然有梯度变化,否则云没有立体效果;
  2. 纹理的重复方式需要注意,建议使用mirror然后对vs部分的代码进行调整;
  3. SPM陡峭视差贴图改进部分,在循环过程中应当对prev_uv进行每一次的更新,防止误差过大;
  4. 静止的uv贴图可以不用z坐标;
  5. 切线空间的使用!
原网站

版权声明
本文为[zczplus]所创,转载请带上原文链接,感谢
https://blog.csdn.net/weixin_42221907/article/details/126127736