selva-compute 1.1.0 → 1.1.2
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/README.md +71 -10
- package/dist/{base-DbB0Ggdq.d.cts → base-dtik4Dlu.d.cts} +20 -0
- package/dist/{base-DbB0Ggdq.d.ts → base-dtik4Dlu.d.ts} +20 -0
- package/dist/chunk-F7DPIDIA.cjs +2 -0
- package/dist/{chunk-G6W5VE7C.cjs.map → chunk-F7DPIDIA.cjs.map} +1 -1
- package/dist/chunk-FMFP7GIM.js +2 -0
- package/dist/chunk-FMFP7GIM.js.map +1 -0
- package/dist/chunk-MK3M76VN.js +3 -0
- package/dist/chunk-MK3M76VN.js.map +1 -0
- package/dist/chunk-PE2FMBXD.cjs +3 -0
- package/dist/chunk-PE2FMBXD.cjs.map +1 -0
- package/dist/{chunk-M2HPEWXH.cjs → chunk-QCHPQ22Y.cjs} +2 -2
- package/dist/{chunk-M2HPEWXH.cjs.map → chunk-QCHPQ22Y.cjs.map} +1 -1
- package/dist/{chunk-QFHFH5BS.js → chunk-WD3ENUAA.js} +2 -2
- package/dist/core.cjs +1 -1
- package/dist/core.d.cts +10 -10
- package/dist/core.d.ts +10 -10
- package/dist/core.js +1 -1
- package/dist/grasshopper.cjs +1 -1
- package/dist/grasshopper.d.cts +9 -6
- package/dist/grasshopper.d.ts +9 -6
- package/dist/grasshopper.js +1 -1
- package/dist/index.cjs +1 -1
- package/dist/index.d.cts +2 -2
- package/dist/index.d.ts +2 -2
- package/dist/index.js +1 -1
- package/dist/{schemas-CQEJ5EZy.d.ts → schemas-BE5ai7jG.d.cts} +17 -4
- package/dist/{schemas-CQEJ5EZy.d.cts → schemas-BE5ai7jG.d.ts} +17 -4
- package/dist/visualization.cjs +1 -1
- package/dist/visualization.cjs.map +1 -1
- package/dist/visualization.d.cts +1 -1
- package/dist/visualization.d.ts +1 -1
- package/dist/visualization.js +1 -1
- package/dist/visualization.js.map +1 -1
- package/package.json +1 -2
- package/dist/chunk-FNWG34KH.cjs +0 -3
- package/dist/chunk-FNWG34KH.cjs.map +0 -1
- package/dist/chunk-G6W5VE7C.cjs +0 -2
- package/dist/chunk-I3OX3G7J.js +0 -2
- package/dist/chunk-I3OX3G7J.js.map +0 -1
- package/dist/chunk-VGM4KAFN.js +0 -3
- package/dist/chunk-VGM4KAFN.js.map +0 -1
- /package/dist/{chunk-QFHFH5BS.js.map → chunk-WD3ENUAA.js.map} +0 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/features/visualization/threejs/three-initializer.ts","../src/features/visualization/threejs/three-helpers.ts","../src/features/visualization/threejs/three-materials.ts","../src/features/visualization/webdisplay/batch-parser.ts","../src/features/visualization/webdisplay/mesh-compression.ts","../src/features/visualization/webdisplay/webdisplay-parser.ts"],"sourcesContent":["import * as THREE from 'three';\nimport { OrbitControls } from 'three/addons/controls/OrbitControls.js';\nimport { RGBELoader } from 'three/addons/loaders/RGBELoader.js';\n\nimport { getLogger } from '@/core';\nimport { ThreeInitializerOptions } from '../types';\n\nconst defaultUp = new THREE.Vector3(0, 0, 1);\n\n/**\n * Initializes a comprehensive Three.js environment with enhanced render quality and flexible configuration.\n *\n * @param canvas - The HTML canvas element to render the scene on.\n * @param options - Configuration options for the Three.js environment.\n * @returns An object containing the scene, camera, controls, renderer, and utility methods.\n */\nexport const initThree = function (\n\tcanvas: HTMLCanvasElement,\n\toptions?: ThreeInitializerOptions\n): {\n\tscene: THREE.Scene;\n\tcamera: THREE.PerspectiveCamera;\n\tcontrols: OrbitControls;\n\trenderer: THREE.WebGLRenderer;\n\tdispose: () => void;\n\tresize: () => void;\n\tfitToView: () => void;\n\tclearSelection: () => void;\n} {\n\tconst config = applyDefaults(options || {});\n\n\t// Initialize core components\n\tconst scene = createScene(config);\n\tconst camera = createCamera(config);\n\tconst renderer = setupRenderer(canvas, config);\n\tconst controls = setupControls(camera, canvas, config);\n\n\t// Setup environment and lighting\n\tsetupEnvironment(scene, config);\n\tsetupLighting(scene, config);\n\n\t// Add floor if enabled\n\tif (config.floor?.enabled) {\n\t\taddFloor(scene, config);\n\t}\n\n\tconst eventHandlers =\n\t\tconfig.events.enableEventHandlers !== false\n\t\t\t? setupEventHandlers(canvas, scene, camera, controls, config)\n\t\t\t: { dispose: () => {}, fitToView: () => {}, clearSelection: () => {} };\n\n\t// Handle resizing\n\tconst { resize, dispose: disposeResize } = setupResponsiveResize(canvas, renderer, camera);\n\n\t// Animation loop\n\tconst { animate, dispose: disposeAnimation } = createAnimationLoop(\n\t\trenderer,\n\t\tscene,\n\t\tcamera,\n\t\tcontrols\n\t);\n\tanimate();\n\n\t// Set scene up vector\n\tconst sceneUp = config.environment?.sceneUp || defaultUp;\n\tscene.up.set(sceneUp.x, sceneUp.y, sceneUp.z);\n\n\t// Comprehensive disposal\n\tconst dispose = () => {\n\t\tdisposeAnimation(); // Stop animation loop\n\t\tdisposeResize(); // Remove resize listeners\n\t\teventHandlers.dispose(); // Remove click/keyboard listeners\n\t\tcontrols.dispose(); // Dispose controls\n\t\trenderer.dispose(); // Dispose renderer\n\n\t\t// Dispose geometries and materials\n\t\tscene.traverse((object) => {\n\t\t\tif (object instanceof THREE.Mesh) {\n\t\t\t\tobject.geometry?.dispose();\n\t\t\t\tif (Array.isArray(object.material)) {\n\t\t\t\t\tobject.material.forEach((material) => material.dispose());\n\t\t\t\t} else {\n\t\t\t\t\tobject.material?.dispose();\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\t};\n\n\treturn {\n\t\tscene,\n\t\tcamera,\n\t\tcontrols,\n\t\trenderer,\n\t\tdispose,\n\t\tresize,\n\t\tfitToView: eventHandlers.fitToView,\n\t\tclearSelection: eventHandlers.clearSelection\n\t};\n};\n\nfunction applyDefaults(options: ThreeInitializerOptions): Required<ThreeInitializerOptions> {\n\tconst scale = options.sceneScale || 'm';\n\n\t// Define sensible defaults for each scale\n\t// Note: All Rhino geometry is normalized to METERS (1 unit = 1 meter), sceneScale just changes the viewing perspective\n\tconst scaleDefaults = {\n\t\tmm: {\n\t\t\t// Geometry scaled UP by 1000x (mm to m conversion for better precision)\n\t\t\tcameraDistance: 20,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 100,\n\t\t\tlightDistance: 10,\n\t\t\tlightHeight: 20,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 100,\n\t\t\tscaleFactor: 1000\n\t\t},\n\t\tcm: {\n\t\t\t// Geometry scaled UP by 100x (cm to m conversion)\n\t\t\tcameraDistance: 20,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 100,\n\t\t\tlightDistance: 25,\n\t\t\tlightHeight: 50,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 100,\n\t\t\tscaleFactor: 100\n\t\t},\n\t\tm: {\n\t\t\t// Natural Three.js scale (1 unit = 1 meter)\n\t\t\tcameraDistance: 10,\n\t\t\tnear: 0.01,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 50,\n\t\t\tlightDistance: 25,\n\t\t\tlightHeight: 50,\n\t\t\tminDistance: 0.001,\n\t\t\tshadowSize: 100,\n\t\t\tscaleFactor: 1\n\t\t},\n\t\tinches: {\n\t\t\t// Geometry scaled UP by ~39.37x (inches to m conversion)\n\t\t\tcameraDistance: 15,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 80,\n\t\t\tlightDistance: 20,\n\t\t\tlightHeight: 40,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 80,\n\t\t\tscaleFactor: 39.37\n\t\t},\n\t\tfeet: {\n\t\t\t// Geometry scaled UP by ~3.28x (feet to m conversion)\n\t\t\tcameraDistance: 8,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 40,\n\t\t\tlightDistance: 15,\n\t\t\tlightHeight: 30,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 60,\n\t\t\tscaleFactor: 3.28084\n\t\t}\n\t};\n\n\tconst defaults = scaleDefaults[scale];\n\n\treturn {\n\t\tsceneScale: scale,\n\t\tcamera: {\n\t\t\tposition:\n\t\t\t\toptions.camera?.position ||\n\t\t\t\tnew THREE.Vector3(\n\t\t\t\t\t-defaults.cameraDistance,\n\t\t\t\t\tdefaults.cameraDistance,\n\t\t\t\t\tdefaults.cameraDistance\n\t\t\t\t),\n\t\t\tfov: options.camera?.fov || 20,\n\t\t\tnear: options.camera?.near || defaults.near,\n\t\t\tfar: options.camera?.far || defaults.far,\n\t\t\ttarget: options.camera?.target || new THREE.Vector3(0, 0, 0)\n\t\t},\n\t\tlighting: {\n\t\t\tenableSunlight: options.lighting?.enableSunlight ?? true,\n\t\t\tsunlightIntensity: options.lighting?.sunlightIntensity || 1,\n\t\t\tsunlightPosition:\n\t\t\t\toptions.lighting?.sunlightPosition ||\n\t\t\t\tnew THREE.Vector3(defaults.lightDistance, defaults.lightHeight, defaults.lightDistance),\n\t\t\tambientLightColor: options.lighting?.ambientLightColor || new THREE.Color(0x404040),\n\t\t\tambientLightIntensity: options.lighting?.ambientLightIntensity || 1,\n\t\t\tsunlightColor: options.lighting?.sunlightColor || 0xffffff // Default to white sunlight\n\t\t},\n\t\tenvironment: {\n\t\t\thdrPath: options.environment?.hdrPath || '/baseHDR.hdr',\n\t\t\tbackgroundColor: options.environment?.backgroundColor || new THREE.Color(0xf0f0f0),\n\t\t\tenableEnvironmentLighting: options.environment?.enableEnvironmentLighting ?? true,\n\t\t\tsceneUp: options.environment?.sceneUp || defaultUp,\n\t\t\tshowEnvironment: options.environment?.showEnvironment ?? false\n\t\t},\n\t\tfloor: {\n\t\t\tenabled: options.floor?.enabled ?? false,\n\t\t\tsize: options.floor?.size || defaults.floorSize,\n\t\t\tcolor: options.floor?.color || new THREE.Color(0x808080),\n\t\t\troughness: options.floor?.roughness || 0.7,\n\t\t\tmetalness: options.floor?.metalness || 0.0,\n\t\t\treceiveShadow: options.floor?.receiveShadow ?? true\n\t\t},\n\t\trender: {\n\t\t\tenableShadows: options.render?.enableShadows ?? true,\n\t\t\tshadowMapSize: options.render?.shadowMapSize || 2048,\n\t\t\tantialias: options.render?.antialias ?? true,\n\t\t\tpixelRatio: options.render?.pixelRatio || Math.min(window.devicePixelRatio, 2),\n\t\t\ttoneMapping: options.render?.toneMapping || THREE.NeutralToneMapping,\n\t\t\ttoneMappingExposure: options.render?.toneMappingExposure || 1,\n\t\t\tpreserveDrawingBuffer: options.render?.preserveDrawingBuffer ?? false\n\t\t},\n\t\tcontrols: {\n\t\t\tenableDamping: options.controls?.enableDamping ?? false,\n\t\t\tdampingFactor: options.controls?.dampingFactor || 0.05,\n\t\t\tautoRotate: options.controls?.autoRotate ?? false,\n\t\t\tautoRotateSpeed: options.controls?.autoRotateSpeed || 0.5,\n\t\t\tenableZoom: options.controls?.enableZoom ?? true,\n\t\t\tenablePan: options.controls?.enablePan ?? true,\n\t\t\tminDistance: options.controls?.minDistance || defaults.minDistance,\n\t\t\tmaxDistance: options.controls?.maxDistance || Infinity\n\t\t},\n\t\tevents: {\n\t\t\tonBackgroundClicked: options.events?.onBackgroundClicked,\n\t\t\tonObjectSelected: options.events?.onObjectSelected,\n\t\t\tonMeshMetadataClicked: options.events?.onMeshMetadataClicked,\n\t\t\tselectionColor: options.events?.selectionColor || '#ff0000', // Default to red\n\t\t\tenableEventHandlers: options.events?.enableEventHandlers ?? true,\n\t\t\tenableKeyboardControls: options.events?.enableKeyboardControls ?? true,\n\t\t\tenableClickToFocus: options.events?.enableClickToFocus ?? true\n\t\t}\n\t};\n}\n\n/**\n * Creates and configures the scene.\n */\nfunction createScene(config: Required<ThreeInitializerOptions>): THREE.Scene {\n\tconst scene = new THREE.Scene();\n\n\t// Clear existing children except floor\n\tscene.children.forEach((child) => {\n\t\tif (child.userData.id !== 'floor') {\n\t\t\tscene.remove(child);\n\t\t}\n\t});\n\n\t// Set background color\n\tconst bgColor =\n\t\ttypeof config.environment.backgroundColor === 'string'\n\t\t\t? new THREE.Color(config.environment.backgroundColor)\n\t\t\t: config.environment.backgroundColor;\n\tscene.background = bgColor || null;\n\n\treturn scene;\n}\n\n/**\n * Creates an optimized animation loop with proper disposal.\n */\nfunction createAnimationLoop(\n\trenderer: THREE.WebGLRenderer,\n\tscene: THREE.Scene,\n\tcamera: THREE.PerspectiveCamera,\n\tcontrols: OrbitControls\n): { animate: () => void; dispose: () => void } {\n\tlet animationId: number | null = null;\n\n\tconst animate = function () {\n\t\tanimationId = requestAnimationFrame(animate);\n\n\t\t// Update controls if damping is enabled\n\t\tif (controls.enableDamping) {\n\t\t\tcontrols.update();\n\t\t}\n\n\t\trenderer.render(scene, camera);\n\t};\n\n\tconst dispose = () => {\n\t\tif (animationId !== null) {\n\t\t\tcancelAnimationFrame(animationId);\n\t\t\tanimationId = null;\n\t\t}\n\t};\n\n\treturn { animate, dispose };\n}\n\n/**\n * Sets up responsive resizing with improved performance and proper parent handling.\n */\nfunction setupResponsiveResize(\n\tcanvas: HTMLCanvasElement,\n\trenderer: THREE.WebGLRenderer,\n\tcamera: THREE.PerspectiveCamera\n): { resize: () => void; dispose: () => void } {\n\tconst parent = canvas.parentElement;\n\tlet resizeTimeout: NodeJS.Timeout;\n\tlet resizeObserver: ResizeObserver | null = null;\n\n\tconst getSize = () => {\n\t\tif (parent) {\n\t\t\treturn {\n\t\t\t\twidth: parent.clientWidth,\n\t\t\t\theight: parent.clientHeight\n\t\t\t};\n\t\t} else {\n\t\t\treturn {\n\t\t\t\twidth: window.innerWidth,\n\t\t\t\theight: window.innerHeight\n\t\t\t};\n\t\t}\n\t};\n\n\tconst handleResize = () => {\n\t\tif (resizeTimeout) {\n\t\t\tclearTimeout(resizeTimeout);\n\t\t}\n\n\t\tresizeTimeout = setTimeout(() => {\n\t\t\tconst { width, height } = getSize();\n\n\t\t\t// Only update if size actually changed\n\t\t\tif (renderer.domElement.width !== width || renderer.domElement.height !== height) {\n\t\t\t\trenderer.setSize(width, height, false);\n\t\t\t\tcamera.aspect = width / height;\n\t\t\t\tcamera.updateProjectionMatrix();\n\t\t\t}\n\t\t}, 16); // ~60fps throttling\n\t};\n\n\t// Set up the appropriate resize observer\n\tif (parent && typeof ResizeObserver !== 'undefined') {\n\t\t// Use ResizeObserver for parent container\n\t\tresizeObserver = new ResizeObserver(handleResize);\n\t\tresizeObserver.observe(parent);\n\t} else {\n\t\t// Fallback to window resize for fullscreen or older browsers\n\t\twindow.addEventListener('resize', handleResize);\n\t}\n\n\tconst dispose = () => {\n\t\tif (resizeTimeout) {\n\t\t\tclearTimeout(resizeTimeout);\n\t\t}\n\t\tif (resizeObserver) {\n\t\t\tresizeObserver.disconnect();\n\t\t} else {\n\t\t\twindow.removeEventListener('resize', handleResize);\n\t\t}\n\t};\n\n\treturn { resize: handleResize, dispose };\n}\n\n/**\n * Sets up environment lighting and HDR.\n */\nfunction setupEnvironment(scene: THREE.Scene, config: Required<ThreeInitializerOptions>) {\n\tif (config.environment.enableEnvironmentLighting) {\n\t\tnew RGBELoader().load(\n\t\t\tconfig.environment.hdrPath || '/baseHDR.hdr',\n\t\t\tfunction (envMap) {\n\t\t\t\tenvMap.mapping = THREE.EquirectangularReflectionMapping;\n\t\t\t\tscene.environment = envMap;\n\t\t\t\tif (config.environment.showEnvironment) {\n\t\t\t\t\tscene.background = envMap;\n\t\t\t\t}\n\t\t\t},\n\t\t\tundefined,\n\t\t\tfunction (error) {\n\t\t\t\tgetLogger().warn('HDR texture could not be loaded, falling back to basic lighting:', error);\n\t\t\t\t// Add basic ambient light as fallback\n\t\t\t\tconst ambientLight = new THREE.AmbientLight(0x404040, 0.4);\n\t\t\t\tscene.add(ambientLight);\n\t\t\t}\n\t\t);\n\t}\n}\n\nfunction setupLighting(scene: THREE.Scene, config: Required<ThreeInitializerOptions>) {\n\t// Add ambient light\n\tconst ambientLight = new THREE.AmbientLight(\n\t\tconfig.lighting.ambientLightColor,\n\t\tconfig.lighting.ambientLightIntensity\n\t);\n\tscene.add(ambientLight);\n\n\t// Add directional light (sunlight)\n\tif (config.lighting.enableSunlight) {\n\t\tconst sunlight = new THREE.DirectionalLight(\n\t\t\tconfig.lighting.sunlightColor ?? 0xffffff,\n\t\t\tconfig.lighting.sunlightIntensity\n\t\t);\n\t\tconst pos = config.lighting.sunlightPosition;\n\t\tif (pos) {\n\t\t\tsunlight.position.set(pos.x, pos.y, pos.z);\n\t\t}\n\n\t\tif (config.render.enableShadows) {\n\t\t\tsunlight.castShadow = true;\n\t\t\tconst shadowSize = config.sceneScale === 'mm' ? 0.1 : config.sceneScale === 'cm' ? 10 : 100;\n\n\t\t\tsunlight.shadow.camera.left = -shadowSize;\n\t\t\tsunlight.shadow.camera.right = shadowSize;\n\t\t\tsunlight.shadow.camera.top = shadowSize;\n\t\t\tsunlight.shadow.camera.bottom = -shadowSize;\n\n\t\t\tconst shadowNear =\n\t\t\t\tconfig.sceneScale === 'mm' ? 0.001 : config.sceneScale === 'cm' ? 0.1 : 0.5;\n\n\t\t\tconst shadowFar = config.sceneScale === 'mm' ? 1 : config.sceneScale === 'cm' ? 100 : 500;\n\n\t\t\tsunlight.shadow.camera.near = shadowNear;\n\t\t\tsunlight.shadow.camera.far = shadowFar;\n\n\t\t\tsunlight.shadow.mapSize.width = config.render.shadowMapSize || 2048;\n\t\t\tsunlight.shadow.mapSize.height = config.render.shadowMapSize || 2048;\n\n\t\t\t// Improved shadow quality\n\t\t\tsunlight.shadow.bias = -0.0001;\n\t\t\tsunlight.shadow.normalBias = 0.02;\n\t\t}\n\n\t\tscene.add(sunlight);\n\t}\n}\n\n/**\n * Adds a floor to the scene with scale-aware sizing.\n */\nfunction addFloor(scene: THREE.Scene, config: Required<ThreeInitializerOptions>) {\n\tconst floorSize = config.floor.size;\n\tconst floorGeometry = new THREE.PlaneGeometry(floorSize, floorSize);\n\n\tconst floorColor =\n\t\ttypeof config.floor.color === 'string'\n\t\t\t? new THREE.Color(config.floor.color)\n\t\t\t: config.floor.color;\n\n\tconst floorMaterial = new THREE.MeshStandardMaterial({\n\t\tcolor: floorColor,\n\t\troughness: config.floor.roughness,\n\t\tmetalness: config.floor.metalness,\n\t\tside: THREE.DoubleSide\n\t});\n\n\tconst floor = new THREE.Mesh(floorGeometry, floorMaterial);\n\tfloor.userData.id = 'floor';\n\tfloor.name = 'floor';\n\tfloor.rotation.x = -Math.PI / 2;\n\tfloor.position.y = 0;\n\n\tif (config.floor.receiveShadow && config.render.enableShadows) {\n\t\tfloor.receiveShadow = true;\n\t}\n\n\tscene.add(floor);\n}\n\n/**\n * Creates and configures the camera with proper aspect ratio.\n */\nfunction createCamera(config: Required<ThreeInitializerOptions>): THREE.PerspectiveCamera {\n\t// Get proper aspect ratio from canvas parent or window\n\tconst canvas = document.querySelector('canvas');\n\tconst parent = canvas?.parentElement;\n\tconst width = parent ? parent.clientWidth : window.innerWidth;\n\tconst height = parent ? parent.clientHeight : window.innerHeight;\n\n\tconst camera = new THREE.PerspectiveCamera(\n\t\tconfig.camera.fov,\n\t\twidth / height,\n\t\tconfig.camera.near,\n\t\tconfig.camera.far\n\t);\n\n\tconst pos = config.camera.position;\n\tif (pos) {\n\t\tcamera.position.set(pos.x, pos.y, pos.z);\n\t}\n\n\treturn camera;\n}\n\n/**\n * Sets up enhanced WebGL renderer with improved quality settings.\n */\nfunction setupRenderer(\n\tcanvas: HTMLCanvasElement,\n\tconfig: Required<ThreeInitializerOptions>\n): THREE.WebGLRenderer {\n\tconst renderer = new THREE.WebGLRenderer({\n\t\tantialias: config.render.antialias,\n\t\tcanvas,\n\t\talpha: true,\n\t\tpowerPreference: 'high-performance',\n\t\tpreserveDrawingBuffer: config.render.preserveDrawingBuffer,\n\t\t// Enable logarithmic depth buffer for extreme scale ranges\n\t\t// This dramatically improves depth precision for mixed scales (mm to km)\n\t\tlogarithmicDepthBuffer: true\n\t});\n\n\t// Get proper dimensions - parent container or window\n\tconst parent = canvas.parentElement;\n\tconst width = parent ? parent.clientWidth : window.innerWidth;\n\tconst height = parent ? parent.clientHeight : window.innerHeight;\n\n\t// Set canvas style to fill parent if it exists\n\tif (parent) {\n\t\tcanvas.style.width = '100%';\n\t\tcanvas.style.height = '100%';\n\t\tcanvas.style.display = 'block';\n\t}\n\n\trenderer.setSize(width, height, false);\n\trenderer.setPixelRatio(config.render.pixelRatio || Math.min(window.devicePixelRatio, 2));\n\n\t// Enhanced shadow settings\n\tif (config.render.enableShadows) {\n\t\trenderer.shadowMap.enabled = true;\n\t\t// Use VSM for better quality with extreme scales\n\t\trenderer.shadowMap.type = THREE.VSMShadowMap;\n\t}\n\n\t// Improved tone mapping and color management\n\trenderer.toneMapping = config.render.toneMapping || THREE.ACESFilmicToneMapping;\n\trenderer.toneMappingExposure = config.render.toneMappingExposure || 1.0;\n\trenderer.outputColorSpace = THREE.SRGBColorSpace;\n\n\t// Additional quality settings for depth rendering\n\trenderer.sortObjects = true; // Ensure proper render order\n\n\treturn renderer;\n}\n\n// Add event handler setup function\nfunction setupEventHandlers(\n\tcanvas: HTMLCanvasElement,\n\tscene: THREE.Scene,\n\tcamera: THREE.PerspectiveCamera,\n\tcontrols: OrbitControls,\n\tconfig: Required<ThreeInitializerOptions>\n): {\n\tdispose: () => void;\n\tfitToView: () => void;\n\tclearSelection: () => void;\n} {\n\tconst selectedObjects = new Set<THREE.Object3D>();\n\tconst originalMaterials = new Map<THREE.Object3D, THREE.Material | THREE.Material[]>();\n\tconst raycaster = new THREE.Raycaster();\n\tconst mouse = new THREE.Vector2();\n\tconst mouseDownPosition = new THREE.Vector2();\n\n\t// Fit scene to view\n\tconst fitToView = () => {\n\t\tconst box = new THREE.Box3();\n\n\t\t// Calculate bounding box of all visible objects (excluding floor)\n\t\tscene.traverse((object) => {\n\t\t\tif (object.visible && object.userData.id !== 'floor' && object instanceof THREE.Mesh) {\n\t\t\t\tbox.expandByObject(object);\n\t\t\t}\n\t\t});\n\n\t\tif (box.isEmpty()) {\n\t\t\tgetLogger().warn('No objects to fit to view');\n\t\t\treturn;\n\t\t}\n\n\t\tconst center = box.getCenter(new THREE.Vector3());\n\t\tconst size = box.getSize(new THREE.Vector3());\n\n\t\t// Calculate distance needed to fit the object\n\t\tconst maxDim = Math.max(size.x, size.y, size.z);\n\t\tconst fov = camera.fov * (Math.PI / 180);\n\t\tlet distance = maxDim / (2 * Math.tan(fov / 2));\n\n\t\t// Add some padding\n\t\tdistance *= 1.5;\n\n\t\t// Position camera\n\t\tconst direction = camera.position.clone().sub(controls.target).normalize();\n\t\tcamera.position.copy(center.clone().add(direction.multiplyScalar(distance)));\n\n\t\t// Update controls target\n\t\tcontrols.target.copy(center);\n\t\tcontrols.update();\n\t};\n\n\t// Parse selection color\n\tconst selectionColorObj =\n\t\ttypeof config.events.selectionColor === 'string'\n\t\t\t? new THREE.Color(config.events.selectionColor)\n\t\t\t: config.events.selectionColor instanceof THREE.Color\n\t\t\t\t? config.events.selectionColor\n\t\t\t\t: new THREE.Color('#ff0000');\n\n\t// Clear selection\n\tconst clearSelection = () => {\n\t\tselectedObjects.forEach((obj) => {\n\t\t\t// Restore original material\n\t\t\tif (obj instanceof THREE.Mesh && originalMaterials.has(obj)) {\n\t\t\t\tobj.material = originalMaterials.get(obj)!;\n\t\t\t\toriginalMaterials.delete(obj);\n\t\t\t}\n\t\t});\n\t\tselectedObjects.clear();\n\t};\n\n\tconst handleMouseDown = (event: MouseEvent) => {\n\t\tmouseDownPosition.set(event.clientX, event.clientY);\n\t};\n\n\t// Handle canvas clicks\n\tconst handleCanvasClick = (event: MouseEvent) => {\n\t\t// Ignore if mouse has moved significantly (drag)\n\t\tconst currentMousePosition = new THREE.Vector2(event.clientX, event.clientY);\n\t\tif (mouseDownPosition.distanceTo(currentMousePosition) > 5) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Calculate mouse position in normalized device coordinates\n\t\tconst rect = canvas.getBoundingClientRect();\n\t\tmouse.x = ((event.clientX - rect.left) / rect.width) * 2 - 1;\n\t\tmouse.y = -((event.clientY - rect.top) / rect.height) * 2 + 1;\n\n\t\t// Raycast to find intersected objects\n\t\traycaster.setFromCamera(mouse, camera);\n\t\tconst intersects = raycaster.intersectObjects(scene.children, true);\n\n\t\tif (intersects.length > 0) {\n\t\t\tconst clickedObject = intersects[0].object;\n\n\t\t\t// Handle object selection\n\t\t\tif (!selectedObjects.has(clickedObject)) {\n\t\t\t\tclearSelection();\n\t\t\t\tselectedObjects.add(clickedObject);\n\n\t\t\t\t// Clone material and apply selection color only to this mesh\n\t\t\t\tif (\n\t\t\t\t\tclickedObject instanceof THREE.Mesh &&\n\t\t\t\t\tclickedObject.material instanceof THREE.Material\n\t\t\t\t) {\n\t\t\t\t\t// Store original material\n\t\t\t\t\toriginalMaterials.set(clickedObject, clickedObject.material);\n\n\t\t\t\t\t// Clone the material so we don't affect other meshes\n\t\t\t\t\tconst clonedMaterial = clickedObject.material.clone();\n\t\t\t\t\t(clonedMaterial as any).emissive = selectionColorObj.clone();\n\t\t\t\t\tclickedObject.material = clonedMaterial;\n\t\t\t\t}\n\n\t\t\t\tconfig.events?.onObjectSelected?.(clickedObject);\n\n\t\t\t\t// Call metadata callback if the mesh has metadata\n\t\t\t\tif (clickedObject instanceof THREE.Mesh && Object.keys(clickedObject.userData).length > 0) {\n\t\t\t\t\tconfig.events?.onMeshMetadataClicked?.(clickedObject.userData);\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\t// Background clicked\n\t\t\tclearSelection();\n\t\t\tconfig.events?.onBackgroundClicked?.({ x: mouse.x, y: mouse.y });\n\t\t}\n\t};\n\n\t// Handle keyboard events\n\tconst handleKeydown = (event: KeyboardEvent) => {\n\t\tif (!config.events?.enableKeyboardControls) return;\n\n\t\tswitch (event.key.toLowerCase()) {\n\t\t\tcase 'f':\n\t\t\t\tevent.preventDefault();\n\t\t\t\tfitToView();\n\t\t\t\tbreak;\n\t\t\tcase 'escape':\n\t\t\t\tevent.preventDefault();\n\t\t\t\tclearSelection();\n\t\t\t\tbreak;\n\t\t\tcase ' ':\n\t\t\t\tevent.preventDefault();\n\t\t\t\tfitToView();\n\t\t\t\tbreak;\n\t\t}\n\t};\n\n\t// Add event listeners\n\tif (config.events?.enableClickToFocus) {\n\t\tcanvas.addEventListener('mousedown', handleMouseDown);\n\t\tcanvas.addEventListener('click', handleCanvasClick);\n\t}\n\n\tif (config.events?.enableKeyboardControls) {\n\t\t// Make canvas focusable\n\t\tcanvas.setAttribute('tabindex', '0');\n\t\t// Only listen for keydown when canvas has focus\n\t\tcanvas.addEventListener('keydown', handleKeydown);\n\t}\n\n\t// Disposal function\n\tconst dispose = () => {\n\t\tcanvas.removeEventListener('mousedown', handleMouseDown);\n\t\tcanvas.removeEventListener('click', handleCanvasClick);\n\t\tcanvas.removeEventListener('keydown', handleKeydown);\n\t\tclearSelection();\n\t};\n\n\treturn { dispose, fitToView, clearSelection };\n}\n\n/**\n * Sets up enhanced orbit controls with scale-aware distances.\n */\nfunction setupControls(\n\tcamera: THREE.PerspectiveCamera,\n\tcanvas: HTMLCanvasElement,\n\tconfig: Required<ThreeInitializerOptions>\n): OrbitControls {\n\tconst controls = new OrbitControls(camera, canvas);\n\n\t// Set target\n\tconst target = config.camera.target;\n\tif (target) {\n\t\tcontrols.target.set(target.x, target.y, target.z);\n\t}\n\n\t// Configure damping\n\tcontrols.enableDamping = config.controls.enableDamping || false;\n\tcontrols.dampingFactor = config.controls.dampingFactor || 0.05;\n\n\t// Configure auto rotation\n\tcontrols.autoRotate = config.controls.autoRotate || false;\n\tcontrols.autoRotateSpeed = config.controls.autoRotateSpeed || 0.5;\n\n\t// Configure interaction limits\n\tcontrols.enableZoom = config.controls.enableZoom || true;\n\tcontrols.enablePan = config.controls.enablePan || true;\n\tcontrols.minDistance = config.controls.minDistance || 0.001;\n\tcontrols.maxDistance = config.controls.maxDistance || Infinity;\n\n\t// Smooth controls\n\tcontrols.screenSpacePanning = false;\n\tcontrols.maxPolarAngle = Math.PI;\n\n\tcontrols.update();\n\treturn controls;\n}\n","import * as THREE from 'three';\nimport { OrbitControls } from 'three/addons/controls/OrbitControls.js';\nimport { getLogger } from '@/core';\n\n/**\n * Updates the scene with the given meshes and camera settings.\n * If initialPositionSet is false, it positions the camera and sets the controls target based on the bounding boxes of the meshes.\n * @param scene - The THREE.Scene object to update.\n * @param meshes - An array of THREE.Mesh objects to add to the scene.\n * @param camera - The THREE.PerspectiveCamera object to position.\n * @param controls - The OrbitControls object to update.\n * @param initialPositionSet - A boolean indicating whether the initial position of the camera and controls have been set.\n */\nexport function updateScene(\n\tscene: THREE.Scene,\n\tmeshes: THREE.Mesh[],\n\tcamera: THREE.PerspectiveCamera,\n\tcontrols: OrbitControls,\n\tinitialPositionSet: boolean\n) {\n\tclearScene(scene);\n\n\tif (meshes.length === 0) return;\n\n\tconst unionBoundingBox = new THREE.Box3();\n\n\tmeshes.forEach((mesh) => {\n\t\tscene.add(mesh);\n\t\tconst boundingBox = new THREE.Box3().setFromObject(mesh);\n\t\tunionBoundingBox.union(boundingBox);\n\t});\n\n\t// Get the center of the union bounding box\n\tconst center = unionBoundingBox.getCenter(new THREE.Vector3());\n\tconst size = unionBoundingBox.getSize(new THREE.Vector3());\n\n\t// Calculate a distance that is slightly larger than the largest dimension of the union bounding box\n\tconst maxDim = Math.max(size.x, size.y, size.z);\n\n\t// Always update camera frustum to ensure geometry is visible\n\t// This prevents clipping when geometry size changes significantly\n\tconst scaleRatio = maxDim / Math.min(size.x || 1, size.y || 1, size.z || 1);\n\n\tif (scaleRatio > 100 || maxDim > 10000) {\n\t\t// Large scale range detected - use logarithmic depth buffer approach\n\t\tcamera.near = maxDim * 0.0001; // 0.01% of max dimension\n\t\tcamera.far = maxDim * 100; // 100x max dimension\n\t} else if (maxDim > 1000) {\n\t\t// Large scene\n\t\tcamera.near = maxDim * 0.001;\n\t\tcamera.far = maxDim * 50;\n\t} else {\n\t\t// Normal scene\n\t\tcamera.near = Math.max(0.01, maxDim * 0.01);\n\t\tcamera.far = Math.max(2000, maxDim * 20);\n\t}\n\n\tcamera.updateProjectionMatrix();\n\n\t// Only reposition camera and controls on first frame\n\tif (!initialPositionSet) {\n\t\tconst distance = maxDim * 4;\n\n\t\tcamera.position.set(center.x + distance * 0.8, center.y + distance, center.z + distance * 1.2);\n\t\tcontrols.target = center;\n\t\tcontrols.minDistance = camera.near * 2;\n\t\tcontrols.maxDistance = camera.far * 0.9;\n\n\t\tcontrols.update();\n\t} else {\n\t\t// Update control constraints to match new frustum\n\t\tcontrols.minDistance = camera.near * 2;\n\t\tcontrols.maxDistance = camera.far * 0.9;\n\t}\n}\n\n// =========================\n// Helper functions\n// =========================\n\n/**\n * Parses a color string in multiple formats to a THREE.Color object.\n * Supported formats:\n * - Hex: \"#C7A5A5\", \"C7A5A5\"\n * - RGB: \"199, 165, 165\"\n * - CSS named colors: \"red\", \"blue\", etc.\n * @param colorString - The color string to parse.\n * @returns A THREE.Color object.\n */\nexport function parseColor(colorString: string): THREE.Color {\n\tif (!colorString || typeof colorString !== 'string') {\n\t\tgetLogger().warn(`Invalid color input: ${colorString}, using white`);\n\t\treturn new THREE.Color(0xffffff);\n\t}\n\n\tconst trimmed = colorString.trim();\n\n\t// Try hex format (#C7A5A5 or C7A5A5)\n\tif (trimmed.startsWith('#') || /^[0-9A-Fa-f]{6}$/.test(trimmed)) {\n\t\ttry {\n\t\t\tconst hex = trimmed.startsWith('#') ? trimmed : `#${trimmed}`;\n\t\t\treturn new THREE.Color(hex);\n\t\t} catch {\n\t\t\tgetLogger().warn(`Invalid hex color: ${colorString}, using white`);\n\t\t\treturn new THREE.Color(0xffffff);\n\t\t}\n\t}\n\n\t// Try RGB format (R, G, B)\n\tif (trimmed.includes(',')) {\n\t\tconst rgb = trimmed.split(',').map((c) => parseInt(c.trim(), 10));\n\t\tif (rgb.length === 3 && rgb.every((n) => !isNaN(n) && n >= 0 && n <= 255)) {\n\t\t\treturn new THREE.Color(rgb[0] / 255, rgb[1] / 255, rgb[2] / 255);\n\t\t}\n\t}\n\n\t// Try CSS named color\n\ttry {\n\t\treturn new THREE.Color(trimmed.toLowerCase());\n\t} catch {\n\t\tgetLogger().warn(`Invalid color string: ${colorString}, using white`);\n\t\treturn new THREE.Color(0xffffff);\n\t}\n}\n\nexport function applyOffset(meshes: THREE.Mesh[], offsetY: number): void {\n\tmeshes.forEach((mesh) => {\n\t\tmesh.position.y -= offsetY;\n\t});\n}\n\nexport function computeCombinedBoundingBox(meshes: THREE.Mesh[]): THREE.Box3 {\n\tconst combinedBoundingBox = new THREE.Box3();\n\tmeshes.forEach((mesh) => {\n\t\tmesh.geometry.computeBoundingBox();\n\t\tif (mesh.geometry.boundingBox) {\n\t\t\tcombinedBoundingBox.union(mesh.geometry.boundingBox);\n\t\t}\n\t});\n\treturn combinedBoundingBox;\n}\n\n/**\n * Updates shadow camera bounds to match scene geometry.\n * This prevents shadow artifacts and ensures proper shadow coverage.\n */\nexport function updateShadowCameraBounds(\n\tscene: THREE.Scene,\n\tdirectionalLight: THREE.DirectionalLight\n): void {\n\tconst bbox = new THREE.Box3();\n\n\tscene.traverse((object) => {\n\t\tif (object instanceof THREE.Mesh && object.userData.id !== 'floor') {\n\t\t\tbbox.expandByObject(object);\n\t\t}\n\t});\n\n\tif (bbox.isEmpty()) return;\n\n\tconst size = bbox.getSize(new THREE.Vector3());\n\tconst center = bbox.getCenter(new THREE.Vector3());\n\tconst maxDim = Math.max(size.x, size.y, size.z);\n\n\t// Position light relative to scene center\n\tconst lightDistance = maxDim * 2;\n\tdirectionalLight.position.set(\n\t\tcenter.x + lightDistance * 0.5,\n\t\tcenter.y + lightDistance,\n\t\tcenter.z + lightDistance * 0.5\n\t);\n\tdirectionalLight.target.position.copy(center);\n\n\t// Adjust shadow camera bounds to scene size with padding\n\tconst padding = maxDim * 0.2;\n\tdirectionalLight.shadow.camera.left = -maxDim / 2 - padding;\n\tdirectionalLight.shadow.camera.right = maxDim / 2 + padding;\n\tdirectionalLight.shadow.camera.top = maxDim / 2 + padding;\n\tdirectionalLight.shadow.camera.bottom = -maxDim / 2 - padding;\n\tdirectionalLight.shadow.camera.near = 0.1;\n\tdirectionalLight.shadow.camera.far = lightDistance * 3;\n\n\t// Improve shadow quality for extreme scales\n\tif (maxDim > 1000) {\n\t\tdirectionalLight.shadow.bias = -0.001;\n\t\tdirectionalLight.shadow.normalBias = 0.05;\n\t} else {\n\t\tdirectionalLight.shadow.bias = -0.0001;\n\t\tdirectionalLight.shadow.normalBias = 0.02;\n\t}\n\n\tdirectionalLight.shadow.camera.updateProjectionMatrix();\n}\n\n/**\n * Clears the given THREE.Scene by removing all meshes and disposing of associated resources.\n * @param scene - The THREE.Scene to clear.\n */\nfunction clearScene(scene: THREE.Scene): void {\n\tconst objectsToRemove: THREE.Object3D[] = [];\n\n\t// Collect all meshes except the floor\n\tscene.traverse((child: THREE.Object3D) => {\n\t\tif (child instanceof THREE.Mesh && child.userData.id !== 'floor') {\n\t\t\tobjectsToRemove.push(child);\n\t\t}\n\t});\n\n\t// Remove and dispose of each object\n\tobjectsToRemove.forEach((object: THREE.Object3D) => {\n\t\tif (object instanceof THREE.Mesh) {\n\t\t\tobject.geometry?.dispose();\n\n\t\t\tconst materials = Array.isArray(object.material) ? object.material : [object.material];\n\t\t\tmaterials.forEach((material) => {\n\t\t\t\tObject.values(material).forEach((value) => {\n\t\t\t\t\tif (value instanceof THREE.Texture) {\n\t\t\t\t\t\tvalue.dispose();\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t\tmaterial.dispose();\n\t\t\t});\n\t\t}\n\n\t\tobject.removeFromParent();\n\t});\n}\n","import * as THREE from 'three';\n\nexport const EMISSIVE_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: 0x000000,\n\temissive: new THREE.Color(0xffffff),\n\temissiveIntensity: 5,\n\tmetalness: 0.0,\n\troughness: 0.2,\n\tclearcoat: 0.3,\n\tclearcoatRoughness: 0.2,\n\tdepthWrite: true,\n\tdepthTest: true,\n\ttransparent: false,\n\talphaTest: 0.0,\n\tpolygonOffset: true,\n\tside: THREE.FrontSide,\n\tdithering: true\n});\n\nexport const METAL_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0x000000),\n\tmetalness: 0.9,\n\troughness: 0.3,\n\tenvMapIntensity: 1.2,\n\tclearcoat: 0.3,\n\tclearcoatRoughness: 0.2,\n\treflectivity: 1,\n\tior: 2.5,\n\tthickness: 1,\n\tdepthWrite: true,\n\ttransparent: false,\n\talphaTest: 0.0,\n\tdepthTest: true,\n\tpolygonOffset: true,\n\tside: THREE.FrontSide,\n\tdithering: true\n});\n\nexport const CONCRETE_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0xcccccc),\n\tmetalness: 0.0,\n\troughness: 0.92,\n\tenvMapIntensity: 0.15,\n\tclearcoat: 0.05,\n\tclearcoatRoughness: 0.9,\n\treflectivity: 0.15,\n\ttransmission: 0.0,\n\tior: 1.45,\n\tthickness: 0.0,\n\tdepthWrite: true,\n\ttransparent: false,\n\talphaTest: 0.5,\n\tdepthTest: true,\n\tpolygonOffset: true,\n\tside: THREE.FrontSide,\n\tdithering: true\n});\n\nexport const PLASTIC_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0xffffff), // Default white plastic\n\tmetalness: 0.0,\n\troughness: 0.3,\n\tenvMapIntensity: 0.5,\n\tclearcoat: 0.5,\n\tclearcoatRoughness: 0.1,\n\treflectivity: 0.5,\n\tior: 1.4,\n\ttransmission: 0.0,\n\ttransparent: false,\n\tdepthWrite: true,\n\tside: THREE.FrontSide,\n\tdithering: true,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n\nexport const GLASS_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0xffffff),\n\tmetalness: 0.0,\n\troughness: 0.0,\n\ttransmission: 0.95,\n\ttransparent: true,\n\topacity: 0.3,\n\tenvMapIntensity: 1.0,\n\tclearcoat: 1.0,\n\tclearcoatRoughness: 0.0,\n\tior: 1.52,\n\treflectivity: 0.9,\n\tthickness: 1.0,\n\tside: THREE.DoubleSide,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n\nexport const RUBBER_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0x1a1a1a),\n\tmetalness: 0.0,\n\troughness: 0.9,\n\tenvMapIntensity: 0.2,\n\tclearcoat: 0.1,\n\tclearcoatRoughness: 0.8,\n\treflectivity: 0.2,\n\tior: 1.3,\n\ttransmission: 0.0,\n\tdepthWrite: true,\n\tside: THREE.FrontSide,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n\nexport const WOOD_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0x885533),\n\tmetalness: 0.0,\n\troughness: 0.7,\n\tenvMapIntensity: 0.3,\n\tclearcoat: 0.3,\n\tclearcoatRoughness: 0.4,\n\treflectivity: 0.3,\n\tior: 1.3,\n\ttransmission: 0.0,\n\tdepthWrite: true,\n\tside: THREE.FrontSide,\n\tdithering: true,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n","import * as THREE from 'three';\n\nimport { parseColor } from '../threejs/three-helpers';\nimport { getLogger } from '@/core';\n\nimport { decompressBatchedMeshData } from './mesh-compression';\n\nimport type { MeshBatch, MaterialGroup, SerializableMaterial } from './types';\n\n/**\n * Parses a batched mesh JSON and creates Three.js meshes.\n *\n * This function handles the optimized batch format where:\n * - Materials are deduplicated and stored once\n * - Meshes are grouped by material for efficient rendering\n * - All geometry data is compressed together and decompressed in a Web Worker\n *\n * @internal Low-level mesh parsing — keep internal to `selva-compute`.\n *\n * @param batchJson - JSON string containing the batched mesh data\n * @param options - Rendering options\n * @returns Promise resolving to array of Three.js mesh objects\n */\nexport async function parseMeshBatch(\n\tbatchJson: string,\n\toptions?: {\n\t\t/** Merge meshes with same material into single geometry*/\n\t\tmergeByMaterial?: boolean;\n\t\t/** Apply coordinate system transformations */\n\t\tapplyTransforms?: boolean;\n\t\t/** Enable performance monitoring */\n\t\tdebug?: boolean;\n\t}\n): Promise<THREE.Mesh[]> {\n\tconst { mergeByMaterial = true, applyTransforms = true, debug = false } = options ?? {};\n\n\tconst perfStart = debug ? performance.now() : 0;\n\tlet parseTime = 0;\n\n\ttry {\n\t\tconst parseStart = performance.now();\n\t\tconst batch: MeshBatch = JSON.parse(batchJson);\n\t\tparseTime = performance.now() - parseStart;\n\n\t\treturn await parseMeshBatchObject(batch, {\n\t\t\tmergeByMaterial,\n\t\t\tapplyTransforms,\n\t\t\tdebug,\n\t\t\tparseTime,\n\t\t\tperfStart\n\t\t});\n\t} catch (error) {\n\t\tgetLogger().error('Error parsing mesh batch:', error);\n\t\treturn [];\n\t}\n}\n\n/**\n * Parses a MeshBatch object and creates Three.js meshes.\n * This is useful when you already have a deserialized MeshBatch object.\n *\n * @internal Low-level mesh parsing — keep internal to `selva-compute`.\n *\n * @param batch - MeshBatch object\n * @param options - Rendering options\n * @returns Promise resolving to array of Three.js mesh objects\n */\nexport async function parseMeshBatchObject(\n\tbatch: MeshBatch,\n\toptions?: {\n\t\t/** Merge meshes with same material into single geometry*/\n\t\tmergeByMaterial?: boolean;\n\t\t/** Apply coordinate system transformations */\n\t\tapplyTransforms?: boolean;\n\t\t/** Scale factor to apply to meshes (e.g., for unit conversion) */\n\t\tscaleFactor?: number;\n\t\t/** Enable performance monitoring */\n\t\tdebug?: boolean;\n\t\t/** Parse time (optional, for debugging) */\n\t\tparseTime?: number;\n\t\t/** Performance start time (optional, for debugging) */\n\t\tperfStart?: number;\n\t}\n): Promise<THREE.Mesh[]> {\n\tconst {\n\t\tmergeByMaterial = true,\n\t\tapplyTransforms = true,\n\t\tscaleFactor = 1,\n\t\tdebug = false,\n\t\tparseTime = 0,\n\t\tperfStart = debug ? performance.now() : 0\n\t} = options ?? {};\n\n\tlet decompressTime = 0,\n\t\tmeshCreateTime = 0;\n\n\ttry {\n\t\tconst decompressStart = performance.now();\n\t\tconst { vertices, faces } = await decompressBatchedMeshData(batch.compressedData);\n\t\tdecompressTime = performance.now() - decompressStart;\n\n\t\tconst compressedSizeMB = ((batch.compressedData.length * 0.75) / 1024 / 1024).toFixed(2); // Base64 overhead\n\t\tconst uncompressedSizeMB = ((vertices.byteLength + faces.byteLength) / 1024 / 1024).toFixed(2);\n\t\tconst compressionRatio = (\n\t\t\t(1 - parseFloat(compressedSizeMB) / parseFloat(uncompressedSizeMB)) *\n\t\t\t100\n\t\t).toFixed(1);\n\n\t\tif (debug) {\n\t\t\tgetLogger().debug('Mesh Batch Stats:');\n\t\t\tgetLogger().debug(` Materials: ${batch.materials.length} | Groups: ${batch.groups.length}`);\n\t\t\tgetLogger().debug(\n\t\t\t\t` Vertices: ${(vertices.length / 3).toLocaleString()} | Faces: ${(faces.length / 3).toLocaleString()}`\n\t\t\t);\n\t\t\tgetLogger().debug(\n\t\t\t\t` Compressed: ${compressedSizeMB} MB | Uncompressed: ${uncompressedSizeMB} MB`\n\t\t\t);\n\t\t\tgetLogger().debug(` Compression Ratio: ${compressionRatio}%`);\n\t\t}\n\n\t\tif (applyTransforms) {\n\t\t\tapplyCoordinateTransform(vertices);\n\t\t}\n\n\t\tconst meshCreateStart = performance.now();\n\t\tconst materials = batch.materials.map(createMaterial);\n\n\t\tconst meshes: THREE.Mesh[] = [];\n\n\t\tfor (const group of batch.groups) {\n\t\t\tif (mergeByMaterial && group.meshes.length > 1) {\n\t\t\t\tconst mergedMesh = createMergedMesh(group, vertices, faces, materials);\n\t\t\t\tmeshes.push(mergedMesh);\n\t\t\t} else {\n\t\t\t\tconst individualMeshes = createIndividualMeshes(group, vertices, faces, materials);\n\t\t\t\tmeshes.push(...individualMeshes);\n\t\t\t}\n\t\t}\n\n\t\t// Apply scaling if needed\n\t\tif (scaleFactor !== 1) {\n\t\t\tfor (const mesh of meshes) {\n\t\t\t\tmesh.scale.set(scaleFactor, scaleFactor, scaleFactor);\n\t\t\t}\n\t\t}\n\n\t\tmeshCreateTime = performance.now() - meshCreateStart;\n\n\t\tif (debug) {\n\t\t\tconst totalTime = performance.now() - perfStart;\n\t\t\tgetLogger().debug('Performance:');\n\t\t\tif (parseTime > 0) getLogger().debug(` Parse JSON: ${parseTime.toFixed(2)}ms`);\n\t\t\tgetLogger().debug(` Decompress: ${decompressTime.toFixed(2)}ms`);\n\t\t\tgetLogger().debug(` Create Meshes: ${meshCreateTime.toFixed(2)}ms`);\n\t\t\tgetLogger().debug(` Total: ${totalTime.toFixed(2)}ms`);\n\t\t}\n\n\t\treturn meshes;\n\t} catch (error) {\n\t\tgetLogger().error('Error parsing mesh batch object:', error);\n\t\treturn [];\n\t}\n}\n\n/**\n * Creates a Three.js material from serializable material data.\n */\nfunction createMaterial(matData: SerializableMaterial): THREE.MeshPhysicalMaterial {\n\tconst color = parseColor(matData.color);\n\n\treturn new THREE.MeshPhysicalMaterial({\n\t\tcolor,\n\t\tmetalness: matData.metalness,\n\t\troughness: matData.roughness,\n\t\topacity: matData.opacity,\n\t\ttransparent: matData.transparent,\n\t\tside: THREE.DoubleSide,\n\t\t// Reduced polygon offset to minimize artifacts\n\t\t// Only use minimal offset to prevent z-fighting on coplanar faces\n\t\tpolygonOffset: true,\n\t\tpolygonOffsetFactor: 0.5,\n\t\tpolygonOffsetUnits: 0.5,\n\t\t// Improve depth rendering\n\t\tdepthWrite: true,\n\t\tdepthTest: true\n\t});\n}\n\n/**\n * Creates a merged mesh from multiple meshes sharing the same material.\n * This is optimal for rendering many small meshes.\n * Optimized to minimize memory allocations and copies.\n */\nfunction createMergedMesh(\n\tgroup: MaterialGroup,\n\tallVertices: Float32Array,\n\tallFaces: Uint32Array,\n\tmaterials: THREE.Material[]\n): THREE.Mesh {\n\tconst geometry = new THREE.BufferGeometry();\n\n\tlet totalVertexFloats = 0;\n\tlet totalFaceIndices = 0;\n\n\tfor (const mesh of group.meshes) {\n\t\ttotalVertexFloats += mesh.vertexCount;\n\t\ttotalFaceIndices += mesh.faceCount;\n\t}\n\n\tconst mergedVertices = new Float32Array(totalVertexFloats);\n\tconst mergedIndices = new Uint32Array(totalFaceIndices);\n\n\tlet vertexWriteOffset = 0;\n\tlet indexWriteOffset = 0;\n\n\tfor (const mesh of group.meshes) {\n\t\tmergedVertices.set(\n\t\t\tallVertices.subarray(mesh.vertexOffset, mesh.vertexOffset + mesh.vertexCount),\n\t\t\tvertexWriteOffset\n\t\t);\n\n\t\tconst faceSlice = allFaces.subarray(mesh.faceOffset, mesh.faceOffset + mesh.faceCount);\n\n\t\t// Face indices are already rebased in the C# batching process\n\t\t// We need to adjust them based on where we're copying the vertices to in the merged array\n\t\tconst originalBaseVertexIndex = Math.floor(mesh.vertexOffset / 3);\n\t\tconst newBaseVertexIndex = Math.floor(vertexWriteOffset / 3);\n\t\tconst indexOffset = newBaseVertexIndex - originalBaseVertexIndex;\n\n\t\tfor (let i = 0; i < faceSlice.length; i++) {\n\t\t\tmergedIndices[indexWriteOffset + i] = faceSlice[i] + indexOffset;\n\t\t}\n\n\t\tvertexWriteOffset += mesh.vertexCount;\n\t\tindexWriteOffset += mesh.faceCount;\n\t}\n\n\tgeometry.setAttribute('position', new THREE.BufferAttribute(mergedVertices, 3));\n\tgeometry.setIndex(new THREE.BufferAttribute(mergedIndices, 1));\n\tgeometry.computeVertexNormals();\n\n\tconst threeMesh = new THREE.Mesh(geometry, materials[group.materialId]);\n\t// Use the first mesh's name, or combine names if multiple meshes\n\tconst meshNames = group.meshes.map((m) => m.name).filter((name) => name && name.length > 0);\n\tthreeMesh.name = meshNames.length > 0 ? meshNames[0] : `merged_material_${group.materialId}`;\n\tthreeMesh.castShadow = true;\n\tthreeMesh.receiveShadow = true;\n\n\tconst allMetadata = group.meshes.map((m) => m.metadata).filter((m) => m);\n\tif (allMetadata.length > 0) {\n\t\tthreeMesh.userData.mergedMetadata = allMetadata;\n\t}\n\n\treturn threeMesh;\n}\n\n/**\n * Creates individual meshes from a material group.\n * This allows independent control of each mesh.\n */\nfunction createIndividualMeshes(\n\tgroup: MaterialGroup,\n\tallVertices: Float32Array,\n\tallFaces: Uint32Array,\n\tmaterials: THREE.Material[]\n): THREE.Mesh[] {\n\tconst meshes: THREE.Mesh[] = [];\n\n\tfor (const meshMeta of group.meshes) {\n\t\tconst geometry = new THREE.BufferGeometry();\n\n\t\tconst vertices = allVertices.subarray(\n\t\t\tmeshMeta.vertexOffset,\n\t\t\tmeshMeta.vertexOffset + meshMeta.vertexCount\n\t\t);\n\n\t\tconst faces = allFaces.subarray(meshMeta.faceOffset, meshMeta.faceOffset + meshMeta.faceCount);\n\n\t\t// Faces are already rebased in C# batching, but we need to rebase them for this\n\t\t// individual mesh since we're using a subarray of vertices starting at 0\n\t\tconst baseIndex = Math.floor(meshMeta.vertexOffset / 3);\n\t\tconst rebasedFaces = new Uint32Array(faces.length);\n\t\tfor (let i = 0; i < faces.length; i++) {\n\t\t\trebasedFaces[i] = faces[i] - baseIndex;\n\t\t}\n\n\t\tgeometry.setAttribute('position', new THREE.BufferAttribute(vertices, 3));\n\t\tgeometry.setIndex(new THREE.BufferAttribute(rebasedFaces, 1));\n\t\tgeometry.computeVertexNormals();\n\n\t\tconst mesh = new THREE.Mesh(geometry, materials[group.materialId]);\n\t\tmesh.name = meshMeta.name;\n\t\tif (meshMeta.metadata) {\n\t\t\tmesh.userData = { ...mesh.userData, ...meshMeta.metadata };\n\t\t}\n\t\tmesh.castShadow = true;\n\t\tmesh.receiveShadow = true;\n\n\t\tmeshes.push(mesh);\n\t}\n\n\treturn meshes;\n}\n\n/**\n * Applies Rhino to Three.js coordinate system transformation.\n * Rhino uses Z-up, Three.js uses Y-up.\n */\nfunction applyCoordinateTransform(vertices: Float32Array): void {\n\tconst cos = Math.cos(-Math.PI / 2);\n\tconst sin = Math.sin(-Math.PI / 2);\n\n\tfor (let i = 0; i < vertices.length; i += 3) {\n\t\tconst x = vertices[i];\n\t\tconst y = vertices[i + 1];\n\t\tconst z = vertices[i + 2];\n\n\t\tvertices[i] = x;\n\t\tvertices[i + 1] = y * cos - z * sin;\n\t\tvertices[i + 2] = y * sin + z * cos;\n\t}\n}\n","import * as fflate from 'fflate';\n\nimport { decodeBase64ToBinary } from '@/core/utils/encoding';\nimport { RhinoComputeError, ErrorCodes } from '@/core/errors';\n\nimport type { DecompressedMeshData } from './types';\n\ninterface MeshData {\n\tverticesArray: Float32Array;\n\tfaceIndicesArray: Uint32Array;\n}\n\n/**\n * Decompresses a base64-encoded string using GZip.\n *\n * @internal Low-level decompression helper — keep internal to `selva-compute`.\n * @param base64String - The base64-encoded string to decompress.\n * @returns The decompressed MeshData.\n * @throws {RhinoComputeError} If decompression fails or data is invalid.\n */\nexport function decompressMeshData(base64String: string): MeshData {\n\ttry {\n\t\tconst bytes = decodeBase64ToBinary(base64String);\n\t\tconst decompressedData = fflate.gunzipSync(bytes);\n\t\treturn parseMeshBinaryData(decompressedData);\n\t} catch (error) {\n\t\tthrow new RhinoComputeError(\n\t\t\terror instanceof RhinoComputeError\n\t\t\t\t? error.message\n\t\t\t\t: `Failed to decompress data: ${error instanceof Error ? error.message : String(error)}`,\n\t\t\terror instanceof RhinoComputeError ? error.code : ErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: { base64StringLength: base64String.length },\n\t\t\t\toriginalError: error instanceof Error ? error : new Error(String(error))\n\t\t\t}\n\t\t);\n\t}\n}\n\n/**\n * Decompresses batched mesh data asynchronously using requestIdleCallback for non-blocking decompression.\n *\n * @internal Low-level decompression helper — keep internal to `selva-compute`.\n * @param base64String - The base64-encoded compressed data.\n * @returns Promise resolving to decompressed vertices and faces arrays.\n * @throws {RhinoComputeError} If decompression fails or data is invalid.\n */\nexport async function decompressBatchedMeshData(\n\tbase64String: string\n): Promise<DecompressedMeshData> {\n\treturn new Promise((resolve, reject) => {\n\t\ttry {\n\t\t\t// Use requestIdleCallback for non-blocking decompression if available\n\t\t\tconst decompressFn = () => {\n\t\t\t\ttry {\n\t\t\t\t\tconst bytes = decodeBase64ToBinary(base64String);\n\t\t\t\t\tconst decompressedData = fflate.gunzipSync(bytes);\n\t\t\t\t\tconst result = parseBatchedMeshBinaryData(decompressedData);\n\t\t\t\t\tresolve(result);\n\t\t\t\t} catch (error) {\n\t\t\t\t\treject(\n\t\t\t\t\t\tnew RhinoComputeError(\n\t\t\t\t\t\t\terror instanceof RhinoComputeError\n\t\t\t\t\t\t\t\t? error.message\n\t\t\t\t\t\t\t\t: `Failed to decompress batched data: ${error instanceof Error ? error.message : String(error)}`,\n\t\t\t\t\t\t\terror instanceof RhinoComputeError ? error.code : ErrorCodes.VALIDATION_ERROR,\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tcontext: { base64StringLength: base64String.length },\n\t\t\t\t\t\t\t\toriginalError: error instanceof Error ? error : new Error(String(error))\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t)\n\t\t\t\t\t);\n\t\t\t\t}\n\t\t\t};\n\n\t\t\tif ('requestIdleCallback' in globalThis) {\n\t\t\t\t(globalThis as any).requestIdleCallback(decompressFn, { timeout: 5000 });\n\t\t\t} else {\n\t\t\t\t// Fallback: use setTimeout with 0 delay to yield to other tasks\n\t\t\t\tsetTimeout(decompressFn, 0);\n\t\t\t}\n\t\t} catch (error) {\n\t\t\treject(\n\t\t\t\tnew RhinoComputeError(\n\t\t\t\t\t`Failed to schedule decompression: ${error instanceof Error ? error.message : String(error)}`,\n\t\t\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t\t\t{ originalError: error instanceof Error ? error : new Error(String(error)) }\n\t\t\t\t)\n\t\t\t);\n\t\t}\n\t});\n}\n\n/**\n * Parses batched binary mesh data (all vertices and faces together).\n * @param binaryMeshData - The binary mesh data to parse.\n * @returns The parsed mesh data with vertices and faces.\n * @throws {RhinoComputeError} If data is invalid or insufficient.\n */\nfunction parseBatchedMeshBinaryData(binaryMeshData: Uint8Array): DecompressedMeshData {\n\tconst dataView = new DataView(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset,\n\t\tbinaryMeshData.byteLength\n\t);\n\tlet offset = 0;\n\n\t// Read vertex data\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of vertex floats.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength, offset } }\n\t\t);\n\t}\n\tconst numVertexFloats = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tif (numVertexFloats % 3 !== 0) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Invalid number of vertex floats; should be divisible by 3.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\tnumVertexFloats,\n\t\t\t\t\tremainder: numVertexFloats % 3,\n\t\t\t\t\ttotalBytes: dataView.byteLength\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst verticesByteLength = numVertexFloats * Float32Array.BYTES_PER_ELEMENT;\n\tif (offset + verticesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read vertices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: verticesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst vertices = new Float32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumVertexFloats\n\t);\n\toffset += verticesByteLength;\n\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength - offset, offset } }\n\t\t);\n\t}\n\tconst numIndices = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tconst indicesByteLength = numIndices * Uint32Array.BYTES_PER_ELEMENT;\n\tif (offset + indicesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: indicesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst faces = new Uint32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumIndices\n\t);\n\n\treturn {\n\t\tvertices,\n\t\tfaces\n\t};\n}\n\n/**\n * Parses binary data and returns mesh data.\n * @param binaryMeshData - The binary mesh data to parse.\n * @returns The parsed mesh data.\n * @throws {RhinoComputeError} If data is invalid or insufficient.\n */\nfunction parseMeshBinaryData(binaryMeshData: Uint8Array): MeshData {\n\tconst dataView = new DataView(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset,\n\t\tbinaryMeshData.byteLength\n\t);\n\tlet offset = 0;\n\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of vertex floats.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength, offset } }\n\t\t);\n\t}\n\tconst numVertexFloats = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tif (numVertexFloats % 3 !== 0) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Invalid number of vertex floats; should be divisible by 3.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { numVertexFloats, remainder: numVertexFloats % 3 } }\n\t\t);\n\t}\n\n\tconst verticesByteLength = numVertexFloats * Float32Array.BYTES_PER_ELEMENT;\n\tif (offset + verticesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read vertices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: verticesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst vertices = new Float32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumVertexFloats\n\t);\n\toffset += verticesByteLength;\n\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength - offset, offset } }\n\t\t);\n\t}\n\tconst numIndices = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tconst indicesByteLength = numIndices * Uint32Array.BYTES_PER_ELEMENT;\n\tif (offset + indicesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: indicesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst faceIndices = new Uint32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumIndices\n\t);\n\n\treturn {\n\t\tverticesArray: vertices,\n\t\tfaceIndicesArray: faceIndices\n\t};\n}\n","import * as THREE from 'three';\n\nimport { applyOffset, computeCombinedBoundingBox } from '../threejs';\nimport { getLogger } from '@/core';\n\nimport { parseMeshBatch } from './batch-parser';\n\nimport type { DataItem, GrasshopperComputeResponse } from '@/features/grasshopper/types';\nimport type { MeshExtractionOptions, MeshBatchParsingOptions } from './types';\n\n// Constants\nexport const SCALE_FACTORS: Record<string, number> = {\n\tMillimeters: 1 / 1000,\n\tCentimeters: 1 / 100,\n\tMeters: 1,\n\tInches: 1 / 39.37,\n\tFeet: 1 / 3.28084\n};\n\nconst DISPLAY_COMPONENT_TYPE = 'Display';\n\n/**\n * Extracts and processes display meshes from a ComputePointerResponse using the Grasshopper WebDisplay component.\n *\n * This is the primary entry point for extracting mesh geometry from Grasshopper compute responses.\n * It handles all aspects of mesh processing: decompression, coordinate transformation, scaling, and positioning.\n *\n * **Note:** Mesh decompression happens asynchronously in a Web Worker to prevent UI blocking.\n *\n * @param data - The ComputePointerResponse containing Grasshopper output trees.\n * @param options - Configuration for mesh extraction and parsing behavior. All options are optional with sensible defaults.\n * @returns Promise resolving to array of THREE.Mesh objects (may be empty).\n * @throws Rethrows unexpected errors after attempting to dispose any created meshes.\n *\n * @remarks\n * - Only works with the WebDisplay component of GHHeadless.\n * - Requires changes to Rhino.Compute (see https://github.com/TheVessen/compute.rhino3d).\n * - Provides a performant way to display mesh data in Three.js.\n * - Decompression is performed in a Web Worker for non-blocking UI updates.\n * - Supports mesh metadata (names, user data) if provided in the compute response.\n *\n * @internal Internal helper: high-level extraction remains public via visualization module, but this\n * function is considered internal implementation detail for mesh extraction.\n *\n * @example\n * ```ts\n * // Simple usage with defaults (all processing enabled)\n * const meshes = await getThreeMeshesFromComputeResponse(response);\n *\n * // With debugging enabled\n * const meshes = await getThreeMeshesFromComputeResponse(response, { debug: true });\n *\n * // With advanced options\n * const meshes = await getThreeMeshesFromComputeResponse(response, {\n * debug: true,\n * allowScaling: true,\n * allowAutoPosition: false,\n * parsing: {\n * mergeByMaterial: false,\n * applyTransforms: true,\n * debug: true,\n * },\n * });\n * ```\n */\nexport async function getThreeMeshesFromComputeResponse(\n\tdata: GrasshopperComputeResponse,\n\toptions?: MeshExtractionOptions\n): Promise<THREE.Mesh[]> {\n\tconst startTime = performance.now();\n\tconst meshes: THREE.Mesh[] = [];\n\n\tconst {\n\t\tallowScaling = true,\n\t\tallowAutoPosition = true,\n\t\tdebug = false,\n\t\tparsing: parsingOptions = {}\n\t} = options ?? {};\n\n\ttry {\n\t\tconst scaleFactor = allowScaling ? getScaleFactor(data.modelunits) : 1;\n\t\tawait extractMeshesFromData(data, meshes, scaleFactor, parsingOptions, debug);\n\n\t\tif (allowAutoPosition) {\n\t\t\tapplyGroundOffset(meshes);\n\t\t}\n\n\t\treturn meshes;\n\t} catch (error) {\n\t\thandleError(error, meshes);\n\t\tthrow error;\n\t} finally {\n\t\tif (debug) {\n\t\t\tlogProcessingTime(startTime);\n\t\t}\n\t}\n}\n\n/**\n * Gets the scale factor for the given unit type.\n */\nfunction getScaleFactor(modelUnits: string): number {\n\treturn SCALE_FACTORS[modelUnits] ?? 1;\n}\n\n/**\n * Extracts meshes from compute response data.\n */\nasync function extractMeshesFromData(\n\tdata: GrasshopperComputeResponse,\n\tmeshes: THREE.Mesh[],\n\tscaleFactor: number,\n\tparsingOptions: MeshBatchParsingOptions,\n\tdebug: boolean\n): Promise<void> {\n\tfor (const value of data.values) {\n\t\tconst innerTree = value.InnerTree as { [key: string]: DataItem[] };\n\n\t\tfor (const path in innerTree) {\n\t\t\tconst branch = innerTree[path];\n\t\t\tif (!branch) continue;\n\n\t\t\tawait processDataBranch(branch, meshes, scaleFactor, parsingOptions, debug);\n\t\t}\n\t}\n}\n\n/**\n * Processes a single data branch to extract MeshBatch display meshes.\n */\nasync function processDataBranch(\n\tbranch: DataItem[],\n\tmeshes: THREE.Mesh[],\n\tscaleFactor: number,\n\tparsingOptions: MeshBatchParsingOptions,\n\tdebug: boolean\n): Promise<void> {\n\tfor (const item of branch) {\n\t\tif (item.type.includes(DISPLAY_COMPONENT_TYPE)) {\n\t\t\tconst mergedParsingOptions = {\n\t\t\t\tmergeByMaterial: true,\n\t\t\t\tapplyTransforms: true,\n\t\t\t\tdebug: false,\n\t\t\t\t...parsingOptions\n\t\t\t};\n\n\t\t\tconst batchMeshes = await parseMeshBatch(item.data, mergedParsingOptions);\n\n\t\t\tif (scaleFactor !== 1) {\n\t\t\t\tfor (const mesh of batchMeshes) {\n\t\t\t\t\tmesh.scale.set(scaleFactor, scaleFactor, scaleFactor);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tmeshes.push(...batchMeshes);\n\n\t\t\tif (debug) {\n\t\t\t\tgetLogger().debug(`Extracted ${batchMeshes.length} meshes from batch`);\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Applies vertical offset to position meshes on the Z=0 plane.\n */\nfunction applyGroundOffset(meshes: THREE.Mesh[]): void {\n\tif (meshes.length === 0) return;\n\n\tconst combinedBoundingBox = computeCombinedBoundingBox(meshes);\n\tconst offsetY = combinedBoundingBox.min.y;\n\tapplyOffset(meshes, offsetY);\n}\n\n/**\n * Handles errors by disposing created meshes and logging.\n */\nfunction handleError(error: unknown, meshes: THREE.Mesh[]): void {\n\tgetLogger().error('An unexpected error occurred:', error);\n\tdisposeMeshes(meshes);\n}\n\n/**\n * Disposes of all meshes and their associated resources.\n */\nfunction disposeMeshes(meshes: THREE.Mesh[]): void {\n\tfor (const mesh of meshes) {\n\t\tif (mesh.geometry) {\n\t\t\tmesh.geometry.dispose();\n\t\t}\n\n\t\tif (mesh.material) {\n\t\t\tif (Array.isArray(mesh.material)) {\n\t\t\t\tmesh.material.forEach((material) => material.dispose());\n\t\t\t} else {\n\t\t\t\tmesh.material.dispose();\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Logs the processing time for mesh extraction.\n */\nfunction logProcessingTime(startTime: number): void {\n\tconst elapsed = performance.now() - startTime;\n\tgetLogger().info('Time to process meshes:', `${elapsed.toFixed(2)}ms`);\n}\n"],"mappings":"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","names":["THREE","OrbitControls","RGBELoader","defaultUp","initThree","canvas","options","config","applyDefaults","scene","createScene","camera","createCamera","renderer","setupRenderer","controls","setupControls","setupEnvironment","setupLighting","addFloor","eventHandlers","setupEventHandlers","resize","disposeResize","setupResponsiveResize","animate","disposeAnimation","createAnimationLoop","sceneUp","object","material","scale","defaults","child","bgColor","animationId","parent","resizeTimeout","resizeObserver","getSize","handleResize","width","height","RGBELoader","envMap","error","getLogger","ambientLight","sunlight","pos","shadowSize","shadowNear","shadowFar","floorSize","floorGeometry","floorColor","floorMaterial","floor","selectedObjects","originalMaterials","raycaster","mouse","mouseDownPosition","fitToView","box","center","size","maxDim","fov","distance","direction","selectionColorObj","clearSelection","obj","handleMouseDown","event","handleCanvasClick","currentMousePosition","rect","intersects","clickedObject","clonedMaterial","handleKeydown","OrbitControls","target","THREE","updateScene","scene","meshes","camera","controls","initialPositionSet","clearScene","unionBoundingBox","mesh","boundingBox","center","size","maxDim","distance","parseColor","colorString","getLogger","trimmed","hex","rgb","c","applyOffset","offsetY","computeCombinedBoundingBox","combinedBoundingBox","clearScene","scene","objectsToRemove","child","object","material","value","three_materials_exports","__export","CONCRETE_MATERIAL","EMISSIVE_MATERIAL","GLASS_MATERIAL","METAL_MATERIAL","PLASTIC_MATERIAL","RUBBER_MATERIAL","WOOD_MATERIAL","THREE","THREE","fflate","init_errors","decompressBatchedMeshData","base64String","resolve","reject","decompressFn","bytes","decodeBase64ToBinary","decompressedData","result","parseBatchedMeshBinaryData","error","RhinoComputeError","ErrorCodes","binaryMeshData","dataView","offset","numVertexFloats","verticesByteLength","vertices","numIndices","indicesByteLength","faces","parseMeshBatch","batchJson","options","mergeByMaterial","applyTransforms","debug","perfStart","parseTime","parseStart","batch","parseMeshBatchObject","error","getLogger","scaleFactor","decompressTime","meshCreateTime","decompressStart","vertices","faces","decompressBatchedMeshData","compressedSizeMB","uncompressedSizeMB","compressionRatio","applyCoordinateTransform","meshCreateStart","materials","createMaterial","meshes","group","mergedMesh","createMergedMesh","individualMeshes","createIndividualMeshes","mesh","totalTime","matData","color","parseColor","allVertices","allFaces","geometry","totalVertexFloats","totalFaceIndices","mergedVertices","mergedIndices","vertexWriteOffset","indexWriteOffset","faceSlice","originalBaseVertexIndex","indexOffset","i","threeMesh","meshNames","m","name","allMetadata","meshMeta","baseIndex","rebasedFaces","cos","sin","x","y","z","SCALE_FACTORS","DISPLAY_COMPONENT_TYPE","getThreeMeshesFromComputeResponse","data","options","startTime","meshes","allowScaling","allowAutoPosition","debug","parsingOptions","scaleFactor","getScaleFactor","extractMeshesFromData","applyGroundOffset","error","handleError","logProcessingTime","modelUnits","value","innerTree","path","branch","processDataBranch","item","mergedParsingOptions","batchMeshes","parseMeshBatch","mesh","getLogger","offsetY","computeCombinedBoundingBox","applyOffset","disposeMeshes","material","elapsed"]}
|
|
1
|
+
{"version":3,"sources":["../src/features/visualization/threejs/three-initializer.ts","../src/features/visualization/threejs/three-helpers.ts","../src/features/visualization/threejs/three-materials.ts","../src/features/visualization/webdisplay/batch-parser.ts","../src/features/visualization/webdisplay/mesh-compression.ts","../src/features/visualization/webdisplay/webdisplay-parser.ts"],"sourcesContent":["import * as THREE from 'three';\nimport { OrbitControls } from 'three/addons/controls/OrbitControls.js';\nimport { RGBELoader } from 'three/addons/loaders/RGBELoader.js';\n\nimport { getLogger } from '@/core';\nimport { ThreeInitializerOptions } from '../types';\n\nconst defaultUp = new THREE.Vector3(0, 0, 1);\n\n/**\n * Initializes a comprehensive Three.js environment with enhanced render quality and flexible configuration.\n *\n * @param canvas - The HTML canvas element to render the scene on.\n * @param options - Configuration options for the Three.js environment.\n * @returns An object containing the scene, camera, controls, renderer, and utility methods.\n */\nexport const initThree = function (\n\tcanvas: HTMLCanvasElement,\n\toptions?: ThreeInitializerOptions\n): {\n\tscene: THREE.Scene;\n\tcamera: THREE.PerspectiveCamera;\n\tcontrols: OrbitControls;\n\trenderer: THREE.WebGLRenderer;\n\tdispose: () => void;\n\tresize: () => void;\n\tfitToView: () => void;\n\tclearSelection: () => void;\n} {\n\tconst config = applyDefaults(options || {});\n\n\t// Initialize core components\n\tconst scene = createScene(config);\n\tconst camera = createCamera(config);\n\tconst renderer = setupRenderer(canvas, config);\n\tconst controls = setupControls(camera, canvas, config);\n\n\t// Setup environment and lighting\n\tsetupEnvironment(scene, config);\n\tsetupLighting(scene, config);\n\n\t// Add floor if enabled\n\tif (config.floor?.enabled) {\n\t\taddFloor(scene, config);\n\t}\n\n\tconst eventHandlers =\n\t\tconfig.events.enableEventHandlers !== false\n\t\t\t? setupEventHandlers(canvas, scene, camera, controls, config)\n\t\t\t: { dispose: () => {}, fitToView: () => {}, clearSelection: () => {} };\n\n\t// Handle resizing\n\tconst { resize, dispose: disposeResize } = setupResponsiveResize(canvas, renderer, camera);\n\n\t// Animation loop\n\tconst { animate, dispose: disposeAnimation } = createAnimationLoop(\n\t\trenderer,\n\t\tscene,\n\t\tcamera,\n\t\tcontrols\n\t);\n\tanimate();\n\n\t// Set scene up vector\n\tconst sceneUp = config.environment?.sceneUp || defaultUp;\n\tscene.up.set(sceneUp.x, sceneUp.y, sceneUp.z);\n\n\t// Comprehensive disposal\n\tconst dispose = () => {\n\t\tdisposeAnimation(); // Stop animation loop\n\t\tdisposeResize(); // Remove resize listeners\n\t\teventHandlers.dispose(); // Remove click/keyboard listeners\n\t\tcontrols.dispose(); // Dispose controls\n\t\trenderer.dispose(); // Dispose renderer\n\n\t\t// Dispose geometries and materials\n\t\tscene.traverse((object) => {\n\t\t\tif (object instanceof THREE.Mesh) {\n\t\t\t\tobject.geometry?.dispose();\n\t\t\t\tif (Array.isArray(object.material)) {\n\t\t\t\t\tobject.material.forEach((material) => material.dispose());\n\t\t\t\t} else {\n\t\t\t\t\tobject.material?.dispose();\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\t};\n\n\treturn {\n\t\tscene,\n\t\tcamera,\n\t\tcontrols,\n\t\trenderer,\n\t\tdispose,\n\t\tresize,\n\t\tfitToView: eventHandlers.fitToView,\n\t\tclearSelection: eventHandlers.clearSelection\n\t};\n};\n\nfunction applyDefaults(options: ThreeInitializerOptions): Required<ThreeInitializerOptions> {\n\tconst scale = options.sceneScale || 'm';\n\n\t// Define sensible defaults for each scale\n\t// Note: All Rhino geometry is normalized to METERS (1 unit = 1 meter), sceneScale just changes the viewing perspective\n\tconst scaleDefaults = {\n\t\tmm: {\n\t\t\t// Geometry scaled UP by 1000x (mm to m conversion for better precision)\n\t\t\tcameraDistance: 20,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 100,\n\t\t\tlightDistance: 10,\n\t\t\tlightHeight: 20,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 100,\n\t\t\tscaleFactor: 1000\n\t\t},\n\t\tcm: {\n\t\t\t// Geometry scaled UP by 100x (cm to m conversion)\n\t\t\tcameraDistance: 20,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 100,\n\t\t\tlightDistance: 25,\n\t\t\tlightHeight: 50,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 100,\n\t\t\tscaleFactor: 100\n\t\t},\n\t\tm: {\n\t\t\t// Natural Three.js scale (1 unit = 1 meter)\n\t\t\tcameraDistance: 10,\n\t\t\tnear: 0.01,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 50,\n\t\t\tlightDistance: 25,\n\t\t\tlightHeight: 50,\n\t\t\tminDistance: 0.001,\n\t\t\tshadowSize: 100,\n\t\t\tscaleFactor: 1\n\t\t},\n\t\tinches: {\n\t\t\t// Geometry scaled UP by ~39.37x (inches to m conversion)\n\t\t\tcameraDistance: 15,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 80,\n\t\t\tlightDistance: 20,\n\t\t\tlightHeight: 40,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 80,\n\t\t\tscaleFactor: 39.37\n\t\t},\n\t\tfeet: {\n\t\t\t// Geometry scaled UP by ~3.28x (feet to m conversion)\n\t\t\tcameraDistance: 8,\n\t\t\tnear: 0.1,\n\t\t\tfar: 2000,\n\t\t\tfloorSize: 40,\n\t\t\tlightDistance: 15,\n\t\t\tlightHeight: 30,\n\t\t\tminDistance: 0.1,\n\t\t\tshadowSize: 60,\n\t\t\tscaleFactor: 3.28084\n\t\t}\n\t};\n\n\tconst defaults = scaleDefaults[scale];\n\n\treturn {\n\t\tsceneScale: scale,\n\t\tcamera: {\n\t\t\tposition:\n\t\t\t\toptions.camera?.position ||\n\t\t\t\tnew THREE.Vector3(\n\t\t\t\t\t-defaults.cameraDistance,\n\t\t\t\t\tdefaults.cameraDistance,\n\t\t\t\t\tdefaults.cameraDistance\n\t\t\t\t),\n\t\t\tfov: options.camera?.fov || 20,\n\t\t\tnear: options.camera?.near || defaults.near,\n\t\t\tfar: options.camera?.far || defaults.far,\n\t\t\ttarget: options.camera?.target || new THREE.Vector3(0, 0, 0)\n\t\t},\n\t\tlighting: {\n\t\t\tenableSunlight: options.lighting?.enableSunlight ?? true,\n\t\t\tsunlightIntensity: options.lighting?.sunlightIntensity || 1,\n\t\t\tsunlightPosition:\n\t\t\t\toptions.lighting?.sunlightPosition ||\n\t\t\t\tnew THREE.Vector3(defaults.lightDistance, defaults.lightHeight, defaults.lightDistance),\n\t\t\tambientLightColor: options.lighting?.ambientLightColor || new THREE.Color(0x404040),\n\t\t\tambientLightIntensity: options.lighting?.ambientLightIntensity || 1,\n\t\t\tsunlightColor: options.lighting?.sunlightColor || 0xffffff // Default to white sunlight\n\t\t},\n\t\tenvironment: {\n\t\t\thdrPath: options.environment?.hdrPath || '/baseHDR.hdr',\n\t\t\tbackgroundColor: options.environment?.backgroundColor || new THREE.Color(0xf0f0f0),\n\t\t\tenableEnvironmentLighting: options.environment?.enableEnvironmentLighting ?? true,\n\t\t\tsceneUp: options.environment?.sceneUp || defaultUp,\n\t\t\tshowEnvironment: options.environment?.showEnvironment ?? false\n\t\t},\n\t\tfloor: {\n\t\t\tenabled: options.floor?.enabled ?? false,\n\t\t\tsize: options.floor?.size || defaults.floorSize,\n\t\t\tcolor: options.floor?.color || new THREE.Color(0x808080),\n\t\t\troughness: options.floor?.roughness || 0.7,\n\t\t\tmetalness: options.floor?.metalness || 0.0,\n\t\t\treceiveShadow: options.floor?.receiveShadow ?? true\n\t\t},\n\t\trender: {\n\t\t\tenableShadows: options.render?.enableShadows ?? true,\n\t\t\tshadowMapSize: options.render?.shadowMapSize || 2048,\n\t\t\tantialias: options.render?.antialias ?? true,\n\t\t\tpixelRatio: options.render?.pixelRatio || Math.min(window.devicePixelRatio, 2),\n\t\t\ttoneMapping: options.render?.toneMapping || THREE.NeutralToneMapping,\n\t\t\ttoneMappingExposure: options.render?.toneMappingExposure || 1,\n\t\t\tpreserveDrawingBuffer: options.render?.preserveDrawingBuffer ?? false\n\t\t},\n\t\tcontrols: {\n\t\t\tenableDamping: options.controls?.enableDamping ?? false,\n\t\t\tdampingFactor: options.controls?.dampingFactor || 0.05,\n\t\t\tautoRotate: options.controls?.autoRotate ?? false,\n\t\t\tautoRotateSpeed: options.controls?.autoRotateSpeed || 0.5,\n\t\t\tenableZoom: options.controls?.enableZoom ?? true,\n\t\t\tenablePan: options.controls?.enablePan ?? true,\n\t\t\tminDistance: options.controls?.minDistance || defaults.minDistance,\n\t\t\tmaxDistance: options.controls?.maxDistance || Infinity\n\t\t},\n\t\tevents: {\n\t\t\tonBackgroundClicked: options.events?.onBackgroundClicked,\n\t\t\tonObjectSelected: options.events?.onObjectSelected,\n\t\t\tonMeshMetadataClicked: options.events?.onMeshMetadataClicked,\n\t\t\tselectionColor: options.events?.selectionColor || '#ff0000', // Default to red\n\t\t\tenableEventHandlers: options.events?.enableEventHandlers ?? true,\n\t\t\tenableKeyboardControls: options.events?.enableKeyboardControls ?? true,\n\t\t\tenableClickToFocus: options.events?.enableClickToFocus ?? true\n\t\t}\n\t};\n}\n\n/**\n * Creates and configures the scene.\n */\nfunction createScene(config: Required<ThreeInitializerOptions>): THREE.Scene {\n\tconst scene = new THREE.Scene();\n\n\t// Clear existing children except floor\n\tscene.children.forEach((child) => {\n\t\tif (child.userData.id !== 'floor') {\n\t\t\tscene.remove(child);\n\t\t}\n\t});\n\n\t// Set background color\n\tconst bgColor =\n\t\ttypeof config.environment.backgroundColor === 'string'\n\t\t\t? new THREE.Color(config.environment.backgroundColor)\n\t\t\t: config.environment.backgroundColor;\n\tscene.background = bgColor || null;\n\n\treturn scene;\n}\n\n/**\n * Creates an optimized animation loop with proper disposal.\n */\nfunction createAnimationLoop(\n\trenderer: THREE.WebGLRenderer,\n\tscene: THREE.Scene,\n\tcamera: THREE.PerspectiveCamera,\n\tcontrols: OrbitControls\n): { animate: () => void; dispose: () => void } {\n\tlet animationId: number | null = null;\n\n\tconst animate = function () {\n\t\tanimationId = requestAnimationFrame(animate);\n\n\t\t// Update controls if damping is enabled\n\t\tif (controls.enableDamping) {\n\t\t\tcontrols.update();\n\t\t}\n\n\t\trenderer.render(scene, camera);\n\t};\n\n\tconst dispose = () => {\n\t\tif (animationId !== null) {\n\t\t\tcancelAnimationFrame(animationId);\n\t\t\tanimationId = null;\n\t\t}\n\t};\n\n\treturn { animate, dispose };\n}\n\n/**\n * Sets up responsive resizing with improved performance and proper parent handling.\n */\nfunction setupResponsiveResize(\n\tcanvas: HTMLCanvasElement,\n\trenderer: THREE.WebGLRenderer,\n\tcamera: THREE.PerspectiveCamera\n): { resize: () => void; dispose: () => void } {\n\tconst parent = canvas.parentElement;\n\tlet resizeTimeout: NodeJS.Timeout;\n\tlet resizeObserver: ResizeObserver | null = null;\n\n\tconst getSize = () => {\n\t\tif (parent) {\n\t\t\treturn {\n\t\t\t\twidth: parent.clientWidth,\n\t\t\t\theight: parent.clientHeight\n\t\t\t};\n\t\t} else {\n\t\t\treturn {\n\t\t\t\twidth: window.innerWidth,\n\t\t\t\theight: window.innerHeight\n\t\t\t};\n\t\t}\n\t};\n\n\tconst handleResize = () => {\n\t\tif (resizeTimeout) {\n\t\t\tclearTimeout(resizeTimeout);\n\t\t}\n\n\t\tresizeTimeout = setTimeout(() => {\n\t\t\tconst { width, height } = getSize();\n\n\t\t\t// Only update if size actually changed\n\t\t\tif (renderer.domElement.width !== width || renderer.domElement.height !== height) {\n\t\t\t\trenderer.setSize(width, height, false);\n\t\t\t\tcamera.aspect = width / height;\n\t\t\t\tcamera.updateProjectionMatrix();\n\t\t\t}\n\t\t}, 16); // ~60fps throttling\n\t};\n\n\t// Set up the appropriate resize observer\n\tif (parent && typeof ResizeObserver !== 'undefined') {\n\t\t// Use ResizeObserver for parent container\n\t\tresizeObserver = new ResizeObserver(handleResize);\n\t\tresizeObserver.observe(parent);\n\t} else {\n\t\t// Fallback to window resize for fullscreen or older browsers\n\t\twindow.addEventListener('resize', handleResize);\n\t}\n\n\tconst dispose = () => {\n\t\tif (resizeTimeout) {\n\t\t\tclearTimeout(resizeTimeout);\n\t\t}\n\t\tif (resizeObserver) {\n\t\t\tresizeObserver.disconnect();\n\t\t} else {\n\t\t\twindow.removeEventListener('resize', handleResize);\n\t\t}\n\t};\n\n\treturn { resize: handleResize, dispose };\n}\n\n/**\n * Sets up environment lighting and HDR.\n */\nfunction setupEnvironment(scene: THREE.Scene, config: Required<ThreeInitializerOptions>) {\n\tif (config.environment.enableEnvironmentLighting) {\n\t\tnew RGBELoader().load(\n\t\t\tconfig.environment.hdrPath || '/baseHDR.hdr',\n\t\t\tfunction (envMap) {\n\t\t\t\tenvMap.mapping = THREE.EquirectangularReflectionMapping;\n\t\t\t\tscene.environment = envMap;\n\t\t\t\tif (config.environment.showEnvironment) {\n\t\t\t\t\tscene.background = envMap;\n\t\t\t\t}\n\t\t\t},\n\t\t\tundefined,\n\t\t\tfunction (error) {\n\t\t\t\tgetLogger().warn('HDR texture could not be loaded, falling back to basic lighting:', error);\n\t\t\t\t// Add basic ambient light as fallback\n\t\t\t\tconst ambientLight = new THREE.AmbientLight(0x404040, 0.4);\n\t\t\t\tscene.add(ambientLight);\n\t\t\t}\n\t\t);\n\t}\n}\n\nfunction setupLighting(scene: THREE.Scene, config: Required<ThreeInitializerOptions>) {\n\t// Add ambient light\n\tconst ambientLight = new THREE.AmbientLight(\n\t\tconfig.lighting.ambientLightColor,\n\t\tconfig.lighting.ambientLightIntensity\n\t);\n\tscene.add(ambientLight);\n\n\t// Add directional light (sunlight)\n\tif (config.lighting.enableSunlight) {\n\t\tconst sunlight = new THREE.DirectionalLight(\n\t\t\tconfig.lighting.sunlightColor ?? 0xffffff,\n\t\t\tconfig.lighting.sunlightIntensity\n\t\t);\n\t\tconst pos = config.lighting.sunlightPosition;\n\t\tif (pos) {\n\t\t\tsunlight.position.set(pos.x, pos.y, pos.z);\n\t\t}\n\n\t\tif (config.render.enableShadows) {\n\t\t\tsunlight.castShadow = true;\n\t\t\tconst shadowSize = config.sceneScale === 'mm' ? 0.1 : config.sceneScale === 'cm' ? 10 : 100;\n\n\t\t\tsunlight.shadow.camera.left = -shadowSize;\n\t\t\tsunlight.shadow.camera.right = shadowSize;\n\t\t\tsunlight.shadow.camera.top = shadowSize;\n\t\t\tsunlight.shadow.camera.bottom = -shadowSize;\n\n\t\t\tconst shadowNear =\n\t\t\t\tconfig.sceneScale === 'mm' ? 0.001 : config.sceneScale === 'cm' ? 0.1 : 0.5;\n\n\t\t\tconst shadowFar = config.sceneScale === 'mm' ? 1 : config.sceneScale === 'cm' ? 100 : 500;\n\n\t\t\tsunlight.shadow.camera.near = shadowNear;\n\t\t\tsunlight.shadow.camera.far = shadowFar;\n\n\t\t\tsunlight.shadow.mapSize.width = config.render.shadowMapSize || 2048;\n\t\t\tsunlight.shadow.mapSize.height = config.render.shadowMapSize || 2048;\n\n\t\t\t// Improved shadow quality\n\t\t\tsunlight.shadow.bias = -0.0001;\n\t\t\tsunlight.shadow.normalBias = 0.02;\n\t\t}\n\n\t\tscene.add(sunlight);\n\t}\n}\n\n/**\n * Adds a floor to the scene with scale-aware sizing.\n */\nfunction addFloor(scene: THREE.Scene, config: Required<ThreeInitializerOptions>) {\n\tconst floorSize = config.floor.size;\n\tconst floorGeometry = new THREE.PlaneGeometry(floorSize, floorSize);\n\n\tconst floorColor =\n\t\ttypeof config.floor.color === 'string'\n\t\t\t? new THREE.Color(config.floor.color)\n\t\t\t: config.floor.color;\n\n\tconst floorMaterial = new THREE.MeshStandardMaterial({\n\t\tcolor: floorColor,\n\t\troughness: config.floor.roughness,\n\t\tmetalness: config.floor.metalness,\n\t\tside: THREE.DoubleSide\n\t});\n\n\tconst floor = new THREE.Mesh(floorGeometry, floorMaterial);\n\tfloor.userData.id = 'floor';\n\tfloor.name = 'floor';\n\tfloor.rotation.x = -Math.PI / 2;\n\tfloor.position.y = 0;\n\n\tif (config.floor.receiveShadow && config.render.enableShadows) {\n\t\tfloor.receiveShadow = true;\n\t}\n\n\tscene.add(floor);\n}\n\n/**\n * Creates and configures the camera with proper aspect ratio.\n */\nfunction createCamera(config: Required<ThreeInitializerOptions>): THREE.PerspectiveCamera {\n\t// Get proper aspect ratio from canvas parent or window\n\tconst canvas = document.querySelector('canvas');\n\tconst parent = canvas?.parentElement;\n\tconst width = parent ? parent.clientWidth : window.innerWidth;\n\tconst height = parent ? parent.clientHeight : window.innerHeight;\n\n\tconst camera = new THREE.PerspectiveCamera(\n\t\tconfig.camera.fov,\n\t\twidth / height,\n\t\tconfig.camera.near,\n\t\tconfig.camera.far\n\t);\n\n\tconst pos = config.camera.position;\n\tif (pos) {\n\t\tcamera.position.set(pos.x, pos.y, pos.z);\n\t}\n\n\treturn camera;\n}\n\n/**\n * Sets up enhanced WebGL renderer with improved quality settings.\n */\nfunction setupRenderer(\n\tcanvas: HTMLCanvasElement,\n\tconfig: Required<ThreeInitializerOptions>\n): THREE.WebGLRenderer {\n\tconst renderer = new THREE.WebGLRenderer({\n\t\tantialias: config.render.antialias,\n\t\tcanvas,\n\t\talpha: true,\n\t\tpowerPreference: 'high-performance',\n\t\tpreserveDrawingBuffer: config.render.preserveDrawingBuffer,\n\t\t// Enable logarithmic depth buffer for extreme scale ranges\n\t\t// This dramatically improves depth precision for mixed scales (mm to km)\n\t\tlogarithmicDepthBuffer: true\n\t});\n\n\t// Get proper dimensions - parent container or window\n\tconst parent = canvas.parentElement;\n\tconst width = parent ? parent.clientWidth : window.innerWidth;\n\tconst height = parent ? parent.clientHeight : window.innerHeight;\n\n\t// Set canvas style to fill parent if it exists\n\tif (parent) {\n\t\tcanvas.style.width = '100%';\n\t\tcanvas.style.height = '100%';\n\t\tcanvas.style.display = 'block';\n\t}\n\n\trenderer.setSize(width, height, false);\n\trenderer.setPixelRatio(config.render.pixelRatio || Math.min(window.devicePixelRatio, 2));\n\n\t// Enhanced shadow settings\n\tif (config.render.enableShadows) {\n\t\trenderer.shadowMap.enabled = true;\n\t\t// Use VSM for better quality with extreme scales\n\t\trenderer.shadowMap.type = THREE.VSMShadowMap;\n\t}\n\n\t// Improved tone mapping and color management\n\trenderer.toneMapping = config.render.toneMapping || THREE.ACESFilmicToneMapping;\n\trenderer.toneMappingExposure = config.render.toneMappingExposure || 1.0;\n\trenderer.outputColorSpace = THREE.SRGBColorSpace;\n\n\t// Additional quality settings for depth rendering\n\trenderer.sortObjects = true; // Ensure proper render order\n\n\treturn renderer;\n}\n\n// Add event handler setup function\nfunction setupEventHandlers(\n\tcanvas: HTMLCanvasElement,\n\tscene: THREE.Scene,\n\tcamera: THREE.PerspectiveCamera,\n\tcontrols: OrbitControls,\n\tconfig: Required<ThreeInitializerOptions>\n): {\n\tdispose: () => void;\n\tfitToView: () => void;\n\tclearSelection: () => void;\n} {\n\tconst selectedObjects = new Set<THREE.Object3D>();\n\tconst originalMaterials = new Map<THREE.Object3D, THREE.Material | THREE.Material[]>();\n\tconst raycaster = new THREE.Raycaster();\n\tconst mouse = new THREE.Vector2();\n\tconst mouseDownPosition = new THREE.Vector2();\n\n\t// Fit scene to view\n\tconst fitToView = () => {\n\t\tconst box = new THREE.Box3();\n\n\t\t// Calculate bounding box of all visible objects (excluding floor)\n\t\tscene.traverse((object) => {\n\t\t\tif (object.visible && object.userData.id !== 'floor' && object instanceof THREE.Mesh) {\n\t\t\t\tbox.expandByObject(object);\n\t\t\t}\n\t\t});\n\n\t\tif (box.isEmpty()) {\n\t\t\tgetLogger().warn('No objects to fit to view');\n\t\t\treturn;\n\t\t}\n\n\t\tconst center = box.getCenter(new THREE.Vector3());\n\t\tconst size = box.getSize(new THREE.Vector3());\n\n\t\t// Calculate distance needed to fit the object\n\t\tconst maxDim = Math.max(size.x, size.y, size.z);\n\t\tconst fov = camera.fov * (Math.PI / 180);\n\t\tlet distance = maxDim / (2 * Math.tan(fov / 2));\n\n\t\t// Add some padding\n\t\tdistance *= 1.5;\n\n\t\t// Position camera\n\t\tconst direction = camera.position.clone().sub(controls.target).normalize();\n\t\tcamera.position.copy(center.clone().add(direction.multiplyScalar(distance)));\n\n\t\t// Update controls target\n\t\tcontrols.target.copy(center);\n\t\tcontrols.update();\n\t};\n\n\t// Parse selection color\n\tconst selectionColorObj =\n\t\ttypeof config.events.selectionColor === 'string'\n\t\t\t? new THREE.Color(config.events.selectionColor)\n\t\t\t: config.events.selectionColor instanceof THREE.Color\n\t\t\t\t? config.events.selectionColor\n\t\t\t\t: new THREE.Color('#ff0000');\n\n\t// Clear selection\n\tconst clearSelection = () => {\n\t\tselectedObjects.forEach((obj) => {\n\t\t\t// Restore original material\n\t\t\tif (obj instanceof THREE.Mesh && originalMaterials.has(obj)) {\n\t\t\t\tobj.material = originalMaterials.get(obj)!;\n\t\t\t\toriginalMaterials.delete(obj);\n\t\t\t}\n\t\t});\n\t\tselectedObjects.clear();\n\t};\n\n\tconst handleMouseDown = (event: MouseEvent) => {\n\t\tmouseDownPosition.set(event.clientX, event.clientY);\n\t};\n\n\t// Handle canvas clicks\n\tconst handleCanvasClick = (event: MouseEvent) => {\n\t\t// Ignore if mouse has moved significantly (drag)\n\t\tconst currentMousePosition = new THREE.Vector2(event.clientX, event.clientY);\n\t\tif (mouseDownPosition.distanceTo(currentMousePosition) > 5) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Calculate mouse position in normalized device coordinates\n\t\tconst rect = canvas.getBoundingClientRect();\n\t\tmouse.x = ((event.clientX - rect.left) / rect.width) * 2 - 1;\n\t\tmouse.y = -((event.clientY - rect.top) / rect.height) * 2 + 1;\n\n\t\t// Raycast to find intersected objects\n\t\traycaster.setFromCamera(mouse, camera);\n\t\tconst intersects = raycaster.intersectObjects(scene.children, true);\n\n\t\tif (intersects.length > 0) {\n\t\t\tconst clickedObject = intersects[0].object;\n\n\t\t\t// Handle object selection\n\t\t\tif (!selectedObjects.has(clickedObject)) {\n\t\t\t\tclearSelection();\n\t\t\t\tselectedObjects.add(clickedObject);\n\n\t\t\t\t// Clone material and apply selection color only to this mesh\n\t\t\t\tif (\n\t\t\t\t\tclickedObject instanceof THREE.Mesh &&\n\t\t\t\t\tclickedObject.material instanceof THREE.Material\n\t\t\t\t) {\n\t\t\t\t\t// Store original material\n\t\t\t\t\toriginalMaterials.set(clickedObject, clickedObject.material);\n\n\t\t\t\t\t// Clone the material so we don't affect other meshes\n\t\t\t\t\tconst clonedMaterial = clickedObject.material.clone();\n\t\t\t\t\t(clonedMaterial as any).emissive = selectionColorObj.clone();\n\t\t\t\t\tclickedObject.material = clonedMaterial;\n\t\t\t\t}\n\n\t\t\t\tconfig.events?.onObjectSelected?.(clickedObject);\n\n\t\t\t\t// Call metadata callback if the mesh has metadata\n\t\t\t\tif (clickedObject instanceof THREE.Mesh && Object.keys(clickedObject.userData).length > 0) {\n\t\t\t\t\tconfig.events?.onMeshMetadataClicked?.(clickedObject.userData);\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\t// Background clicked\n\t\t\tclearSelection();\n\t\t\tconfig.events?.onBackgroundClicked?.({ x: mouse.x, y: mouse.y });\n\t\t}\n\t};\n\n\t// Handle keyboard events\n\tconst handleKeydown = (event: KeyboardEvent) => {\n\t\tif (!config.events?.enableKeyboardControls) return;\n\n\t\tswitch (event.key.toLowerCase()) {\n\t\t\tcase 'f':\n\t\t\t\tevent.preventDefault();\n\t\t\t\tfitToView();\n\t\t\t\tbreak;\n\t\t\tcase 'escape':\n\t\t\t\tevent.preventDefault();\n\t\t\t\tclearSelection();\n\t\t\t\tbreak;\n\t\t\tcase ' ':\n\t\t\t\tevent.preventDefault();\n\t\t\t\tfitToView();\n\t\t\t\tbreak;\n\t\t}\n\t};\n\n\t// Add event listeners\n\tif (config.events?.enableClickToFocus) {\n\t\tcanvas.addEventListener('mousedown', handleMouseDown);\n\t\tcanvas.addEventListener('click', handleCanvasClick);\n\t}\n\n\tif (config.events?.enableKeyboardControls) {\n\t\t// Make canvas focusable\n\t\tcanvas.setAttribute('tabindex', '0');\n\t\t// Only listen for keydown when canvas has focus\n\t\tcanvas.addEventListener('keydown', handleKeydown);\n\t}\n\n\t// Disposal function\n\tconst dispose = () => {\n\t\tcanvas.removeEventListener('mousedown', handleMouseDown);\n\t\tcanvas.removeEventListener('click', handleCanvasClick);\n\t\tcanvas.removeEventListener('keydown', handleKeydown);\n\t\tclearSelection();\n\t};\n\n\treturn { dispose, fitToView, clearSelection };\n}\n\n/**\n * Sets up enhanced orbit controls with scale-aware distances.\n */\nfunction setupControls(\n\tcamera: THREE.PerspectiveCamera,\n\tcanvas: HTMLCanvasElement,\n\tconfig: Required<ThreeInitializerOptions>\n): OrbitControls {\n\tconst controls = new OrbitControls(camera, canvas);\n\n\t// Set target\n\tconst target = config.camera.target;\n\tif (target) {\n\t\tcontrols.target.set(target.x, target.y, target.z);\n\t}\n\n\t// Configure damping\n\tcontrols.enableDamping = config.controls.enableDamping || false;\n\tcontrols.dampingFactor = config.controls.dampingFactor || 0.05;\n\n\t// Configure auto rotation\n\tcontrols.autoRotate = config.controls.autoRotate || false;\n\tcontrols.autoRotateSpeed = config.controls.autoRotateSpeed || 0.5;\n\n\t// Configure interaction limits\n\tcontrols.enableZoom = config.controls.enableZoom || true;\n\tcontrols.enablePan = config.controls.enablePan || true;\n\tcontrols.minDistance = config.controls.minDistance || 0.001;\n\tcontrols.maxDistance = config.controls.maxDistance || Infinity;\n\n\t// Smooth controls\n\tcontrols.screenSpacePanning = false;\n\tcontrols.maxPolarAngle = Math.PI;\n\n\tcontrols.update();\n\treturn controls;\n}\n","import * as THREE from 'three';\nimport { OrbitControls } from 'three/addons/controls/OrbitControls.js';\nimport { getLogger } from '@/core';\n\n// Camera configuration constants\nconst CAMERA_CONFIG = {\n\tHUGE_THRESHOLD: 10000,\n\tLARGE_THRESHOLD: 1000,\n\tSCALE_RATIO_THRESHOLD: 100,\n\tNEAR_PLANE_FACTOR: {\n\t\tTINY: 0.0001,\n\t\tSMALL: 0.001,\n\t\tNORMAL: 0.01,\n\t},\n\tFAR_PLANE_FACTOR: {\n\t\tHUGE: 100,\n\t\tLARGE: 50,\n\t\tNORMAL: 20,\n\t},\n\tInitialDistanceMultiplier: 4,\n};\n\n/**\n * Updates the scene with the given meshes and camera settings.\n * If initialPositionSet is false, it positions the camera and sets the controls target based on the bounding boxes of the meshes.\n * @param scene - The THREE.Scene object to update.\n * @param meshes - An array of THREE.Mesh objects to add to the scene.\n * @param camera - The THREE.PerspectiveCamera object to position.\n * @param controls - The OrbitControls object to update.\n * @param initialPositionSet - A boolean indicating whether the initial position of the camera and controls have been set.\n */\nexport function updateScene(\n\tscene: THREE.Scene,\n\tmeshes: THREE.Mesh[],\n\tcamera: THREE.PerspectiveCamera,\n\tcontrols: OrbitControls,\n\tinitialPositionSet: boolean\n) {\n\tclearScene(scene);\n\n\tif (meshes.length === 0) return;\n\n\t// Add new meshes to scene\n\tmeshes.forEach((mesh) => {\n\t\tscene.add(mesh);\n\t});\n\n\t// Calculate bounds of the new content\n\tconst unionBoundingBox = computeCombinedBoundingBox(meshes);\n\n\t// Get the center of the union bounding box\n\tconst center = unionBoundingBox.getCenter(new THREE.Vector3());\n\tconst size = unionBoundingBox.getSize(new THREE.Vector3());\n\n\t// Calculate a distance that is slightly larger than the largest dimension of the union bounding box\n\tconst maxDim = Math.max(size.x, size.y, size.z);\n\n\t// Always update camera frustum to ensure geometry is visible\n\t// This prevents clipping when geometry size changes significantly\n\tconst scaleRatio = maxDim / Math.min(size.x || 1, size.y || 1, size.z || 1);\n\n\tif (scaleRatio > CAMERA_CONFIG.SCALE_RATIO_THRESHOLD || maxDim > CAMERA_CONFIG.HUGE_THRESHOLD) {\n\t\t// Large scale range detected - use logarithmic depth buffer approach\n\t\tcamera.near = maxDim * CAMERA_CONFIG.NEAR_PLANE_FACTOR.TINY;\n\t\tcamera.far = maxDim * CAMERA_CONFIG.FAR_PLANE_FACTOR.HUGE;\n\t} else if (maxDim > CAMERA_CONFIG.LARGE_THRESHOLD) {\n\t\t// Large scene\n\t\tcamera.near = maxDim * CAMERA_CONFIG.NEAR_PLANE_FACTOR.SMALL;\n\t\tcamera.far = maxDim * CAMERA_CONFIG.FAR_PLANE_FACTOR.LARGE;\n\t} else {\n\t\t// Normal scene\n\t\tcamera.near = Math.max(0.01, maxDim * CAMERA_CONFIG.NEAR_PLANE_FACTOR.NORMAL);\n\t\tcamera.far = Math.max(2000, maxDim * CAMERA_CONFIG.FAR_PLANE_FACTOR.NORMAL);\n\t}\n\n\tcamera.updateProjectionMatrix();\n\n\t// Only reposition camera and controls on first frame\n\tif (!initialPositionSet) {\n\t\tconst distance = maxDim * CAMERA_CONFIG.InitialDistanceMultiplier;\n\n\t\tcamera.position.set(center.x + distance * 0.8, center.y + distance, center.z + distance * 1.2);\n\t\tcontrols.target.copy(center);\n\t\tcontrols.minDistance = camera.near * 2;\n\t\tcontrols.maxDistance = camera.far * 0.9;\n\n\t\tcontrols.update();\n\t} else {\n\t\t// Update control constraints to match new frustum\n\t\tcontrols.minDistance = camera.near * 2;\n\t\tcontrols.maxDistance = camera.far * 0.9;\n\t}\n}\n\n// =========================\n// Helper functions\n// =========================\n\n/**\n * Parses a color string in multiple formats to a THREE.Color object.\n * Supported formats:\n * - Hex: \"#C7A5A5\", \"C7A5A5\"\n * - RGB: \"199, 165, 165\"\n * - CSS named colors: \"red\", \"blue\", etc.\n * @param colorString - The color string to parse.\n * @returns A THREE.Color object.\n */\nexport function parseColor(colorString: string): THREE.Color {\n\tif (!colorString || typeof colorString !== 'string') {\n\t\tgetLogger().warn(`Invalid color input: ${colorString}, using white`);\n\t\treturn new THREE.Color(0xffffff);\n\t}\n\n\tconst trimmed = colorString.trim();\n\n\t// Try hex format (#C7A5A5 or C7A5A5)\n\tif (trimmed.startsWith('#') || /^[0-9A-Fa-f]{6}$/.test(trimmed)) {\n\t\ttry {\n\t\t\tconst hex = trimmed.startsWith('#') ? trimmed : `#${trimmed}`;\n\t\t\treturn new THREE.Color(hex);\n\t\t} catch {\n\t\t\tgetLogger().warn(`Invalid hex color: ${colorString}, using white`);\n\t\t\treturn new THREE.Color(0xffffff);\n\t\t}\n\t}\n\n\t// Try RGB format (R, G, B)\n\tif (trimmed.includes(',')) {\n\t\tconst rgb = trimmed.split(',').map((c) => parseInt(c.trim(), 10));\n\t\tif (rgb.length === 3 && rgb.every((n) => !isNaN(n) && n >= 0 && n <= 255)) {\n\t\t\t// THREE.Color constructor accepts r, g, b in range [0, 1]\n\t\t\treturn new THREE.Color(rgb[0] / 255, rgb[1] / 255, rgb[2] / 255);\n\t\t}\n\t}\n\n\t// Try CSS named color\n\ttry {\n\t\treturn new THREE.Color(trimmed.toLowerCase());\n\t} catch {\n\t\tgetLogger().warn(`Invalid color string: ${colorString}, using white`);\n\t\treturn new THREE.Color(0xffffff);\n\t}\n}\n\nexport function applyOffset(meshes: THREE.Mesh[], offsetY: number): void {\n\tmeshes.forEach((mesh) => {\n\t\tmesh.position.y -= offsetY;\n\t\t// Ensure world matrix is pending update if needed\n\t\tmesh.updateMatrix();\n\t});\n}\n\n/**\n * Computes the combined AI world-axis-aligned bounding box of a set of meshes.\n * Correctly accounts for mesh transformations (rotation, position, scale).\n */\nexport function computeCombinedBoundingBox(meshes: THREE.Mesh[]): THREE.Box3 {\n\tconst combinedBoundingBox = new THREE.Box3();\n\tmeshes.forEach((mesh) => {\n\t\t// Ensure the world matrix is up to date before calculating the box\n\t\tmesh.updateMatrixWorld(true);\n\t\tconst bbox = new THREE.Box3().setFromObject(mesh);\n\t\tcombinedBoundingBox.union(bbox);\n\t});\n\treturn combinedBoundingBox;\n}\n\n/**\n * Updates shadow camera bounds to match scene geometry.\n * This prevents shadow artifacts and ensures proper shadow coverage.\n * @param scene - The scene containing geometry.\n * @param directionalLight - The light to update.\n * @param optionalBounds - Optional pre-calculated bounds to avoid scene traversal.\n */\nexport function updateShadowCameraBounds(\n\tscene: THREE.Scene,\n\tdirectionalLight: THREE.DirectionalLight,\n\toptionalBounds?: THREE.Box3\n): void {\n\tlet bbox = optionalBounds;\n\n\tif (!bbox) {\n\t\tbbox = new THREE.Box3();\n\t\tscene.traverse((object) => {\n\t\t\tif (object instanceof THREE.Mesh && object.userData.id !== 'floor') {\n\t\t\t\t// Use setFromObject to ensure world transforms are respected\n\t\t\t\tconst objBbox = new THREE.Box3().setFromObject(object);\n\t\t\t\tbbox!.union(objBbox);\n\t\t\t}\n\t\t});\n\t}\n\n\tif (!bbox || bbox.isEmpty()) return;\n\n\tconst size = bbox.getSize(new THREE.Vector3());\n\tconst center = bbox.getCenter(new THREE.Vector3());\n\tconst maxDim = Math.max(size.x, size.y, size.z);\n\n\t// Position light relative to scene center\n\tconst lightDistance = maxDim * 2;\n\tdirectionalLight.position.set(\n\t\tcenter.x + lightDistance * 0.5,\n\t\tcenter.y + lightDistance,\n\t\tcenter.z + lightDistance * 0.5\n\t);\n\tdirectionalLight.target.position.copy(center);\n\n\t// Adjust shadow camera bounds to scene size with padding\n\tconst padding = maxDim * 0.2;\n\tdirectionalLight.shadow.camera.left = -maxDim / 2 - padding;\n\tdirectionalLight.shadow.camera.right = maxDim / 2 + padding;\n\tdirectionalLight.shadow.camera.top = maxDim / 2 + padding;\n\tdirectionalLight.shadow.camera.bottom = -maxDim / 2 - padding;\n\tdirectionalLight.shadow.camera.near = 0.1;\n\tdirectionalLight.shadow.camera.far = lightDistance * 3;\n\n\t// Improve shadow quality for extreme scales\n\tif (maxDim > CAMERA_CONFIG.LARGE_THRESHOLD) {\n\t\tdirectionalLight.shadow.bias = -0.001;\n\t\tdirectionalLight.shadow.normalBias = 0.05;\n\t} else {\n\t\tdirectionalLight.shadow.bias = -0.0001;\n\t\tdirectionalLight.shadow.normalBias = 0.02;\n\t}\n\n\tdirectionalLight.shadow.camera.updateProjectionMatrix();\n}\n\n/**\n * Clears the given THREE.Scene by removing all meshes and disposing of associated resources.\n * @param scene - The THREE.Scene to clear.\n */\nfunction clearScene(scene: THREE.Scene): void {\n\tconst objectsToRemove: THREE.Object3D[] = [];\n\n\t// Collect all meshes except the floor\n\tscene.traverse((child: THREE.Object3D) => {\n\t\tif (child instanceof THREE.Mesh && child.userData.id !== 'floor') {\n\t\t\tobjectsToRemove.push(child);\n\t\t}\n\t});\n\n\t// Remove and dispose of each object\n\tobjectsToRemove.forEach((object: THREE.Object3D) => {\n\t\tif (object instanceof THREE.Mesh) {\n\t\t\tobject.geometry?.dispose();\n\n\t\t\tconst materials = Array.isArray(object.material) ? object.material : [object.material];\n\t\t\tmaterials.forEach((material) => {\n\t\t\t\t// Dispose textures first\n\t\t\t\tfor (const key in material) {\n\t\t\t\t\tconst value = material[key as keyof THREE.Material];\n\t\t\t\t\tif (value && value instanceof THREE.Texture) {\n\t\t\t\t\t\tvalue.dispose();\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\tmaterial.dispose();\n\t\t\t});\n\t\t}\n\n\t\tobject.removeFromParent();\n\t});\n}\n","import * as THREE from 'three';\n\nexport const EMISSIVE_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: 0x000000,\n\temissive: new THREE.Color(0xffffff),\n\temissiveIntensity: 5,\n\tmetalness: 0.0,\n\troughness: 0.2,\n\tclearcoat: 0.3,\n\tclearcoatRoughness: 0.2,\n\tdepthWrite: true,\n\tdepthTest: true,\n\ttransparent: false,\n\talphaTest: 0.0,\n\tpolygonOffset: true,\n\tside: THREE.FrontSide,\n\tdithering: true\n});\n\nexport const METAL_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0x000000),\n\tmetalness: 0.9,\n\troughness: 0.3,\n\tenvMapIntensity: 1.2,\n\tclearcoat: 0.3,\n\tclearcoatRoughness: 0.2,\n\treflectivity: 1,\n\tior: 2.5,\n\tthickness: 1,\n\tdepthWrite: true,\n\ttransparent: false,\n\talphaTest: 0.0,\n\tdepthTest: true,\n\tpolygonOffset: true,\n\tside: THREE.FrontSide,\n\tdithering: true\n});\n\nexport const CONCRETE_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0xcccccc),\n\tmetalness: 0.0,\n\troughness: 0.92,\n\tenvMapIntensity: 0.15,\n\tclearcoat: 0.05,\n\tclearcoatRoughness: 0.9,\n\treflectivity: 0.15,\n\ttransmission: 0.0,\n\tior: 1.45,\n\tthickness: 0.0,\n\tdepthWrite: true,\n\ttransparent: false,\n\talphaTest: 0.5,\n\tdepthTest: true,\n\tpolygonOffset: true,\n\tside: THREE.FrontSide,\n\tdithering: true\n});\n\nexport const PLASTIC_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0xffffff), // Default white plastic\n\tmetalness: 0.0,\n\troughness: 0.3,\n\tenvMapIntensity: 0.5,\n\tclearcoat: 0.5,\n\tclearcoatRoughness: 0.1,\n\treflectivity: 0.5,\n\tior: 1.4,\n\ttransmission: 0.0,\n\ttransparent: false,\n\tdepthWrite: true,\n\tside: THREE.FrontSide,\n\tdithering: true,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n\nexport const GLASS_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0xffffff),\n\tmetalness: 0.0,\n\troughness: 0.0,\n\ttransmission: 0.95,\n\ttransparent: true,\n\topacity: 0.3,\n\tenvMapIntensity: 1.0,\n\tclearcoat: 1.0,\n\tclearcoatRoughness: 0.0,\n\tior: 1.52,\n\treflectivity: 0.9,\n\tthickness: 1.0,\n\tside: THREE.DoubleSide,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n\nexport const RUBBER_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0x1a1a1a),\n\tmetalness: 0.0,\n\troughness: 0.9,\n\tenvMapIntensity: 0.2,\n\tclearcoat: 0.1,\n\tclearcoatRoughness: 0.8,\n\treflectivity: 0.2,\n\tior: 1.3,\n\ttransmission: 0.0,\n\tdepthWrite: true,\n\tside: THREE.FrontSide,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n\nexport const WOOD_MATERIAL = new THREE.MeshPhysicalMaterial({\n\tcolor: new THREE.Color(0x885533),\n\tmetalness: 0.0,\n\troughness: 0.7,\n\tenvMapIntensity: 0.3,\n\tclearcoat: 0.3,\n\tclearcoatRoughness: 0.4,\n\treflectivity: 0.3,\n\tior: 1.3,\n\ttransmission: 0.0,\n\tdepthWrite: true,\n\tside: THREE.FrontSide,\n\tdithering: true,\n\tpolygonOffset: true,\n\tpolygonOffsetFactor: 1,\n\tpolygonOffsetUnits: 1\n});\n","import * as THREE from 'three';\n\nimport { parseColor } from '../threejs/three-helpers';\nimport { getLogger } from '@/core';\n\nimport { decompressBatchedMeshData } from './mesh-compression';\n\nimport type { MeshBatch, MaterialGroup, SerializableMaterial } from './types';\n\n/**\n * Parses a batched mesh JSON and creates Three.js meshes.\n *\n * This function handles the optimized batch format where:\n * - Materials are deduplicated and stored once\n * - Meshes are grouped by material for efficient rendering\n * - All geometry data is compressed together and decompressed in a Web Worker\n *\n * @internal Low-level mesh parsing — keep internal to `selva-compute`.\n *\n * @param batchJson - JSON string containing the batched mesh data\n * @param options - Rendering options\n * @returns Promise resolving to array of Three.js mesh objects\n */\nexport async function parseMeshBatch(\n\tbatchJson: string,\n\toptions?: {\n\t\t/** Merge meshes with same material into single geometry*/\n\t\tmergeByMaterial?: boolean;\n\t\t/** Apply coordinate system transformations */\n\t\tapplyTransforms?: boolean;\n\t\t/** Enable performance monitoring */\n\t\tdebug?: boolean;\n\t}\n): Promise<THREE.Mesh[]> {\n\tconst { mergeByMaterial = true, applyTransforms = true, debug = false } = options ?? {};\n\n\tconst perfStart = debug ? performance.now() : 0;\n\tlet parseTime = 0;\n\n\ttry {\n\t\tconst parseStart = performance.now();\n\t\tconst batch: MeshBatch = JSON.parse(batchJson);\n\t\tparseTime = performance.now() - parseStart;\n\n\t\treturn await parseMeshBatchObject(batch, {\n\t\t\tmergeByMaterial,\n\t\t\tapplyTransforms,\n\t\t\tdebug,\n\t\t\tparseTime,\n\t\t\tperfStart\n\t\t});\n\t} catch (error) {\n\t\tgetLogger().error('Error parsing mesh batch:', error);\n\t\treturn [];\n\t}\n}\n\n/**\n * Parses a MeshBatch object and creates Three.js meshes.\n * This is useful when you already have a deserialized MeshBatch object.\n *\n * @internal Low-level mesh parsing — keep internal to `selva-compute`.\n *\n * @param batch - MeshBatch object\n * @param options - Rendering options\n * @returns Promise resolving to array of Three.js mesh objects\n */\nexport async function parseMeshBatchObject(\n\tbatch: MeshBatch,\n\toptions?: {\n\t\t/** Merge meshes with same material into single geometry*/\n\t\tmergeByMaterial?: boolean;\n\t\t/** Apply coordinate system transformations */\n\t\tapplyTransforms?: boolean;\n\t\t/** Scale factor to apply to meshes (e.g., for unit conversion) */\n\t\tscaleFactor?: number;\n\t\t/** Enable performance monitoring */\n\t\tdebug?: boolean;\n\t\t/** Parse time (optional, for debugging) */\n\t\tparseTime?: number;\n\t\t/** Performance start time (optional, for debugging) */\n\t\tperfStart?: number;\n\t}\n): Promise<THREE.Mesh[]> {\n\tconst {\n\t\tmergeByMaterial = true,\n\t\tapplyTransforms = true,\n\t\tscaleFactor = 1,\n\t\tdebug = false,\n\t\tparseTime = 0,\n\t\tperfStart = debug ? performance.now() : 0\n\t} = options ?? {};\n\n\tlet decompressTime = 0,\n\t\tmeshCreateTime = 0;\n\n\ttry {\n\t\tconst decompressStart = performance.now();\n\t\tconst { vertices, faces } = await decompressBatchedMeshData(batch.compressedData);\n\t\tdecompressTime = performance.now() - decompressStart;\n\n\t\tconst compressedSizeMB = ((batch.compressedData.length * 0.75) / 1024 / 1024).toFixed(2); // Base64 overhead\n\t\tconst uncompressedSizeMB = ((vertices.byteLength + faces.byteLength) / 1024 / 1024).toFixed(2);\n\t\tconst compressionRatio = (\n\t\t\t(1 - parseFloat(compressedSizeMB) / parseFloat(uncompressedSizeMB)) *\n\t\t\t100\n\t\t).toFixed(1);\n\n\t\tif (debug) {\n\t\t\tgetLogger().debug('Mesh Batch Stats:');\n\t\t\tgetLogger().debug(` Materials: ${batch.materials.length} | Groups: ${batch.groups.length}`);\n\t\t\tgetLogger().debug(\n\t\t\t\t` Vertices: ${(vertices.length / 3).toLocaleString()} | Faces: ${(faces.length / 3).toLocaleString()}`\n\t\t\t);\n\t\t\tgetLogger().debug(\n\t\t\t\t` Compressed: ${compressedSizeMB} MB | Uncompressed: ${uncompressedSizeMB} MB`\n\t\t\t);\n\t\t\tgetLogger().debug(` Compression Ratio: ${compressionRatio}%`);\n\t\t}\n\n\t\tif (applyTransforms) {\n\t\t\tapplyCoordinateTransform(vertices);\n\t\t}\n\n\t\tconst meshCreateStart = performance.now();\n\t\tconst materials = batch.materials.map(createMaterial);\n\n\t\tconst meshes: THREE.Mesh[] = [];\n\n\t\tfor (const group of batch.groups) {\n\t\t\tif (mergeByMaterial && group.meshes.length > 1) {\n\t\t\t\tconst mergedMesh = createMergedMesh(group, vertices, faces, materials);\n\t\t\t\tmeshes.push(mergedMesh);\n\t\t\t} else {\n\t\t\t\tconst individualMeshes = createIndividualMeshes(group, vertices, faces, materials);\n\t\t\t\tmeshes.push(...individualMeshes);\n\t\t\t}\n\t\t}\n\n\t\t// Apply scaling if needed\n\t\tif (scaleFactor !== 1) {\n\t\t\tfor (const mesh of meshes) {\n\t\t\t\tmesh.scale.set(scaleFactor, scaleFactor, scaleFactor);\n\t\t\t}\n\t\t}\n\n\t\tmeshCreateTime = performance.now() - meshCreateStart;\n\n\t\tif (debug) {\n\t\t\tconst totalTime = performance.now() - perfStart;\n\t\t\tgetLogger().debug('Performance:');\n\t\t\tif (parseTime > 0) getLogger().debug(` Parse JSON: ${parseTime.toFixed(2)}ms`);\n\t\t\tgetLogger().debug(` Decompress: ${decompressTime.toFixed(2)}ms`);\n\t\t\tgetLogger().debug(` Create Meshes: ${meshCreateTime.toFixed(2)}ms`);\n\t\t\tgetLogger().debug(` Total: ${totalTime.toFixed(2)}ms`);\n\t\t}\n\n\t\treturn meshes;\n\t} catch (error) {\n\t\tgetLogger().error('Error parsing mesh batch object:', error);\n\t\treturn [];\n\t}\n}\n\n/**\n * Creates a Three.js material from serializable material data.\n */\nfunction createMaterial(matData: SerializableMaterial): THREE.MeshPhysicalMaterial {\n\tconst color = parseColor(matData.color);\n\n\treturn new THREE.MeshPhysicalMaterial({\n\t\tcolor,\n\t\tmetalness: matData.metalness,\n\t\troughness: matData.roughness,\n\t\topacity: matData.opacity,\n\t\ttransparent: matData.transparent,\n\t\tside: THREE.DoubleSide,\n\t\t// Reduced polygon offset to minimize artifacts\n\t\t// Only use minimal offset to prevent z-fighting on coplanar faces\n\t\tpolygonOffset: true,\n\t\tpolygonOffsetFactor: 0.5,\n\t\tpolygonOffsetUnits: 0.5,\n\t\t// Improve depth rendering\n\t\tdepthWrite: true,\n\t\tdepthTest: true\n\t});\n}\n\n/**\n * Creates a merged mesh from multiple meshes sharing the same material.\n * This is optimal for rendering many small meshes.\n * Optimized to minimize memory allocations and copies.\n */\nfunction createMergedMesh(\n\tgroup: MaterialGroup,\n\tallVertices: Float32Array,\n\tallFaces: Uint32Array,\n\tmaterials: THREE.Material[]\n): THREE.Mesh {\n\tconst geometry = new THREE.BufferGeometry();\n\n\tlet totalVertexFloats = 0;\n\tlet totalFaceIndices = 0;\n\n\tfor (const mesh of group.meshes) {\n\t\ttotalVertexFloats += mesh.vertexCount;\n\t\ttotalFaceIndices += mesh.faceCount;\n\t}\n\n\tconst mergedVertices = new Float32Array(totalVertexFloats);\n\tconst mergedIndices = new Uint32Array(totalFaceIndices);\n\n\tlet vertexWriteOffset = 0;\n\tlet indexWriteOffset = 0;\n\n\tfor (const mesh of group.meshes) {\n\t\tmergedVertices.set(\n\t\t\tallVertices.subarray(mesh.vertexOffset, mesh.vertexOffset + mesh.vertexCount),\n\t\t\tvertexWriteOffset\n\t\t);\n\n\t\tconst faceSlice = allFaces.subarray(mesh.faceOffset, mesh.faceOffset + mesh.faceCount);\n\n\t\t// Face indices are already rebased in the C# batching process\n\t\t// We need to adjust them based on where we're copying the vertices to in the merged array\n\t\tconst originalBaseVertexIndex = Math.floor(mesh.vertexOffset / 3);\n\t\tconst newBaseVertexIndex = Math.floor(vertexWriteOffset / 3);\n\t\tconst indexOffset = newBaseVertexIndex - originalBaseVertexIndex;\n\n\t\tfor (let i = 0; i < faceSlice.length; i++) {\n\t\t\tmergedIndices[indexWriteOffset + i] = faceSlice[i] + indexOffset;\n\t\t}\n\n\t\tvertexWriteOffset += mesh.vertexCount;\n\t\tindexWriteOffset += mesh.faceCount;\n\t}\n\n\tgeometry.setAttribute('position', new THREE.BufferAttribute(mergedVertices, 3));\n\tgeometry.setIndex(new THREE.BufferAttribute(mergedIndices, 1));\n\tgeometry.computeVertexNormals();\n\n\tconst threeMesh = new THREE.Mesh(geometry, materials[group.materialId]);\n\t// Use the first mesh's name, or combine names if multiple meshes\n\tconst meshNames = group.meshes.map((m) => m.name).filter((name) => name && name.length > 0);\n\tthreeMesh.name = meshNames.length > 0 ? meshNames[0] : `merged_material_${group.materialId}`;\n\tthreeMesh.castShadow = true;\n\tthreeMesh.receiveShadow = true;\n\n\tconst allMetadata = group.meshes.map((m) => m.metadata).filter((m) => m);\n\tif (allMetadata.length > 0) {\n\t\tthreeMesh.userData.mergedMetadata = allMetadata;\n\t}\n\n\treturn threeMesh;\n}\n\n/**\n * Creates individual meshes from a material group.\n * This allows independent control of each mesh.\n */\nfunction createIndividualMeshes(\n\tgroup: MaterialGroup,\n\tallVertices: Float32Array,\n\tallFaces: Uint32Array,\n\tmaterials: THREE.Material[]\n): THREE.Mesh[] {\n\tconst meshes: THREE.Mesh[] = [];\n\n\tfor (const meshMeta of group.meshes) {\n\t\tconst geometry = new THREE.BufferGeometry();\n\n\t\tconst vertices = allVertices.subarray(\n\t\t\tmeshMeta.vertexOffset,\n\t\t\tmeshMeta.vertexOffset + meshMeta.vertexCount\n\t\t);\n\n\t\tconst faces = allFaces.subarray(meshMeta.faceOffset, meshMeta.faceOffset + meshMeta.faceCount);\n\n\t\t// Faces are already rebased in C# batching, but we need to rebase them for this\n\t\t// individual mesh since we're using a subarray of vertices starting at 0\n\t\tconst baseIndex = Math.floor(meshMeta.vertexOffset / 3);\n\t\tconst rebasedFaces = new Uint32Array(faces.length);\n\t\tfor (let i = 0; i < faces.length; i++) {\n\t\t\trebasedFaces[i] = faces[i] - baseIndex;\n\t\t}\n\n\t\tgeometry.setAttribute('position', new THREE.BufferAttribute(vertices, 3));\n\t\tgeometry.setIndex(new THREE.BufferAttribute(rebasedFaces, 1));\n\t\tgeometry.computeVertexNormals();\n\n\t\tconst mesh = new THREE.Mesh(geometry, materials[group.materialId]);\n\t\tmesh.name = meshMeta.name;\n\t\tif (meshMeta.metadata) {\n\t\t\tmesh.userData = { ...mesh.userData, ...meshMeta.metadata };\n\t\t}\n\t\tmesh.castShadow = true;\n\t\tmesh.receiveShadow = true;\n\n\t\tmeshes.push(mesh);\n\t}\n\n\treturn meshes;\n}\n\n/**\n * Applies Rhino to Three.js coordinate system transformation.\n * Rhino uses Z-up, Three.js uses Y-up.\n */\nfunction applyCoordinateTransform(vertices: Float32Array): void {\n\tconst cos = Math.cos(-Math.PI / 2);\n\tconst sin = Math.sin(-Math.PI / 2);\n\n\tfor (let i = 0; i < vertices.length; i += 3) {\n\t\tconst x = vertices[i];\n\t\tconst y = vertices[i + 1];\n\t\tconst z = vertices[i + 2];\n\n\t\tvertices[i] = x;\n\t\tvertices[i + 1] = y * cos - z * sin;\n\t\tvertices[i + 2] = y * sin + z * cos;\n\t}\n}\n","import * as fflate from 'fflate';\n\nimport { decodeBase64ToBinary } from '@/core/utils/encoding';\nimport { RhinoComputeError, ErrorCodes } from '@/core/errors';\n\nimport type { DecompressedMeshData } from './types';\n\ninterface MeshData {\n\tverticesArray: Float32Array;\n\tfaceIndicesArray: Uint32Array;\n}\n\n/**\n * Decompresses a base64-encoded string using GZip.\n *\n * @internal Low-level decompression helper — keep internal to `selva-compute`.\n * @param base64String - The base64-encoded string to decompress.\n * @returns The decompressed MeshData.\n * @throws {RhinoComputeError} If decompression fails or data is invalid.\n */\nexport function decompressMeshData(base64String: string): MeshData {\n\ttry {\n\t\tconst bytes = decodeBase64ToBinary(base64String);\n\t\tconst decompressedData = fflate.gunzipSync(bytes);\n\t\treturn parseMeshBinaryData(decompressedData);\n\t} catch (error) {\n\t\tthrow new RhinoComputeError(\n\t\t\terror instanceof RhinoComputeError\n\t\t\t\t? error.message\n\t\t\t\t: `Failed to decompress data: ${error instanceof Error ? error.message : String(error)}`,\n\t\t\terror instanceof RhinoComputeError ? error.code : ErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: { base64StringLength: base64String.length },\n\t\t\t\toriginalError: error instanceof Error ? error : new Error(String(error))\n\t\t\t}\n\t\t);\n\t}\n}\n\n/**\n * Decompresses batched mesh data asynchronously using requestIdleCallback for non-blocking decompression.\n *\n * @internal Low-level decompression helper — keep internal to `selva-compute`.\n * @param base64String - The base64-encoded compressed data.\n * @returns Promise resolving to decompressed vertices and faces arrays.\n * @throws {RhinoComputeError} If decompression fails or data is invalid.\n */\nexport async function decompressBatchedMeshData(\n\tbase64String: string\n): Promise<DecompressedMeshData> {\n\treturn new Promise((resolve, reject) => {\n\t\ttry {\n\t\t\t// Use requestIdleCallback for non-blocking decompression if available\n\t\t\tconst decompressFn = () => {\n\t\t\t\ttry {\n\t\t\t\t\tconst bytes = decodeBase64ToBinary(base64String);\n\t\t\t\t\tconst decompressedData = fflate.gunzipSync(bytes);\n\t\t\t\t\tconst result = parseBatchedMeshBinaryData(decompressedData);\n\t\t\t\t\tresolve(result);\n\t\t\t\t} catch (error) {\n\t\t\t\t\treject(\n\t\t\t\t\t\tnew RhinoComputeError(\n\t\t\t\t\t\t\terror instanceof RhinoComputeError\n\t\t\t\t\t\t\t\t? error.message\n\t\t\t\t\t\t\t\t: `Failed to decompress batched data: ${error instanceof Error ? error.message : String(error)}`,\n\t\t\t\t\t\t\terror instanceof RhinoComputeError ? error.code : ErrorCodes.VALIDATION_ERROR,\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tcontext: { base64StringLength: base64String.length },\n\t\t\t\t\t\t\t\toriginalError: error instanceof Error ? error : new Error(String(error))\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t)\n\t\t\t\t\t);\n\t\t\t\t}\n\t\t\t};\n\n\t\t\tif ('requestIdleCallback' in globalThis) {\n\t\t\t\t(globalThis as any).requestIdleCallback(decompressFn, { timeout: 5000 });\n\t\t\t} else {\n\t\t\t\t// Fallback: use setTimeout with 0 delay to yield to other tasks\n\t\t\t\tsetTimeout(decompressFn, 0);\n\t\t\t}\n\t\t} catch (error) {\n\t\t\treject(\n\t\t\t\tnew RhinoComputeError(\n\t\t\t\t\t`Failed to schedule decompression: ${error instanceof Error ? error.message : String(error)}`,\n\t\t\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t\t\t{ originalError: error instanceof Error ? error : new Error(String(error)) }\n\t\t\t\t)\n\t\t\t);\n\t\t}\n\t});\n}\n\n/**\n * Parses batched binary mesh data (all vertices and faces together).\n * @param binaryMeshData - The binary mesh data to parse.\n * @returns The parsed mesh data with vertices and faces.\n * @throws {RhinoComputeError} If data is invalid or insufficient.\n */\nfunction parseBatchedMeshBinaryData(binaryMeshData: Uint8Array): DecompressedMeshData {\n\tconst dataView = new DataView(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset,\n\t\tbinaryMeshData.byteLength\n\t);\n\tlet offset = 0;\n\n\t// Read vertex data\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of vertex floats.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength, offset } }\n\t\t);\n\t}\n\tconst numVertexFloats = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tif (numVertexFloats % 3 !== 0) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Invalid number of vertex floats; should be divisible by 3.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\tnumVertexFloats,\n\t\t\t\t\tremainder: numVertexFloats % 3,\n\t\t\t\t\ttotalBytes: dataView.byteLength\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst verticesByteLength = numVertexFloats * Float32Array.BYTES_PER_ELEMENT;\n\tif (offset + verticesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read vertices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: verticesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst vertices = new Float32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumVertexFloats\n\t);\n\toffset += verticesByteLength;\n\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength - offset, offset } }\n\t\t);\n\t}\n\tconst numIndices = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tconst indicesByteLength = numIndices * Uint32Array.BYTES_PER_ELEMENT;\n\tif (offset + indicesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: indicesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst faces = new Uint32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumIndices\n\t);\n\n\treturn {\n\t\tvertices,\n\t\tfaces\n\t};\n}\n\n/**\n * Parses binary data and returns mesh data.\n * @param binaryMeshData - The binary mesh data to parse.\n * @returns The parsed mesh data.\n * @throws {RhinoComputeError} If data is invalid or insufficient.\n */\nfunction parseMeshBinaryData(binaryMeshData: Uint8Array): MeshData {\n\tconst dataView = new DataView(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset,\n\t\tbinaryMeshData.byteLength\n\t);\n\tlet offset = 0;\n\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of vertex floats.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength, offset } }\n\t\t);\n\t}\n\tconst numVertexFloats = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tif (numVertexFloats % 3 !== 0) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Invalid number of vertex floats; should be divisible by 3.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { numVertexFloats, remainder: numVertexFloats % 3 } }\n\t\t);\n\t}\n\n\tconst verticesByteLength = numVertexFloats * Float32Array.BYTES_PER_ELEMENT;\n\tif (offset + verticesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read vertices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: verticesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst vertices = new Float32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumVertexFloats\n\t);\n\toffset += verticesByteLength;\n\n\tif (offset + 4 > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read the number of face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{ context: { expectedBytes: 4, availableBytes: dataView.byteLength - offset, offset } }\n\t\t);\n\t}\n\tconst numIndices = dataView.getUint32(offset, true);\n\toffset += 4;\n\n\tconst indicesByteLength = numIndices * Uint32Array.BYTES_PER_ELEMENT;\n\tif (offset + indicesByteLength > dataView.byteLength) {\n\t\tthrow new RhinoComputeError(\n\t\t\t'Insufficient data to read face indices.',\n\t\t\tErrorCodes.VALIDATION_ERROR,\n\t\t\t{\n\t\t\t\tcontext: {\n\t\t\t\t\texpectedBytes: indicesByteLength,\n\t\t\t\t\tavailableBytes: dataView.byteLength - offset,\n\t\t\t\t\toffset\n\t\t\t\t}\n\t\t\t}\n\t\t);\n\t}\n\n\tconst faceIndices = new Uint32Array(\n\t\tbinaryMeshData.buffer,\n\t\tbinaryMeshData.byteOffset + offset,\n\t\tnumIndices\n\t);\n\n\treturn {\n\t\tverticesArray: vertices,\n\t\tfaceIndicesArray: faceIndices\n\t};\n}\n","import * as THREE from 'three';\n\nimport { applyOffset, computeCombinedBoundingBox } from '../threejs';\nimport { getLogger } from '@/core';\n\nimport { parseMeshBatch } from './batch-parser';\n\nimport type { DataItem, GrasshopperComputeResponse } from '@/features/grasshopper/types';\nimport type { MeshExtractionOptions, MeshBatchParsingOptions } from './types';\n\n// Constants\nexport const SCALE_FACTORS: Record<string, number> = {\n\tMillimeters: 1 / 1000,\n\tCentimeters: 1 / 100,\n\tMeters: 1,\n\tInches: 1 / 39.37,\n\tFeet: 1 / 3.28084\n};\n\nconst DISPLAY_COMPONENT_TYPE = 'Display';\n\n/**\n * Extracts and processes display meshes from a ComputePointerResponse using the Grasshopper WebDisplay component.\n *\n * This is the primary entry point for extracting mesh geometry from Grasshopper compute responses.\n * It handles all aspects of mesh processing: decompression, coordinate transformation, scaling, and positioning.\n *\n * **Note:** Mesh decompression happens asynchronously in a Web Worker to prevent UI blocking.\n *\n * @param data - The ComputePointerResponse containing Grasshopper output trees.\n * @param options - Configuration for mesh extraction and parsing behavior. All options are optional with sensible defaults.\n * @returns Promise resolving to array of THREE.Mesh objects (may be empty).\n * @throws Rethrows unexpected errors after attempting to dispose any created meshes.\n *\n * @remarks\n * - Only works with the WebDisplay component of GHHeadless.\n * - Requires changes to Rhino.Compute (see https://github.com/TheVessen/compute.rhino3d).\n * - Provides a performant way to display mesh data in Three.js.\n * - Decompression is performed in a Web Worker for non-blocking UI updates.\n * - Supports mesh metadata (names, user data) if provided in the compute response.\n *\n * @internal Internal helper: high-level extraction remains public via visualization module, but this\n * function is considered internal implementation detail for mesh extraction.\n *\n * @example\n * ```ts\n * // Simple usage with defaults (all processing enabled)\n * const meshes = await getThreeMeshesFromComputeResponse(response);\n *\n * // With debugging enabled\n * const meshes = await getThreeMeshesFromComputeResponse(response, { debug: true });\n *\n * // With advanced options\n * const meshes = await getThreeMeshesFromComputeResponse(response, {\n * debug: true,\n * allowScaling: true,\n * allowAutoPosition: false,\n * parsing: {\n * mergeByMaterial: false,\n * applyTransforms: true,\n * debug: true,\n * },\n * });\n * ```\n */\nexport async function getThreeMeshesFromComputeResponse(\n\tdata: GrasshopperComputeResponse,\n\toptions?: MeshExtractionOptions\n): Promise<THREE.Mesh[]> {\n\tconst startTime = performance.now();\n\tconst meshes: THREE.Mesh[] = [];\n\n\tconst {\n\t\tallowScaling = true,\n\t\tallowAutoPosition = true,\n\t\tdebug = false,\n\t\tparsing: parsingOptions = {}\n\t} = options ?? {};\n\n\ttry {\n\t\tconst scaleFactor = allowScaling ? getScaleFactor(data.modelunits) : 1;\n\t\tawait extractMeshesFromData(data, meshes, scaleFactor, parsingOptions, debug);\n\n\t\tif (allowAutoPosition) {\n\t\t\tapplyGroundOffset(meshes);\n\t\t}\n\n\t\treturn meshes;\n\t} catch (error) {\n\t\thandleError(error, meshes);\n\t\tthrow error;\n\t} finally {\n\t\tif (debug) {\n\t\t\tlogProcessingTime(startTime);\n\t\t}\n\t}\n}\n\n/**\n * Gets the scale factor for the given unit type.\n */\nfunction getScaleFactor(modelUnits: string): number {\n\treturn SCALE_FACTORS[modelUnits] ?? 1;\n}\n\n/**\n * Extracts meshes from compute response data.\n */\nasync function extractMeshesFromData(\n\tdata: GrasshopperComputeResponse,\n\tmeshes: THREE.Mesh[],\n\tscaleFactor: number,\n\tparsingOptions: MeshBatchParsingOptions,\n\tdebug: boolean\n): Promise<void> {\n\tfor (const value of data.values) {\n\t\tconst innerTree = value.InnerTree as { [key: string]: DataItem[] };\n\n\t\tfor (const path in innerTree) {\n\t\t\tconst branch = innerTree[path];\n\t\t\tif (!branch) continue;\n\n\t\t\tawait processDataBranch(branch, meshes, scaleFactor, parsingOptions, debug);\n\t\t}\n\t}\n}\n\n/**\n * Processes a single data branch to extract MeshBatch display meshes.\n */\nasync function processDataBranch(\n\tbranch: DataItem[],\n\tmeshes: THREE.Mesh[],\n\tscaleFactor: number,\n\tparsingOptions: MeshBatchParsingOptions,\n\tdebug: boolean\n): Promise<void> {\n\tfor (const item of branch) {\n\t\tif (item.type.includes(DISPLAY_COMPONENT_TYPE)) {\n\t\t\tconst mergedParsingOptions = {\n\t\t\t\tmergeByMaterial: true,\n\t\t\t\tapplyTransforms: true,\n\t\t\t\tdebug: false,\n\t\t\t\t...parsingOptions\n\t\t\t};\n\n\t\t\tconst batchMeshes = await parseMeshBatch(item.data, mergedParsingOptions);\n\n\t\t\tif (scaleFactor !== 1) {\n\t\t\t\tfor (const mesh of batchMeshes) {\n\t\t\t\t\tmesh.scale.set(scaleFactor, scaleFactor, scaleFactor);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tmeshes.push(...batchMeshes);\n\n\t\t\tif (debug) {\n\t\t\t\tgetLogger().debug(`Extracted ${batchMeshes.length} meshes from batch`);\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Applies vertical offset to position meshes on the Z=0 plane.\n */\nfunction applyGroundOffset(meshes: THREE.Mesh[]): void {\n\tif (meshes.length === 0) return;\n\n\tconst combinedBoundingBox = computeCombinedBoundingBox(meshes);\n\tconst offsetY = combinedBoundingBox.min.y;\n\tapplyOffset(meshes, offsetY);\n}\n\n/**\n * Handles errors by disposing created meshes and logging.\n */\nfunction handleError(error: unknown, meshes: THREE.Mesh[]): void {\n\tgetLogger().error('An unexpected error occurred:', error);\n\tdisposeMeshes(meshes);\n}\n\n/**\n * Disposes of all meshes and their associated resources.\n */\nfunction disposeMeshes(meshes: THREE.Mesh[]): void {\n\tfor (const mesh of meshes) {\n\t\tif (mesh.geometry) {\n\t\t\tmesh.geometry.dispose();\n\t\t}\n\n\t\tif (mesh.material) {\n\t\t\tif (Array.isArray(mesh.material)) {\n\t\t\t\tmesh.material.forEach((material) => material.dispose());\n\t\t\t} else {\n\t\t\t\tmesh.material.dispose();\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Logs the processing time for mesh extraction.\n */\nfunction logProcessingTime(startTime: number): void {\n\tconst elapsed = performance.now() - startTime;\n\tgetLogger().info('Time to process meshes:', `${elapsed.toFixed(2)}ms`);\n}\n"],"mappings":"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","names":["THREE","OrbitControls","RGBELoader","defaultUp","initThree","canvas","options","config","applyDefaults","scene","createScene","camera","createCamera","renderer","setupRenderer","controls","setupControls","setupEnvironment","setupLighting","addFloor","eventHandlers","setupEventHandlers","resize","disposeResize","setupResponsiveResize","animate","disposeAnimation","createAnimationLoop","sceneUp","object","material","scale","defaults","child","bgColor","animationId","parent","resizeTimeout","resizeObserver","getSize","handleResize","width","height","RGBELoader","envMap","error","getLogger","ambientLight","sunlight","pos","shadowSize","shadowNear","shadowFar","floorSize","floorGeometry","floorColor","floorMaterial","floor","selectedObjects","originalMaterials","raycaster","mouse","mouseDownPosition","fitToView","box","center","size","maxDim","fov","distance","direction","selectionColorObj","clearSelection","obj","handleMouseDown","event","handleCanvasClick","currentMousePosition","rect","intersects","clickedObject","clonedMaterial","handleKeydown","OrbitControls","target","THREE","CAMERA_CONFIG","updateScene","scene","meshes","camera","controls","initialPositionSet","clearScene","mesh","unionBoundingBox","computeCombinedBoundingBox","center","size","maxDim","distance","parseColor","colorString","getLogger","trimmed","hex","rgb","c","n","applyOffset","offsetY","combinedBoundingBox","bbox","clearScene","scene","objectsToRemove","child","object","material","key","value","three_materials_exports","__export","CONCRETE_MATERIAL","EMISSIVE_MATERIAL","GLASS_MATERIAL","METAL_MATERIAL","PLASTIC_MATERIAL","RUBBER_MATERIAL","WOOD_MATERIAL","THREE","THREE","fflate","init_errors","decompressBatchedMeshData","base64String","resolve","reject","decompressFn","bytes","decodeBase64ToBinary","decompressedData","result","parseBatchedMeshBinaryData","error","RhinoComputeError","ErrorCodes","binaryMeshData","dataView","offset","numVertexFloats","verticesByteLength","vertices","numIndices","indicesByteLength","faces","parseMeshBatch","batchJson","options","mergeByMaterial","applyTransforms","debug","perfStart","parseTime","parseStart","batch","parseMeshBatchObject","error","getLogger","scaleFactor","decompressTime","meshCreateTime","decompressStart","vertices","faces","decompressBatchedMeshData","compressedSizeMB","uncompressedSizeMB","compressionRatio","applyCoordinateTransform","meshCreateStart","materials","createMaterial","meshes","group","mergedMesh","createMergedMesh","individualMeshes","createIndividualMeshes","mesh","totalTime","matData","color","parseColor","allVertices","allFaces","geometry","totalVertexFloats","totalFaceIndices","mergedVertices","mergedIndices","vertexWriteOffset","indexWriteOffset","faceSlice","originalBaseVertexIndex","indexOffset","i","threeMesh","meshNames","m","name","allMetadata","meshMeta","baseIndex","rebasedFaces","cos","sin","x","y","z","SCALE_FACTORS","DISPLAY_COMPONENT_TYPE","getThreeMeshesFromComputeResponse","data","options","startTime","meshes","allowScaling","allowAutoPosition","debug","parsingOptions","scaleFactor","getScaleFactor","extractMeshesFromData","applyGroundOffset","error","handleError","logProcessingTime","modelUnits","value","innerTree","path","branch","processDataBranch","item","mergedParsingOptions","batchMeshes","parseMeshBatch","mesh","getLogger","offsetY","computeCombinedBoundingBox","applyOffset","disposeMeshes","material","elapsed"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "selva-compute",
|
|
3
|
-
"version": "1.1.
|
|
3
|
+
"version": "1.1.2",
|
|
4
4
|
"description": "TypeScript library for Rhino Compute Server - Grasshopper and RhinoCommon",
|
|
5
5
|
"author": "VektorNode",
|
|
6
6
|
"license": "MIT",
|
|
@@ -66,7 +66,6 @@
|
|
|
66
66
|
},
|
|
67
67
|
"sideEffects": false,
|
|
68
68
|
"dependencies": {
|
|
69
|
-
"compute-rhino3d": "0.13.0-beta",
|
|
70
69
|
"fflate": "^0.8.2",
|
|
71
70
|
"rhino3dm": "8.9.0"
|
|
72
71
|
},
|
package/dist/chunk-FNWG34KH.cjs
DELETED
|
@@ -1,3 +0,0 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }var E=Object.defineProperty;var S=Object.getOwnPropertyDescriptor;var I=Object.getOwnPropertyNames;var x=Object.prototype.hasOwnProperty;var $=(r,e,t)=>e in r?E(r,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):r[e]=t;var f=(r,e)=>()=>(r&&(e=r(r=0)),e);var _=(r,e)=>{for(var t in e)E(r,t,{get:e[t],enumerable:!0})},L=(r,e,t,o)=>{if(e&&typeof e=="object"||typeof e=="function")for(let n of I(e))!x.call(r,n)&&n!==t&&E(r,n,{get:()=>e[n],enumerable:!(o=S(e,n))||o.enumerable});return r};var F=r=>L(E({},"__esModule",{value:!0}),r);var d=(r,e,t)=>$(r,typeof e!="symbol"?e+"":e,t);var p,w= exports.e =f(()=>{"use strict";p= exports.d =class extends Error{constructor(t,o="UNKNOWN_ERROR",n){super(t);d(this,"code");d(this,"statusCode");d(this,"context");d(this,"originalError");this.name="RhinoComputeError",this.code=o,this.statusCode=_optionalChain([n, 'optionalAccess', _2 => _2.statusCode]),this.context=_optionalChain([n, 'optionalAccess', _3 => _3.context]),this.originalError=_optionalChain([n, 'optionalAccess', _4 => _4.originalError]),"cause"in Error.prototype&&Object.defineProperty(this,"cause",{value:_optionalChain([n, 'optionalAccess', _5 => _5.originalError]),enumerable:!0})}}});var i,N= exports.g =f(()=>{"use strict";i= exports.f ={NETWORK_ERROR:"NETWORK_ERROR",AUTH_ERROR:"AUTH_ERROR",VALIDATION_ERROR:"VALIDATION_ERROR",COMPUTATION_ERROR:"COMPUTATION_ERROR",TIMEOUT_ERROR:"TIMEOUT_ERROR",CORS_ERROR:"CORS_ERROR",UNKNOWN_ERROR:"UNKNOWN_ERROR",INVALID_STATE:"INVALID_STATE",INVALID_INPUT:"INVALID_INPUT",INVALID_CONFIG:"INVALID_CONFIG",BROWSER_ONLY:"BROWSER_ONLY",ENVIRONMENT_ERROR:"ENVIRONMENT_ERROR",ENCODING_ERROR:"ENCODING_ERROR"}});var A={};_(A,{ErrorCodes:()=>i,RhinoComputeError:()=>p});var C=f(()=>{"use strict";w();N()});function l(){return v}function b(r){r===null?v=new O:"debug"in r&&"info"in r&&"warn"in r&&"error"in r?v=r:v=new T}function M(){b(new T)}var O,T,v,y= exports.m =f(()=>{"use strict";O=class{debug(){}info(){}warn(){}error(){}},T=class{debug(e,...t){console.debug(e,...t)}info(e,...t){console.info(e,...t)}warn(e,...t){console.warn(e,...t)}error(e,...t){console.error(e,...t)}},v=new O});C();y();var _computerhino3d = require('compute-rhino3d'); var _computerhino3d2 = _interopRequireDefault(_computerhino3d);function D(r,e,t,o,n,R){let{status:a,statusText:u}=r,c={url:e,requestId:t,method:"POST",requestSize:o,serverUrl:n},h={401:{message:`HTTP ${a}: ${u}`,code:i.AUTH_ERROR},403:{message:`HTTP ${a}: ${u}`,code:i.AUTH_ERROR},404:{message:`Endpoint not found: ${e}`,code:i.NETWORK_ERROR},413:{message:`Request too large: ${(o/1024).toFixed(2)}KB`,code:i.VALIDATION_ERROR},429:{message:"Rate limit exceeded",code:i.NETWORK_ERROR},500:{message:`Server error: ${R||u}`,code:i.COMPUTATION_ERROR},502:{message:`Service unavailable: ${u}`,code:i.NETWORK_ERROR},503:{message:`Service unavailable: ${u}`,code:i.NETWORK_ERROR},504:{message:`Service unavailable: ${u}`,code:i.NETWORK_ERROR}}[a]||{message:`HTTP ${a}: ${u}`,code:i.UNKNOWN_ERROR};throw new p(h.message,h.code,{statusCode:a,context:c})}function K(r,e){let t=e.replace(/\/+$/,""),o=r.replace(/^\/+/,"");return`${t}/${o}`}function k(r){try{let e=new URL(r).host;return/^(localhost|127\.0\.0\.1|::1)(:\d+)?$/i.test(e)}catch (e2){return/(localhost|127\.0\.0\.1)/i.test(r)}}function P(r,e){let t={"User-Agent":`compute.rhino3d.js/${_computerhino3d2.default.version}`,"X-Request-ID":r,"Content-Type":"application/json",...e.authToken&&{Authorization:e.authToken},...e.apiKey&&{RhinoComputeKey:e.apiKey}};return!e.apiKey&&!k(e.serverUrl)&&l().warn(`\u26A0\uFE0F [Rhino Compute] Request [${r}] targets remote server (${e.serverUrl}) but no API key is configured. Requests may fail or be rate-limited.`),t}function H(){return`${Date.now()}-${Math.random().toString(36).substring(2,11)}`}function m(r,e){e&&l().debug(r)}async function V(r,e,t,o,n,R,a){let u=Math.round(performance.now()-R);if(!r.ok){let c=await r.text();if(a&&(m(`\u274C Request [${t}] failed with HTTP ${r.status} in ${u}ms`,!0),m(` URL: ${e}`,!0),m(` Status: ${r.status} ${r.statusText}`,!0),c&&m(` Response body: ${c.substring(0,500)}${c.length>500?"...":""}`,!0)),r.status===500)try{let s=JSON.parse(c);if(_optionalChain([s, 'optionalAccess', _6 => _6.values])&&(s.errors||s.warnings))return a&&(m(`\u26A0\uFE0F Request [${t}] completed with Grasshopper errors in ${u}ms`,!0),_optionalChain([s, 'access', _7 => _7.errors, 'optionalAccess', _8 => _8.length])>0&&m(` Errors: ${JSON.stringify(s.errors,null,2)}`,!0),_optionalChain([s, 'access', _9 => _9.warnings, 'optionalAccess', _10 => _10.length])>0&&m(` Warnings: ${JSON.stringify(s.warnings,null,2)}`,!0)),s;_optionalChain([s, 'optionalAccess', _11 => _11.Message])?c=`${s.ExceptionType?s.ExceptionType+": ":""}${s.Message}
|
|
2
|
-
${s.StackTrace||""}`:_optionalChain([s, 'optionalAccess', _12 => _12.error])&&(c=typeof s.error=="string"?s.error:JSON.stringify(s.error,null,2))}catch(s){a&&m(` Failed to parse error body as JSON: ${s}`,!0)}D(r,e,t,o,n,c)}m(`\u2705 Request [${t}] completed in ${u}ms`,a);try{return await r.json()}catch(c){throw new p("Failed to parse JSON response",i.NETWORK_ERROR,{statusCode:r.status,context:{url:e,requestId:t},originalError:c instanceof Error?c:new Error(String(c))})}}async function W(r,e,t){let o=H(),n=JSON.stringify(e),R=n.length,a=K(r,t.serverUrl);if(t.debug){let s=(R/1024).toFixed(2),h=R>1e5?"\u26A0\uFE0F":"\u{1F680}";m(`${h} Starting compute request [${o}]: ${r} (${s}KB)`,!0)}let u=new AbortController,c=t.timeoutMs?setTimeout(()=>u.abort(),t.timeoutMs):null;try{let s=performance.now(),h=await fetch(a,{method:"POST",body:n,headers:P(o,t),signal:u.signal});return await V(h,a,o,R,t.serverUrl,s,t.debug)}catch(s){throw s instanceof Error&&s.name==="AbortError"&&t.timeoutMs?new p(`Request timed out after ${t.timeoutMs}ms`,i.TIMEOUT_ERROR,{context:{serverUrl:t.serverUrl,timeoutMs:t.timeoutMs,url:a,requestId:o,args:e}}):s instanceof TypeError?new p(`Network error: ${s.message}`,i.NETWORK_ERROR,{context:{serverUrl:t.serverUrl,url:a,requestId:o,endpoint:r},originalError:s}):s}finally{c!==null&&clearTimeout(c)}}C();y();var g=class{constructor(e,t){d(this,"serverUrl");d(this,"apiKey");d(this,"disposed",!1);d(this,"activeMonitors",new Set);d(this,"activeTimeouts",new Set);if(!_optionalChain([e, 'optionalAccess', _13 => _13.trim, 'call', _14 => _14()]))throw new p("serverUrl is required",i.INVALID_CONFIG,{context:{serverUrl:e}});if(!e.match(/^https?:\/\//))throw new p(`Invalid serverUrl: "${e}". Must start with "http://" or "https://". For example: "http://localhost:5000" or "https://example.com"`,i.INVALID_CONFIG,{context:{serverUrl:e}});try{new URL(e)}catch(o){throw new p(`Invalid serverUrl: "${e}". Must be a valid URL. Received error: ${o instanceof Error?o.message:String(o)}`,i.INVALID_CONFIG,{context:{serverUrl:e},originalError:o instanceof Error?o:void 0})}this.apiKey=t,this.serverUrl=e.replace(/\/+$/,"")}buildHeaders(){let e={"Content-Type":"application/json"};return this.apiKey&&(e.RhinoComputeKey=this.apiKey),e}async isServerOnline(){this.ensureNotDisposed();let e=`${this.serverUrl}/healthcheck`,t={headers:this.buildHeaders(),method:"GET"};try{return(await fetch(e,t)).ok}catch(o){return l().debug("[ComputeServerStats] Fetch error:",o),!1}}async getActiveChildren(){this.ensureNotDisposed();try{let e=await fetch(`${this.serverUrl}/activechildren`,{headers:this.buildHeaders()});if(!e.ok)return l().warn("[ComputeServerStats] Failed to fetch active children:",e.status),null;let t=await e.text(),o=parseInt(t.trim(),10);return isNaN(o)?(l().warn("[ComputeServerStats] Invalid active children response:",t),null):o}catch(e){return l().warn("[ComputeServerStats] Error fetching active children:",e),null}}async getVersion(){this.ensureNotDisposed();try{let e=await fetch(`${this.serverUrl}/version`,{headers:this.buildHeaders()});if(!e.ok)return l().warn("[ComputeServerStats] Failed to fetch version:",e.status),null;try{let t=await e.json();return{rhino:_nullishCoalesce(t.rhino, () => ("")),compute:_nullishCoalesce(t.compute, () => ("")),git_sha:_nullishCoalesce(t.git_sha, () => (null))}}catch (e3){return{rhino:await e.text(),compute:"",git_sha:null}}}catch(e){return l().warn("[ComputeServerStats] Error fetching version:",e),null}}async getServerStats(){if(this.ensureNotDisposed(),!await this.isServerOnline())return{isOnline:!1};let[t,o]=await Promise.all([this.getVersion(),this.getActiveChildren()]);return{isOnline:!0,...t&&{version:t},...o!==null&&{activeChildren:o}}}monitor(e,t=5e3){this.ensureNotDisposed();let o=!0,n=null;l().info(`\u{1F504} Starting server stats monitoring every ${t}ms`);let R=async()=>{if(n!==null&&(this.activeTimeouts.delete(n),n=null),!o||this.disposed)return;let u=await this.getServerStats();!o||this.disposed||(e(u),o&&!this.disposed&&(n=setTimeout(()=>{R()},t),this.activeTimeouts.add(n)))},a=()=>{o=!1,n!==null&&(clearTimeout(n),this.activeTimeouts.delete(n),n=null),this.activeMonitors.delete(a)};return this.activeMonitors.add(a),R(),a}async dispose(){if(!this.disposed){this.disposed=!0;for(let e of this.activeMonitors)e();this.activeMonitors.clear();for(let e of this.activeTimeouts)clearTimeout(e);this.activeTimeouts.clear()}}ensureNotDisposed(){if(this.disposed)throw new p("ComputeServerStats has been disposed and cannot be used",i.INVALID_STATE,{context:{disposed:this.disposed}})}};w();N();y();exports.a = _; exports.b = F; exports.c = d; exports.d = p; exports.e = w; exports.f = i; exports.g = N; exports.h = A; exports.i = C; exports.j = l; exports.k = b; exports.l = M; exports.m = y; exports.n = W; exports.o = g;
|
|
3
|
-
//# sourceMappingURL=chunk-FNWG34KH.cjs.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["/home/runner/work/selva-compute/selva-compute/dist/chunk-FNWG34KH.cjs","../src/core/errors/base.ts","../src/core/errors/error-codes.ts","../src/core/errors/index.ts","../src/core/utils/logger.ts","../src/core/compute-fetch/compute-fetch.ts","../src/core/server/compute-server-stats.ts"],"names":["RhinoComputeError","init_base","__esmMin","message","code","options","__publicField","ErrorCodes","init_error_codes","errors_exports","__export","init_errors","getLogger","internalLogger","setLogger","logger","NoOpLogger","ConsoleLogger","enableDebugLogging","init_logger","args","throwHttpError","response","fullUrl","requestId","requestSize","serverUrl","errorBody","status","statusText","context","error"],"mappings":"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","file":"/home/runner/work/selva-compute/selva-compute/dist/chunk-FNWG34KH.cjs","sourcesContent":[null,"/**\n * Simplified error for Rhino Compute operations\n *\n * @public Use this for error handling with error codes and context.\n */\nexport class RhinoComputeError extends Error {\n\tpublic readonly code: string;\n\tpublic readonly statusCode?: number;\n\tpublic readonly context?: Record<string, unknown>;\n\tpublic readonly originalError?: Error;\n\n\tconstructor(\n\t\tmessage: string,\n\t\tcode: string = 'UNKNOWN_ERROR',\n\t\toptions?: { statusCode?: number; context?: Record<string, unknown>; originalError?: Error }\n\t) {\n\t\tsuper(message);\n\t\tthis.name = 'RhinoComputeError';\n\t\tthis.code = code;\n\t\tthis.statusCode = options?.statusCode;\n\t\tthis.context = options?.context;\n\t\tthis.originalError = options?.originalError;\n\n\t\t// Support error chaining (Node.js 16.9+, TypeScript 4.6+)\n\t\tif ('cause' in Error.prototype) {\n\t\t\tObject.defineProperty(this, 'cause', {\n\t\t\t\tvalue: options?.originalError,\n\t\t\t\tenumerable: true\n\t\t\t});\n\t\t}\n\t}\n}\n","export const ErrorCodes = {\n\tNETWORK_ERROR: 'NETWORK_ERROR',\n\tAUTH_ERROR: 'AUTH_ERROR',\n\tVALIDATION_ERROR: 'VALIDATION_ERROR',\n\tCOMPUTATION_ERROR: 'COMPUTATION_ERROR',\n\tTIMEOUT_ERROR: 'TIMEOUT_ERROR',\n\tCORS_ERROR: 'CORS_ERROR',\n\tUNKNOWN_ERROR: 'UNKNOWN_ERROR',\n\tINVALID_STATE: 'INVALID_STATE',\n\tINVALID_INPUT: 'INVALID_INPUT',\n\tINVALID_CONFIG: 'INVALID_CONFIG',\n\tBROWSER_ONLY: 'BROWSER_ONLY',\n\tENVIRONMENT_ERROR: 'ENVIRONMENT_ERROR',\n\tENCODING_ERROR: 'ENCODING_ERROR'\n} as const;\n\nexport type ErrorCode = (typeof ErrorCodes)[keyof typeof ErrorCodes];\n","/**\n * Error handling - explicit public API\n */\n\nexport { RhinoComputeError } from './base';\nexport { ErrorCodes } from './error-codes';\nexport type { ErrorCode } from './error-codes';\n","/**\n * Logger interface for structured logging\n *\n * @public Implement this interface to provide custom logging behavior.\n */\nexport interface Logger {\n\tdebug(message: string, ...args: unknown[]): void;\n\tinfo(message: string, ...args: unknown[]): void;\n\twarn(message: string, ...args: unknown[]): void;\n\terror(message: string, ...args: unknown[]): void;\n}\n\n/**\n * No-op logger implementation (default)\n * @internal\n */\nclass NoOpLogger implements Logger {\n\tdebug(): void {}\n\tinfo(): void {}\n\twarn(): void {}\n\terror(): void {}\n}\n\n/**\n * Console logger implementation\n * @internal\n */\nclass ConsoleLogger implements Logger {\n\tdebug(message: string, ...args: unknown[]): void {\n\t\tconsole.debug(message, ...args);\n\t}\n\n\tinfo(message: string, ...args: unknown[]): void {\n\t\tconsole.info(message, ...args);\n\t}\n\n\twarn(message: string, ...args: unknown[]): void {\n\t\tconsole.warn(message, ...args);\n\t}\n\n\terror(message: string, ...args: unknown[]): void {\n\t\tconsole.error(message, ...args);\n\t}\n}\n\n/**\n * Internal logger instance\n * @internal\n */\nlet internalLogger: Logger = new NoOpLogger();\n\n/**\n * Get the current logger instance\n *\n * @returns The current logger instance\n */\nexport function getLogger(): Logger {\n\treturn internalLogger;\n}\n\n/**\n * Set a custom logger instance\n *\n * @public Use this to configure custom logging behavior.\n *\n * @param logger - Custom logger implementation or null to disable logging\n *\n * @example\n * ```typescript\n * import { setLogger } from 'selva-compute';\n *\n * // Enable console logging\n * setLogger(console);\n *\n * // Use a custom logger\n * setLogger({\n * debug: (msg, ...args) => myLogger.debug(msg, ...args),\n * info: (msg, ...args) => myLogger.info(msg, ...args),\n * warn: (msg, ...args) => myLogger.warn(msg, ...args),\n * error: (msg, ...args) => myLogger.error(msg, ...args)\n * });\n *\n * // Disable logging\n * setLogger(null);\n * ```\n */\nexport function setLogger(logger: Logger | Console | null): void {\n\tif (logger === null) {\n\t\tinternalLogger = new NoOpLogger();\n\t} else if ('debug' in logger && 'info' in logger && 'warn' in logger && 'error' in logger) {\n\t\tinternalLogger = logger as Logger;\n\t} else {\n\t\tinternalLogger = new ConsoleLogger();\n\t}\n}\n\n/**\n * Enable debug logging to console\n *\n * @public Convenience method to enable console logging.\n *\n * @example\n * ```typescript\n * import { enableDebugLogging } from 'selva-compute';\n *\n * enableDebugLogging();\n * ```\n */\nexport function enableDebugLogging(): void {\n\tsetLogger(new ConsoleLogger());\n}\n","import RhinoCompute from 'compute-rhino3d';\n\nimport { RhinoComputeError, ErrorCodes } from '../errors';\nimport { getLogger } from '../utils/logger';\n\nimport type { ComputeConfig } from '../types';\nimport type {\n\tGrasshopperComputeConfig,\n\tGrasshopperComputeResponse,\n\tIoResponseSchema\n} from '@/features/grasshopper/types';\n\n/**\n * Valid endpoints for Rhino Compute\n */\nexport type Endpoint = 'grasshopper' | 'io' | string;\n\nexport type EndpointResponseMap = {\n\tgrasshopper: GrasshopperComputeResponse;\n\tio: IoResponseSchema;\n};\n\nexport type ComputeResponseFor<E extends string> = E extends keyof EndpointResponseMap\n\t? EndpointResponseMap[E]\n\t: unknown;\n\n// ============================================================================\n// Error Handling\n// ============================================================================\n\nfunction throwHttpError(\n\tresponse: Response,\n\tfullUrl: string,\n\trequestId: string,\n\trequestSize: number,\n\tserverUrl: string,\n\terrorBody: string\n): never {\n\tconst { status, statusText } = response;\n\tconst context = { url: fullUrl, requestId, method: 'POST', requestSize, serverUrl };\n\n\tconst errorMap: Record<number, { message: string; code: string }> = {\n\t\t401: { message: `HTTP ${status}: ${statusText}`, code: ErrorCodes.AUTH_ERROR },\n\t\t403: { message: `HTTP ${status}: ${statusText}`, code: ErrorCodes.AUTH_ERROR },\n\t\t404: { message: `Endpoint not found: ${fullUrl}`, code: ErrorCodes.NETWORK_ERROR },\n\t\t413: {\n\t\t\tmessage: `Request too large: ${(requestSize / 1024).toFixed(2)}KB`,\n\t\t\tcode: ErrorCodes.VALIDATION_ERROR\n\t\t},\n\t\t429: { message: 'Rate limit exceeded', code: ErrorCodes.NETWORK_ERROR },\n\t\t500: {\n\t\t\tmessage: `Server error: ${errorBody || statusText}`,\n\t\t\tcode: ErrorCodes.COMPUTATION_ERROR\n\t\t},\n\t\t502: { message: `Service unavailable: ${statusText}`, code: ErrorCodes.NETWORK_ERROR },\n\t\t503: { message: `Service unavailable: ${statusText}`, code: ErrorCodes.NETWORK_ERROR },\n\t\t504: { message: `Service unavailable: ${statusText}`, code: ErrorCodes.NETWORK_ERROR }\n\t};\n\n\tconst error = errorMap[status] || {\n\t\tmessage: `HTTP ${status}: ${statusText}`,\n\t\tcode: ErrorCodes.UNKNOWN_ERROR\n\t};\n\n\tthrow new RhinoComputeError(error.message, error.code, { statusCode: status, context });\n}\n\n// ============================================================================\n// Request Helpers\n// ============================================================================\n\nfunction buildUrl(endpoint: string, serverUrl: string): string {\n\tconst base = serverUrl.replace(/\\/+$/, '');\n\tconst path = endpoint.replace(/^\\/+/, '');\n\treturn `${base}/${path}`;\n}\n\nfunction isLocalhost(serverUrl: string): boolean {\n\ttry {\n\t\tconst host = new URL(serverUrl).host;\n\t\treturn /^(localhost|127\\.0\\.0\\.1|::1)(:\\d+)?$/i.test(host);\n\t} catch {\n\t\treturn /(localhost|127\\.0\\.0\\.1)/i.test(serverUrl);\n\t}\n}\n\nfunction buildHeaders(requestId: string, config: ComputeConfig): HeadersInit {\n\tconst headers: HeadersInit = {\n\t\t'User-Agent': `compute.rhino3d.js/${RhinoCompute.version}`,\n\t\t'X-Request-ID': requestId,\n\t\t'Content-Type': 'application/json',\n\t\t...(config.authToken && { Authorization: config.authToken }),\n\t\t...(config.apiKey && { RhinoComputeKey: config.apiKey })\n\t};\n\n\tif (!config.apiKey && !isLocalhost(config.serverUrl)) {\n\t\tgetLogger().warn(\n\t\t\t`⚠️ [Rhino Compute] Request [${requestId}] targets remote server (${config.serverUrl}) but no API key is configured. Requests may fail or be rate-limited.`\n\t\t);\n\t}\n\n\treturn headers;\n}\n\nfunction generateRequestId(): string {\n\treturn `${Date.now()}-${Math.random().toString(36).substring(2, 11)}`;\n}\n\nfunction log(message: string, debug?: boolean): void {\n\tif (debug) getLogger().debug(message);\n}\n\n// ============================================================================\n// Response Processing\n// ============================================================================\n\nasync function handleResponse(\n\tresponse: Response,\n\tfullUrl: string,\n\trequestId: string,\n\trequestSize: number,\n\tserverUrl: string,\n\tstartTime: number,\n\tdebug?: boolean\n): Promise<any> {\n\tconst responseTime = Math.round(performance.now() - startTime);\n\n\tif (!response.ok) {\n\t\t// Read body once and reuse\n\t\tlet errorBody = await response.text();\n\n\t\t// Enhanced logging for errors\n\t\tif (debug) {\n\t\t\tlog(\n\t\t\t\t`❌ Request [${requestId}] failed with HTTP ${response.status} in ${responseTime}ms`,\n\t\t\t\ttrue\n\t\t\t);\n\t\t\tlog(` URL: ${fullUrl}`, true);\n\t\t\tlog(` Status: ${response.status} ${response.statusText}`, true);\n\t\t\tif (errorBody) {\n\t\t\t\tlog(\n\t\t\t\t\t` Response body: ${errorBody.substring(0, 500)}${errorBody.length > 500 ? '...' : ''}`,\n\t\t\t\t\ttrue\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\n\t\t// Check if it's a valid compute response with errors/warnings\n\t\tif (response.status === 500) {\n\t\t\ttry {\n\t\t\t\tconst parsed = JSON.parse(errorBody);\n\t\t\t\t// If it has values, it's a partial success with errors\n\t\t\t\tif (parsed?.values && (parsed.errors || parsed.warnings)) {\n\t\t\t\t\tif (debug) {\n\t\t\t\t\t\tlog(\n\t\t\t\t\t\t\t`⚠️ Request [${requestId}] completed with Grasshopper errors in ${responseTime}ms`,\n\t\t\t\t\t\t\ttrue\n\t\t\t\t\t\t);\n\t\t\t\t\t\tif (parsed.errors?.length > 0) {\n\t\t\t\t\t\t\tlog(` Errors: ${JSON.stringify(parsed.errors, null, 2)}`, true);\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (parsed.warnings?.length > 0) {\n\t\t\t\t\t\t\tlog(` Warnings: ${JSON.stringify(parsed.warnings, null, 2)}`, true);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\treturn parsed;\n\t\t\t\t}\n\n\t\t\t\t// If it's a raw exception from the server (like ArgumentException), include it in the error message\n\t\t\t\tif (parsed?.Message) {\n\t\t\t\t\terrorBody = `${parsed.ExceptionType ? parsed.ExceptionType + ': ' : ''}${parsed.Message}\\n${parsed.StackTrace || ''}`;\n\t\t\t\t} else if (parsed?.error) {\n\t\t\t\t\terrorBody =\n\t\t\t\t\t\ttypeof parsed.error === 'string' ? parsed.error : JSON.stringify(parsed.error, null, 2);\n\t\t\t\t}\n\t\t\t} catch (e) {\n\t\t\t\tif (debug) {\n\t\t\t\t\tlog(` Failed to parse error body as JSON: ${e}`, true);\n\t\t\t\t}\n\t\t\t\t// Not valid JSON, proceed with HTTP error\n\t\t\t}\n\t\t}\n\n\t\tthrowHttpError(response, fullUrl, requestId, requestSize, serverUrl, errorBody);\n\t}\n\n\tlog(`✅ Request [${requestId}] completed in ${responseTime}ms`, debug);\n\n\ttry {\n\t\treturn await response.json();\n\t} catch (error) {\n\t\tthrow new RhinoComputeError('Failed to parse JSON response', ErrorCodes.NETWORK_ERROR, {\n\t\t\tstatusCode: response.status,\n\t\t\tcontext: {\n\t\t\t\turl: fullUrl,\n\t\t\t\trequestId\n\t\t\t},\n\t\t\toriginalError: error instanceof Error ? error : new Error(String(error))\n\t\t});\n\t}\n}\n\n// ============================================================================\n// Main Function\n// ============================================================================\n\n//TODO: VALIDATE IF THIS DOC IS ACCURATE\n\n/**\n * Generic Rhino Compute fetch function.\n * Sends a POST request to any Compute endpoint with pre-prepared arguments.\n *\n * @public Use this for advanced low-level control over compute requests. For most use cases, prefer higher-level APIs.\n *\n * @param endpoint - The Compute API endpoint (e.g., 'grasshopper', 'io', 'mesh').\n * @param args - Pre-prepared arguments for the request.\n * @param config - Compute configuration (server URL, API key, timeout, debug).\n * @returns The parsed JSON response from the server.\n *\n * @example\n * ```typescript\n * const response = await fetchRhinoCompute(\n * 'grasshopper',\n * {\n * pointer: { url: 'https://example.com/definition.gh' },\n * values: [{ ParamName: 'x', InnerTree: { '0': [{ type: 'System.Double', data: 10 }] } }]\n * },\n * { serverUrl: 'https://compute.rhino3d.com', debug: true, timeoutMs: 30000 }\n * );\n * ```\n */\nexport async function fetchRhinoCompute<E extends Endpoint>(\n\tendpoint: E,\n\targs: Record<string, any>,\n\tconfig: ComputeConfig | GrasshopperComputeConfig\n): Promise<ComputeResponseFor<E>> {\n\tconst requestId = generateRequestId();\n\tconst body = JSON.stringify(args);\n\tconst requestSize = body.length;\n\tconst fullUrl = buildUrl(endpoint, config.serverUrl);\n\n\tif (config.debug) {\n\t\tconst sizeKb = (requestSize / 1024).toFixed(2);\n\t\tconst emoji = requestSize > 100000 ? '⚠️' : '🚀';\n\t\tlog(`${emoji} Starting compute request [${requestId}]: ${endpoint} (${sizeKb}KB)`, true);\n\t}\n\n\tconst controller = new AbortController();\n\tconst timeoutId = config.timeoutMs\n\t\t? setTimeout(() => controller.abort(), config.timeoutMs)\n\t\t: null;\n\n\ttry {\n\t\tconst startTime = performance.now();\n\t\tconst response = await fetch(fullUrl, {\n\t\t\tmethod: 'POST',\n\t\t\tbody,\n\t\t\theaders: buildHeaders(requestId, config),\n\t\t\tsignal: controller.signal\n\t\t});\n\n\t\treturn await handleResponse(\n\t\t\tresponse,\n\t\t\tfullUrl,\n\t\t\trequestId,\n\t\t\trequestSize,\n\t\t\tconfig.serverUrl,\n\t\t\tstartTime,\n\t\t\tconfig.debug\n\t\t);\n\t} catch (error) {\n\t\tif (error instanceof Error && error.name === 'AbortError' && config.timeoutMs) {\n\t\t\tthrow new RhinoComputeError(\n\t\t\t\t`Request timed out after ${config.timeoutMs}ms`,\n\t\t\t\tErrorCodes.TIMEOUT_ERROR,\n\t\t\t\t{\n\t\t\t\t\tcontext: {\n\t\t\t\t\t\tserverUrl: config.serverUrl,\n\t\t\t\t\t\ttimeoutMs: config.timeoutMs,\n\t\t\t\t\t\turl: fullUrl,\n\t\t\t\t\t\trequestId,\n\t\t\t\t\t\targs\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t);\n\t\t}\n\n\t\t// Handle fetch errors (network issues, connection refused, etc.)\n\t\tif (error instanceof TypeError) {\n\t\t\tthrow new RhinoComputeError(`Network error: ${error.message}`, ErrorCodes.NETWORK_ERROR, {\n\t\t\t\tcontext: {\n\t\t\t\t\tserverUrl: config.serverUrl,\n\t\t\t\t\turl: fullUrl,\n\t\t\t\t\trequestId,\n\t\t\t\t\tendpoint\n\t\t\t\t},\n\t\t\t\toriginalError: error\n\t\t\t});\n\t\t}\n\n\t\tthrow error;\n\t} finally {\n\t\tif (timeoutId !== null) clearTimeout(timeoutId);\n\t}\n}\n","import { RhinoComputeError, ErrorCodes } from '../errors';\nimport { getLogger } from '../utils/logger';\n\n/**\n * ComputeServerStats provides methods to query Rhino Compute server statistics.\n *\n * @public Use this for server health monitoring and statistics.\n *\n * @example\n * ```typescript\n * const stats = new ComputeServerStats('http://localhost:6500', 'your-api-key');\n *\n * try {\n * const isOnline = await stats.isServerOnline();\n * const children = await stats.getActiveChildren();\n * const version = await stats.getVersion();\n *\n * // Or get everything at once\n * const allStats = await stats.getServerStats();\n * } finally {\n * await stats.dispose(); // Clean up resources\n * }\n * ```\n */\nexport default class ComputeServerStats {\n\tprivate readonly serverUrl: string;\n\tprivate readonly apiKey?: string;\n\tprivate disposed = false;\n\tprivate activeMonitors: Set<() => void> = new Set();\n\tprivate activeTimeouts: Set<ReturnType<typeof setTimeout>> = new Set();\n\n\t/**\n\t * @param serverUrl - Base URL of the Rhino Compute server with http:// or https:// scheme (e.g., 'http://localhost:6500')\n\t * @param apiKey - Optional API key for authentication\n\t */\n\tconstructor(serverUrl: string, apiKey?: string) {\n\t\tif (!serverUrl?.trim()) {\n\t\t\tthrow new RhinoComputeError('serverUrl is required', ErrorCodes.INVALID_CONFIG, {\n\t\t\t\tcontext: { serverUrl }\n\t\t\t});\n\t\t}\n\n\t\t// Validate URL has http:// or https:// scheme\n\t\tif (!serverUrl.match(/^https?:\\/\\//)) {\n\t\t\tthrow new RhinoComputeError(\n\t\t\t\t`Invalid serverUrl: \"${serverUrl}\". Must start with \"http://\" or \"https://\". ` +\n\t\t\t\t\t`For example: \"http://localhost:5000\" or \"https://example.com\"`,\n\t\t\t\tErrorCodes.INVALID_CONFIG,\n\t\t\t\t{ context: { serverUrl } }\n\t\t\t);\n\t\t}\n\n\t\ttry {\n\t\t\tnew URL(serverUrl);\n\t\t} catch (err) {\n\t\t\tthrow new RhinoComputeError(\n\t\t\t\t`Invalid serverUrl: \"${serverUrl}\". Must be a valid URL. ` +\n\t\t\t\t\t`Received error: ${err instanceof Error ? err.message : String(err)}`,\n\t\t\t\tErrorCodes.INVALID_CONFIG,\n\t\t\t\t{\n\t\t\t\t\tcontext: { serverUrl },\n\t\t\t\t\toriginalError: err instanceof Error ? err : undefined\n\t\t\t\t}\n\t\t\t);\n\t\t}\n\n\t\tthis.apiKey = apiKey;\n\t\tthis.serverUrl = serverUrl.replace(/\\/+$/, '');\n\t}\n\n\t/**\n\t * Build request headers with optional API key.\n\t */\n\tprivate buildHeaders(): HeadersInit {\n\t\tconst headers: HeadersInit = {\n\t\t\t'Content-Type': 'application/json'\n\t\t};\n\n\t\tif (this.apiKey) {\n\t\t\theaders['RhinoComputeKey'] = this.apiKey;\n\t\t}\n\n\t\treturn headers;\n\t}\n\n\t/**\n\t * Check if the server is online.\n\t */\n\tpublic async isServerOnline(): Promise<boolean> {\n\t\tthis.ensureNotDisposed();\n\n\t\tconst url = `${this.serverUrl}/healthcheck`;\n\t\tconst init: RequestInit = { headers: this.buildHeaders(), method: 'GET' };\n\n\t\ttry {\n\t\t\tconst response = await fetch(url, init);\n\n\t\t\treturn response.ok;\n\t\t} catch (err) {\n\t\t\tgetLogger().debug('[ComputeServerStats] Fetch error:', err);\n\t\t\treturn false;\n\t\t}\n\t}\n\n\t/**\n\t * Get the number of active child processes on the server.\n\t *\n\t * @returns Number of active children, or null if unavailable\n\t */\n\tpublic async getActiveChildren(): Promise<number | null> {\n\t\tthis.ensureNotDisposed();\n\n\t\ttry {\n\t\t\tconst response = await fetch(`${this.serverUrl}/activechildren`, {\n\t\t\t\theaders: this.buildHeaders()\n\t\t\t});\n\t\t\tif (!response.ok) {\n\t\t\t\tgetLogger().warn('[ComputeServerStats] Failed to fetch active children:', response.status);\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tconst text = await response.text();\n\t\t\tconst count = parseInt(text.trim(), 10);\n\n\t\t\tif (isNaN(count)) {\n\t\t\t\tgetLogger().warn('[ComputeServerStats] Invalid active children response:', text);\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\treturn count;\n\t\t} catch (err) {\n\t\t\tgetLogger().warn('[ComputeServerStats] Error fetching active children:', err);\n\t\t\treturn null;\n\t\t}\n\t}\n\n\t/**\n\t * Get the server version information.\n\t *\n\t * @returns Version object with rhino, compute, and git_sha, or null if unavailable\n\t */\n\tpublic async getVersion(): Promise<{\n\t\trhino: string;\n\t\tcompute: string;\n\t\tgit_sha: string | null;\n\t} | null> {\n\t\tthis.ensureNotDisposed();\n\n\t\ttry {\n\t\t\tconst response = await fetch(`${this.serverUrl}/version`, {\n\t\t\t\theaders: this.buildHeaders()\n\t\t\t});\n\n\t\t\tif (!response.ok) {\n\t\t\t\tgetLogger().warn('[ComputeServerStats] Failed to fetch version:', response.status);\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\ttry {\n\t\t\t\tconst json = await response.json();\n\t\t\t\treturn {\n\t\t\t\t\trhino: json.rhino ?? '',\n\t\t\t\t\tcompute: json.compute ?? '',\n\t\t\t\t\tgit_sha: json.git_sha ?? null\n\t\t\t\t};\n\t\t\t} catch {\n\t\t\t\t// Fallback: parse as plain text\n\t\t\t\tconst text = await response.text();\n\t\t\t\treturn { rhino: text, compute: '', git_sha: null };\n\t\t\t}\n\t\t} catch (err) {\n\t\t\tgetLogger().warn('[ComputeServerStats] Error fetching version:', err);\n\t\t\treturn null;\n\t\t}\n\t}\n\n\t/**\n\t * Get comprehensive server statistics.\n\t * Fetches all available server information in parallel.\n\t *\n\t * @returns Object containing server status and available stats\n\t */\n\tpublic async getServerStats(): Promise<{\n\t\tisOnline: boolean;\n\t\tversion?: { rhino: string; compute: string; git_sha: string | null };\n\t\tactiveChildren?: number;\n\t}> {\n\t\tthis.ensureNotDisposed();\n\n\t\tconst isOnline = await this.isServerOnline();\n\n\t\tif (!isOnline) {\n\t\t\treturn { isOnline: false };\n\t\t}\n\n\t\tconst [version, activeChildren] = await Promise.all([\n\t\t\tthis.getVersion(),\n\t\t\tthis.getActiveChildren()\n\t\t]);\n\n\t\treturn {\n\t\t\tisOnline: true,\n\t\t\t...(version && { version }),\n\t\t\t...(activeChildren !== null && { activeChildren })\n\t\t};\n\t}\n\n\t/**\n\t * Continuously monitor server stats at specified interval.\n\t *\n\t * @param callback - Function called with stats on each interval\n\t * @param intervalMs - Milliseconds between checks (default: 5000)\n\t * @returns Function to stop monitoring\n\t *\n\t * @example\n\t * ```typescript\n\t * const stopMonitoring = stats.monitor((data) => {\n\t * console.log('Server stats:', data);\n\t * }, 3000);\n\t *\n\t * // Later...\n\t * stopMonitoring();\n\t * ```\n\t */\n\tpublic monitor(\n\t\tcallback: (stats: Awaited<ReturnType<typeof this.getServerStats>>) => void,\n\t\tintervalMs: number = 5000\n\t): () => void {\n\t\tthis.ensureNotDisposed();\n\n\t\tlet active = true;\n\t\tlet currentTimeoutId: ReturnType<typeof setTimeout> | null = null;\n\n\t\tgetLogger().info(`🔄 Starting server stats monitoring every ${intervalMs}ms`);\n\n\t\tconst check = async () => {\n\t\t\t// Clear current timeout from tracking since it has fired\n\t\t\tif (currentTimeoutId !== null) {\n\t\t\t\tthis.activeTimeouts.delete(currentTimeoutId);\n\t\t\t\tcurrentTimeoutId = null;\n\t\t\t}\n\n\t\t\tif (!active || this.disposed) return;\n\n\t\t\tconst _stats = await this.getServerStats();\n\n\t\t\t// Check again after async operation to prevent race condition\n\t\t\tif (!active || this.disposed) return;\n\n\t\t\tcallback(_stats);\n\n\t\t\tif (active && !this.disposed) {\n\t\t\t\tcurrentTimeoutId = setTimeout(() => void check(), intervalMs);\n\t\t\t\tthis.activeTimeouts.add(currentTimeoutId);\n\t\t\t}\n\t\t};\n\n\t\tconst stopMonitoring = () => {\n\t\t\tactive = false;\n\n\t\t\t// Clear any pending timeout\n\t\t\tif (currentTimeoutId !== null) {\n\t\t\t\tclearTimeout(currentTimeoutId);\n\t\t\t\tthis.activeTimeouts.delete(currentTimeoutId);\n\t\t\t\tcurrentTimeoutId = null;\n\t\t\t}\n\n\t\t\tthis.activeMonitors.delete(stopMonitoring);\n\t\t};\n\n\t\tthis.activeMonitors.add(stopMonitoring);\n\n\t\t// Explicitly mark as fire-and-forget since we don't need to await the initial call\n\t\tvoid check();\n\n\t\treturn stopMonitoring;\n\t}\n\n\t/**\n\t * Disposes of all resources and stops all active monitors.\n\t * Call this when you're done using the stats instance.\n\t */\n\tpublic async dispose(): Promise<void> {\n\t\tif (this.disposed) return;\n\n\t\tthis.disposed = true;\n\n\t\t// Stop all active monitors (this will also clear their timeouts)\n\t\tfor (const stopMonitor of this.activeMonitors) {\n\t\t\tstopMonitor();\n\t\t}\n\t\tthis.activeMonitors.clear();\n\n\t\t// Clear any remaining timeouts (defensive cleanup)\n\t\tfor (const timeoutId of this.activeTimeouts) {\n\t\t\tclearTimeout(timeoutId);\n\t\t}\n\t\tthis.activeTimeouts.clear();\n\t}\n\n\t/**\n\t * Ensures the instance hasn't been disposed.\n\t */\n\tprivate ensureNotDisposed(): void {\n\t\tif (this.disposed) {\n\t\t\tthrow new RhinoComputeError(\n\t\t\t\t'ComputeServerStats has been disposed and cannot be used',\n\t\t\t\tErrorCodes.INVALID_STATE,\n\t\t\t\t{ context: { disposed: this.disposed } }\n\t\t\t);\n\t\t}\n\t}\n}\n"]}
|