此文上接一系列 WebGL 文章,从基础概念开始, 上一篇讲的是动画。
在 WebGL 中如何使用纹理?你可能会从二维图像处理的文章 中得到启发,如果我们讲的再深入一点可能更好理解。
首先需要调整着色器以便使用纹理,这里是顶点着色器的修改部分, 我们需要传递纹理坐标,在这个例子中直接将它们传到片断着色器中。
#version 300 es
in vec4 a_position;
*in vec2 a_texcoord;
uniform mat4 u_matrix;
+// 定义一个传递到片段着色器的纹理坐标变量
+out vec2 v_texcoord;
void main() {
// 将位置和矩阵相乘
gl_Position = u_matrix * a_position;
+ // 传递纹理坐标到片断着色器
+ v_texcoord = a_texcoord;
}
在片断着色器中声明一个 sampler2D 类型的全局变量,可以让我们引用一个纹理,
然后使用从顶点着色器传入的纹理坐标调用 texture
方法,
在纹理上找到对应的颜色。
#version 300 es
precision highp float;
// 从顶点着色器中传入的值
*in vec2 v_texcoord;
*// 纹理
*uniform sampler2D u_texture;
out vec4 outColor;
void main() {
* outColor = texture(u_texture, v_texcoord);
}
我们需要设置纹理坐标
// 找到顶点坐标中的属性
var positionAttributeLocation = gl.getAttribLocation(program, "a_position");
*var texcoordAttributeLocation = gl.getAttribLocation(program, "a_texcoord");
...
*// 创建一个纹理坐标缓冲,使其成为当前的 ARRAY_BUFFER 并复制 纹理坐标 值
*var texcoordBuffer = gl.createBuffer();
*gl.bindBuffer(gl.ARRAY_BUFFER, texcoordBuffer);
*setTexcoords(gl);
*
*// 启用属性
*gl.enableVertexAttribArray(texcoordAttributeLocation);
*
*// 告诉属性如何从 纹理坐标缓冲 (ARRAY_BUFFER) 中获取数据
*var size = 2; // 每次迭代运行提取两个单位数据
*var type = gl.FLOAT; // 每个单位的数据类型是32位浮点型
*var normalize = true; // 转换 0-255 为 0.0-1.0
*var stride = 0; // 0 = 移动单位数量 * 每个单位占用内存(sizeof(type)每次迭代获取下一个纹理坐标
*var offset = 0; // 从缓冲起始位置开始读取
*gl.vertexAttribPointer(
* texcoordAttributeLocation, size, type, normalize, stride, offset);
如你所见,我们将图像映射到 ‘F’ 中的每个矩形面上。
*// 为 F 设置纹理坐标缓冲
*function setTexcoords(gl) {
* gl.bufferData(
* gl.ARRAY_BUFFER,
* new Float32Array([
* // 正面左竖
* 0, 0,
* 0, 1,
* 1, 0,
* 0, 1,
* 1, 1,
* 1, 0,
*
* // 正面上横
* 0, 0,
* 0, 1,
* 1, 0,
* 0, 1,
* 1, 1,
* 1, 0,
* ...
* ]),
* gl.STATIC_DRAW);
我们还需要一个纹理,我们可以从头做一个但在这个例子中就直接加载一个图像把, 因为那可能是常用的做法。
这是我们将要使用的图像
一颗赛艇的图像!事实上使用一个带有 ‘F’ 的图像能够在结果中清楚的分辨出纹理的方向。
加载图像的过程是异步的,我们请求图像资源后浏览器需要一段时间去下载。 通常有两种处理方法,一种是等纹理下载完成后再开始绘制,另一种是在图像加载前使用生成的纹理, 这种方式可以立即启动渲染,一旦图像下载完成就拷贝到纹理。我们将使用下方的方法。
*// 创建一个纹理
*var texture = gl.createTexture();
*gl.bindTexture(gl.TEXTURE_2D, texture);
*
*// 用 1x1 个蓝色像素填充纹理
*gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, 1, 1, 0, gl.RGBA, gl.UNSIGNED_BYTE,
* new Uint8Array([0, 0, 255, 255]));
*
*// 异步加载图像
*var image = new Image();
*image.src = "resources/f-texture.png";
*image.addEventListener('load', function() {
* // 现在图像加载完成,拷贝到纹理中
* gl.bindTexture(gl.TEXTURE_2D, texture);
* gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA,gl.UNSIGNED_BYTE, image);
* gl.generateMipmap(gl.TEXTURE_2D);
*});
这是结果
如果我只想使用一部分图像覆盖 ‘F’ 的正面怎么办,纹理是通过“纹理坐标”来引用的, 纹理坐标 0.0 到 1.0 对应纹理从左到右,0.0 到 1.0 对应第一个像素所在行到最后一行。 注意我没有使用上或者下,上下在纹理坐标空间中是没有意义的,因为绘制一些东西后再重定向后, 是没有上下的概念的,主要是依据传递给 WebGL 的纹理数据,纹理数据的开头对应纹理坐标 0, 0, 结尾对应纹理坐标 1, 1
我将纹理载入到 Photoshop 中得到一些点的坐标。
可以像这样将像素坐标转换到纹理坐标
texcoordX = pixelCoordX / (width - 1)
texcoordY = pixelCoordY / (height - 1)
这是正面的纹理坐标
// 正面左竖
38 / 255, 44 / 255,
38 / 255, 223 / 255,
113 / 255, 44 / 255,
38 / 255, 223 / 255,
113 / 255, 223 / 255,
113 / 255, 44 / 255,
// 正面上横
113 / 255, 44 / 255,
113 / 255, 85 / 255,
218 / 255, 44 / 255,
113 / 255, 85 / 255,
218 / 255, 85 / 255,
218 / 255, 44 / 255,
// 正面中横
113 / 255, 112 / 255,
113 / 255, 151 / 255,
203 / 255, 112 / 255,
113 / 255, 151 / 255,
203 / 255, 151 / 255,
203 / 255, 112 / 255,
对背面也使用相同的纹理坐标,得到这样的结果。
并不是非常好看,但是希望这样能展示出纹理坐标的用法。 如果你使用代码生成几何体(立方体,球体,等等), 通常情况下计算期望的纹理坐标也是比较容易的。 另一方面如果通过软件例如 Blender, Maya, 3D Studio Max 制作几何体, 那么你的美术(或者你自己)就会 在这些软件中用 UV 编辑器调整纹理坐标 .
如果纹理坐标再 0.0 到 1.0 之外会怎样?WebGL 默认会重复纹理, 0.0 到 1.0 是一份纹理的“拷贝”,1.0 到 2.0 是另外一份拷贝, -4.0 到 -3.0 也是另外一份拷贝。让我们在一个平面上使用这些纹理坐标。
-3, -1,
2, -1,
-3, 4,
-3, 4,
2, -1,
2, 4,
这是结果
你可以使用CLAMP_TO_EDGE
告诉 WebGL 再某个方向不需要重复,例如
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
you can also tell WebGL to mirror the texture when it repeat use gl.MIRRORED_REPEAT
.
您还可以在 WebGL 中使用 gl.MIRRORED_REPEAT
进行重复时使用镜像纹理。
点击上方示例中的按钮,观察结果。
你可能注意到在加载纹理时调用了 gl.generateMipmap
,那是干什么的?
假设我们有这样一个 16×16 像素的纹理。
假设我们要将它绘制在屏幕的 2×2 个像素上,那么这 4 个像素应该使用什么颜色? 这里有 256 个像素可以选择,如果在 Photoshop 中将 16×16 的图像缩放到 2×2, 它会将每个角 8×8 的像素的平均值赋给这四个像素。不幸的是绘制 64 个像素再求平均在 GPU 中是非常慢的。假设你有一个 2048x2084 像素的纹理想要绘制成 2x2 个像素, 就需要对 1024x1024 或 100 万个像素求平均 4 次,这需要很多运算同时速度要快。
事实上 GPU 使用的是一个纹理贴图(mipmap),纹理贴图是一个逐渐缩小的图像集合, 每一个是前一个的四分之一大小,16×16 纹理的纹理贴图看起来像这样。
通常每个子图都是前一级的双线性插值,这就是 gl.generateMipmap
做的事情,
它根据原始图像创建所有的缩小级别,你也可以自己提供缩小级别的图像。
现在如果你想将 16x16 像素的纹理绘制到屏幕的 2×2 个像素上, WebGL 会从创建的贴图中找到从之前级别贴图插值出的 2×2 贴图来使用。
你可以为纹理选择不同的贴图筛选条件来控制 WebGL 的插值, 一共有这 6 种模式
NEAREST
= 从最大的贴图中选择 1 个像素LINEAR
= 从最大的贴图中选择 4 个像素然后混合NEAREST_MIPMAP_NEAREST
= 选择最合适的贴图,然后从上面找到一个像素LINEAR_MIPMAP_NEAREST
= 选择最合适的贴图,然后取出 4 个像素进行混合NEAREST_MIPMAP_LINEAR
= 选择最合适的两个贴图,从每个上面选择 1 个像素然后混合LINEAR_MIPMAP_LINEAR
= 选择最合适的两个贴图,从每个上选择 4 个像素然后混合你可以通过这两个例子看到贴图的重要性,第一个显示的是使用 NEAREST
或 LINEAR
,
只从最大的体贴图上选择像素,当物体运动时就会出现抖动。由于每个像素都从最大的图上选择,
随着位置和大小的改变,可能会在不同的时间选择不同的像素,从而出现抖动。
上面的例子被夸大了,以说明问题。
观察发现左边和中间的抖动会多于右边。
由于右边的使用多级贴图并且混合颜色,绘制的越小 WebGL 挑选的像素离原图关系越远。
相反的中间的小图虽然使用了 LINEAR
混合 4 个像素的不同颜色,但这 4 个像素是从不同角落的 16x16 的大图中选出来,
不同的选择会有较大的差别,所以还是抖动明显。右下角的图保持颜色一致是从右中图中挑选的像素。
第二个例子显示了一些深入屏幕中的多边形。
6 个深入屏幕的横梁使用的是之前 6 种不同的筛选模式。左上使用的是 NEAREST
,
你会感受到明显的块状感;中上使用的是 LINEAR
也没有好到哪里去;
右上使用的是 NEAREST_MIPMAP_NEAREST
,点击图像切换纹理,每个贴图都是不同的颜色,
就可以清除的看出它使用的是哪个贴图;左下使用的是 LINEAR_MIPMAP_NEAREST
,
意思是挑选最合适贴图种的 4 个像素进行混合,你会发现贴图切换的部分非常突兀;
中下使用的是 NEAREST_MIPMAP_LINEAR
,也就是找到最合适的两个贴图各取一点进行混合,
如果仔细看会发现仍然有块状感,尤其是水平方向;右下使用的是 LINEAR_MIPMAP_LINEAR
,
也就是选出最合适的两个贴图各取 4 个点进行混合。
你可能会想既然理论上 LINEAR_MIPMAP_LINEAR
是最好的选择为什么还要有其他选择,
一个原因是 LINEAR_MIPMAP_LINEAR
是最慢的,读 8 个像素比读 1 个像素慢一些,
在现代的 GPU 上如果一次使用一个贴图可能没什么问题,但是现在的游戏可能一次就需要 2
到 4 个贴图,4 贴图 * 8 像素每贴图 = 绘制每个像素需要读取 32 个像素,那就会慢很多了。
另一个原因是如果想实现特定的效果,比如做一些复古的东西可能就需要使用 NEAREST
。
贴图也占用内存,事实上它占用额外 33% 的内存,那是非常多的内存,
尤其是使用很大的纹理例如想要在游戏的标题屏幕上绘制的东西。如果你不会绘制比最大的贴图要小的东西,
为什么要把内存浪费在较小的贴图上,直接使用 NEAREST
或 LINEAR
就只使用第一个贴图。
设置筛选器可以调用 gl.texParameter
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_LINEAR);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);
TEXTURE_MIN_FILTER
是当绘制的比最大贴图小的时候。
TEXTURE_MAG_FILTER
是绘制的比最大的贴图大的时候。
对于 TEXTURE_MAG_FILTER
只有 NEAREST
和 LINEAR
两个可选设置。
需要注意的是,WebGL2 要求纹理是“纹理完整”的,否则它们将无法渲染。 “纹理完整”意味着要么
您已经设置了过滤,因此它只使用第一个贴图级别,这意味着 将“TEXTURE_MIN_FILTER”设置为“LINEAR”或“NEAREST”。
如果您使用的是多个贴图,那么它们需要是正确的尺寸,并且您必须提供所有这些 缩小到 1x1 大小。
最简单的方法是调用 gl.generateMipmap
。 否则,如果您提供自己的贴图,则需要提供
所有这些,否则你会得到一个错误。
一个常见的问题是“如何将不同的图像应用于立方体的每个面?”。 例如,假设我们有这 6 张图片。
脑中出现了 3 个答案
制作一个复杂的着色器,引用 6 个纹理,传入一些额外的顶点信息表明使用的纹理是什么。不要这样做! 稍微一想就知道你要写一大堆不同的着色器应用于不同面数量的图形之类。
绘制 6 个面代替立方体,这是常用的解决办法,不错但是只能用在简单的图形例如立方体。 如果有一个包含 1000 个方形的图形就要绘制 1000 个面,会非常慢。
我敢说,最好的方法就是将图像放在一个纹理中,然后利用纹理坐标映射不同的图像到每个面, 这是很多高性能应用(读作游戏)使用的技术。例如我们将所有的图像放入这样一个纹理中 像这样
然后为立方体的每个面设置不同的纹理坐标。
// 选择左上图
0 , 0 ,
0 , 0.5,
0.25, 0 ,
0 , 0.5,
0.25, 0.5,
0.25, 0 ,
// 选择中上图
0.25, 0 ,
0.5 , 0 ,
0.25, 0.5,
0.25, 0.5,
0.5 , 0 ,
0.5 , 0.5,
// 选择右上图
0.5 , 0 ,
0.5 , 0.5,
0.75, 0 ,
0.5 , 0.5,
0.75, 0.5,
0.75, 0 ,
// 选择左下图
0 , 0.5,
0.25, 0.5,
0 , 1 ,
0 , 1 ,
0.25, 0.5,
0.25, 1 ,
// 选择中下图
0.25, 0.5,
0.25, 1 ,
0.5 , 0.5,
0.25, 1 ,
0.5 , 1 ,
0.5 , 0.5,
// 选择右下图
0.5 , 0.5,
0.75, 0.5,
0.5 , 1 ,
0.5 , 1 ,
0.75, 0.5,
0.75, 1 ,
然后得到
这种将多个图像通过一个纹理提供的方法通常被叫做 纹理图集. 它是最好的方式,因为只需要加载一个贴图,着色器也会因为只用一个贴图而保持简单, 不同于多个平面需要多次调用绘制,这样只需要调用一次绘制。
你可能还想知道关于纹理的其他重要内容, 一个是 纹理单元状态如何工作. 一个是 如何一次使用多个纹理,还有一个是 如何使用其他域名下的图像.
接下来讲的是 让我们开始简化用更少的代码获得更多的乐趣.