threejs在透视相机模式下,绘制像素大小固定的元素

要求:在透视相机模式下绘制一个图标,图标大小始终为32*32px。图标如下:

实现思路:

使用THREE.Sprite。因为 SpriteMaterial 支持配置 sizeAttenuation 使Sprite大小不随相机的深度而衰减。所以我们只要保证sprite的初始的大小合适,在以后的相机深度变化的时候就不会改变大小了。

开始操作

第一次的操作:

javascript 复制代码
drawAddSprite(type: InterActiveType = InterActiveType.Default) {
    // 这个sprite我是自己用canvas画出来的,比较简单,就不贴了
    const texture = generateAddIconCanvasTexture(type);
    const material = new THREE.SpriteMaterial({ map: texture, sizeAttenuation: false });
    material.map.colorSpace = 'srgb';
    const sprite = new THREE.Sprite(material);
    scene.add(sprite)
}

我们看一下效果

我设置sizeAttenuation: false之前,我以为这个图标会占满屏(高度上满屏),毕竟这个图标跟相机没关系了。但是并没有。后来意识到sizeAttenuation: false 只是设置了相机的深度跟Sprite没有关系,但是透视相机的fov还是会影响到sprite的大小的。我们去验证一下,当我不断修改相机的fov时,图标随着fov的增大而减小。

思路:

  1. 第一步,我们找到一个fov值,在这个值下,我们看到的图标是占满屏幕的

  2. 在第一步的fov值下,我们要求图标大小是32*32的,那只需要设置Sprite的scale为 32 / canvas.clientHeight

我们来实现一下上面的思路:

  1. 我们来找这个fov

这个d就是相机的深度,既然设置了 sizeAttenuation: false,sprite不随相机的深度变化而变化,那这个d就要有一个默认值。我盲猜这个默认值是1。那这个fov的值计算如下

javascript 复制代码
Math.atan(0.5) * 180 / Math.PI * 2

这个值经过计算是53.13010235415598

现在我们设置相机fov为这个值去看一下效果

果然是满屏的。所以这个d确实是1.

找到了这个fov,我们设计相机的fov为这个值,现在我们去设置sprite的scale去改变Sprite的大小

javascript 复制代码
drawAddSprite(type: InterActiveType = InterActiveType.Default) {
    const texture = generateAddIconCanvasTexture(type);
    const material = new THREE.SpriteMaterial({ map: texture, sizeAttenuation: false });
    material.map.colorSpace = 'srgb';
    const sprite = new THREE.Sprite(material);
    const scale = 32 / canvas.clientHeight;
    sprite.scale.set(scale, scale, 1);
    scene.add(sprite)
}

看一下效果,已经实现了。

但是呢,我们如果我们的fov不是固定的怎么办呢,只需要加上如下配置就可以动态的根据fov去获得这个scale了。

javascript 复制代码
drawAddSprite(type: InterActiveType = InterActiveType.Default) {
    const texture = generateAddIconCanvasTexture(type);
    const material = new THREE.SpriteMaterial({ map: texture, sizeAttenuation: false });
    material.map.colorSpace = 'srgb';
    const sprite = new THREE.Sprite(material);
    const fullFov = ((Math.atan(0.5) * 180) / Math.PI) * 2;
    const { fov } = camera;
    let scale = 32 / dom.clientHeight;
    scale *= Math.tan((fov / 2 / 180) * Math.PI) / Math.tan((fullFov / 2 / 180) * Math.PI);
    sprite.scale.set(scale, scale, 1);
    scene.add(sprite)
}

完成。

相关推荐
爱看书的小沐4 天前
【小沐学WebGIS】基于Three.JS绘制飞行轨迹Flight Tracker(Three.JS/ vue / react / WebGL)
javascript·vue·webgl·three.js·航班·航迹·飞行轨迹
GISer_Jing5 天前
前端GIS篇——WebGIS、WebGL、Java后端篇
java·前端·webgl
爱看书的小沐7 天前
【小沐学GIS】基于C++瓦片地图下载工具(高德/天地图/谷歌/必应/OSM/MapBox/ArcGIS)第十三期
c++·webgl·谷歌地图·earth·osm·瓦片地图下载·mapdowloader
xhload3d13 天前
智慧停车场合集 | 图扑数字孪生静态交通一网统管
物联网·3d·智慧城市·html5·webgl·数字孪生·可视化·工业互联网·三维建模·智慧停车·智慧交通·轻量化·电力能源·智慧停车场·智慧停车楼
ssshooter13 天前
WebGL 切换 Shader 的策略
前端·webgl
Zuckjet_13 天前
第 5 篇:WebGL 从 2D 到 3D - 坐标系、透视与相机
前端·javascript·3d·webgl
ssshooter14 天前
linkProgram 和 useProgram 分别执行了什么动作?
前端·webgl
猪哥帅过吴彦祖15 天前
第 5 篇:WebGL 从 2D 到 3D - 坐标系、透视与相机
前端·javascript·webgl
_JinHao15 天前
Cesium Viewer对象详解——Cesium基础笔记(快速入门)
前端·javascript·笔记·3d·webgl
Zuckjet_15 天前
开启 3D 之旅 - 你的第一个 WebGL 三角形
前端·javascript·3d·webgl