跳转至

GAMES101 Computer Graphics

  • https://sites.cs.ucsb.edu/~lingqi/teaching/games101.html
  • https://www.bilibili.com/video/BV1X7411F744

Overview of Computer Graphics (L1)

Review of Linear Algebra (L2)

Transformation - 2D (L3)

逆时针旋转 \(\theta\)

\(R_\theta = \begin{bmatrix} \cos\theta & -\sin\theta \\ \sin\theta & \cos\theta \end{bmatrix}\)

\(R_\theta = R_{-\theta} = R_\theta^T\)

Homogenous Coordinates

  • 2D point \(\begin{pmatrix} x \\ y \\ 1 \end{pmatrix}\)
  • 2D vector \(\begin{pmatrix} x \\ y \\ 0 \end{pmatrix}\)
  • \(\begin{pmatrix} x' \\ y' \\ 1 \end{pmatrix} = \begin{pmatrix} a & b & t_x \\ c & d & t_y \\ 0 & 0 & 1 \end{pmatrix} \cdot \begin{pmatrix} x \\ y \\ 1 \end{pmatrix}\)
  • \(\begin{pmatrix} x \\ y \\ w \end{pmatrix} = \begin{pmatrix} x/w \\ y/w \\ 1 \end{pmatrix}\) 表示一个二维点
  • 两点相加得到中点

变换叠加

\(M \begin{pmatrix} x \\ y \\ 1 \end{pmatrix} = A_n \cdots A_2 A_1 \begin{pmatrix} x \\ y \\ 1 \end{pmatrix}\)

Transformation - 3D (L4)

三维旋转

\(R_{xyz}(\alpha,\beta,\gamma) = R_x(\alpha) R_y(\beta) R_z(\gamma)\)

头:上下看 pitch 左右转 yaw 向日葵歪头 roll

Rodrigues’ Rotation Formula

绕轴 \(\vec{n}\) 旋转 \(\alpha\)\(R(\vec{n}, \alpha) = \cos \alpha \cdot I + (1 - \cos\alpha) \cdot \vec{n} \vec{n}^T+\sin \alpha \cdot \begin{pmatrix} 0 & -n_{z} & n_{y} \\ n_{z} & 0 & -n_{x} \\ -n_{y} & n_{x} & 0 \end{pmatrix}\)

四元数:为了解决旋转矩阵的插值问题

MVP

model view projection

相机

相机:位置 \(\vec{e}\) 正对的方向 \(\hat{g}\) 朝上的方向 \(\hat{t}\)

把相机放到 \((0,0,0)\) 正对 \(-z\) 方向 头顶为 \(y\) 方向

\(M = R \cdot T\) 表示先将相机放到原点 然后把相机的方向旋转到上述方向

\(M = R \cdot T = (R^{-1})^{-1} \cdot T = \begin{bmatrix} x_{\hat{g}\times \hat{t}} & x_t & x_{-g} & 0 \\ y_{\hat{g}\times \hat{t}} & y_t & y_{-g} & 0 \\ z_{\hat{g}\times \hat{t}} & z_t & z_{-g} & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix}^T \cdot \begin{bmatrix} 1 & 0 & 0 & -x_e \\ 0 & 1 & 0 & -y_e \\ 0 & 0 & 1 & -z_e \\ 0 & 0 & 0 & 1 \end{bmatrix}\)

正交投影

将原空间放到原点 再在三个轴上缩放到 \([-1, 1]\)

top/bottom n(靠前的面 成像位置)/f(靠后的面 实际位置) left/right

\(M=\begin{bmatrix} \frac{2}{r-l} & 0 & 0 & 0 \\ 0 & \frac{2}{t-b} & 0 & 0 \\ 0 & 0 & \frac{2}{n-f} & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix}\begin{bmatrix} 1 & 0 & 0 & -\frac{r+l}{2} \\ 0 & 1 & 0 & -\frac{t+b}{2} \\ 0 & 0 & 1 & -\frac{n+f}{2} \\ 0 & 0 & 0 & 1 \end{bmatrix}\)

透视投影

先将平面缩放到与期望的画布同大小 类似相似三角形 然后\(z\)对应的矩阵行用待定系数法确定 \(M = \begin{bmatrix} n & 0 & 0 & 0 \\ 0 & n & 0 & 0 \\ 0 & 0 & n+f & -nf \\ 0 & 0 & 1 & 0 \end{bmatrix}\)

做完这一步之后再使用正交投影将\(f\)处的画面拉到\(n\)处即可

Rasterization - Triangles (L5)

MVP做完之后我们会得到一个 \([-1,1]^3\) 的规范立方体

首先将这个立方体变换到 \([0,width],[0,height],[-1,1]\)

\(M = \begin{bmatrix} w/2 & 0 & 0 & w/2 \\ 0 & h/2 & 0 & h/2 \\ 0 & 0 & 1 & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix}\)

然后要将三角形变到像素(这里暂时不考虑深度)

for pixel in image {
    pixel.toRasterize = isInside(triangle, pixel.centerX, pixel.centerY)
}

isInside()的实现:判断三角形三边与要判断的点的关系

\(\overrightarrow{AP} \times \overrightarrow{AB},\overrightarrow{BP} \times \overrightarrow{BC},\overrightarrow{CP} \times \overrightarrow{CA}\) 的结果同向

加速光栅化:不对所有的像素遍历 只遍历可能的像素

Rasterization - Antialiasing (L6)

走样 - alias:以同一种频率采样两种不同频率的信号得到相同的结果

不希望看到的结果 - artifacts

aliasing artifacts 采样的速度跟不上信号变化的速度

解决方案

  • 提高屏幕分辨率
  • 先对三角形做模糊将高频分量抑制 然后再采样(回忆时域卷积-频率相乘)
  • MSAA(multi-sample) 将像素分割为小像素然后计算出颜色比例
  • FXAA(fast-approximate) 对有锯齿的图形作后期处理
  • TAA(temporal) 利用两帧之间的运动信息

其他:超分辨率 用深度学习模型来猜缺失的采样点

Rasterization - Z-Buffer (L7)

画家算法

  • 先画远处的东西 将近处的东西依次覆盖其上
  • 对于三角形不适用

Z-Buffer

  • 并行处理所有的三角形 光栅化后得到像素和深度
  • 得到一个像素的深度后 和depth buffer存储的现有最小深度作比较 如果近的话 就将像素的颜色覆盖到frame buffer

Shading - Illumination (L7 L8)

  • shading - 着色/上色 主要考虑的是一个小三角形面的材质对光线的反射 是局部的
    • shading point 小面的中心点
    • \(\hat{n}\) 小面的法向量 注意面有厚度/有里外
    • \(\hat{l}\) 光线入射的方向
    • \(\hat{v}\) camera的方向
  • shadow - 阴影 考虑的是物体间的遮挡关系

shading Blinn-Phong

  • \(L = L_s + L_d + L_a\)
  • specular highlights - 高光 反射
    • 经验判断:出射光和观察方向差不多 也即入射光和观察方向的平均与法向差不多
    • \(L_s = k_s \frac{I}{r^2} \max (0,\hat{n}\cdot\hat{h})\)
      • \(L_s\) 高光反射的光强度
      • \(k_s\) 高光系数
      • \(\hat{h} = \frac{\hat{v}+\hat{l}}{2}\) 半程向量
      • \(p\) p越大 高光越明显 光点越小 一般取64/128
  • diffuse reflection - 漫反射 光线入射后从半球均匀射出
    • \(L_d = k_d \frac{I}{r^2} \max(0,\hat{n}\cdot\hat{l})\)
      • \(L_d\) 漫反射的光强度
      • \(k_d\) 漫反射系数 越大表示漫反射越强 也就是越亮
      • \(\frac{I}{r^2}\) 点光源发出的光强平方衰减
      • \(\cos\theta = \hat{n}\cdot\hat{l}\) 入射光线越倾斜 小面得到光强就越小
      • \(\max(0,\hat{n}\cdot\hat{l})\) 考虑光线从物体后方入射 这时没有反射
  • ambient lighting - 环境光 假定是常数
    • \(L_a = k_a I_a\)

Shading - Frequencies (L8)

  • Flat shading - shade each triangle
    • 选取三角形的法向
  • Gouraud shading - shade each vertex
    • 用顶点周围的三角形面的法向做平均得到方向
  • Phong shading - shade each pixel
    • 用重心坐标差值得到

Shading - Graphics (Real-time Rendering) Pipeline (L8)

  • Application
  • Vertex Processing (Programmable: for each vertex)
  • Triancle Processing
  • Rasterization
  • Fragment Processing (Programmable: for each pixel/fragment)
  • Framebuffer Operations
  • Display

在线编写shader https://www.shadertoy.com

Shading - Texture Mapping (L8 L9)

  • 目的:将屏幕上的一个像素着色
  • 过程:
    • 找到屏幕上的像素中心对应的三维三角形 (\(x_sy_s \to xyz\))
    • 找到这个像素中心在三角形的重心坐标系中的位置 (\(xyz \to \alpha\beta\gamma\))
    • 用材质上面的像素点的颜色去插值得到需要显示的颜色 (\(\alpha\beta\gamma \to uv\)) 如果材质的三角形顶点上有属性 也可以插值得到屏幕上那一点的属性
      • 颜色插值方法
        • Nearest 点在材质的那个像素里面 就用这个像素的颜色
        • Bilinear 点所在位置四周的像素的颜色做一个位置平均(先上下 再左右)

遇到的问题

  • 采样走样:远处的像素看不出纹理
    • 发生原因:远处屏幕上一个像素应该用纹理的一个很大的区域来平均 而不是直接取一个点附近的四个像素平均 这样不足以代表这个像素
    • 解决方案:使用Mipmap
      • 提前对纹理进行正方形区域的平均值计算 存储多占用33% 但是查询非常快
      • 在查询时可以查询两层之间的值 只要做插值就行
  • Mipmap斜的"像素"看起来很奇怪
    • 原因:Mipmap只支持正方形查询
    • 解决方案:各向异性过滤 Anisotropic Filtering
      • 在纹理的存储图中存储矩形 存储量增大
      • EWA filtering 使用椭圆切线处的纹理采样

Barycentric Coordinates

点在三角形内部的话 \(0 < \alpha,\beta,\gamma < 1\)

\(\begin{aligned} \alpha &=\frac{-\left(x-x_{B}\right)\left(y_{C}-y_{B}\right)+\left(y-y_{B}\right)\left(x_{C}-x_{B}\right)}{-\left(x_{A}-x_{B}\right)\left(y_{C}-y_{B}\right)+\left(y_{A}-y_{B}\right)\left(x_{C}-x_{B}\right)} \\ \beta &=\frac{-\left(x-x_{C}\right)\left(y_{A}-y_{C}\right)+\left(y-y_{C}\right)\left(x_{A}-x_{C}\right)}{-\left(x_{B}-x_{C}\right)\left(y_{A}-y_{C}\right)+\left(y_{B}-y_{C}\right)\left(x_{A}-x_{C}\right)} \\ \gamma &=1-\alpha-\beta \end{aligned}\)

Application of Textures

将环境光记录在球面上