zoukankan      html  css  js  c++  java
  • Three.js学习笔记04--纹理

    1 纹理由图片组成

     3D世界的纹理由图片组成。

    将纹理以一定的规则映射到几何体上,一般是三角形上,那么这个几何体就有纹理皮肤了。

    首先应该有一个纹理类,其次是有一个加载图片的方法,将这张图片和这个纹理类捆绑起来。

    在threejs中,纹理类由THREE.Texture表示,其构造函数如下所示:

    THREE.Texture( image, mapping, wrapS, wrapT, magFilter, minFilter, format, type, anisotropy )

    各个参数的意义是:

    Image:这是一个图片类型,基本上它有ImageUtils来加载,如下代码

    var image = THREE.ImageUtils.loadTexture(url); // url 是一个http://xxxx/aaa.jpg 的类似地址,javascript没有从本地加载数据的能力,所以没有办法从您电脑的C盘加载数据。

    Mapping:是一个THREE.UVMapping()类型,它表示的是纹理坐标。

    wrapS:表示x轴的纹理的回环方式,就是当纹理的宽度小于需要贴图的平面的宽度的时候,平面剩下的部分应该以何种方式贴图的问题。

    wrapT:表示y轴的纹理回环方式。

    magFilter和minFilter表示过滤的方式,这是OpenGL的基本概念。

    format:表示加载的图片的格式,这个参数可以取值THREE.RGBAFormat,RGBFormat等。THREE.RGBAFormat表示每个像素点要使用四个分量表示,分别是红、绿、蓝、透明来表示。RGBFormat则不使用透明,也就是说纹理不会有透明的效果。

    type:表示存储纹理的内存的每一个字节的格式,是有符号,还是没有符号,是整形,还是浮点型。不过这里默认是无符号型(THREE.UnsignedByteType)。

    anisotropy:各向异性过滤。使用各向异性过滤能够使纹理的效果更好,但是会消耗更多的内存、CPU、GPU时间

    2 纹理坐标

    在正常的情况下,你在0.0到1.0的范围内指定纹理坐标。我们来简单看一下纹理坐标如下图:

    当我们用一幅图来做纹理的时候,那么这幅图就隐示的被赋予了如图一样的纹理坐标,这个纹理坐标将被对应到一个形状上。

    (1) 画一个平面

    通过PlaneGemotry可以画一个平面,代码如下:

    var geometry = new THREE.PlaneGeometry( 500, 300, 1, 1 );

    这个平面的宽度是500,高度是300.

    (2)为平面赋予纹理坐标

    平面有4个顶点,所以我们只需要指定4个纹理坐标就行了。纹理坐标由顶点的uv成员来表示,uv被定义为一个二维向量THREE.Vector2(),我们可以通过如下代码来为平面定义纹理:

    geometry.vertices[0].uv = new THREE.Vector2(0,0);

    geometry.vertices[1].uv = new THREE.Vector2(1,0);

    geometry.vertices[2].uv = new THREE.Vector2(1,1);

    geometry.vertices[3].uv = new THREE.Vector2(0,1);

    注意,4个顶点分别对应了纹理的4个顶点。还要注意(0,0),(1,0),(1,1),(0,1)他们之间的顺序是逆时针方向。大家在给平面赋纹理坐标的时候也要注意方向,不然three.js是分不清楚的。

    (3)加载纹理

    纹理作为一张图片,可以来源于互联网,或者本地服务器,但是就是不能来源于类似C:pica.jpg这样的本地路径。这是因为javascript没有加载本地路径文件的权限。如果你尝试这么做,会报错误。所以务必在你的电脑上搭建一个tomcat,apache,iis等服务器中的一种,并把图片资源放到服务器中,并通过http://localhost:8080/img/a.jpg这样的方式去访问它。这样就能解决上面的交叉域问题。

    这里加载纹理使用了上面介绍的loadTexture函数,代码如下:

    var texture = THREE.ImageUtils.loadTexture("textures/a.jpg",null,function(t)

    {

    });

    这个函数的第一个参数是一个相对路径,表示与您的网页之间的相对路径。相对路径对应了一个纹理图片textures/a.jpg。

    第二个参数为null,表示时候要传入一个纹理坐标参数,来覆盖前面在geometry中的参数。

    第三个表示一个回调函数,表示成功加载纹理后需要执行的函数,参数t是传入的texture。

    最后,这个函数的返回值是加载的纹理。

    (4) 将纹理应用于材质

    加载好纹理,只需要将纹理映射到材质就可以了。我们这里使用了一个普通的材质THREE.MeshBasicMaterial,材质中有一个map属性,可以直接接受纹理,我们可以这样定义一个带纹理的材质:

    var material = new THREE.MeshBasicMaterial({map:texture});

    接下来直接将纹理甩给Mesh,同时也别忘了Mesh也需要geometry,他们的关系如下:

    var mesh = new THREE.Mesh( geometry,material );

    最后的最后,将这个mesh加入场景中:

    scene.add( mesh );

    2 白话纹理原理,通过html中的canvas来作为纹理

    从本质上来说,纹理只是图片而已,它是由像素点组成。无论在内存还是显存中,它都是由4个分量组成,这四个分量是R、G、B和A。唯一的不同的,在显存中,会比内存中更快的渲染到显示器上。这是毋庸置疑的,因为显存中的帧缓冲本来就是和显示器上 的像素一一对应的。

    从上面的概念中,我们就能够引申出一些重要的理解了,就是只要是图像数据,准确的说是内存或者显存中的图像数据,都可以作为纹理,显示在屏幕中。

    它们两者之间有很多差别,这个差别就是图片和canvas的差别,图片是通过图像处理软件,如photoshop来处理的。而canvas是通过浏览器的绘图API来绘制的。显示canvas能够给程序员更多的想象空间,从而做出更有意思的效果出来。

     

    3 模型的加载

    在three.js中,模型是怎么加载到浏览器中的呢?

    上图的顺序是:

    1、服务器上的模型文件以文本的方式存储,除了以three.js自定义的文本方式存储之外,当然也可以以二进制的方式存储。

    2、浏览器下载文件到本地

    3、Javascript解析模型文件,生成Mesh网格模型

    4、显示在场景中。

    对照上面这幅图,我们对需要注意的几点重点说明一下:

    1、服务器上的模型文件大多是存储的模型的顶点信息,这些信息可以以文本的方式存储的(并不一定需要用文本的方式存储)。Three.js支持很多种3D模型格式,例如ply,stl,obj,vtk等等。随着three.js的升级,会支持越来越多的文件格式,到目前为止,three.js已经能够支持市面上大多数3D模型格式了。

    2、第二步是浏览器下载文本文件,这是一件很普通的事情,只需要使用javascript的异步请求就可以实现了。

    3、Javascript解析文本文件并生成一个geometry,最终生成Mesh。

    4、当产生Mesh后,将其加入到场景。

    转载自webgl中文网

  • 相关阅读:
    <转载>大白话系列之C#委托与事件讲解(二)
    <转载>C# 类型基础
    <转载>大白话系列之C#委托与事件讲解(三)
    <转载>大白话系列之C#委托与事件讲解大结局
    <转载>C#中父窗口和子窗口之间实现控件互操作
    <转载>C# 中的泛型
    <转载>C# 中的委托和事件
    mailto的用法
    计算器
    终于搞清楚了这句代码的意思
  • 原文地址:https://www.cnblogs.com/yhhzxcvbnm/p/8251654.html
Copyright © 2011-2022 走看看