时间过的好快啊~再一次感叹,忙忙碌碌一年又过去了,新年第一帖,新的一年也要加油呀!
简单介绍下Three.js吧,Three.js是基于原生WebGL封装运行的三维引擎,在所有WebGL引擎中,Three.js是国内文资料最多、使用最广泛的三维引擎。因为使用简单,入门比较容易。
Three.js的具体介绍和使用有很多教程,可以根据自己需要搜索。它重要的三个属性:场景(scene),相机(camera),渲染器(renderer),组合构成一个三维画面。
创建一个div容器,用于加载模型
初始化数据
loadModel: true, // 模型加载状态
scene: null, // 3D模型场景
camera: null, // 摄像机
renderer: null, // 渲染器
mixer: null, // 动画帧
clock: null, // 锁
glbLoader: null,
dracoLoader: null,
渲染模型
// 渲染模型
init() {
// eslint-disable-next-line @typescript-eslint/no-this-alias
const self = this;
self.scene = new THREE.Scene();
self.camera = new THREE.PerspectiveCamera(45, self.$refs.threeBox.clientWidth / self.$refs.threeBox.clientHeight, 0.1, 10000);
self.renderer = new THREE.WebGLRenderer();
self.renderer.setClearColor('0xffffff', 0);
self.renderer.setSize(self.$refs.threeBox.clientWidth * 1.07, self.$refs.threeBox.clientHeight);
self.$refs.threeBox.appendChild(this.renderer.domElement);
// self.scene.add(new THREE.AxesHelper(1000));
self.controls = new OrbitControls(self.camera, self.renderer.domElement);
self.controls.target.set(0, 0, 0);
self.controls.update();
const boxHelper = new THREE.BoxHelper();
// 加载glb模型
const fbxLoader = new FBXLoader();
const actives = [];
self.glbLoader = new GLTFLoader();
self.dracoLoader = new DRACOLoader();
self.dracoLoader.setDecoderPath('draco/');
self.dracoLoader.setDecoderConfig({ type: 'js' });
self.dracoLoader.preload();
self.glbLoader.setDRACOLoader(self.dracoLoader);
self.glbLoader.load(self.modelPath, (obj) => {
const object = obj.scene;
boxHelper.setFromObject(object);
const { center } = boxHelper.geometry.boundingSphere;
const { radius } = boxHelper.geometry.boundingSphere;
const cameraPos = new THREE.Vector3(center.x + object.position.x, center.y + object.position.y, radius * 2.2 + center.z + object.position.z);
const lookPos = new THREE.Vector3(center.x + object.position.x, center.y + object.position.y, center.z - object.position.z);
object.rotation.y = (-90 * Math.PI) / 180;
object.rotation.z = (-45 * Math.PI) / 180;
self.camera.position.copy(cameraPos);
self.controls.target = lookPos;
self.camera.lookAt(lookPos);
self.scene.add(object);
self.loadModel = false;
// 加载动画帧
self.mixer = new THREE.AnimationMixer(object);
obj.animations.forEach((item) => {
actives.push(self.mixer.clipAction(item));
});
actives[0].play();
});
// 环境光
const ambient = new THREE.AmbientLight(0xffffff);
this.scene.add(ambient);
// 点光源
const point1 = new THREE.PointLight(0xD3D3D3);
point1.position.set(200, 400, 300);
self.scene.add(point1);
const point2 = new THREE.PointLight(0xD3D3D3);
point2.position.set(-200, -400, -300);
self.scene.add(point2);
self.camera.position.set(100, 200, 300);
self.camera.lookAt(0, 0, 0);
},
其中modelPath是放在public下面的模型路径,这里以glb模型为例,可以改为自己对应的glb模型地址。该模型中里面包含了动画,在加载完模型的时候,加载动画帧(mixer)
modelPath: '/model/model-processed.glb', // glb模型路径
加载动画
// 加载动画
animate() {
if (this.mixer) {
this.mixer.update(this.clock.getDelta());
}
requestAnimationFrame(this.animate);
this.renderer.clear();
this.renderer.render(this.scene, this.camera);
this.renderer.clearDepth();
},
初始化调用
mounted() {
this.$nextTick(() => {
this.clock = new THREE.Clock();
this.init();
this.animate();
});
},
最终效果图