首页 > 其他分享 >WebGL系列教程七(二维及三维旋转、平移、缩放)

WebGL系列教程七(二维及三维旋转、平移、缩放)

时间:2024-09-13 23:51:05浏览次数:16  
标签:cos 1.0 缩放 WebGL 教程 旋转 bmatrix theta sin

目录

1 前言

  上一篇我们讲了如何进行立方体的纹理贴图,为了方便立方体的贴图后效果的展示,我们对立方体进行了旋转,但代码中并没有进行体现,这一节我们来讲讲旋转、平移、缩放。实际上,在WebGL中进行旋转、平移、缩放只需要乘以相应的旋转、平移、缩放矩阵即可。

2 二维

  在二维空间中,平移、旋转和缩放是常见的几何变换。为了系统地表示这些变换,我们通常使用齐次坐标,即将二维坐标 ( x , y ) (x, y) (x,y) 扩展为三维齐次坐标 ( x , y , 1 ) (x, y, 1) (x,y,1)。这样可以统一表示线性变换和平移变换。下面分别推导这三种变换的矩阵表示。

2.1 平移

在这里插入图片描述

定义:将任意一个点 ( x , y ) (x, y) (x,y) 平移 ( t x , t y ) (t_x, t_y) (tx​,ty​),得到新点 ( x ′ , y ′ ) (x', y') (x′,y′),即

{ x ′ = x + t x y ′ = y + t y \begin{cases} x' = x + t_x \\ y' = y + t_y \end{cases} {x′=x+tx​y′=y+ty​​

齐次坐标表示
[ x ′ y ′ 1 ] = [ 1 0 t x 0 1 t y 0 0 1 ] [ x y 1 ] \begin{bmatrix} x' \\ y' \\ 1 \end{bmatrix}=\begin{bmatrix} 1 & 0 & t_x \\ 0 & 1 & t_y \\ 0 & 0 & 1 \end{bmatrix}\begin{bmatrix} x \\ y \\ 1 \end{bmatrix} ​x′y′1​ ​= ​100​010​tx​ty​1​ ​xy1​

因此,二维平移矩阵为:

T = [ 1 0 t x 0 1 t y 0 0 1 ] T = \begin{bmatrix} 1 & 0 & t_x \\ 0 & 1 & t_y \\ 0 & 0 & 1 \end{bmatrix} T= ​100​010​tx​ty​1​

2.2 旋转

在这里插入图片描述

定义:将任意一个点 ( x , y ) (x, y) (x,y) 绕原点逆时针旋转角度 θ \theta θ,得到新点 ( x ′ , y ′ ) (x', y') (x′,y′),即

{ x ′ = x cos ⁡ θ − y sin ⁡ θ y ′ = x sin ⁡ θ + y cos ⁡ θ \begin{cases} x' = x \cos\theta - y \sin\theta \\ y' = x \sin\theta + y \cos\theta \end{cases} {x′=xcosθ−ysinθy′=xsinθ+ycosθ​

齐次坐标表示

[ x ′ y ′ 1 ] = [ cos ⁡ θ − sin ⁡ θ 0 sin ⁡ θ cos ⁡ θ 0 0 0 1 ] [ x y 1 ] \begin{bmatrix} x' \\ y' \\ 1 \end{bmatrix} =\begin{bmatrix} \cos\theta & -\sin\theta & 0 \\ \sin\theta & \cos\theta & 0 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x \\ y \\ 1 \end{bmatrix} ​x′y′1​ ​= ​cosθsinθ0​−sinθcosθ0​001​ ​xy1​
因此,二维旋转矩阵为:

R ( θ ) = [ cos ⁡ θ − sin ⁡ θ 0 sin ⁡ θ cos ⁡ θ 0 0 0 1 ] R(\theta) = \begin{bmatrix} \cos\theta & -\sin\theta & 0 \\ \sin\theta & \cos\theta & 0 \\ 0 & 0 & 1 \end{bmatrix} R(θ)= ​cosθsinθ0​−sinθcosθ0​001​

2.3 缩放

在这里插入图片描述

定义:将任意一个点 ( x , y ) (x, y) (x,y) 沿 x x x 轴缩放因子 s x s_x sx​,沿 y y y 轴缩放因子 s y s_y sy​,得到新点 ( x ′ , y ′ ) (x', y') (x′,y′),即

{ x ′ = s x ⋅ x y ′ = s y ⋅ y \begin{cases} x' = s_x \cdot x \\ y' = s_y \cdot y \end{cases} {x′=sx​⋅xy′=sy​⋅y​

齐次坐标表示

[ x ′ y ′ 1 ] = [ s x 0 0 0 s y 0 0 0 1 ] [ x y 1 ] \begin{bmatrix} x' \\ y' \\ 1 \end{bmatrix} =\begin{bmatrix} s_x & 0 & 0 \\ 0 & s_y & 0 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x \\ y \\ 1 \end{bmatrix} ​x′y′1​ ​= ​sx​00​0sy​0​001​ ​xy1​

因此,二维缩放矩阵为:

S = [ s x 0 0 0 s y 0 0 0 1 ] S = \begin{bmatrix} s_x & 0 & 0 \\ 0 & s_y & 0 \\ 0 & 0 & 1 \end{bmatrix} S= ​sx​00​0sy​0​001​

3 三维

  在三维空间中,平移、旋转和缩放是类似的,我们通常也使用齐次坐标,即将三维坐标 ( x , y , z ) (x, y, z) (x,y,z) 扩展为四维齐次坐标 ( x , y , z , 1 ) (x, y, z, 1) (x,y,z,1)。这样可以统一表示线性变换和平移变换。下面分别推导这三种变换的矩阵表示。

3.1 平移

定义:将任意一个点 ( x , y , z ) (x, y, z) (x,y,z) 平移 ( t x , t y , t z ) (t_x, t_y, t_z) (tx​,ty​,tz​),得到新点 ( x ′ , y ′ , z ′ ) (x', y', z') (x′,y′,z′),即

{ x ′ = x + t x y ′ = y + t y z ′ = z + t z \begin{cases} x' = x + t_x \\ y' = y + t_y \\ z' = z + t_z \end{cases} ⎩ ⎧​x′=x+tx​y′=y+ty​z′=z+tz​​

齐次坐标表示

[ x ′ y ′ z ′ 1 ] = [ 1 0 0 t x 0 1 0 t y 0 0 1 t z 0 0 0 1 ] [ x y z 1 ] \begin{bmatrix} x' \\ y' \\ z' \\ 1 \end{bmatrix}=\begin{bmatrix} 1 & 0 & 0 & t_x \\ 0 & 1 & 0 & t_y \\ 0 & 0 & 1 & t_z \\ 0 & 0 & 0 & 1 \end{bmatrix}\begin{bmatrix} x \\ y \\ z \\ 1 \end{bmatrix} ​x′y′z′1​ ​= ​1000​0100​0010​tx​ty​tz​1​ ​xyz1​

因此,三维平移矩阵为:
T = [ 1 0 0 t x 0 1 0 t y 0 0 1 t z 0 0 0 1 ] T = \begin{bmatrix} 1 & 0 & 0 & t_x \\ 0 & 1 & 0 & t_y \\ 0 & 0 & 1 & t_z \\ 0 & 0 & 0 & 1 \end{bmatrix} T= ​1000​0100​0010​tx​ty​tz​1​

3.2 旋转

在三维空间中,旋转可以绕任意一个坐标轴进行。下面分别给出绕 X X X 轴、 Y Y Y 轴和 Z Z Z 轴旋转角度 θ \theta θ 的旋转矩阵。在三维空间中的旋转看着复杂其实很简单,因为绕X轴旋转其实X的值并不会产生变化,只有Y和Z的值会产生变化,那不就相当于在二维空间中的旋转吗?是的,我们依次来讨论。
在这里插入图片描述

3.2.1 绕 X X X 轴旋转

定义:将任意一个点 ( x , y , z ) (x, y, z) (x,y,z) 绕 X X X 轴逆时针旋转角度 θ \theta θ,得到新点 ( x ′ , y ′ , z ′ ) (x', y', z') (x′,y′,z′),x的大小其实是不动的,y和z的值和二维平面类似,即

{ x ′ = x y ′ = y cos ⁡ θ − z sin ⁡ θ z ′ = y sin ⁡ θ + z cos ⁡ θ \begin{cases} x' = x \\ y' = y \cos\theta - z \sin\theta \\ z' = y \sin\theta + z \cos\theta \end{cases} ⎩ ⎧​x′=xy′=ycosθ−zsinθz′=ysinθ+zcosθ​

齐次坐标表示

R x ( θ ) = [ 1 0 0 0 0 cos ⁡ θ − sin ⁡ θ 0 0 sin ⁡ θ cos ⁡ θ 0 0 0 0 1 ] R_x(\theta) = \begin{bmatrix} 1 & 0 & 0 & 0 \\ 0 & \cos\theta & -\sin\theta & 0 \\ 0 & \sin\theta & \cos\theta & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix} Rx​(θ)= ​1000​0cosθsinθ0​0−sinθcosθ0​0001​

3.2.2 绕 Y Y Y 轴旋转

定义:将任意一个点 ( x , y , z ) (x, y, z) (x,y,z) 绕 Y Y Y 轴逆时针旋转角度 θ \theta θ,得到新点 ( x ′ , y ′ , z ′ ) (x', y', z') (x′,y′,z′),y值也是不动的,x和z的值与二维平面类似,即

{ x ′ = x cos ⁡ θ + z sin ⁡ θ y ′ = y z ′ = − x sin ⁡ θ + z cos ⁡ θ \begin{cases} x' = x \cos\theta + z \sin\theta \\ y' = y \\ z' = -x \sin\theta + z \cos\theta \end{cases} ⎩ ⎧​x′=xcosθ+zsinθy′=yz′=−xsinθ+zcosθ​

齐次坐标表示

R y ( θ ) = [ cos ⁡ θ 0 sin ⁡ θ 0 0 1 0 0 − sin ⁡ θ 0 cos ⁡ θ 0 0 0 0 1 ] R_y(\theta) = \begin{bmatrix} \cos\theta & 0 & \sin\theta & 0 \\ 0 & 1 & 0 & 0 \\ -\sin\theta & 0 & \cos\theta & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix} Ry​(θ)= ​cosθ0−sinθ0​0100​sinθ0cosθ0​0001​

3.2.3 绕 Z Z Z 轴旋转

定义:将一个点 ( x , y , z ) (x, y, z) (x,y,z) 绕 Z Z Z 轴逆时针旋转角度 θ \theta θ,得到新点 ( x ′ , y ′ , z ′ ) (x', y', z') (x′,y′,z′),z值是不变的,x和y的值与二维平面类似,即

{ x ′ = x cos ⁡ θ − y sin ⁡ θ y ′ = x sin ⁡ θ + y cos ⁡ θ z ′ = z \begin{cases} x' = x \cos\theta - y \sin\theta \\ y' = x \sin\theta + y \cos\theta \\ z' = z \end{cases} ⎩ ⎧​x′=xcosθ−ysinθy′=xsinθ+ycosθz′=z​

齐次坐标表示

R z ( θ ) = [ cos ⁡ θ − sin ⁡ θ 0 0 sin ⁡ θ cos ⁡ θ 0 0 0 0 1 0 0 0 0 1 ] R_z(\theta) = \begin{bmatrix} \cos\theta & -\sin\theta & 0 & 0 \\ \sin\theta & \cos\theta & 0 & 0 \\ 0 & 0 & 1 & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix} Rz​(θ)= ​cosθsinθ00​−sinθcosθ00​0010​0001​

因此,三维旋转矩阵可以根据旋转轴选择对应的 R x ( θ ) R_x(\theta) Rx​(θ)、 R y ( θ ) R_y(\theta) Ry​(θ) 或 R z ( θ ) R_z(\theta) Rz​(θ)。

3.2.4 绕任意轴旋转

  怎么表示任意轴呢?用一个向量便可以表示任意轴。那怎么绕任意轴旋转呢?答案是先将物体平移到原点去,分解为分别绕XYZ的旋转,而绕这三个轴旋转的角度就被叫做欧拉角,旋转完毕后再平移到原来的地方就可以了。这个思想也就是著名的罗德里格斯旋转公式所采用的思想。

  设旋转轴为单位向量 u = ( u x , u y , u z ) \mathbf{u} = (u_x, u_y, u_z) u=(ux​,uy​,uz​),旋转角度为 θ \theta θ。为了确保计算的准确性,旋转轴必须为单位向量,即:

u x 2 + u y 2 + u z 2 = 1 u_x^2 + u_y^2 + u_z^2 = 1 ux2​+uy2​+uz2​=1
罗德里格斯旋转公式提供了一种通过旋转轴和旋转角度来计算旋转矩阵的方法。旋转矩阵 R \mathbf{R} R 可以表示为:

R = I cos ⁡ θ + ( 1 − cos ⁡ θ ) u u T + K sin ⁡ θ \mathbf{R} = \mathbf{I} \cos\theta + (1 - \cos\theta) \mathbf{u} \mathbf{u}^T + \mathbf{K} \sin\theta R=Icosθ+(1−cosθ)uuT+Ksinθ

其中:

  • I \mathbf{I} I 是 3 × 3 3 \times 3 3×3 的单位矩阵。
  • u u T \mathbf{u} \mathbf{u}^T uuT 是旋转轴的外积矩阵。
  • K \mathbf{K} K 是旋转轴的反对称矩阵,定义为:

K = [ 0 − u z u y u z 0 − u x − u y u x 0 ] \mathbf{K} = \begin{bmatrix} 0 & -u_z & u_y \\ u_z & 0 & -u_x \\ -u_y & u_x & 0 \end{bmatrix} K= ​0uz​−uy​​−uz​0ux​​uy​−ux​0​
结合上述推导,绕任意单位向量 u = ( u x , u y , u z ) \mathbf{u} = (u_x, u_y, u_z) u=(ux​,uy​,uz​) 旋转角度 θ \theta θ 的旋转矩阵 R \mathbf{R} R 为:

R = [ cos ⁡ θ + u x 2 ( 1 − cos ⁡ θ ) u x u y ( 1 − cos ⁡ θ ) − u z sin ⁡ θ u x u z ( 1 − cos ⁡ θ ) + u y sin ⁡ θ u y u x ( 1 − cos ⁡ θ ) + u z sin ⁡ θ cos ⁡ θ + u y 2 ( 1 − cos ⁡ θ ) u y u z ( 1 − cos ⁡ θ ) − u x sin ⁡ θ u z u x ( 1 − cos ⁡ θ ) − u y sin ⁡ θ u z u y ( 1 − cos ⁡ θ ) + u x sin ⁡ θ cos ⁡ θ + u z 2 ( 1 − cos ⁡ θ ) ] \mathbf{R} = \begin{bmatrix} \cos\theta + u_x^2 (1 - \cos\theta) & u_x u_y (1 - \cos\theta) - u_z \sin\theta & u_x u_z (1 - \cos\theta) + u_y \sin\theta \\ u_y u_x (1 - \cos\theta) + u_z \sin\theta & \cos\theta + u_y^2 (1 - \cos\theta) & u_y u_z (1 - \cos\theta) - u_x \sin\theta \\ u_z u_x (1 - \cos\theta) - u_y \sin\theta & u_z u_y (1 - \cos\theta) + u_x \sin\theta & \cos\theta + u_z^2 (1 - \cos\theta) \end{bmatrix} R= ​cosθ+ux2​(1−cosθ)uy​ux​(1−cosθ)+uz​sinθuz​ux​(1−cosθ)−uy​sinθ​ux​uy​(1−cosθ)−uz​sinθcosθ+uy2​(1−cosθ)uz​uy​(1−cosθ)+ux​sinθ​ux​uz​(1−cosθ)+uy​sinθuy​uz​(1−cosθ)−ux​sinθcosθ+uz2​(1−cosθ)​

3.3 缩放

定义:将一个点 ( x , y , z ) (x, y, z) (x,y,z) 沿 x x x 轴缩放因子 s x s_x sx​,沿 y y y 轴缩放因子 s y s_y sy​,沿 z z z 轴缩放因子 s z s_z sz​,得到新点 ( x ′ , y ′ , z ′ ) (x', y', z') (x′,y′,z′),即

{ x ′ = s x ⋅ x y ′ = s y ⋅ y z ′ = s z ⋅ z \begin{cases} x' = s_x \cdot x \\ y' = s_y \cdot y \\ z' = s_z \cdot z \end{cases} ⎩ ⎧​x′=sx​⋅xy′=sy​⋅yz′=sz​⋅z​

齐次坐标表示

S = [ s x 0 0 0 0 s y 0 0 0 0 s z 0 0 0 0 1 ] S = \begin{bmatrix} s_x & 0 & 0 & 0 \\ 0 & s_y & 0 & 0 \\ 0 & 0 & s_z & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix} S= ​sx​000​0sy​00​00sz​0​0001​

4 WebGL中怎么实现旋转、平移、缩放

  OK.终于讲完理论了,现在我们来讲下实操。通过上面的讲解,大家应该明白了,要旋转一个立方体,只要把立方体中的每一个点都乘以一个旋转矩阵,就可以了。类似的,要实现平移,只要把立方体中的每一个点都乘以一个平移矩阵就可以了。我们以旋转立方体为例。

4.1 声明顶点着色器和片元着色器

<script id="vertex-shader" type="x-shader/x-vertex">
    attribute vec4 a_Position;
    attribute vec4 a_Color;
    uniform mat4 u_RorateMatrix;
    varying vec4 v_Color;
    void main(){
    	//给所有点都乘以一个旋转矩阵
        gl_Position =  uRotateMatrix * a_Position;
        v_Color = a_Color;
        v_TexCoord = a_TexCoord;
    }
</script>
<script id="fragment-shader" type="x-shader/x-fragment">
    precision highp float;
    varying vec4 v_Color;
    void main(){
        gl_FragColor = v_Color;
    }
</script>

4.2 计算旋转矩阵

  计算旋转矩阵的类库太多了,比如gl-Matrix、Cesiumjs、Threejs、Babylonjs都可以实现。本文中使用的是WebGL编程指南中自带的方法,只要调用时,指明要旋转的角度和要绕哪个轴旋转即可。

var modelMatrix = new Matrix4();
//绕Y轴【0,1,0】旋转60度
modelMatrix.setRotate(60,0,1,0);
//赋值给u_RorateMatrix
var u_RorateMatrix= gl.getUniformLocation(program, 'u_RorateMatrix');
//false表示不使用归一化
gl.uniformMatrix4fv(u_RorateMatrix,false,modelMatrix.elements);

旋转方法源码如下:

/**
 * Set the matrix for rotation.
 * The vector of rotation axis may not be normalized.
 * @param angle The angle of rotation (degrees)
 * @param x The X coordinate of vector of rotation axis.
 * @param y The Y coordinate of vector of rotation axis.
 * @param z The Z coordinate of vector of rotation axis.
 * @return this
 */
Matrix4.prototype.setRotate = function(angle, x, y, z) {
  var e, s, c, len, rlen, nc, xy, yz, zx, xs, ys, zs;

  angle = Math.PI * angle / 180;
  e = this.elements;

  s = Math.sin(angle);
  c = Math.cos(angle);

  if (0 !== x && 0 === y && 0 === z) {
    // Rotation around X axis
    if (x < 0) {
      s = -s;
    }
    e[0] = 1;  e[4] = 0;  e[ 8] = 0;  e[12] = 0;
    e[1] = 0;  e[5] = c;  e[ 9] =-s;  e[13] = 0;
    e[2] = 0;  e[6] = s;  e[10] = c;  e[14] = 0;
    e[3] = 0;  e[7] = 0;  e[11] = 0;  e[15] = 1;
  } else if (0 === x && 0 !== y && 0 === z) {
    // Rotation around Y axis
    if (y < 0) {
      s = -s;
    }
    e[0] = c;  e[4] = 0;  e[ 8] = s;  e[12] = 0;
    e[1] = 0;  e[5] = 1;  e[ 9] = 0;  e[13] = 0;
    e[2] =-s;  e[6] = 0;  e[10] = c;  e[14] = 0;
    e[3] = 0;  e[7] = 0;  e[11] = 0;  e[15] = 1;
  } else if (0 === x && 0 === y && 0 !== z) {
    // Rotation around Z axis
    if (z < 0) {
      s = -s;
    }
    e[0] = c;  e[4] =-s;  e[ 8] = 0;  e[12] = 0;
    e[1] = s;  e[5] = c;  e[ 9] = 0;  e[13] = 0;
    e[2] = 0;  e[6] = 0;  e[10] = 1;  e[14] = 0;
    e[3] = 0;  e[7] = 0;  e[11] = 0;  e[15] = 1;
  } else {
    // Rotation around another axis
    len = Math.sqrt(x*x + y*y + z*z);
    if (len !== 1) {
      rlen = 1 / len;
      x *= rlen;
      y *= rlen;
      z *= rlen;
    }
    nc = 1 - c;
    xy = x * y;
    yz = y * z;
    zx = z * x;
    xs = x * s;
    ys = y * s;
    zs = z * s;

    e[ 0] = x*x*nc +  c;
    e[ 1] = xy *nc + zs;
    e[ 2] = zx *nc - ys;
    e[ 3] = 0;

    e[ 4] = xy *nc - zs;
    e[ 5] = y*y*nc +  c;
    e[ 6] = yz *nc + xs;
    e[ 7] = 0;

    e[ 8] = zx *nc + ys;
    e[ 9] = yz *nc - xs;
    e[10] = z*z*nc +  c;
    e[11] = 0;

    e[12] = 0;
    e[13] = 0;
    e[14] = 0;
    e[15] = 1;
  }

  return this;
};

4.3 绘制立方体并进行旋转完整代码

const verticesColors = new Float32Array([
    // 前面
    -1.0, -1.0,  1.0,     1.0, 0.0,1.0,//v2 红色
     1.0, -1.0,  1.0,     1.0, 0.0,1.0,//v3 红色
     1.0,  1.0,  1.0,     1.0, 0.0,1.0,//v0 红色
    -1.0,  1.0,  1.0,     1.0, 0.0,1.0,//v1 红色

    // 后面
    -1.0, -1.0, -1.0,     0.0, 1.0, 0.0,//v5 绿色
     1.0, -1.0, -1.0,     0.0, 1.0, 0.0,//v4 绿色
     1.0,  1.0, -1.0,     0.0, 1.0, 0.0,//v7 绿色
    -1.0,  1.0, -1.0,     0.0, 1.0, 0.0,//v6 绿色

    // 上面
    -1.0,  1.0,  1.0,     0.0, 0.0,1.0,//v1 蓝色
     1.0,  1.0,  1.0,     0.0, 0.0,1.0,//v0 蓝色
     1.0,  1.0, -1.0,     0.0, 0.0,1.0,//v7 蓝色
    -1.0,  1.0, -1.0,     0.0, 0.0,1.0,//v6 蓝色

    // 下面
    -1.0, -1.0,  1.0,     0.0, 0.0,0.0,//v2 黑色
     1.0, -1.0,  1.0,     0.0, 0.0,0.0,//v3 黑色
     1.0, -1.0, -1.0,     0.0, 0.0,0.0,//v4 黑色
    -1.0, -1.0, -1.0,     0.0, 0.0,0.0,//v5 黑色

    // 左面
    -1.0, -1.0, -1.0,     0.0, 1.0,1.0,//v5 青色
    -1.0, -1.0,  1.0,     0.0, 1.0,1.0,//v2 青色
    -1.0,  1.0,  1.0,     0.0, 1.0,1.0,//v1 青色
    -1.0,  1.0, -1.0,     0.0, 1.0,1.0,//v6 青色

    // 右面
     1.0, -1.0,  1.0,     1.0, 1.0,1.0,//v3 白色
     1.0, -1.0, -1.0,     1.0, 1.0,1.0,//v4 白色
     1.0,  1.0, -1.0,     1.0, 1.0,1.0,//v7 白色
     1.0,  1.0,  1.0,     1.0, 1.0,1.0,//v0 白色
]);
const indices = new Uint8Array([
    0, 1, 2, 0, 2, 3, // 前面
    4, 5, 6, 4, 6, 7, // 后面
    8, 9, 10, 8, 10, 11, // 上面
    12, 13, 14, 12, 14, 15, // 下面
    16, 17, 18, 16, 18, 19, // 左面
    20, 21, 22, 20, 22, 23  // 右面
]);


//使webgl视口和canvas画板一样大
gl.viewport(0, 0, canvas.width, canvas.height);
//开启深度检测
gl.enable(gl.DEPTH_TEST);
//顶点
let vertexColorBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER,vertexColorBuffer);
gl.bufferData(gl.ARRAY_BUFFER,verticesColors,gl.STATIC_DRAW);
let FSIZE = verticesColors.BYTES_PER_ELEMENT;
let a_Position = gl.getAttribLocation(program,'a_Position');
gl.vertexAttribPointer(a_Position,3,gl.FLOAT,false,FSIZE*6,0);
gl.enableVertexAttribArray(a_Position);
//颜色
var a_Color = gl.getAttribLocation(program, 'a_Color');
gl.vertexAttribPointer(a_Color, 3, gl.FLOAT, false, FSIZE * 6, FSIZE * 3);
gl.enableVertexAttribArray(a_Color);
//旋转
var modelMatrix = new Matrix4();
modelMatrix.setRotate(60,0,1,0);

//赋值给uRotateMatrix
var u_RorateMatrix= gl.getUniformLocation(program, 'u_RorateMatrix');
gl.uniformMatrix4fv(u_RorateMatrix,false,modelMatrix .elements);
//绑定索引缓冲
let indexBuffer =  gl.createBuffer();
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER,indexBuffer);
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER,indices,gl.STATIC_DRAW);
gl.drawElements(gl.TRIANGLES, 36, gl.UNSIGNED_BYTE, 0);

4.4 效果

在这里插入图片描述

5 总结

  本文我们通过梳理二维及三维的旋转、平移、缩放矩阵,搞明白了旋转、平移、缩放对应的矩阵,并在WebGL中进行了实验操作,实现了一个立方体的旋转。本文在理解上有一定的难度,尤其是矩阵的变换,需要读者具有一定的线性代数知识,希望读者仔细体会,回见~

标签:cos,1.0,缩放,WebGL,教程,旋转,bmatrix,theta,sin
From: https://blog.csdn.net/xiangshangdemayi/article/details/142212154

相关文章

  • Java教程:SE进阶【十万字详解】(上)
    ✨博客主页:https://blog.csdn.net/m0_63815035?type=blog......
  • Angular Material 18+ 高级教程 – Datepicker の Calendar & Custom DateAdapter (Te
    前言本篇只会教AngularMaterialDatepicker里最关键的组件--Calendar组件。还有如何自定义DateAdapter,让Calendar支持 TC39Temporal。有兴趣想学完整Datepicker的人,请移步官网。只对Calendar组件和自定义DateAdapter感兴趣的,留下!Let'sstart......
  • Java学习全套视频教程网盘资源下载
    如大家所了解的,Java是一门面向对象编程语言,可以编写桌面应用程序、Web应用程序、分布式系统和嵌入式系统应用程序。Java之所以可以让计算机识别,是因为Java的编译器和Java虚拟机,它会将人按照一定规范写成的信息翻译成机器所能识别二进制的信息,这样就可以让我们很好的和计算机进......
  • 【HBuilderX-从下载到项目创建】编程初学者适用的HBuilderX开发环境(超详细的)下载安装
    简介:HBuilderX是一款由DCloud公司开发的集成开发环境(IDE),专为前端开发设计,同时也支持多平台应用开发。它支持HTML、CSS、JavaScript、Vue、React、Uni-app等多种编程语言和框架,具备代码编辑、调试、测试等功能,并且提供了丰富的插件生态系统以扩展其功能。“......
  • Cursor超详细使用教程,从安装、订阅到高级技巧的全面教程-解锁 Cursor 的全部功能
    最近,编程圈子里大家都在讨论一个新工具——Cursor。朋友们都在说:“Cursor真是太棒了,我的编程效率飞速提升!”还有人说:“我彻底被Cursor给圈粉了,现在离不开它了!”看到大家对Cursor这么热衷,我也忍不住试用了两天。不出意外,我也被它的强大功能深深吸引了。Cursor确实是一......
  • 全网最适合入门的面向对象编程教程:47 Python函数方法与接口-回调函数Callback
    全网最适合入门的面向对象编程教程:47Python函数方法与接口-回调函数Callback摘要:回调函数是编程中一种非常常见的模式,用于将函数作为参数传递给其他函数或方法。这种模式在Python中广泛应用于事件处理、异步编程、函数式编程等场景。原文链接:FreakStudio的博客往......
  • 十年电商经验分享:从0-1打单品保姆级教程(下)
    接着上一篇《十年电商经验分享:从0-1打单品保姆级教程(上)》,各位觉得有参考意义的商家朋友们可以点赞收藏一下。5、制作sku图片以及sku名称+卖点这里很多商家可能不太注意,这个也是优化转化率最好的一块内容,因为从正常顾客购买流程来看,sku是一个必选的内容,你sku,图片卖点的表......
  • Stable Diffusion|揭秘小红书疯狂涨粉爆款治愈壁纸,保姆级制作教程!
    在这个日新月异的数字时代,AI技术如同一股不可阻挡的潮流,正以前所未有的速度改变着我们的生活方式,也深刻影响着艺术创作的边界。今天,我们推出一系列AI治愈系壁纸,它们不仅是技术进步的结晶,更是科技与美学完美融合的产物。这些治愈系插画壁纸,点赞、评论、收藏数据非常好,可实现......