WebGL 多重纹理的使用介绍

 更新时间:2023年04月19日 14:21:23   作者:H_World  
这篇文章主要为大家介绍了WebGL 多重纹理的使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

基于上篇的内容,纹理中最小的单位是纹素,若干纹素组成一个纹理单元,每个纹理单元用一个number值来管理,那么我们来看看它是如何工作的。

激活

在使用纹理单元之前,得需要激活纹理单元

activeTexture()

参数:

  • 指定准备激活的纹理单元:webgl.TEXTURE0、webgl.TEXTURE1...最后的数字表示纹理单元的编号

绑定

这一步,需要告诉WebGL系统纹理对象使用的是何种类型的纹理,在对该对象的操作之前,需要先绑定该对象,这和之前的缓冲区对象的数据写操作类似。

bindTexture(webgl.TEXTURE_2D, texture0)

参数:

  • webgl.TEXTURE_2D: 二维纹理;webgl.TEXTURE_CUBE_MAP: 立方体纹理
  • texture0:需要绑定指定的纹理对象

该方法开启纹理对象,以及将纹理对象texture0绑定到纹理单元webgl.TEXTURE0上,之后通过操作纹理单元去操作纹理对象。

配置

这儿主要是对纹理对象具体的信息操作了,例如纹理单元类型、尺寸、是否裁剪、纹理的展示方式(放大或缩小)等,下面我们再一一分析。

texParameteri(target, pname, param)

参数:

  • target: webgl.TEXTURE_2D或webgl.TEXTURE_CUBE_MAP
  • pname: webgl.TEXTURE_MAG_FILTER纹理放大,例如16x16的纹理图像映射到32x32的像素空间中;webgl.TEXTURE_MIN_FILTER纹理缩小, 应用场景相反;webgl.TEXTURE_WRAP_S纹理水平轴(ST坐标系统)方向填充;webgl.TEXTURE_WRAP_T纹理T轴方向填充。
  • param: webgl.LINEAR线性变换,使用距离像素中心最近的四个点,进行线性平均加权,然后作为新的颜色值; webgl.NEAREST最近的距离优先,新的像素值取中心像素的值;CLAMP_TO_EDGE: 设置纹理 S/T 轴方向上的边缘环绕方式为CLAMP_TO_EDGE,即在超出边缘的部分使用纹理边缘的像素颜色填充. webgl.REPEAT纹理重复补充;webgl.MIRRORED_REPEAT:纹理镜像对称式重复

会发现参数pname、param中的参数常量分类比较多,对应的分类是

  • pname : TEXTURE_MAG_FILTER、TEXTURE_MIN_FILTER
  • param: NEAREST、LINEAR

  • pname : TEXTURE_WRAP_S、TEXTURE_WRAP_T
  • param: REPEAT、MIRRORED_REPEAT、CLAMP_TO_EDGE

分配

纹理对象信息配置好后,接着可以将纹理图像分配给纹理对象

webgl.texImage2D(target, level, internalformat, format type, image)

参数:

  • target: webgl.TEXTURE_2D或webgl.TEXTURE_CUBE_MAP
  • level: 指定纹理缓冲区中要加载的纹理级别,一般为 0
  • internalformat: 纹理对象内部数据格式,通常使用webgl.RGBA
  • format: 纹理数据格式,必须与internalformat相同的值
  • type: 纹理数据类型
  • image: 纹理对象中image对象

其中的纹理数据类型有:

webgl.UNSIGNED_BYTE: 无符号整型,每个颜色分量占据1字节
webgl.UNSIGNED_SHORT_5_6_5:RGB
webgl.UNSIGNED_SHORT_4_4_4:RGBA
webgl.UNSIGNED_SHORT_5_5_5_1:RGBA

传递

经过上面一系列准备后,我们就可以把最终的纹理单元传递给片元着色器了,这儿就用到了方法 webgl.uniform1i(),上篇这个方法已经说过了,就不再说明了。

主要看看片元着色器中是怎么做的

uniform sampler2D texture;
varying vec2 inUV;
vec4 color1=texture2D(texture, vec2(inUV.x, 1.0 - inUV.y));
gl_FragColor=color1;
  • 定义了一个可以在外部访问的一种特殊的、专用于纹理对象的数据类型sampler2D,就是对应的纹理单元类型webgl.TEXTURE_2D;如果是webgl.TEXTURE_CUBE_MAP呢,则是samplerCube.
  • 通过uniform1i方法,将纹理对象传递到了片元着色器中的texture变量
  • inUV接收顶点着色器中的顶点的纹理坐标,当然这里的纹理坐标是在外部设置的
  • texture2D()在片元着色器中根据纹理坐标获取纹理图像上的纹素的颜色
  • 把获取到的颜色赋值给片元着色器

经过对纹理的加载、设置、映射,剩下的就是绘画了。

webgl.drawArrays(webgl.TRIANGLE_STRIP, 0, 4)

好了,晦涩难懂的概念介绍完了,就把具体的代码贴出来吧!

效果

代码

着色器设置

const VSHADER_SOURCE = `
  attribute vec2 a_position;
  attribute vec2 outUV;
  varying vec2 inUV;
  void main(void){
    gl_Position=vec4(a_position, 0.0, 1.0);
    inUV=outUV;
  }
`
const FSHADER_SOURCE = `
  precision mediump float;
  uniform sampler2D texture;
  uniform sampler2D texture1;
  varying vec2 inUV;
  uniform float anim;
  void main(void){
    vec4 color1=texture2D(texture, vec2(inUV.x, 1.0 - inUV.y));
    vec4 color2=texture2D(texture1, vec2(inUV.x + anim, 1.0 - inUV.y));
    gl_FragColor=color1+color2;
  }
`

数据缓存区的设置

const initBuffer = async () => {
  let positions = [
    0.8, -0.8,
    -0.8, -0.8,
    0.8, 0.8,
    -0.8, 0.8,
  ]
  let pointPosition = new Float32Array(positions)
  let aPsotion = webgl.getAttribLocation(webgl.program, "a_position")
  let triangleBuffer = webgl.createBuffer()
  webgl.bindBuffer(webgl.ARRAY_BUFFER, triangleBuffer)
  webgl.bufferData(webgl.ARRAY_BUFFER, pointPosition, webgl.STATIC_DRAW)
  webgl.enableVertexAttribArray(aPsotion)
  webgl.vertexAttribPointer(aPsotion, 2, webgl.FLOAT, false, 0, 0)
  var texCoords = [
    1, 0,
    0, 0,
    1, 1,
    0, 1,
  ]
  const attribOutUV = webgl.getAttribLocation(webgl.program, "outUV")
  let texCoordBuffer = webgl.createBuffer()
  webgl.bindBuffer(webgl.ARRAY_BUFFER, texCoordBuffer)
  webgl.bufferData(webgl.ARRAY_BUFFER, new Float32Array(texCoords), webgl.STATIC_DRAW)
  webgl.enableVertexAttribArray(attribOutUV)
  webgl.vertexAttribPointer(attribOutUV, 2, webgl.FLOAT, false, 0, 0)
  uniformTexture = webgl.getUniformLocation(webgl.program, "texture")
  uniformTexture1 = webgl.getUniformLocation(webgl.program, "texture1")
  texture0 = await initTexture('/web-gl/landscape.png')
  texture1 = await initTexture("/web-gl/fog.png")
}

主要是设置顶点坐标和纹理坐标的数据

纹理对象的初始化

const initTexture = (imageSrc: string): Promise<WebGLTexture> => {
  return new Promise((resolve, reject) => {
    let textureHandle = webgl.createTexture()
    const image = new Image()
    image.onload = function () {
      webgl.bindTexture(webgl.TEXTURE_2D, textureHandle)
      webgl.texParameteri(
        webgl.TEXTURE_2D,
        webgl.TEXTURE_WRAP_S,
        webgl.CLAMP_TO_EDGE
      )
      webgl.texParameteri(
        webgl.TEXTURE_2D,
        webgl.TEXTURE_WRAP_T,
        webgl.CLAMP_TO_EDGE
      )
      webgl.texParameteri(
        webgl.TEXTURE_2D,
        webgl.TEXTURE_MIN_FILTER,
        webgl.LINEAR
      )
      webgl.texImage2D(
        webgl.TEXTURE_2D,
        0,
        webgl.RGBA,
        webgl.RGBA,
        webgl.UNSIGNED_BYTE,
        image
      )
      resolve(textureHandle)
    }
    image.onerror = reject
    image.src = imageSrc
  })
}

这儿需要使用异步方法,因为需要在image.onload方法中返回设置信息数据后的纹理对象。 在 webgl 的纹理加载中,由于图片是异步加载的,因此我们需要使用 Promise 来处理加载完毕后的回调函数。

绘制和动态变化

const updateCount = () => {
  count = count + 0.001
  if (count >= 1.14) {
    count = -1.0
  }
}
const draw = () => {
  webgl.clearColor(0.0, 1.0, 1.0, 1.0)
  webgl.clear(webgl.COLOR_BUFFER_BIT | webgl.DEPTH_BUFFER_BIT)
  webgl.enable(webgl.DEPTH_TEST)
  //纹理变动
  uniformAnim = webgl.getUniformLocation(webgl.program, "anim");
  updateCount()
  webgl.uniform1f(uniformAnim, count);
  webgl.activeTexture(webgl.TEXTURE0)
  webgl.bindTexture(webgl.TEXTURE_2D, texture0)
  webgl.uniform1i(uniformTexture, 0)
  webgl.activeTexture(webgl.TEXTURE1)
  webgl.bindTexture(webgl.TEXTURE_2D, texture1)
  webgl.uniform1i(uniformTexture1, 1)
  webgl.drawArrays(webgl.TRIANGLE_STRIP, 0, 4)
  requestAnimationFrame(draw)
}

具体的代码就是这些了。

小结一下

到目前为止,我们已经了解了顶点着色器、片元着色器、顶点缓冲区、矩阵绘制和变换、纹理图像、纹理叠加等,这样我们已经基本上对二维绘图掌握了,一个面我们了解了,之后我们可能就去看看多个面的物体和场景。

那么就让我们一起看看三维世界吧!

以上就是WebGL 多重纹理的详细内容,更多关于WebGL 多重纹理的资料请关注脚本之家其它相关文章!

相关文章

  • 基于JS实现点击图片在弹出层显示大图效果

    基于JS实现点击图片在弹出层显示大图效果

    Javascript是个好东西。Jquery是基于这个好东西的一个强大的库。本文将利用JavaScript实现点击图片在弹出层显示大图功能,感兴趣的可以了解一下
    2022-08-08
  • JS模板实现方法

    JS模板实现方法

    我们在使用JS渲染DOM时,一般使用字符串创建DOM然后附加到父元素上,如果附加的DOM是动态易变的,那需要在函数中写大量逻辑。如果在控件实现过程中,这带来的问题更为严重。
    2013-04-04
  • JavaScript定时器实现的原理分析

    JavaScript定时器实现的原理分析

    JavaScript中的定时器大家基本在平时的开发中都遇见过吧,但是又有多少人去深入的理解其中的原理呢?本文我们就来分析一下定时器的实现原理、定时器的妙用、定时器使用注意事项,有兴趣的朋友可以看下
    2016-12-12
  • JS正则截取两个字符串之间及字符串前后内容的方法

    JS正则截取两个字符串之间及字符串前后内容的方法

    这篇文章主要介绍了JS正则截取两个字符串之间及字符串前后内容的方法,结合实例形式简单分析了JS正则截取字符串操作的常用技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2017-01-01
  • JavaScript 报表展示实现代码

    JavaScript 报表展示实现代码

    以下是从网上找到的一段JavaScript实现图形报表的代码,对于想客户端显示报表的朋友可以参考下。
    2009-12-12
  • 详解Js模块化的作用原理和方案

    详解Js模块化的作用原理和方案

    这篇文章主要介绍了Js模块化的作用原理和方案,对JS模块化感兴趣的同学,可以参考下
    2021-04-04
  • JavaScipt验证URL新方法详解(2023 年版)

    JavaScipt验证URL新方法详解(2023 年版)

    这篇文章主要为大家介绍了JavaScipt验证URL新方法示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • 微信小程序 wxParse插件显示视频问题

    微信小程序 wxParse插件显示视频问题

    这篇文章主要介绍了微信小程序 wxParse插件显示视频问题,文中给大家提到了微信小程序插件wxParse的使用,需要的朋友可以参考下
    2019-09-09
  • 纯html+css+javascript实现楼层跳跃式的页面布局(实例代码)

    纯html+css+javascript实现楼层跳跃式的页面布局(实例代码)

    这篇文章主要介绍了纯html+css+javascript实现楼层跳跃式的页面布局,需要的朋友可以参考下
    2017-10-10
  • Bootstrap Search Suggest使用例子

    Bootstrap Search Suggest使用例子

    这篇文章主要为大家详细介绍了Bootstrap Search Suggest使用例子,教大家使用bootstrap搜索建议插件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12

最新评论