Unity Shader 凹凸映射 法线纹理 在世界空间下计算

Shader "MyShader/NormalMapWorldMapSpace"
{
    Properties
    {
        _Color("Color Tint",Color) = (1,1,1,1)
        _MainTex("Main Tex",2D) = "white"{}
        _BumpMap("Normal Map",2D) = "bump"{}
        _BumpScale("Bump Scale",Float) = 1.0
        _Specular("Specular",Color) = (1,1,1,1)
        _Gloss("Gloss",Range(8.0,256)) = 20
    }
        SubShader
        {
            Pass
            {
                Tags{"LightMode" = "ForwardBase"}

                CGPROGRAM

    #pragma vertex vert
    #pragma fragment frag

    #include "Lighting.cginc"

                fixed4 _Color;
                sampler2D _MainTex;
                float4 _MainTex_ST;
                sampler2D _BumpMap;
                float4 _BumpMap_ST;
                float _BumpScale;
                fixed4 _Specular;
                float _Gloss;

                struct a2v
                {
                    float4 vertex:POSITION;
                    float3 normal:NORMAL;
                    float4 tangent:TANGENT;
                    float4 texcoord : TEXCOORD0;
                };

                struct v2f
                {
                    float4 pos:SV_POSITION;
                    float4 uv:TEXCOORD0;
                    float4 TtoW0:TEXCOORD1;
                    float4 TtoW1:TEXCOORD2;
                    float4 TtoW2:TEXCOORD3;
                };

                v2f vert(a2v v)
                {
                    v2f o;

                    o.pos = UnityObjectToClipPos(v.vertex);

                    o.uv.xy = v.texcoord.xy * _MainTex_ST.xy + _MainTex_ST.zw;
                    o.uv.zw = v.texcoord.xy * _BumpMap_ST.xy + _BumpMap_ST.zw;

                    float3 worldPos = mul(unity_ObjectToWorld, v.vertex).xyz;
                    fixed3 worldNormal = UnityObjectToWorldNormal(v.normal);
                    fixed3 worldTangent = UnityObjectToWorldDir(v.tangent.xyz);
                    fixed3 worldBinormal = cross(worldNormal, worldTangent)*v.tangent.w;

                    o.TtoW0 = float4(worldTangent.x, worldBinormal.x, worldNormal.x, worldPos.x);
                    o.TtoW1 = float4(worldTangent.y, worldBinormal.y, worldNormal.y, worldPos.y);
                    o.TtoW2 = float4(worldTangent.z, worldBinormal.z, worldNormal.z, worldPos.z);

                    return o;
                }

                fixed4 frag(v2f i) :SV_Target
                {
                    float3 worldPos = float3(i.TtoW0.w,i.TtoW1.w,i.TtoW2.w);
                    fixed3 lightDir = normalize(UnityWorldSpaceLightDir(worldPos));
                    fixed3 viewDir = normalize(UnityWorldSpaceViewDir(worldPos));

                    fixed3 bump = UnpackNormal(tex2D(_BumpMap, i.uv.zw));
                    bump *= _BumpScale;
                    bump.z = sqrt(1.0 - saturate(dot(bump.xy, bump.xy)));
                    bump = normalize(half3(dot(i.TtoW0.xyz, bump), dot(i.TtoW1.xyz, bump), dot(i.TtoW2.xyz, bump)));

                    fixed3 albedo = tex2D(_MainTex, i.uv).rgb * _Color.rgb;
                    fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz * albedo;
                    fixed3 diffuse = _LightColor0.rgb * albedo * max(0, dot(bump, lightDir));
                    fixed3 halfDir = normalize(lightDir + viewDir);
                    fixed3 specular = _LightColor0.rgb * _Specular.rgb * pow(max(0, dot(bump, halfDir)), _Gloss);
                    return fixed4(ambient + diffuse + specular, 1.0);
                }
            ENDCG
        }
        }
            Fallback "Specualr"
}

效果,右边那个

参考 我买的 unity shader入门精要

Unity Shader中,求取法线矩阵的逆矩阵通常是为了进行光照计算,尤其是在使用了表面着色器(Surface Shader)时。法线矩阵是由模型变换矩阵的上3x3部分构成的,用于保持法线向量在空间变换后的方向正确。如果模型变换包含了非均匀缩放,则直接使用模型矩阵的逆转置作为法线矩阵会导致光照计算错误。 法线矩阵(Normal Matrix)是模型变换矩阵中用于变换法线的矩阵,通常是模型矩阵的逆转置(Inverse Transpose)。在Shader中,可以通过内置函数`UNITY_MATRIX_IT_MV`来直接获取模型视图空间下的法线矩阵,或者使用`UNITY_MATRIX_I_MV`获取模型视图空间下的逆转置矩阵,再进行逆矩阵计算Unity Shader语言中并没有直接的函数可以获取法线矩阵的逆矩阵,但是可以使用`transpose`函数来得到逆转置矩阵,然后使用`inverse`函数来得到逆矩阵。不过,这通常不必要,因为大多数情况下我们只需要逆转置矩阵来正确变换法线向量。如果确实需要法线矩阵的逆矩阵(而非逆转置矩阵),则需要首先使用`transpose`函数得到逆转置矩阵,然后使用`inverse`函数来计算最终的逆矩阵。 以下是一个简单的例子,展示如何在Unity Shader中获取并使用法线矩阵(逆转置)来变换法线向量: ```csharp Shader "Custom/NormalMatrixExample" { Properties { _MainTex ("Texture", 2D) = "white" {} } SubShader { Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag #include "UnityCG.cginc" struct appdata { float4 vertex : POSITION; float3 normal : NORMAL; float2 uv : TEXCOORD0; }; struct v2f { float2 uv : TEXCOORD0; float4 vertex : SV_POSITION; float3 normal : TEXCOORD1; }; sampler2D _MainTex; float4 _MainTex_ST; v2f vert (appdata v) { v2f o; o.vertex = UnityObjectToClipPos(v.vertex); o.normal = UnityObjectToWorldNormal(v.normal); o.uv = TRANSFORM_TEX(v.uv, _MainTex); return o; } fixed4 frag (v2f i) : SV_Target { // 使用法线矩阵(逆转置)变换法线向量 float3 normal = mul((float3x3)UNITY_MATRIX_IT_MV, i.normal); // 进行光照计算... return fixed4(normal, 1.0); } ENDCG } } FallBack "Diffuse" } ``` 请注意,上面的代码仅用于演示如何获取和使用法线矩阵,并未包含完整的光照模型计算
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值