filestash/public/assets/pages/viewerpage/application_3d/init.js
MickaelK 71b14e6eaf feature (3d): embed 3d viewer anywhere
This contains a bunch of things packaged in 1:

1) UI improvements for the 3D viewer to support all sort of file types
   and create a nice rendering in a clean way with all sort of options

2) enable people to use Filestash as an SDK so we can embed the 3d viewer
   elsewhere
2024-12-23 18:50:23 +11:00

137 lines
8.9 KiB
JavaScript

import { createElement, onDestroy } from "../../../lib/skeleton/index.js";
import { join } from "../../../lib/path.js";
import * as THREE from "../../../lib/vendor/three/three.module.js";
import { OrbitControls } from "../../../lib/vendor/three/OrbitControls.js";
import { toCreasedNormals } from "../../../lib/vendor/three/utils/BufferGeometryUtils.js";
import { GLTFLoader } from "../../../lib/vendor/three/GLTFLoader.js";
import { OBJLoader } from "../../../lib/vendor/three/OBJLoader.js";
import { STLLoader } from "../../../lib/vendor/three/STLLoader.js";
import { FBXLoader } from "../../../lib/vendor/three/FBXLoader.js";
import { Rhino3dmLoader } from "../../../lib/vendor/three/3DMLoader.js";
export default function({ $page, $menubar, mesh, refresh }) {
// setup the dom
const renderer = new THREE.WebGLRenderer({ antialias: true, shadowMapEnabled: true });
renderer.setSize($page.clientWidth, $page.clientHeight);
renderer.setPixelRatio(window.devicePixelRatio);
renderer.setClearColor(0xf5f5f5);
renderer.shadowMap.enabled = true;
renderer.shadowMap.type = THREE.PCFSoftShadowMap;
$page.appendChild(renderer.domElement);
// center everything
const box = new THREE.Box3().setFromObject(mesh);
const center = box.getCenter(new THREE.Vector3());
const size = box.getSize(new THREE.Vector3());
const maxDim = Math.max(size.x, size.y, size.z);
// setup the scene, camera and controls
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(
45,
$page.clientWidth / $page.clientHeight,
Math.max(0.1, maxDim / 100),
maxDim * 10,
);
const controls = new OrbitControls(camera, renderer.domElement);
scene.add(mesh);
mesh.castShadow = true;
mesh.receiveShadow = true;
camera.position.set(center.x, center.y, center.z + maxDim * 1.8);
controls.target.copy(center);
// enable animation if present
const mixer = new THREE.AnimationMixer(mesh);
if (mesh.animations.length > 0) {
const ICON = {
PLAY: "",
PAUSE: "",
};
const $button = createElement(`<img class="component_icon" draggable="false" src="${ICON.PLAY}" alt="play">`);
const action = mixer.clipAction(mesh.animations[0]);
let isPlaying = false;
$button.onclick = () => {
if (isPlaying === false) action.play();
else action.stop();
isPlaying = !isPlaying;
$button.setAttribute("src", isPlaying ? ICON.PAUSE : ICON.PLAY);
};
$menubar.add($button);
}
// sizing of the window
const onResize = () => {
camera.aspect = $page.clientWidth / $page.clientHeight;
camera.updateProjectionMatrix();
renderer.setSize($page.clientWidth, $page.clientHeight);
};
window.addEventListener("resize", onResize);
onDestroy(() => window.removeEventListener("resize", onResize));
// stuff we refresh constantly
const clock = new THREE.Clock();
refresh.push(() => {
controls.update();
renderer.render(scene, camera);
mixer.update(clock.getDelta());
});
return { renderer, scene, camera, controls, box };
}
export function getLoader(mime) {
const identity = (s) => s;
switch (mime) {
case "application/object":
return [
new OBJLoader(),
(obj) => {
obj.traverse((child) => {
if (child.isMesh) {
child.material = new THREE.MeshPhongMaterial({
color: 0x40464b,
emissive: 0x40464b,
specular: 0xf9f9fa,
shininess: 10,
transparent: true,
});
// smooth the edges: https://discourse.threejs.org/t/how-to-smooth-an-obj-with-threejs/3950/16
child.geometry = toCreasedNormals(child.geometry, (30 / 180) * Math.PI);
}
});
return obj;
},
];
case "model/3dm":
THREE.Object3D.DEFAULT_UP.set(0, 0, 1);
const loader = new Rhino3dmLoader();
loader.setLibraryPath(join(import.meta.url, "../../../lib/vendor/three/rhino3dm/"));
return [loader, identity];
case "model/gtlt-binary":
case "model/gltf+json":
return [new GLTFLoader(), (gltf) => gltf.scene];
case "model/stl":
return [new STLLoader(), (geometry) => {
const material = new THREE.MeshPhongMaterial({
emissive: 0x40464b,
specular: 0xf9f9fa,
shininess: 15,
transparent: true,
});
if (geometry.hasColors) material.vertexColors = true;
else material.color = material.emissive;
return new THREE.Mesh(geometry, material);
}];
case "application/fbx":
return [new FBXLoader(), (obj) => {
obj.traverse((child) => {
// console.log(child);
});
return obj;
}];
default:
return [null, null];
}
}