Unity Shader URP 学习 《URP中光照的应用》

我们要明确 realtime 和 bake两种状态

Shader "Custom/A"
{
    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
        [Noraml]_Normal("Normal", 2D) = "bump"{}
        _SpecularScale("Specular scale",float) = 1
        _SpecularStrength("Specular strength",float) = 1
    }
    SubShader
    {
        //renderPipleine决定shader使用那种管线模式
         Tags { "RenderType" = "Opaque" "RenderPipeline" = "UniversalRenderPipeline"}
        LOD 100

        Pass
        {
            //LightMode——URP中已经对Pass进行了集中管理,如果我们没有在URPAsset里设置或
           //RenderFeature方法中添加,自定义的pass将不会起作用
            //https://blog.csdn.net/enk_2/article/details/109081930
            name "Forward"
            Tags{"LightMode" = "UniversalForward"}
             HLSLPROGRAM
            #pragma prefer_hlslcc gles
            #pragma exclude_renderers d3d11_9x
            #pragma target 2.0
            
            //Mainlight受光 阴影相关
            #pragma multi_compile _ _MAIN_LIGHT_SHADOWS
            #pragma multi_compile _ _MAIN_LIGHT_SHADOWS_CASCADE
            #pragma multi_compile _ _ADDITIONAL_LIGHT_SHADOWS
            #pragma multi_compile _ _SHADOWS_SOFT
            
            //额外光照
            #pragma multi_compile _ _ADDITIONAL_LIGHTS_VERTEX _ADDITIONAL_LIGHTS
            //lightmap
            #pragma multi_compile _ LIGHTMAP_ON
            
            #pragma multi_compile_fog

            #pragma vertex vert
            #pragma fragment frag
          
             #pragma multi_compile_instancing


             #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Lighting.hlsl"

            TEXTURE2D(_MainTex);
            SAMPLER(sampler_MainTex);
            TEXTURE2D(_Normal);
            SAMPLER(sampler_MainTex);            

            //CBUFFER中的属性需要和properties中保持一致这样才可以正确SRP Batch,一般放除贴图以外的变量,不支持properties中不包含的传入属性
            CBUFFER_START(UnityPerMaterial)
            float4 _MainTex_ST;
            float _SpecularScale;
            float _SpecularStrength;
            CBUFFER_END


            struct Attributes
            {
                float4 positionOS   : POSITION;
                float3 normalOS     : NORMAL;
                float4 tangentOS    : TANGENT;
                float2 uv           : TEXCOORD0;
                float2 lightmapUV   : TEXCOORD1;
                UNITY_VERTEX_INPUT_INSTANCE_ID
            };

            struct Varyings
            {
               float4 positionCS : SV_POSITION;
               float2 uv         : TEXCOORD0;
                //有lightmap定义lightmap 有顶点定义顶点
                DECLARE_LIGHTMAP_OR_SH(lightmapUV, vertexSH, 1);
                float4 positionWSAndFogfactor : TEXCOORD2;
                float3 noramlWS  : TEXCOORD3;
                float3 tangetnWS : TEXCOORD4;
                float3 bitangentWS:TEXCOORD5;
                #if defined(REQUIRES_VERTEX_SHADOW_COORD_INTERPOLATOR)
                    float4 shadowCoord              : TEXCOORD7;
                #endif
                   UNITY_VERTEX_INPUT_INSTANCE_ID
                   UNITY_VERTEX_OUTPUT_STEREO
            };


            Varyings vert (Attributes input)
            {
                Varyings output;
                UNITY_SETUP_INSTANCE_ID(input);
                UNITY_TRANSFER_INSTANCE_ID(input, output);
                UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(output);
                //vertex和normal的便捷方法,不需要自己繁琐的转换。
                VertexPositionInputs vertexInput = GetVertexPositionInputs(input.positionOS.xyz);
                VertexNormalInputs normalInput = GetVertexNormalInputs(input.normalOS, input.tangentOS);
                output.positionCS = vertexInput.positionCS;
                output.uv = TRANSFORM_TEX(input.uv, _MainTex);

                output.positionWSAndFogfactor.xyz = vertexInput.positionWS;
                output.normalWS = noramlInput.normalWS;
                output.tangetnWS = normlaInput.tangentWS;
                output.bitangentWS = normalInput.bitangentWS;
                //绘制接受阴影
                #if defined(REQUIRES_VERTEX_SHADOW_COORD_INTERPOLATOR)
                    output.shadowCoord = TransformWorldToShadowCoord(vertexInput.positionWS);
              
                #endif 
                
                //input lightmapUV; output lightmapUV;
                OUTPUT_LIGHTMAP_UV(input.lightmapUV, unity_LightmapST, output.lightmapUV);

                OUTPUT_SH(output.normalWS, output.vertexSH);//in normal out vertexSH

                o.positionWSAndFogfactor.w = ComputeFogFactor(output.positionCS.z);
                
                return output;
            }

            half4 frag (Varyings input) : SV_Target
            {
                 UNITY_SETUP_INSTANCE_ID(input);
                 UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
                  
                    Light mainLight;
                    #if defined(REQUIRES_VERTEX_SHADOW_COORD_INTERPOLATOR)
                    mainLight = GetMainLight(input.shadowCoord);
                    #else
                    mainLight = GetMainLight();
                    #endif

                float fogFactor = input.positionWSAndFogfactor.w;
                float3 positionWS = input.positionWSAndFogfactor.xyz;
                float3x3 tangnetTransform = float3x3(input.tangentWS,
                                                     input.bitangentWS,
                                                     input.normalWS);
                // sample the texture
                half4 col = SAMPLE_TEXTURE2D(_MainTex,sampler_MainTex, input.uv);
                
                float3 normal = UnpackNormal(SAMPLE_TEXTURE2D(_Normal, sampler_Normal, input.uv));
                float3 worldNormal = mul(normal, tangnetTransform);

                float NdotL = max(0, dot(worldNormal, mainLight.direcion));
                float3 viewDir = normalize(_WorldSpaceCameraPos.xyz - positionWS);
                float3 halfDir =normalize( mainLight.direction + viewDir);
                float NdotH = max(0, dot(worldNormal, halfDir));
                float spec = pow(NdotH, _SpecularScale) * _SpecularStrength;

                half4 color = 1;
                  color.rgb = col.rgb * NdotL;
                  color.a = col.a;
                half3 bakedGI = SAMPLE_GI(input.lightmapUV,input.vertexSH, worldNormal);
              
                    //实时光   
                    float3 ambient = col.rgb * bakedGI;
                    color.rgb = color.rgb * mainLight.shadowAttenuation + ambient + spec;
                    //额外光照
                    #ifdef _ADDITIONAL_LIGHTS
                        uint pixelLightCount = GetAdditionalLightsCount();
                        for (uint lightIndex = 0u; lightIndex < pixelLightCount; ++lightIndex)
                        {
                            Light addlight = GetAdditionalLight(lightIndex, positionWS);

                            color.rgb += col.rgb * (max(0, dot(addlight.direction, i.normalWS)) * addlight.color.rgb * addlight.distanceAttenuation * addlight.shadowAttenuation);
                        }
                    #endif
          
                color.rgb = MixFog(color.rgb,fogfactor);
                
                // apply fog
                return color;
            }
            ENDHLSL
        }
    }
}

 

 

 

 

  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
深入学习Unity Shader学习图形学知识和URP渲染管线,可以按照以下步骤进行: 1. 学习基础知识:首先,了解图形学的基本概念和原理,包括光照、阴影、纹理映射等。可以阅读相关的图形学教材或者在线资源,掌握相关的数学知识,如向量、矩阵等。 2. 学习Shader语言:Unity使用的是一种类似于C的着色器语言,称为ShaderLab。学习ShaderLab语言,并了解Unity Shader的基本语法和结构是非常重要的。可以参考Unity官方文档和教程,也可以寻找一些书籍或在线教程来学习。 3. 实践编写Shader:通过实践编写自己的Shader来加深对Shader语言的理解和应用。可以从简单的表面着色器开始,逐步探索更复杂的效果,如法线贴图、镜面反射等。可以参考Unity官方文档的示例和教程,也可以寻找一些开源的Shader代码来学习。 4. 学习URP渲染管线:URP(Universal Render Pipeline)是Unity提供的高性能、可定制的渲染管线。学习URP渲染管线可以帮助你理解渲染过程和优化技术。可以阅读Unity官方文档关于URP的介绍和教程,也可以参考一些社区资源和教程。 5. 参考资源:除了官方文档和教程外,还可以参考一些优秀的书籍、博客和视频教程来加深对Unity Shader和图形学知识的理解。一些推荐的资源包括《Unity Shader入门精要》、《Unity Shader之高级进阶》、CSDN博客等。 记住,学习Shader和图形学是一个需要不断实践和探索的过程,通过不断尝试和学习,你会逐渐掌握这些知识并且提升自己的技能。祝你学习顺利!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值