首页 > 其他分享 >【Unity3D】空间和变换

【Unity3D】空间和变换

时间:2023-03-21 21:01:09浏览次数:57  
标签:Unity3D 变换 矩阵 空间 z0 坐标系 向量

1 空间

1.1 左右手坐标系及其法则

1.1.1 左右手坐标系

img 左手坐标系与右手坐标系

​ Unity 局部空间、世界空间、裁剪空间、屏幕空间都采用左手坐标系,只有观察空间采用右手坐标系。

​ 左右手坐标系除了坐标系朝向(旋向性)不同,还存在以下差异:

  • 左手坐标系下旋转正方向的定义遵循左手法则,右手坐标系下旋转正方向的定义遵循右手法则;
  • 左手坐标系下向量叉乘遵循左手法则,右手坐标系下向量叉乘遵循右手法则。

1.1.2 左右手法则

​ 左手坐标系和右手坐标系下旋转正方形定义:

img 左手旋转法则与右手旋转法则

​ 左手坐标系和右手坐标系下向量叉乘方向定义:

img 左手叉乘法则与右手叉乘法则

1.2 四维空间

​ 在三维空间中,将向量 u 变换到向量 v,如果该变换是一个线性变换,一般可以使用一个不依赖于 u、v 的 3x3 的矩阵描述,即 A · u = v,其中 A 是描述该线性变换的变换矩阵。但是,并不是所有线性变换都能使用 3x3 矩阵描述,如:平移变换和对称变换,对于这些变换,需要将空间扩展到四维空间中,再使用 4x4 矩阵描述。

​ Unity 模型变换主要有平移、旋转、缩放、对称,其中旋转和缩放变换可以使用 3x3 矩阵描述,平移和对称变换必须使用 4x4 矩阵描述,为了统一描述这些变换,将三维空间扩展到四维空间中,扩展方法如下:

  • 三维空间中的坐标原点 [0, 0, 0]' 映射到四维空间中的 [0, 0, 0, 1]' 点;
  • 三维空间中的 x、y、z 轴正方向单位向量 [1, 0, 0]'、[0, 1, 0]'、[0, 0, 1]' 分别映射到四维空间中的 [1, 0, 0, 0]'、 [0, 1, 0, 0]'、 [0, 0, 1, 0]' 向量;
  • 新添加的 w 轴 正方向单位向量为 [0, 0, 0, 1]'。

​ 注意:三维空间中的原点 [0, 0, 0]' 并没有映射到四维空间中的原点 [0, 0, 0, 0]‘,而是映射到 [0, 0, 0, 1]’。

​ 由以上空间映射关系可知:

  • 三维空间中的任意点 [x, y, z]',对应的四维空间坐标为:[x, y, z, 1]';
  • 三维空间中的任意向量 [x, y, z]',对应的四维空间坐标为:[x, y, z, 0]'。

1.3 Unity 空间

1.3.1 Unity 空间变换过程

img

  • 从模型空间到裁剪空间的变换过程在顶点着色器中完成,顶点着色器输入模型局部坐标,输出裁剪空间中的坐标;
  • 从裁剪空间到屏幕空间的变换过程由系统控制,用户不能控制该阶段变换。

1.3.2 Unity 空间变换示意图

1) 模型空间、世界空间、 观察空间

img 模型空间、世界空间、 观察空间

​ 模型空间和世界空间的坐标系是左手坐标系,观察空间的坐标系是右手坐标系。

2) 观察空间

img 观察空间(透视相机和正交相机)

​ 近平面和远平面间棱台称为视锥体,表示可见区域范围,视锥体以外的顶点数据将被裁剪丢弃。

3) 裁剪空间

img 裁剪空间(透视投影和正交投影)

​ 裁剪空间中的点满足条件:-w < x < w,-w < y < w,-w < z < w,不满足该条件的点将被踢出掉。

4)归一化的设备空间

​ 将裁剪空间中的点 x、y、z 坐标都除以 w(齐次除法、透视除法),就会得到归一化的设备空间,该空间中的点满足条件:-1 < x < 1,-1 < y < 1,-1 < z < 1。

img 归一化的设备空间

​ Unity 和 OpenGL 归一化的设备空间中的点 z 值值域是 [-1, 1];DirectX 归一化的设备空间中的点 z 值值域是 [0, 1]。

5)屏幕空间

img 屏幕空间

​ Unity 和 OpenGL 屏幕空间坐标系原点都在左下角,向右和向上分别为 x 轴、y 轴正方向;DirectX 屏幕空间坐标系原点都在左上角,向右和向下分别为 x 轴、y 轴正方向。

2 变换

2.1 模型变换

2.1.1 平移变换

​ 对于任意点 [x, y, z, 1]',将其平移 [a, b, c, 0]',可以使用以下矩阵运算描述平移变换:

img

​ 对于任意向量 [x, y, z, 0]',将其平移 [a, b, c, 0]',平移后仍然是 [x, y, z, 0]',如下:

img

2.1.2 旋转变换

​ 绕 x 轴旋转 α 度,对应的旋转矩阵如下:

img

​ 绕 y 轴旋转 α 度,对应的旋转矩阵如下:

img

​ 绕 z 轴旋转 α 度,对应的旋转矩阵如下:

img

​ 当旋转角度为 [α, β, γ]' 时,对应的复合旋转矩阵为:

img

2.1.3 缩放变换

​ 对于缩放系数 [kx, ky, kz]',对应的缩放矩阵如下:

img

​ 当 kx = ky = kz 时,该缩放变换称为统一缩放(uniform scale),否则称为非统一缩放(nonuniform scale)。

2.1.4 对称变换

​ 对于任意点 [x, y, z, 1]',关于点 [a, b, c, 1]' 对称的点,可以使用以下矩阵运算描述对称变换:

img

2.2 观察变换

​ 将世界坐标系按照 1.2 节映射方法扩展到四维空间中,假设 x、y、z、w 轴正方向的的单位向量分别为 e1、e2、e3、e4,相机在世界坐标系下的坐标为 [a, b, c, 1]',其向右、向上、向前方向的单位向量分别为:r、u、f,则向量 r、u、-f、e4 在一组基向量 e1、e2、e3、e4 下的表示如下:(f 取负是因为观察坐标系是右手坐标系,其 z 轴的正方向与相机的前方方向相反)

img

​ 由于向量 r、u、-f、e4 两两垂直,并且都是单位向量,因此由这 4 个向量组成的矩阵是正交矩阵,即 A-1 = A'。由此可知,向量 e1、e2、e3、e4 在一组基向量 r、u、-f、e4 下的表示如下:

img

​ 右侧的矩阵就是坐标轴的旋转变换矩阵,另外还需要进行坐标原点的平移变换,由 1.2.1 节平移变换矩阵可知,观察变换矩阵如下:

img

2.3 投影变换

2.3.1 透视投影

1)透视投影相机参数

​ Unity 提供给用户调整透视投影矩阵的参数如下:

img

  • Near:近平面距离相机的距离;
  • Far:远平面距离相机的距离;
  • Field of View:视锥体在竖直方向上的张角(以下简称 FOV);
  • Viewport Rect:视口起点和宽高(左下角为原点,向右宽度增加,向上高度增加)

​ 根据 Near、Far、FOV 的值,可以计算近平面和远平面高度分别为:

img

​ 近平面和远平面的宽度受相机(或屏幕)宽高比影响,假设相机(或屏幕)宽高比为 Aspect,则近平面和远平面的宽度计算如下:

img

2)透视投影矩阵推导

img

​ 投影的目的是:将模型顶点投影到近平面上,如下,将观察坐标系下的任意顶点 [x0, y0, z0]' 投影到近平面上,投影后的坐标为 [x1, y1, z1]’(z1 = -Near)。

img

​ 根据三角形相似原理,存在以下函数关系:

img

​ 为了方便后续进行屏幕映射,需要将 [x1, y1]' 进行标准化,即将 x1、y1 映射到区间 [-1, 1],当前 -nearClipPlaneWidth / 2 ≤ x1 ≤ nearClipPlaneWidth / 2,-nearClipPlaneHeight / 2 ≤ y1 ≤ nearClipPlaneHeight / 2,假设归一化后的坐标为 [x2, y2],因此存在以下关系:

img

​ 将 x1、y1、nearClipPlaneWidth、nearClipPlaneHeight 使用 x0、y0、z0、Aspect、FOV 替换得:

img

​ x2、y2 已标准化,但是当前 -Far ≤ z0 ≤ -Near,我们期望将 z0 也标准化,假设标准化后的变量为 z2,则 -1 ≤ z2 ≤ 1。x2 与 x0、y2 与 y0 的关系式中,都存在 (-1 / z0),我们期望 z2 与 z0 的关系式中也存在 (-1 / z0),因此,我们假设 z2 与 z0 的关系如下:

img

​ 将 [-Far, 1]'、[-Near, -1]' 代入求解得:(注意:这里不能代入 [-Far, -1]'、[-Near, 1]',因为裁剪坐标系的 z 轴和观察坐标系的 z 轴方向相反)

img

​ 将 k、b 代入 z2 与 z0 的关系式中得:

img

​ 整理 x2、y2、z2 与 x0、y0、z0 的关系如下:

img

​ 由于 z0 是变量,在矩阵前面乘以 (-1 / z0),使得透视变换不是线性变换,因此我们将原本的透视变换拆分为以下两步:

  • 对 [x0, y0, z0, 1]' 左乘透视矩阵;
  • 将第一步的结果除以 (-z0)。

​ 为保证透视变换的线性性质,我们将第二步变换移到屏幕映射中处理,并将其定义为齐次除法(或透视除法),而将第一步变换作为透视变换,其对应的矩阵如下:

img

​ 说明:P(4, 3) 定义为 -1 是为了将观察坐标系中顶点的深度信息 (-z) 传递给下一步(齐次除法或透视除法)处理(用 w 存储,即 w = -z),避免深度信息丢失。经透视变换后,顶点坐标的 x、y、z 分量将约束在 [z, -z] 区间,即 [-w, w],在该区间外的顶点将被丢弃。

​ 观察空间和裁剪空间视锥体的对比如下:

img点击并拖拽以移动

2.3.2 正交投影

1)正交投影相机参数

​ Unity 提供给用户调整正交投影矩阵的参数如下:

img

  • Size:视锥体在竖直方向上的高度的一半;
  • Near:近平面距离相机的距离;
  • Far:远平面距离相机的距离;
  • Viewport Rect:视口起点和宽高(左下角为原点,向右宽度增加,向上高度增加)

2)正交投影矩阵推导

img

​ 投影的目的是:将模型顶点投影到近平面上,如下,将观察坐标系下的任意顶点 [x0, y0, z0]' 投影到近平面上,投影后的坐标为 [x0, y0, -Near]'。

img

​ 为了方便后续进行屏幕映射,需要将 [x0, y0, z0]' 进行标准化,即将 x0、y0、z0 映射到区间 [-1, 1],当前 -Aspect · Size ≤ x1 ≤ Aspect · Size,-Size ≤ y1 ≤ Size,-Far ≤ z0 ≤ -Near,假设归一化后的坐标为 [x1, y1, z1]',因此存在以下关系:

img

​ 说明:正交投影变换后,w 分量的值仍然是 1,从而保证 [x1, y1, z1]' 经齐次除法(除以 w1)后,仍然是标准化坐标(即值域为 [-1, 1]),这样做的好处是:下游不用区分上游传递过来的数据是透视投影还是正交投影处理后的数据。经正交投影变换后,顶点坐标的 x、y、z 分量将约束在 [-1, 1] 区间,在该区间外的顶点将被丢弃。

​ 观察空间和裁剪空间视锥体的对比如下:

img

2.4 齐次除法和屏幕映射

2.4.1 齐次除法

​ 经透视投影或正交投影后,将坐标 [x, y, z, w] 中的 x、y、z 分量都除以其 w 分量的值,使得 x、y、z 都约束在 [-1, 1] 区间,该过程称为齐次除法(或透视除法),得到的坐标称为归一化的设备坐标NDC)。

img

2.4.2 屏幕映射

​ 经齐次除法后,将坐标的 x、y 值映射到屏幕像素位置,该过程称为屏幕映射。映射前 x、y 的值域为 [-1, 1],映射后 x 的值域为 [0, pixelWidth],y 的值域为 [0, pixelHeight],屏幕左下角坐标为 [0, 0],右上角坐标为 [pixelWidth, pixelHeight]。屏幕映射公式如下:

img

2.5 法线变换

​ 假设模型变换为 M,模型空间中某点法线向量为 n,如果模型变换中包含非统一缩放(即 x、y、z 的缩放系数不全相等), 此时若按照 M · n 计算法线的世界坐标,就会出现变换后的法线与切面不垂直,如下图。

img

​ 法线由切线计算而来,在模型空间中 A 点的切线向量为 v1,法线向量为 n1,经过模型变换(矩阵 M)后,切线向量为 v2,法线向量为 n2,假设法线向量的变换矩阵为 G,因此存在以下关系:

img

​ Unity 中线性变换主要有平移、旋转、缩放,由于向量不受平移变换影响,因此,对于法线向量而言,只受旋转和缩放影响。

  • 当 M 只包含旋转变换时,M 是正交矩阵,,因此 G = M;
  • 当 M 只包含统一缩放变换时,M = k·E,因此 G = 1/k·E = 1/(k^2)·M,由于法线向量只需要方向,后面会进行归一化,因此可以简写 G = M;
  • 当 M 只包含旋转变换和统一缩放变换时,G = 1/(k^2)·M,由于法线向量只需要方向,后面会进行归一化,因此可以简写 G = M;

​ Unity 中法线变换源码如下:

​ UnityCG.cginc

// 局部空间->世界空间
float3 UnityObjectToWorldNormal(float3 norm) {
#ifdef UNITY_ASSUME_UNIFORM_SCALING // 统一缩放(x、y、z分量缩放系数一致)
    return UnityObjectToWorldDir(norm); // normalize(mul((float3x3)unity_ObjectToWorld, norm))
#else
    return normalize(mul(norm, (float3x3)unity_WorldToObject)); // mul(IT_M, norm) => mul(norm, I_M)
#endif
}

​ 声明:本文转自【Unity3D】空间和变换

标签:Unity3D,变换,矩阵,空间,z0,坐标系,向量
From: https://www.cnblogs.com/zhyan8/p/17238160.html

相关文章

  • 【Unity3D】激光灯、碰撞特效
    1需求描述​本文将模拟激光灯(或碰撞)特效,详细需求如下:从鼠标位置发射屏幕射线,检测是否与物体发生碰撞当与物体发生碰撞时,在物体表面覆盖一层激光灯(或碰撞)特效​......
  • Opengl ES之矩阵变换(上)
    前言说到矩阵变换,我们第一时间想到的就是大学时代的线性代数这些复杂的东西,突然有了一种令人从入门到放弃的念头,不慌,作为了一个应用层的CV工程师,在实际应用中线性代数哪......
  • 【Unity3D】基于模板测试和顶点膨胀的描边方法
    1前言​选中物体描边特效中介绍了基于模板纹理模糊膨胀的描边方法,该方法实现了软描边,效果较好,但是为了得到模糊纹理,对屏幕像素进行了多次渲染,效率欠佳。本文将介绍......
  • 【Unity3D】基于AssetBundle实现资源热更新
    1前言​Unity3D本地资源一般放在Resources目录下,但是Resouces文件夹的大小不能超过2G,使用AssetBundle管理资源可以解决Resources文件夹受限问题。​......
  • 【Unity3D】半球卷屏特效
    1原理​凸镜贴图和渐变凸镜贴图中介绍了使用OpenGL实现凸镜贴图及其原理,通过顶点坐标映射到纹理坐标,并构造三角形网格,构建了真正的三维凸镜模型。本文通过Shad......
  • 【Unity3D】卷轴特效
    1原理​当一个圆在地面上沿直线匀速滚动时,圆上固定点的运动轨迹称为旋轮线(或摆线、圆滚线)。本文实现的卷轴特效使用了旋轮线相关理论。​以下是卷轴特效原理及......
  • 【Unity3D】水波特效
    1水波特效原理​水波特效属于Unity3D后处理特效,其原理是:对渲染后的纹理进行局部挤压和拉升变换,即对局部uv坐标进行周期性的偏移运动,实现波纹效果。​1)波形......
  • 【Unity3D】选中物体描边特效
    1前言​描边的难点在于如何检测和识别边缘,当前实现描边特效的方法主要有以下几种:​1)基于顶点膨胀的描边方法​在SubShader中开2个Pass渲染通道,第一......
  • 【Unity3D】绘制物体表面三角形网格
    1仅绘制三角形网格​1)创建游戏对象​创建一个空对象,重命名为Grid,并在其下添加需要绘制网格的对象,如下:​场景显示如下:​2)添加脚本组件​Grid......
  • 【Unity3D】绘制物体外框线条盒子
    1需求描述​点选物体、框选物体、绘制外边框中介绍了物体投影到屏幕上的二维外框绘制方法,本文将介绍物体外框线条盒子绘制方法。内框:选中物体后,绘制物体的内框(紧......