webGL(web图形库)是一个中JavaScript API,用于在web浏览器中呈现交互式2D与3D图形,而且无需使用插件。
ThreeJs 是一款WebGL框架(在其API接口基础上又进行了一层封装),
Three.js是基于原生WebGL封装运行的三维引擎
git地址
下载dev 压缩包
工程化开发,在vue里面npm下载three.js
import 引入
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<script src="../three.js-dev/build/three.js"></script>
<script src="../three.js-dev/examples/js/controls/OrbitControls.js"></script>
</head>
<body>
<script>
//虚拟场景+相机+渲染器=效果
// //创建一个三维场景
const scene = new THREE.Scene()
// //创建一个几何体长宽高
const geometry = new THREE.BoxGeometry(100, 100, 100)
// //设置材质和颜色
const material = new THREE.MeshLambertMaterial({
transparent: true, //开启透明
opacity: 0.5, //设置透明度
color: 0xc708fd
}) //设置完会全黑,需要设置光源
// //创个一个网格模型对象
const mesh = new THREE.Mesh(geometry, material)
// //给网格模型添加到三维场景
scene.add(mesh);
// mesh.position.set(100, 100, 100) //设置在坐标轴的位置
// mesh.rotateX(Math.PI / 4) //旋转
//设置环境光源
const Ambient = new THREE.AmbientLight(0xffffff, 0.5); //环境光
scene.add(Ambient);
//设置点光源
const Point = new THREE.PointLight(0xffffff, 1);
Point.position.set(100, 100, 100); //200,200,200是在三个角的夹角
scene.add(Point);
//设置可视化点光源 一个白色三角 光照过来的位置
const pointLightHelper = new THREE.PointLightHelper(Point, 10)
scene.add(pointLightHelper)
//设置坐标轴
const axesHelper = new THREE.AxesHelper(150);
scene.add(axesHelper);
// //怎么呈现到web网页上:通过相机和渲染器把三维场景scene渲染出来
// //1.创建一个透视相机
const width = 800
const height = 500
//45:视场角度,width/height:canvas画布宽高比,1:近裁截面,3000,远裁截面
const camera = new THREE.PerspectiveCamera(45, width / height, 1, 1000) //透视投影
camera.position.set(200, 200, 200) //相机所在位置
camera.lookAt(0, 0, 0) //相机所照方向 (0,0,0原点)位置+坐标会构成一条线决定你的观察方向
// //创建webGl渲染器
const renderer = new THREE.WebGLRenderer()
renderer.setSize(width, height) //设置渲染区域尺寸
renderer.setClearColor(0xb9d3ff, 1); //设置背景颜色
renderer.render(scene, camera) //渲染操作
document.body.appendChild(renderer.domElement) //body元素中插入canvas对象,相当于canvas的画布
//使用OrbitControls轨道控制器需要引入(可以360°转动)
const controls = new THREE.OrbitControls(camera, renderer.domElement)
// //使用OrbitControls轨道控制器需,重新进行更新
// controls.addEventListener('change', function () {
// renderer.render(scene, camera) //重新渲染
// })
//动画使用 requestAnimationFrame()实现动画效果,有动画渲染就不需要controls.addEventListener('change', function () {})
//动画的原理就是一张一张拼起来的照片,只要运动够快就是动画
//1.通过照相每次改变都需要从新呈现
const clock = new THREE.Clock() //获取上一次和这次渲染时间间隔
function render() {
const spt = clock.getDelta * 1000 //渲染时间间隔毫秒
const zl = 1000 / spt //渲染帧率 一秒渲染多少次 比如一秒渲染60次
renderer.render(scene, camera)
mesh.rotateY(0.01)
requestAnimationFrame(render)
}
render()
</script>
</body>
</html>
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<script src="../three.js-dev/build/three.js"></script>
<script src="../three.js-dev/examples/js/controls/OrbitControls.js"></script>
</head>
<body>
<script>
//虚拟场景+相机+渲染器=效果
// //创建一个三维场景
const scene = new THREE.Scene()
// //创建一个几何体长宽高
const geometry = new THREE.BoxGeometry(100, 100, 100)
// //设置材质和颜色
const material = new THREE.MeshLambertMaterial({
transparent: true, //开启透明
opacity: 0.5, //设置透明度
color: 0xc708fd
}) //设置完会全黑,需要设置光源
// //创个一个网格模型对象
// const mesh = new THREE.Mesh(geometry, material)
// // //给网格模型添加到三维场景
// scene.add(mesh);
for (let i = 0; i < 10; i++) { //批量创建很多个立方体
for (let j = 0; j < 10; j++) {
const mesh = new THREE.Mesh(geometry, material)
mesh.position.set(i * 200, 0, j * 200, )
scene.add(mesh);
}
}
// mesh.position.set(100, 100, 100) //设置在坐标轴的位置
// mesh.rotateX(Math.PI / 4) //旋转
//设置环境光源
const Ambient = new THREE.AmbientLight(0xffffff, 0.5); //环境光
scene.add(Ambient);
//设置点光源
const Point = new THREE.PointLight(0xffffff, 1);
Point.position.set(100, 100, 100); //200,200,200是在三个角的夹角
scene.add(Point);
//设置可视化点光源 一个白色三角 光照过来的位置
// const pointLightHelper = new THREE.PointLightHelper(Point, 10)
// scene.add(pointLightHelper)
//设置坐标轴
const axesHelper = new THREE.AxesHelper(150);
scene.add(axesHelper);
// //怎么呈现到web网页上:通过相机和渲染器把三维场景scene渲染出来
// //1.创建一个透视相机
const width = 800
const height = 500
//45:视场角度,width/height:canvas画布宽高比,1:近裁截面,3000,远裁截面
const camera = new THREE.PerspectiveCamera(45, width / height, 1, 3000) //透视投影
camera.position.set(800, 800, 800) //相机所在位置
camera.lookAt(1000, 0, 1000) //相机所照方向 (0,0,0原点)位置+坐标会构成一条线决定你的观察方向
// lookAt会影响OrbitControls轨道控制器的区域
// //创建webGl渲染器
const renderer = new THREE.WebGLRenderer()
renderer.setSize(width, height) //设置渲染区域尺寸
renderer.setClearColor(0xb9d3ff, 1); //设置背景颜色
renderer.render(scene, camera) //渲染操作
document.body.appendChild(renderer.domElement) //body元素中插入canvas对象,相当于canvas的画布
//使用OrbitControls轨道控制器需要引入(可以360°转动)
const controls = new THREE.OrbitControls(camera, renderer.domElement)
controls.target.set(1000, 0, 1000)
controls.update()
// //使用OrbitControls轨道控制器需,重新进行更新
// controls.addEventListener('change', function () {
// renderer.render(scene, camera) //重新渲染
// })
//动画使用 requestAnimationFrame()实现动画效果,有动画渲染就不需要controls.addEventListener('change', function () {})
//动画的原理就是一张一张拼起来的照片,只要运动够快就是动画
//1.通过照相每次改变都需要从新呈现
const clock = new THREE.Clock() //获取上一次和这次渲染时间间隔
function render() {
const spt = clock.getDelta * 1000 //渲染时间间隔毫秒
const zl = 1000 / spt //渲染帧率 一秒渲染多少次 比如一秒渲染60次
renderer.render(scene, camera)
// mesh.rotateY(0.01)
requestAnimationFrame(render)
}
render()
</script>
</body>
</html>
//几大难点
1.太阳光源:方向光DirectionalLight
1.确认太阳位置:通过SunCalc.js js库 传入经纬度时间可以得出太阳的高度角,方位角,然后通过three的方法角度转弧度,转换成方向光的position
2.设置阴影的属性:密度,最远.最近距离,清晰度....
3.阴影水波纹的问题: this.directionalLight.shadow.bias = -0.01;(调整阴影偏差)
4.画模型加压缩:用SketchUp(草图大师),导出gltf模型,gltf-pipeline压缩,放到七牛云上
5.载入模型:gltf-pipeline压缩后使用DRACOLoader加载
6.网页端和移动端:mousedown,touchstart/touchmove/touchend
7.小程序镶嵌网页:日期格式,window获取不到问题
<template>
<div id="sunshine" style="width: 100vw; height: 100vh; overflow: hidden">
<div class="compass">
<img
src="../assets/images/compass.png"
class="icon"
style="transform: rotate(0deg)"
/>
</div>
<div class="page-wrappr" id="app2">
<div class="bottom-card">
<div class="top">
<h4 class="title">{{ newhouseName }}</h4>
<div class="date" style="float: right">
<span class="text"
><el-date-picker
v-model="date"
type="date"
:editable="false"
placeholder="选择日期"
format="yyyy/MM/dd"
value-format="yyyy/MM/dd"
@change="changDate"
@focus="focusDate"
>
</el-date-picker></span
><i class="van-icon van-icon-arrow"> </i>
</div>
</div>
<div class="time-wrapper">
<div class="play-btn play-btn-bf" @click="bfBtnFun()">
<svg
v-if="isPlay"
t="1689756307920"
class="icon"
viewBox="0 0 1024 1024"
version="1.1"
xmlns="http://www.w3.org/2000/svg"
p-id="19031"
width="48"
height="48"
>
<path
d="M426.666667 138.666667v746.666666a53.393333 53.393333 0 0 1-53.333334 53.333334H266.666667a53.393333 53.393333 0 0 1-53.333334-53.333334V138.666667a53.393333 53.393333 0 0 1 53.333334-53.333334h106.666666a53.393333 53.393333 0 0 1 53.333334 53.333334z m330.666666-53.333334H650.666667a53.393333 53.393333 0 0 0-53.333334 53.333334v746.666666a53.393333 53.393333 0 0 0 53.333334 53.333334h106.666666a53.393333 53.393333 0 0 0 53.333334-53.333334V138.666667a53.393333 53.393333 0 0 0-53.333334-53.333334z"
fill="#f01e03"
p-id="19032"
></path>
</svg>
<svg
v-else
t="1689756265979"
class="icon"
viewBox="0 0 1024 1024"
version="1.1"
xmlns="http://www.w3.org/2000/svg"
p-id="17742"
width="48"
height="48"
>
<path
d="M870.2 466.333333l-618.666667-373.28a53.333333 53.333333 0 0 0-80.866666 45.666667v746.56a53.206667 53.206667 0 0 0 80.886666 45.666667l618.666667-373.28a53.333333 53.333333 0 0 0 0-91.333334z"
fill="#f01e03"
p-id="17743"
></path>
</svg>
</div>
<div class="time-slider">
<span class="time">04:49</span>
<div class="van-slider van-slider" style="height: 4px">
<div
class="van-slider__bar"
style="width: 0; height: 4px; background: rgb(255, 204, 204)"
>
<div
role="slider"
tabindex="0"
class="van-slider__button-wrapper"
@mousedown.prevent="SunIconMousestartPC"
style="position: absolute; top: -15px; left: 0%"
v-if="isPc"
>
<div class="custom-button">
<img
src=""
alt=""
class="sun-icon"
/>
</div>
</div>
<div
role="slider"
tabindex="0"
class="van-slider__button-wrapper"
@touchstart.prevent="SunIconMousestart"
@touchmove.prevent="SunIconMousemove"
@touchend.prevent="SunIconMouseend"
style="position: absolute; top: -15px; left: 0%"
v-else
>
<div class="custom-button">
<img
src=""
alt=""
class="sun-icon"
/>
</div>
</div>
</div>
</div>
<span class="time">19:40</span>
</div>
<span class="current-time">10:43</span>
</div>
<div class="time-labels">
<span class="rise-time">日出时间</span
><span class="set-time">日落时间</span><span class="value"> </span>
</div>
<div class="dates">
<button class="btn" @click="jq(0)">大寒</button
><button class="btn" @click="jq(1)">春分</button
><button class="btn" @click="jq(2)">夏至</button
><button class="btn" @click="jq(3)">秋分</button
><button class="btn" @click="jq(4)">立冬</button
><button class="btn" @click="jq(5)">冬至</button>
</div>
</div>
</div>
</div>
</template>
<script>
import dayjs from "dayjs";
import * as THREE from "three";
import { OrbitControls } from "three/examples/jsm/controls/OrbitControls";//控制器
import * as SunCalc from "../assets/js/suncalc.js";//JS库用来获取太阳位置,日出日落时间
import { GLTFLoader } from "three/examples/jsm/loaders/GLTFLoader.js";//glft模型
import { getSunshineNewhouse } from "@/api/house/newhouse";
import { DRACOLoader } from "three/examples/jsm/loaders/DRACOLoader.js";//gltf-pipeline压缩后使用DRACOLoader加载
export default {
name: "sunshine",
metaInfo() {
return {
title: "房大地-日照分析",
};
},
data() {
return {
imageUrl: "https://****.com/",
renderer: null,
latitude: 39.899746, //经纬度
longitude: 116.409337,
date: null,
camera: null, //相机
scene: null, //屏幕
ambientLight: null, //光照
directionalLight: null, //光照
SunIcon: null, //滑动块
vanSliderBar: null, //阴影进度条
vanSlider: null, //最长进度条
bfBtn: null, //播放按钮
ztBtn: null, //暂停按钮
HS: 0, //开始小时
MS: 0, //开始分钟
HL: 0, //结束小时
ML: 0, //结束分钟
timer: null, //计时器
sunrise: null, // 日出时间
sunset: null, // 日落时间
timeDifference: null, //时间差日落-日出
cube: null,
plane: null,
cube2: null,
controls: null, //性能插件
isPlay: true, //播放按钮
DTURL: "", //底图url
HouseURL: "", //3d模型url
building: null,
newhouseName: "",
offsetDegree: 0,
modelX: 0, //模型位置x
modelY: 0, //模型位置y
modelZ: 0, //模型位置z
modelScale: "", //模型比例
planeScale: "", //底图比例
x1: 0,
x2: 0, //滑块距离左边的位置
distance: 0,
isPc: true,
timerNum: 100, //
intervalNum: 10,
isDivMove: false, //小太阳是否拖拽
intersects: [], //几何体合计-点击事件
};
},
watch: {},
created() {},
mounted() {
//判断是什么环境,手机还是电脑
if (document.documentElement.clientWidth < 720) {
this.isPc = false;
this.timerNum = 1;
this.intervalNum = 100;
} else {
this.isPc = true;
this.timerNum = 1;
this.intervalNum = 10;
}
this.getSunshineFun();
},
methods: {
//获取3d模型和底图数据
getSunshineFun() {
getSunshineNewhouse(this.$route.params.id).then((res) => {
if (res.data) {
this.DTURL = this.imageUrl + res.data.backUrl;
this.HouseURL = this.imageUrl + res.data.modelUrl;
this.offsetDegree = res.data.offsetDegree;
this.latitude = res.data.newhouseLat;
this.longitude = res.data.newhouseLng;
this.newhouseName = res.data.newhouseName;
this.modelX = res.data.modelX;
this.modelY = res.data.modelY;
this.modelZ = res.data.modelZ;
this.modelScale = res.data.modelScale;
this.planeScale = res.data.planeScale;
//获取今天日期
var D = new Date();
this.date = new Date(D.getFullYear(), D.getMonth(), D.getDate());
this.SunIcon = document.querySelector(".van-slider__button-wrapper");
this.vanSliderBar = document.querySelector(".van-slider__bar");
this.vanSlider = document.querySelector(".van-slider");
this.bfBtn = document.querySelector(".play-btn-bf"); //播放按钮
this.ztBtn = document.querySelector(".play-btn-zt"); //暂停按钮
this.initObj();
this.initRender();
this.initScene();
this.initCamera();
this.initLight();
this.initModel();
this.initControls();
this.animate();
this.SunCalcGetTimes();
window.addEventListener("resize", this.onWindowResize);
}
});
},
//初始相机
initCamera() {
this.camera = new THREE.PerspectiveCamera(
45,
window.innerWidth / window.innerHeight,
1,
3000
);
this.camera.position.set(0, 300, 250); //相机所在的位置,默认为(0,0,0)个单位;
this.camera.lookAt(new THREE.Vector3(0, 0, 0));
//点击事件
},
//初始虚拟场景
initScene() {
this.scene = new THREE.Scene();
this.scene.background = new THREE.Color("#b2e0ff");
this.scene.add(new THREE.AmbientLight(0xffffff, 1)); // 环境光
this.scene.receiveShadow = true;
},
//初始three 虚拟场景+相机+渲染器=场景
initRender() {
this.renderer = new THREE.WebGLRenderer({
antialias: true,
logarithmicDepthBuffer: true,
alpha: true,
});
this.renderer.setSize(window.innerWidth, window.innerHeight);
//告诉渲染器需要阴影效果
this.renderer.shadowMap.enabled = true;
this.renderer.outputEncoding = THREE.SRGBColorSpace; //RGB模式编码(sRGBEncoding)进行对材质进行渲染,SRGBColorSpace
this.renderer.shadowMap.type = THREE.PCFSoftShadowMap; // 默认的是,没有设置的这个清晰 THREE.PCFShadowMap
this.renderer.toneMapping = THREE.ReinhardToneMapping;
this.renderer.toneMappingExposure = 1.1; // 默认1为了让场景更明亮
document.querySelector("#sunshine").appendChild(this.renderer.domElement);
},
// 加载楼盘模型
initObj() {
let _this = this;
var loader = new GLTFLoader();
//gltf-pipeline压缩后使用DRACOLoader加载, gltf-pipeline -i 111.gltf -o modelDraco.gltf -d
const dracoLoader = new DRACOLoader();
// 设置Draco解码库 该路径就是解压文件所在的位置,https://www.gstatic.com/draco/versioned/decoders/1.5.6/或者把node_modules/three/examples/jsm/libs/draco文件复制到public文件下
dracoLoader.setDecoderPath("/draco/");
// 使用js方式解压
dracoLoader.setDecoderConfig({ type: "js" });
// 初始化_initDecoder 解码器
dracoLoader.preload();
// 设置gltf加载器dracoLoader解码器
loader.setDRACOLoader(dracoLoader);
loader.load(
this.HouseURL,
(gltf) => {
const model = gltf.scene;
gltf.scene.name = "模型1";
// 遍历模型中的所有子对象,设置阴影接收和投射属性
model.traverse(function (child) {
if (child.isMesh) {
// 重新设置材质
let scaleNum = _this.modelScale ? _this.modelScale : 2;
child.scale.set(scaleNum, scaleNum, scaleNum); // 设置模型大小缩放
model.position.x = _this.modelX != 0 ? _this.modelX : 300;
model.position.y = _this.modelY != 0 ? _this.modelY : 0;
model.position.z = _this.modelZ != 0 ? _this.modelZ : -300;
child.material.side = THREE.DoubleSide;
//物体遮挡阴影
child.castShadow = true;
child.receiveShadow = true;
}
});
_this.building = gltf.scene;
_this.scene.add(gltf.scene);
},
undefined,
function (error) {
console.error(error);
}
);
},
//底部草地+底图
initModel() {
//底图
var planeGeometry = new THREE.CircleGeometry(110, 64);
var texture = new THREE.TextureLoader().load(this.DTURL);
texture.encoding = THREE.SRGBColorSpace;
var planeMaterial = new THREE.MeshStandardMaterial({
map: texture,
});
this.plane = new THREE.Mesh(planeGeometry, planeMaterial);
this.plane.rotation.x = -0.5 * Math.PI;
this.plane.position.y = 0;
//物体遮挡阴影
this.plane.castShadow = true;
//告诉底部平面需要接收阴影
this.plane.receiveShadow = true;
this.scene.add(this.plane);
//草地
var planeGeometry1 = new THREE.CircleGeometry(150, 64);
var texture1 = new THREE.TextureLoader().load("/bj.png");
var planeMaterial1 = new THREE.MeshStandardMaterial({
map: texture1,
});
this.plane1 = new THREE.Mesh(planeGeometry1, planeMaterial1);
this.plane1.rotation.x = -0.5 * Math.PI;
this.plane1.position.y = -1;
//物体遮挡阴影
this.plane1.castShadow = true;
//告诉底部平面需要接收阴影
this.plane1.receiveShadow = true;
this.scene.add(this.plane1);
},
//日出日落时间
SunCalcGetTimes() {
if (typeof this.date == "string") {
this.date = new Date(this.date);
} else {
this.date = new Date(
this.date.getFullYear(),
this.date.getMonth(),
this.date.getDate()
); // 使用当前日期和时间
}
var times = SunCalc.getTimes(this.date, this.latitude, this.longitude);
let sunriseD = dayjs(times.sunrise);
let sunriseF = sunriseD.format("HH:mm");
// 日出
this.sunrise = sunriseF;
// 日落
this.sunset = dayjs(times.sunset).format("HH:mm");
document.querySelectorAll(".time")[0].innerHTML = this.sunrise;
document.querySelectorAll(".time")[1].innerHTML = this.sunset;
this.timeDifference = this.calculateTimeDifference(
this.sunrise,
this.sunset
);
document.querySelector(".time-labels .value").innerHTML =
"昼长:" +
this.timeDifference.hours +
"小时" +
this.timeDifference.minutes +
"分钟";
this.HS = this.sunrise.slice(0, 2) * 1;
this.MS = this.sunrise.slice(3, 5) * 1;
this.HL = this.sunset.slice(0, 2) * 1;
this.ML = this.sunset.slice(3, 5) * 1;
this.interval();
},
//计算太阳位置
SunCalcGetPosition(date) {
var sunPosition = SunCalc.getPosition(
date,
this.latitude,
this.longitude
);
// 太阳在Three.js坐标系中的位置向量
const sunDirection = new THREE.Vector3();
if (this.offsetDegree > 0) {
//角度转弧度
var offSetRad = THREE.MathUtils.degToRad(offsetDegree);
sunDirection.setFromSphericalCoords(
1,
Math.PI / 2 - sunPosition.altitude,
-sunPosition.azimuth - offSetRad
);
} else {
sunDirection.setFromSphericalCoords(
1,
Math.PI / 2 - sunPosition.altitude,
-sunPosition.azimuth
);
}
sunDirection.normalize();
var sunlightPosition = sunDirection.clone().multiplyScalar(200); //光源到原点的距离
//设置太阳位置
this.directionalLight.position.copy(sunlightPosition);
this.directionalLight.target.position.set(0, 0, 0);
},
//太阳光阴影
initLight(a, b) {
this.directionalLight = new THREE.DirectionalLight(0xffffff);
this.directionalLight.visible = true;
this.directionalLight.intensity = 3; //光线的密度,默认为1。 光照越强,物体表面就更明亮
this.directionalLight.shadow.camera.near = -300; //产生阴影的最近距离
this.directionalLight.shadow.camera.far = 300; //产生阴影的最远距离
this.directionalLight.shadow.camera.left = -300; //产生阴影距离位置的最左边位置
this.directionalLight.shadow.camera.right = 300; //最右边
this.directionalLight.shadow.camera.top = 300; //最上边
this.directionalLight.shadow.camera.bottom = -300; //最下面
this.directionalLight.shadow.bias = -0.01; //用于解决阴影水波纹条纹阴影的问题
this.directionalLight.shadow.mapSize.set(2048, 2048); //阴影清晰度
//告诉平行光需要开启阴影投射,物体遮挡阴影
this.directionalLight.castShadow = true;
this.scene.add(this.directionalLight);
},
//按住滑块拖动PC端
SunIconMousestart(e) {
if (this.timer) {
clearInterval(this.timer);
this.timer = null;
}
// 获取鼠标在当前事件源的位置
this.x1 = e.touches[0].clientX;
this.x2 = this.SunIcon.style.left.split("px")[0] * 1;
},
// 小太阳滑块PC端
SunIconMousemove(e) {
this.distance = e.touches[0].clientX - this.x1;
if (this.SunIcon.style.left.split("px")[0] * 1 <= 0) {
this.SunIcon.style.left = 0;
this.vanSliderBar.style.width = 0 + "px";
this.HS = this.sunrise.slice(0, 2) * 1;
this.MS = this.sunrise.slice(3, 5) * 1;
} else if (
this.SunIcon.style.left.split("px")[0] * 1 >=
this.vanSlider.clientWidth - this.SunIcon.clientWidth
) {
this.SunIcon.style.left =
this.vanSlider.clientWidth - this.SunIcon.clientWidth + "px";
this.vanSliderBar.style.width =
this.vanSlider.clientWidth - this.SunIcon.clientWidth + "px";
this.HS = this.sunset.slice(0, 2) * 1;
this.MS = this.sunset.slice(3, 5) * 1;
} else {
this.SunIcon.style.left = this.x2 + this.distance + "px";
this.vanSliderBar.style.width =
this.x2 + this.distance + this.SunIcon.clientWidth + "px";
}
//百分比
var bfb = (
(this.SunIcon.style.left.split("px")[0] * 1) /
(this.vanSlider.clientWidth - this.SunIcon.clientWidth)
).toFixed(2);
//进度条右侧时间
//最后时间
var l = this.sunset.slice(0, 2) * 60 + this.sunset.slice(3, 5) * 1;
//开始时间
var s = this.sunrise.slice(0, 2) * 60 + this.sunrise.slice(3, 5) * 1;
var a = (l - s) * bfb;
var h = Math.floor((a + s) / 60);
var m = ((s + a) % 60).toFixed(0) * 1;
this.HS = h;
this.MS = m;
document.querySelector(".current-time").innerHTML =
(h < 10 ? "0" + h : h) + ":" + (m < 10 ? "0" + m : m);
},
// 松开小太阳移动端
SunIconMouseend(e) {
this.isPlay = true;
this.interval();
},
//按住太阳拖拽移动端
SunIconMousestartPC(e) {
this.isDivMove = true;
if (this.timer) {
clearInterval(this.timer);
this.timer = null;
}
// 获取鼠标在当前事件源的位置
var x1 = e.clientX - this.vanSlider.offsetLeft - this.SunIcon.offsetLeft;
var _this = this;
document.onmousemove = function (e) {
var x2 = e.clientX;
let distance = x2 - _this.vanSlider.offsetLeft - x1;
console.log(distance);
if (distance <= 0) {
distance = 0;
_this.HS = _this.sunrise.slice(0, 2) * 1;
_this.MS = _this.sunrise.slice(3, 5) * 1;
} else if (
distance >=
_this.vanSlider.clientWidth - _this.SunIcon.clientWidth
) {
distance = _this.vanSlider.clientWidth - _this.SunIcon.clientWidth;
_this.HS = _this.sunset.slice(0, 2) * 1;
_this.MS = _this.sunset.slice(3, 5) * 1;
} else {
_this.SunIcon.style.left = distance + "px";
_this.vanSliderBar.style.width =
distance + _this.SunIcon.clientWidth + "px";
}
//百分比
var bfb = (
distance /
(_this.vanSlider.clientWidth - _this.SunIcon.clientWidth)
).toFixed(2);
//进度条右侧时间
//最后时间
var l = _this.sunset.slice(0, 2) * 60 + _this.sunset.slice(3, 5) * 1;
//开始时间
var s = _this.sunrise.slice(0, 2) * 60 + _this.sunrise.slice(3, 5) * 1;
var a = (l - s) * bfb;
var h = Math.floor((a + s) / 60);
var m = ((s + a) % 60).toFixed(0) * 1;
_this.HS = h;
_this.MS = m;
document.querySelector(".current-time").innerHTML =
(h < 10 ? "0" + h : h) + ":" + (m < 10 ? "0" + m : m);
};
document.onmouseup = function (e) {
_this.isDivMove = false;
_this.isPlay = true;
_this.interval();
document.onmousemove = null;
document.onmouseup = null;
};
},
//初始化性能插件
initControls() {
this.controls = new OrbitControls(this.camera, this.renderer.domElement);
// 使动画循环使用时阻尼或自转 意思是否有惯性
this.controls.enableDamping = true;
//是否可以缩放
this.controls.enableZoom = true;
//是否自动旋转
this.controls.autoRotate = false;
//设置相机距离原点的最远距离
this.controls.minDistance = 200;
//设置相机距离原点的最远距离
this.controls.maxDistance = 600;
//设置相机上下旋转最大角度最大到平面
this.controls.minPolarAngle = 0;
this.controls.maxPolarAngle = Math.PI / 2 - 0.1;
this.controls.addEventListener("change", (e) => {
// 获取对象的旋转角度;
const rotation = this.camera.rotation;
// 判断中心轴旋转方向
// 定义变量来跟踪旋转角度
let rotationDegrees = {
x: THREE.MathUtils.radToDeg(rotation.x),
y: THREE.MathUtils.radToDeg(rotation.y),
z: THREE.MathUtils.radToDeg(rotation.z),
};
//左上角指南针转动
document.querySelector(".compass img").style.transform =
"rotate(" + rotationDegrees.z + "deg)";
});
},
render() {
this.renderer.render(this.scene, this.camera);
},
//窗口变动触发的函数
onWindowResize() {
this.$nextTick(() => {
this.camera.aspect = window.innerWidth / window.innerHeight;
this.camera.updateProjectionMatrix();
this.render();
this.renderer.setSize(window.innerWidth, window.innerHeight);
});
},
animate() {
//更新控制器
this.render();
//更新性能插件
requestAnimationFrame(this.animate);
},
//计时器
interval() {
if (this.timer) {
clearInterval(this.timer);
this.timer = null;
}
//控制分钟增加
this.timer = setInterval(() => {
if (this.MS >= 60) {
this.MS = 1;
this.HS = this.HS + 1;
} else {
this.MS += this.timerNum;
}
if (this.HS >= this.HL && this.MS >= this.ML) {
this.HS = this.sunrise.slice(0, 2) * 1;
this.MS = this.sunrise.slice(3, 5) * 1;
}
// 更新时间
this.date = new Date(
this.date.getFullYear(),
this.date.getMonth(),
this.date.getDate(),
this.HS,
this.MS
);
// 更新光照的方向和颜色
this.SunCalcGetPosition(this.date);
//计算相差时间比
//进度条右侧时间
const timeDifference2 = this.calculateTimeDifference(
this.sunrise,
this.HS + ":" + this.MS
);
let baifenb =
(
(timeDifference2.hours * 60 + timeDifference2.minutes) /
(this.timeDifference.hours * 60 + this.timeDifference.minutes)
).toFixed(2) * 100;
let currentTimeEle = document.querySelector(".current-time");
currentTimeEle.innerHTML =
(this.HS < 10 ? "0" + this.HS : this.HS) +
":" +
(this.MS < 10 ? "0" + this.MS : this.MS);
//设置阴影滚动条长度
this.vanSliderBar.style.width =
((this.vanSlider.clientWidth - this.SunIcon.clientWidth) * baifenb) /
100 +
"px";
this.SunIcon.style.left =
((this.vanSlider.clientWidth - this.SunIcon.clientWidth) * baifenb) /
100 +
"px";
}, this.intervalNum);
},
//点击播放按钮
bfBtnFun() {
console.log(this.isPlay);
if (this.isPlay) {
clearInterval(this.timer);
this.timer = null;
this.isPlay = false;
} else {
this.interval();
this.isPlay = true;
}
},
//计算两时间差
calculateTimeDifference(startTime, endTime) {
const start = new Date();
const end = new Date();
// 将时间字符串转换为 Date 对象
const [startHour, startMinute] = startTime.split(":");
const [endHour, endMinute] = endTime.split(":");
start.setHours(startHour, startMinute, 0);
end.setHours(endHour, endMinute, 0);
// 计算时间间隔(毫秒为单位)
const timeDiff = end.getTime() - start.getTime();
// 将毫秒转换为小时和分钟
const hours = Math.floor(timeDiff / (1000 * 60 * 60));
const minutes = Math.floor((timeDiff % (1000 * 60 * 60)) / (1000 * 60));
return {
hours,
minutes,
};
},
//计算节气日期
jq(num) {
var solarTerm = new Array("小寒", "春分", "夏至", "秋分", "立冬", "冬至");
var sTermInfo = new Array(0, 107014, 240693, 375494, 440795, 504758);
function sTermDate(y, n) {
return new Date(
31556925974.7 * (y - 1900) +
sTermInfo[n] * 60000 +
Date.UTC(1900, 0, 6, 2, 5)
);
}
//获取2013年第一个节气小寒的公历日期
var Y = new Date().getFullYear();
var sterm = sTermDate(Y, num);
if (this.timer) {
clearInterval(this.timer);
this.timer = null;
}
this.isPlay = true;
this.date = sterm;
this.SunCalcGetTimes();
},
//改变日期
changDate(e) {
var that = this;
var date = new Date(e);
var y = date.getFullYear(); // 年
var m = date.getMonth() + 1; // 月
m = m < 10 ? "0" + m : m;
var d = date.getDate(); // 日
d = d < 10 ? "0" + d : d;
that.date = y + "/" + m + "/" + d; //拼在一起
this.SunCalcGetTimes();
this.isPlay = true;
},
//日期选择器获取焦点
focusDate() {
//禁止软键盘弹出
document.activeElement.blur();
clearInterval(this.timer);
this.timer = null;
},
},
};
</script>
<style rel="stylesheet/scss" lang="scss" scoped>
::v-deep .el-date-editor.el-input,
.el-date-editor.el-input__inner {
width: 150px;
}
#sunshine {
position: relative;
}
@media only screen and (min-width: 500px) {
.bottom-card {
position: absolute;
top: auto;
bottom: 0;
left: 50%;
width: 100vw;
transform-origin: center bottom;
transform: translateX(-50%) scale(0.46);
}
}
.bottom-card {
position: relative;
top: -4.26667vw;
padding: 4.26667vw 4vw;
background: #fff;
border-radius: 4.26667vw 4.26667vw 0 0;
z-index: 999;
color: #333330;
}
#app2 {
font-family: Avenir, Helvetica, Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
color: #333;
overflow: hidden;
font-size: 3.73333vw;
}
.page-wrapper {
position: relative;
height: 100vh;
overflow: hidden;
display: flex;
flex-flow: column nowrap;
background: #fff;
}
.canvas-container {
width: 100vw !important;
flex: 1;
}
.bottom-card {
position: relative;
top: -4.26667vw;
padding: 4.26667vw 4vw;
background: #fff;
border-radius: 4.26667vw 4.26667vw 0 0;
z-index: 999;
color: #333330;
}
.bottom-card .top {
display: flex;
justify-content: space-between;
line-height: 4.26667vw;
}
.bottom-card .top .title {
flex: 1;
font-size: 4.8vw;
font-family: PingFangSC-Semibold, PingFang SC;
font-weight: 600;
color: #333330;
line-height: 6.66667vw;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.bottom-card .top .date {
display: flex;
justify-content: flex-end;
align-items: center;
font-size: 3.73333vw;
}
.top .date .text {
margin-right: 0.8vw;
}
.bottom-card .time-wrapper {
display: flex;
flex-flow: row nowrap;
align-items: center;
justify-content: center;
line-height: 5.33333vw;
}
.bottom-card .time-wrapper .play-btn {
width: 9.86667vw;
height: 9.86667vw;
font-size: 0;
margin-left: -1.6vw;
border: 1px solid #ff5d5d;
border-radius: 50%;
box-shadow: 0px 0px 5px #ffa2a2;
text-align: center;
line-height: 9.86667vw;
}
.bottom-card .time-wrapper .play-btn svg {
vertical-align: middle;
}
.van-slider {
position: relative;
width: 100%;
height: 2px;
background-color: #ebedf0;
border-radius: 999px;
cursor: pointer;
}
.bottom-card .time-wrapper .time-slider {
flex: 1;
display: flex;
align-items: center;
padding: 0 1.06667vw;
font-size: 3.73333vw;
}
.bottom-card .time-wrapper .time-slider .van-slider {
margin: 0 2.66667vw;
}
.bottom-card .time-wrapper .time-slider .van-slider .custom-button {
display: flex;
align-items: center;
justify-content: center;
}
.bottom-card .time-wrapper .time-slider .van-slider .custom-button .sun-icon {
width: 4.26667vw;
height: 4.26667vw;
}
.bottom-card .time-wrapper .current-time {
width: 19.46667vw;
height: 9.86667vw;
line-height: 9.86667vw;
text-align: center;
font-size: 3.2vw;
font-family: PingFangSC-Semibold, PingFang SC;
font-weight: 600;
color: #f44;
margin-right: -1.6vw;
background: url();
background-size: 100% 100%;
}
.bottom-card .time-labels {
position: relative;
width: 65.86667vw;
margin-left: 8.26667vw;
font-size: 3.2vw;
text-align: center;
}
.bottom-card .time-labels .rise-time {
position: absolute;
left: -0.8vw;
}
.bottom-card .time-labels .set-time {
position: absolute;
right: -0.8vw;
}
.bottom-card .times {
display: flex;
flex-flow: row nowrap;
justify-content: space-around;
}
.bottom-card .times .value {
margin-left: 1.06667vw;
color: #ff3a3a;
}
.bottom-card .dates {
display: flex;
justify-content: space-between;
margin-top: 4vw;
}
.bottom-card .dates .btn {
flex: 1;
line-height: 6.4vw;
font-size: 3.46667vw;
text-align: center;
background: #eee;
border-radius: 1.06667vw;
color: #333330;
border: none;
padding: 0;
margin: 0 1.6vw;
}
.bottom-card .dates .btn:first-child {
margin-left: 0;
}
.bottom-card .dates .btn:last-child {
margin-right: 0;
}
.bottom-card .dates .active {
color: #fff;
background: #f44;
}
.compass {
position: absolute;
top: 5.33333vw;
left: 5.33333vw;
font-size: 0;
border-radius: 50%;
overflow: hidden;
z-index: 999;
}
.compass .icon {
width: 18.13333vw;
height: 18.13333vw;
}
.van-calendar {
padding-bottom: 6.4vw;
}
.ipx .bottom-card {
padding-bottom: 8.53333vw;
}
.ipx .van-calendar {
padding-bottom: 13.33333vw;
}
.user-guide {
position: absolute;
top: 0;
width: 100%;
height: 100%;
background: rgba(0, 0, 0, 0.7);
display: flex;
flex-flow: column;
align-items: center;
justify-content: center;
z-index: 1000;
}
.user-guide .touch-guide {
width: 44.26667vw;
height: 42.66667vw;
}
.user-guide .close-btn {
width: 43.2vw;
height: 15.33333vw;
margin-top: 5.33333vw;
}
.css2DLabel {
position: relative;
font-size: 2.4vw;
line-height: 4vw;
border-radius: 0.53333vw;
color: #fff;
padding: 0 1.06667vw;
background: rgba(0, 0, 0, 0.4);
}
.css2DLabel:after {
content: "";
position: absolute;
top: 100%;
left: 50%;
width: 1px;
height: 4.26667vw;
border: none;
background: linear-gradient(180deg, #fff, hsla(0, 0%, 100%, 0) 80%);
}
@media only screen and (min-width: 500px) {
#app {
font-size: 24px;
}
.page-wrapper {
display: block;
}
.canvas-container {
width: 100vw !important;
height: 100vh !important;
}
.bottom-card {
position: absolute;
top: auto;
bottom: 0;
left: 50%;
width: 100vw;
transform-origin: center bottom;
transform: translateX(-50%) scale(0.46);
}
.compass {
top: 20px;
left: 20px;
}
.compass .icon {
width: 80px;
height: 80px;
}
.css2DLabel {
font-size: 12px;
line-height: 1.5;
border-radius: 4px;
padding: 0 6px;
}
.css2DLabel:after {
width: 1px;
height: 16px;
}
}
@media only screen and (min-width: 1200px) {
.bottom-card {
transform: translateX(-50%) scale(0.33);
}
.bottom-card .time-wrapper .time-slider .van-slider .custom-button {
padding: 0;
}
}
.loading {
position: fixed;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
z-index: 9999;
background-color: #313538;
}
.loading:before {
content: "";
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
font-size: 2.66667vw;
width: 2.13333vw;
height: 2.13333vw;
color: #fff;
border-radius: 50%;
text-indent: -9999em;
animation: load-effect 1s linear infinite;
}
@keyframes load-effect {
0% {
box-shadow: 0 -3em 0 0.2em #fff, 2em -2em 0 0 #fff, 3em 0 0 -0.5em #fff,
2em 2em 0 -0.5em #fff, 0 3em 0 -0.5em #fff, -2em 2em 0 -0.5em #fff,
-3em 0 0 -0.5em #fff, -2em -2em 0 0 #fff;
}
12.5% {
box-shadow: 0 -3em 0 0 #fff, 2em -2em 0 0.2em #fff, 3em 0 0 0 #fff,
2em 2em 0 -0.5em #fff, 0 3em 0 -0.5em #fff, -2em 2em 0 -0.5em #fff,
-3em 0 0 -0.5em #fff, -2em -2em 0 -0.5em #fff;
}
25% {
box-shadow: 0 -3em 0 -0.5em #fff, 2em -2em 0 0 #fff, 3em 0 0 0.2em #fff,
2em 2em 0 0 #fff, 0 3em 0 -0.5em #fff, -2em 2em 0 -0.5em #fff,
-3em 0 0 -0.5em #fff, -2em -2em 0 -0.5em #fff;
}
37.5% {
box-shadow: 0 -3em 0 -0.5em #fff, 2em -2em 0 -0.5em #fff, 3em 0 0 0 #fff,
2em 2em 0 0.2em #fff, 0 3em 0 0 #fff, -2em 2em 0 -0.5em #fff,
-3em 0 0 -0.5em #fff, -2em -2em 0 -0.5em #fff;
}
50% {
box-shadow: 0 -3em 0 -0.5em #fff, 2em -2em 0 -0.5em #fff,
3em 0 0 -0.5em #fff, 2em 2em 0 0 #fff, 0 3em 0 0.2em #fff,
-2em 2em 0 0 #fff, -3em 0 0 -0.5em #fff, -2em -2em 0 -0.5em #fff;
}
62.5% {
box-shadow: 0 -3em 0 -0.5em #fff, 2em -2em 0 -0.5em #fff,
3em 0 0 -0.5em #fff, 2em 2em 0 -0.5em #fff, 0 3em 0 0 #fff,
-2em 2em 0 0.2em #fff, -3em 0 0 0 #fff, -2em -2em 0 -0.5em #fff;
}
75% {
box-shadow: 0 -3em 0 -0.5em #fff, 2em -2em 0 -0.5em #fff,
3em 0 0 -0.5em #fff, 2em 2em 0 -0.5em #fff, 0 3em 0 -0.5em #fff,
-2em 2em 0 0 #fff, -3em 0 0 0.2em #fff, -2em -2em 0 0 #fff;
}
87.5% {
box-shadow: 0 -3em 0 0 #fff, 2em -2em 0 -0.5em #fff, 3em 0 0 -0.5em #fff,
2em 2em 0 -0.5em #fff, 0 3em 0 -0.5em #fff, -2em 2em 0 0 #fff,
-3em 0 0 0 #fff, -2em -2em 0 0.2em #fff;
}
to {
box-shadow: 0 -3em 0 0.2em #fff, 2em -2em 0 0 #fff, 3em 0 0 -0.5em #fff,
2em 2em 0 -0.5em #fff, 0 3em 0 -0.5em #fff, -2em 2em 0 -0.5em #fff,
-3em 0 0 -0.5em #fff, -2em -2em 0 0 #fff;
}
}
@media only screen and (min-width: 500px) {
.loading:before {
font-size: 12px;
width: 12px;
height: 12px;
}
.bottom-card .time-wrapper .time-slider .van-slider .custom-button {
padding: 0;
}
::v-deep .el-date-editor.el-input,
.el-date-editor.el-input__inner {
transform: scale(2.5);
}
.bottom-card .top .title {
flex: none;
}
.bottom-card .top .date {
margin-right: 100px;
}
}
@media only screen and (min-width: 300px) and (max-width: 800px) {
#sunshine .page-wrappr {
position: absolute;
bottom: -20px;
border-radius: 20px 20px 0 0;
width: 100%;
}
}
.bottom-card .time-wrapper .play-btn svg {
width: 4.8vw;
height: 4.8vw;
}
.bottom-card .time-wrapper .time-slider .van-slider .custom-button {
padding-left: 0;
padding-right: 0;
}
</style>