最近有朋友需要渲染模型,就帮忙写了个简单的案例,下面分享下代码

image.png
代码块分享
<template>
<div class="threejs-container">
<div class="title">
<p>{{ findRes.title }}</p>
</div>
<div class="subTitle"><img
src=""
alt="">
<p class="subTitle-val">{{ findRes.subTitle }}</p> <img
src=""
alt="">
</div>
<div ref="canvasContainer"></div>
<div v-if="isLoading" class="loading-overlay">
<div class="loading-bar"
:style="{ background: `conic-gradient(#4caf50 ${loadingProgress}%, #eee ${loadingProgress}%)` }">
<span class="loading-text">{{ loadingProgress.toFixed(0) }}%</span>
</div>
</div>
</div>
</template>
<script>
import * as THREE from 'three';
import { FBXLoader } from 'three/examples/jsm/loaders/FBXLoader.js';
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls.js';
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader.js';
export default {
name: 'MyView',
data() {
return {
findRes: {},
isLoading: true,
loadingProgress: 0
}
},
methods: {
initSence() {
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
const renderer = new THREE.WebGLRenderer({ alpha: true, antialias: true });
renderer.setSize(window.innerWidth, window.innerHeight);
this.$refs.canvasContainer.appendChild(renderer.domElement);
// 添加光源
const lightPower = this.findRes.lightPower || 1
const ambientLight = new THREE.AmbientLight(0xffffff, lightPower);
scene.add(ambientLight);
const directionalLight = new THREE.DirectionalLight(0xffffff, lightPower);
directionalLight.position.set(1, 1, 1);
scene.add(directionalLight);
const pointLight = new THREE.PointLight(0xffffff, 2, 100);
pointLight.position.set(5, 5, 5);
scene.add(pointLight);
const loadingManager = new THREE.LoadingManager();
loadingManager.onStart = () => {
this.isLoading = true;
};
loadingManager.onLoad = () => {
//this.loadingProgress = 100;
setTimeout(() => {
this.isLoading = false;
}, 500)
};
loadingManager.onProgress = (url, itemsLoaded, itemsTotal) => {
this.loadingProgress = Math.floor((itemsLoaded / itemsTotal) * 10) + 90; // 更新进度
};
const loader = this.findRes.modelPath.endsWith('.gltf') || this.findRes.modelPath.endsWith('.glb') ?
new GLTFLoader(loadingManager) : new FBXLoader(loadingManager); // 根据文件类型选择加载器
let object; // 声明一个变量来存储加载的模型
let isMouseControlled = false; // 添加标志来跟踪鼠标控制状态
loader.load(this.findRes.modelPath, (loadedObject) => {
object = loadedObject.scene || loadedObject; // 将加载的模型赋值给变量
scene.add(object);
}, (xhr) => {
// 使用 xhr 对象获取更精确的加载进度
const percentComplete = Math.round((xhr.loaded / xhr.total) * 100);
this.loadingProgress = percentComplete>90 ? 90 : percentComplete;
//console.log('xhr',percentComplete);
}, (error) => {
console.error('加载模型时出错:', error);
});
const cameraRng = this.findRes.cameraRng ? this.findRes.cameraRng : [0, 20, 22]
//console.log('cameraRng', cameraRng);
camera.position.set(...cameraRng);
const controls = new OrbitControls(camera, renderer.domElement);
// 监听鼠标控制事件
controls.addEventListener('start', () => {
isMouseControlled = true; // 开始控制时设置标志
});
const animate = function () {
requestAnimationFrame(animate);
if (object && !isMouseControlled) {
object.rotation.y += 0.005; // 使模型自动旋转
}
renderer.render(scene, camera);
};
animate();
}
},
mounted() {
const routeParams = this.$route.query;
//const findRes = window.config.lists.find(item => item.id == routeParams.id);
//console.log('obj:', findRes);
const findRes = {
id: 1,
title: '解放战争时期粟裕大将使用的臂搁',
subTitle: '新四军江南指挥部纪念馆',
modelPath: "/model/1/model.fbx",
cameraRng: [0, 3, 30], //相机配置
lightPower:2 //光源
}
this.findRes = findRes || window.config.lists[0];
this.initSence();
}
};
</script>
<style scoped>
.loading-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: rgba(0, 0, 0, 0.5);
display: flex;
align-items: center;
justify-content: center;
z-index: 9999;
/* 确保在最上层 */
}
.loading-bar {
width: 100px;
/* 圆形的直径 */
height: 100px;
/* 圆形的直径 */
border-radius: 50%;
/* 使其成为圆形 */
position: relative;
/* 使文本能够绝对定位 */
transition: background 0.5s ease;
/* 添加过渡效果 */
background: conic-gradient(gray 0%, transparent 0%);
/* 默认颜色为灰色 */
}
.loading-text {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
/* 居中对齐 */
color: #000;
/* 文本颜色为黑色 */
font-size: 16px;
/* 文本大小 */
font-weight: bold;
/* 文本加粗 */
background: #fff;
/* 白色背景 */
border-radius: 50%;
/* 使其成为圆形 */
width: 80px;
/* 圆形的直径 */
height: 80px;
/* 圆形的直径 */
display: flex;
align-items: center;
justify-content: center;
/* 居中对齐文本 */
}
.threejs-container {
width: 100vw;
height: 100vh;
position: relative;
overflow: hidden;
background: url('/pc_bg.png') 50% / cover no-repeat;
z-index: 1;
}
.threejs-container .title {
position: absolute;
top: 20px;
left: 50%;
z-index: 10;
-webkit-transform: translateX(-50%);
transform: translateX(-50%);
font-size: 26px;
font-weight: 700;
color: #ffeac0;
text-align: center;
display: -webkit-box;
display: -ms-flexbox;
display: flex
}
.threejs-container .subTitle {
position: absolute;
top: 80px;
left: 50%;
z-index: 10;
-webkit-transform: translateX(-50%);
transform: translateX(-50%);
font-size: 22px;
color: #ffeac0;
opacity: .5;
text-align: center;
display: -webkit-box;
display: -ms-flexbox;
display: flex;
}
.subTitle-val {
margin: 0px 10px;
}
canvas {
display: block;
position: absolute;
top: 0;
left: 0;
z-index: 2;
}
</style>