使用ArcGIS API和Three.js在三维场景中实现动态立体墙效果

使用ArcGIS API和Three.js在三维场景中实现动态立体墙效果

废话不多说,直接先来看下最终实现的动态立体墙效果图。

动态立体墙效果图.gif

如果图片还不够直观,那么点击链接查看在线示例

首先我们需要用到ArcGIS API中的externalRenderers类将外部的Three.js渲染器加载到地图三维场景中,如果不知道怎么使用的可以查看我的这篇文章《ArcGIS API在视图中渲染Three.js场景》。那篇文章中加载的是一个三维模型,而本示例中只需加载一面“墙”,也就是一个平面,并增加一个动态效果。所以重点就是怎么加载一个垂直于地球表面的平面,以及如何实现动态效果。

1 垂直于地球表面的墙

如图所示,先确定出两个“墙角”的坐标。


墙角坐标点.png
let points = [
  [104.06179498614645, 30.659871702738265], // 坐标1
  [104.06494384459816, 30.659931252383917], // 坐标2
];

现在我们有了两个经纬度坐标的点,但是我们需要4个顶点才能构成一个矩形面,所以我们还需要2个点。假设2个墙角坐标贴近于地面,那么它们的高度就为0,那就再只需要2个同样经纬度坐标但高度大于0的点就能构成一个在地面上并且垂直于地面的矩形面了。所以在我们定义的myRenderer对象中添加一个height属性。

let myRenderer = {
  // ... 其它属性
  height: 100, // 墙的高度
  // ... 其它属性、方法
};

现在我们有4个由经纬度加高度构成的点,如果要在视图中渲染成一个矩形面,我们要先将这4个点转换为渲染坐标系中的点,再将每3个顶点为一组构成一个三角面,最后由2个三角面构成一个矩形面。这样做是因为在Three.js中所有的模型都是由顶点加三角面构成的。

1.1 顶点转换

在顶点转换之前,我们还需要做一个操作,那就是将我们的经纬度坐标转换为XY坐标。这需要用到ArcGIS API中的webMercatorUtils工具中的lngLatToXY方法,该方法将给定的经度和纬度转换为Web Mercator的XY值。

points.forEach((point) => {
  // 将经纬度坐标转换为xy值
  let pointXY = webMercatorUtils.lngLatToXY(point[0], point[1]);
});

然后需要用到Three.js的数学库中的四维矩阵Matrix4类以及ArcGIS API中externalRenderers对象上的renderCoordinateTransformAt方法将点转换为渲染坐标系中的点坐标。

let transform = new THREE.Matrix4(); // 变换矩阵
let transformation = new Array(16);
let vector3List = []; // 顶点数组
points.forEach((point) => {
  // 将经纬度坐标转换为xy值
  let pointXY = webMercatorUtils.lngLatToXY(point[0], point[1]);
  // 先转换高度为0的点
  transform.fromArray(
    externalRenderers.renderCoordinateTransformAt(
      this.view,
      [pointXY[0], pointXY[1], 0], // 坐标在地面上的点[x值, y值, 高度值]
      this.view.spatialReference,
      transformation
    )
  );
  vector3List.push(
    new THREE.Vector3(
      transform.elements[12],
      transform.elements[13],
      transform.elements[14]
    )
  );
  // 再转换距离地面高度为height的点
  transform.fromArray(
    externalRenderers.renderCoordinateTransformAt(
      this.view,
      [pointXY[0], pointXY[1], this.height], // 坐标在空中的点[x值, y值, 高度值]
      this.view.spatialReference,
      transformation
    )
  );
  vector3List.push(
    new THREE.Vector3(
      transform.elements[12],
      transform.elements[13],
      transform.elements[14]
    )
  );
});

renderCoordinateTransformAt方法的作用是计算一个4x4变换矩阵,该矩阵构成从局部笛卡尔坐标系到虚拟世界坐标系的线性坐标变换。该方法传入4个参数:
1 view,ArcGIS API生成的三维视图。
2 origin,局部笛卡尔坐标系中原点的全局坐标,也就是[经纬度转换后的X坐标, 经纬度转换后的y坐标, 高度值]。
3 srcSpatialReference,原点坐标的空间参考。
4 dest,存储16个矩阵元素的数组的引用。生成的矩阵遵循OpenGL约定,其中转换组件占据第13、14和第15个元素。

现在,vector3List变量中存储的就是每个顶点转换后的三维向量,一共为4个顶点。顺序是[第一个经纬度的地面顶点, 第一个经纬度的空中顶点, 第二个经纬度的地面顶点, 第二个经纬度的空中顶点],这个顶点的顺序很重要,后面会用到。

1.2 生成三角面以及面的UV队列

因为Three.js中的面都是由小三角面构成的,所以我们需要根据顶点列表中的顶点来组成三角面,每三个顶点构成一个三角面,一定要注意构成三角面的的顶点顺序,因为要和面的UV队列一一对应起来,这样给每个面贴的纹理材质才能正确显示出来。
纹理贴图的坐标系统是这样的:图片左下角为原点(0, 0),右下角为(1, 0),右上角为(1, 1),左上角为(0, 1),这和图片的大小宽高无关。如下图所示:

纹理贴图示意图.png

将纹理坐标关系转换为二维向量表示。

const t0 = new THREE.Vector2(0, 0); // 图片左下角
const t1 = new THREE.Vector2(1, 0); // 图片右下角
const t2 = new THREE.Vector2(1, 1); // 图片右上角
const t3 = new THREE.Vector2(0, 1); // 图片左上角

一个简单的矩形面由4个顶点和2个小三角面构成,顶点和三角面关系如下图所示:

顶点顺序示意图.png

图中0、1、2、3序号代表vector3List变量中顶点的顺序。按照逆时针规则画出2个三角面,下三角面为绿色三角面[0, 2, 1],上三角面为蓝色三角面[1, 2, 3]。例如,要将纹理贴图和绿色三角面映射起来,那么绿色三角面对应的UV就是[t0, t1, t3],蓝色三角面对应的UV就是[t3, t1, t2]。
顶点和纹理坐标向量对应图.png

根据以上原理生成三角面列表以及UV队列。

let faceList = []; // 三角面数组
let faceVertexUvs = []; // 面的 UV 层的队列,该队列用于将纹理和几何信息进行映射

for (let i = 0; i < vector3List.length - 2; i++) {
  if (i % 2 === 0) { // 下三角面
    faceList.push(new THREE.Face3(i, i + 2, i + 1));
    faceVertexUvs.push([t0, t1, t3]);
  } else { // 上三角面
    faceList.push(new THREE.Face3(i, i + 1, i + 2));
    faceVertexUvs.push([t3, t1, t2]);
  }
}

1.3 生成几何体

使用Three.js中的Geometry构造函数来生成自定义几何体。

const geometry = new THREE.Geometry(); // 生成几何体
geometry.vertices = vector3List; // 几何体顶点
geometry.faces = faceList; // 几何体三角面
geometry.faceVertexUvs[0] = faceVertexUvs; // 面的UV队列,用于将纹理信息映射到几何体上

geometry.faceVertexUvs的属性值为数组是因为有多组UV。颜色贴图、法线贴图、高光贴图、金属度贴图等共用一组纹理坐标UV即geometry.faceVertexUvs[0],设置阴影的光照贴图lightMap使用另外一组纹理坐标,也就是geometry.faceVertexUvs[1]。默认情况下,geometry.faceVertexUvs属性中会存在一个元素,所以可以直接对geometry.faceVertexUvs[0]进行赋值操作。
注意:对于缓冲区类型几何体也就是通过BufferGeometry构造函数生成的几何体,是通过设置.attributes.uv和.attributes.uv2两个属性分别定义两组顶点纹理坐标。

2 实现墙的动态效果

动态效果的原理其实是纹理贴图实现的,一共两层贴图,一层颜色从上到下越来越不透明,给人一面墙的感觉,另一层从上到下越来越透明,然后每次渲染都改变第二层纹理在垂直方向上的偏移量,这样就有了滚动起来的效果。
因为当第一层半透明和第二层半透明的效果都叠加到一个几何体上时,这个几何体就会变得更加的透明,显示效果上就不是很好,所以我们把这两层效果放到两个几何体上,只需要把上面创建好的几何体克隆一遍。

const geometry2 = geometry.clone();

2.1 利用材质的alphaMap贴图实现半透明效果

我们选用基础网络材质MeshBasicMaterial,该材质不受光照的影响,所以不需要在场景中再额外的添加光源,省时省力~。该材质对象上的alphaMap贴图属性可以用来控制整个表面的不透明度,黑色完全透明,白色完全不透明。如下图所示,从上到下越来越白,也就是也来越不透明。

alphaMap贴图.png

加载alphaMap的纹理贴图资源,创建材质,和第一个几何体生成网格,然后添加到场景中。

this.alphaMap = new THREE.TextureLoader().load( // 加载alpha贴图资源
  '../images/texture_1.png'
);
// 创建材质
const material = new THREE.MeshBasicMaterial({
  color: 0xff0000,
  side: THREE.DoubleSide,
  transparent: true, // 必须设置为true,alphaMap才有效果
  depthWrite: false, // 渲染此材质是否对深度缓冲区有任何影响
  alphaMap: this.alphaMap, // alpha贴图,控制透明度
});
const mesh = new THREE.Mesh(geometry, material); // 第一个几何体和第一个材质
this.scene.add(mesh);

注意
side属性要设置为THREE.DoubleSide,这样才能两个面都进行绘制,也就是说从前后两个方向都能看到几何体。
transparent属性一定要设置为true,不然alphaMap贴图是没有效果的,看不出透明效果。
depthWrite属性一定要设置为false,才能正确渲染后方的半透明物体。

效果如图所示:

第一层透明效果图.png

2.2 利用材质的map颜色贴图实现渐变半透明效果

MeshBasicMaterial材质的map属性为颜色贴图。可以设置为半透明的PNG格式图片,也就达到了透明的效果。

PNG格式纹理.png

加载PNG格式的纹理贴图资源,创建材质,和克隆出来的几何体生成网格,然后添加到场景中。

this.texture = new THREE.TextureLoader().load(
  '../images/texture_2.png'
);
this.texture.wrapS = THREE.RepeatWrapping; // 水平方向重复
this.texture.wrapT = THREE.RepeatWrapping; // 垂直方向重复
const material2 = new THREE.MeshBasicMaterial({
  side: THREE.DoubleSide,
  transparent: true,
  depthWrite: false, // 渲染此材质是否对深度缓冲区有任何影响
  map: this.texture, // 颜色贴图,加载PNG图片达到透明效果
});
const mesh2 = new THREE.Mesh(geometry2, material2);
this.scene.add(mesh2);

注意
因为需要在垂直方向上存在偏移量,形成滚动的效果,所以必须设置纹理的包裹方式为重复,wrapSwrapT属性设置为THREE.RepeatWrapping。具体可查看文档

叠加到场景中的效果如图所示:

第二层叠加后效果图.png

2.3 效果动起来

现在大体效果已经差不多了,最后只要动起来就完工了。要实现动起来的效果只需要在render函数中添加更新纹理贴图偏移量的代码,每渲染一次就更新一次偏移量。

render() {
  // ... 其它代码
  if (this.offset <= 0) {
    this.offset = 1;
  } else {
    this.offset -= 0.02; // 每次渲染就向上移动0.02个单位,如果想要速度快就增大该值
  }
  if (this.texture) {
    this.texture.offset.set(0, this.offset); // 水平偏移量0,垂直方向偏移量为offset
  }
  // ... 其它代码
}

texture对象上存在offset属性,该属性值类型为二维向量Vector2,用来设置水平和垂直方向上的偏移量,值的范围在0.0到1.0之间。

最终效果如图所示:

动态立体墙效果图.gif

3 总结

至此,我们的立体动态墙效果就已经实现了。重要点就是通过顶点向量加三角面构成自定义的平面矩形几何体,通过设置纹理贴图以及改变纹理贴图的偏移量来实现动起来的效果。


点击链接查看完整代码
点击链接查看在线示例

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