Web学习之怎么使用纹理贴图

为了使图形能获得接近于真实物体的材质效果,一般会使用贴图,贴图类型主要包括两种:漫反射贴图和镜面高光贴图。其中漫反射贴图可以同时实现漫反射光和环境光的效果。
实际效果请看demo:纹理贴图

纹理贴图

2D纹理

实现贴图就需要用到纹理,常用的纹理格式有:2D纹理,立方体纹理,3D纹理。我们使用最基本的2D纹理就能实现本节需要的效果,我们来看一下使用纹理需要的api。相关教程:js视频教程

因为纹理的坐标原点位于左下角,和我们通常的左上角坐标原点刚好相反,下面就是将它按Y轴进行反转,方便我们设置坐标。

gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, 1);

登录后复制

激活和绑定纹理,gl.TEXTURE0 表示0号纹理,可以从0一直往上递增。TEXTURE_2D 则是表示2D纹理。

gl.activeTexture(gl.TEXTURE0);//激活纹理gl.bindTexture(gl.TEXTURE_2D, texture);//绑定纹理

登录后复制

接着就是设置纹理参数,这个api非常重要,也是纹理最复杂的部分。

gl.texParameteri(target, pname, param) ,将param的值赋给绑定到目标的纹理对象的pname参数上。参数:

target: gl.TEXTURE_2Dgl.TEXTURE_CUBE_MAP

pname: 可指定4个纹理参数

放大(gl.TEXTURE_MAP_FILTER):当纹理的绘制范围比纹理本身更大时,如何获取纹理颜色。比如,将16*16的纹理图像映射到32*32像素的空间时,纹理的尺寸变为原始的两倍。默认值为gl.LINEAR。缩小(gl.TEXTURE_MIN_FILTER): 当纹理的绘制返回比纹理本身更小时,如何获取纹素颜色。比如,将32*32的纹理图像映射到16*16像素空间里,纹理的尺寸就只有原始的一般。默认值为gl.NEAREST_MIPMAP_LINEAR。水平填充(gl.TEXTURE_WRAP_S): 表示如何对纹理图像左侧或右侧区域进行填充。默认值为gl.REPEAT。垂直填充(gl.TEXTURE_WRAP_T): 表示如何对纹理图像上方和下方的区域进行填充。默认值为gl.REPEAT。

param: 纹理参数的值

可赋给 gl.TEXTURE_MAP_FILTERgl.TEXTURE_MIN_FILTER 参数的值

gl.NEAREST: 使用原纹理上距离映射后像素中心最近的那个像素的颜色值,作为新像素的值。

gl.LINEAR: 使用距离新像素中心最近的四个像素的颜色值的加权平均,作为新像素的值(和gl.NEAREST相比,该方法图像质量更好,但也会有较大的开销。)

可赋给 gl.TEXTURE_WRAP_Sgl.TEXTURE_WRAP_T 的常量:

gl.REPEAT: 平铺式的重复纹理

gl.MIRRORED_REPEAT: 镜像对称的重复纹理

gl.CLAMP_TO_EDGE: 使用纹理图像边缘值

设置样例如下所示:

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);gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);

登录后复制

gl.texImage2D,将 pixels 指定给绑定的纹理对象,这个api在 WebGL1WebGL2 中的重载函数多达十几个,格式类型非常多样。pixels参数既可以是图像,canvas,也可以是视频,我们只看 WebGL1中的调用形式。

// WebGL1:void gl.texImage2D(target, level, internalformat, width, height, border, format, type, ArrayBufferView? pixels);void gl.texImage2D(target, level, internalformat, format, type, ImageData? pixels);void gl.texImage2D(target, level, internalformat, format, type, HTMLImageElement? pixels);void gl.texImage2D(target, level, internalformat, format, type, HTMLCanvasElement? pixels);void gl.texImage2D(target, level, internalformat, format, type, HTMLVideoElement? pixels);void gl.texImage2D(target, level, internalformat, format, type, ImageBitmap? pixels);// WebGL2://...

登录后复制

我封装出了一个纹理加载函数,每个api的调用格式可以查看资料,还是先实现我们想要的效果。

function loadTexture(url) {    const texture = gl.createTexture();    gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, 1);    gl.activeTexture(gl.TEXTURE0);    gl.bindTexture(gl.TEXTURE_2D, texture);        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);    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);        let textureInfo = {        width: 1,        height: 1,        texture: texture,    };    const img = new Image();    return new Promise((resolve,reject) => {        img.onload = function() {            textureInfo.width = img.width;            textureInfo.height = img.height;            gl.bindTexture(gl.TEXTURE_2D, textureInfo.texture);            gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, img);            resolve(textureInfo);        };        img.src = url;    });}

登录后复制

漫反射贴图

首先实现漫反射光贴图,从网上下载了个地板的贴图,里面包含了各种类型的贴图。

缓冲区要增加顶点对应的纹理坐标,这样才能通过纹理坐标找到对应的纹理像素,简称纹素。

const arrays = {    position: [        -1, 0, -1,        -1, 0, 1,        1, 0, -1,        1, 0, 1    ],    texcoord: [        0.0, 1.0,        0.0, 0.0,        1.0, 1.0,        1.0, 0.0    ],    normal: [ 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1 ],};

登录后复制

顶点着色器唯一区别是增加了纹理坐标,需要插值传入片元着色器

//...attribute vec2 a_texcoord;varying vec2 v_texcoord;void main() {         //...    v_texcoord = a_texcoord;}

登录后复制

片元着色器修改的多一些。主要是使用 texture2D 获取对应坐标下的纹素,代替之前的颜色就可以了。下面就是片元着色器相关代码

//...vec3 normal = normalize(v_normal);vec4 diffMap = texture2D(u_samplerD, v_texcoord);//光线方向vec3 lightDirection = normalize(u_lightPosition - v_position);// 计算光线方向和法向量夹角float nDotL = max(dot(lightDirection, normal), 0.0);// 漫反射光亮度vec3 diffuse = u_diffuseColor * nDotL * diffMap.rgb;// 环境光亮度vec3 ambient = u_ambientColor * diffMap.rgb;//...

登录后复制

js部分加载贴图对应的图片,传递纹理单元,然后渲染

//...(async function (){    const ret = await loadTexture('/model/floor_tiles_06_diff_1k.jpg')    setUniforms(program, {        u_samplerD: 0//0号纹理    });    //...    draw();})()

登录后复制

效果如下,镜面高光部分似乎太刺眼了,因为地板是不会有镜子一样光滑强烈的反光的。

漫反射贴图

镜面高光贴图

为了实现更逼真的高光效果,继续实现高光贴图,实现原理和漫反射一样,把对应的高光颜色替换成高光贴图纹素就可以了。
下面就是片元着色器增加修改高光部分

//...vec3 normal = normalize(v_normal);vec4 diffMap = texture2D(u_samplerD, v_texcoord);vec4 specMap = texture2D(u_samplerS, v_texcoord);//光线方向vec3 lightDirection = normalize(u_lightPosition - v_position);// 计算光线方向和法向量夹角float nDotL = max(dot(lightDirection, normal), 0.0);// 漫反射光亮度vec3 diffuse = u_diffuseColor * nDotL * diffMap.rgb;// 环境光亮度vec3 ambient = u_ambientColor * diffMap.rgb;// 镜面高光vec3 eyeDirection = normalize(u_viewPosition - v_position);// 反射方向vec3 halfwayDir = normalize(lightDirection + eyeDirection);float specularIntensity = pow(max(dot(normal, halfwayDir), 0.0), u_shininess);vec3 specular = (vec3(0.2,0.2,0.2) + specMap.rgb) * specularIntensity;//...

登录后复制

js同时加载漫反射和高光贴图

//...(async function (){    const ret = await Promise.all([        loadTexture('/model/floor_tiles_06_diff_1k.jpg'),        loadTexture('/model/floor_tiles_06_spec_1k.jpg',1)    ]);    setUniforms(program, {        u_samplerD: 0,//0号纹理        u_samplerS: 1 //1号纹理    });    //...    draw();})()

登录后复制

最后实现的效果如下,明显更加接近真实的地板

高光贴图

以上就是Web学习之怎么使用纹理贴图的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2731068.html

(0)
上一篇 2025年3月8日 00:24:47
下一篇 2025年2月25日 08:18:47

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • OpenLayer怎么实现小车按路径运动

     本篇文章主要讲述的是在openlayer上实现路径运动,下面和小编一起去看看具体都是如何实现的吧,感兴趣的朋友可以看看。 一、需求分析      客户需要的功能就是能在一张Gis图上实现小车根据路径进行移动,为什么一定要Gis呢(这是客户…

    2025年3月8日
    200
  • 带你轻松理解KMP算法

    kmp(the knuth-morris-pratt algorithm)算法用于字符串匹配,从字符串中找出给定的子字符串。但它并不是很好理解和掌握。而理解它概念中的部分匹配表,是理解 kmp 算法的关键。 这里的讨论绕开其背后晦涩难懂的逻…

    编程技术 2025年3月8日
    200
  • js是什么编程语言?

    js全称JavaScript,是一种具有函数优先的轻量级,直译式、解释型或即时编译型的高级编程语言,是一种属于网络的高级脚本语言;JavaScript基于原型编程、多范式的动态脚本语言,并且支持面向对象、命令式和声明式,如函数式编程。 本教…

    2025年3月8日
    200
  • js中怎么写ajax

    在JavaScript中使用ajax有两个作用: 1.让js去读服务器上面的数据. 2.无刷新的情况下读取服务器上面的数据,例如:验证账号和密码是否正确等. 对于网络请求我们知道有Get 和Post两种,它们之间的区别是什么呢? get方式…

    2025年3月8日
    200
  • JQuery属于什么语言

    jquery是基于JavaScript语言写出来的一个框架,它封装JavaScript常用的功能代码,提供一种简便的JavaScript设计模式,但实质上还是js,所以JQuery也属于网页编程语言。 随着计算机行业的迅速发展,出现了各种各…

    2025年3月8日
    200
  • js中隐藏元素用什么方法

    js中隐藏元素可以使用html中的style属性来实现。具体方法如:【style.display=”block”】,表示控件可见;【style.display=”none”】,表示控件不可见。…

    2025年3月8日
    200
  • 学js需要什么基础?

    javascript一种直译式脚本语言,简称js,是一种动态类型、弱类型、基于原型的语言,内置支持类型。它的解释器被称为javascript引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在html网页上使用,用来给html网页增加…

    2025年3月8日
    200
  • json解析是什么?

    json是一种传递对象的语法,对象可以是name/value对,数组和其他对象。json(javascript object notation) 是一种轻量级的数据交换格式。 它基于JavaScript(Standard ECMA-262 …

    2025年3月8日
    200
  • js全称是什么?

    js全称为javascript,是一种直译式脚本语言,是一种动态类型、弱类型、基于原型的语言,内置支持类型。 javascript的解释器被称为JavaScript引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在HTML(标准通用…

    2025年3月8日
    200
  • js闭包是什么

    闭包就是能够读取其他函数内部变量的函数。由于在javascript中,只有函数内部的子函数才能读取局部变量,所以闭包可以理解成“定义在一个函数内部的函数“。在本质上,闭包是将函数内部和函数外部连接起来的桥梁。 JavaScript闭包 在J…

    2025年3月8日
    200

发表回复

登录后才能评论