翼度科技»论坛 编程开发 JavaScript 查看内容

webgl 系列 —— 三角形

5

主题

5

帖子

15

积分

新手上路

Rank: 1

积分
15
三角形

有人说三维模型的基本单元是三角形。比如复杂的游戏角色,也只是用许多三角形画出来的。
不管上述说法是否属实,本篇先把三角形画出来。
如何绘制一个三角形

鼠标点击绘点示例我们写了这样的代码:
  1. points.forEach(item => {
  2.     gl.vertexAttrib3f(a_Position, item.x, item.y, 0.0);
  3.     gl.drawArrays(gl.POINTS, 0, 1);
  4. })
复制代码

这种方法一次只能绘制一个点。
比如需要绘制一个三角形,应该是一个连贯的动作。比如在顶点着色器中一次性画三个点,然后用线连接;而不是绘制一个点,在绘制一个点,在绘制一个点...,不应该是零散的
Tip:提前透露 - 只要一次性将三个点绘制出来,其实三角形也就画出来了。
一次性绘制多个顶点可以使用缓冲区对象。
缓冲区对象

可以使用 webgl 中的缓冲区对象(buffer object) 一次性的向着色器传入多个顶点数据。
使用缓冲区对象向顶点着色器传入多个顶点数据,需要如下5个步骤:

  • 创建缓冲区对象
  • 绑定缓冲区对象
  • 写入数据到缓冲区对象
  • 分配缓冲区对象给一个 attribute 变量
  • 开启 attribute 变量
一次绘制三个点

效果


实现

完整代码如下:
  1. // 一次绘制三个点
  2. const VSHADER_SOURCE = `
  3. attribute vec4 a_Position;
  4. void main() {
  5.   gl_Position = a_Position;
  6.   gl_PointSize = 10.0;               
  7. }
  8. `
  9. const FSHADER_SOURCE = `
  10. void main() {
  11.   gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);
  12. }
  13. `
  14. function main() {
  15.     const canvas = document.getElementById('webgl');
  16.     const gl = canvas.getContext("webgl");
  17.     if (!gl) {
  18.         console.log('Failed to get the rendering context for WebGL');
  19.         return;
  20.     }
  21.     if (!initShaders(gl, VSHADER_SOURCE, FSHADER_SOURCE)) {
  22.         console.log('Failed to intialize shaders.');
  23.         return;
  24.     }
  25.     gl.clearColor(0, 0, 0, 1);
  26.     gl.clear(gl.COLOR_BUFFER_BIT);
  27.     // 顶点数据
  28.     const vertices = {
  29.         // 顶点数据。Float32Array 当做普通数组,对应 C 语言的 float
  30.         data: new Float32Array([
  31.             0.0, 0.5,
  32.             -0.5, -0.5,
  33.             0.5, -0.5
  34.         ]),
  35.         // 顶点数
  36.         vertexNumber: 3,
  37.         // 每个顶点分量数,例如第一个点(0.0, 0.5)
  38.         count: 2,
  39.     }
  40.     // 将顶点的位置写入顶点着色器
  41.     initVertexBuffers(gl, vertices)
  42.     // gl.drawArrays(mode, first, count)
  43.     // 还是画点(gl.POINTS),从缓冲区的第一个位置(0)开始画,绘制3(vertices.vertexNumber)个点
  44.     gl.drawArrays(gl.POINTS, 0, vertices.vertexNumber);
  45. }
  46. // 将三个顶点通过缓冲对象一次写入着色器
  47. function initVertexBuffers(gl, {data, count}) {
  48.     // 1. 创建缓冲区对象
  49.     const vertexBuffer = gl.createBuffer();
  50.     if (!vertexBuffer) {
  51.         console.log('创建缓冲区对象失败');
  52.         return -1;
  53.     }
  54.     // 绑定缓冲区对象绑定到`目标`。只能通过目标向缓冲区写数据
  55.     gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);
  56.     // 将数据写入缓冲区
  57.     gl.bufferData(gl.ARRAY_BUFFER, data, gl.STATIC_DRAW);
  58.     const a_Position = gl.getAttribLocation(gl.program, 'a_Position');
  59.     if (a_Position < 0) {
  60.         console.log('Failed to get the storage location of a_Position');
  61.         return -1;
  62.     }
  63.     // 将整个缓冲区对象分配给 atttibute(a_Position) 变量
  64.     gl.vertexAttribPointer(a_Position, count, gl.FLOAT, false, 0, 0);
  65.     // 激活 attribute 变量,使缓冲区对 attribute 变量分配生效
  66.     gl.enableVertexAttribArray(a_Position);
  67. }
复制代码
通过 initVertexBuffers() 将3个顶点分配给 attribute 变量,执行 gl.drawArrays(gl.POINTS, 0, vertices.vertexNumber) 时,顶点着色器执行了3次。顶点着色器执行过程和缓冲区数据传入过程如下:

绘制出所有点后,颜色缓冲区的内容就会自动显示在浏览器中。
initVertexBuffers() 里面涉及了许多概念和api,请往下阅读。
Float32Array

绘制三维图形 webgl 需要处理大量相同的数据,例如顶点坐标。为了优化性能,就引入了一种特殊的数组(类型化数组),浏览器事先知道数组中的数据类型,处理起来就更有效率。
Float32Array 就是其中一种类型化数组,通常用于存储顶点坐标或颜色。
Tip: webgl 中很多操作都需要用到类型化数组。
webgl 中有如下类型化数组:
数组类型每个元素所占字节数对应C语言的数据类型Int8Array18 位有符号整数Uint8Array18 位无符号整型Int16Array216 位有符号整数Uint16Array216 位无符号整型Int32Array432 位有符号的整型Uint32Array432 位无符号的整型Float32Array432 位的浮点数 floatFloat64Array864 位的浮点数 doublebindBuffer

gl.bindBuffer(target, buffer) - 绑定缓冲区。例如示例中的: gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);
为什么要将绑定缓冲区对象?不能直接向缓冲区写入数据,只能通过目标写入数据。就像这样:

目标表示缓冲区的用途。例如这里的 gl.ARRAY_BUFFER 指包含顶点属性(例如顶点坐标、纹理坐标数据或顶点颜色数据)的缓冲区。
bufferData

gl.bufferData(target, size, usage) - 将数据写入缓冲区。例如示例中的:gl.bufferData(gl.ARRAY_BUFFER, data, gl.STATIC_DRAW);
gl.STATIC_DRAW 是 usage 中的一种,指缓冲区的内容可能经常使用,而不会经常更改。内容被写入缓冲区,但不被读取。
此方法执行后,webgl 系统内部状态如下:

vertexAttribPointer

g.vertexAttribPointer(index, size, type, normalized, stride, offset) - 将整个缓冲区对象分配给 atttibute 变量。
例如示例中的:
  1. // count - 指定缓冲区每个顶点的分量个数,缺少则按照 vertexAttrib3f 的补全方式
  2. // gl.FLOAT 与上文的 Float32Array 对应
  3. // normalized - 对于类型gl.FLOAT和gl.HALF_FLOAT,此参数无效
  4. // stride - 指定相邻两个顶点间的字节数。默认为0
  5. // offset - 指定缓冲区对象中的偏移量。即attribute 从缓冲区何处开始存储
  6. gl.vertexAttribPointer(a_Position, count, gl.FLOAT, false, 0, 0);
复制代码
此方法执行后,webgl 系统内部状态如下:

疑惑: 说vertexAttribPointer最后一个参数(offset)必须是类型的字节长度的倍数。尝试将 0 改成 4,效果却是:

enableVertexAttribArray

gl.enableVertexAttribArray(index) - 激活 attribute 变量,使缓冲区对 attribute 变量分配生效。
此方法执行后,webgl 系统内部状态如下:

drawArrays

gl.drawArrays(mode, first, count) - 执行顶点着色器,按照 mode 指定的参数绘制图形。first 指定从哪个点开始绘制,count 指绘制需要几个点。
能绘制的图形有:

Tip: drawArrays 更多细节请看这里
三角形

只要一次性将三个点绘制出来,其实三角形也就画出来了。
修改上面示例一行代码就好(gl.POINTS -> gl.LINE_LOOP):
  1. // 前
  2. gl.drawArrays(gl.POINTS, 0, vertices.vertexNumber);
  3. // 后
  4. gl.drawArrays(gl.LINE_LOOP, 0, vertices.vertexNumber);
复制代码
效果如下:

矩形

效果如下:

核心代码如下:
  1.     // 定义四个点
  2.     const vertices = {
  3.         data: new Float32Array([
  4.             -0.5, 0.5,
  5.             -0.5, -0.5,
  6.             0.5, 0.5,
  7.             0.5, -0.5,
  8.         ]),
  9.         // 顶点数
  10.         vertexNumber: 4,
  11.         count: 2,
  12.     }
  13.     initVertexBuffers(gl, vertices)
  14.     // TRIANGLE_STRIP
  15.     gl.drawArrays(gl.TRIANGLE_STRIP, 0, vertices.vertexNumber);
复制代码
Tip:四个点的顺序有要求
三角扇

效果如下:

在矩形的基础上,改为 gl.TRIANGLE_FAN 即可。
Tip:GL_TRIANGLE_FAN - 绘制各三角形形成一个扇形序列,以 v0 为起点:(v0, v1, v2)、(v0, v2, v3)、(v0, v3, v4)

来源:https://www.cnblogs.com/pengjiali/p/17183771.html
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具