总结起来有两个,一个是基本的转换,单纯的模型空间转换到世界空间,第二个是需要法线贴图时(Bump Textrue)的时候就转换到切线空间下进行计算。

1.从“模型空间”到“世界空间”(Object To World):

(1)方法1,使用和“顶点”到“世界”变换矩阵的“逆转置矩阵“对法线进行相同的变换,因此先得到顶点的模型到世界的变换矩阵的“逆矩阵”(Unity_WorldToObject),然后通过调换它在mul函数中的位置,得到和转置矩阵相同的矩阵乘法,由于法线是一个三维矢量,因此我们只需要街截取unity_WorldToObject的前三行前三列:

worldNormal = normalize(mul(v.normal,(float3x3)unity_WorldToObject))

最外侧的运算“normalize(x)”表示归一化。

(2)使用Unity内置的语法“UnityObjectToWorldNormal()”。
这是Unity5.x在中,可以通过“内置着色器”中的“Unity.cginc”中找到相关的用法,截一小段代码:

// Transforms normal from object to world space
inline float3 UnityObjectToWorldNormal( in float3 norm )
{
#ifdef UNITY_ASSUME_UNIFORM_SCALINGreturn UnityObjectToWorldDir(norm);
#else// mul(IT_M, norm) => mul(norm, I_M) => {dot(norm, I_M.col0), dot(norm, I_M.col1), dot(norm, I_M.col2)}return normalize(mul(norm, (float3x3)unity_WorldToObject));
#endif
}

注释已经写的很清楚了“把法线从模型空间变换到世界空间”,里面做了个判断。如果是“统一缩放”那么这个方法用的是“UnityObjectToWorldDir()”,如果是非统一缩放,则用“normalize(mul(norm, (float3x3)unity_WorldToObject)”,这个就是和“方法1”是一模一样的。之所以这种写法,就是为了保证模型“非统一缩放”也不会得到错误的法线。
而Unity封装的这个方法,就解决了,不管模型是不是统一缩放,用这个肯定没有错。在Unity4.x和之前,是不用管是不是统一缩放的,因为当模型缩放后,unity会在背后重新生成一个模型,和缩放后的一样大,从而法线肯定不会错。但是unity5.x及以后,就必须要注意这个问题。
下面看一下在shader中,怎么使用这个方法:

normalDir = UnityObjectToWorldNormal(v.normal)

具体的推导公式可以参考冯乐乐的书p86。

1.先得到一个“从切线空间到世界空间”的矩阵。
为什么要这个矩阵?我们平时看到的法线贴图记录的都是切线空间下的法线信息(实质上是坐标信息),我们想要用这张图参与接下来的光照计算,其它的如“顶点坐标”““灯光位置”等等都是在世界空间的,自然的这张图也需要在世界空间才能参与计算。因此需要这个“从切线空间到世界空间”的矩阵,来对这张法线图进行变换。一般的用的矩阵是:

float3x3 tangentTransform = float3x3(i.tangentDir,i.bitangentDir,i.normalDir);

矩阵的三个参数:切线、副切线、法线,注意三个参数要进行归一化!

2.对纹理进行采样
既然是使用纹理,项diffusecolor(basecolor)一样,需要对纹理进行采样,用到两个函数:

TRANSFORM_TEX() //计算贴图的UV
tex2D()//对纹理进行采样

当然也可以在一步完成如:

packedNormal = tex2D(_BumpMap,TRANSFORM_TEX(i.uv0,_BumpMap))

3.对“法线贴图”进行解包(Unpack)操作。
(1)上面说到法线贴图记录的是切线空间下的法线信息。而“法线贴图”本身从名字来看,它本身是张图,既然是图,也就是意味着它记录的是颜色信息(RGB)。所以法线贴图本身是有一个映射关系的。
法线本身分量是在[-1,1],而法线贴图(RGB)是以像素为单位的,像素的范围是[0,1],法线变成法线图,必然有个映射关系,那就是“pixel = (normal + 1) / 2”。
(2)另外这个公式还可以解释为什么法线贴图是浅蓝色的,因为法线图是切线空间下的,而切线空间下的法线基本上都在“顶点的切线空间的Z轴方向”扰动,也就是基本上都是(0,0,1)。带入公式,像素值=(0.5,0.5,1)
这个值正是浅蓝色。
(3)既然法线贴图要参与下面的计算,就要映射回去,由上面的公式逆推一下,法线=pixel * 2 - 1,带入公式计算:

//接着上面的代码:
float3 tangentNormal
tangentNormal.xy = (packedNormal.xy * 2 - 1) * _BumpScale;
tangentNormal.z = sqrt(1.0 - saturate(dot(tangentNormal.xy, tangentNormal.xy)));

①上面第一句的意思是首先把“法线贴图”的xy分量按公式映射回法线方向,然后乘以_MumpScale(控制法线强度的值)来得到切线空间下法线的 x y分量。
②第二句意思是:由x y分量,计算Z分量。这是由于法线是单位矢量,所以Z分量可以这么计算。由于我们使用的是切线空间下的法线贴图,因此可以保证法线的Z分量为正(可以参考下面的图)

image.png

(4)理论上这么做是可以的。但是Unity为了优化贴图,在进行“反映射”的同时对纹理进行了压缩优化。具体操作就是把导入的法线贴图,在其属性面板中,把选项标记成“NormalMap”这个操作(即使你不标记,Unity也会提醒你的),在shader中的体现就是“UnpackNormal()”函数。这个函数把压缩贴图和“反映射”同步进行了,具体的可以参考“UnityCG.cginc”下面的定义:

inline fixed3 UnpackNormalDXT5nm (fixed4 packednormal)
{fixed3 normal;normal.xy = packednormal.wy * 2 - 1;normal.z = sqrt(1 - saturate(dot(normal.xy, normal.xy)));return normal;
}inline fixed3 UnpackNormal(fixed4 packednormal)
{
#if defined(UNITY_NO_DXT5nm)return packednormal.xyz * 2 - 1;
#elsereturn UnpackNormalDXT5nm(packednormal);
#endif
}

①可以看到“UnpackNormal()”函数包含了“UnpackNormalDXT5nm()函数”,“UnpackNormal()”函数里面只是做了一个简单的判断(判断是否进行纹理压缩),而真正进行计算的是“UnpackNormalDXT5nm()函数” 其中“DXT5”应该看着很眼熟,它正是压缩方法的其中一种。
②“具体的细节是:把“w”分量(纹理的a通道)对应法线的 x 分量。g通道对应了法线的y分量,而纹理的r和b通道则会被舍弃,法线的Z分量可以由x y 推导得出。更具体的细节以后再看吧。可以看到“UnpackNormalDXT5nm()函数”的输入参数是“packednormal”,(采样后的法线贴图),但是“packednormal”的参数只有RGB,没有w y 所以可以推断这个压缩过程是在外部进行的。而“UnpackNormalDXT5nm()函数”只是对压缩后的“packednormal”新的RGB进行反映射。

(5)反映射后,如果还想要调节法线的强度,同样的还需要乘上调节强度的参数(_BumpScale)具体可以这样写:

fixed3 tangentNormal;
tangentNormal = UnpackNormal(packedNormal);
tangentNormal.xy *= _BumpScale;
tangentNormal.z = sqrt(1.0 - saturate(dot(tangentNormal.xy, tangentNormal.xy)));

可以看到只是把(3)手动映射的计算,换成UnpackNormal()操计算。这样才能保证得到的正确法的法线。
这里还要提一句:不能把得到的“tangentNormal”直接带入“反映射公式”去计算,如:
tangentNormal = tangentNormal.xyz * 2-1,因为Z分量是由 x y 分量推导出来的。

4.得到反映射的法线后,就可以 用“1”得到的变换矩阵去变换了:

float3 normalDirection = normalize(mul(tangentNormal,tangentTransform));
//用矩阵变换后,执行归一化

5.到这里,需要用法线贴图的基本shader的法线变换就讲完了。

转载链接:https://www.jianshu.com/p/c6a94d3b64bc

shader中的法线变换相关推荐

  1. 法线变换详解 和 3D 变换中法向量变换矩阵的推导

    两篇文章 法线变换详解(Normal Transform) 在图形学中,同样的一个模型视图变换矩阵可以用来变换点.线.多边形以及其它几何体,也可以变换多边形表面的切向量.比如: posEyeSpace ...

  2. unity shader中 CG和GLSL之间矩阵存储方式的差异

    在unity shader中计算切线空间光照的时候发现TBN矩阵在计算后并没有转置,这和在GLSL.线代中的计算方法相差一个转置的步骤,因为在GLSL中直接求得的TBN矩阵是切线空间到世界空间的变换, ...

  3. 凹凸映射中的法线纹理贴图

    凹凸映射(bump mapping) 凹凸映射的目的:使用一张纹理来修改模型表面的法线,让模型看上去"凹凸不平". 主要有两种方法:一.高度纹理来模拟表面位移,得到一个修改后的法线 ...

  4. Unity Shaders and Effects Cookbook (7-2) Surface Shader 中实现 顶点动画

    上一节中说了,在 Surface Shader 中,添加顶点函数,我们可以在 顶点函数中获取到 顶点数据,比如顶点颜色.顶点坐标等. 这一节学习获取顶点坐标,并且修改顶点坐标,来实现顶点动画. 简单介 ...

  5. 【Unity】环境光探究,在shader中如何计算 : Ambient Color, Sky Color, Equator Color, Ground Color

    环境光 接触过光照模型的小伙伴肯定对环境光不陌生,比如phong光照模型,渲染结果就是下面几项的线性叠加: 环境光+漫反射+高光 而一般情况我们在Shader中计算环境光也是简单的不能再简单了,直接可 ...

  6. 【Unity Shaders】Reflecting Your World —— Unity3D中的法线贴图和反射

    本系列主要参考<Unity Shaders and Effects Cookbook>一书(感谢原书作者),同时会加上一点个人理解或拓展. 这里是本书所有的插图.这里是本书所需的代码和资源 ...

  7. 如何在Unity实现从纹理中生成法线贴图?

    本文主要讲解从纹理中生成法线贴图的基本方法,并在 Unity 中进行实现和测试. 预备知识 法线贴图和基本的图形学知识,基本的向量和极限的知识. 高度图或灰度图 一张二维纹理有两个维度 u 和 v,但 ...

  8. Unity Shader中各部分定义内容详解

    Unity Shader中各部分定义内容详解 样板 Shader "Practice/Unlit/SimpleUnlit" {Properties{_MainTex (" ...

  9. 【OpenGL】理解VAO、VBO、EBO和 shader中vertex、fragment交互。

    1:首先了解GPU和渲染管道工作原理. 2:解释VAO.VBO.EBO 顶点数组对象:Vertex Array Object,VAO 顶点缓冲对象:Vertex Buffer Object,VBO 元 ...

最新文章

  1. Oracle SCN
  2. Visual Studio 2010Beta与Silverlight的更新
  3. java 64进制转10进制_java进制转换
  4. JavaEE是什么?
  5. 【每周CV论文】初学深度学习图像风格化要读的文章
  6. 操作系统实验报告6:进程间通信—共享内存
  7. php 织梦模板 防盗,dedecms 软件下载频道防盗链php代码
  8. mysql emoji 显示不出来_解决 Mysql 存取 emoji 表情的问题
  9. C++ 模板何时被实例化
  10. [luogu1880] [NOI1995]石子合并
  11. JDBC+Servlet+JSP实现基本的增删改查(简易通讯录)
  12. 光盘重装linux系统教程视频,重装系统?一步一步安装系统详细教程【带视频】。...
  13. Android 音乐播放器SD卡本地播放器实现
  14. 配置VScode上基于WSL的lc3汇编语言环境
  15. OA课程--word2013实用技巧大全-目录
  16. 拉伯证券|社会消费复苏将是2023年主旋律
  17. 读淘宝页面字节流提取宝贝图片地址宝贝标题宝贝价格
  18. Win10无法更改账户名称怎么办
  19. 幸运数 c++程序(详解,附完整代码)
  20. Java学习导航汇总

热门文章

  1. 吉大 杨博 计算机,杨博-计算机科学学院
  2. 火热朝天?现在的技术VC该何去何从
  3. 打印机池的设计与实现
  4. etc apdu指令
  5. hpy计算机维护系统,HPY计算机维护系统2013(W8PE+03PE+DOS,ISO量产版)
  6. 【stm32】手把手用cubemx配置血氧传感器(MAX30102)
  7. c语言数组转bin,C语言数组转换为BIN文件工具
  8. java把汉字转换url_url中汉字编码互相转换
  9. 微弱直流电压/电流信号的采样电路 --滤波跟随放大
  10. 文件系统-- 安装根文件系统阶段(安装rootfs文件系统)