【threejs】3D模型加载

3D 模型加载

在实际开发中, 不可能使用 Three.js 提供的几何体、材质等方式来编写 3D 模型, 通常是使用建模软件建模, 导出 3D 模型, 使用 Three.js 完成交互功能。

添加了模型之后, 必须添加灯光才能看到模型, 否则模型会被黑色覆盖, 因为三维场景和现实世界一样, 没有光照亮, 能看到的只有黑色。

Three.js 提供了多种 3D 模型加载器,常用的有:

  • GLTFLoader - 加载 .gltf/.glb 模型(官方推荐的格式,支持性最好)。
  • DRACOLoader - 用于压缩 .gltf/.glb 模型
  • FBXLoader - 加载 .fbx 模型。
  • ColladaLoader - 加载 .dae 模型。
  • OBJLoader - 加载 .obj 模型,与 MTLLoader 配合使用。
  • MTLLoader - 加载 .mtl 材质,与 OBJLoader 配合使用。
  • STLLoader - 加载 .stl 模型。

文档:https://threejs.rocyuan.top/docs/#examples/zh/loaders/GLTFLoader

GLTFLoader 使用

加载器的使用基本大同小异,以 GLTFLoader 为例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
import { GLTFLoader } from "three/examples/jsm/loaders/GLTFLoader.js";
import { DRACOLoader } from "three/examples/jsm/loaders/DRACOLoader.js";

// 加载 GLFT 三维模型
const gltfLoader = new GLTFLoader();

// 指定 DRACO 解码器(可选)
const dracoLoader = new DRACOLoader();
dracoLoader.setDecoderPath("/examples/jsm/libs/draco/");
gltfLoader.setDRACOLoader(dracoLoader);

gltfLoader.load(
"/model/wooden_table_02_4k/wooden_table_02_4k.gltf",
(glft) => {
// 主要的模型操作在此处编写...
console.log(glft);
const model = glft.scene;
// 给加载的三维模型中的网格模型开启阴影
model.traverse((child) => {
if (child.isMesh) {
child.castShadow = true;
child.receiveShadow = false;
}
});
scene.add(model);
},
(e) => {
console.log("加载进度:", `${(e.loaded / e.total) * 100}%`);
},
(err) => {
console.log("加载错误:", err);
}
);

完整示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
<template>
<div ref="threeContainer" class="three-container"></div>
</template>
<script setup>
import { onMounted, ref } from "vue";
import * as THREE from "three";
import { OrbitControls } from "three/examples/jsm/controls/OrbitControls.js";
import { GLTFLoader } from "three/examples/jsm/loaders/GLTFLoader.js";

const threeContainer = ref(null);

function init3D() {
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(75, threeContainer.value.clientWidth / threeContainer.value.clientHeight, 0.1, 1000);
camera.position.set(-1, 1.5, 1.5);
const renderer = new THREE.WebGLRenderer({ antialias: true });
renderer.setSize(threeContainer.value.clientWidth, threeContainer.value.clientHeight);
renderer.setPixelRatio(window.devicePixelRatio);
threeContainer.value.appendChild(renderer.domElement);
// 1. 渲染器中启用阴影
renderer.shadowMap.enabled = true;
renderer.shadowMap.type = THREE.PCFSoftShadowMap;

const ambientLight = new THREE.AmbientLight(0x404040);
scene.add(ambientLight);
const directionalLight = new THREE.DirectionalLight(0xffffff, 0.5);
directionalLight.position.y = 200;
directionalLight.position.x = 200;
directionalLight.position.z = 100;
// 2. 开启平行光投射阴影
directionalLight.castShadow = true;
scene.add(directionalLight);

// 创建球体
const sphereGeometry = new THREE.SphereGeometry(0.5, 32, 32);
const sphereMaterial = new THREE.MeshStandardMaterial({
color: 0x00ff00,
});
const sphere = new THREE.Mesh(sphereGeometry, sphereMaterial);
sphere.position.set(0, 1, 0);
// 3. 开启物体投射阴影(castShadow) 与 接收阴影(receiveShadow)
sphere.castShadow = true;
sphere.receiveShadow = true;
// scene.add(sphere)

// 创建平板
const planeGeometry = new THREE.PlaneGeometry(5, 5);
const planeMaterial = new THREE.MeshStandardMaterial({
color: 0xffffff,
side: THREE.DoubleSide,
});
const plane = new THREE.Mesh(planeGeometry, planeMaterial);
plane.rotation.x = Math.PI / 2;
// 4. 开启平面(模拟地面)接收阴影
plane.receiveShadow = true;
scene.add(plane);

// 加载 GLFT 三维模型
const gltfLoader = new GLTFLoader();
gltfLoader.load(
"/model/wooden_table_02_4k.gltf/wooden_table_02_4k.gltf",
(glft) => {
// 主要的模型操作在此处编写...
console.log(glft);
const model = glft.scene;
// 给加载的三维模型中的网格模型开启阴影
model.traverse((child) => {
if (child.isMesh) {
child.castShadow = true;
child.receiveShadow = false;
}
});
scene.add(model);
},
(e) => {
console.log("加载进度:", `${(e.loaded / e.total) * 100}%`);
},
(err) => {
console.log("加载错误:", err);
}
);

const controls = new OrbitControls(camera, renderer.domElement);
controls.enableDamping = true;
controls.dampingFactor = 0.05;

function animate() {
requestAnimationFrame(animate);
controls.update();
renderer.render(scene, camera);
}
animate();
}

onMounted(() => {
init3D();
});
</script>
<style scoped>
.three-container {
width: 100vw;
height: calc(100vh - 20px);
}
</style>

效果

加载3d模型效果

打印模型加载完成的数据可见, 模型中包含了网格模型、材质、动画、位置、旋转、缩放等数据, 均可在加载完成后操作。