three.js全景漫游实践

时间:2022-10-01 20:58:33

Hello 小伙伴们,如果觉得本文还不错,记得给个 star , 小伙伴们的 star 是我持续更新的动力!GitHub 地址

简介

全景图分两种

  1. 由六张正方形图片组成的SkyBox
  2. 一整张的宽高比为2比1的全景图片。

今天我就实现一整张全景图的案例。

思路

我们超赞的设计师画的中秋全景图(利用透视网格辅助PS绘制)

three.js全景漫游实践

 
创建一个球体网格,对网格进行x轴反转,使所有的面点向内
let geometry = new THREE.SphereGeometry( 500, 60, 40 );
geometry.scale( -1, 1, 1 );

使用上面的全景贴图创建基础材质

let material = new THREE.MeshBasicMaterial({
map: new THREE.TextureLoader().load( 'panorama.jpg'),
depthTest: false//此参数控制是否使用像素深度来计算新像素的值
}); let mesh = new THREE.Mesh( geometry, material );
scene.add( mesh );

把相机设置为球的中心点

let camera = new THREE.PerspectiveCamera( 100, window.innerWidth / window.innerHeight, 1, 1100 );
camera.target = new THREE.Vector3( 0, 0, 0 );
camera.position.set(0, 0, 0);

陀螺仪相机控制器,实现移动端陀螺仪控制相机

let controls = new THREE.DeviceOrientationControls( camera );

此时还没有动画效果,还需要增加一个实时更新渲染动画

function animate() {
render();
requestAnimationFrame(animate);
}
function render() {
//更新控制器
controls.update();
camera.lookAt( camera.target );
renderer.render(scene, camera);
}

简单案例代码

DEMO: songdy.github.io/panorama/si…

这就简单实现了一个全景图,贴出以上的全部代码

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>简单的全景图</title>
<link rel="stylesheet" type="text/css" href="./css/simple-index.css">
</head>
<body>
<div id="container"></div>
<script type="text/javascript" src="./js/three.min.js"></script>
<script type="text/javascript" src="./js/DeviceOrientationControls.js"></script>
<script type="text/javascript">
class panorama{
constructor () {
this.scene = new THREE.Scene();
this.initCamera();
this.initMesh();
this.initRenderer();
this.animate();
} initCamera () {
let camera = this.camera = new THREE.PerspectiveCamera( 100, window.innerWidth / window.innerHeight, 1, 1100 );
camera.position.set(0, 0, 0); this.controls = new THREE.DeviceOrientationControls( camera );
this.controls.connect();
}
initMesh () {
let geometry = new THREE.SphereGeometry( 500, 60, 40 );
geometry.scale( -1, 1, 1 );
geometry.rotateY(-Math.PI / 2) let material = new THREE.MeshBasicMaterial({
map: new THREE.TextureLoader().load('./textures/SphericalMap.jpg')
}); let mesh = new THREE.Mesh( geometry, material );
this.scene.add( mesh );
}
initRenderer () {
let container = document.getElementById( 'container' );
let renderer = this.renderer = new THREE.WebGLRenderer({ logarithmicDepthBuffer: true });
renderer.setPixelRatio( window.devicePixelRatio );
renderer.setSize( window.innerWidth, window.innerHeight );
renderer.sortObjects = false;
renderer.autoClear = false;
container.appendChild( renderer.domElement );
}
animate() {
this.render();
requestAnimationFrame( ()=>{this.animate()});
}
render() {
//更新控制器
this.controls.update();
this.renderer.render(this.scene, this.camera);
}
}
new panorama();
</script>
</body>
</html>

  

相机

直接上图,常规的全景漫游的进场效果:

three.js全景漫游实践

左边是效果,右边是相机辅助效果。

思路分析

相机起始在球体接近顶部位置,从上往下看

let camera = new THREE.PerspectiveCamera( 150, window.innerWidth / window.innerHeight, 1, 2000 );
camera.position.set(0, 450, 0);//相机定位在y轴450
camera.target = new THREE.Vector3( 0, -500, 0 );//设置目标点
camera.lookAt( camera.target );//看向y轴负方向

  相机有上往下移动到求的中心点(0, 0, 0)。同时,相机目标点从底部(0, -500, 0)转到背面(0, 0, -500)。把fov从150调整为100,效果更赞了。

new TWEEN.Tween( { y : 450, lat : 0, fov : 150 } )
.to( { y : 0, lat : 90, fov : 100 }, 2500 )
.onUpdate(function() {
camera.position.y = this.y;
let phi = THREE.Math.degToRad( this.lat );
camera.target.y = -500 * Math.cos( phi );
camera.target.z = -500 * Math.sin( phi );
camera.fov = this.fov;
camera.updateProjectionMatrix();
})

  

进场案例代码

DEMO: songdy.github.io/panorama/ca…

辅助理解DEMO: songdy.github.io/panorama/ca…

把简单版加入进场效果

<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>玩转相机</title>
<link rel="stylesheet" type="text/css" href="./css/simple-index.css">
</head>
<body>
<div id="container"></div>
<script type="text/javascript" src="./js/three.min.js"></script>
<script type="text/javascript" src="./js/DeviceOrientationControls.js"></script>
<script type="text/javascript" src="./js/Tween.js"></script>
<script type="text/javascript">
class panorama{
constructor () {
this.SCREEN_WIDTH = window.innerWidth;
this.SCREEN_HEIGHT = window.innerHeight;
this.scene = new THREE.Scene();
this.initCamera();
this.initMesh();
this.initRenderer();
this.animate();
this.start();
}
initCamera () {
let aspect = this.SCREEN_WIDTH / this.SCREEN_HEIGHT;
this.camera = new THREE.PerspectiveCamera( 150, 0.5 * aspect, 1, 2000 );
this.camera.position.set(0, 450, 0);
this.camera.target = new THREE.Vector3( 0, -1, 0 ); }
initMesh () {
let geometry = new THREE.SphereGeometry( 500, 60, 40 );
geometry.scale( -1, 1, 1 );
geometry.rotateY(-Math.PI / 2)
let material = new THREE.MeshBasicMaterial({
map: new THREE.TextureLoader().load('./textures/SphericalMap.jpg')
});
this.mesh = new THREE.Mesh( geometry, material );
this.scene.add( this.mesh );
}
initRenderer () {
let container = document.getElementById( 'container' );
let renderer = this.renderer = new THREE.WebGLRenderer({ logarithmicDepthBuffer: true });
renderer.setPixelRatio( window.devicePixelRatio );
renderer.setSize( window.innerWidth, window.innerHeight );
renderer.sortObjects = false;
renderer.autoClear = false;
container.appendChild( renderer.domElement );
}
start () {
let {camera} = this;
new TWEEN.Tween( { lat : 0, y : camera.position.y, fov : camera.fov } )
.to( { lat: 90, y : 0, fov : 100 }, 2500 )
.delay(1000)
.easing(TWEEN.Easing.Cubic.InOut)
.repeat(Infinity)
.onUpdate(function() {
let phi = THREE.Math.degToRad( this.lat );
camera.target.y = -500 * Math.cos( phi );
camera.target.z = -500 * Math.sin( phi );
camera.position.y = this.y;
camera.fov = this.fov;
camera.updateProjectionMatrix();
})
.start()
}
animate() {
this.render();
requestAnimationFrame( ()=>{this.animate()});
}
render() {
TWEEN.update();
this.camera.lookAt( this.camera.target );
this.renderer.clear();
this.renderer.render( this.scene, this.camera );
}
}
new panorama();
</script>
</body>
</html>

  

 

three.js全景漫游实践的更多相关文章

  1. &lt&semi;Three&period;js&gt&semi;(第三节)全景漫游

    一.实验内容 通过上次实验,了解了Three.js创建场景的基本步骤.这一节,我们将通过Three.js实现全景漫游功能.如下图: 全景图是获取一个3D场景中的不同角度的图片,然后通过拼接.融合实现3 ...

  2. 打造H5里的&OpenCurlyDoubleQuote;3D全景漫游”秘籍

    近来风生水起的VR虚拟现实技术,抽空想起年初完成的“星球计划”项目,总结篇文章与各位分享一下制作基于Html5的3D全景漫游秘籍. QQ物联与深圳市天文台合作,在手Q“发现新设备”-“公共设备”里,连 ...

  3. 打造自己的3D全景漫游

    three.js 示例: ​ 打造H5里的"3D全景漫游"秘籍 - 腾讯ISUX ​ QQ物联星球计划 通过pano2vr直接将鱼眼全景图生成立体空间的六个面:也可通过Photos ...

  4. JS编程最佳实践

    最近花了一周时间把<编写可维护的js> 阅读了一遍, 现将全书提到的JS编程最佳实践总结如下, 已追来者! 1.return 之后不可直接换行, 否则会导致ASI(自动分号插入机制)会在r ...

  5. Atitit&period;angular&period;js 使用最佳实践 原理与常见问题解决与列表显示案例 attilax总结

    Atitit.angular.js 使用最佳实践 原理与常见问题解决与列表显示案例 attilax总结 1. 本文范围 1 2. Angular的优点 1 2.1. 双向数据绑定 1 2.2. dsl ...

  6. Cookie和Session在Node&period;JS中的实践(三)

    Cookie和Session在Node.JS中的实践(三) 前面作者写的COOKIE篇.SESSION篇,算是已经比较详细的说明了两者间的区别.机制.联系了.阅读时间可能稍长,因为作者本身作图也做了不 ...

  7. Cookie和Session在Node&period;JS中的实践(二)

    Cookie和Session在Node.JS中的实践(二) cookie篇在作者的上一篇文章Cookie和Session在Node.JS中的实践(一)已经是写得算是比较详细了,有兴趣可以翻看,这篇是s ...

  8. 《Vue&period;js 2&period;x实践指南》 已出版

    <Vue.js 2.x实践指南>其实在一年前就已经完稿了,只是由于疫情的缘故耽搁了很久才下厂印刷.阅读本书需要具备HTML.CSS和JS基础,本书针对的用户群体主要是:想要快速学习vue技 ...

  9. vue&period;js&plus;boostrap最佳实践

    一.为什么要写这篇文章 最近忙里偷闲学了一下vue.js,同时也复习了一下boostrap,发现这两种东西如果同时运用到一起,可以发挥很强大的作用,boostrap优雅的样式和丰富的组件使得页面开发变 ...

随机推荐

  1. 前端学HTTP之代理

    前面的话 Web代理(proxy)服务器是网络的中间实体,位于客户端和服务器之间,扮演“中间人”的角色,在各端点之间来回传送HTTP报文.本文将介绍HTTP代理服务器相关内容 中间实体 Web上的代理 ...

  2. JavaScript学习总结【12】、JS AJAX应用

    1.AJAX 简介 AJAX(音译为:阿贾克斯) = Asynchronous JavaScript and XML(异步的 JavaScript 和 XML),是指一种创建交互式网页应用的网页开发技 ...

  3. Solr -- Solr Facet 2

    solr将以导航为目的的查询结果称为facet. 它并不会修改查询结果信息, 只是在查询结果上根据分类添加了count信息, 然后用户根据count信息做进一步的查询, 比如淘宝的查询列表中, 上面会 ...

  4. Oracle case 关键字的使用

    select e.salary, --case 语句开始 case then salary else salary end new_salary --case 语句结束,可见也和存储过程等结束方式一样 ...

  5. 用Python学分析 - 散点图

    # 运用散点图对数据分布得到直观的认识 import numpy as np import matplotlib.pyplot as plt # 设计 x, y 轴 n = 10000 x = np. ...

  6. 多线程控制工具类--倒计时器CountDownLatch的使用(模仿火箭发射)

    package com.thread.test.Lock; import java.util.Random; import java.util.concurrent.CountDownLatch; i ...

  7. BInsertSort

    #include <bits/stdc++.h> using namespace std; #define MAXSIZE 200000 typedef int KeyType; type ...

  8. jquery实现分页&plus;单删批删

    //定义一个分页的方法 public function fenye(){ //查询满足条件的总条数 $count = M("regis")->count(); //设置每页显 ...

  9. socke&plus;epoll

    读: #define V5CLI_READ_MSG_LEN 1024 char readMsg[V5CLI_READ_MSG_LEN]; ; ; )) > ){ n += nread; }//读 ...

  10. 记一次MSSQL到MySQL大数据迁移过程

    工作中遇到一个需求 要将MSSQL数据库*计12张表的数据大概1000W行数据迁移到MySQL数据库中,去年另一个同事负责这件事情,他采用的方法是先将MSSQL数据库里的数据生成同MySQL数据库表 ...