brep-io-kernel 1.0.196 → 1.0.197
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/apiExamples/BREP_Booleans.html +2 -2
- package/dist/apiExamples/BREP_Export.html +2 -2
- package/dist/apiExamples/BREP_Primitives.html +2 -2
- package/dist/apiExamples/BREP_Transforms.html +2 -2
- package/dist/apiExamples/Embeded_2D_Sketcher.html +2 -2
- package/dist/apiExamples/Embeded_CAD.html +2 -2
- package/dist/apiExamples/Embeded_CAD_Integration_Test.html +2 -2
- package/dist/assets/{SimulationWorkbenchManager-CZU2PaTA.js → SimulationWorkbenchManager-BlXiJ2tl.js} +121 -37
- package/dist/assets/{apiExample_BREP_Booleans-D4_za_tB.js → apiExample_BREP_Booleans-C2hoasf2.js} +1 -1
- package/dist/assets/{apiExample_BREP_Export-u2fTdaNT.js → apiExample_BREP_Export-B9OOBBCX.js} +1 -1
- package/dist/assets/{apiExample_BREP_Primitives-DAoWeXoq.js → apiExample_BREP_Primitives-Do8y8b_w.js} +1 -1
- package/dist/assets/{apiExample_BREP_Transforms-DFpbDf74.js → apiExample_BREP_Transforms-Dg6kSpF5.js} +1 -1
- package/dist/assets/{apiExample_Embeded_2D_Sketcher-BTLB-XJ1.js → apiExample_Embeded_2D_Sketcher-mfrI-BcL.js} +1 -1
- package/dist/assets/{apiExample_Embeded_CAD-BN7Rap4P.js → apiExample_Embeded_CAD-GLEJ5EGM.js} +1 -1
- package/dist/assets/{apiExample_Embeded_CAD_Integration_Test-Dl4HRxKv.js → apiExample_Embeded_CAD_Integration_Test-CsWfKgCS.js} +1 -1
- package/dist/assets/{brep-kernel-CTSjNzN4.js → brep-kernel-D6hhTKa0.js} +1 -1
- package/dist/assets/{browserTests-BeGjo-jd.js → browserTests-gCJmMaIE.js} +1 -1
- package/dist/assets/{index.es-DiqUPvrb.js → index.es-CN_DUH2L.js} +1 -1
- package/dist/assets/{javascript-Bao00j9Y.js → javascript-CamWQzyi.js} +1 -1
- package/dist/assets/{main-cad-A9axj9C3.js → main-cad-BgZQtzIN.js} +5 -5
- package/dist/assets/{test-BiekV37i.js → test-q3OlLTks.js} +3 -3
- package/dist/cad.html +1 -1
- package/dist/test.html +1 -1
- package/dist/viewer.html +1 -1
- package/dist-kernel/brep-kernel.js +1 -1
- package/package.json +1 -1
- package/src/simulation/SimulationWorkbenchManager.js +121 -37
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["assets/brep-kernel-
|
|
1
|
+
const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["assets/brep-kernel-D6hhTKa0.js","assets/preload-helper-ZNr0Qq7Q.js"])))=>i.map(i=>d[i]);
|
|
2
2
|
var H=Object.defineProperty;var c=(e,t)=>H(e,"name",{value:t,configurable:!0});import"./modulepreload-polyfill-BdX5DvLD.js";import{_ as B}from"./preload-helper-ZNr0Qq7Q.js";const $=document.getElementById("status"),J=document.getElementById("log"),w=document.getElementById("btn-create"),V=document.getElementById("btn-destroy"),T=document.getElementById("btn-apply-css"),F=document.getElementById("btn-apply-theme"),L=document.getElementById("btn-export-svg"),G=document.getElementById("geometry-color"),_=document.getElementById("point-color"),N=document.getElementById("constraint-color"),A=document.getElementById("background-color"),P=document.getElementById("point-size-px"),O=document.getElementById("curve-thickness-px"),v=document.getElementById("sidebar-expanded"),g=document.getElementById("grid-visible"),R=document.getElementById("grid-spacing"),D=document.getElementById("css-input"),K=document.getElementById("sketch-status"),U=document.getElementById("sketch-host"),p=document.getElementById("event-output"),y=document.getElementById("svg-preview"),C=document.getElementById("path-output");let n=null,S=0,E=0,k=0;const W=5,a=c((...e)=>{const t=e.map(s=>{if(typeof s=="string")return s;try{return JSON.stringify(s)}catch{return String(s)}}).join(" ");J.textContent+=`${t}
|
|
3
3
|
`,console.log(...e)},"log"),u=c((e,t)=>{$.textContent=e,$.className=`status ${t}`},"setStatus"),o=c(e=>{K.textContent=e},"setSketchStatus"),x=c(e=>{w.disabled=e,V.disabled=!e,T.disabled=!e,F.disabled=!e,L.disabled=!e},"setSketchButtons"),h=c((e,t=null)=>{const s=new Date().toLocaleTimeString(),i=t==null?`[${s}] ${e}`:`[${s}] ${e} ${JSON.stringify(t)}`,l=p.textContent==="(No sketch events yet)"?[]:p.textContent.split(`
|
|
4
4
|
`).filter(Boolean),d=[i,...l].slice(0,W);p.textContent=d.length?d.join(`
|
|
5
|
-
`):"(No sketch events yet)"},"pushEvent"),M=c(()=>({geometryColor:G.value,pointColor:_.value,constraintColor:N.value,backgroundColor:A.value,pointSizePx:Math.max(1,Number(P.value)||10),curveThicknessPx:Math.max(.5,Number(O.value)||2)}),"currentTheme"),b=c(()=>{const e=Number(R.value);return Number.isFinite(e)&&e>0?e:1},"currentGridSpacing"),X=c(async()=>{if(n)return;const{Sketcher2DEmbed:e}=await B(async()=>{const{Sketcher2DEmbed:t}=await import("./brep-kernel-
|
|
6
|
-
`):"(No sketch geometry to export)",o(`Exported ${e.paths.length} SVG paths.`),y.scrollIntoView({behavior:"smooth",block:"start"})},"exportSvg");w.addEventListener("click",()=>{X().catch(e=>{console.error(e),o(`Failed to create sketcher: ${e?.message||String(e)}`)})});V.addEventListener("click",()=>{Y().catch(e=>{console.error(e),o(`Failed to destroy sketcher: ${e?.message||String(e)}`)})});T.addEventListener("click",()=>{q().catch(e=>{console.error(e),o(`Failed to apply CSS: ${e?.message||String(e)}`)})});L.addEventListener("click",()=>{j().catch(e=>{console.error(e),o(`Failed to export SVG: ${e?.message||String(e)}`)})});F.addEventListener("click",()=>{r().catch(e=>{console.error(e),o(`Failed to apply theme: ${e?.message||String(e)}`)})});[G,_,N,A,P,O].forEach(e=>{e.addEventListener("input",()=>{n&&r().catch(t=>{console.error(t),o(`Failed to apply theme: ${t?.message||String(t)}`)})})});v.addEventListener("change",()=>{n&&r().catch(e=>{console.error(e),o(`Failed to apply sidebar state: ${e?.message||String(e)}`)})});g.addEventListener("change",()=>{n&&r().catch(e=>{console.error(e),o(`Failed to apply grid visibility: ${e?.message||String(e)}`)})});R.addEventListener("input",()=>{n&&r().catch(e=>{console.error(e),o(`Failed to apply grid spacing: ${e?.message||String(e)}`)})});x(!1);C.textContent="(No SVG exported yet)";(async()=>{u("Importing bundle...","pending");const e=await B(()=>import("./brep-kernel-
|
|
5
|
+
`):"(No sketch events yet)"},"pushEvent"),M=c(()=>({geometryColor:G.value,pointColor:_.value,constraintColor:N.value,backgroundColor:A.value,pointSizePx:Math.max(1,Number(P.value)||10),curveThicknessPx:Math.max(.5,Number(O.value)||2)}),"currentTheme"),b=c(()=>{const e=Number(R.value);return Number.isFinite(e)&&e>0?e:1},"currentGridSpacing"),X=c(async()=>{if(n)return;const{Sketcher2DEmbed:e}=await B(async()=>{const{Sketcher2DEmbed:t}=await import("./brep-kernel-D6hhTKa0.js");return{Sketcher2DEmbed:t}},__vite__mapDeps([0,1]));n=new e({cssText:D.value,...M(),sidebarExpanded:v.checked,gridVisible:g.checked,gridSpacing:b(),onChange:c(t=>{S+=1;const s=Array.isArray(t?.geometries)?t.geometries.length:0;o(`Sketch updated (${S}). Geometries: ${s}`),h("onChange",{geometries:s})},"onChange"),onFinished:c(t=>{E+=1;const s=Array.isArray(t?.geometries)?t.geometries.length:0;o(`Sketch finished (${E}). Geometries: ${s}`),h("onFinished",{geometries:s}),j().catch(i=>{console.error(i),o(`Failed to export SVG after Finish: ${i?.message||String(i)}`)})},"onFinished"),onCancelled:c(()=>{k+=1,o(`Sketch cancelled (${k}).`),h("onCancelled")},"onCancelled")}),await n.mount(U),await n.getSketch(),x(!0),o("Sketcher iframe mounted. Draw geometry and click Export SVG Paths.")},"attachSketcher"),Y=c(async()=>{n&&(await n.destroy(),n=null,S=0,E=0,k=0,x(!1),o("Sketcher destroyed."),p.textContent="(No sketch events yet)",y.innerHTML="",C.textContent="")},"detachSketcher"),q=c(async()=>{n&&(await n.setCss(D.value),o("Custom CSS applied to iframe."))},"applySketchCss"),r=c(async()=>{n&&(await n.setTheme(M()),await n.setSidebarExpanded(v.checked),typeof n.setGrid=="function"?await n.setGrid({visible:g.checked,spacing:b()}):(typeof n.setGridVisible=="function"&&await n.setGridVisible(g.checked),typeof n.setGridSpacing=="function"&&await n.setGridSpacing(b())),o("Theme + sidebar + grid state applied to iframe."))},"applySketchTheme"),j=c(async()=>{if(!n)return;const e=await n.exportSVG({flipY:!0,precision:3,stroke:"#111111",strokeWidth:1.5,fill:"none",padding:12});await n.getSketch({preferCached:!0}),y.innerHTML=e.svg,C.textContent=e.paths.length?e.paths.map(t=>`id=${t.id} type=${t.type} d="${t.d}"`).join(`
|
|
6
|
+
`):"(No sketch geometry to export)",o(`Exported ${e.paths.length} SVG paths.`),y.scrollIntoView({behavior:"smooth",block:"start"})},"exportSvg");w.addEventListener("click",()=>{X().catch(e=>{console.error(e),o(`Failed to create sketcher: ${e?.message||String(e)}`)})});V.addEventListener("click",()=>{Y().catch(e=>{console.error(e),o(`Failed to destroy sketcher: ${e?.message||String(e)}`)})});T.addEventListener("click",()=>{q().catch(e=>{console.error(e),o(`Failed to apply CSS: ${e?.message||String(e)}`)})});L.addEventListener("click",()=>{j().catch(e=>{console.error(e),o(`Failed to export SVG: ${e?.message||String(e)}`)})});F.addEventListener("click",()=>{r().catch(e=>{console.error(e),o(`Failed to apply theme: ${e?.message||String(e)}`)})});[G,_,N,A,P,O].forEach(e=>{e.addEventListener("input",()=>{n&&r().catch(t=>{console.error(t),o(`Failed to apply theme: ${t?.message||String(t)}`)})})});v.addEventListener("change",()=>{n&&r().catch(e=>{console.error(e),o(`Failed to apply sidebar state: ${e?.message||String(e)}`)})});g.addEventListener("change",()=>{n&&r().catch(e=>{console.error(e),o(`Failed to apply grid visibility: ${e?.message||String(e)}`)})});R.addEventListener("input",()=>{n&&r().catch(e=>{console.error(e),o(`Failed to apply grid spacing: ${e?.message||String(e)}`)})});x(!1);C.textContent="(No SVG exported yet)";(async()=>{u("Importing bundle...","pending");const e=await B(()=>import("./brep-kernel-D6hhTKa0.js"),__vite__mapDeps([0,1]));a("Bundle exports:",Object.keys(e));const{BREP:t}=e;if(!t)throw new Error("BREP export missing from bundle");u("Running kernel checks...","pending");const s=new t.Cube({x:2,y:3,z:4,name:"Cube"}),i=s.volume();a("Cube volume:",i);const l=new t.Sphere({r:1,resolution:24,name:"Sphere"}),d=l.getTriangleCount();a("Sphere triangles:",d);const f=s.union(l),z=f.volume();a("Union volume:",z),a("Union triangles:",f.getTriangleCount());const m=24,I=Math.abs(i-m)<1e-6;if(a("Cube volume check:",I?"OK":"FAIL","(expected",m,")"),!I)throw new Error(`Cube volume mismatch: ${i} vs ${m}`);u("Success: bundle + WASM OK","ok")})().catch(e=>{console.error(e),a("ERROR:",e&&(e.stack||e.message||String(e))),u("Failed: see log","err")});
|
package/dist/cad.html
CHANGED
|
@@ -19,7 +19,7 @@
|
|
|
19
19
|
<script type="module" crossorigin src="/assets/ListEntityBase-DIaWtqmZ.js"></script>
|
|
20
20
|
<script type="module" crossorigin src="/assets/PartHistory-CW7D2BZZ.js"></script>
|
|
21
21
|
<script type="module" crossorigin src="/assets/AnnotationRegistry-DxNMMWXW.js"></script>
|
|
22
|
-
<script type="module" crossorigin src="/assets/main-cad-
|
|
22
|
+
<script type="module" crossorigin src="/assets/main-cad-BgZQtzIN.js"></script>
|
|
23
23
|
<link rel="stylesheet" crossorigin href="/assets/main-cad-C3E_anfF.css">
|
|
24
24
|
</head>
|
|
25
25
|
|
package/dist/test.html
CHANGED
|
@@ -167,7 +167,7 @@
|
|
|
167
167
|
font-size: 13px;
|
|
168
168
|
}
|
|
169
169
|
</style>
|
|
170
|
-
<script type="module" crossorigin src="/assets/test-
|
|
170
|
+
<script type="module" crossorigin src="/assets/test-q3OlLTks.js"></script>
|
|
171
171
|
<link rel="modulepreload" crossorigin href="/assets/modulepreload-polyfill-BdX5DvLD.js">
|
|
172
172
|
<link rel="modulepreload" crossorigin href="/assets/preload-helper-ZNr0Qq7Q.js">
|
|
173
173
|
</head>
|
package/dist/viewer.html
CHANGED
|
@@ -19,7 +19,7 @@
|
|
|
19
19
|
<script type="module" crossorigin src="/assets/ListEntityBase-DIaWtqmZ.js"></script>
|
|
20
20
|
<script type="module" crossorigin src="/assets/PartHistory-CW7D2BZZ.js"></script>
|
|
21
21
|
<script type="module" crossorigin src="/assets/AnnotationRegistry-DxNMMWXW.js"></script>
|
|
22
|
-
<script type="module" crossorigin src="/assets/main-cad-
|
|
22
|
+
<script type="module" crossorigin src="/assets/main-cad-BgZQtzIN.js"></script>
|
|
23
23
|
<link rel="stylesheet" crossorigin href="/assets/main-cad-C3E_anfF.css">
|
|
24
24
|
</head>
|
|
25
25
|
|
|
@@ -304369,7 +304369,7 @@ const G1e = class G1e {
|
|
|
304369
304369
|
async _ensureSimulationWorkbenchManager() {
|
|
304370
304370
|
return this.simulationWorkbenchManager ? this.simulationWorkbenchManager : this._simulationWorkbenchManagerPromise ? this._simulationWorkbenchManagerPromise : (this._simulationWorkbenchManagerPromise = (async () => {
|
|
304371
304371
|
try {
|
|
304372
|
-
const A = new URL("data:text/javascript;base64,import * as THREE from 'three';
import { CombinedTransformControls } from '../UI/controls/CombinedTransformControls.js';

const FIXED_METADATA_KEY = 'fixed';
const TRANSFORM_METADATA_KEY = 'simulationTransform';
const SIM_PROXY_PREFIX = '__SIM_PROXY__';
const VHACD_OPTIONS = Object.freeze({
  maxHulls: 64,
  voxelResolution: 1000000,
  maxVerticesPerHull: 64,
  shrinkWrap: true,
  fillMode: 'flood',
  findBestPlane: true,
  minEdgeLength: 1,
  messages: 'none',
});
const DEFAULT_TRANSFORM = Object.freeze({
  position: [0, 0, 0],
  rotationEuler: [0, 0, 0],
});

function cloneVec3(value, fallback = [0, 0, 0]) {
  const src = Array.isArray(value) ? value : fallback;
  return [
    Number.isFinite(Number(src[0])) ? Number(src[0]) : fallback[0],
    Number.isFinite(Number(src[1])) ? Number(src[1]) : fallback[1],
    Number.isFinite(Number(src[2])) ? Number(src[2]) : fallback[2],
  ];
}

function normalizeText(value, fallback = '') {
  const next = String(value == null ? '' : value).trim();
  return next || fallback;
}

function normalizeTransform(value) {
  const raw = value && typeof value === 'object' ? value : {};
  return {
    position: cloneVec3(raw.position, DEFAULT_TRANSFORM.position),
    rotationEuler: cloneVec3(raw.rotationEuler, DEFAULT_TRANSFORM.rotationEuler),
  };
}

function transformIsIdentity(transform) {
  const position = cloneVec3(transform?.position, DEFAULT_TRANSFORM.position);
  const rotationEuler = cloneVec3(transform?.rotationEuler, DEFAULT_TRANSFORM.rotationEuler);
  return position.every((value) => Math.abs(value) <= 1e-9)
    && rotationEuler.every((value) => Math.abs(value) <= 1e-9);
}

function copyLocalPose(object) {
  return {
    position: object.position.clone(),
    quaternion: object.quaternion.clone(),
    scale: object.scale.clone(),
  };
}

function buildLocalMatrix(pose) {
  return new THREE.Matrix4().compose(
    pose.position.clone(),
    pose.quaternion.clone(),
    pose.scale.clone(),
  );
}

function poseToOffset(basePose, currentObject) {
  const baseMatrix = buildLocalMatrix(basePose);
  const currentMatrix = new THREE.Matrix4().compose(
    currentObject.position.clone(),
    currentObject.quaternion.clone(),
    currentObject.scale.clone(),
  );
  const relative = new THREE.Matrix4().copy(baseMatrix).invert().multiply(currentMatrix);
  const position = new THREE.Vector3();
  const quaternion = new THREE.Quaternion();
  const scale = new THREE.Vector3();
  relative.decompose(position, quaternion, scale);
  const rotationEuler = new THREE.Euler().setFromQuaternion(quaternion, 'XYZ');
  return {
    position: [position.x, position.y, position.z],
    rotationEuler: [rotationEuler.x, rotationEuler.y, rotationEuler.z],
  };
}

function normalizeMeshArrays(mesh) {
  if (!mesh) return null;
  const stride = Math.max(3, Number(mesh.numProp) || 3);
  const sourcePositions = Array.isArray(mesh.vertProperties) ? mesh.vertProperties : Array.from(mesh.vertProperties || []);
  const sourceIndices = Array.isArray(mesh.triVerts) ? mesh.triVerts : Array.from(mesh.triVerts || []);
  if (sourcePositions.length < 9 || sourceIndices.length < 3) return null;
  const positions = new Float64Array((sourcePositions.length / stride) * 3);
  for (let src = 0, dst = 0; src + 2 < sourcePositions.length; src += stride, dst += 3) {
    positions[dst + 0] = Number(sourcePositions[src + 0]) || 0;
    positions[dst + 1] = Number(sourcePositions[src + 1]) || 0;
    positions[dst + 2] = Number(sourcePositions[src + 2]) || 0;
  }
  return {
    positions,
    indices: new Uint32Array(sourceIndices),
  };
}

function buildProxyMaterial(baseColor, index, total) {
  const color = new THREE.Color(baseColor || '#9ca3af');
  const hsl = { h: 0, s: 0, l: 0 };
  color.getHSL(hsl);
  const offset = total > 1 ? (index / Math.max(1, total)) * 0.18 : 0;
  color.setHSL((hsl.h + offset) % 1, Math.min(1, Math.max(0.35, hsl.s || 0.55)), Math.min(0.72, Math.max(0.42, hsl.l)));
  return new THREE.MeshStandardMaterial({
    color,
    transparent: true,
    opacity: 0.55,
    roughness: 0.55,
    metalness: 0.05,
    depthWrite: true,
  });
}

function disposeHierarchy(root) {
  if (!root?.traverse) return;
  root.traverse((node) => {
    if (node?.geometry?.dispose) {
      try { node.geometry.dispose(); } catch {}
    }
    if (Array.isArray(node?.material)) {
      for (const material of node.material) {
        try { material?.dispose?.(); } catch {}
      }
    } else {
      try { node?.material?.dispose?.(); } catch {}
    }
  });
}

export class SimulationWorkbenchManager {
  constructor(viewer) {
    this.viewer = viewer || null;
    this._active = false;
    this._baseLocalPose = new Map();
    this._runtimeState = new Map();
    this._sourceVisibility = new Map();
    this._decompositionCache = new Map();
    this._proxyGroups = new Map();
    this._transformSession = null;
    this._rapier = null;
    this._rapierLoadPromise = null;
    this._vhacd = null;
    this._vhacdLoadPromise = null;
    this._physicsWorld = null;
    this._bodyState = new Map();
    this._listeners = new Set();
    this._removeStateManagerListener = null;
    this._isPlaying = false;
    this._motionState = new Map();
    this._movedSolidIds = new Set();
    this._raf = 0;
    this._lastStepTime = 0;
    this._scratch = {
      parentInverse: new THREE.Matrix4(),
      parentWorldMatrix: new THREE.Matrix4(),
      localMatrix: new THREE.Matrix4(),
      worldMatrix: new THREE.Matrix4(),
      deltaMatrix: new THREE.Matrix4(),
      worldPosition: new THREE.Vector3(),
      localPosition: new THREE.Vector3(),
      worldQuaternion: new THREE.Quaternion(),
      parentQuaternion: new THREE.Quaternion(),
      parentQuaternionInverse: new THREE.Quaternion(),
      box: new THREE.Box3(),
      center: new THREE.Vector3(),
      axis: new THREE.Vector3(),
      translation: new THREE.Vector3(),
    };
    this._bindStateManager();
  }

  dispose() {
    if (typeof this._removeStateManagerListener === 'function') {
      try { this._removeStateManagerListener(); } catch {}
    }
    this._removeStateManagerListener = null;
    this.setActive(false);
  }

  isActive() {
    return this._active;
  }

  isSimulationWorkbenchActive() {
    return this.viewer?._getActiveWorkbenchId?.() === 'SIMULATION';
  }

  addListener(listener) {
    if (typeof listener !== 'function') return () => {};
    this._listeners.add(listener);
    return () => {
      try { this._listeners.delete(listener); } catch {}
    };
  }

  removeListener(listener) {
    if (typeof listener !== 'function') return;
    try { this._listeners.delete(listener); } catch {}
  }

  isPlaying() {
    return this._isPlaying;
  }

  setPlaying(playing) {
    const next = !!playing;
    if (next === this._isPlaying) return this._isPlaying;
    this._isPlaying = next;
    if (this._active) {
      if (next) {
        void this._prepareSimulationAssets().then(() => {
          this._ensurePhysicsLoop();
        });
      } else {
        this._stopPhysicsLoop();
      }
    }
    this._emit();
    try { this.viewer?.render?.(); } catch {}
    return this._isPlaying;
  }

  resetSimulationState() {
    this._stopTransformSession();
    this._stopPhysicsLoop();
    this._destroyPhysicsWorld();
    this._isPlaying = false;
    this._movedSolidIds.clear();
    this._motionState.clear();
    for (const solid of this._listSceneSolids()) {
      this._runtimeState.set(solid.uuid, normalizeTransform(null));
      this.setStoredTransform(solid, null);
    }
    this._restoreBaseLocalPoses();
    for (const solid of this._listSceneSolids()) {
      this._syncProxyGroupTransform(solid);
    }
    if (this._active) {
      this._resetMotionRuntime();
    }
    this._emit();
    try { this.viewer?.render?.(); } catch {}
  }

  getSolidFixed(solid) {
    if (!solid?.name) return false;
    const own = this.viewer?.partHistory?.metadataManager?.getOwnMetadata?.(solid.name) || {};
    return own[FIXED_METADATA_KEY] === true;
  }

  setSolidFixed(solid, fixed) {
    if (!solid?.name) return false;
    const manager = this.viewer?.partHistory?.metadataManager;
    if (!manager) return false;
    const data = manager.getOwnMetadata(solid.name);
    if (fixed) data[FIXED_METADATA_KEY] = true;
    else delete data[FIXED_METADATA_KEY];
    manager.setMetadataObject(solid.name, data);
    if (this._active && this._physicsWorld) {
      void this._rebuildPhysicsWorld();
    }
    return true;
  }

  getStoredTransform(solid) {
    if (!solid?.name) return normalizeTransform(null);
    const own = this.viewer?.partHistory?.metadataManager?.getOwnMetadata?.(solid.name) || {};
    return normalizeTransform(own[TRANSFORM_METADATA_KEY]);
  }

  setStoredTransform(solid, transform) {
    if (!solid?.name) return false;
    const manager = this.viewer?.partHistory?.metadataManager;
    if (!manager) return false;
    const normalized = normalizeTransform(transform);
    const data = manager.getOwnMetadata(solid.name);
    if (transformIsIdentity(normalized)) delete data[TRANSFORM_METADATA_KEY];
    else data[TRANSFORM_METADATA_KEY] = normalized;
    manager.setMetadataObject(solid.name, data);
    return true;
  }

  setActive(active) {
    const next = !!active;
    if (next === this._active) {
      if (next) {
        this._applyRuntimeTransforms();
      } else {
        this._restoreBaseLocalPoses();
      }
      return;
    }
    this._active = next;
    if (this._active) {
      this._isPlaying = false;
      this._stopPhysicsLoop();
      this._destroyPhysicsWorld();
      this._captureBaseLocalPoses();
      this._captureSourceVisibility();
      this._hydrateRuntimeStateFromMetadata();
      this._resetMotionRuntime();
      this._applyRuntimeTransforms();
      this._setSourceSolidsVisible(true);
      void this._prepareVisualAssets();
      this._emit();
      return;
    }
    this._stopTransformSession();
    this._destroyPhysicsWorld();
    this._stopPhysicsLoop();
    this._clearProxyGroups();
    this._setSourceSolidsVisible(true);
    this._restoreBaseLocalPoses();
    this._runtimeState.clear();
    this._baseLocalPose.clear();
    this._sourceVisibility.clear();
    this._decompositionCache.clear();
    this._motionState.clear();
    this._movedSolidIds.clear();
    this._isPlaying = false;
    this._emit();
  }

  toggleSolidTransform(solid) {
    if (!solid || solid.type !== 'SOLID') return;
    if (!this.isSimulationWorkbenchActive()) return;
    const session = this._transformSession;
    if (session && session.solid === solid) {
      const currentMode = (typeof session.controls?.getMode === 'function')
        ? session.controls.getMode()
        : (session.controls?.mode || session.mode || 'translate');
      if (currentMode === 'translate') {
        try { session.controls?.setMode('rotate'); } catch { session.controls.mode = 'rotate'; }
        session.mode = 'rotate';
        try { session.globalState?.updateForCamera?.(); } catch {}
        try { this.viewer?.render?.(); } catch {}
        return;
      }
      this._stopTransformSession();
      return;
    }
    this._startTransformSession(solid);
  }

  _captureBaseLocalPoses() {
    for (const solid of this._listSceneSolids()) {
      if (!this._baseLocalPose.has(solid.uuid)) {
        this._baseLocalPose.set(solid.uuid, copyLocalPose(solid));
      }
    }
  }

  _captureSourceVisibility() {
    for (const solid of this._listSceneSolids()) {
      if (!this._sourceVisibility.has(solid.uuid)) {
        this._sourceVisibility.set(solid.uuid, solid.visible !== false);
      }
    }
  }

  _hydrateRuntimeStateFromMetadata() {
    for (const solid of this._listSceneSolids()) {
      this._runtimeState.set(solid.uuid, normalizeTransform(this.getStoredTransform(solid)));
    }
  }

  _restoreBaseLocalPoses() {
    for (const solid of this._listSceneSolids()) {
      const base = this._baseLocalPose.get(solid.uuid);
      if (!base) continue;
      solid.position.copy(base.position);
      solid.quaternion.copy(base.quaternion);
      solid.scale.copy(base.scale);
      solid.updateMatrixWorld?.(true);
    }
    try { this.viewer?.render?.(); } catch {}
  }

  _applyRuntimeTransforms() {
    this._captureBaseLocalPoses();
    for (const solid of this._listSceneSolids()) {
      this._applyRuntimeTransformToSolid(solid);
    }
    try { this.viewer?.render?.(); } catch {}
  }

  _applyRuntimeTransformToSolid(solid) {
    const base = this._baseLocalPose.get(solid.uuid);
    if (!base) return;
    const runtime = normalizeTransform(this._runtimeState.get(solid.uuid));
    const offsetQuaternion = new THREE.Quaternion().setFromEuler(
      new THREE.Euler(...runtime.rotationEuler, 'XYZ'),
    );
    const nextMatrix = buildLocalMatrix(base).multiply(
      new THREE.Matrix4().compose(
        new THREE.Vector3(...runtime.position),
        offsetQuaternion,
        new THREE.Vector3(1, 1, 1),
      ),
    );
    nextMatrix.decompose(solid.position, solid.quaternion, solid.scale);
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }

  _updateRuntimeTransformFromSolid(solid, { persist = false } = {}) {
    const base = this._baseLocalPose.get(solid.uuid);
    if (!base) return;
    const offset = normalizeTransform(poseToOffset(base, solid));
    this._runtimeState.set(solid.uuid, offset);
    if (persist) {
      this.setStoredTransform(solid, offset);
      this.viewer?.partHistory?.queueHistorySnapshot?.({ debounceMs: 0, reason: 'simulation-transform' });
    }
    this._syncProxyGroupTransform(solid);
  }

  async _prepareSimulationAssets() {
    if (!this._active) return;
    try {
      await Promise.all(this._listSceneSolids().map((solid) => this._ensureDecomposition(solid)));
      this._buildProxyGroups();
      await this._rebuildPhysicsWorld();
    } catch (error) {
      console.warn('[SimulationWorkbench] Failed to prepare simulation assets:', error);
    }
  }

  async _prepareVisualAssets() {
    if (!this._active) return;
    try {
      await Promise.all(this._listSceneSolids().map((solid) => this._ensureDecomposition(solid)));
      this._buildProxyGroups();
    } catch (error) {
      console.warn('[SimulationWorkbench] Failed to prepare simulation visuals:', error);
    }
  }

  async _loadRapier() {
    if (this._rapier) return this._rapier;
    if (!this._rapierLoadPromise) {
      this._rapierLoadPromise = import('@dimforge/rapier3d/rapier.js')
        .then((module) => module?.default || module)
        .then((rapier) => {
          this._rapier = rapier;
          return rapier;
        })
        .catch((error) => {
          this._rapierLoadPromise = null;
          throw error;
        });
    }
    return this._rapierLoadPromise;
  }

  async _loadVhacd() {
    if (this._vhacd) return this._vhacd;
    if (!this._vhacdLoadPromise) {
      this._vhacdLoadPromise = import('vhacd-js/lib/vhacd.js')
        .then((module) => module?.ConvexMeshDecomposition?.create?.())
        .then((instance) => {
          this._vhacd = instance || null;
          return this._vhacd;
        })
        .catch((error) => {
          this._vhacdLoadPromise = null;
          throw error;
        });
    }
    return this._vhacdLoadPromise;
  }

  async _ensureDecomposition(solid) {
    if (!solid?.uuid) return [];
    const cached = this._decompositionCache.get(solid.uuid);
    if (cached) return cached.hulls;
    const rawMesh = this._extractSolidMesh(solid);
    if (!rawMesh) {
      this._decompositionCache.set(solid.uuid, { hulls: [] });
      return [];
    }
    let hulls = [];
    try {
      const vhacd = await this._loadVhacd();
      if (vhacd?.computeConvexHulls) {
        hulls = vhacd.computeConvexHulls(rawMesh, VHACD_OPTIONS) || [];
      }
    } catch (error) {
      console.warn('[SimulationWorkbench] Failed to decompose mesh with vhacd-js:', error);
    }
    if (!Array.isArray(hulls) || hulls.length === 0) {
      hulls = [rawMesh];
    }
    this._decompositionCache.set(solid.uuid, { hulls });
    return hulls;
  }

  _extractSolidMesh(solid) {
    let mesh = null;
    try {
      mesh = solid.getMesh?.();
      return normalizeMeshArrays(mesh);
    } catch {
      return null;
    } finally {
      try { if (mesh && typeof mesh.delete === 'function') mesh.delete(); } catch {}
    }
  }

  _buildProxyGroups() {
    this._clearProxyGroups();
    for (const solid of this._listSceneSolids()) {
      const record = this._decompositionCache.get(solid.uuid);
      const hulls = Array.isArray(record?.hulls) ? record.hulls : [];
      const group = this._createProxyGroup(solid, hulls);
      if (!group) continue;
      this._proxyGroups.set(solid.uuid, group);
      this._syncProxyGroupTransform(solid);
      try { this.viewer?.render?.(); } catch {}
    }
  }

  _createProxyGroup(solid, hulls) {
    const parent = solid.parent || this.viewer?.scene || null;
    if (!parent?.add || !Array.isArray(hulls) || hulls.length === 0) return null;
    const group = new THREE.Group();
    group.name = `${SIM_PROXY_PREFIX}:${solid.name || solid.uuid || ''}`;
    group.userData.excludeFromFit = false;
    group.userData.simulationProxy = true;
    group.visible = false;
    const metadataColor = this.viewer?.partHistory?.metadataManager?.getMetadata?.(solid.name || '')?.color || null;
    hulls.forEach((hull, index) => {
      const geometry = new THREE.BufferGeometry();
      geometry.setAttribute('position', new THREE.Float32BufferAttribute(Array.from(hull.positions || []), 3));
      geometry.setIndex(Array.from(hull.indices || []));
      geometry.computeVertexNormals();
      const mesh = new THREE.Mesh(geometry, buildProxyMaterial(metadataColor, index, hulls.length));
      mesh.userData.excludeFromFit = false;
      mesh.userData.simulationProxy = true;
      mesh.userData.sourceSolidUuid = solid.uuid;
      group.add(mesh);

      const edges = new THREE.LineSegments(
        new THREE.EdgesGeometry(geometry, 20),
        new THREE.LineBasicMaterial({ color: 0x22d3ee, transparent: true, opacity: 0.9 }),
      );
      edges.userData.excludeFromFit = false;
      edges.userData.simulationProxy = true;
      group.add(edges);
    });
    parent.add(group);
    return group;
  }

  _syncProxyGroupTransform(solid) {
    const group = this._proxyGroups.get(solid.uuid);
    if (!group) return;
    group.position.copy(solid.position);
    group.quaternion.copy(solid.quaternion);
    group.scale.copy(solid.scale);
    group.visible = false;
    group.updateMatrixWorld?.(true);
  }

  _clearProxyGroups() {
    for (const group of this._proxyGroups.values()) {
      try { group.parent?.remove?.(group); } catch {}
      disposeHierarchy(group);
    }
    this._proxyGroups.clear();
  }

  _setSourceSolidsVisible(visible) {
    for (const solid of this._listSceneSolids()) {
      if (visible) {
        solid.visible = this._sourceVisibility.get(solid.uuid) !== false;
      } else {
        solid.visible = false;
      }
    }
    for (const group of this._proxyGroups.values()) {
      group.visible = false;
    }
    try { this.viewer?.render?.(); } catch {}
  }

  async _rebuildPhysicsWorld() {
    if (!this._active) return;
    const [rapier] = await Promise.all([
      this._loadRapier(),
      Promise.all(this._listSceneSolids().map((solid) => this._ensureDecomposition(solid))),
    ]);
    if (!this._active || !rapier) return;
    this._destroyPhysicsWorld();
    this._bodyState.clear();
    const world = new rapier.World({ x: 0, y: -9.81, z: 0 });
    world.maxCcdSubsteps = 2;
    this._physicsWorld = world;
    const selectedSolid = this._transformSession?.solid || null;
    for (const solid of this._listSceneSolids()) {
      const bodyDesc = (selectedSolid && solid === selectedSolid)
        ? rapier.RigidBodyDesc.kinematicPositionBased()
        : (this._isSolidMotionDriven(solid)
          ? rapier.RigidBodyDesc.kinematicPositionBased()
          : (this.getSolidFixed(solid) ? rapier.RigidBodyDesc.fixed() : rapier.RigidBodyDesc.dynamic()));
      const position = solid.getWorldPosition(new THREE.Vector3());
      const quaternion = solid.getWorldQuaternion(new THREE.Quaternion());
      bodyDesc.setTranslation(position.x, position.y, position.z);
      bodyDesc.setRotation({ x: quaternion.x, y: quaternion.y, z: quaternion.z, w: quaternion.w });
      bodyDesc.setLinearDamping(2.5);
      bodyDesc.setAngularDamping(2.5);
      bodyDesc.setCcdEnabled(true);
      const body = world.createRigidBody(bodyDesc);
      const hulls = this._decompositionCache.get(solid.uuid)?.hulls || [];
      for (const hull of hulls) {
        const collider = this._createColliderForHull(rapier, hull);
        if (!collider) continue;
        collider.setRestitution(0.05);
        collider.setFriction(0.9);
        world.createCollider(collider, body);
      }
      this._bodyState.set(solid.uuid, {
        solid,
        body,
        proxyGroup: this._proxyGroups.get(solid.uuid) || null,
      });
    }
  }

  _destroyPhysicsWorld() {
    try { this._physicsWorld?.free?.(); } catch {}
    this._physicsWorld = null;
    this._bodyState.clear();
  }

  _createColliderForHull(rapier, hull) {
    const positions = hull?.positions ? new Float32Array(hull.positions) : null;
    const indices = hull?.indices ? new Uint32Array(hull.indices) : null;
    if (!positions || positions.length < 9 || !indices || indices.length < 3) return null;
    const collider = rapier.ColliderDesc.convexMesh(positions, indices);
    if (collider) return collider;
    return rapier.ColliderDesc.convexHull(positions);
  }

  _ensurePhysicsLoop() {
    if (this._raf) return;
    this._lastStepTime = 0;
    const step = (ts) => {
      this._raf = 0;
      if (!this._active) return;
      this._stepPhysics(ts);
      this._raf = requestAnimationFrame(step);
    };
    this._raf = requestAnimationFrame(step);
  }

  _stopPhysicsLoop() {
    if (this._raf) {
      cancelAnimationFrame(this._raf);
      this._raf = 0;
    }
    this._lastStepTime = 0;
  }

  _stepPhysics(timestamp) {
    const world = this._physicsWorld;
    if (!world) return;
    if (!this._isPlaying) {
      this._lastStepTime = timestamp;
      return;
    }
    const dt = this._lastStepTime > 0
      ? Math.min(1 / 20, Math.max(1 / 240, (timestamp - this._lastStepTime) / 1000))
      : 1 / 60;
    this._lastStepTime = timestamp;
    world.timestep = dt;

    const selectedSolid = this._transformSession?.solid || null;
    let changed = false;
    if (this._isPlaying) {
      changed = this._applyMotionStep(dt) || changed;
    }
    if (selectedSolid) {
      const bodyState = this._bodyState.get(selectedSolid.uuid);
      if (bodyState?.body) {
        const worldPosition = selectedSolid.getWorldPosition(new THREE.Vector3());
        const worldQuaternion = selectedSolid.getWorldQuaternion(new THREE.Quaternion());
        bodyState.body.setNextKinematicTranslation(worldPosition);
        bodyState.body.setNextKinematicRotation(worldQuaternion);
      }
    }
    if (this._isPlaying) {
      for (const solid of this._listSceneSolids()) {
        if (!this._isSolidMotionDriven(solid)) continue;
        const bodyState = this._bodyState.get(solid.uuid);
        if (!bodyState?.body) continue;
        const worldPosition = solid.getWorldPosition(new THREE.Vector3());
        const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
        bodyState.body.setNextKinematicTranslation(worldPosition);
        bodyState.body.setNextKinematicRotation(worldQuaternion);
      }
    }

    world.step();
    for (const bodyState of this._bodyState.values()) {
      const { solid, body } = bodyState;
      if (!solid || !body) continue;
      if (selectedSolid && solid === selectedSolid) {
        this._syncProxyGroupTransform(solid);
        continue;
      }
      if (this._isSolidMotionDriven(solid)) {
        this._syncProxyGroupTransform(solid);
        continue;
      }
      if (this.getSolidFixed(solid)) {
        this._syncProxyGroupTransform(solid);
        continue;
      }
      const translation = body.translation();
      const rotation = body.rotation();
      if (!translation || !rotation) continue;
      this._setSolidWorldPose(solid, translation, rotation);
      this._updateRuntimeTransformFromSolid(solid, { persist: false });
      this._movedSolidIds.add(solid.uuid);
      changed = true;
    }
    if (changed) {
      try { this.viewer?.render?.(); } catch {}
    }
  }

  _setSolidWorldPose(solid, translation, rotation) {
    const parent = solid.parent || this.viewer?.scene || null;
    const scratch = this._scratch;
    scratch.worldPosition.set(translation.x, translation.y, translation.z);
    scratch.worldQuaternion.set(rotation.x, rotation.y, rotation.z, rotation.w);
    if (parent?.isObject3D) {
      parent.updateMatrixWorld?.(true);
      scratch.parentInverse.copy(parent.matrixWorld).invert();
      parent.getWorldQuaternion(scratch.parentQuaternion);
      scratch.parentQuaternionInverse.copy(scratch.parentQuaternion).invert();
      scratch.localPosition.copy(scratch.worldPosition).applyMatrix4(scratch.parentInverse);
      solid.position.copy(scratch.localPosition);
      solid.quaternion.copy(scratch.parentQuaternionInverse.multiply(scratch.worldQuaternion));
    } else {
      solid.position.copy(scratch.worldPosition);
      solid.quaternion.copy(scratch.worldQuaternion);
    }
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }

  _startTransformSession(solid) {
    if (!solid || !this._active) return;
    this._stopTransformSession();
    const controls = new CombinedTransformControls(this.viewer.camera, this.viewer.renderer?.domElement);
    try { controls.setMode('translate'); } catch { controls.mode = 'translate'; }
    const target = new THREE.Object3D();
    target.name = `SimulationTransformTarget:${solid.name || solid.uuid || ''}`;
    this.viewer.scene.updateMatrixWorld?.(true);
    solid.updateMatrixWorld?.(true);
    const box = this._scratch.box;
    const center = box.setFromObject(this._proxyGroups.get(solid.uuid) || solid).isEmpty()
      ? solid.getWorldPosition(this._scratch.center)
      : box.getCenter(this._scratch.center);
    const worldPosition = solid.getWorldPosition(new THREE.Vector3());
    const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
    const offsetLocal = worldPosition.clone().sub(center).applyQuaternion(worldQuaternion.clone().invert());
    target.position.copy(center);
    target.quaternion.copy(worldQuaternion);
    this.viewer.scene.add(target);
    controls.attach(target);
    this.viewer.scene.add(controls);

    const applyToSolid = (persist = false) => {
      const targetWorldPosition = target.getWorldPosition(new THREE.Vector3());
      const targetWorldQuaternion = target.getWorldQuaternion(new THREE.Quaternion());
      const solidWorldPosition = targetWorldPosition.clone().add(offsetLocal.clone().applyQuaternion(targetWorldQuaternion));
      this._setSolidWorldPose(solid, solidWorldPosition, targetWorldQuaternion);
      this._updateRuntimeTransformFromSolid(solid, { persist });
      if (persist) {
        if (this._physicsWorld) {
          void this._rebuildPhysicsWorld();
        }
      }
      try { this.viewer?.render?.(); } catch {}
    };

    const changeHandler = () => applyToSolid(false);
    const dragHandler = (event) => {
      const dragging = !!event?.value;
      try { if (this.viewer?.controls) this.viewer.controls.enabled = !dragging; } catch {}
      if (!dragging) applyToSolid(true);
    };
    const objectChangeHandler = () => {
      if (!controls.dragging) applyToSolid(true);
    };
    const updateForCamera = () => {
      try { controls.update?.(); } catch {}
    };
    const cameraChangeHandler = () => updateForCamera();
    controls.addEventListener('change', changeHandler);
    controls.addEventListener('dragging-changed', dragHandler);
    controls.addEventListener('objectChange', objectChangeHandler);
    try { this.viewer?.controls?.addEventListener?.('change', cameraChangeHandler); } catch {}

    const globalState = { controls, viewer: this.viewer, target, updateForCamera };
    try { window.__BREP_activeXform = globalState; } catch {}
    this._transformSession = {
      solid,
      controls,
      target,
      changeHandler,
      dragHandler,
      objectChangeHandler,
      cameraChangeHandler,
      globalState,
      mode: 'translate',
    };
    if (this._physicsWorld) {
      void this._rebuildPhysicsWorld();
    }
    updateForCamera();
    applyToSolid(false);
  }

  _stopTransformSession() {
    const session = this._transformSession;
    if (!session) return;
    try { session.controls?.removeEventListener('change', session.changeHandler); } catch {}
    try { session.controls?.removeEventListener('dragging-changed', session.dragHandler); } catch {}
    try { session.controls?.removeEventListener('objectChange', session.objectChangeHandler); } catch {}
    try { this.viewer?.controls?.removeEventListener?.('change', session.cameraChangeHandler); } catch {}
    try { session.controls?.detach?.(); } catch {}
    try { this.viewer?.scene?.remove?.(session.controls); } catch {}
    try { this.viewer?.scene?.remove?.(session.target); } catch {}
    try { session.controls?.dispose?.(); } catch {}
    try {
      if (window.__BREP_activeXform === session.globalState) {
        window.__BREP_activeXform = null;
      }
    } catch {}
    this._transformSession = null;
    try { if (this.viewer?.controls) this.viewer.controls.enabled = true; } catch {}
    if (this._physicsWorld) {
      void this._rebuildPhysicsWorld();
    }
    try { this.viewer?.render?.(); } catch {}
  }

  _listSceneSolids() {
    const scene = this.viewer?.partHistory?.scene || this.viewer?.scene || null;
    const solids = [];
    if (!scene?.traverse) return solids;
    scene.traverse((obj) => {
      if (obj?.type === 'SOLID') solids.push(obj);
    });
    return solids;
  }

  _bindStateManager() {
    const manager = this.viewer?.partHistory?.simulationStateManager;
    if (!manager?.addListener) return;
    this._removeStateManagerListener = manager.addListener(() => {
      this._reconcileMotionState();
      if (this._active && this._physicsWorld) {
        void this._rebuildPhysicsWorld();
      }
      this._emit();
    });
  }

  _emit() {
    if (!this._listeners || this._listeners.size === 0) return;
    const payload = {
      active: this._active,
      playing: this._isPlaying,
      manager: this,
    };
    for (const listener of Array.from(this._listeners)) {
      try { listener(payload); } catch {}
    }
  }

  _getMotionEntries() {
    const motions = this.viewer?.partHistory?.simulationStateManager?.getMotions?.();
    return Array.isArray(motions)
      ? motions.map((entry) => (entry?.inputParams && typeof entry.inputParams === 'object') ? entry.inputParams : entry).filter(Boolean)
      : [];
  }

  _resetMotionRuntime() {
    this._motionState.clear();
    for (const motion of this._getMotionEntries()) {
      this._motionState.set(motion.id, { progress: 0, completed: false });
    }
  }

  _reconcileMotionState() {
    const activeIds = new Set();
    for (const motion of this._getMotionEntries()) {
      activeIds.add(motion.id);
      if (!this._motionState.has(motion.id)) {
        this._motionState.set(motion.id, { progress: 0, completed: false });
      }
    }
    for (const id of Array.from(this._motionState.keys())) {
      if (!activeIds.has(id)) this._motionState.delete(id);
    }
  }

  _isSolidMotionDriven(solid) {
    if (!solid || !this._isPlaying) return false;
    const name = normalizeText(solid.name, '');
    if (!name) return false;
    return this._getMotionEntries().some((motion) => normalizeText(this._resolveReferenceName(motion?.solid), '') === name);
  }

  _applyMotionStep(dt) {
    let changed = false;
    for (const motion of this._getMotionEntries()) {
      const solid = this._resolveMotionSolid(motion);
      if (!solid) continue;
      if (motion.type === 'linear') changed = this._applyLinearMotionStep(solid, motion, dt) || changed;
      else changed = this._applyRotationMotionStep(solid, motion, dt) || changed;
    }
    return changed;
  }

  _resolveMotionSolid(motion) {
    const name = normalizeText(this._resolveReferenceName(motion?.solid), '');
    if (!name) return null;
    const object = this.viewer?.partHistory?.getObjectByName?.(name) || null;
    return object?.type === 'SOLID' ? object : null;
  }

  _resolveMotionAxis(motion) {
    const objectName = normalizeText(this._resolveReferenceName(motion?.axis), '');
    const object = objectName ? this.viewer?.partHistory?.getObjectByName?.(objectName) : null;
    if (!object) return null;
    const live = this._extractAxisPointsFromObject(object);
    if (!live?.start || !live?.end) return null;
    const startVec = new THREE.Vector3(Number(live.start[0]) || 0, Number(live.start[1]) || 0, Number(live.start[2]) || 0);
    const endVec = new THREE.Vector3(Number(live.end[0]) || 0, Number(live.end[1]) || 0, Number(live.end[2]) || 0);
    if (startVec.distanceToSquared(endVec) <= 1e-12) return null;
    return { start: startVec, end: endVec };
  }

  _resolveReferenceName(value) {
    if (Array.isArray(value)) {
      return this._resolveReferenceName(value[0] || null);
    }
    if (value && typeof value === 'object') {
      return normalizeText(value.name || value.objectName || value.label || '', '');
    }
    return normalizeText(value, '');
  }

  _extractAxisPointsFromObject(object) {
    if (!object) return null;
    try {
      if (typeof object.points === 'function') {
        const points = object.points(true);
        if (Array.isArray(points) && points.length >= 2) {
          const first = points[0];
          const last = points[points.length - 1];
          return {
            start: [Number(first?.x) || 0, Number(first?.y) || 0, Number(first?.z) || 0],
            end: [Number(last?.x) || 0, Number(last?.y) || 0, Number(last?.z) || 0],
          };
        }
      }
    } catch {}
    try {
      object.updateMatrixWorld?.(true);
      const attr = object.geometry?.getAttribute?.('position');
      if (!attr || attr.count < 2) return null;
      const first = new THREE.Vector3(attr.getX(0), attr.getY(0), attr.getZ(0)).applyMatrix4(object.matrixWorld);
      const lastIndex = attr.count - 1;
      const last = new THREE.Vector3(attr.getX(lastIndex), attr.getY(lastIndex), attr.getZ(lastIndex)).applyMatrix4(object.matrixWorld);
      return {
        start: [first.x, first.y, first.z],
        end: [last.x, last.y, last.z],
      };
    } catch {
      return null;
    }
  }

  _applyRotationMotionStep(solid, motion, dt) {
    const state = this._motionState.get(motion.id) || { progress: 0, completed: false };
    if (state.completed) return;
    const axis = this._resolveMotionAxis(motion);
    if (!axis) return;
    const speedDeg = Number(motion?.speed);
    if (!Number.isFinite(speedDeg) || Math.abs(speedDeg) <= 1e-9) return;
    let deltaDeg = speedDeg * dt;
    const limitDeg = Number.isFinite(Number(motion?.angle)) ? Number(motion.angle) : null;
    if (limitDeg != null) {
      const remaining = limitDeg - state.progress;
      if (Math.abs(remaining) <= 1e-9) {
        state.completed = true;
        this._motionState.set(motion.id, state);
        return false;
      }
      if (Math.sign(deltaDeg || 1) !== Math.sign(remaining || 1)) {
        deltaDeg = Math.sign(remaining || 1) * Math.abs(deltaDeg);
      }
      if (Math.abs(deltaDeg) > Math.abs(remaining)) deltaDeg = remaining;
    }
    if (Math.abs(deltaDeg) <= 1e-9) return false;
    this._rotateSolidAroundWorldAxis(solid, axis.start, axis.end, THREE.MathUtils.degToRad(deltaDeg));
    state.progress += deltaDeg;
    if (limitDeg != null && Math.abs(limitDeg - state.progress) <= 1e-9) state.completed = true;
    this._motionState.set(motion.id, state);
    this._updateRuntimeTransformFromSolid(solid, { persist: false });
    this._movedSolidIds.add(solid.uuid);
    return true;
  }

  _applyLinearMotionStep(solid, motion, dt) {
    const state = this._motionState.get(motion.id) || { progress: 0, completed: false };
    if (state.completed) return;
    const axis = this._resolveMotionAxis(motion);
    if (!axis) return;
    const speed = Number(motion?.speed);
    if (!Number.isFinite(speed) || Math.abs(speed) <= 1e-9) return;
    let delta = speed * dt;
    const limit = Number.isFinite(Number(motion?.distance)) ? Number(motion.distance) : null;
    if (limit != null) {
      const remaining = limit - state.progress;
      if (Math.abs(remaining) <= 1e-9) {
        state.completed = true;
        this._motionState.set(motion.id, state);
        return false;
      }
      if (Math.sign(delta || 1) !== Math.sign(remaining || 1)) {
        delta = Math.sign(remaining || 1) * Math.abs(delta);
      }
      if (Math.abs(delta) > Math.abs(remaining)) delta = remaining;
    }
    if (Math.abs(delta) <= 1e-9) return false;
    this._translateSolidAlongWorldAxis(solid, axis.start, axis.end, delta);
    state.progress += delta;
    if (limit != null && Math.abs(limit - state.progress) <= 1e-9) state.completed = true;
    this._motionState.set(motion.id, state);
    this._updateRuntimeTransformFromSolid(solid, { persist: false });
    this._movedSolidIds.add(solid.uuid);
    return true;
  }

  _rotateSolidAroundWorldAxis(solid, axisStart, axisEnd, angleRad) {
    const parent = solid.parent || this.viewer?.scene || null;
    const scratch = this._scratch;
    scratch.axis.copy(axisEnd).sub(axisStart).normalize();
    scratch.parentWorldMatrix.copy(parent?.matrixWorld || new THREE.Matrix4());
    scratch.parentInverse.copy(scratch.parentWorldMatrix).invert();
    scratch.worldMatrix.copy(solid.matrixWorld);
    scratch.deltaMatrix.makeTranslation(axisStart.x, axisStart.y, axisStart.z);
    scratch.deltaMatrix.multiply(new THREE.Matrix4().makeRotationAxis(scratch.axis, angleRad));
    scratch.deltaMatrix.multiply(new THREE.Matrix4().makeTranslation(-axisStart.x, -axisStart.y, -axisStart.z));
    scratch.worldMatrix.premultiply(scratch.deltaMatrix);
    scratch.localMatrix.copy(scratch.parentInverse).multiply(scratch.worldMatrix);
    scratch.localMatrix.decompose(solid.position, solid.quaternion, solid.scale);
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }

  _translateSolidAlongWorldAxis(solid, axisStart, axisEnd, distance) {
    const parent = solid.parent || this.viewer?.scene || null;
    const scratch = this._scratch;
    scratch.axis.copy(axisEnd).sub(axisStart).normalize();
    scratch.translation.copy(scratch.axis).multiplyScalar(distance);
    scratch.parentWorldMatrix.copy(parent?.matrixWorld || new THREE.Matrix4());
    scratch.parentInverse.copy(scratch.parentWorldMatrix).invert();
    scratch.worldMatrix.copy(solid.matrixWorld);
    scratch.deltaMatrix.makeTranslation(scratch.translation.x, scratch.translation.y, scratch.translation.z);
    scratch.worldMatrix.premultiply(scratch.deltaMatrix);
    scratch.localMatrix.copy(scratch.parentInverse).multiply(scratch.worldMatrix);
    scratch.localMatrix.decompose(solid.position, solid.quaternion, solid.scale);
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }
}
", import.meta.url).href, { SimulationWorkbenchManager: e } = await import(
|
|
304372
|
+
const A = new URL("data:text/javascript;base64,import * as THREE from 'three';
import { CombinedTransformControls } from '../UI/controls/CombinedTransformControls.js';

const FIXED_METADATA_KEY = 'fixed';
const TRANSFORM_METADATA_KEY = 'simulationTransform';
const SIM_PROXY_PREFIX = '__SIM_PROXY__';
const VHACD_OPTIONS = Object.freeze({
  maxHulls: 64,
  voxelResolution: 1000000,
  maxVerticesPerHull: 64,
  shrinkWrap: true,
  fillMode: 'flood',
  findBestPlane: true,
  minEdgeLength: 1,
  messages: 'none',
});
const DEFAULT_TRANSFORM = Object.freeze({
  position: [0, 0, 0],
  rotationEuler: [0, 0, 0],
});

function cloneVec3(value, fallback = [0, 0, 0]) {
  const src = Array.isArray(value) ? value : fallback;
  return [
    Number.isFinite(Number(src[0])) ? Number(src[0]) : fallback[0],
    Number.isFinite(Number(src[1])) ? Number(src[1]) : fallback[1],
    Number.isFinite(Number(src[2])) ? Number(src[2]) : fallback[2],
  ];
}

function normalizeText(value, fallback = '') {
  const next = String(value == null ? '' : value).trim();
  return next || fallback;
}

function normalizeOptionalMotionLimit(value) {
  const numeric = Number(value);
  if (!Number.isFinite(numeric)) return null;
  return Math.abs(numeric) <= 1e-9 ? null : numeric;
}

function normalizeTransform(value) {
  const raw = value && typeof value === 'object' ? value : {};
  return {
    position: cloneVec3(raw.position, DEFAULT_TRANSFORM.position),
    rotationEuler: cloneVec3(raw.rotationEuler, DEFAULT_TRANSFORM.rotationEuler),
  };
}

function transformIsIdentity(transform) {
  const position = cloneVec3(transform?.position, DEFAULT_TRANSFORM.position);
  const rotationEuler = cloneVec3(transform?.rotationEuler, DEFAULT_TRANSFORM.rotationEuler);
  return position.every((value) => Math.abs(value) <= 1e-9)
    && rotationEuler.every((value) => Math.abs(value) <= 1e-9);
}

function copyLocalPose(object) {
  return {
    position: object.position.clone(),
    quaternion: object.quaternion.clone(),
    scale: object.scale.clone(),
  };
}

function buildLocalMatrix(pose) {
  return new THREE.Matrix4().compose(
    pose.position.clone(),
    pose.quaternion.clone(),
    pose.scale.clone(),
  );
}

function poseToOffset(basePose, currentObject) {
  const baseMatrix = buildLocalMatrix(basePose);
  const currentMatrix = new THREE.Matrix4().compose(
    currentObject.position.clone(),
    currentObject.quaternion.clone(),
    currentObject.scale.clone(),
  );
  const relative = new THREE.Matrix4().copy(baseMatrix).invert().multiply(currentMatrix);
  const position = new THREE.Vector3();
  const quaternion = new THREE.Quaternion();
  const scale = new THREE.Vector3();
  relative.decompose(position, quaternion, scale);
  const rotationEuler = new THREE.Euler().setFromQuaternion(quaternion, 'XYZ');
  return {
    position: [position.x, position.y, position.z],
    rotationEuler: [rotationEuler.x, rotationEuler.y, rotationEuler.z],
  };
}

function normalizeMeshArrays(mesh) {
  if (!mesh) return null;
  const stride = Math.max(3, Number(mesh.numProp) || 3);
  const sourcePositions = Array.isArray(mesh.vertProperties) ? mesh.vertProperties : Array.from(mesh.vertProperties || []);
  const sourceIndices = Array.isArray(mesh.triVerts) ? mesh.triVerts : Array.from(mesh.triVerts || []);
  if (sourcePositions.length < 9 || sourceIndices.length < 3) return null;
  const positions = new Float64Array((sourcePositions.length / stride) * 3);
  for (let src = 0, dst = 0; src + 2 < sourcePositions.length; src += stride, dst += 3) {
    positions[dst + 0] = Number(sourcePositions[src + 0]) || 0;
    positions[dst + 1] = Number(sourcePositions[src + 1]) || 0;
    positions[dst + 2] = Number(sourcePositions[src + 2]) || 0;
  }
  return {
    positions,
    indices: new Uint32Array(sourceIndices),
  };
}

function buildProxyMaterial(baseColor, index, total) {
  const color = new THREE.Color(baseColor || '#9ca3af');
  const hsl = { h: 0, s: 0, l: 0 };
  color.getHSL(hsl);
  const offset = total > 1 ? (index / Math.max(1, total)) * 0.18 : 0;
  color.setHSL((hsl.h + offset) % 1, Math.min(1, Math.max(0.35, hsl.s || 0.55)), Math.min(0.72, Math.max(0.42, hsl.l)));
  return new THREE.MeshStandardMaterial({
    color,
    transparent: true,
    opacity: 0.55,
    roughness: 0.55,
    metalness: 0.05,
    depthWrite: true,
  });
}

function disposeHierarchy(root) {
  if (!root?.traverse) return;
  root.traverse((node) => {
    if (node?.geometry?.dispose) {
      try { node.geometry.dispose(); } catch {}
    }
    if (Array.isArray(node?.material)) {
      for (const material of node.material) {
        try { material?.dispose?.(); } catch {}
      }
    } else {
      try { node?.material?.dispose?.(); } catch {}
    }
  });
}

export class SimulationWorkbenchManager {
  constructor(viewer) {
    this.viewer = viewer || null;
    this._active = false;
    this._baseLocalPose = new Map();
    this._runtimeState = new Map();
    this._sourceVisibility = new Map();
    this._decompositionCache = new Map();
    this._proxyGroups = new Map();
    this._transformSession = null;
    this._rapier = null;
    this._rapierLoadPromise = null;
    this._vhacd = null;
    this._vhacdLoadPromise = null;
    this._physicsWorld = null;
    this._bodyState = new Map();
    this._listeners = new Set();
    this._removeStateManagerListener = null;
    this._isPlaying = false;
    this._motionState = new Map();
    this._movedSolidIds = new Set();
    this._raf = 0;
    this._lastStepTime = 0;
    this._scratch = {
      parentInverse: new THREE.Matrix4(),
      parentWorldMatrix: new THREE.Matrix4(),
      localMatrix: new THREE.Matrix4(),
      worldMatrix: new THREE.Matrix4(),
      deltaMatrix: new THREE.Matrix4(),
      worldPosition: new THREE.Vector3(),
      localPosition: new THREE.Vector3(),
      worldQuaternion: new THREE.Quaternion(),
      parentQuaternion: new THREE.Quaternion(),
      parentQuaternionInverse: new THREE.Quaternion(),
      box: new THREE.Box3(),
      center: new THREE.Vector3(),
      axis: new THREE.Vector3(),
      translation: new THREE.Vector3(),
      velocity: new THREE.Vector3(),
      angularVelocity: new THREE.Vector3(),
      radius: new THREE.Vector3(),
    };
    this._bindStateManager();
  }

  dispose() {
    if (typeof this._removeStateManagerListener === 'function') {
      try { this._removeStateManagerListener(); } catch {}
    }
    this._removeStateManagerListener = null;
    this.setActive(false);
  }

  isActive() {
    return this._active;
  }

  isSimulationWorkbenchActive() {
    return this.viewer?._getActiveWorkbenchId?.() === 'SIMULATION';
  }

  addListener(listener) {
    if (typeof listener !== 'function') return () => {};
    this._listeners.add(listener);
    return () => {
      try { this._listeners.delete(listener); } catch {}
    };
  }

  removeListener(listener) {
    if (typeof listener !== 'function') return;
    try { this._listeners.delete(listener); } catch {}
  }

  isPlaying() {
    return this._isPlaying;
  }

  setPlaying(playing) {
    const next = !!playing;
    if (next === this._isPlaying) return this._isPlaying;
    this._isPlaying = next;
    if (this._active) {
      if (next) {
        void this._prepareSimulationAssets().then(() => {
          this._ensurePhysicsLoop();
        });
      } else {
        this._stopPhysicsLoop();
      }
    }
    this._emit();
    try { this.viewer?.render?.(); } catch {}
    return this._isPlaying;
  }

  resetSimulationState() {
    this._stopTransformSession();
    this._stopPhysicsLoop();
    this._destroyPhysicsWorld();
    this._isPlaying = false;
    this._movedSolidIds.clear();
    this._motionState.clear();
    for (const solid of this._listSceneSolids()) {
      this._runtimeState.set(solid.uuid, normalizeTransform(null));
      this.setStoredTransform(solid, null);
    }
    this._restoreBaseLocalPoses();
    for (const solid of this._listSceneSolids()) {
      this._syncProxyGroupTransform(solid);
    }
    if (this._active) {
      this._resetMotionRuntime();
    }
    this._emit();
    try { this.viewer?.render?.(); } catch {}
  }

  getSolidFixed(solid) {
    if (!solid?.name) return false;
    const own = this.viewer?.partHistory?.metadataManager?.getOwnMetadata?.(solid.name) || {};
    return own[FIXED_METADATA_KEY] === true;
  }

  setSolidFixed(solid, fixed) {
    if (!solid?.name) return false;
    const manager = this.viewer?.partHistory?.metadataManager;
    if (!manager) return false;
    const data = manager.getOwnMetadata(solid.name);
    if (fixed) data[FIXED_METADATA_KEY] = true;
    else delete data[FIXED_METADATA_KEY];
    manager.setMetadataObject(solid.name, data);
    if (this._active && this._physicsWorld) {
      void this._rebuildPhysicsWorld();
    }
    return true;
  }

  getStoredTransform(solid) {
    if (!solid?.name) return normalizeTransform(null);
    const own = this.viewer?.partHistory?.metadataManager?.getOwnMetadata?.(solid.name) || {};
    return normalizeTransform(own[TRANSFORM_METADATA_KEY]);
  }

  setStoredTransform(solid, transform) {
    if (!solid?.name) return false;
    const manager = this.viewer?.partHistory?.metadataManager;
    if (!manager) return false;
    const normalized = normalizeTransform(transform);
    const data = manager.getOwnMetadata(solid.name);
    if (transformIsIdentity(normalized)) delete data[TRANSFORM_METADATA_KEY];
    else data[TRANSFORM_METADATA_KEY] = normalized;
    manager.setMetadataObject(solid.name, data);
    return true;
  }

  setActive(active) {
    const next = !!active;
    if (next === this._active) {
      if (next) {
        this._applyRuntimeTransforms();
      } else {
        this._restoreBaseLocalPoses();
      }
      return;
    }
    this._active = next;
    if (this._active) {
      this._isPlaying = false;
      this._stopPhysicsLoop();
      this._destroyPhysicsWorld();
      this._captureBaseLocalPoses();
      this._captureSourceVisibility();
      this._hydrateRuntimeStateFromMetadata();
      this._resetMotionRuntime();
      this._applyRuntimeTransforms();
      this._setSourceSolidsVisible(true);
      void this._prepareVisualAssets();
      this._emit();
      return;
    }
    this._stopTransformSession();
    this._destroyPhysicsWorld();
    this._stopPhysicsLoop();
    this._clearProxyGroups();
    this._setSourceSolidsVisible(true);
    this._restoreBaseLocalPoses();
    this._runtimeState.clear();
    this._baseLocalPose.clear();
    this._sourceVisibility.clear();
    this._decompositionCache.clear();
    this._motionState.clear();
    this._movedSolidIds.clear();
    this._isPlaying = false;
    this._emit();
  }

  toggleSolidTransform(solid) {
    if (!solid || solid.type !== 'SOLID') return;
    if (!this.isSimulationWorkbenchActive()) return;
    const session = this._transformSession;
    if (session && session.solid === solid) {
      const currentMode = (typeof session.controls?.getMode === 'function')
        ? session.controls.getMode()
        : (session.controls?.mode || session.mode || 'translate');
      if (currentMode === 'translate') {
        try { session.controls?.setMode('rotate'); } catch { session.controls.mode = 'rotate'; }
        session.mode = 'rotate';
        try { session.globalState?.updateForCamera?.(); } catch {}
        try { this.viewer?.render?.(); } catch {}
        return;
      }
      this._stopTransformSession();
      return;
    }
    this._startTransformSession(solid);
  }

  _captureBaseLocalPoses() {
    for (const solid of this._listSceneSolids()) {
      if (!this._baseLocalPose.has(solid.uuid)) {
        this._baseLocalPose.set(solid.uuid, copyLocalPose(solid));
      }
    }
  }

  _captureSourceVisibility() {
    for (const solid of this._listSceneSolids()) {
      if (!this._sourceVisibility.has(solid.uuid)) {
        this._sourceVisibility.set(solid.uuid, solid.visible !== false);
      }
    }
  }

  _hydrateRuntimeStateFromMetadata() {
    for (const solid of this._listSceneSolids()) {
      this._runtimeState.set(solid.uuid, normalizeTransform(this.getStoredTransform(solid)));
    }
  }

  _restoreBaseLocalPoses() {
    for (const solid of this._listSceneSolids()) {
      const base = this._baseLocalPose.get(solid.uuid);
      if (!base) continue;
      solid.position.copy(base.position);
      solid.quaternion.copy(base.quaternion);
      solid.scale.copy(base.scale);
      solid.updateMatrixWorld?.(true);
    }
    try { this.viewer?.render?.(); } catch {}
  }

  _applyRuntimeTransforms() {
    this._captureBaseLocalPoses();
    for (const solid of this._listSceneSolids()) {
      this._applyRuntimeTransformToSolid(solid);
    }
    try { this.viewer?.render?.(); } catch {}
  }

  _applyRuntimeTransformToSolid(solid) {
    const base = this._baseLocalPose.get(solid.uuid);
    if (!base) return;
    const runtime = normalizeTransform(this._runtimeState.get(solid.uuid));
    const offsetQuaternion = new THREE.Quaternion().setFromEuler(
      new THREE.Euler(...runtime.rotationEuler, 'XYZ'),
    );
    const nextMatrix = buildLocalMatrix(base).multiply(
      new THREE.Matrix4().compose(
        new THREE.Vector3(...runtime.position),
        offsetQuaternion,
        new THREE.Vector3(1, 1, 1),
      ),
    );
    nextMatrix.decompose(solid.position, solid.quaternion, solid.scale);
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }

  _updateRuntimeTransformFromSolid(solid, { persist = false } = {}) {
    const base = this._baseLocalPose.get(solid.uuid);
    if (!base) return;
    const offset = normalizeTransform(poseToOffset(base, solid));
    this._runtimeState.set(solid.uuid, offset);
    if (persist) {
      this.setStoredTransform(solid, offset);
      this.viewer?.partHistory?.queueHistorySnapshot?.({ debounceMs: 0, reason: 'simulation-transform' });
    }
    this._syncProxyGroupTransform(solid);
  }

  async _prepareSimulationAssets() {
    if (!this._active) return;
    try {
      await Promise.all(this._listSceneSolids().map((solid) => this._ensureDecomposition(solid)));
      this._buildProxyGroups();
      await this._rebuildPhysicsWorld();
    } catch (error) {
      console.warn('[SimulationWorkbench] Failed to prepare simulation assets:', error);
    }
  }

  async _prepareVisualAssets() {
    if (!this._active) return;
    try {
      await Promise.all(this._listSceneSolids().map((solid) => this._ensureDecomposition(solid)));
      this._buildProxyGroups();
    } catch (error) {
      console.warn('[SimulationWorkbench] Failed to prepare simulation visuals:', error);
    }
  }

  async _loadRapier() {
    if (this._rapier) return this._rapier;
    if (!this._rapierLoadPromise) {
      this._rapierLoadPromise = import('@dimforge/rapier3d/rapier.js')
        .then((module) => module?.default || module)
        .then((rapier) => {
          this._rapier = rapier;
          return rapier;
        })
        .catch((error) => {
          this._rapierLoadPromise = null;
          throw error;
        });
    }
    return this._rapierLoadPromise;
  }

  async _loadVhacd() {
    if (this._vhacd) return this._vhacd;
    if (!this._vhacdLoadPromise) {
      this._vhacdLoadPromise = import('vhacd-js/lib/vhacd.js')
        .then((module) => module?.ConvexMeshDecomposition?.create?.())
        .then((instance) => {
          this._vhacd = instance || null;
          return this._vhacd;
        })
        .catch((error) => {
          this._vhacdLoadPromise = null;
          throw error;
        });
    }
    return this._vhacdLoadPromise;
  }

  async _ensureDecomposition(solid) {
    if (!solid?.uuid) return [];
    const cached = this._decompositionCache.get(solid.uuid);
    if (cached) return cached.hulls;
    const rawMesh = this._extractSolidMesh(solid);
    if (!rawMesh) {
      this._decompositionCache.set(solid.uuid, { hulls: [] });
      return [];
    }
    let hulls = [];
    try {
      const vhacd = await this._loadVhacd();
      if (vhacd?.computeConvexHulls) {
        hulls = vhacd.computeConvexHulls(rawMesh, VHACD_OPTIONS) || [];
      }
    } catch (error) {
      console.warn('[SimulationWorkbench] Failed to decompose mesh with vhacd-js:', error);
    }
    if (!Array.isArray(hulls) || hulls.length === 0) {
      hulls = [rawMesh];
    }
    this._decompositionCache.set(solid.uuid, { hulls });
    return hulls;
  }

  _extractSolidMesh(solid) {
    let mesh = null;
    try {
      mesh = solid.getMesh?.();
      return normalizeMeshArrays(mesh);
    } catch {
      return null;
    } finally {
      try { if (mesh && typeof mesh.delete === 'function') mesh.delete(); } catch {}
    }
  }

  _buildProxyGroups() {
    this._clearProxyGroups();
    for (const solid of this._listSceneSolids()) {
      const record = this._decompositionCache.get(solid.uuid);
      const hulls = Array.isArray(record?.hulls) ? record.hulls : [];
      const group = this._createProxyGroup(solid, hulls);
      if (!group) continue;
      this._proxyGroups.set(solid.uuid, group);
      this._syncProxyGroupTransform(solid);
      try { this.viewer?.render?.(); } catch {}
    }
  }

  _createProxyGroup(solid, hulls) {
    const parent = solid.parent || this.viewer?.scene || null;
    if (!parent?.add || !Array.isArray(hulls) || hulls.length === 0) return null;
    const group = new THREE.Group();
    group.name = `${SIM_PROXY_PREFIX}:${solid.name || solid.uuid || ''}`;
    group.userData.excludeFromFit = false;
    group.userData.simulationProxy = true;
    group.visible = false;
    const metadataColor = this.viewer?.partHistory?.metadataManager?.getMetadata?.(solid.name || '')?.color || null;
    hulls.forEach((hull, index) => {
      const geometry = new THREE.BufferGeometry();
      geometry.setAttribute('position', new THREE.Float32BufferAttribute(Array.from(hull.positions || []), 3));
      geometry.setIndex(Array.from(hull.indices || []));
      geometry.computeVertexNormals();
      const mesh = new THREE.Mesh(geometry, buildProxyMaterial(metadataColor, index, hulls.length));
      mesh.userData.excludeFromFit = false;
      mesh.userData.simulationProxy = true;
      mesh.userData.sourceSolidUuid = solid.uuid;
      group.add(mesh);

      const edges = new THREE.LineSegments(
        new THREE.EdgesGeometry(geometry, 20),
        new THREE.LineBasicMaterial({ color: 0x22d3ee, transparent: true, opacity: 0.9 }),
      );
      edges.userData.excludeFromFit = false;
      edges.userData.simulationProxy = true;
      group.add(edges);
    });
    parent.add(group);
    return group;
  }

  _syncProxyGroupTransform(solid) {
    const group = this._proxyGroups.get(solid.uuid);
    if (!group) return;
    group.position.copy(solid.position);
    group.quaternion.copy(solid.quaternion);
    group.scale.copy(solid.scale);
    group.visible = false;
    group.updateMatrixWorld?.(true);
  }

  _clearProxyGroups() {
    for (const group of this._proxyGroups.values()) {
      try { group.parent?.remove?.(group); } catch {}
      disposeHierarchy(group);
    }
    this._proxyGroups.clear();
  }

  _setSourceSolidsVisible(visible) {
    for (const solid of this._listSceneSolids()) {
      if (visible) {
        solid.visible = this._sourceVisibility.get(solid.uuid) !== false;
      } else {
        solid.visible = false;
      }
    }
    for (const group of this._proxyGroups.values()) {
      group.visible = false;
    }
    try { this.viewer?.render?.(); } catch {}
  }

  async _rebuildPhysicsWorld() {
    if (!this._active) return;
    const [rapier] = await Promise.all([
      this._loadRapier(),
      Promise.all(this._listSceneSolids().map((solid) => this._ensureDecomposition(solid))),
    ]);
    if (!this._active || !rapier) return;
    this._destroyPhysicsWorld();
    this._bodyState.clear();
    const world = new rapier.World({ x: 0, y: -9.81, z: 0 });
    world.maxCcdSubsteps = 2;
    this._physicsWorld = world;
    const selectedSolid = this._transformSession?.solid || null;
    for (const solid of this._listSceneSolids()) {
      const bodyDesc = (selectedSolid && solid === selectedSolid)
        ? rapier.RigidBodyDesc.kinematicPositionBased()
        : (this._isSolidKinematicMotionDriven(solid)
          ? rapier.RigidBodyDesc.kinematicPositionBased()
          : (this.getSolidFixed(solid) ? rapier.RigidBodyDesc.fixed() : rapier.RigidBodyDesc.dynamic()));
      const position = solid.getWorldPosition(new THREE.Vector3());
      const quaternion = solid.getWorldQuaternion(new THREE.Quaternion());
      bodyDesc.setTranslation(position.x, position.y, position.z);
      bodyDesc.setRotation({ x: quaternion.x, y: quaternion.y, z: quaternion.z, w: quaternion.w });
      bodyDesc.setLinearDamping(2.5);
      bodyDesc.setAngularDamping(2.5);
      bodyDesc.setCcdEnabled(true);
      const body = world.createRigidBody(bodyDesc);
      const hulls = this._decompositionCache.get(solid.uuid)?.hulls || [];
      for (const hull of hulls) {
        const collider = this._createColliderForHull(rapier, hull);
        if (!collider) continue;
        collider.setRestitution(0.05);
        collider.setFriction(0.9);
        world.createCollider(collider, body);
      }
      this._bodyState.set(solid.uuid, {
        solid,
        body,
        proxyGroup: this._proxyGroups.get(solid.uuid) || null,
      });
    }
  }

  _destroyPhysicsWorld() {
    try { this._physicsWorld?.free?.(); } catch {}
    this._physicsWorld = null;
    this._bodyState.clear();
  }

  _createColliderForHull(rapier, hull) {
    const positions = hull?.positions ? new Float32Array(hull.positions) : null;
    const indices = hull?.indices ? new Uint32Array(hull.indices) : null;
    if (!positions || positions.length < 9 || !indices || indices.length < 3) return null;
    const collider = rapier.ColliderDesc.convexMesh(positions, indices);
    if (collider) return collider;
    return rapier.ColliderDesc.convexHull(positions);
  }

  _ensurePhysicsLoop() {
    if (this._raf) return;
    this._lastStepTime = 0;
    const step = (ts) => {
      this._raf = 0;
      if (!this._active) return;
      this._stepPhysics(ts);
      this._raf = requestAnimationFrame(step);
    };
    this._raf = requestAnimationFrame(step);
  }

  _stopPhysicsLoop() {
    if (this._raf) {
      cancelAnimationFrame(this._raf);
      this._raf = 0;
    }
    this._lastStepTime = 0;
  }

  _stepPhysics(timestamp) {
    const world = this._physicsWorld;
    if (!world) return;
    if (!this._isPlaying) {
      this._lastStepTime = timestamp;
      return;
    }
    const dt = this._lastStepTime > 0
      ? Math.min(1 / 20, Math.max(1 / 240, (timestamp - this._lastStepTime) / 1000))
      : 1 / 60;
    this._lastStepTime = timestamp;
    world.timestep = dt;

    const selectedSolid = this._transformSession?.solid || null;
    let changed = false;
    if (this._isPlaying) {
      changed = this._applyMotionStep(dt) || changed;
    }
    if (selectedSolid) {
      const bodyState = this._bodyState.get(selectedSolid.uuid);
      if (bodyState?.body) {
        const worldPosition = selectedSolid.getWorldPosition(new THREE.Vector3());
        const worldQuaternion = selectedSolid.getWorldQuaternion(new THREE.Quaternion());
        bodyState.body.setNextKinematicTranslation(worldPosition);
        bodyState.body.setNextKinematicRotation(worldQuaternion);
      }
    }
    world.step();
    for (const bodyState of this._bodyState.values()) {
      const { solid, body } = bodyState;
      if (!solid || !body) continue;
      if (selectedSolid && solid === selectedSolid) {
        this._syncProxyGroupTransform(solid);
        continue;
      }
      if (this.getSolidFixed(solid)) {
        this._syncProxyGroupTransform(solid);
        continue;
      }
      const translation = body.translation();
      const rotation = body.rotation();
      if (!translation || !rotation) continue;
      this._setSolidWorldPose(solid, translation, rotation);
      this._updateRuntimeTransformFromSolid(solid, { persist: false });
      this._movedSolidIds.add(solid.uuid);
      changed = true;
    }
    if (changed) {
      try { this.viewer?.render?.(); } catch {}
    }
  }

  _setSolidWorldPose(solid, translation, rotation) {
    const parent = solid.parent || this.viewer?.scene || null;
    const scratch = this._scratch;
    scratch.worldPosition.set(translation.x, translation.y, translation.z);
    scratch.worldQuaternion.set(rotation.x, rotation.y, rotation.z, rotation.w);
    if (parent?.isObject3D) {
      parent.updateMatrixWorld?.(true);
      scratch.parentInverse.copy(parent.matrixWorld).invert();
      parent.getWorldQuaternion(scratch.parentQuaternion);
      scratch.parentQuaternionInverse.copy(scratch.parentQuaternion).invert();
      scratch.localPosition.copy(scratch.worldPosition).applyMatrix4(scratch.parentInverse);
      solid.position.copy(scratch.localPosition);
      solid.quaternion.copy(scratch.parentQuaternionInverse.multiply(scratch.worldQuaternion));
    } else {
      solid.position.copy(scratch.worldPosition);
      solid.quaternion.copy(scratch.worldQuaternion);
    }
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }

  _startTransformSession(solid) {
    if (!solid || !this._active) return;
    this._stopTransformSession();
    const controls = new CombinedTransformControls(this.viewer.camera, this.viewer.renderer?.domElement);
    try { controls.setMode('translate'); } catch { controls.mode = 'translate'; }
    const target = new THREE.Object3D();
    target.name = `SimulationTransformTarget:${solid.name || solid.uuid || ''}`;
    this.viewer.scene.updateMatrixWorld?.(true);
    solid.updateMatrixWorld?.(true);
    const box = this._scratch.box;
    const center = box.setFromObject(this._proxyGroups.get(solid.uuid) || solid).isEmpty()
      ? solid.getWorldPosition(this._scratch.center)
      : box.getCenter(this._scratch.center);
    const worldPosition = solid.getWorldPosition(new THREE.Vector3());
    const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
    const offsetLocal = worldPosition.clone().sub(center).applyQuaternion(worldQuaternion.clone().invert());
    target.position.copy(center);
    target.quaternion.copy(worldQuaternion);
    this.viewer.scene.add(target);
    controls.attach(target);
    this.viewer.scene.add(controls);

    const applyToSolid = (persist = false) => {
      const targetWorldPosition = target.getWorldPosition(new THREE.Vector3());
      const targetWorldQuaternion = target.getWorldQuaternion(new THREE.Quaternion());
      const solidWorldPosition = targetWorldPosition.clone().add(offsetLocal.clone().applyQuaternion(targetWorldQuaternion));
      this._setSolidWorldPose(solid, solidWorldPosition, targetWorldQuaternion);
      this._updateRuntimeTransformFromSolid(solid, { persist });
      if (persist) {
        if (this._physicsWorld) {
          void this._rebuildPhysicsWorld();
        }
      }
      try { this.viewer?.render?.(); } catch {}
    };

    const changeHandler = () => applyToSolid(false);
    const dragHandler = (event) => {
      const dragging = !!event?.value;
      try { if (this.viewer?.controls) this.viewer.controls.enabled = !dragging; } catch {}
      if (!dragging) applyToSolid(true);
    };
    const objectChangeHandler = () => {
      if (!controls.dragging) applyToSolid(true);
    };
    const updateForCamera = () => {
      try { controls.update?.(); } catch {}
    };
    const cameraChangeHandler = () => updateForCamera();
    controls.addEventListener('change', changeHandler);
    controls.addEventListener('dragging-changed', dragHandler);
    controls.addEventListener('objectChange', objectChangeHandler);
    try { this.viewer?.controls?.addEventListener?.('change', cameraChangeHandler); } catch {}

    const globalState = { controls, viewer: this.viewer, target, updateForCamera };
    try { window.__BREP_activeXform = globalState; } catch {}
    this._transformSession = {
      solid,
      controls,
      target,
      changeHandler,
      dragHandler,
      objectChangeHandler,
      cameraChangeHandler,
      globalState,
      mode: 'translate',
    };
    if (this._physicsWorld) {
      void this._rebuildPhysicsWorld();
    }
    updateForCamera();
    applyToSolid(false);
  }

  _stopTransformSession() {
    const session = this._transformSession;
    if (!session) return;
    try { session.controls?.removeEventListener('change', session.changeHandler); } catch {}
    try { session.controls?.removeEventListener('dragging-changed', session.dragHandler); } catch {}
    try { session.controls?.removeEventListener('objectChange', session.objectChangeHandler); } catch {}
    try { this.viewer?.controls?.removeEventListener?.('change', session.cameraChangeHandler); } catch {}
    try { session.controls?.detach?.(); } catch {}
    try { this.viewer?.scene?.remove?.(session.controls); } catch {}
    try { this.viewer?.scene?.remove?.(session.target); } catch {}
    try { session.controls?.dispose?.(); } catch {}
    try {
      if (window.__BREP_activeXform === session.globalState) {
        window.__BREP_activeXform = null;
      }
    } catch {}
    this._transformSession = null;
    try { if (this.viewer?.controls) this.viewer.controls.enabled = true; } catch {}
    if (this._physicsWorld) {
      void this._rebuildPhysicsWorld();
    }
    try { this.viewer?.render?.(); } catch {}
  }

  _listSceneSolids() {
    const scene = this.viewer?.partHistory?.scene || this.viewer?.scene || null;
    const solids = [];
    if (!scene?.traverse) return solids;
    scene.traverse((obj) => {
      if (obj?.type === 'SOLID') solids.push(obj);
    });
    return solids;
  }

  _bindStateManager() {
    const manager = this.viewer?.partHistory?.simulationStateManager;
    if (!manager?.addListener) return;
    this._removeStateManagerListener = manager.addListener(() => {
      this._reconcileMotionState();
      if (this._active && this._physicsWorld) {
        void this._rebuildPhysicsWorld();
      }
      this._emit();
    });
  }

  _emit() {
    if (!this._listeners || this._listeners.size === 0) return;
    const payload = {
      active: this._active,
      playing: this._isPlaying,
      manager: this,
    };
    for (const listener of Array.from(this._listeners)) {
      try { listener(payload); } catch {}
    }
  }

  _getMotionEntries() {
    const motions = this.viewer?.partHistory?.simulationStateManager?.getMotions?.();
    return Array.isArray(motions)
      ? motions.map((entry) => (entry?.inputParams && typeof entry.inputParams === 'object') ? entry.inputParams : entry).filter(Boolean)
      : [];
  }

  _resetMotionRuntime() {
    this._motionState.clear();
    for (const motion of this._getMotionEntries()) {
      this._motionState.set(motion.id, { progress: 0, completed: false });
    }
  }

  _reconcileMotionState() {
    const activeIds = new Set();
    for (const motion of this._getMotionEntries()) {
      activeIds.add(motion.id);
      if (!this._motionState.has(motion.id)) {
        this._motionState.set(motion.id, { progress: 0, completed: false });
      }
    }
    for (const id of Array.from(this._motionState.keys())) {
      if (!activeIds.has(id)) this._motionState.delete(id);
    }
  }

  _isSolidMotionDriven(solid) {
    if (!solid || !this._isPlaying) return false;
    const name = normalizeText(solid.name, '');
    if (!name) return false;
    return this._getMotionEntries().some((motion) => normalizeText(this._resolveReferenceName(motion?.solid), '') === name);
  }

  _isSolidKinematicMotionDriven(solid) {
    return !!solid && this.getSolidFixed(solid) && this._isSolidMotionDriven(solid);
  }

  _applyMotionStep(dt) {
    const selectedSolid = this._transformSession?.solid || null;
    const commands = new Map();
    for (const solid of this._listSceneSolids()) {
      if (!this._isSolidMotionDriven(solid)) continue;
      if (selectedSolid && solid === selectedSolid) continue;
      if (this._isSolidKinematicMotionDriven(solid)) continue;
      const body = this._bodyState.get(solid.uuid)?.body;
      if (!body) continue;
      body.setLinvel({ x: 0, y: 0, z: 0 }, true);
      body.setAngvel({ x: 0, y: 0, z: 0 }, true);
    }
    for (const motion of this._getMotionEntries()) {
      const solid = this._resolveMotionSolid(motion);
      if (!solid) continue;
      if (selectedSolid && solid === selectedSolid) continue;
      if (motion.type === 'linear') this._applyLinearMotionStep(solid, motion, dt, commands);
      else this._applyRotationMotionStep(solid, motion, dt, commands);
    }
    for (const [solidUuid, command] of commands.entries()) {
      const body = this._bodyState.get(solidUuid)?.body;
      if (!body) continue;
      body.setLinvel(command.linear, true);
      body.setAngvel(command.angular, true);
    }
    return commands.size > 0;
  }

  _resolveMotionSolid(motion) {
    const name = normalizeText(this._resolveReferenceName(motion?.solid), '');
    if (!name) return null;
    const object = this.viewer?.partHistory?.getObjectByName?.(name) || null;
    return object?.type === 'SOLID' ? object : null;
  }

  _resolveMotionAxis(motion) {
    const objectName = normalizeText(this._resolveReferenceName(motion?.axis), '');
    const object = objectName ? this.viewer?.partHistory?.getObjectByName?.(objectName) : null;
    if (!object) return null;
    const live = this._extractAxisPointsFromObject(object);
    if (!live?.start || !live?.end) return null;
    const startVec = new THREE.Vector3(Number(live.start[0]) || 0, Number(live.start[1]) || 0, Number(live.start[2]) || 0);
    const endVec = new THREE.Vector3(Number(live.end[0]) || 0, Number(live.end[1]) || 0, Number(live.end[2]) || 0);
    if (startVec.distanceToSquared(endVec) <= 1e-12) return null;
    return { start: startVec, end: endVec };
  }

  _resolveReferenceName(value) {
    if (Array.isArray(value)) {
      return this._resolveReferenceName(value[0] || null);
    }
    if (value && typeof value === 'object') {
      return normalizeText(value.name || value.objectName || value.label || '', '');
    }
    return normalizeText(value, '');
  }

  _extractAxisPointsFromObject(object) {
    if (!object) return null;
    try {
      if (typeof object.points === 'function') {
        const points = object.points(true);
        if (Array.isArray(points) && points.length >= 2) {
          const first = points[0];
          const last = points[points.length - 1];
          return {
            start: [Number(first?.x) || 0, Number(first?.y) || 0, Number(first?.z) || 0],
            end: [Number(last?.x) || 0, Number(last?.y) || 0, Number(last?.z) || 0],
          };
        }
      }
    } catch {}
    try {
      object.updateMatrixWorld?.(true);
      const attr = object.geometry?.getAttribute?.('position');
      if (!attr || attr.count < 2) return null;
      const first = new THREE.Vector3(attr.getX(0), attr.getY(0), attr.getZ(0)).applyMatrix4(object.matrixWorld);
      const lastIndex = attr.count - 1;
      const last = new THREE.Vector3(attr.getX(lastIndex), attr.getY(lastIndex), attr.getZ(lastIndex)).applyMatrix4(object.matrixWorld);
      return {
        start: [first.x, first.y, first.z],
        end: [last.x, last.y, last.z],
      };
    } catch {
      return null;
    }
  }

  _getMotionCommand(commands, solid) {
    if (!solid?.uuid) return null;
    let command = commands.get(solid.uuid);
    if (!command) {
      command = {
        linear: { x: 0, y: 0, z: 0 },
        angular: { x: 0, y: 0, z: 0 },
      };
      commands.set(solid.uuid, command);
    }
    return command;
  }

  _applyRotationMotionStep(solid, motion, dt, commands) {
    const state = this._motionState.get(motion.id) || { progress: 0, completed: false };
    if (state.completed) return false;
    const axis = this._resolveMotionAxis(motion);
    if (!axis) return false;
    const body = this._bodyState.get(solid.uuid)?.body || null;
    if (!body) return false;
    const speedDeg = Number(motion?.speed);
    if (!Number.isFinite(speedDeg) || Math.abs(speedDeg) <= 1e-9) return false;
    let deltaDeg = speedDeg * dt;
    const limitDeg = normalizeOptionalMotionLimit(motion?.angle);
    if (limitDeg != null) {
      const remaining = limitDeg - state.progress;
      if (Math.abs(remaining) <= 1e-9) {
        state.completed = true;
        this._motionState.set(motion.id, state);
        return false;
      }
      if (Math.sign(deltaDeg || 1) !== Math.sign(remaining || 1)) {
        deltaDeg = Math.sign(remaining || 1) * Math.abs(deltaDeg);
      }
      if (Math.abs(deltaDeg) > Math.abs(remaining)) deltaDeg = remaining;
    }
    if (Math.abs(deltaDeg) <= 1e-9) return false;
    if (this._isSolidKinematicMotionDriven(solid)) {
      this._rotateSolidAroundWorldAxis(solid, axis.start, axis.end, THREE.MathUtils.degToRad(deltaDeg));
      const worldPosition = solid.getWorldPosition(new THREE.Vector3());
      const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
      body.setNextKinematicTranslation(worldPosition);
      body.setNextKinematicRotation(worldQuaternion);
      this._updateRuntimeTransformFromSolid(solid, { persist: false });
      this._movedSolidIds.add(solid.uuid);
      state.progress += deltaDeg;
      if (limitDeg != null && Math.abs(limitDeg - state.progress) <= 1e-9) state.completed = true;
      this._motionState.set(motion.id, state);
      return true;
    }
    const omega = THREE.MathUtils.degToRad(deltaDeg) / Math.max(dt, 1e-9);
    const scratch = this._scratch;
    const command = this._getMotionCommand(commands, solid);
    const translation = body.translation();
    scratch.axis.copy(axis.end).sub(axis.start).normalize().multiplyScalar(omega);
    scratch.radius.set(
      Number(translation?.x) || 0,
      Number(translation?.y) || 0,
      Number(translation?.z) || 0,
    ).sub(axis.start);
    scratch.velocity.copy(scratch.axis).cross(scratch.radius);
    command.angular.x += scratch.axis.x;
    command.angular.y += scratch.axis.y;
    command.angular.z += scratch.axis.z;
    command.linear.x += scratch.velocity.x;
    command.linear.y += scratch.velocity.y;
    command.linear.z += scratch.velocity.z;
    state.progress += deltaDeg;
    if (limitDeg != null && Math.abs(limitDeg - state.progress) <= 1e-9) state.completed = true;
    this._motionState.set(motion.id, state);
    return true;
  }

  _applyLinearMotionStep(solid, motion, dt, commands) {
    const state = this._motionState.get(motion.id) || { progress: 0, completed: false };
    if (state.completed) return false;
    const axis = this._resolveMotionAxis(motion);
    if (!axis) return false;
    const body = this._bodyState.get(solid.uuid)?.body || null;
    if (!body) return false;
    const speed = Number(motion?.speed);
    if (!Number.isFinite(speed) || Math.abs(speed) <= 1e-9) return false;
    let delta = speed * dt;
    const limit = normalizeOptionalMotionLimit(motion?.distance);
    if (limit != null) {
      const remaining = limit - state.progress;
      if (Math.abs(remaining) <= 1e-9) {
        state.completed = true;
        this._motionState.set(motion.id, state);
        return false;
      }
      if (Math.sign(delta || 1) !== Math.sign(remaining || 1)) {
        delta = Math.sign(remaining || 1) * Math.abs(delta);
      }
      if (Math.abs(delta) > Math.abs(remaining)) delta = remaining;
    }
    if (Math.abs(delta) <= 1e-9) return false;
    if (this._isSolidKinematicMotionDriven(solid)) {
      this._translateSolidAlongWorldAxis(solid, axis.start, axis.end, delta);
      const worldPosition = solid.getWorldPosition(new THREE.Vector3());
      const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
      body.setNextKinematicTranslation(worldPosition);
      body.setNextKinematicRotation(worldQuaternion);
      this._updateRuntimeTransformFromSolid(solid, { persist: false });
      this._movedSolidIds.add(solid.uuid);
      state.progress += delta;
      if (limit != null && Math.abs(limit - state.progress) <= 1e-9) state.completed = true;
      this._motionState.set(motion.id, state);
      return true;
    }
    const velocityScale = delta / Math.max(dt, 1e-9);
    const scratch = this._scratch;
    const command = this._getMotionCommand(commands, solid);
    const translation = body.translation();
    scratch.axis.copy(axis.end).sub(axis.start).normalize();
    scratch.translation.copy(scratch.axis).multiplyScalar(delta);
    body.setTranslation({
      x: (Number(translation?.x) || 0) + scratch.translation.x,
      y: (Number(translation?.y) || 0) + scratch.translation.y,
      z: (Number(translation?.z) || 0) + scratch.translation.z,
    }, true);
    scratch.axis.multiplyScalar(velocityScale);
    command.linear.x += scratch.axis.x;
    command.linear.y += scratch.axis.y;
    command.linear.z += scratch.axis.z;
    body.setAngvel({ x: 0, y: 0, z: 0 }, true);
    state.progress += delta;
    if (limit != null && Math.abs(limit - state.progress) <= 1e-9) state.completed = true;
    this._motionState.set(motion.id, state);
    return true;
  }

  _rotateSolidAroundWorldAxis(solid, axisStart, axisEnd, angleRad) {
    const parent = solid.parent || this.viewer?.scene || null;
    const scratch = this._scratch;
    scratch.axis.copy(axisEnd).sub(axisStart).normalize();
    scratch.parentWorldMatrix.copy(parent?.matrixWorld || new THREE.Matrix4());
    scratch.parentInverse.copy(scratch.parentWorldMatrix).invert();
    scratch.worldMatrix.copy(solid.matrixWorld);
    scratch.deltaMatrix.makeTranslation(axisStart.x, axisStart.y, axisStart.z);
    scratch.deltaMatrix.multiply(new THREE.Matrix4().makeRotationAxis(scratch.axis, angleRad));
    scratch.deltaMatrix.multiply(new THREE.Matrix4().makeTranslation(-axisStart.x, -axisStart.y, -axisStart.z));
    scratch.worldMatrix.premultiply(scratch.deltaMatrix);
    scratch.localMatrix.copy(scratch.parentInverse).multiply(scratch.worldMatrix);
    scratch.localMatrix.decompose(solid.position, solid.quaternion, solid.scale);
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }

  _translateSolidAlongWorldAxis(solid, axisStart, axisEnd, distance) {
    const parent = solid.parent || this.viewer?.scene || null;
    const scratch = this._scratch;
    scratch.axis.copy(axisEnd).sub(axisStart).normalize();
    scratch.translation.copy(scratch.axis).multiplyScalar(distance);
    scratch.parentWorldMatrix.copy(parent?.matrixWorld || new THREE.Matrix4());
    scratch.parentInverse.copy(scratch.parentWorldMatrix).invert();
    scratch.worldMatrix.copy(solid.matrixWorld);
    scratch.deltaMatrix.makeTranslation(scratch.translation.x, scratch.translation.y, scratch.translation.z);
    scratch.worldMatrix.premultiply(scratch.deltaMatrix);
    scratch.localMatrix.copy(scratch.parentInverse).multiply(scratch.worldMatrix);
    scratch.localMatrix.decompose(solid.position, solid.quaternion, solid.scale);
    solid.updateMatrixWorld?.(true);
    this._syncProxyGroupTransform(solid);
  }
}
", import.meta.url).href, { SimulationWorkbenchManager: e } = await import(
|
|
304373
304373
|
/* @vite-ignore */
|
|
304374
304374
|
A
|
|
304375
304375
|
);
|
package/package.json
CHANGED
|
@@ -33,6 +33,12 @@ function normalizeText(value, fallback = '') {
|
|
|
33
33
|
return next || fallback;
|
|
34
34
|
}
|
|
35
35
|
|
|
36
|
+
function normalizeOptionalMotionLimit(value) {
|
|
37
|
+
const numeric = Number(value);
|
|
38
|
+
if (!Number.isFinite(numeric)) return null;
|
|
39
|
+
return Math.abs(numeric) <= 1e-9 ? null : numeric;
|
|
40
|
+
}
|
|
41
|
+
|
|
36
42
|
function normalizeTransform(value) {
|
|
37
43
|
const raw = value && typeof value === 'object' ? value : {};
|
|
38
44
|
return {
|
|
@@ -171,6 +177,9 @@ export class SimulationWorkbenchManager {
|
|
|
171
177
|
center: new THREE.Vector3(),
|
|
172
178
|
axis: new THREE.Vector3(),
|
|
173
179
|
translation: new THREE.Vector3(),
|
|
180
|
+
velocity: new THREE.Vector3(),
|
|
181
|
+
angularVelocity: new THREE.Vector3(),
|
|
182
|
+
radius: new THREE.Vector3(),
|
|
174
183
|
};
|
|
175
184
|
this._bindStateManager();
|
|
176
185
|
}
|
|
@@ -606,7 +615,7 @@ export class SimulationWorkbenchManager {
|
|
|
606
615
|
for (const solid of this._listSceneSolids()) {
|
|
607
616
|
const bodyDesc = (selectedSolid && solid === selectedSolid)
|
|
608
617
|
? rapier.RigidBodyDesc.kinematicPositionBased()
|
|
609
|
-
: (this.
|
|
618
|
+
: (this._isSolidKinematicMotionDriven(solid)
|
|
610
619
|
? rapier.RigidBodyDesc.kinematicPositionBased()
|
|
611
620
|
: (this.getSolidFixed(solid) ? rapier.RigidBodyDesc.fixed() : rapier.RigidBodyDesc.dynamic()));
|
|
612
621
|
const position = solid.getWorldPosition(new THREE.Vector3());
|
|
@@ -695,18 +704,6 @@ export class SimulationWorkbenchManager {
|
|
|
695
704
|
bodyState.body.setNextKinematicRotation(worldQuaternion);
|
|
696
705
|
}
|
|
697
706
|
}
|
|
698
|
-
if (this._isPlaying) {
|
|
699
|
-
for (const solid of this._listSceneSolids()) {
|
|
700
|
-
if (!this._isSolidMotionDriven(solid)) continue;
|
|
701
|
-
const bodyState = this._bodyState.get(solid.uuid);
|
|
702
|
-
if (!bodyState?.body) continue;
|
|
703
|
-
const worldPosition = solid.getWorldPosition(new THREE.Vector3());
|
|
704
|
-
const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
|
|
705
|
-
bodyState.body.setNextKinematicTranslation(worldPosition);
|
|
706
|
-
bodyState.body.setNextKinematicRotation(worldQuaternion);
|
|
707
|
-
}
|
|
708
|
-
}
|
|
709
|
-
|
|
710
707
|
world.step();
|
|
711
708
|
for (const bodyState of this._bodyState.values()) {
|
|
712
709
|
const { solid, body } = bodyState;
|
|
@@ -715,10 +712,6 @@ export class SimulationWorkbenchManager {
|
|
|
715
712
|
this._syncProxyGroupTransform(solid);
|
|
716
713
|
continue;
|
|
717
714
|
}
|
|
718
|
-
if (this._isSolidMotionDriven(solid)) {
|
|
719
|
-
this._syncProxyGroupTransform(solid);
|
|
720
|
-
continue;
|
|
721
|
-
}
|
|
722
715
|
if (this.getSolidFixed(solid)) {
|
|
723
716
|
this._syncProxyGroupTransform(solid);
|
|
724
717
|
continue;
|
|
@@ -923,15 +916,36 @@ export class SimulationWorkbenchManager {
|
|
|
923
916
|
return this._getMotionEntries().some((motion) => normalizeText(this._resolveReferenceName(motion?.solid), '') === name);
|
|
924
917
|
}
|
|
925
918
|
|
|
919
|
+
_isSolidKinematicMotionDriven(solid) {
|
|
920
|
+
return !!solid && this.getSolidFixed(solid) && this._isSolidMotionDriven(solid);
|
|
921
|
+
}
|
|
922
|
+
|
|
926
923
|
_applyMotionStep(dt) {
|
|
927
|
-
|
|
924
|
+
const selectedSolid = this._transformSession?.solid || null;
|
|
925
|
+
const commands = new Map();
|
|
926
|
+
for (const solid of this._listSceneSolids()) {
|
|
927
|
+
if (!this._isSolidMotionDriven(solid)) continue;
|
|
928
|
+
if (selectedSolid && solid === selectedSolid) continue;
|
|
929
|
+
if (this._isSolidKinematicMotionDriven(solid)) continue;
|
|
930
|
+
const body = this._bodyState.get(solid.uuid)?.body;
|
|
931
|
+
if (!body) continue;
|
|
932
|
+
body.setLinvel({ x: 0, y: 0, z: 0 }, true);
|
|
933
|
+
body.setAngvel({ x: 0, y: 0, z: 0 }, true);
|
|
934
|
+
}
|
|
928
935
|
for (const motion of this._getMotionEntries()) {
|
|
929
936
|
const solid = this._resolveMotionSolid(motion);
|
|
930
937
|
if (!solid) continue;
|
|
931
|
-
if (
|
|
932
|
-
|
|
938
|
+
if (selectedSolid && solid === selectedSolid) continue;
|
|
939
|
+
if (motion.type === 'linear') this._applyLinearMotionStep(solid, motion, dt, commands);
|
|
940
|
+
else this._applyRotationMotionStep(solid, motion, dt, commands);
|
|
933
941
|
}
|
|
934
|
-
|
|
942
|
+
for (const [solidUuid, command] of commands.entries()) {
|
|
943
|
+
const body = this._bodyState.get(solidUuid)?.body;
|
|
944
|
+
if (!body) continue;
|
|
945
|
+
body.setLinvel(command.linear, true);
|
|
946
|
+
body.setAngvel(command.angular, true);
|
|
947
|
+
}
|
|
948
|
+
return commands.size > 0;
|
|
935
949
|
}
|
|
936
950
|
|
|
937
951
|
_resolveMotionSolid(motion) {
|
|
@@ -994,15 +1008,30 @@ export class SimulationWorkbenchManager {
|
|
|
994
1008
|
}
|
|
995
1009
|
}
|
|
996
1010
|
|
|
997
|
-
|
|
1011
|
+
_getMotionCommand(commands, solid) {
|
|
1012
|
+
if (!solid?.uuid) return null;
|
|
1013
|
+
let command = commands.get(solid.uuid);
|
|
1014
|
+
if (!command) {
|
|
1015
|
+
command = {
|
|
1016
|
+
linear: { x: 0, y: 0, z: 0 },
|
|
1017
|
+
angular: { x: 0, y: 0, z: 0 },
|
|
1018
|
+
};
|
|
1019
|
+
commands.set(solid.uuid, command);
|
|
1020
|
+
}
|
|
1021
|
+
return command;
|
|
1022
|
+
}
|
|
1023
|
+
|
|
1024
|
+
_applyRotationMotionStep(solid, motion, dt, commands) {
|
|
998
1025
|
const state = this._motionState.get(motion.id) || { progress: 0, completed: false };
|
|
999
|
-
if (state.completed) return;
|
|
1026
|
+
if (state.completed) return false;
|
|
1000
1027
|
const axis = this._resolveMotionAxis(motion);
|
|
1001
|
-
if (!axis) return;
|
|
1028
|
+
if (!axis) return false;
|
|
1029
|
+
const body = this._bodyState.get(solid.uuid)?.body || null;
|
|
1030
|
+
if (!body) return false;
|
|
1002
1031
|
const speedDeg = Number(motion?.speed);
|
|
1003
|
-
if (!Number.isFinite(speedDeg) || Math.abs(speedDeg) <= 1e-9) return;
|
|
1032
|
+
if (!Number.isFinite(speedDeg) || Math.abs(speedDeg) <= 1e-9) return false;
|
|
1004
1033
|
let deltaDeg = speedDeg * dt;
|
|
1005
|
-
const limitDeg =
|
|
1034
|
+
const limitDeg = normalizeOptionalMotionLimit(motion?.angle);
|
|
1006
1035
|
if (limitDeg != null) {
|
|
1007
1036
|
const remaining = limitDeg - state.progress;
|
|
1008
1037
|
if (Math.abs(remaining) <= 1e-9) {
|
|
@@ -1016,24 +1045,53 @@ export class SimulationWorkbenchManager {
|
|
|
1016
1045
|
if (Math.abs(deltaDeg) > Math.abs(remaining)) deltaDeg = remaining;
|
|
1017
1046
|
}
|
|
1018
1047
|
if (Math.abs(deltaDeg) <= 1e-9) return false;
|
|
1019
|
-
this.
|
|
1048
|
+
if (this._isSolidKinematicMotionDriven(solid)) {
|
|
1049
|
+
this._rotateSolidAroundWorldAxis(solid, axis.start, axis.end, THREE.MathUtils.degToRad(deltaDeg));
|
|
1050
|
+
const worldPosition = solid.getWorldPosition(new THREE.Vector3());
|
|
1051
|
+
const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
|
|
1052
|
+
body.setNextKinematicTranslation(worldPosition);
|
|
1053
|
+
body.setNextKinematicRotation(worldQuaternion);
|
|
1054
|
+
this._updateRuntimeTransformFromSolid(solid, { persist: false });
|
|
1055
|
+
this._movedSolidIds.add(solid.uuid);
|
|
1056
|
+
state.progress += deltaDeg;
|
|
1057
|
+
if (limitDeg != null && Math.abs(limitDeg - state.progress) <= 1e-9) state.completed = true;
|
|
1058
|
+
this._motionState.set(motion.id, state);
|
|
1059
|
+
return true;
|
|
1060
|
+
}
|
|
1061
|
+
const omega = THREE.MathUtils.degToRad(deltaDeg) / Math.max(dt, 1e-9);
|
|
1062
|
+
const scratch = this._scratch;
|
|
1063
|
+
const command = this._getMotionCommand(commands, solid);
|
|
1064
|
+
const translation = body.translation();
|
|
1065
|
+
scratch.axis.copy(axis.end).sub(axis.start).normalize().multiplyScalar(omega);
|
|
1066
|
+
scratch.radius.set(
|
|
1067
|
+
Number(translation?.x) || 0,
|
|
1068
|
+
Number(translation?.y) || 0,
|
|
1069
|
+
Number(translation?.z) || 0,
|
|
1070
|
+
).sub(axis.start);
|
|
1071
|
+
scratch.velocity.copy(scratch.axis).cross(scratch.radius);
|
|
1072
|
+
command.angular.x += scratch.axis.x;
|
|
1073
|
+
command.angular.y += scratch.axis.y;
|
|
1074
|
+
command.angular.z += scratch.axis.z;
|
|
1075
|
+
command.linear.x += scratch.velocity.x;
|
|
1076
|
+
command.linear.y += scratch.velocity.y;
|
|
1077
|
+
command.linear.z += scratch.velocity.z;
|
|
1020
1078
|
state.progress += deltaDeg;
|
|
1021
1079
|
if (limitDeg != null && Math.abs(limitDeg - state.progress) <= 1e-9) state.completed = true;
|
|
1022
1080
|
this._motionState.set(motion.id, state);
|
|
1023
|
-
this._updateRuntimeTransformFromSolid(solid, { persist: false });
|
|
1024
|
-
this._movedSolidIds.add(solid.uuid);
|
|
1025
1081
|
return true;
|
|
1026
1082
|
}
|
|
1027
1083
|
|
|
1028
|
-
_applyLinearMotionStep(solid, motion, dt) {
|
|
1084
|
+
_applyLinearMotionStep(solid, motion, dt, commands) {
|
|
1029
1085
|
const state = this._motionState.get(motion.id) || { progress: 0, completed: false };
|
|
1030
|
-
if (state.completed) return;
|
|
1086
|
+
if (state.completed) return false;
|
|
1031
1087
|
const axis = this._resolveMotionAxis(motion);
|
|
1032
|
-
if (!axis) return;
|
|
1088
|
+
if (!axis) return false;
|
|
1089
|
+
const body = this._bodyState.get(solid.uuid)?.body || null;
|
|
1090
|
+
if (!body) return false;
|
|
1033
1091
|
const speed = Number(motion?.speed);
|
|
1034
|
-
if (!Number.isFinite(speed) || Math.abs(speed) <= 1e-9) return;
|
|
1092
|
+
if (!Number.isFinite(speed) || Math.abs(speed) <= 1e-9) return false;
|
|
1035
1093
|
let delta = speed * dt;
|
|
1036
|
-
const limit =
|
|
1094
|
+
const limit = normalizeOptionalMotionLimit(motion?.distance);
|
|
1037
1095
|
if (limit != null) {
|
|
1038
1096
|
const remaining = limit - state.progress;
|
|
1039
1097
|
if (Math.abs(remaining) <= 1e-9) {
|
|
@@ -1047,12 +1105,38 @@ export class SimulationWorkbenchManager {
|
|
|
1047
1105
|
if (Math.abs(delta) > Math.abs(remaining)) delta = remaining;
|
|
1048
1106
|
}
|
|
1049
1107
|
if (Math.abs(delta) <= 1e-9) return false;
|
|
1050
|
-
this.
|
|
1108
|
+
if (this._isSolidKinematicMotionDriven(solid)) {
|
|
1109
|
+
this._translateSolidAlongWorldAxis(solid, axis.start, axis.end, delta);
|
|
1110
|
+
const worldPosition = solid.getWorldPosition(new THREE.Vector3());
|
|
1111
|
+
const worldQuaternion = solid.getWorldQuaternion(new THREE.Quaternion());
|
|
1112
|
+
body.setNextKinematicTranslation(worldPosition);
|
|
1113
|
+
body.setNextKinematicRotation(worldQuaternion);
|
|
1114
|
+
this._updateRuntimeTransformFromSolid(solid, { persist: false });
|
|
1115
|
+
this._movedSolidIds.add(solid.uuid);
|
|
1116
|
+
state.progress += delta;
|
|
1117
|
+
if (limit != null && Math.abs(limit - state.progress) <= 1e-9) state.completed = true;
|
|
1118
|
+
this._motionState.set(motion.id, state);
|
|
1119
|
+
return true;
|
|
1120
|
+
}
|
|
1121
|
+
const velocityScale = delta / Math.max(dt, 1e-9);
|
|
1122
|
+
const scratch = this._scratch;
|
|
1123
|
+
const command = this._getMotionCommand(commands, solid);
|
|
1124
|
+
const translation = body.translation();
|
|
1125
|
+
scratch.axis.copy(axis.end).sub(axis.start).normalize();
|
|
1126
|
+
scratch.translation.copy(scratch.axis).multiplyScalar(delta);
|
|
1127
|
+
body.setTranslation({
|
|
1128
|
+
x: (Number(translation?.x) || 0) + scratch.translation.x,
|
|
1129
|
+
y: (Number(translation?.y) || 0) + scratch.translation.y,
|
|
1130
|
+
z: (Number(translation?.z) || 0) + scratch.translation.z,
|
|
1131
|
+
}, true);
|
|
1132
|
+
scratch.axis.multiplyScalar(velocityScale);
|
|
1133
|
+
command.linear.x += scratch.axis.x;
|
|
1134
|
+
command.linear.y += scratch.axis.y;
|
|
1135
|
+
command.linear.z += scratch.axis.z;
|
|
1136
|
+
body.setAngvel({ x: 0, y: 0, z: 0 }, true);
|
|
1051
1137
|
state.progress += delta;
|
|
1052
1138
|
if (limit != null && Math.abs(limit - state.progress) <= 1e-9) state.completed = true;
|
|
1053
1139
|
this._motionState.set(motion.id, state);
|
|
1054
|
-
this._updateRuntimeTransformFromSolid(solid, { persist: false });
|
|
1055
|
-
this._movedSolidIds.add(solid.uuid);
|
|
1056
1140
|
return true;
|
|
1057
1141
|
}
|
|
1058
1142
|
|