[译]为任意网格计算tangent空间的基向量
[译]为任意网格计算tangent空间的基向量
Computing Tangent Space Basis Vectors for an Arbitrary Mesh (Lengyel’s Method)
Modern bump mapping (also known as normal mapping) requires that tangent plane basis vectors be calculated for each vertex in a mesh. This article presents the theory behind the computation of per-vertex tangent spaces for an arbitrary triangle mesh and provides source code that implements the proper mathematics.
现代凹凸映射(也被称为法线映射)要求为网格中的每个顶点计算出其tangent平面基向量。本文描述了对任意三角形网格计算逐顶点tangent空间的数学理论,还提供了实现数学计算的源代码。
Mathematical Derivation 数学推导
[This derivation also appears in Mathematics for 3D Game Programming and Computer Graphics, 3rd ed., Section 7.8.]
[这一推导也出现在Mathematics for 3D Game Programming and Computer Graphics, 3rd ed.,章节7.8。]
We want our tangent space to be aligned such that the x axis corresponds to the u direction in the bump map and the y axis corresponds to the v direction in the bump map. That is, if Q represents a point inside the triangle, we would like to be able to write
我们想让我们的tangent空间这样布局:其x轴对应凹凸贴图的u方向,y轴对应凹凸贴图的v方向。(译者注:这话的意思是,想象一下,将正方形贴图拿起来,将其(0, 0)位置固定到顶点上,此时,其u方向就是我们要找的tangent空间的x轴,其v方向就是我们要找的tangent空间的y轴,三角形面的法线方向就是我们要找的tangent空间的z轴。显然,这样的x轴y轴对,有无数个,我们需要的,是其中唯一的一个。是哪个呢?继续看原文。。。)换句话说,设Q代表三角形内部的一个点,我们希望有
where P0 is the position of one of the vertices of the triangle, and (u0, v0) are the texture coordinates at that vertex. The vectors T and B are the tangent and bitangent vectors aligned to the texture map, and these are what we’d like to calculate.
其中P0是三角形的某个顶点,(u0, v0)是此顶点的纹理坐标。向量T和B分别是tangent 和bitangent 向量,它们分别与纹理贴图的轴平行,是我们想求得的未知数。(译者注:这个公式的意思是,当Q位于P0P1的中点时,u就位于u0u1的中点,等等,当Q位于P0P1P2内部的某点时,通过缩放纹理,会让(u, v)位于同贴图上同样的位置上。顶点P0P1P2的位置构成的三角形A,与顶点P0P1P2的纹理坐标构成的三角形B,两者上的点,即Q和(u, v),建立了一个线性的对应关系。能够使这2个三角形A和B的这样线性关系成立的那个T和B,就是我们需要的唯一的结果。)
Suppose that we have a triangle whose vertex positions are given by the points P0, P1, and P2, and whose corresponding texture coordinates are given by (u0, v0), (u1, v1), and (u2, v2). Our calculations can be made much simpler by working relative to the vertex P0, so we let
假设我们有一个三角形,其顶点位置由P0、P1和P2给定,其对应的纹理坐标由(u0, v0)、(u1, v1)和(u2, v2)给定。通过相对于P0(进行归零),我们的计算可以简化很多,所以我们让
and
且
(译者注:这其实就是让P0和(u0, v0)归零,将三角形A和B都放到原点上。由于A和B是线性关系,所以这样的平移不会影响其线性关系。)
We need to solve the following equations for T and B.
我们需要求解下述等式中的T和B。
This is a linear system with six unknowns (three for each T and B) and six equations (the x, y, and z components of the two vector equations). We can write this in matrix form as follows.
这是个线性系统,有6个未知数(T和B各3个)和6个等式(2个向量等式的x、y和z分量)。我们可以将它改写为下述矩阵形式。
Multiplying both sides by the inverse of the (s, t) matrix, we have
两边同时乘以(s, t)矩阵的逆矩阵(译者注:然后将等号左右互换位置),我们有
This gives us the (unnormalized) T and B vectors for the triangle whose vertices are P0, P1, and P2. To find the tangent vectors for a single vertex, we average the tangents for all triangles sharing that vertex in a manner similar to the way in which vertex normals are commonly calculated. In the case that neighboring triangles have discontinuous texture mapping, vertices along the border are generally already duplicated since they have different mapping coordinates anyway. We do not average tangents from such triangles because the result would not accurately represent the orientation of the bump map for either triangle.
这给出了顶点为P0、P1和P2的三角形的(译者注:即face的)(未标准化的)T和B向量。为了找到顶点的tangent向量,我们对所有共享此顶点的三角形的tangent取平均值,类似于计算顶点的法线那样的方式。当相邻的三角形的纹理映射不连续时,其共享边上的顶点一般都已经被复制了一份,因为它们的纹理坐标毕竟是不同的。(译者注:想象一个人头部模型的三维网格及其贴图,贴图上的后脑勺部分肯定是被分开的,鼻子部分肯定是挨着的,所以网格上描述后脑勺的顶点是被复制了一份,是可以“掰开”的。还是不能想象的话,上网找找模型及其贴图,或者看看电影《画皮》。)我们不把这样的三角形计入平均化过程,因为其结果无法准确地代表(两个三角形中任意一个三角形的)凹凸贴图的朝向。(译者注:由于后脑勺的顶点被复制了一份,就不再共享边了,所以也不会计入平均化过程。)
Once we have the normal vector N and the tangent vectors T and B for a vertex, we can transform from tangent space into object space using the matrix
一旦我们有了顶点的法线向量N和tangent向量T和B,我们就可以实施从tangent空间到object空间的转换了(用矩阵的形式)
To transform in the opposite direction (from object space to tangent space—what we want to do to the light direction), we can simply use the inverse of this matrix. It is not necessarily true that the tangent vectors are perpendicular to each other or to the normal vector, so the inverse of this matrix is not generally equal to its transpose. It is safe to assume, however, that the three vectors will at least be close to orthogonal, so using the Gram-Schmidt algorithm to orthogonalize them should not cause any unacceptable distortions. Using this process, new (still unnormalized) tangent vectors T′ and B′ are given by
为了向反方向变换(从object空间到tangent空间——我们想对光照方向这样做),我们直接使用此矩阵的逆矩阵即可。2个tangent向量未必是互相垂直的,它们也未必与法线向量垂直,所以此矩阵的逆矩阵未必等于其转置矩阵。(译者注:这是个数学定理吧,由3个互相垂直的向量构成的矩阵,其逆矩阵等于其转置矩阵。书到用时方恨少,有空学时最贪玩。)但可以安全地假设,这3个向量是接近互相垂直的,所以使用Gram-Schmidt算法来正交化它们,应该不会引起任何不可接受的失真。使用这个方法,新的(仍旧是未标准化的)tangent向量T′和B′由下述公式给出
Normalizing these vectors
and storing them as the tangent and bitangent for a vertex lets us use the
matrix
标准化这些向量,将其保存为顶点的tangent和bitangent,这样我们就可以使用矩阵
to transform the direction
to light from object space into tangent space. Taking the dot product of the
transformed light direction with a sample from the bump map then produces the
correct Lambertian diffuse lighting value.
来将光线方向从object空间变换到tangent空间。对(变换后的光照方向)和(来自凹凸贴图的采样值)使用点积,就会产生正确的Lambertian漫反射光照值。
It is not necessary to store
an extra array containing the per-vertex bitangent since the cross
product N × T′ can be used to obtain mB′,
where m = ±1 represents the handedness of the tangent
space. The handedness value must be stored per-vertex since the bitangent B′ obtained
from N × T′ may point in the wrong direction.
The value of m is equal to the determinant of the matrix in
Equation (*). You might find it convenient to store the per-vertex tangent
vector T′ as a four-dimensional entity whose w coordinate
holds the value of m. Then the bitangent B′ can be
computed using the formula
没有必要用一个数组来保存逐顶点的bitangent,因为叉积N × T′可以用来获取mB′,其中m = ±1,表示tangent空间的惯用手(译者注:左手系or右手系)。惯用手值必须逐顶点地保存,因为用N × T′得到的bitangent B′可能指向错误的(译者注:即相反的)方向。m的值等于等式(*)中的矩阵的行列式(译者注:行列式其实就是一个数值)。你可能发现了,用四维向量保存逐顶点的tangent向量T′,其w分量保存m,比较方便。这样,bitangent B′就可以用下述公式计算
where the cross product
ignores the w coordinate. This works nicely for vertex shaders
by avoiding the need to specify an additional array containing the
per-vertex m values.
其中的叉积忽略了w分量。这样就可以在顶点着色器中,避免指定一个额外的用于保存逐顶点的m值的数组,棒棒哒。
Bitangent versus Binormal 副切线 versus 副法线
The term binormal is
commonly used as the name of the second tangent direction (that is
perpendicular to the surface normal and u-aligned tangent
direction). This is a misnomer. The term binormal pops up in the study of curves and
completes what is known as a Frenet frame about a particular point on a curve.
Curves have a single tangent direction and two orthogonal normal directions,
hence the terms normal and binormal. When discussing a coordinate frame at a
point on a surface, there is one normal direction and two tangent
directions, which should be called the tangent and bitangent.
名词binormal 通常用作第二个切线方向的名字(即垂直于表面法线和u方向的切线)。这是个误称。名词binormal出现在对曲线(curves )的研究中,它是Frenet坐标系的一部分,用于描述曲线上的点。曲线有一个切线方向和2个垂直的法线方向,因此有了名词normal和binormal。当讨论一个表面(surface)上一点处的坐标系时,有1个法线和2个切线方向,所以应当被称为tangent和bitangent。
Source Code 源代码
The code below generates a
four-component tangent T in which the handedness of the local
coordinate system is stored as ±1 in the w-coordinate. The bitangent
vector B is then given by B = (N × T) · Tw.
下述代码计算了四维tangent
T,其tangent坐标系的惯用手(其值为±1)保存在w分量。之后,bitangent向量B可以通过B = (N × T) · Tw给出。
#include "Vector4D.h" struct Triangle
{
unsigned short index[];
}; void CalculateTangentArray(long vertexCount, const Point3D *vertex, const Vector3D *normal,
const Point2D *texcoord, long triangleCount, const Triangle *triangle, Vector4D *tangent)
{
Vector3D *tan1 = new Vector3D[vertexCount * ];
Vector3D *tan2 = tan1 + vertexCount;
ZeroMemory(tan1, vertexCount * sizeof(Vector3D) * ); for (long a = ; a < triangleCount; a++)
{
long i1 = triangle->index[];
long i2 = triangle->index[];
long i3 = triangle->index[]; const Point3D& v1 = vertex[i1];
const Point3D& v2 = vertex[i2];
const Point3D& v3 = vertex[i3]; const Point2D& w1 = texcoord[i1];
const Point2D& w2 = texcoord[i2];
const Point2D& w3 = texcoord[i3]; float x1 = v2.x - v1.x;
float x2 = v3.x - v1.x;
float y1 = v2.y - v1.y;
float y2 = v3.y - v1.y;
float z1 = v2.z - v1.z;
float z2 = v3.z - v1.z; float s1 = w2.x - w1.x;
float s2 = w3.x - w1.x;
float t1 = w2.y - w1.y;
float t2 = w3.y - w1.y; float r = 1.0F / (s1 * t2 - s2 * t1);
Vector3D sdir((t2 * x1 - t1 * x2) * r, (t2 * y1 - t1 * y2) * r,
(t2 * z1 - t1 * z2) * r);
Vector3D tdir((s1 * x2 - s2 * x1) * r, (s1 * y2 - s2 * y1) * r,
(s1 * z2 - s2 * z1) * r); tan1[i1] += sdir;
tan1[i2] += sdir;
tan1[i3] += sdir; tan2[i1] += tdir;
tan2[i2] += tdir;
tan2[i3] += tdir; triangle++;
} for (long a = ; a < vertexCount; a++)
{
const Vector3D& n = normal[a];
const Vector3D& t = tan1[a]; // Gram-Schmidt orthogonalize
tangent[a] = (t - n * Dot(n, t)).Normalize(); // Calculate handedness
tangent[a].w = (Dot(Cross(n, t), tan2[a]) < 0.0F) ? -1.0F : 1.0F;
} delete[] tan1;
}
CalculateTangentArray
How to cite this article 如何引用本文
Lengyel, Eric. “Computing Tangent Space Basis Vectors for an Arbitrary Mesh”. Terathon Software, 2001. http://terathon.com/code/tangent.html
Copyright © 2001–2017, Terathon Software LLC
[译]为任意网格计算tangent空间的基向量的更多相关文章
- 我发起了一个 网格计算 协议 开源项目 GridP
GridP 是 Grid Protocol 的 全称 . 我在 <关于软件产业的两个契机> https://www.cnblogs.com/KSongKing/p/95319 ...
- Grid Virtual Server 和 网格计算
Grid Virtual Server 的 Virtual Server 源于 LVS (Linux Virtual Server) , LVS 的意思就是把 多个 Linux 服务器 联合起来构成一 ...
- [.NET网格计算框架] Alchemi
Alchemi [.NET网格计算框架] 是 一个以使用简易为目的的Windows下的网格计算框架.它提供了:a)开发网格软件的编程环境 和 b)建造网格和运行网格软件的运行机制. A ...
- opencv —— moments 矩的计算(空间矩/几何矩、中心距、归一化中心距、Hu矩)
计算矩的目的 从一幅图像计算出来的矩集,不仅可以描述图像形状的全局特征,而且可以提供大量关于该图像不同的几何特征信息,如大小,位置.方向和形状等.这种描述能力广泛应用于各种图像处理.计算机视觉和机器人 ...
- Unity Shader-法线贴图(Normal)及其原理
简介 以前经常听说“模型不好看啊,怎么办啊?”答曰“加法线”,”做了个高模,准备烘一下法线贴图”,“有的美术特别屌,直接画法线贴图”.....法线贴图到底是个什么鬼,当年天真的我真的被这个图形学的奇淫 ...
- android计算每个目录剩余空间丶总空间以及SD卡剩余空间
ublic class MemorySpaceCheck { /** * 计算剩余空间 * @param path * @return */ public static String getAvail ...
- C#实现任意大数的计算和简单逻辑命题的证明——前言
介绍 这是本人毕业设计的项目,一直想将其整理成文,可一不小心4年就过去了(这个时间又可以读个大学了).现在给自己定一个目标,一个月时间里将项目的所有关键点都整理出来.不然真怕一眨眼又一个4年过去了,而 ...
- Android为TV端助力 计算每个目录剩余空间丶总空间以及SD卡剩余空间
ublic class MemorySpaceCheck { /** * 计算剩余空间 * @param path * @return */ public static String getAvail ...
- Spatial Transformer Networks(空间变换神经网络)
Reference:Spatial Transformer Networks [Google.DeepMind]Reference:[Theano源码,基于Lasagne] 闲扯:大数据不如小数据 这 ...
随机推荐
- Linux升级OpenSSL版本
安装nginx的时候,出现了这样的问题: nginx : Depends: libssl1.0.0 (>= 1.0.2~beta3) but 1.0.1f-1ubuntu2.11 is to b ...
- C#每天进步一点--引用类型和值类型
在刚参加工作面试时,我们经常会遇到有关值类型和引用类型的问题,你回答的怎么样直接影响你在别人心目中的印象,你回答的不好说明你对C#没有深入的了解学习,今天我带大家回顾下C#中的引用类型和值类型. CL ...
- SQL基础复习1
一.概述 SQL语言组成:DDL,DCL,DML 二.数据定义 1.模式定义(Schema) Schema这个东西一直感觉不大明白,一直以为就是对表的字段定义则被称为Schema,在复习数据库理论中才 ...
- RequestMappingHandlerAdapter和RequestParam原理分析
我们要使用定义了RequestMapping方法或者类是,需要先准备好所需要的参数.如何准备参数,我们应该考虑些上面问题. 都有哪些参数需要绑定? 除了方法确定的参数,还有两个方法的参数需要绑定,那就 ...
- sed命令和find命令的结合的使用
linux中查找当前目录及其子目录下的所有test.txt文件,并将文件中的oldboy替换成oldgirl 首先查找出当前目录及其子目录下的所有的test.txt文件 [root@zxl zxl]# ...
- vuex分模块4
Vuex下Store的模块化拆分实践 https://segmentfault.com/a/1190000007667542 vue.js vuex 猫切 2016年12月02日发布 赞 | 1 ...
- 使用docker部署zabbix
1 官方地址 官方写的很详细并且是中文的,一步步按照操作就可以 https://www.zabbix.com/documentation/3.4/zh/manual/installation/cont ...
- shell写的俄罗斯方块
共享一下. #!/bin/bash # Tetris Game # xhchen<[email]xhchen@winbond.com.tw[/email]> #APP declaratio ...
- Qt之股票组件-自选股--列表可以拖拽、右键常用菜单
目录 一.开头嘴一嘴 二.效果展示 三.自选股列表 1.列表初始化 2.添加Item 3.右键菜单 4.拖拽Item 5.刷新数据 四.相关文章 原文链接:Qt之股票组件-自选股--列表可以拖拽.右键 ...
- Python中的函数及函数参数的使用
函数:一个工具,随调随用 降级代码冗余 增加代码的复用性,提高开发效率,为了不成为cv战士 提高程序扩展性 函数有两个阶段:定义阶段,调用阶段. 定义时:只检查函数体内代码语法,不执行函数体内代码. ...