ThreeJs 认识纹理

一、前言

这篇文章,我们主要来了解一下 ThreeJs 中纹理相关的知识。渲染一个 3D 物体时,网格 Mesh 决定了这个物体的形状态,如一个球,一辆车,一个人等。而纹理决定了这个物体的表面具体长什么样子。一个球包上一层篮球的花纹就是篮球了,而如果包上的是一层足球的花纹那可能就是足球了。

二、概述

ThreeJs 中为定义了多种多样的纹理,其类图如下。

Texture.jpg

纹理的基类是 Texture,一般我们都使用这个类,通过给其属性 Image 传入一个图片从而构造出一个纹理。纹理是材质的属性,材质和几何体 Gemotry 构成 Mesh ,然后被添加到 Scene 中进行渲染。纹理决定了物体的表面该是什么样子,而材质则决定了物体具备什么样的“气质”。后面还会再专门学习材质,下面再详细地看一看各个纹理的作用及其使用场景。

三、认识纹理

1、Texture

TextureXmind.jpg

纹理的属性何其多,但在一般情况下,并不要需要一一来设置,而是取默认值即可。并且,我们一般都会通过 TextureLoader 来为我们构造一个纹理,而不是直接构造。例子如下。

var texture = new THREE.TextureLoader().load( "textures/water.jpg" );
texture.wrapS = THREE.RepeatWrapping;
texture.wrapT = THREE.RepeatWrapping;
texture.repeat.set( 4, 4 );

一般情况下,我们这么简单的设置下就可以了。但实际应用中,我们可能会碰到需要指定自己的纹理坐标的情况。那这个时候,我们就需要自己计算好纹理坐标,然后给 geometry 添加属性 "uv",从而应用我们自己的纹理坐标。

geom.addAttribute('uv', new THREE.BufferAttribute(uvArr, 2))

2、CanvasTexture

CanvasTexture.jpg

CanvasTexture 主要是用于将 Dom 元素 <canvas> 中绘制的内容,以纹理的方式贴到模型上。在 canvas 中我们可以绘制任何我们想要绘制的图形或者文字。以下代码就是在 Canvas 中绘制文本,从而创建一个 CanvasTexture。

let m = 4;
let textDiv = document.createElement('canvas');
let context2D = textDiv.getContext('2d');
context2D.font = ' ' +  font.fontSize + 'px ' + font.fontFamily;
context2D.fontWeight = font.fontWeight;
context2D.fillStyle = font.fontColor;

let s = context2D.measureText(text).width,height = y(1.4 * font.fontSize + 2 * m),u = height - m, c = u / 2, width = y(s + c + 2 * m);

textDiv.width = width;
textDiv.height = height;

context2D.font = ' ' +  font.fontSize + 'px ' + font.fontFamily;
context2D.fontWeight = font.fontWeight;
context2D.fillStyle = font.fontColor;
context2D.fillText(text,textDiv.width / 2 - s / 2,textDiv.height / 2 + 0.3 * font.fontSize);
let texture = new THREE.CanvasTexture(textDiv)
image.png

图中红圈里的内容就是通过 CanvasTexture 来实现的。

3、CompressedTexture

CompressedTexture.jpg

4、CubeTexture

CubeTexture.jpg

一般都是在环境贴图中,其效果就是模型的材质会呈现世界中图像的镜像,就像是镜子一样。

var loader = new THREE.CubeTextureLoader();
loader.setPath( 'textures/cube/pisa/' );

var textureCube = loader.load( [
    'px.png', 'nx.png',
    'py.png', 'ny.png',
    'pz.png', 'nz.png'
] );

var material = new THREE.MeshBasicMaterial( { color: 0xffffff, envMap: textureCube } );

类似效果如下。

image.png

5、DataTexture

DataTexture.jpg
// create a buffer with color data

var size = width * height;
var data = new Uint8Array( 3 * size );

var r = Math.floor( color.r * 255 );
var g = Math.floor( color.g * 255 );
var b = Math.floor( color.b * 255 );

for ( var i = 0; i < size; i ++ ) {

    var stride = i * 3;

    data[ stride ] = r;
    data[ stride + 1 ] = g;
    data[ stride + 2 ] = b;

}

// used the buffer to create a DataTexture

var texture = new THREE.DataTexture( data, width, height, THREE.RGBFormat );
texture.needsUpdate = true
image.png

6、DataTexture3D

DataTexture3D.jpg

相对 DataTexture 多有一个深度维度。

var texture = new THREE.DataTexture3D( array, 256, 256, 109 );

texture.format = THREE.RedFormat;
texture.type = THREE.UnsignedByteType;
image.png

7、DepthTexture

DepthTexture.jpg
depthTexture = new THREE.DepthTexture();
depthTexture.type = THREE.UnsignedShortType;
image.png

对于深度纹理贴图,这里首先要弄明白什么是深度图。深度图像包含了普通的RGB三通道彩色图像和Depth Map。而通俗的讲,深度就是每个像素距离当时相机所在位置的距离。

image.png

上面就是一个深度图的展示,通过颜色的明暗程度,我们可以看出图片中所呈现的物体的远近关系的。

我们进一步来看,对于深度图的实际应用,应该能更进一步帮助我们理解。首先看一小段 frag_shader 的代码。主要关注其中的 tDepth 的应用。这是一个纹理采样器。

<script id="post-frag" type="x-shader/x-fragment">
        #include <packing>

        varying vec2 vUv;
        uniform sampler2D tDiffuse;
        // 定义一个用于深度图片的纹理采样器
        uniform sampler2D tDepth;
        uniform float cameraNear;
        uniform float cameraFar;

       // 读取深度信息
        float readDepth( sampler2D depthSampler, vec2 coord ) {
            float fragCoordZ = texture2D( depthSampler, coord ).x;
            float viewZ = perspectiveDepthToViewZ( fragCoordZ, cameraNear, cameraFar );
            return viewZToOrthographicDepth( viewZ, cameraNear, cameraFar );
        }

        void main() {
            //vec3 diffuse = texture2D( tDiffuse, vUv ).rgb;
            float depth = readDepth( tDepth, vUv );
       // 应用深度信息
            gl_FragColor.rgb = 1.0 - vec3( depth );
            gl_FragColor.a = 1.0;
        }
</script>

这里的关键的 2 步就是首先从纹理中读取深度值,然后进行相应的应用,也就是颜色上的计算,并全部赋值给 gl_FragColor.rgb。具体怎么计算就需要根据对应的需求来定了。

8、VideoTexture

VideoTexture.jpg

常规的用法

<video id="video" autoplay loop crossOrigin="anonymous" webkit-playsinline style="display:none">
    <source src="textures/sintel.ogv" type='video/ogg; codecs="theora, vorbis"'>
    <source src="textures/sintel.mp4" type='video/mp4; codecs="avc1.42E01E, mp4a.40.2"'>
</video>

video = document.getElementById( 'video' );
texture = new THREE.VideoTexture( video );

视频作为纹理的一个效果

image.png

四、总结

文章走马观花式的把 ThreeJs 所支持的几种纹理都过了一遍,大概看了每个纹理的用法以及效果。

而实际项目中,作者目前用到的只是 Texture(一般纹理贴图)、CanvasTexture(用于文字绘制)、CubeTexutre(用于环境贴图),其他几个相对比较高级,目前还没有实际去用到。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容

  • 本文首发于个人博客:Lam's Blog - 【OpenGL-ES】二维纹理,文章由MarkDown语法编写,可能...
    格子林ll阅读 3,758评论 0 9
  • 转载自VR设计云课堂[https://www.jianshu.com/u/c7ffdc4b379e]Unity S...
    水月凡阅读 1,010评论 0 0
  • 一、纹理基础 3D图形渲染中最基本的操作就是对一个表面应用纹理。纹理可以表现只从网格的几何形状无法得到的附加细节。...
    cain_huang阅读 8,727评论 0 7
  • 一、纹理综述 物理世界中,视域内的颜色会发生快速的变化。你可以看到很多物体表面都会呈现出丰富的颜色,并且在狭小的面...
    凡几多阅读 1,026评论 1 5
  • 立方体贴图(Cubemap) 我们之前一直使用的是2D纹理,还有更多的纹理类型我们没有探索过,本教程中我们讨论的纹...
    IceMJ阅读 9,210评论 2 9