ModelViewer.tsx 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899
  1. import { useEffect, useRef, useState } from 'react';
  2. import { useTranslation } from 'react-i18next';
  3. import * as THREE from 'three';
  4. import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls.js';
  5. import { mergeGeometries } from 'three/examples/jsm/utils/BufferGeometryUtils.js';
  6. import { STLLoader } from 'three/examples/jsm/loaders/STLLoader.js';
  7. import JSZip from 'jszip';
  8. import { Loader2, RotateCcw, ZoomIn, ZoomOut } from 'lucide-react';
  9. import { Button } from './Button';
  10. import { getAuthToken } from '../api/client';
  11. interface BuildVolume {
  12. x: number;
  13. y: number;
  14. z: number;
  15. }
  16. interface ModelViewerProps {
  17. url: string;
  18. fileType?: string;
  19. buildVolume?: BuildVolume;
  20. filamentColors?: string[];
  21. selectedPlateId?: number | null;
  22. className?: string;
  23. }
  24. interface MeshData {
  25. vertices: number[];
  26. triangles: number[];
  27. extruder: number; // Per-mesh extruder index for coloring
  28. }
  29. interface ObjectData {
  30. id: string;
  31. meshes: MeshData[];
  32. defaultExtruder: number; // Default extruder for object (used if mesh doesn't have specific one)
  33. plateId?: number | null;
  34. }
  35. interface BuildItem {
  36. objectId: string;
  37. transform: THREE.Matrix4;
  38. extruder?: number; // Can override object's extruder
  39. plateId?: number | null;
  40. }
  41. interface Parsed3MFData {
  42. objects: Map<string, ObjectData>;
  43. buildItems: BuildItem[];
  44. plateBounds: Map<number, { minX: number; minY: number; maxX: number; maxY: number }>;
  45. plateOffsets: Map<number, { offsetX: number; offsetY: number }>;
  46. }
  47. // Parse 3MF transform - keep in 3MF coordinate space (Z-up)
  48. function parseTransform3MF(transformStr: string | null): THREE.Matrix4 {
  49. const matrix = new THREE.Matrix4();
  50. if (!transformStr) {
  51. return matrix; // Identity matrix
  52. }
  53. // 3MF transform is a 3x4 affine matrix in row-major order:
  54. // "m00 m01 m02 m10 m11 m12 m20 m21 m22 m30 m31 m32"
  55. // Where (m30, m31, m32) is the translation vector
  56. const values = transformStr.trim().split(/\s+/).map(parseFloat);
  57. if (values.length >= 12) {
  58. // Three.js Matrix4.set takes row-major order arguments:
  59. // set(n11, n12, n13, n14, n21, n22, n23, n24, n31, n32, n33, n34, n41, n42, n43, n44)
  60. // 3MF row-major: m00, m01, m02, m10, m11, m12, m20, m21, m22, m30, m31, m32
  61. matrix.set(
  62. values[0], values[1], values[2], values[9], // m00, m01, m02, tx
  63. values[3], values[4], values[5], values[10], // m10, m11, m12, ty
  64. values[6], values[7], values[8], values[11], // m20, m21, m22, tz
  65. 0, 0, 0, 1
  66. );
  67. }
  68. return matrix;
  69. }
  70. // Alias for backwards compatibility
  71. const parseTransform = parseTransform3MF;
  72. async function parseMeshFromDoc(doc: Document, defaultExtruder: number = 0): Promise<MeshData[]> {
  73. const meshes: MeshData[] = [];
  74. const meshElements = doc.getElementsByTagName('mesh');
  75. for (let j = 0; j < meshElements.length; j++) {
  76. const meshEl = meshElements[j];
  77. const vertices: number[] = [];
  78. const triangles: number[] = [];
  79. const vertexElements = meshEl.getElementsByTagName('vertex');
  80. for (let k = 0; k < vertexElements.length; k++) {
  81. const v = vertexElements[k];
  82. vertices.push(
  83. parseFloat(v.getAttribute('x') || '0'),
  84. parseFloat(v.getAttribute('y') || '0'),
  85. parseFloat(v.getAttribute('z') || '0')
  86. );
  87. }
  88. const triangleElements = meshEl.getElementsByTagName('triangle');
  89. for (let k = 0; k < triangleElements.length; k++) {
  90. const t = triangleElements[k];
  91. triangles.push(
  92. parseInt(t.getAttribute('v1') || '0'),
  93. parseInt(t.getAttribute('v2') || '0'),
  94. parseInt(t.getAttribute('v3') || '0')
  95. );
  96. }
  97. if (vertices.length > 0 && triangles.length > 0) {
  98. meshes.push({ vertices, triangles, extruder: defaultExtruder });
  99. }
  100. }
  101. return meshes;
  102. }
  103. function parsePlateIdFromAttributes(element: Element): number | null {
  104. const plateAttribute = Array.from(element.attributes).find((attr) => {
  105. const name = attr.name.toLowerCase();
  106. return (
  107. name === 'plate_id' ||
  108. name === 'plater_id' ||
  109. name === 'plateid' ||
  110. name === 'platerid' ||
  111. name.endsWith(':plate_id') ||
  112. name.endsWith(':plater_id')
  113. );
  114. });
  115. if (!plateAttribute?.value) return null;
  116. const parsed = Number.parseInt(plateAttribute.value, 10);
  117. return Number.isFinite(parsed) ? parsed : null;
  118. }
  119. async function parse3MF(arrayBuffer: ArrayBuffer): Promise<Parsed3MFData> {
  120. let zip: JSZip;
  121. try {
  122. zip = await JSZip.loadAsync(arrayBuffer);
  123. } catch {
  124. throw new Error('Unsupported file format');
  125. }
  126. const objects = new Map<string, ObjectData>();
  127. const buildItems: BuildItem[] = [];
  128. const plateBounds = new Map<number, { minX: number; minY: number; maxX: number; maxY: number }>();
  129. const plateOffsets = new Map<number, { offsetX: number; offsetY: number }>();
  130. const parser = new DOMParser();
  131. // Helper to load and parse a model file from the zip
  132. async function loadModelFile(path: string): Promise<Document | null> {
  133. // Normalize path (remove leading slash)
  134. const normalizedPath = path.startsWith('/') ? path.slice(1) : path;
  135. const file = zip.files[normalizedPath];
  136. if (!file) return null;
  137. const content = await file.async('string');
  138. return parser.parseFromString(content, 'application/xml');
  139. }
  140. // Parse model_settings.config to get extruder assignments
  141. // Maps: object ID -> default extruder, and (object ID, part ID) -> part-specific extruder
  142. const extruderMapById = new Map<string, number>();
  143. const partExtruderMap = new Map<string, number>(); // Key: "objectId:partId"
  144. const objectNameById = new Map<string, string>();
  145. const plateAssignmentsByObjectId = new Map<string, number>();
  146. const modelSettingsFile = zip.files['Metadata/model_settings.config'];
  147. if (modelSettingsFile) {
  148. try {
  149. const content = await modelSettingsFile.async('string');
  150. const doc = parser.parseFromString(content, 'application/xml');
  151. const objectElements = doc.getElementsByTagName('object');
  152. for (let i = 0; i < objectElements.length; i++) {
  153. const objEl = objectElements[i];
  154. const objectId = objEl.getAttribute('id');
  155. if (!objectId) continue;
  156. // Find object-level extruder + name
  157. const directMetadata = Array.from(objEl.children).filter(
  158. (el) => el.tagName === 'metadata' && el.getAttribute('key') === 'extruder'
  159. );
  160. if (directMetadata.length > 0) {
  161. const extruderVal = directMetadata[0].getAttribute('value');
  162. if (extruderVal) {
  163. extruderMapById.set(objectId, Math.max(0, parseInt(extruderVal, 10) - 1));
  164. }
  165. }
  166. const nameMetadata = Array.from(objEl.children).find(
  167. (el) => el.tagName === 'metadata' && el.getAttribute('key') === 'name'
  168. );
  169. const objectName = nameMetadata?.getAttribute('value');
  170. if (objectName) {
  171. objectNameById.set(objectId, objectName);
  172. }
  173. // Find part-level extruders
  174. const partElements = objEl.getElementsByTagName('part');
  175. for (let j = 0; j < partElements.length; j++) {
  176. const partEl = partElements[j];
  177. const partId = partEl.getAttribute('id');
  178. if (!partId) continue;
  179. // Look for extruder in part's direct children
  180. const partMetadata = Array.from(partEl.children).filter(
  181. (el) => el.tagName === 'metadata' && el.getAttribute('key') === 'extruder'
  182. );
  183. if (partMetadata.length > 0) {
  184. const extruderVal = partMetadata[0].getAttribute('value');
  185. if (extruderVal) {
  186. partExtruderMap.set(`${objectId}:${partId}`, Math.max(0, parseInt(extruderVal, 10) - 1));
  187. }
  188. }
  189. }
  190. }
  191. // Parse plate -> object assignments
  192. const plateElements = doc.getElementsByTagName('plate');
  193. for (let i = 0; i < plateElements.length; i++) {
  194. const plateEl = plateElements[i];
  195. let plateId: number | null = null;
  196. const metadataElements = plateEl.getElementsByTagName('metadata');
  197. let plateOffsetX = 0;
  198. let plateOffsetY = 0;
  199. for (let j = 0; j < metadataElements.length; j++) {
  200. const metaEl = metadataElements[j];
  201. const key = metaEl.getAttribute('key');
  202. if (key === 'plater_id' || key === 'plate_id') {
  203. const value = metaEl.getAttribute('value');
  204. if (value) {
  205. const parsed = Number.parseInt(value, 10);
  206. if (Number.isFinite(parsed)) {
  207. plateId = parsed;
  208. }
  209. }
  210. } else if (key === 'pos_x') {
  211. const value = metaEl.getAttribute('value');
  212. const parsed = value ? Number.parseFloat(value) : Number.NaN;
  213. if (Number.isFinite(parsed)) {
  214. plateOffsetX = parsed;
  215. }
  216. } else if (key === 'pos_y') {
  217. const value = metaEl.getAttribute('value');
  218. const parsed = value ? Number.parseFloat(value) : Number.NaN;
  219. if (Number.isFinite(parsed)) {
  220. plateOffsetY = parsed;
  221. }
  222. }
  223. }
  224. if (plateId == null) continue;
  225. if (plateOffsetX !== 0 || plateOffsetY !== 0) {
  226. plateOffsets.set(plateId, { offsetX: plateOffsetX, offsetY: plateOffsetY });
  227. }
  228. const modelInstances = plateEl.getElementsByTagName('model_instance');
  229. for (let j = 0; j < modelInstances.length; j++) {
  230. const instanceEl = modelInstances[j];
  231. const instanceMetadata = instanceEl.getElementsByTagName('metadata');
  232. for (let k = 0; k < instanceMetadata.length; k++) {
  233. const metaEl = instanceMetadata[k];
  234. if (metaEl.getAttribute('key') === 'object_id') {
  235. const value = metaEl.getAttribute('value');
  236. if (value) {
  237. plateAssignmentsByObjectId.set(value, plateId);
  238. }
  239. }
  240. }
  241. }
  242. }
  243. } catch {
  244. // Silently ignore model_settings.config parsing errors
  245. }
  246. }
  247. // Parse plate_*.json for plate assignments by object name (source-only / unsliced files)
  248. const plateAssignmentsByName = new Map<string, number>();
  249. const plateJsonNames = Object.keys(zip.files).filter(
  250. (name) => name.startsWith('Metadata/plate_') && name.endsWith('.json')
  251. );
  252. for (const name of plateJsonNames) {
  253. const match = name.match(/^Metadata\/plate_(\d+)\.json$/);
  254. if (!match) continue;
  255. const plateIndex = Number.parseInt(match[1], 10);
  256. if (!Number.isFinite(plateIndex)) continue;
  257. try {
  258. const payload = await zip.files[name].async('string');
  259. const json = JSON.parse(payload) as { bbox_objects?: Array<{ name?: string }>; bbox_all?: number[] };
  260. const objectsList = json.bbox_objects ?? [];
  261. for (const entry of objectsList) {
  262. if (entry?.name) {
  263. plateAssignmentsByName.set(entry.name, plateIndex);
  264. }
  265. }
  266. if (Array.isArray(json.bbox_all) && json.bbox_all.length >= 4) {
  267. const [minX, minY, maxX, maxY] = json.bbox_all;
  268. if ([minX, minY, maxX, maxY].every((value) => Number.isFinite(value))) {
  269. plateBounds.set(plateIndex, { minX, minY, maxX, maxY });
  270. }
  271. }
  272. } catch {
  273. // Ignore plate json parsing errors
  274. }
  275. }
  276. // Find the main 3D model file
  277. const mainModelPath = Object.keys(zip.files).find(
  278. (name) => name === '3D/3dmodel.model' || name.endsWith('/3dmodel.model')
  279. );
  280. if (!mainModelPath) {
  281. // Fallback: try to find any .model file
  282. const anyModelPath = Object.keys(zip.files).find((name) => name.endsWith('.model'));
  283. if (anyModelPath) {
  284. const doc = await loadModelFile(anyModelPath);
  285. if (doc) {
  286. const meshes = await parseMeshFromDoc(doc, 0);
  287. if (meshes.length > 0) {
  288. objects.set('1', { id: '1', meshes, defaultExtruder: 0 });
  289. }
  290. }
  291. }
  292. return { objects, buildItems, plateBounds, plateOffsets };
  293. }
  294. const mainDoc = await loadModelFile(mainModelPath);
  295. if (!mainDoc) return { objects, buildItems, plateBounds, plateOffsets };
  296. // Parse objects - Bambu Studio uses components to reference external files
  297. const objectElements = mainDoc.getElementsByTagName('object');
  298. for (let i = 0; i < objectElements.length; i++) {
  299. const objEl = objectElements[i];
  300. const objectId = objEl.getAttribute('id');
  301. if (!objectId) continue;
  302. const objectPlateId = parsePlateIdFromAttributes(objEl) ?? plateAssignmentsByObjectId.get(objectId) ?? null;
  303. // Get default extruder from model_settings.config map, falling back to attribute or default
  304. let defaultExtruder = extruderMapById.get(objectId) ?? -1;
  305. if (defaultExtruder < 0) {
  306. const extruderAttr = objEl.getAttribute('p:extruder') || objEl.getAttributeNS('http://schemas.microsoft.com/3dmanufacturing/production/2015/06', 'extruder') || '1';
  307. defaultExtruder = Math.max(0, parseInt(extruderAttr, 10) - 1);
  308. }
  309. const meshes: MeshData[] = [];
  310. // Check for direct mesh in this object
  311. const objMeshElements = objEl.getElementsByTagName('mesh');
  312. for (let j = 0; j < objMeshElements.length; j++) {
  313. const meshEl = objMeshElements[j];
  314. const vertices: number[] = [];
  315. const triangles: number[] = [];
  316. const vertexElements = meshEl.getElementsByTagName('vertex');
  317. for (let k = 0; k < vertexElements.length; k++) {
  318. const v = vertexElements[k];
  319. vertices.push(
  320. parseFloat(v.getAttribute('x') || '0'),
  321. parseFloat(v.getAttribute('y') || '0'),
  322. parseFloat(v.getAttribute('z') || '0')
  323. );
  324. }
  325. const triangleElements = meshEl.getElementsByTagName('triangle');
  326. for (let k = 0; k < triangleElements.length; k++) {
  327. const t = triangleElements[k];
  328. triangles.push(
  329. parseInt(t.getAttribute('v1') || '0'),
  330. parseInt(t.getAttribute('v2') || '0'),
  331. parseInt(t.getAttribute('v3') || '0')
  332. );
  333. }
  334. if (vertices.length > 0 && triangles.length > 0) {
  335. meshes.push({ vertices, triangles, extruder: defaultExtruder });
  336. }
  337. }
  338. // Check for component references (Bambu Studio style)
  339. const componentElements = objEl.getElementsByTagName('component');
  340. for (let j = 0; j < componentElements.length; j++) {
  341. const compEl = componentElements[j];
  342. // p:path attribute contains the external file reference
  343. const extPath = compEl.getAttribute('p:path') || compEl.getAttributeNS('http://schemas.microsoft.com/3dmanufacturing/production/2015/06', 'path');
  344. // objectid in component corresponds to part id in model_settings
  345. const compObjectId = compEl.getAttribute('objectid');
  346. if (extPath) {
  347. const extDoc = await loadModelFile(extPath);
  348. if (extDoc) {
  349. // Look up per-part extruder, falling back to object's default
  350. const partKey = compObjectId ? `${objectId}:${compObjectId}` : null;
  351. const compExtruder = partKey ? (partExtruderMap.get(partKey) ?? defaultExtruder) : defaultExtruder;
  352. const extMeshes = await parseMeshFromDoc(extDoc, compExtruder);
  353. // Apply component transform if present
  354. const compTransformStr = compEl.getAttribute('transform');
  355. const compTransform = parseTransform(compTransformStr);
  356. for (const mesh of extMeshes) {
  357. if (compTransformStr) {
  358. // Apply transform to vertices (in 3MF coordinate space, before Y/Z swap)
  359. const transformedVertices: number[] = [];
  360. for (let k = 0; k < mesh.vertices.length; k += 3) {
  361. const v = new THREE.Vector3(mesh.vertices[k], mesh.vertices[k + 1], mesh.vertices[k + 2]);
  362. v.applyMatrix4(compTransform);
  363. transformedVertices.push(v.x, v.y, v.z);
  364. }
  365. meshes.push({ vertices: transformedVertices, triangles: mesh.triangles, extruder: mesh.extruder });
  366. } else {
  367. meshes.push(mesh);
  368. }
  369. }
  370. }
  371. }
  372. }
  373. if (meshes.length > 0) {
  374. objects.set(objectId, { id: objectId, meshes, defaultExtruder, plateId: objectPlateId });
  375. }
  376. }
  377. // Parse build items (placement on build plate)
  378. const buildElements = mainDoc.getElementsByTagName('build');
  379. if (buildElements.length > 0) {
  380. const itemElements = buildElements[0].getElementsByTagName('item');
  381. for (let i = 0; i < itemElements.length; i++) {
  382. const itemEl = itemElements[i];
  383. const objectId = itemEl.getAttribute('objectid');
  384. if (!objectId) continue;
  385. const transform = parseTransform(itemEl.getAttribute('transform'));
  386. const itemPlateId = parsePlateIdFromAttributes(itemEl);
  387. const objectPlateId = objects.get(objectId)?.plateId ?? null;
  388. const objectName = objectNameById.get(objectId);
  389. const namePlateId = objectName ? plateAssignmentsByName.get(objectName) ?? null : null;
  390. buildItems.push({ objectId, transform, plateId: itemPlateId ?? objectPlateId ?? namePlateId ?? null });
  391. }
  392. }
  393. return { objects, buildItems, plateBounds, plateOffsets };
  394. }
  395. function createGeometryFromMesh(mesh: MeshData): THREE.BufferGeometry {
  396. const geometry = new THREE.BufferGeometry();
  397. // Convert from 3MF Z-up to Three.js Y-up coordinate system
  398. // 3MF: X right, Y back, Z up -> Three.js: X right, Y up, Z forward
  399. const positions = new Float32Array(mesh.vertices.length);
  400. for (let i = 0; i < mesh.vertices.length; i += 3) {
  401. positions[i] = mesh.vertices[i]; // X stays X
  402. positions[i + 1] = mesh.vertices[i + 2]; // Y becomes Z (up)
  403. positions[i + 2] = mesh.vertices[i + 1]; // Z becomes Y
  404. }
  405. geometry.setAttribute('position', new THREE.BufferAttribute(positions, 3));
  406. geometry.setIndex(mesh.triangles);
  407. // Compute normals
  408. geometry.computeVertexNormals();
  409. return geometry;
  410. }
  411. function disposeGroup(group: THREE.Group) {
  412. group.traverse((child) => {
  413. if (child instanceof THREE.Mesh) {
  414. child.geometry.dispose();
  415. if (Array.isArray(child.material)) {
  416. for (const material of child.material) {
  417. material.dispose();
  418. }
  419. } else {
  420. child.material.dispose();
  421. }
  422. }
  423. });
  424. }
  425. function buildModelGroup(
  426. parsedData: Parsed3MFData,
  427. selectedPlateId: number | null,
  428. filamentColors?: string[],
  429. ): THREE.Group {
  430. const { objects, buildItems } = parsedData;
  431. const group = new THREE.Group();
  432. // Create materials for each extruder color
  433. const getMaterial = (extruder: number): THREE.MeshPhongMaterial => {
  434. const defaultColor = '#00ae42';
  435. const colorStr = filamentColors?.[extruder] || defaultColor;
  436. // Convert hex color string to THREE.js color
  437. const color = new THREE.Color(colorStr);
  438. return new THREE.MeshPhongMaterial({
  439. color,
  440. shininess: 30,
  441. flatShading: false,
  442. });
  443. };
  444. // Group geometries by extruder index (using per-mesh extruder)
  445. const geometriesByExtruder = new Map<number, THREE.BufferGeometry[]>();
  446. const hasPlateAssignments = buildItems.some((item) => item.plateId != null);
  447. const plateFilteredItems = selectedPlateId == null || !hasPlateAssignments
  448. ? buildItems
  449. : buildItems.filter((item) => item.plateId === selectedPlateId);
  450. const activeBuildItems = plateFilteredItems.length > 0 ? plateFilteredItems : buildItems;
  451. // If we have build items, use them for positioning
  452. if (activeBuildItems.length > 0) {
  453. for (const item of activeBuildItems) {
  454. const objectData = objects.get(item.objectId);
  455. if (!objectData) continue;
  456. for (const meshData of objectData.meshes) {
  457. // Use mesh's extruder, or item override, or object default
  458. const extruder = item.extruder ?? meshData.extruder;
  459. // Apply build transform to vertices in 3MF space BEFORE coordinate conversion
  460. const transformedVertices: number[] = [];
  461. for (let k = 0; k < meshData.vertices.length; k += 3) {
  462. const v = new THREE.Vector3(
  463. meshData.vertices[k],
  464. meshData.vertices[k + 1],
  465. meshData.vertices[k + 2]
  466. );
  467. v.applyMatrix4(item.transform);
  468. transformedVertices.push(v.x, v.y, v.z);
  469. }
  470. // Now create geometry with coordinate conversion
  471. const geometry = createGeometryFromMesh({
  472. vertices: transformedVertices,
  473. triangles: meshData.triangles,
  474. extruder: extruder,
  475. });
  476. if (!geometriesByExtruder.has(extruder)) {
  477. geometriesByExtruder.set(extruder, []);
  478. }
  479. geometriesByExtruder.get(extruder)!.push(geometry);
  480. }
  481. }
  482. } else {
  483. // Fallback: just add all objects without transforms
  484. for (const objectData of objects.values()) {
  485. for (const meshData of objectData.meshes) {
  486. // Use per-mesh extruder
  487. const extruder = meshData.extruder;
  488. const geometry = createGeometryFromMesh(meshData);
  489. if (!geometriesByExtruder.has(extruder)) {
  490. geometriesByExtruder.set(extruder, []);
  491. }
  492. geometriesByExtruder.get(extruder)!.push(geometry);
  493. }
  494. }
  495. }
  496. // Create meshes for each extruder group
  497. for (const [extruder, geometries] of geometriesByExtruder) {
  498. if (geometries.length === 0) continue;
  499. const mergedGeometry = geometries.length === 1
  500. ? geometries[0]
  501. : mergeGeometries(geometries, false);
  502. if (mergedGeometry) {
  503. const material = getMaterial(extruder);
  504. const mesh = new THREE.Mesh(mergedGeometry, material);
  505. group.add(mesh);
  506. }
  507. // Dispose individual geometries if merged
  508. if (geometries.length > 1) {
  509. for (const geom of geometries) {
  510. geom.dispose();
  511. }
  512. }
  513. }
  514. return group;
  515. }
  516. export function ModelViewer({
  517. url,
  518. fileType,
  519. buildVolume = { x: 256, y: 256, z: 256 },
  520. filamentColors,
  521. selectedPlateId = null,
  522. className = '',
  523. }: ModelViewerProps) {
  524. const { t } = useTranslation();
  525. const containerRef = useRef<HTMLDivElement>(null);
  526. const rendererRef = useRef<THREE.WebGLRenderer | null>(null);
  527. const sceneRef = useRef<THREE.Scene | null>(null);
  528. const cameraRef = useRef<THREE.PerspectiveCamera | null>(null);
  529. const controlsRef = useRef<OrbitControls | null>(null);
  530. const modelGroupRef = useRef<THREE.Group | null>(null);
  531. const plateRef = useRef<THREE.Mesh | null>(null);
  532. const gridRef = useRef<THREE.GridHelper | null>(null);
  533. const [loading, setLoading] = useState(true);
  534. const [error, setError] = useState<string | null>(null);
  535. const [parsedData, setParsedData] = useState<Parsed3MFData | null>(null);
  536. const [stlGeometry, setStlGeometry] = useState<THREE.BufferGeometry | null>(null);
  537. useEffect(() => {
  538. if (!containerRef.current) return;
  539. const container = containerRef.current;
  540. const width = container.clientWidth;
  541. const height = container.clientHeight;
  542. // Scene
  543. const scene = new THREE.Scene();
  544. scene.background = new THREE.Color(0x1a1a1a);
  545. sceneRef.current = scene;
  546. // Camera
  547. const camera = new THREE.PerspectiveCamera(45, width / height, 0.1, 10000);
  548. camera.position.set(150, 150, 150);
  549. cameraRef.current = camera;
  550. // Renderer
  551. const renderer = new THREE.WebGLRenderer({ antialias: true });
  552. renderer.setSize(width, height);
  553. renderer.setPixelRatio(window.devicePixelRatio);
  554. container.appendChild(renderer.domElement);
  555. rendererRef.current = renderer;
  556. // Controls
  557. const controls = new OrbitControls(camera, renderer.domElement);
  558. controls.enableDamping = true;
  559. controls.dampingFactor = 0.05;
  560. controlsRef.current = controls;
  561. // Lights
  562. const ambientLight = new THREE.AmbientLight(0xffffff, 0.6);
  563. scene.add(ambientLight);
  564. const directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);
  565. directionalLight.position.set(100, 100, 100);
  566. scene.add(directionalLight);
  567. const directionalLight2 = new THREE.DirectionalLight(0xffffff, 0.4);
  568. directionalLight2.position.set(-100, 50, -100);
  569. scene.add(directionalLight2);
  570. // Grid - use the larger dimension for the grid size
  571. const gridSize = Math.max(buildVolume.x, buildVolume.y);
  572. const gridDivisions = Math.ceil(gridSize / 16);
  573. const gridHelper = new THREE.GridHelper(gridSize, gridDivisions, 0x444444, 0x333333);
  574. scene.add(gridHelper);
  575. gridRef.current = gridHelper;
  576. // Build plate indicator
  577. const plateGeometry = new THREE.PlaneGeometry(buildVolume.x, buildVolume.y);
  578. const plateMaterial = new THREE.MeshBasicMaterial({
  579. color: 0x00ae42,
  580. transparent: true,
  581. opacity: 0.15,
  582. side: THREE.DoubleSide,
  583. });
  584. const plate = new THREE.Mesh(plateGeometry, plateMaterial);
  585. plate.rotation.x = -Math.PI / 2;
  586. plate.position.y = -0.5; // Slightly below Y=0 so models sit on top
  587. scene.add(plate);
  588. plateRef.current = plate;
  589. // Animation loop - keep it simple for reliability
  590. let animationId: number;
  591. const animate = () => {
  592. animationId = requestAnimationFrame(animate);
  593. controls.update();
  594. renderer.render(scene, camera);
  595. };
  596. animate();
  597. setLoading(true);
  598. setError(null);
  599. setParsedData(null);
  600. setStlGeometry(null);
  601. const normalizedType = (fileType || url.split('?')[0].split('.').pop() || '').toLowerCase();
  602. // Build auth headers for fetch
  603. const headers: HeadersInit = {};
  604. const token = getAuthToken();
  605. if (token) {
  606. headers['Authorization'] = `Bearer ${token}`;
  607. }
  608. if (normalizedType === 'stl') {
  609. fetch(url, { headers })
  610. .then((res) => {
  611. if (!res.ok) throw new Error(t('modelViewer.errors.failedToLoad'));
  612. return res.arrayBuffer();
  613. })
  614. .then((buffer) => {
  615. const loader = new STLLoader();
  616. const geometry = loader.parse(buffer);
  617. geometry.computeVertexNormals();
  618. geometry.rotateX(-Math.PI / 2);
  619. setStlGeometry(geometry);
  620. })
  621. .catch((err) => {
  622. setError(err.message);
  623. setLoading(false);
  624. });
  625. } else if (normalizedType === '3mf') {
  626. fetch(url, { headers })
  627. .then((res) => {
  628. if (!res.ok) throw new Error(t('modelViewer.errors.failedToLoad'));
  629. return res.arrayBuffer();
  630. })
  631. .then(parse3MF)
  632. .then((parsed) => {
  633. if (parsed.objects.size === 0) {
  634. throw new Error(t('modelViewer.errors.noMeshes'));
  635. }
  636. setParsedData(parsed);
  637. })
  638. .catch((err) => {
  639. setError(err.message);
  640. setLoading(false);
  641. });
  642. } else {
  643. setError(t('modelViewer.errors.unsupportedFormat'));
  644. setLoading(false);
  645. }
  646. // Handle resize (window + container)
  647. const handleResize = () => {
  648. if (!container) return;
  649. const w = container.clientWidth;
  650. const h = container.clientHeight;
  651. if (w === 0 || h === 0) return;
  652. camera.aspect = w / h;
  653. camera.updateProjectionMatrix();
  654. renderer.setSize(w, h);
  655. };
  656. window.addEventListener('resize', handleResize);
  657. const resizeObserver = new ResizeObserver(() => {
  658. handleResize();
  659. });
  660. resizeObserver.observe(container);
  661. return () => {
  662. window.removeEventListener('resize', handleResize);
  663. resizeObserver.disconnect();
  664. cancelAnimationFrame(animationId);
  665. controls.dispose();
  666. renderer.dispose();
  667. container.removeChild(renderer.domElement);
  668. modelGroupRef.current = null;
  669. plateRef.current = null;
  670. gridRef.current = null;
  671. };
  672. }, [url, buildVolume, fileType, t]);
  673. useEffect(() => {
  674. if (!sceneRef.current || !cameraRef.current || !controlsRef.current) return;
  675. if (!parsedData && !stlGeometry) return;
  676. if (modelGroupRef.current) {
  677. sceneRef.current.remove(modelGroupRef.current);
  678. disposeGroup(modelGroupRef.current);
  679. }
  680. const isStlModel = !!stlGeometry;
  681. const group = isStlModel
  682. ? (() => {
  683. const materialColor = filamentColors?.[0] || '#00ae42';
  684. const material = new THREE.MeshPhongMaterial({ color: new THREE.Color(materialColor), shininess: 30 });
  685. const mesh = new THREE.Mesh(stlGeometry!, material);
  686. const stlGroup = new THREE.Group();
  687. stlGroup.add(mesh);
  688. return stlGroup;
  689. })()
  690. : buildModelGroup(parsedData!, selectedPlateId ?? null, filamentColors);
  691. modelGroupRef.current = group;
  692. sceneRef.current.add(group);
  693. // Get bounding box to position model
  694. const box = new THREE.Box3().setFromObject(group);
  695. const center = box.getCenter(new THREE.Vector3());
  696. // Always place models on the build plate (Y=0)
  697. group.position.y = -box.min.y;
  698. const selectedPlateBounds = (!isStlModel && selectedPlateId != null && parsedData!.buildItems.length > 0)
  699. ? parsedData!.plateBounds.get(selectedPlateId)
  700. : undefined;
  701. const selectedPlateOffset = (!isStlModel && selectedPlateId != null)
  702. ? parsedData!.plateOffsets.get(selectedPlateId)
  703. : undefined;
  704. const shouldCenterOnPlate = isStlModel
  705. || parsedData!.buildItems.length === 0
  706. || (selectedPlateId != null && !selectedPlateBounds && !selectedPlateOffset);
  707. const centerOffsetX = shouldCenterOnPlate ? -center.x : 0;
  708. const centerOffsetZ = shouldCenterOnPlate ? -center.z : 0;
  709. let plateOffsetX = 0;
  710. let plateOffsetZ = 0;
  711. if (!isStlModel && selectedPlateId != null && parsedData!.buildItems.length > 0 && selectedPlateBounds) {
  712. const plateBox = new THREE.Box3().setFromObject(group);
  713. plateOffsetX = plateBox.min.x - selectedPlateBounds.minX;
  714. plateOffsetZ = plateBox.min.z - selectedPlateBounds.minY;
  715. }
  716. const plateCenterX = buildVolume.x / 2;
  717. const plateCenterZ = buildVolume.y / 2;
  718. if (!isStlModel && selectedPlateId != null && parsedData!.buildItems.length > 0 && selectedPlateBounds) {
  719. group.position.x = centerOffsetX - plateOffsetX;
  720. group.position.z = centerOffsetZ - plateOffsetZ;
  721. } else if (!isStlModel && selectedPlateId != null && selectedPlateOffset) {
  722. group.position.x = centerOffsetX + (plateCenterX - selectedPlateOffset.offsetX);
  723. group.position.z = centerOffsetZ + (plateCenterZ - selectedPlateOffset.offsetY);
  724. } else if (shouldCenterOnPlate) {
  725. group.position.x = centerOffsetX + plateCenterX;
  726. group.position.z = centerOffsetZ + plateCenterZ;
  727. } else {
  728. group.position.x = centerOffsetX;
  729. group.position.z = centerOffsetZ;
  730. }
  731. if (plateRef.current) {
  732. plateRef.current.position.x = plateCenterX;
  733. plateRef.current.position.z = plateCenterZ;
  734. }
  735. if (gridRef.current) {
  736. gridRef.current.position.x = plateCenterX;
  737. gridRef.current.position.z = plateCenterZ;
  738. }
  739. // Recalculate bounding box after positioning
  740. const finalBox = new THREE.Box3().setFromObject(group);
  741. const finalCenter = finalBox.getCenter(new THREE.Vector3());
  742. const finalSize = finalBox.getSize(new THREE.Vector3());
  743. // Adjust camera to fit model
  744. const maxDim = Math.max(finalSize.x, finalSize.y, finalSize.z);
  745. const cameraDistance = maxDim * 1.8;
  746. cameraRef.current.position.set(
  747. finalCenter.x + cameraDistance * 0.7,
  748. finalCenter.y + cameraDistance * 0.5,
  749. finalCenter.z + cameraDistance * 0.7
  750. );
  751. controlsRef.current.target.copy(finalCenter);
  752. controlsRef.current.update();
  753. setLoading(false);
  754. }, [parsedData, stlGeometry, selectedPlateId, filamentColors, buildVolume]);
  755. const resetView = () => {
  756. if (cameraRef.current && controlsRef.current) {
  757. cameraRef.current.position.set(150, 150, 150);
  758. controlsRef.current.target.set(0, 50, 0);
  759. controlsRef.current.update();
  760. }
  761. };
  762. const zoom = (factor: number) => {
  763. if (cameraRef.current) {
  764. cameraRef.current.position.multiplyScalar(factor);
  765. }
  766. };
  767. return (
  768. <div className={`relative ${className}`}>
  769. <div ref={containerRef} className="w-full h-full min-h-[400px]" />
  770. {loading && (
  771. <div className="absolute inset-0 flex items-center justify-center bg-bambu-dark/80">
  772. <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
  773. </div>
  774. )}
  775. {error && (
  776. <div className="absolute inset-0 flex items-center justify-center bg-bambu-dark/80">
  777. <p className="text-red-400">{error}</p>
  778. </div>
  779. )}
  780. {!loading && !error && (
  781. <div className="absolute bottom-4 right-4 flex gap-2">
  782. <Button variant="secondary" size="sm" onClick={() => zoom(0.8)}>
  783. <ZoomIn className="w-4 h-4" />
  784. </Button>
  785. <Button variant="secondary" size="sm" onClick={() => zoom(1.25)}>
  786. <ZoomOut className="w-4 h-4" />
  787. </Button>
  788. <Button variant="secondary" size="sm" onClick={resetView}>
  789. <RotateCcw className="w-4 h-4" />
  790. </Button>
  791. </div>
  792. )}
  793. </div>
  794. );
  795. }