filestash/public/assets/pages/viewerpage/application_3d/toolbar.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

97 lines
5.2 KiB
JavaScript

import { createElement } from "../../../lib/skeleton/index.js";
import { qs } from "../../../lib/dom.js";
import * as THREE from "../../../lib/vendor/three/three.module.js";
export default function(render, { camera, controls, mesh, $menubar, $toolbar }) {
if (mesh.children.length <= 1) return;
$menubar.add(buttonLayers({ $toolbar }));
render(createChild(
document.createDocumentFragment(),
mesh,
0,
{ camera, controls }
));
}
function buttonLayers({ $toolbar }) {
const $button = createElement(`<img class="component_icon" draggable="false" src="" alt="layers">`);
$button.onclick = () => $toolbar.classList.toggle("open");
return $button;
}
function createChild($fragment, mesh, child = 0, opts) {
if (["Bone"].indexOf(mesh.type) >= 0) return;
buildDOM($fragment, mesh, child, opts);
if (mesh.children.length > 0 && child < 4) {
for (let i=0; i<mesh.children.length; i++) {
createChild($fragment, mesh.children[i], child + 1, opts);
}
}
return $fragment;
}
function buildDOM($fragment, child, left, { camera, controls }) {
const $label = createElement(`
<label class="no-select" style="padding-left: ${left*20}px">
<div class="component_checkbox">
<input type="checkbox" ${child.visible ? "checked" : ""} />
<span class="indicator"></span>
</div>
<span class="text">${name(child)}</span>
</label>
`);
qs($label, "input").onchange = () => child.visible = !child.visible;
let block = false; let blockID = null;
$label.onclick = async(e) => {
if (e.target.nodeName === "INPUT" || e.target.classList.contains("component_checkbox")) return;
e.preventDefault(); e.stopPropagation();
block = true;
clearTimeout(blockID);
blockID = setTimeout(() => block = false, 2000);
$label.onmouseenter();
await flyTo({ mesh: child, camera, controls });
};
$label.onmouseenter = () => block === false && getRootObject(child).traverse((c) => {
if (!c.material) return;
c.material.opacity = c.id === child.id || c.parent.id === child.id ? 1 : 0.2;
c.material.depthWrite = c.material.opacity === 1;
});
$label.onmouseleave = () => block === false && getRootObject(child).traverse((c) => {
if (!c.material) return;
c.material.depthWrite = true;
c.material.opacity = 1;
});
$fragment.appendChild($label);
}
async function flyTo({ mesh, camera, controls }) {
const box = new THREE.Box3().setFromObject(mesh);
const size = box.getSize(new THREE.Vector3());
const targetLookAt = box.getCenter(new THREE.Vector3());
const targetDistance = Math.max(size.x, size.y, size.z) * 1.1;
const targetPosition = targetLookAt.clone().add(new THREE.Vector3(targetDistance, targetDistance, targetDistance));
const [startPosition, startLookAt] = [camera.position.clone(), controls.target.clone()];
const startTime = performance.now();
return new Promise((resolve) => (function animate() {
const t = Math.min((performance.now() - startTime) / 500, 1);
camera.position.lerpVectors(startPosition, targetPosition, t);
controls.target.lerpVectors(startLookAt, targetLookAt, t);
controls.update();
t < 1 ? requestAnimationFrame(animate) : resolve();
})());
}
function getRootObject(mesh) {
if (mesh.type === "Scene" || mesh.parent.type === "Scene") return mesh;
return getRootObject(mesh.parent);
}
function name(mesh) {
if (mesh.name) return mesh.name;
else if (mesh.isGroup && mesh.uuid) return `group: ${mesh.uuid}`;
else if (mesh.uuid) return mesh.uuid;
return "N/A";
}