Unity内置Shader解读11——VertexLit

1.Shader在什么情况下使用

VertexLit作为大部分shader的FallBack,实现了基础的光照、Lightmapped和阴影三个Pass。

2.Shader的价值(用的多不多),Shader的难度

直接用的可能不多,但绝大部分shader最终都FallBack到了这里,难度中,阴影部分还没搞清楚

3.代码详细注释

Shader "Legacy Shaders/VertexLit" {
    Properties {
        _Color ("Main Color", Color) = (1,1,1,1)
        _SpecColor ("Spec Color", Color) = (1,1,1,1)//镜面反射
        _Emission ("Emissive Color", Color) = (0,0,0,0)//自发光
        [PowerSlider(5.0)] _Shininess ("Shininess", Range (0.01, 1)) = 0.7//光泽度
        _MainTex ("Base (RGB)", 2D) = "white" {}
    }

    SubShader {
        Tags { "RenderType"="Opaque" }
        LOD 100

        // Non-lightmapped
        Pass {
            Tags { "LightMode" = "Vertex" }
            //使用最基础的渲染模式Fixed Shader实现基础效果
            Material {
                Diffuse [_Color]
                Ambient [_Color]
                Shininess [_Shininess]
                Specular [_SpecColor]
                Emission [_Emission]
            }
            Lighting On
            SeparateSpecular On
            SetTexture [_MainTex] {
                constantColor (1,1,1,1)
                Combine texture * primary DOUBLE, constant // UNITY_OPAQUE_ALPHA_FFP
            }
        }

        // Lightmapped
        Pass
        {
            Tags{ "LIGHTMODE" = "VertexLM" "RenderType" = "Opaque" }

            CGPROGRAM

            #pragma vertex vert
            #pragma fragment frag
            #pragma target 2.0
            #include "UnityCG.cginc"
            #pragma multi_compile_fog
            #define USING_FOG (defined(FOG_LINEAR) || defined(FOG_EXP) || defined(FOG_EXP2))

            float4 unity_Lightmap_ST;
            float4 _MainTex_ST;

            struct appdata
            {
                float3 pos : POSITION;
                float3 uv1 : TEXCOORD1;
                float3 uv0 : TEXCOORD0;
                UNITY_VERTEX_INPUT_INSTANCE_ID
            };

            struct v2f
            {
                float2 uv0 : TEXCOORD0;
                float2 uv1 : TEXCOORD1;
                float2 uv2 : TEXCOORD2;
                #if USING_FOG
                    fixed fog : TEXCOORD3;
                #endif
                float4 pos : SV_POSITION;
                UNITY_VERTEX_OUTPUT_STEREO
            };

            v2f vert(appdata IN)
            {
                v2f o;
                UNITY_SETUP_INSTANCE_ID(IN);
                UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(o);

                //获取Lightmap的UV
                //将模型顶点的uv和Tiling、Offset两个变量进行运算,计算出实际显示用的顶点uv
                //乘以Tiling再加Offset
                o.uv0 = IN.uv1.xy * unity_LightmapST.xy + unity_LightmapST.zw;
                //o.uv1完全没有使用到,不知道这里写了有啥用,知道大佬麻烦告知下
                o.uv1 = IN.uv1.xy * unity_Lightmap_ST.xy + unity_Lightmap_ST.zw;
                //获取主纹理的UV
                o.uv2 = IN.uv0.xy * _MainTex_ST.xy + _MainTex_ST.zw;

                #if USING_FOG
                    //转到View空间下
                    float3 eyePos = UnityObjectToViewPos(IN.pos);
                    //length返回输入向量的长度,离view点越近的点雾效越淡
                    float fogCoord = length(eyePos.xyz);
                    //计算雾浓度
                    //具体可参考:https://blog.csdn.net/zengjunjie59/article/details/112346424
                    UNITY_CALC_FOG_FACTOR_RAW(fogCoord);
                    //限制o.fog值从0-1
                    o.fog = saturate(unityFogFactor);
                #endif

                //MVP转换
                o.pos = UnityObjectToClipPos(IN.pos);
                return o;
            }

            sampler2D _MainTex;
            fixed4 _Color;

            fixed4 frag(v2f IN) : SV_Target
            {
                fixed4 col;

                //采样Lightmap
                fixed4 tex = UNITY_SAMPLE_TEX2D(unity_Lightmap, IN.uv0.xy);
                //DecodeLightmap解码光照贴图 
                half4 bakedColor = half4(DecodeLightmap(tex), 1.0);

                //将烘焙结果和_Color混合
                col = bakedColor * _Color;

                //采样主纹理
                tex = tex2D(_MainTex, IN.uv2.xy);
                //将col和主纹理颜色混合
                col.rgb = tex.rgb * col.rgb;

                //alpha控制为1
                col.a = 1.0f;

                #if USING_FOG
                    //混合雾效
                    col.rgb = lerp(unity_FogColor.rgb, col.rgb, IN.fog);
                #endif

                return col;
            }

            ENDCG
        }

        //VertexLit作为所有需要阴影的shader的FallBack,其实所有的阴影都是走的这个Pass
        // Pass to render object as a shadow caster 传递以将对象渲染为阴影投射器
        Pass {
            Name "ShadowCaster"
            Tags { "LightMode" = "ShadowCaster" }

            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #pragma target 2.0
            #pragma multi_compile_shadowcaster
            #pragma multi_compile_instancing // allow instanced shadow pass for most of the shaders允许大多数着色器使用实例化shadow pass
            #include "UnityCG.cginc"

            struct v2f {
                //阴影宏定义,主要定义了float4 pos:SV_POTISION;
                V2F_SHADOW_CASTER;
                //INSTANCE_ID
                UNITY_VERTEX_OUTPUT_STEREO
            };

            v2f vert( appdata_base v )
            {
                v2f o;
                //INSTANCE_ID
                UNITY_SETUP_INSTANCE_ID(v);
                UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(o);

                //Unity提供的阴影计算方法,搜不到解释,只有这篇解释了一半https://blog.csdn.net/yanyangxu01/article/details/81987064
                //既然大家都不求甚解只管用,我也先放一放吧能力还不够-。-,后面有能力了再开专篇讲吧
                TRANSFER_SHADOW_CASTER_NORMALOFFSET(o)
                return o;
            }

            float4 frag( v2f i ) : SV_Target
            {
                //同上
                SHADOW_CASTER_FRAGMENT(i)
            }
            ENDCG

        }
    }
}

4.Shader编写思路,用到的知识点

第一个Pass使用Fixed Shader实现基础的光照,第二个实现Lightmapped,最后一个实现阴影,作为绝大部分shader的FallBack终点,其实很多阴影都是用的这边的pass。知识点不太多,主要是阴影部分还没搞明白,也搜不到太多的资料,后面再更吧。


Legacy Shaders基本已经讨论完了,同类型相似的不再赘述。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值