• Unity3D教程宝典之Shader篇:第二十讲法线贴图


    转载自风宇冲Unity3D教程学院

                            






    上一讲我们讲了凹凸贴图以及生成法线贴图。
    这一讲来谈谈怎么使用法线贴图。
    一:法线贴图的原理
    二:法线贴图的实现
    三:法线贴图的使用
    四:法线贴图的格式
     
    一:法线贴图的原理
        光照效果很大程度上是由垂直于物体表面的法线决定的,因为法线影响反射光的方向。均匀垂直的法线是镜面贴图。但是有时候我们会给一个平面使用砖墙贴图,砖墙应该是凹凸不平的,而如果让砖墙使用该平面的法线的话,画面就会很假,神马?一面墙像镜子一样反光=。=
    而如果按真实砖墙去做模型的话,即做高精度模型,一方面制作麻烦,另一方面运行时对性能损耗大。
    法线贴图就是来解决这个问题的。法线贴图就是把法线信息储存在一张图里。使用法线贴图时,通常顶点数和三角形面数只有高精度模型的十分之一不到。
     
    二:法线贴图的实现
        将材质贴图对应的法线 绘制在一张贴图上。将贴图对应点的单位法线向量信息float3(x,y,z) 储存在图对应的颜色里color(r,g,b)里,其中x,y,z分别对应r,g,b。单位法线向量 float3(x,y,z),x,y,z的取值范围是 [-1,1]。在法线贴图中被压缩在颜色的范围[0,1]中,所以需要转换:
    颜色 = 0.5 * 法线 + 0.5;
    线 = 2 * (颜色 - 0.5);    
     
    三:法线贴图的使用
    主要步骤
    (1)对法线贴图进行采样,取得压缩在颜色空间[0,1]里的法线
    float4 packedNormal = tex2D(_NormalMap, IN.uv_MainTex);
    (2)将压缩在[0,1]里的法线转换至3D空间[-1,1]  (因为是单位向量)
    float3 expand(float3 v) { return (v - 0.5) * 2; }
       之后使用该法线即可,方法与16讲里一样。
     
       具体实现详见本文末的脚本。
     
    四:法线贴图的格式
    法线贴图主要分为2个类别:
    (1)RGB法线贴图,即上面使用的。通常呈蓝色。(后缀可以是常见的.png  .jpg等)
    (2)压缩格式的法线贴图。例如DXT5nm(后缀名为.dds)
    dds是DirectDraw Surface的缩写,实际上,它是DirectX纹理压缩(DirectX Texture Compression,简称DXTC)的产物。DXTC减少了纹理内存消耗的50%甚至更多,有3种DXTC的格式可供使用,它们分别是DXT1,DXT3和DXT5。 
    
     
    压缩法线贴图的原理
    法线(x,y,z)是一条单位向量。故X2 + Y2 +Z =1。所以知道了x,y,z里的任意两个,剩下的那个就可以通过计算得出。所以我们就可以使用2个通道的图储存x,y,z里的两个值,将xyz里剩余的值省略,通过计算得出。
     
    压缩法线贴图的好处
    压缩后的法线贴图,大小只有原来的1/4左右,故可以使用更大或者更多的贴图来提升画面品质。
     
    Unity3d的法线贴图
    Unity3d使用的压缩法线贴图是DXT5nm格式的。有A和G两个通道。对于法线(x,y,z) A对应x,G对应y。
    对压缩法线贴图的采样依然是如下函数:

    float4 packedNormal = tex2D(_NormalMap, IN.uv_MainTex);

    packedNormal.w对应A通道,即法线的x。

    packedNormal.y对应G通道,即法线的y。

    范围依然是[0,1], 依然需要转换至[-1,1]。
    对DXT5nm法线贴图进行转换的函数如下,其中v传入packedNormal
     
    float3 expand(float3 v)
    { 
        fixed3 normal;
        normal.xy = v.wy * 2 - 1;
        normal.z = sqrt(1 - normal.x*normal.x - normal.y * normal.y);
        return normal;
    }
    

      

     
    Unity3d的标准法线解压函数是fixed3 UnpackNormal(fixed4 packednormal)。
    打开UnityCG.cginc找到对应函数:
     
    inline fixed3 UnpackNormal(fixed4 packednormal)
    {
        #if defined(SHADER_API_GLES) && defined(SHADER_API_MOBILE)
            return packednormal.xyz * 2 - 1;
        #else
            fixed3 normal;
            normal.xy = packednormal.wy * 2 - 1;
            normal.z = sqrt(1 - normal.x*normal.x - normal.y * normal.y);
            return normal;
        #endif
    }    
    

      

     
    该函数定义的如果是移动平台或者OpenGL ES,那么断定使用的是RGB法线贴图,否则则为DXT5nm贴图。
    但实际上移动平台也可以用压缩格式的法线贴图,而Windows也能使用RGB法线贴图。故不建议使用UnpackNormal函数,建议根据法线贴图的具体格式来使用自己写的对应函数。
     
    ================================================================================================
    ================================================================================================
    ================================================================================================
     
    脚本:
    //  Shader: 带法线贴图的Surface Shader
    //  Author: 风宇冲
    Shader "Custom/3_NormalMap" {
        Properties
        {
            _MainTex ("Texture", 2D) = "white" {}
            _NormalMap ("NormalMap", 2D) = "white" {}
        }
     
        Subshader
        {
            CGPROGRAM
            #pragma surface surf BlinnPhong 
            struct Input
            {
                float2 uv_MainTex;
            };
     
            //法线范围转换:单位法线 float3(x,y,z),x,y,z的取值范围是 [-1,1]。在法线贴图中被压缩在颜色的范围[0,1]中,所以需要转换
            //(1)RGB法线贴图
            float3 expand(float3 v) { return (v - 0.5) * 2; }
            //(2)DXT5nm法线贴图
            float3 expand2(float4 v)
            { 
                fixed3 normal;
                normal.xy = v.wy * 2 - 1;
                normal.z = sqrt(1 - normal.x*normal.x - normal.y * normal.y);
                return normal;
            }
     
            sampler2D _MainTex;
            sampler2D _NormalMap;
     
            void surf(Input IN,inout SurfaceOutput o)
            {
                half4 c = tex2D(_MainTex, IN.uv_MainTex);
                o.Albedo = c.rgb;
                o.Alpha = c.a;
     
                //对法线贴图进行采样,取得压缩在颜色空间里的法线([0,1])
                float4 packedNormal = tex2D(_NormalMap, IN.uv_MainTex);
     
                //要将颜色空间里的法线[0,1],转换至真正3D空间里的法线范围[-1,1]
                //注意:范围基本都是从[0,1]转换至[-1,1].主要是图的通道与法线xyz的对应关系要根据法线贴图格式而定
                //UnpackNormal, UnityCG.cginc里的函数
                //o.Normal = UnpackNormal(packedNormal);
     
                //expand,标准法线解压函数
                o.Normal = expand(packedNormal.xyz);
            }
            ENDCG
        }
    }
    

      

  • 相关阅读:
    拼接表达式树的原理
    ql Server 2012完全卸载方法
    jquery tmpl 详解
    Entity Framework(EF) Code First将实体中的string属性映射成text类型的几种方式
    Entity Framework 数据生成选项DatabaseGenerated
    Entity Framework 复杂类型
    EF Code First 学习笔记:约定配置
    比特币转账流程
    mmap 的理解
    copy_to_user,copy_from_user,get_user,put_user函数比较
  • 原文地址:https://www.cnblogs.com/zdlbbg/p/4329502.html
Copyright © 2020-2023  润新知