经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 软件/图像 » WebGL » 查看文章
WebGL学习之纹理贴图
来源:cnblogs  作者:Jeff.Zhong  时间:2019/4/30 8:39:25  对本文有异议

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

纹理贴图

2D纹理

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

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

  1. gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, 1);

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

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

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

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

  • target: gl.TEXTURE_2Dgl.TEXTURE_CUBE_MAP

  • pname: 可指定4个纹理参数

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

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

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

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

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

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

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

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

设置样例如下所示:

  1. gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
  2. gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
  3. gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);
  4. gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);

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

  1. // WebGL1:
  2. void gl.texImage2D(target, level, internalformat, width, height, border, format, type, ArrayBufferView? pixels);
  3. void gl.texImage2D(target, level, internalformat, format, type, ImageData? pixels);
  4. void gl.texImage2D(target, level, internalformat, format, type, HTMLImageElement? pixels);
  5. void gl.texImage2D(target, level, internalformat, format, type, HTMLCanvasElement? pixels);
  6. void gl.texImage2D(target, level, internalformat, format, type, HTMLVideoElement? pixels);
  7. void gl.texImage2D(target, level, internalformat, format, type, ImageBitmap? pixels);
  8. // WebGL2:
  9. //...

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

  1. function loadTexture(url) {
  2. const texture = gl.createTexture();
  3. gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, 1);
  4. gl.activeTexture(gl.TEXTURE0);
  5. gl.bindTexture(gl.TEXTURE_2D, texture);
  6. gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
  7. gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
  8. gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);
  9. let textureInfo = {
  10. width: 1,
  11. height: 1,
  12. texture: texture,
  13. };
  14. const img = new Image();
  15. return new Promise((resolve,reject) => {
  16. img.onload = function() {
  17. textureInfo.width = img.width;
  18. textureInfo.height = img.height;
  19. gl.bindTexture(gl.TEXTURE_2D, textureInfo.texture);
  20. gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, img);
  21. resolve(textureInfo);
  22. };
  23. img.src = url;
  24. });
  25. }

漫反射贴图

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

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

  1. const arrays = {
  2. position: [
  3. -1, 0, -1,
  4. -1, 0, 1,
  5. 1, 0, -1,
  6. 1, 0, 1
  7. ],
  8. texcoord: [
  9. 0.0, 1.0,
  10. 0.0, 0.0,
  11. 1.0, 1.0,
  12. 1.0, 0.0
  13. ],
  14. normal: [ 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1 ],
  15. };

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

  1. //...
  2. attribute vec2 a_texcoord;
  3. varying vec2 v_texcoord;
  4. void main() {
  5. //...
  6. v_texcoord = a_texcoord;
  7. }

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

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

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

  1. //...
  2. (async function (){
  3. const ret = await loadTexture('/model/floor_tiles_06_diff_1k.jpg')
  4. setUniforms(program, {
  5. u_samplerD: 0//0号纹理
  6. });
  7. //...
  8. draw();
  9. })()

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

漫反射贴图

镜面高光贴图

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

  1. //...
  2. vec3 normal = normalize(v_normal);
  3. vec4 diffMap = texture2D(u_samplerD, v_texcoord);
  4. vec4 specMap = texture2D(u_samplerS, v_texcoord);
  5. //光线方向
  6. vec3 lightDirection = normalize(u_lightPosition - v_position);
  7. // 计算光线方向和法向量夹角
  8. float nDotL = max(dot(lightDirection, normal), 0.0);
  9. // 漫反射光亮度
  10. vec3 diffuse = u_diffuseColor * nDotL * diffMap.rgb;
  11. // 环境光亮度
  12. vec3 ambient = u_ambientColor * diffMap.rgb;
  13. // 镜面高光
  14. vec3 eyeDirection = normalize(u_viewPosition - v_position);// 反射方向
  15. vec3 halfwayDir = normalize(lightDirection + eyeDirection);
  16. float specularIntensity = pow(max(dot(normal, halfwayDir), 0.0), u_shininess);
  17. vec3 specular = (vec3(0.2,0.2,0.2) + specMap.rgb) * specularIntensity;
  18. //...

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

  1. //...
  2. (async function (){
  3. const ret = await Promise.all([
  4. loadTexture('/model/floor_tiles_06_diff_1k.jpg'),
  5. loadTexture('/model/floor_tiles_06_spec_1k.jpg',1)
  6. ]);
  7. setUniforms(program, {
  8. u_samplerD: 0,//0号纹理
  9. u_samplerS: 1 //1号纹理
  10. });
  11. //...
  12. draw();
  13. })()

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

原文链接:http://www.cnblogs.com/edwardloveyou/p/10790882.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号