@babylonjs/core 9.9.2 → 9.10.1

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.
Files changed (34) hide show
  1. package/DeviceInput/webDeviceInputSystem.js +7 -1
  2. package/DeviceInput/webDeviceInputSystem.js.map +1 -1
  3. package/Engines/abstractEngine.pure.js +2 -2
  4. package/Engines/abstractEngine.pure.js.map +1 -1
  5. package/Engines/engine.common.js +61 -1
  6. package/Engines/engine.common.js.map +1 -1
  7. package/Materials/GaussianSplatting/gaussianSplattingDebugMaterialPlugin.pure.d.ts +8 -0
  8. package/Materials/GaussianSplatting/gaussianSplattingDebugMaterialPlugin.pure.js +26 -0
  9. package/Materials/GaussianSplatting/gaussianSplattingDebugMaterialPlugin.pure.js.map +1 -1
  10. package/Meshes/GaussianSplatting/gaussianSplattingDebugger.pure.d.ts +2 -0
  11. package/Meshes/GaussianSplatting/gaussianSplattingDebugger.pure.js +17 -0
  12. package/Meshes/GaussianSplatting/gaussianSplattingDebugger.pure.js.map +1 -1
  13. package/Meshes/GaussianSplatting/gaussianSplattingMesh.pure.js +4 -0
  14. package/Meshes/GaussianSplatting/gaussianSplattingMesh.pure.js.map +1 -1
  15. package/Meshes/GaussianSplatting/gaussianSplattingMeshBase.pure.d.ts +6 -0
  16. package/Meshes/GaussianSplatting/gaussianSplattingMeshBase.pure.js +8 -0
  17. package/Meshes/GaussianSplatting/gaussianSplattingMeshBase.pure.js.map +1 -1
  18. package/Misc/tools.pure.js +1 -1
  19. package/Misc/tools.pure.js.map +1 -1
  20. package/Shaders/ShadersInclude/pbrBlockFinalLitComponents.js +1 -1
  21. package/Shaders/ShadersInclude/pbrBlockFinalLitComponents.js.map +1 -1
  22. package/ShadersWGSL/ShadersInclude/pbrBlockFinalLitComponents.js +1 -1
  23. package/ShadersWGSL/ShadersInclude/pbrBlockFinalLitComponents.js.map +1 -1
  24. package/SmartAssets/index.d.ts +1 -1
  25. package/SmartAssets/index.js +1 -1
  26. package/SmartAssets/index.js.map +1 -1
  27. package/SmartAssets/pure.d.ts +1 -0
  28. package/SmartAssets/pure.js +1 -0
  29. package/SmartAssets/pure.js.map +1 -1
  30. package/SmartAssets/{smartAssetManager.js → smartAssetManager.pure.js} +30 -17
  31. package/SmartAssets/smartAssetManager.pure.js.map +1 -0
  32. package/package.json +1 -1
  33. package/SmartAssets/smartAssetManager.js.map +0 -1
  34. /package/SmartAssets/{smartAssetManager.d.ts → smartAssetManager.pure.d.ts} +0 -0
@@ -117,6 +117,20 @@ export function _CommonDispose(commonEngine, canvas) {
117
117
  * @returns an object containing ascent, height and descent
118
118
  */
119
119
  export function GetFontOffset(font) {
120
+ const domFontOffset = GetFontOffsetFromDom(font);
121
+ if (domFontOffset) {
122
+ return domFontOffset;
123
+ }
124
+ const canvasFontOffset = GetFontOffsetFromCanvas(font);
125
+ if (canvasFontOffset) {
126
+ return canvasFontOffset;
127
+ }
128
+ return GetFallbackFontOffset(font);
129
+ }
130
+ function GetFontOffsetFromDom(font) {
131
+ if (!IsDocumentAvailable() || !document.body) {
132
+ return null;
133
+ }
120
134
  const text = document.createElement("span");
121
135
  text.textContent = "Hg";
122
136
  text.style.font = font;
@@ -140,7 +154,53 @@ export function GetFontOffset(font) {
140
154
  finally {
141
155
  document.body.removeChild(div);
142
156
  }
143
- return { ascent: fontAscent, height: fontHeight, descent: fontHeight - fontAscent };
157
+ const offset = { ascent: fontAscent, height: fontHeight, descent: fontHeight - fontAscent };
158
+ return IsValidFontOffset(offset) ? offset : null;
159
+ }
160
+ function GetFontOffsetFromCanvas(font) {
161
+ let canvas = null;
162
+ try {
163
+ if (typeof OffscreenCanvas !== "undefined") {
164
+ canvas = new OffscreenCanvas(64, 64);
165
+ }
166
+ else if (IsDocumentAvailable() && typeof document.createElement === "function") {
167
+ canvas = document.createElement("canvas");
168
+ canvas.width = 64;
169
+ canvas.height = 64;
170
+ }
171
+ const context = canvas?.getContext("2d");
172
+ if (!context) {
173
+ return null;
174
+ }
175
+ context.font = font;
176
+ const metrics = context.measureText("Hg");
177
+ const ascent = Number(metrics.actualBoundingBoxAscent ?? metrics.fontBoundingBoxAscent);
178
+ const descent = Number(metrics.actualBoundingBoxDescent ?? metrics.fontBoundingBoxDescent);
179
+ const offset = { ascent, height: ascent + descent, descent };
180
+ return IsValidFontOffset(offset) ? offset : null;
181
+ }
182
+ catch {
183
+ return null;
184
+ }
185
+ finally {
186
+ const disposableCanvas = canvas;
187
+ if (typeof disposableCanvas?.dispose === "function") {
188
+ disposableCanvas.dispose();
189
+ }
190
+ }
191
+ }
192
+ function GetFallbackFontOffset(font) {
193
+ const size = Math.max(1, GetCssPixelFontSize(font));
194
+ const ascent = size * 0.8;
195
+ const descent = size * 0.2;
196
+ return { ascent, height: ascent + descent, descent };
197
+ }
198
+ function GetCssPixelFontSize(font) {
199
+ const match = /(?:^|\s)([0-9]+(?:\.[0-9]+)?)px(?:\/|\s|$)/.exec(String(font || ""));
200
+ return match ? Number(match[1]) : 16;
201
+ }
202
+ function IsValidFontOffset(offset) {
203
+ return Number.isFinite(offset.ascent) && Number.isFinite(offset.height) && Number.isFinite(offset.descent) && offset.height > 0;
144
204
  }
145
205
  /** @internal */
146
206
  export async function CreateImageBitmapFromSource(engine, imageSource, options) {
@@ -1 +1 @@
1
- {"version":3,"file":"engine.common.js","sourceRoot":"","sources":["../../../../dev/core/src/Engines/engine.common.ts"],"names":[],"mappings":"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","sourcesContent":["import { IsDocumentAvailable } from \"../Misc/domManagement\";\r\nimport { type Nullable } from \"../types\";\r\nimport { AbstractEngine, type AbstractEngineOptions } from \"./abstractEngine\";\r\nimport { EngineStore } from \"./engineStore\";\r\n\r\n/** @internal */\r\nfunction DisableTouchAction(canvas: Nullable<HTMLCanvasElement>): void {\r\n if (!canvas || !canvas.setAttribute) {\r\n return;\r\n }\r\n\r\n canvas.setAttribute(\"touch-action\", \"none\");\r\n canvas.style.touchAction = \"none\";\r\n (canvas.style as any).webkitTapHighlightColor = \"transparent\";\r\n}\r\n\r\n/** @internal */\r\nexport function _CommonInit(commonEngine: AbstractEngine, canvas: HTMLCanvasElement, creationOptions: AbstractEngineOptions) {\r\n commonEngine._onCanvasFocus = () => {\r\n commonEngine.onCanvasFocusObservable.notifyObservers(commonEngine);\r\n };\r\n\r\n commonEngine._onCanvasBlur = () => {\r\n commonEngine.onCanvasBlurObservable.notifyObservers(commonEngine);\r\n };\r\n\r\n commonEngine._onCanvasContextMenu = (evt: Event) => {\r\n if (commonEngine.disableContextMenu) {\r\n evt.preventDefault();\r\n }\r\n };\r\n\r\n canvas.addEventListener(\"focus\", commonEngine._onCanvasFocus);\r\n canvas.addEventListener(\"blur\", commonEngine._onCanvasBlur);\r\n canvas.addEventListener(\"contextmenu\", commonEngine._onCanvasContextMenu);\r\n\r\n commonEngine._onBlur = () => {\r\n if (commonEngine.disablePerformanceMonitorInBackground) {\r\n commonEngine.performanceMonitor.disable();\r\n }\r\n commonEngine._windowIsBackground = true;\r\n };\r\n\r\n commonEngine._onFocus = () => {\r\n if (commonEngine.disablePerformanceMonitorInBackground) {\r\n commonEngine.performanceMonitor.enable();\r\n }\r\n commonEngine._windowIsBackground = false;\r\n };\r\n\r\n commonEngine._onCanvasPointerOut = (ev) => {\r\n // Check that the element at the point of the pointer out isn't the canvas and if it isn't, notify observers\r\n // Note: This is a workaround for a bug with Safari\r\n if (document.elementFromPoint(ev.clientX, ev.clientY) !== canvas) {\r\n commonEngine.onCanvasPointerOutObservable.notifyObservers(ev);\r\n }\r\n };\r\n\r\n const hostWindow = commonEngine.getHostWindow(); // it calls IsWindowObjectExist()\r\n if (hostWindow && typeof hostWindow.addEventListener === \"function\") {\r\n hostWindow.addEventListener(\"blur\", commonEngine._onBlur);\r\n hostWindow.addEventListener(\"focus\", commonEngine._onFocus);\r\n }\r\n\r\n canvas.addEventListener(\"pointerout\", commonEngine._onCanvasPointerOut);\r\n\r\n if (!creationOptions.doNotHandleTouchAction) {\r\n DisableTouchAction(canvas);\r\n }\r\n\r\n // Create Audio Engine if needed.\r\n if (!AbstractEngine.audioEngine && creationOptions.audioEngine && AbstractEngine.AudioEngineFactory) {\r\n AbstractEngine.audioEngine = AbstractEngine.AudioEngineFactory(commonEngine.getRenderingCanvas(), commonEngine.getAudioContext(), commonEngine.getAudioDestination());\r\n }\r\n\r\n if (IsDocumentAvailable()) {\r\n // Fullscreen\r\n commonEngine._onFullscreenChange = () => {\r\n commonEngine.isFullscreen = !!document.fullscreenElement;\r\n\r\n // Pointer lock\r\n if (commonEngine.isFullscreen && commonEngine._pointerLockRequested && canvas) {\r\n RequestPointerlock(canvas);\r\n }\r\n };\r\n\r\n document.addEventListener(\"fullscreenchange\", commonEngine._onFullscreenChange, false);\r\n document.addEventListener(\"webkitfullscreenchange\", commonEngine._onFullscreenChange, false);\r\n\r\n // Pointer lock\r\n commonEngine._onPointerLockChange = () => {\r\n commonEngine.isPointerLock = document.pointerLockElement === canvas;\r\n };\r\n\r\n document.addEventListener(\"pointerlockchange\", commonEngine._onPointerLockChange, false);\r\n document.addEventListener(\"webkitpointerlockchange\", commonEngine._onPointerLockChange, false);\r\n }\r\n\r\n commonEngine.enableOfflineSupport = AbstractEngine.OfflineProviderFactory !== undefined;\r\n\r\n commonEngine._deterministicLockstep = !!creationOptions.deterministicLockstep;\r\n commonEngine._lockstepMaxSteps = creationOptions.lockstepMaxSteps || 0;\r\n commonEngine._timeStep = creationOptions.timeStep || 1 / 60;\r\n}\r\n\r\n/** @internal */\r\nexport function _CommonDispose(commonEngine: AbstractEngine, canvas: Nullable<HTMLCanvasElement>) {\r\n // Release audio engine\r\n if (EngineStore.Instances.length === 1 && AbstractEngine.audioEngine) {\r\n AbstractEngine.audioEngine.dispose();\r\n AbstractEngine.audioEngine = null;\r\n }\r\n\r\n // Events\r\n const hostWindow = commonEngine.getHostWindow(); // it calls IsWindowObjectExist()\r\n if (hostWindow && typeof hostWindow.removeEventListener === \"function\") {\r\n hostWindow.removeEventListener(\"blur\", commonEngine._onBlur);\r\n hostWindow.removeEventListener(\"focus\", commonEngine._onFocus);\r\n }\r\n\r\n if (canvas) {\r\n canvas.removeEventListener(\"focus\", commonEngine._onCanvasFocus);\r\n canvas.removeEventListener(\"blur\", commonEngine._onCanvasBlur);\r\n canvas.removeEventListener(\"pointerout\", commonEngine._onCanvasPointerOut);\r\n canvas.removeEventListener(\"contextmenu\", commonEngine._onCanvasContextMenu);\r\n }\r\n\r\n if (IsDocumentAvailable()) {\r\n document.removeEventListener(\"fullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"mozfullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"webkitfullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"msfullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"pointerlockchange\", commonEngine._onPointerLockChange);\r\n document.removeEventListener(\"mspointerlockchange\", commonEngine._onPointerLockChange);\r\n document.removeEventListener(\"mozpointerlockchange\", commonEngine._onPointerLockChange);\r\n document.removeEventListener(\"webkitpointerlockchange\", commonEngine._onPointerLockChange);\r\n }\r\n}\r\n\r\n/**\r\n * Get Font size information\r\n * @param font font name\r\n * @returns an object containing ascent, height and descent\r\n */\r\nexport function GetFontOffset(font: string): { ascent: number; height: number; descent: number } {\r\n const text = document.createElement(\"span\");\r\n text.textContent = \"Hg\";\r\n text.style.font = font;\r\n\r\n const block = document.createElement(\"div\");\r\n block.style.display = \"inline-block\";\r\n block.style.width = \"1px\";\r\n block.style.height = \"0px\";\r\n block.style.verticalAlign = \"bottom\";\r\n\r\n const div = document.createElement(\"div\");\r\n div.style.whiteSpace = \"nowrap\";\r\n div.appendChild(text);\r\n div.appendChild(block);\r\n\r\n document.body.appendChild(div);\r\n\r\n let fontAscent: number;\r\n let fontHeight: number;\r\n try {\r\n fontHeight = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\r\n block.style.verticalAlign = \"baseline\";\r\n fontAscent = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\r\n } finally {\r\n document.body.removeChild(div);\r\n }\r\n return { ascent: fontAscent, height: fontHeight, descent: fontHeight - fontAscent };\r\n}\r\n\r\n/** @internal */\r\nexport async function CreateImageBitmapFromSource(engine: AbstractEngine, imageSource: string, options?: ImageBitmapOptions): Promise<ImageBitmap> {\r\n return await new Promise<ImageBitmap>((resolve, reject) => {\r\n const image = new Image();\r\n image.onload = () => {\r\n // eslint-disable-next-line @typescript-eslint/no-floating-promises, github/no-then\r\n image.decode().then(() => {\r\n // eslint-disable-next-line @typescript-eslint/no-floating-promises, github/no-then\r\n engine.createImageBitmap(image, options).then((imageBitmap) => {\r\n resolve(imageBitmap);\r\n });\r\n });\r\n };\r\n image.onerror = () => {\r\n // eslint-disable-next-line @typescript-eslint/prefer-promise-reject-errors\r\n reject(`Error loading image ${image.src}`);\r\n };\r\n\r\n image.src = imageSource;\r\n });\r\n}\r\n\r\n/** @internal */\r\nexport function ResizeImageBitmap(engine: AbstractEngine, image: HTMLImageElement | ImageBitmap, bufferWidth: number, bufferHeight: number): Uint8Array {\r\n const canvas = engine.createCanvas(bufferWidth, bufferHeight);\r\n const context = canvas.getContext(\"2d\");\r\n\r\n if (!context) {\r\n throw new Error(\"Unable to get 2d context for resizeImageBitmap\");\r\n }\r\n\r\n context.drawImage(image, 0, 0);\r\n\r\n // Create VertexData from map data\r\n // Cast is due to wrong definition in lib.d.ts from ts 1.3 - https://github.com/Microsoft/TypeScript/issues/949\r\n const buffer = <Uint8Array>(<any>context.getImageData(0, 0, bufferWidth, bufferHeight).data);\r\n return buffer;\r\n}\r\n\r\n/**\r\n * Ask the browser to promote the current element to fullscreen rendering mode\r\n * @param element defines the DOM element to promote\r\n */\r\nexport function RequestFullscreen(element: HTMLElement): void {\r\n const requestFunction = element.requestFullscreen || (<any>element).webkitRequestFullscreen;\r\n if (!requestFunction) {\r\n return;\r\n }\r\n void requestFunction.call(element);\r\n}\r\n\r\n/**\r\n * Asks the browser to exit fullscreen mode\r\n */\r\nexport function ExitFullscreen(): void {\r\n const anyDoc = document as any;\r\n\r\n if (document.exitFullscreen) {\r\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\r\n document.exitFullscreen();\r\n } else if (anyDoc.webkitCancelFullScreen) {\r\n anyDoc.webkitCancelFullScreen();\r\n }\r\n}\r\n\r\n/**\r\n * Ask the browser to promote the current element to pointerlock mode\r\n * @param element defines the DOM element to promote\r\n */\r\nexport function RequestPointerlock(element: HTMLElement): void {\r\n if (element.requestPointerLock) {\r\n // In some browsers, requestPointerLock returns a promise.\r\n // Handle possible rejections to avoid an unhandled top-level exception.\r\n const promise: unknown = element.requestPointerLock();\r\n if (promise instanceof Promise) {\r\n promise\r\n // eslint-disable-next-line github/no-then\r\n .then(() => {\r\n element.focus();\r\n })\r\n // eslint-disable-next-line github/no-then\r\n .catch(() => {});\r\n } else {\r\n element.focus();\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Asks the browser to exit pointerlock mode\r\n */\r\nexport function ExitPointerlock(): void {\r\n if (document.exitPointerLock) {\r\n document.exitPointerLock();\r\n }\r\n}\r\n"]}
1
+ {"version":3,"file":"engine.common.js","sourceRoot":"","sources":["../../../../dev/core/src/Engines/engine.common.ts"],"names":[],"mappings":"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","sourcesContent":["import { IsDocumentAvailable } from \"../Misc/domManagement\";\r\nimport { type Nullable } from \"../types\";\r\nimport { AbstractEngine, type AbstractEngineOptions } from \"./abstractEngine\";\r\nimport { EngineStore } from \"./engineStore\";\r\n\r\n/** @internal */\r\nfunction DisableTouchAction(canvas: Nullable<HTMLCanvasElement>): void {\r\n if (!canvas || !canvas.setAttribute) {\r\n return;\r\n }\r\n\r\n canvas.setAttribute(\"touch-action\", \"none\");\r\n canvas.style.touchAction = \"none\";\r\n (canvas.style as any).webkitTapHighlightColor = \"transparent\";\r\n}\r\n\r\n/** @internal */\r\nexport function _CommonInit(commonEngine: AbstractEngine, canvas: HTMLCanvasElement, creationOptions: AbstractEngineOptions) {\r\n commonEngine._onCanvasFocus = () => {\r\n commonEngine.onCanvasFocusObservable.notifyObservers(commonEngine);\r\n };\r\n\r\n commonEngine._onCanvasBlur = () => {\r\n commonEngine.onCanvasBlurObservable.notifyObservers(commonEngine);\r\n };\r\n\r\n commonEngine._onCanvasContextMenu = (evt: Event) => {\r\n if (commonEngine.disableContextMenu) {\r\n evt.preventDefault();\r\n }\r\n };\r\n\r\n canvas.addEventListener(\"focus\", commonEngine._onCanvasFocus);\r\n canvas.addEventListener(\"blur\", commonEngine._onCanvasBlur);\r\n canvas.addEventListener(\"contextmenu\", commonEngine._onCanvasContextMenu);\r\n\r\n commonEngine._onBlur = () => {\r\n if (commonEngine.disablePerformanceMonitorInBackground) {\r\n commonEngine.performanceMonitor.disable();\r\n }\r\n commonEngine._windowIsBackground = true;\r\n };\r\n\r\n commonEngine._onFocus = () => {\r\n if (commonEngine.disablePerformanceMonitorInBackground) {\r\n commonEngine.performanceMonitor.enable();\r\n }\r\n commonEngine._windowIsBackground = false;\r\n };\r\n\r\n commonEngine._onCanvasPointerOut = (ev) => {\r\n // Check that the element at the point of the pointer out isn't the canvas and if it isn't, notify observers\r\n // Note: This is a workaround for a bug with Safari\r\n if (document.elementFromPoint(ev.clientX, ev.clientY) !== canvas) {\r\n commonEngine.onCanvasPointerOutObservable.notifyObservers(ev);\r\n }\r\n };\r\n\r\n const hostWindow = commonEngine.getHostWindow(); // it calls IsWindowObjectExist()\r\n if (hostWindow && typeof hostWindow.addEventListener === \"function\") {\r\n hostWindow.addEventListener(\"blur\", commonEngine._onBlur);\r\n hostWindow.addEventListener(\"focus\", commonEngine._onFocus);\r\n }\r\n\r\n canvas.addEventListener(\"pointerout\", commonEngine._onCanvasPointerOut);\r\n\r\n if (!creationOptions.doNotHandleTouchAction) {\r\n DisableTouchAction(canvas);\r\n }\r\n\r\n // Create Audio Engine if needed.\r\n if (!AbstractEngine.audioEngine && creationOptions.audioEngine && AbstractEngine.AudioEngineFactory) {\r\n AbstractEngine.audioEngine = AbstractEngine.AudioEngineFactory(commonEngine.getRenderingCanvas(), commonEngine.getAudioContext(), commonEngine.getAudioDestination());\r\n }\r\n\r\n if (IsDocumentAvailable()) {\r\n // Fullscreen\r\n commonEngine._onFullscreenChange = () => {\r\n commonEngine.isFullscreen = !!document.fullscreenElement;\r\n\r\n // Pointer lock\r\n if (commonEngine.isFullscreen && commonEngine._pointerLockRequested && canvas) {\r\n RequestPointerlock(canvas);\r\n }\r\n };\r\n\r\n document.addEventListener(\"fullscreenchange\", commonEngine._onFullscreenChange, false);\r\n document.addEventListener(\"webkitfullscreenchange\", commonEngine._onFullscreenChange, false);\r\n\r\n // Pointer lock\r\n commonEngine._onPointerLockChange = () => {\r\n commonEngine.isPointerLock = document.pointerLockElement === canvas;\r\n };\r\n\r\n document.addEventListener(\"pointerlockchange\", commonEngine._onPointerLockChange, false);\r\n document.addEventListener(\"webkitpointerlockchange\", commonEngine._onPointerLockChange, false);\r\n }\r\n\r\n commonEngine.enableOfflineSupport = AbstractEngine.OfflineProviderFactory !== undefined;\r\n\r\n commonEngine._deterministicLockstep = !!creationOptions.deterministicLockstep;\r\n commonEngine._lockstepMaxSteps = creationOptions.lockstepMaxSteps || 0;\r\n commonEngine._timeStep = creationOptions.timeStep || 1 / 60;\r\n}\r\n\r\n/** @internal */\r\nexport function _CommonDispose(commonEngine: AbstractEngine, canvas: Nullable<HTMLCanvasElement>) {\r\n // Release audio engine\r\n if (EngineStore.Instances.length === 1 && AbstractEngine.audioEngine) {\r\n AbstractEngine.audioEngine.dispose();\r\n AbstractEngine.audioEngine = null;\r\n }\r\n\r\n // Events\r\n const hostWindow = commonEngine.getHostWindow(); // it calls IsWindowObjectExist()\r\n if (hostWindow && typeof hostWindow.removeEventListener === \"function\") {\r\n hostWindow.removeEventListener(\"blur\", commonEngine._onBlur);\r\n hostWindow.removeEventListener(\"focus\", commonEngine._onFocus);\r\n }\r\n\r\n if (canvas) {\r\n canvas.removeEventListener(\"focus\", commonEngine._onCanvasFocus);\r\n canvas.removeEventListener(\"blur\", commonEngine._onCanvasBlur);\r\n canvas.removeEventListener(\"pointerout\", commonEngine._onCanvasPointerOut);\r\n canvas.removeEventListener(\"contextmenu\", commonEngine._onCanvasContextMenu);\r\n }\r\n\r\n if (IsDocumentAvailable()) {\r\n document.removeEventListener(\"fullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"mozfullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"webkitfullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"msfullscreenchange\", commonEngine._onFullscreenChange);\r\n document.removeEventListener(\"pointerlockchange\", commonEngine._onPointerLockChange);\r\n document.removeEventListener(\"mspointerlockchange\", commonEngine._onPointerLockChange);\r\n document.removeEventListener(\"mozpointerlockchange\", commonEngine._onPointerLockChange);\r\n document.removeEventListener(\"webkitpointerlockchange\", commonEngine._onPointerLockChange);\r\n }\r\n}\r\n\r\n/**\r\n * Get Font size information\r\n * @param font font name\r\n * @returns an object containing ascent, height and descent\r\n */\r\nexport function GetFontOffset(font: string): { ascent: number; height: number; descent: number } {\r\n const domFontOffset = GetFontOffsetFromDom(font);\r\n if (domFontOffset) {\r\n return domFontOffset;\r\n }\r\n\r\n const canvasFontOffset = GetFontOffsetFromCanvas(font);\r\n if (canvasFontOffset) {\r\n return canvasFontOffset;\r\n }\r\n\r\n return GetFallbackFontOffset(font);\r\n}\r\n\r\nfunction GetFontOffsetFromDom(font: string): Nullable<{ ascent: number; height: number; descent: number }> {\r\n if (!IsDocumentAvailable() || !document.body) {\r\n return null;\r\n }\r\n\r\n const text = document.createElement(\"span\");\r\n text.textContent = \"Hg\";\r\n text.style.font = font;\r\n\r\n const block = document.createElement(\"div\");\r\n block.style.display = \"inline-block\";\r\n block.style.width = \"1px\";\r\n block.style.height = \"0px\";\r\n block.style.verticalAlign = \"bottom\";\r\n\r\n const div = document.createElement(\"div\");\r\n div.style.whiteSpace = \"nowrap\";\r\n div.appendChild(text);\r\n div.appendChild(block);\r\n\r\n document.body.appendChild(div);\r\n\r\n let fontAscent: number;\r\n let fontHeight: number;\r\n try {\r\n fontHeight = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\r\n block.style.verticalAlign = \"baseline\";\r\n fontAscent = block.getBoundingClientRect().top - text.getBoundingClientRect().top;\r\n } finally {\r\n document.body.removeChild(div);\r\n }\r\n\r\n const offset = { ascent: fontAscent, height: fontHeight, descent: fontHeight - fontAscent };\r\n return IsValidFontOffset(offset) ? offset : null;\r\n}\r\n\r\nfunction GetFontOffsetFromCanvas(font: string): Nullable<{ ascent: number; height: number; descent: number }> {\r\n let canvas: Nullable<OffscreenCanvas | HTMLCanvasElement> = null;\r\n try {\r\n if (typeof OffscreenCanvas !== \"undefined\") {\r\n canvas = new OffscreenCanvas(64, 64);\r\n } else if (IsDocumentAvailable() && typeof document.createElement === \"function\") {\r\n canvas = document.createElement(\"canvas\");\r\n canvas.width = 64;\r\n canvas.height = 64;\r\n }\r\n\r\n const context = canvas?.getContext(\"2d\") as Nullable<CanvasRenderingContext2D | OffscreenCanvasRenderingContext2D>;\r\n if (!context) {\r\n return null;\r\n }\r\n\r\n context.font = font;\r\n const metrics = context.measureText(\"Hg\");\r\n const ascent = Number(metrics.actualBoundingBoxAscent ?? metrics.fontBoundingBoxAscent);\r\n const descent = Number(metrics.actualBoundingBoxDescent ?? metrics.fontBoundingBoxDescent);\r\n const offset = { ascent, height: ascent + descent, descent };\r\n return IsValidFontOffset(offset) ? offset : null;\r\n } catch {\r\n return null;\r\n } finally {\r\n const disposableCanvas = canvas as Nullable<{ dispose?: () => void }>;\r\n if (typeof disposableCanvas?.dispose === \"function\") {\r\n disposableCanvas.dispose();\r\n }\r\n }\r\n}\r\n\r\nfunction GetFallbackFontOffset(font: string): { ascent: number; height: number; descent: number } {\r\n const size = Math.max(1, GetCssPixelFontSize(font));\r\n const ascent = size * 0.8;\r\n const descent = size * 0.2;\r\n return { ascent, height: ascent + descent, descent };\r\n}\r\n\r\nfunction GetCssPixelFontSize(font: string): number {\r\n const match = /(?:^|\\s)([0-9]+(?:\\.[0-9]+)?)px(?:\\/|\\s|$)/.exec(String(font || \"\"));\r\n return match ? Number(match[1]) : 16;\r\n}\r\n\r\nfunction IsValidFontOffset(offset: { ascent: number; height: number; descent: number }): boolean {\r\n return Number.isFinite(offset.ascent) && Number.isFinite(offset.height) && Number.isFinite(offset.descent) && offset.height > 0;\r\n}\r\n\r\n/** @internal */\r\nexport async function CreateImageBitmapFromSource(engine: AbstractEngine, imageSource: string, options?: ImageBitmapOptions): Promise<ImageBitmap> {\r\n return await new Promise<ImageBitmap>((resolve, reject) => {\r\n const image = new Image();\r\n image.onload = () => {\r\n // eslint-disable-next-line @typescript-eslint/no-floating-promises, github/no-then\r\n image.decode().then(() => {\r\n // eslint-disable-next-line @typescript-eslint/no-floating-promises, github/no-then\r\n engine.createImageBitmap(image, options).then((imageBitmap) => {\r\n resolve(imageBitmap);\r\n });\r\n });\r\n };\r\n image.onerror = () => {\r\n // eslint-disable-next-line @typescript-eslint/prefer-promise-reject-errors\r\n reject(`Error loading image ${image.src}`);\r\n };\r\n\r\n image.src = imageSource;\r\n });\r\n}\r\n\r\n/** @internal */\r\nexport function ResizeImageBitmap(engine: AbstractEngine, image: HTMLImageElement | ImageBitmap, bufferWidth: number, bufferHeight: number): Uint8Array {\r\n const canvas = engine.createCanvas(bufferWidth, bufferHeight);\r\n const context = canvas.getContext(\"2d\");\r\n\r\n if (!context) {\r\n throw new Error(\"Unable to get 2d context for resizeImageBitmap\");\r\n }\r\n\r\n context.drawImage(image, 0, 0);\r\n\r\n // Create VertexData from map data\r\n // Cast is due to wrong definition in lib.d.ts from ts 1.3 - https://github.com/Microsoft/TypeScript/issues/949\r\n const buffer = <Uint8Array>(<any>context.getImageData(0, 0, bufferWidth, bufferHeight).data);\r\n return buffer;\r\n}\r\n\r\n/**\r\n * Ask the browser to promote the current element to fullscreen rendering mode\r\n * @param element defines the DOM element to promote\r\n */\r\nexport function RequestFullscreen(element: HTMLElement): void {\r\n const requestFunction = element.requestFullscreen || (<any>element).webkitRequestFullscreen;\r\n if (!requestFunction) {\r\n return;\r\n }\r\n void requestFunction.call(element);\r\n}\r\n\r\n/**\r\n * Asks the browser to exit fullscreen mode\r\n */\r\nexport function ExitFullscreen(): void {\r\n const anyDoc = document as any;\r\n\r\n if (document.exitFullscreen) {\r\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\r\n document.exitFullscreen();\r\n } else if (anyDoc.webkitCancelFullScreen) {\r\n anyDoc.webkitCancelFullScreen();\r\n }\r\n}\r\n\r\n/**\r\n * Ask the browser to promote the current element to pointerlock mode\r\n * @param element defines the DOM element to promote\r\n */\r\nexport function RequestPointerlock(element: HTMLElement): void {\r\n if (element.requestPointerLock) {\r\n // In some browsers, requestPointerLock returns a promise.\r\n // Handle possible rejections to avoid an unhandled top-level exception.\r\n const promise: unknown = element.requestPointerLock();\r\n if (promise instanceof Promise) {\r\n promise\r\n // eslint-disable-next-line github/no-then\r\n .then(() => {\r\n element.focus();\r\n })\r\n // eslint-disable-next-line github/no-then\r\n .catch(() => {});\r\n } else {\r\n element.focus();\r\n }\r\n }\r\n}\r\n\r\n/**\r\n * Asks the browser to exit pointerlock mode\r\n */\r\nexport function ExitPointerlock(): void {\r\n if (document.exitPointerLock) {\r\n document.exitPointerLock();\r\n }\r\n}\r\n"]}
@@ -102,6 +102,7 @@ export declare class GaussianSplattingDebugMaterialPlugin extends MaterialPlugin
102
102
  private _partShOrder2s;
103
103
  private _partShOrder3s;
104
104
  private _partShOrder4s;
105
+ private readonly _partArrays;
105
106
  private _dbgPartDataTexture;
106
107
  private _textureDirty;
107
108
  private _maxPartCount;
@@ -132,6 +133,13 @@ export declare class GaussianSplattingDebugMaterialPlugin extends MaterialPlugin
132
133
  * @param partIndex The zero-based part index.
133
134
  */
134
135
  clearPartOptions(partIndex: number): void;
136
+ /**
137
+ * Removes the per-part override slot at `removedIndex` and shifts all higher-indexed
138
+ * slots down by one, keeping the arrays aligned with the compound mesh's new part layout.
139
+ * @param removedIndex The original (pre-removal) part index.
140
+ * @internal
141
+ */
142
+ shiftPartOptions(removedIndex: number): void;
135
143
  /**
136
144
  * World-space axis-aligned clipping box. Splats outside are not rendered.
137
145
  * Set to null to disable clipping.
@@ -77,6 +77,18 @@ export class GaussianSplattingDebugMaterialPlugin extends MaterialPluginBase {
77
77
  this._dbgPartDataTexture = null;
78
78
  this._textureDirty = false;
79
79
  this._maxPartCount = 0;
80
+ this._partArrays = [
81
+ this._partClippingBoxes,
82
+ this._partOpacityCullings,
83
+ this._partSizeCullings,
84
+ this._partOpacityScales,
85
+ this._partOpacitySaturates,
86
+ this._partShDcs,
87
+ this._partShOrder1s,
88
+ this._partShOrder2s,
89
+ this._partShOrder3s,
90
+ this._partShOrder4s,
91
+ ];
80
92
  }
81
93
  _isAnyFeatureActive() {
82
94
  return (this._clippingBox !== null ||
@@ -180,6 +192,20 @@ export class GaussianSplattingDebugMaterialPlugin extends MaterialPluginBase {
180
192
  this._partShOrder4s[partIndex] = null;
181
193
  this._markDirty();
182
194
  }
195
+ /**
196
+ * Removes the per-part override slot at `removedIndex` and shifts all higher-indexed
197
+ * slots down by one, keeping the arrays aligned with the compound mesh's new part layout.
198
+ * @param removedIndex The original (pre-removal) part index.
199
+ * @internal
200
+ */
201
+ shiftPartOptions(removedIndex) {
202
+ for (const arr of this._partArrays) {
203
+ if (removedIndex < arr.length) {
204
+ arr.splice(removedIndex, 1);
205
+ }
206
+ }
207
+ this._markDirty();
208
+ }
183
209
  /**
184
210
  * World-space axis-aligned clipping box. Splats outside are not rendered.
185
211
  * Set to null to disable clipping.
@@ -1 +1 @@
1
- {"version":3,"file":"gaussianSplattingDebugMaterialPlugin.pure.js","sourceRoot":"","sources":["../../../../../dev/core/src/Materials/GaussianSplatting/gaussianSplattingDebugMaterialPlugin.pure.ts"],"names":[],"mappings":"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","sourcesContent":["/** This file must only contain pure code and pure imports */\n\nimport { type Nullable } from \"../../types\";\nimport { type Scene } from \"../../scene\";\nimport { type AbstractEngine } from \"../../Engines/abstractEngine\";\nimport { type SubMesh } from \"../../Meshes/subMesh\";\nimport { type UniformBuffer } from \"../uniformBuffer\";\nimport { MaterialDefines } from \"../materialDefines\";\nimport { MaterialPluginBase } from \"../materialPluginBase.pure\";\nimport { ShaderLanguage } from \"../shaderLanguage\";\nimport { RegisterClass } from \"../../Misc/typeStore\";\nimport { type Vector3 } from \"../../Maths/math.vector\";\nimport { Logger } from \"../../Misc/logger\";\nimport { type GaussianSplattingMaterial, GetGaussianSplattingMaxPartCount } from \"./gaussianSplattingMaterial.pure\";\nimport { RawTexture } from \"../Textures/rawTexture\";\nimport { Constants } from \"../../Engines/constants\";\n\n/** @internal */\nclass GaussianSplattingDebugDefines extends MaterialDefines {\n /** Defines whether any debug feature is active */\n GS_DBG_ENABLED: boolean = false;\n /** Defines whether world-space clipping box is enabled (0: off, 1: on) */\n GS_DBG_CLIP: number = 0;\n /** Defines whether opacity culling is enabled (0: off, 1: on) */\n GS_DBG_CULL_OPACITY: number = 0;\n /** Defines whether size culling is enabled (0: off, 1: on) */\n GS_DBG_CULL_SIZE: number = 0;\n /** Defines whether per-splat opacity scaling is enabled (0: off, 1: on) */\n GS_DBG_OPACITY_SCALE: number = 0;\n /** Defines whether opacity saturation (flat disk) is enabled (0: off, 1: on) */\n GS_DBG_OPACITY_SATURATE: number = 0;\n /** Defines whether the DC (base) SH color is included (0: off, 1: on) */\n GS_DBG_SH_DC: number = 1;\n /** Defines whether SH band 1 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER1: number = 1;\n /** Defines whether SH band 2 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER2: number = 1;\n /** Defines whether SH band 3 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER3: number = 1;\n /** Defines whether SH band 4 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER4: number = 1;\n}\n\n/**\n * Per-part debug options for compound Gaussian splat meshes.\n * Each field is optional; unset fields fall back to the global setting on the plugin,\n * and if that is also unset, a neutral value is used (no culling, full SH, etc.).\n */\nexport interface IGaussianSplattingDebugOptions {\n /** World-space axis-aligned clipping box, or null to disable. */\n clippingBox: Nullable<{ min: Vector3; max: Vector3 }>;\n /** Opacity culling range [0..1], or null to disable. */\n opacityCulling: Nullable<{ min: number; max: number }>;\n /** Size culling range, or null to disable. */\n sizeCulling: Nullable<{ min: number; max: number }>;\n /** Scalar opacity multiplier. 1.0 = no change. */\n opacityScale: number;\n /** When true, replaces Gaussian falloff with flat disk opacity. */\n opacitySaturate: boolean;\n /** Include the DC (base) SH color. */\n shDc: boolean;\n /** Include SH band 1 contribution. */\n shOrder1: boolean;\n /** Include SH band 2 contribution. */\n shOrder2: boolean;\n /** Include SH band 3 contribution. */\n shOrder3: boolean;\n /** Include SH band 4 contribution. */\n shOrder4: boolean;\n}\n\n/**\n * Debug plugin for GaussianSplattingMaterial.\n * Provides runtime controls for clipping, opacity/size culling, opacity scaling,\n * opacity saturation, and per-SH-order toggling. All features are gated behind\n * the GS_DBG_ENABLED shader define — when every option is at its default value\n * the define is absent and the shader compiles to identical code as without the plugin.\n *\n * In compound mode (partCount \\> 0), per-part overrides can be set via setPartOptions().\n * Global settings act as defaults; per-part settings override them for that part index.\n */\nexport class GaussianSplattingDebugMaterialPlugin extends MaterialPluginBase {\n private _clippingBox: Nullable<{ min: Vector3; max: Vector3 }> = null;\n private _opacityCulling: Nullable<{ min: number; max: number }> = null;\n private _sizeCulling: Nullable<{ min: number; max: number }> = null;\n private _opacityScale: number = 1.0;\n private _opacitySaturate: boolean = false;\n private _shDc: boolean = true;\n private _shOrder1: boolean = true;\n private _shOrder2: boolean = true;\n private _shOrder3: boolean = true;\n private _shOrder4: boolean = true;\n\n // Per-part state (only populated when partCount > 0)\n private _partCount: number = 0;\n private _partClippingBoxes: Array<Nullable<{ min: Vector3; max: Vector3 }>> = [];\n private _partOpacityCullings: Array<Nullable<{ min: number; max: number }>> = [];\n private _partSizeCullings: Array<Nullable<{ min: number; max: number }>> = [];\n private _partOpacityScales: Array<Nullable<number>> = [];\n private _partOpacitySaturates: Array<Nullable<boolean>> = [];\n private _partShDcs: Array<Nullable<boolean>> = [];\n private _partShOrder1s: Array<Nullable<boolean>> = [];\n private _partShOrder2s: Array<Nullable<boolean>> = [];\n private _partShOrder3s: Array<Nullable<boolean>> = [];\n private _partShOrder4s: Array<Nullable<boolean>> = [];\n\n // Per-part debug LUT: MAX_PART_COUNT × 5 RGBA float texture (dbgPartData)\n private _dbgPartDataTexture: Nullable<RawTexture> = null;\n private _textureDirty: boolean = false;\n private _maxPartCount: number = 0;\n\n /**\n * Creates a new GaussianSplattingDebugMaterialPlugin.\n * @param material The GaussianSplattingMaterial to attach the plugin to.\n */\n constructor(material: GaussianSplattingMaterial) {\n super(material, \"GaussianSplattingDebug\", 200, new GaussianSplattingDebugDefines(), true, true);\n }\n\n private _isAnyFeatureActive(): boolean {\n return (\n this._clippingBox !== null ||\n this._opacityCulling !== null ||\n this._sizeCulling !== null ||\n this._opacityScale !== 1.0 ||\n this._opacitySaturate ||\n !this._shDc ||\n !this._shOrder1 ||\n !this._shOrder2 ||\n !this._shOrder3 ||\n !this._shOrder4 ||\n this._partClippingBoxes.some((b) => b !== null) ||\n this._partOpacityCullings.some((b) => b !== null) ||\n this._partSizeCullings.some((b) => b !== null) ||\n this._partOpacityScales.some((s) => s !== null) ||\n this._partOpacitySaturates.some((s) => s !== null) ||\n this._partShDcs.some((s) => s !== null) ||\n this._partShOrder1s.some((s) => s !== null) ||\n this._partShOrder2s.some((s) => s !== null) ||\n this._partShOrder3s.some((s) => s !== null) ||\n this._partShOrder4s.some((s) => s !== null)\n );\n }\n\n private _markDirty(): void {\n this.markAllDefinesAsDirty();\n this._textureDirty = true;\n }\n\n // ----- Public API -----\n\n /**\n * Number of parts in compound mode. Set automatically by GaussianSplattingDebugger.addMesh().\n * When 0 (non-compound), setPartOptions() logs an error.\n * @returns the part count\n */\n public get partCount(): number {\n return this._partCount;\n }\n public set partCount(count: number) {\n if (this._partCount !== count) {\n this._partCount = count;\n this._markDirty();\n }\n }\n\n /**\n * Sets per-part debug overrides for the given part index.\n * Only valid on compound meshes (partCount \\> 0); logs an error otherwise.\n * @param partIndex The zero-based part index.\n * @param options Partial set of debug options to override for this part.\n */\n public setPartOptions(partIndex: number, options: Partial<IGaussianSplattingDebugOptions>): void {\n if (this._partCount === 0) {\n Logger.Error(\"GaussianSplattingDebugMaterialPlugin.setPartOptions: called on a non-compound mesh. partCount is 0.\");\n return;\n }\n if (options.clippingBox !== undefined) {\n this._partClippingBoxes[partIndex] = options.clippingBox;\n }\n if (options.opacityCulling !== undefined) {\n this._partOpacityCullings[partIndex] = options.opacityCulling;\n }\n if (options.sizeCulling !== undefined) {\n this._partSizeCullings[partIndex] = options.sizeCulling;\n }\n if (options.opacityScale !== undefined) {\n this._partOpacityScales[partIndex] = options.opacityScale;\n }\n if (options.opacitySaturate !== undefined) {\n this._partOpacitySaturates[partIndex] = options.opacitySaturate;\n }\n if (options.shDc !== undefined) {\n this._partShDcs[partIndex] = options.shDc;\n }\n if (options.shOrder1 !== undefined) {\n this._partShOrder1s[partIndex] = options.shOrder1;\n }\n if (options.shOrder2 !== undefined) {\n this._partShOrder2s[partIndex] = options.shOrder2;\n }\n if (options.shOrder3 !== undefined) {\n this._partShOrder3s[partIndex] = options.shOrder3;\n }\n if (options.shOrder4 !== undefined) {\n this._partShOrder4s[partIndex] = options.shOrder4;\n }\n this._markDirty();\n }\n\n /**\n * Clears all per-part debug overrides for the given part index,\n * falling back to global settings.\n * @param partIndex The zero-based part index.\n */\n public clearPartOptions(partIndex: number): void {\n this._partClippingBoxes[partIndex] = null;\n this._partOpacityCullings[partIndex] = null;\n this._partSizeCullings[partIndex] = null;\n this._partOpacityScales[partIndex] = null;\n this._partOpacitySaturates[partIndex] = null;\n this._partShDcs[partIndex] = null;\n this._partShOrder1s[partIndex] = null;\n this._partShOrder2s[partIndex] = null;\n this._partShOrder3s[partIndex] = null;\n this._partShOrder4s[partIndex] = null;\n this._markDirty();\n }\n\n /**\n * World-space axis-aligned clipping box. Splats outside are not rendered.\n * Set to null to disable clipping.\n * Example: `{ min: new Vector3(-2,-2,-2), max: new Vector3(2,2,2) }`\n */\n public get clippingBox(): Nullable<{ min: Vector3; max: Vector3 }> {\n return this._clippingBox;\n }\n public set clippingBox(value: Nullable<{ min: Vector3; max: Vector3 }>) {\n this._clippingBox = value;\n this._markDirty();\n }\n\n /**\n * Opacity culling range [0..1]. Splats whose stored opacity falls outside this\n * range are not rendered. Set to null to disable.\n */\n public get opacityCulling(): Nullable<{ min: number; max: number }> {\n return this._opacityCulling;\n }\n public set opacityCulling(value: Nullable<{ min: number; max: number }>) {\n this._opacityCulling = value;\n this._markDirty();\n }\n\n /**\n * Size culling range. Size is pow(|det(Σ)|, 1/6) of the 3D covariance matrix,\n * equal to the geometric mean of the principal radii. Use GaussianSplattingMeshBase.splatSizeRange\n * to find the asset's range. Set to null to disable.\n */\n public get sizeCulling(): Nullable<{ min: number; max: number }> {\n return this._sizeCulling;\n }\n public set sizeCulling(value: Nullable<{ min: number; max: number }>) {\n this._sizeCulling = value;\n this._markDirty();\n }\n\n /**\n * Scalar multiplier applied to every splat's opacity after all other modifiers.\n * 1.0 (default) = no change.\n */\n public get opacityScale(): number {\n return this._opacityScale;\n }\n public set opacityScale(value: number) {\n this._opacityScale = value;\n this._markDirty();\n }\n\n /**\n * When true, replaces the Gaussian spatial falloff with a flat uniform opacity,\n * making each splat appear as a solid disk with its raw alpha value.\n */\n public get opacitySaturate(): boolean {\n return this._opacitySaturate;\n }\n public set opacitySaturate(value: boolean) {\n this._opacitySaturate = value;\n this._markDirty();\n }\n\n /** Include the DC (base) color from colorsTexture. Default: true. */\n public get shDc(): boolean {\n return this._shDc;\n }\n public set shDc(value: boolean) {\n this._shDc = value;\n this._markDirty();\n }\n\n /** Include SH band 1 contribution. Default: true. */\n public get shOrder1(): boolean {\n return this._shOrder1;\n }\n public set shOrder1(value: boolean) {\n this._shOrder1 = value;\n this._markDirty();\n }\n\n /** Include SH band 2 contribution. Default: true. */\n public get shOrder2(): boolean {\n return this._shOrder2;\n }\n public set shOrder2(value: boolean) {\n this._shOrder2 = value;\n this._markDirty();\n }\n\n /** Include SH band 3 contribution. Default: true. */\n public get shOrder3(): boolean {\n return this._shOrder3;\n }\n public set shOrder3(value: boolean) {\n this._shOrder3 = value;\n this._markDirty();\n }\n\n /** Include SH band 4 contribution. Default: true. */\n public get shOrder4(): boolean {\n return this._shOrder4;\n }\n public set shOrder4(value: boolean) {\n this._shOrder4 = value;\n this._markDirty();\n }\n\n // ----- Plugin overrides -----\n\n /**\n * Adds the per-part debug data texture name to the sampler list so the effect can bind it.\n * @param samplers the sampler list to populate\n */\n public override getSamplers(samplers: string[]): void {\n samplers.push(\"dbgPartData\");\n }\n\n /** @returns the class name of this plugin */\n public override getClassName(): string {\n return \"GaussianSplattingDebugMaterialPlugin\";\n }\n\n /**\n * @param shaderLanguage the shader language to check\n * @returns true for GLSL and WGSL\n */\n public override isCompatible(shaderLanguage: ShaderLanguage): boolean {\n switch (shaderLanguage) {\n case ShaderLanguage.GLSL:\n case ShaderLanguage.WGSL:\n return true;\n default:\n return false;\n }\n }\n\n /**\n * Always ready — no async resources.\n * @param _defines unused\n * @param _scene unused\n * @param _engine unused\n * @param _subMesh unused\n * @returns true\n */\n public override isReadyForSubMesh(_defines: MaterialDefines, _scene: Scene, _engine: AbstractEngine, _subMesh: SubMesh): boolean {\n return true;\n }\n\n /**\n * Sets shader defines from current property state. GS_DBG_ENABLED is set to true\n * only when at least one feature is non-default, ensuring zero overhead otherwise.\n * Sub-flags also check per-part arrays so compound-only overrides activate the correct\n * code paths even when the global setting is at its default.\n * @param defines the defines object\n */\n public override prepareDefines(defines: GaussianSplattingDebugDefines): void {\n defines.GS_DBG_ENABLED = this._isAnyFeatureActive();\n defines.GS_DBG_CLIP = this._clippingBox !== null || this._partClippingBoxes.some((b) => b !== null) ? 1 : 0;\n defines.GS_DBG_CULL_OPACITY = this._opacityCulling !== null || this._partOpacityCullings.some((b) => b !== null) ? 1 : 0;\n defines.GS_DBG_CULL_SIZE = this._sizeCulling !== null || this._partSizeCullings.some((b) => b !== null) ? 1 : 0;\n defines.GS_DBG_OPACITY_SCALE = this._opacityScale !== 1.0 || this._partOpacityScales.some((s) => s !== null) ? 1 : 0;\n defines.GS_DBG_OPACITY_SATURATE = this._opacitySaturate || this._partOpacitySaturates.some((s) => s !== null) ? 1 : 0;\n defines.GS_DBG_SH_DC = !this._shDc || this._partShDcs.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER1 = !this._shOrder1 || this._partShOrder1s.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER2 = !this._shOrder2 || this._partShOrder2s.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER3 = !this._shOrder3 || this._partShOrder3s.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER4 = !this._shOrder4 || this._partShOrder4s.some((s) => s !== null) ? 0 : 1;\n }\n\n /**\n * Returns shader code injections for the debug features.\n * @param shaderType \"vertex\" or \"fragment\"\n * @param shaderLanguage GLSL or WGSL\n * @returns map of injection-point name to injected code, or null\n */\n public override getCustomCode(shaderType: string, shaderLanguage = ShaderLanguage.GLSL): Nullable<{ [pointName: string]: string }> {\n if (shaderLanguage === ShaderLanguage.WGSL) {\n return this._getCustomCodeWGSL(shaderType);\n }\n return this._getCustomCodeGLSL(shaderType);\n }\n\n private _getCustomCodeGLSL(shaderType: string): Nullable<{ [pointName: string]: string }> {\n if (shaderType === \"vertex\") {\n return {\n CUSTOM_VERTEX_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\nuniform vec3 dbgClipMin;\nuniform vec3 dbgClipMax;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\nuniform float dbgMinOpacity;\nuniform float dbgMaxOpacity;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\nuniform float dbgMinSize;\nuniform float dbgMaxSize;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\nuniform float dbgOpacityScale;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nuniform sampler2D dbgPartData;\nvarying float vPartIndex;\n#endif\n`,\n CUSTOM_VERTEX_UPDATE: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\n if (worldPos.x < dbgClipMin.x || worldPos.x > dbgClipMax.x ||\n worldPos.y < dbgClipMin.y || worldPos.y > dbgClipMax.y ||\n worldPos.z < dbgClipMin.z || worldPos.z > dbgClipMax.z) {\n scale = vec2(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\n if (splat.color.w < dbgMinOpacity || splat.color.w > dbgMaxOpacity) {\n scale = vec2(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\n {\n float _d0 = splat.covA.x; float _d1 = splat.covA.y; float _d2 = splat.covA.z;\n float _d3 = splat.covA.w; float _d4 = splat.covB.x; float _d5 = splat.covB.y;\n float _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n float _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < dbgMinSize || _sz > dbgMaxSize) {\n scale = vec2(0.0);\n }\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\n vColor.w *= dbgOpacityScale;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n {\n int _pIdx = int(splat.partIndex);\n vPartIndex = float(splat.partIndex);\n vec4 _row0 = texelFetch(dbgPartData, ivec2(_pIdx, 0), 0);\n vec4 _row1 = texelFetch(dbgPartData, ivec2(_pIdx, 1), 0);\n vec4 _row2 = texelFetch(dbgPartData, ivec2(_pIdx, 2), 0);\n vec3 _clipMin = _row0.xyz;\n vec3 _clipMax = vec3(_row0.w, _row1.xy);\n float _minOp = _row1.z;\n float _maxOp = _row1.w;\n float _minSz = _row2.x;\n float _maxSz = _row2.y;\n float _opSc = _row2.z;\n #if GS_DBG_CLIP == 1\n if (worldPos.x < _clipMin.x || worldPos.x > _clipMax.x ||\n worldPos.y < _clipMin.y || worldPos.y > _clipMax.y ||\n worldPos.z < _clipMin.z || worldPos.z > _clipMax.z) {\n scale = vec2(0.0);\n }\n #endif\n #if GS_DBG_CULL_OPACITY == 1\n if (splat.color.w < _minOp || splat.color.w > _maxOp) {\n scale = vec2(0.0);\n }\n #endif\n #if GS_DBG_CULL_SIZE == 1\n {\n float _d0 = splat.covA.x; float _d1 = splat.covA.y; float _d2 = splat.covA.z;\n float _d3 = splat.covA.w; float _d4 = splat.covB.x; float _d5 = splat.covB.y;\n float _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n float _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < _minSz || _sz > _maxSz) {\n scale = vec2(0.0);\n }\n }\n #endif\n #if GS_DBG_OPACITY_SCALE == 1\n vColor.w *= _opSc;\n #endif\n }\n#endif\n`,\n };\n } else if (shaderType === \"fragment\") {\n return {\n CUSTOM_FRAGMENT_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nuniform sampler2D dbgPartData;\nvarying float vPartIndex;\n#endif\n`,\n CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n if (texelFetch(dbgPartData, ivec2(int(vPartIndex + 0.5), 2), 0).w > 0.5) { finalColor.a = vColor.a; }\n#elif defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SATURATE == 1\n finalColor.a = vColor.a;\n#endif\n`,\n };\n }\n return null;\n }\n\n private _getCustomCodeWGSL(shaderType: string): Nullable<{ [pointName: string]: string }> {\n if (shaderType === \"vertex\") {\n return {\n CUSTOM_VERTEX_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\nuniform dbgClipMin: vec3f;\nuniform dbgClipMax: vec3f;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\nuniform dbgMinOpacity: f32;\nuniform dbgMaxOpacity: f32;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\nuniform dbgMinSize: f32;\nuniform dbgMaxSize: f32;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\nuniform dbgOpacityScale: f32;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nvar dbgPartData: texture_2d<f32>;\nvarying vPartIndex: f32;\n#endif\n`,\n CUSTOM_VERTEX_UPDATE: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\n if (worldPos.x < uniforms.dbgClipMin.x || worldPos.x > uniforms.dbgClipMax.x ||\n worldPos.y < uniforms.dbgClipMin.y || worldPos.y > uniforms.dbgClipMax.y ||\n worldPos.z < uniforms.dbgClipMin.z || worldPos.z > uniforms.dbgClipMax.z) {\n scale = vec2f(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\n if (splat.color.w < uniforms.dbgMinOpacity || splat.color.w > uniforms.dbgMaxOpacity) {\n scale = vec2f(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\n {\n let _d0 = splat.covA.x; let _d1 = splat.covA.y; let _d2 = splat.covA.z;\n let _d3 = splat.covA.w; let _d4 = splat.covB.x; let _d5 = splat.covB.y;\n let _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n let _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < uniforms.dbgMinSize || _sz > uniforms.dbgMaxSize) {\n scale = vec2f(0.0);\n }\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\n vertexOutputs.vColor.w *= uniforms.dbgOpacityScale;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n {\n let _pIdx = i32(splat.partIndex);\n vertexOutputs.vPartIndex = f32(splat.partIndex);\n let _row0 = textureLoad(dbgPartData, vec2i(_pIdx, 0), 0);\n let _row1 = textureLoad(dbgPartData, vec2i(_pIdx, 1), 0);\n let _row2 = textureLoad(dbgPartData, vec2i(_pIdx, 2), 0);\n let _clipMin = _row0.xyz;\n let _clipMax = vec3f(_row0.w, _row1.xy);\n let _minOp = _row1.z;\n let _maxOp = _row1.w;\n let _minSz = _row2.x;\n let _maxSz = _row2.y;\n let _opSc = _row2.z;\n #if GS_DBG_CLIP == 1\n if (worldPos.x < _clipMin.x || worldPos.x > _clipMax.x ||\n worldPos.y < _clipMin.y || worldPos.y > _clipMax.y ||\n worldPos.z < _clipMin.z || worldPos.z > _clipMax.z) {\n scale = vec2f(0.0);\n }\n #endif\n #if GS_DBG_CULL_OPACITY == 1\n if (splat.color.w < _minOp || splat.color.w > _maxOp) {\n scale = vec2f(0.0);\n }\n #endif\n #if GS_DBG_CULL_SIZE == 1\n {\n let _d0 = splat.covA.x; let _d1 = splat.covA.y; let _d2 = splat.covA.z;\n let _d3 = splat.covA.w; let _d4 = splat.covB.x; let _d5 = splat.covB.y;\n let _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n let _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < _minSz || _sz > _maxSz) {\n scale = vec2f(0.0);\n }\n }\n #endif\n #if GS_DBG_OPACITY_SCALE == 1\n vertexOutputs.vColor.w *= _opSc;\n #endif\n }\n#endif\n`,\n };\n } else if (shaderType === \"fragment\") {\n return {\n CUSTOM_FRAGMENT_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nvar dbgPartData: texture_2d<f32>;\nvarying vPartIndex: f32;\n#endif\n`,\n CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n if (textureLoad(dbgPartData, vec2i(i32(fragmentInputs.vPartIndex + 0.5), 2), 0).w > 0.5) {\n let _gsdbgA: f32 = -dot(fragmentInputs.vPosition, fragmentInputs.vPosition);\n if (_gsdbgA > -4.0) { finalColor.a = fragmentInputs.vColor.a; }\n }\n#elif defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SATURATE == 1\n {\n let _gsdbgA: f32 = -dot(fragmentInputs.vPosition, fragmentInputs.vPosition);\n if (_gsdbgA > -4.0) {\n finalColor.a = fragmentInputs.vColor.a;\n }\n }\n#endif\n`,\n };\n }\n return null;\n }\n\n /**\n * Declares the non-compound scalar debug uniform names as external so the Effect can\n * resolve their locations. WGSL uniforms are declared inline in getCustomCode() injections.\n * @returns uniform descriptor with externalUniforms list\n */\n public override getUniforms(): {\n ubo?: Array<{ name: string; size?: number; type?: string; arraySize?: number }>;\n vertex?: string;\n fragment?: string;\n externalUniforms?: string[];\n } {\n return {\n externalUniforms: [\"dbgClipMin\", \"dbgClipMax\", \"dbgMinOpacity\", \"dbgMaxOpacity\", \"dbgMinSize\", \"dbgMaxSize\", \"dbgOpacityScale\"],\n };\n }\n\n private _buildTextureData(partCount: number): Float32Array {\n const maxPartCount = this._maxPartCount;\n const data = new Float32Array(maxPartCount * 5 * 4);\n for (let i = 0; i < maxPartCount; i++) {\n const r0 = i * 4;\n const r1 = (maxPartCount + i) * 4;\n const r2 = (maxPartCount * 2 + i) * 4;\n const r3 = (maxPartCount * 3 + i) * 4;\n const r4 = (maxPartCount * 4 + i) * 4;\n\n // Row 0: clipMin.xyz, clipMax.x — Row 1: clipMax.yz, minOpacity, maxOpacity\n const box = i < partCount ? (this._partClippingBoxes[i] ?? this._clippingBox) : this._clippingBox;\n if (box) {\n data[r0] = box.min.x;\n data[r0 + 1] = box.min.y;\n data[r0 + 2] = box.min.z;\n data[r0 + 3] = box.max.x;\n data[r1] = box.max.y;\n data[r1 + 1] = box.max.z;\n } else {\n data[r0] = -1e9;\n data[r0 + 1] = -1e9;\n data[r0 + 2] = -1e9;\n data[r0 + 3] = 1e9;\n data[r1] = 1e9;\n data[r1 + 1] = 1e9;\n }\n const opCull = i < partCount ? (this._partOpacityCullings[i] ?? this._opacityCulling) : this._opacityCulling;\n data[r1 + 2] = opCull ? opCull.min : 0.0;\n data[r1 + 3] = opCull ? opCull.max : 1.0;\n\n // Row 2: minSize, maxSize, opacityScale, opacitySaturate\n const szCull = i < partCount ? (this._partSizeCullings[i] ?? this._sizeCulling) : this._sizeCulling;\n data[r2] = szCull ? szCull.min : 0.0;\n data[r2 + 1] = szCull ? szCull.max : 1e9;\n data[r2 + 2] = i < partCount ? (this._partOpacityScales[i] ?? this._opacityScale) : this._opacityScale;\n const sat = i < partCount ? (this._partOpacitySaturates[i] ?? this._opacitySaturate) : this._opacitySaturate;\n data[r2 + 3] = sat ? 1.0 : 0.0;\n\n // Row 3: shDc, shOrder1, shOrder2, shOrder3\n data[r3] = (i < partCount ? (this._partShDcs[i] ?? this._shDc) : this._shDc) ? 1.0 : 0.0;\n data[r3 + 1] = (i < partCount ? (this._partShOrder1s[i] ?? this._shOrder1) : this._shOrder1) ? 1.0 : 0.0;\n data[r3 + 2] = (i < partCount ? (this._partShOrder2s[i] ?? this._shOrder2) : this._shOrder2) ? 1.0 : 0.0;\n data[r3 + 3] = (i < partCount ? (this._partShOrder3s[i] ?? this._shOrder3) : this._shOrder3) ? 1.0 : 0.0;\n\n // Row 4: shOrder4, padding\n data[r4] = (i < partCount ? (this._partShOrder4s[i] ?? this._shOrder4) : this._shOrder4) ? 1.0 : 0.0;\n }\n return data;\n }\n\n private _updateOrCreateTexture(scene: Scene): void {\n if (!this._maxPartCount) {\n this._maxPartCount = GetGaussianSplattingMaxPartCount(scene.getEngine());\n }\n const data = this._buildTextureData(this._partCount);\n if (!this._dbgPartDataTexture) {\n this._dbgPartDataTexture = RawTexture.CreateRGBATexture(\n data,\n this._maxPartCount,\n 5,\n scene,\n false,\n false,\n Constants.TEXTURE_NEAREST_SAMPLINGMODE,\n Constants.TEXTURETYPE_FLOAT\n );\n } else {\n this._dbgPartDataTexture.update(data);\n }\n this._textureDirty = false;\n }\n\n /**\n * Binds uniform values each frame. Scalar uniforms are uploaded for non-compound mode;\n * the per-part data texture is updated and bound for compound mode.\n * @param _uniformBuffer unused\n * @param _scene the current scene\n * @param _engine unused\n * @param subMesh the submesh being rendered\n */\n public override bindForSubMesh(_uniformBuffer: UniformBuffer, _scene: Scene, _engine: AbstractEngine, subMesh: SubMesh): void {\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n\n if (this._partCount > 0 && this._isAnyFeatureActive()) {\n // Compound mode: pack per-part data into a texture (only when the debug path is active)\n if (this._textureDirty || !this._dbgPartDataTexture) {\n this._updateOrCreateTexture(_scene);\n }\n if (this._dbgPartDataTexture) {\n effect.setTexture(\"dbgPartData\", this._dbgPartDataTexture);\n }\n } else if (this._partCount === 0) {\n // Non-compound: upload scalar uniforms\n if (this._clippingBox) {\n effect.setVector3(\"dbgClipMin\", this._clippingBox.min);\n effect.setVector3(\"dbgClipMax\", this._clippingBox.max);\n }\n if (this._opacityCulling) {\n effect.setFloat(\"dbgMinOpacity\", this._opacityCulling.min);\n effect.setFloat(\"dbgMaxOpacity\", this._opacityCulling.max);\n }\n if (this._sizeCulling) {\n effect.setFloat(\"dbgMinSize\", this._sizeCulling.min);\n effect.setFloat(\"dbgMaxSize\", this._sizeCulling.max);\n }\n if (this._opacityScale !== 1.0) {\n effect.setFloat(\"dbgOpacityScale\", this._opacityScale);\n }\n }\n }\n\n /**\n * Disposes the per-part data texture.\n * @param _forceDisposeTextures unused; the LUT texture is always disposed as it is owned by this plugin\n */\n public override dispose(_forceDisposeTextures?: boolean): void {\n this._dbgPartDataTexture?.dispose();\n this._dbgPartDataTexture = null;\n }\n}\n\nlet _Registered = false;\n/**\n * Register side effects for GaussianSplattingDebugMaterialPlugin.\n * Safe to call multiple times; only the first call has an effect.\n */\nexport function RegisterGaussianSplattingDebugMaterialPlugin(): void {\n if (_Registered) {\n return;\n }\n _Registered = true;\n\n RegisterClass(\"BABYLON.GaussianSplattingDebugMaterialPlugin\", GaussianSplattingDebugMaterialPlugin);\n}\n"]}
1
+ {"version":3,"file":"gaussianSplattingDebugMaterialPlugin.pure.js","sourceRoot":"","sources":["../../../../../dev/core/src/Materials/GaussianSplatting/gaussianSplattingDebugMaterialPlugin.pure.ts"],"names":[],"mappings":"AAAA,6DAA6D;AAO7D,OAAO,EAAE,eAAe,EAAE,MAAM,oBAAoB,CAAC;AACrD,OAAO,EAAE,kBAAkB,EAAE,MAAM,4BAA4B,CAAC;AAEhE,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAErD,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAC3C,OAAO,EAAkC,gCAAgC,EAAE,MAAM,kCAAkC,CAAC;AACpH,OAAO,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AACpD,OAAO,EAAE,SAAS,EAAE,MAAM,yBAAyB,CAAC;AAEpD,gBAAgB;AAChB,MAAM,6BAA8B,SAAQ,eAAe;IAA3D;;QACI,kDAAkD;QAClD,mBAAc,GAAY,KAAK,CAAC;QAChC,0EAA0E;QAC1E,gBAAW,GAAW,CAAC,CAAC;QACxB,iEAAiE;QACjE,wBAAmB,GAAW,CAAC,CAAC;QAChC,8DAA8D;QAC9D,qBAAgB,GAAW,CAAC,CAAC;QAC7B,2EAA2E;QAC3E,yBAAoB,GAAW,CAAC,CAAC;QACjC,gFAAgF;QAChF,4BAAuB,GAAW,CAAC,CAAC;QACpC,yEAAyE;QACzE,iBAAY,GAAW,CAAC,CAAC;QACzB,yEAAyE;QACzE,qBAAgB,GAAW,CAAC,CAAC;QAC7B,yEAAyE;QACzE,qBAAgB,GAAW,CAAC,CAAC;QAC7B,yEAAyE;QACzE,qBAAgB,GAAW,CAAC,CAAC;QAC7B,yEAAyE;QACzE,qBAAgB,GAAW,CAAC,CAAC;IACjC,CAAC;CAAA;AA8BD;;;;;;;;;GASG;AACH,MAAM,OAAO,oCAAqC,SAAQ,kBAAkB;IAiCxE;;;OAGG;IACH,YAAY,QAAmC;QAC3C,KAAK,CAAC,QAAQ,EAAE,wBAAwB,EAAE,GAAG,EAAE,IAAI,6BAA6B,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;QArC5F,iBAAY,GAA6C,IAAI,CAAC;QAC9D,oBAAe,GAA2C,IAAI,CAAC;QAC/D,iBAAY,GAA2C,IAAI,CAAC;QAC5D,kBAAa,GAAW,GAAG,CAAC;QAC5B,qBAAgB,GAAY,KAAK,CAAC;QAClC,UAAK,GAAY,IAAI,CAAC;QACtB,cAAS,GAAY,IAAI,CAAC;QAC1B,cAAS,GAAY,IAAI,CAAC;QAC1B,cAAS,GAAY,IAAI,CAAC;QAC1B,cAAS,GAAY,IAAI,CAAC;QAElC,qDAAqD;QAC7C,eAAU,GAAW,CAAC,CAAC;QACvB,uBAAkB,GAAoD,EAAE,CAAC;QACzE,yBAAoB,GAAkD,EAAE,CAAC;QACzE,sBAAiB,GAAkD,EAAE,CAAC;QACtE,uBAAkB,GAA4B,EAAE,CAAC;QACjD,0BAAqB,GAA6B,EAAE,CAAC;QACrD,eAAU,GAA6B,EAAE,CAAC;QAC1C,mBAAc,GAA6B,EAAE,CAAC;QAC9C,mBAAc,GAA6B,EAAE,CAAC;QAC9C,mBAAc,GAA6B,EAAE,CAAC;QAC9C,mBAAc,GAA6B,EAAE,CAAC;QAKtD,0EAA0E;QAClE,wBAAmB,GAAyB,IAAI,CAAC;QACjD,kBAAa,GAAY,KAAK,CAAC;QAC/B,kBAAa,GAAW,CAAC,CAAC;QAQ9B,IAAI,CAAC,WAAW,GAAG;YACf,IAAI,CAAC,kBAAkB;YACvB,IAAI,CAAC,oBAAoB;YACzB,IAAI,CAAC,iBAAiB;YACtB,IAAI,CAAC,kBAAkB;YACvB,IAAI,CAAC,qBAAqB;YAC1B,IAAI,CAAC,UAAU;YACf,IAAI,CAAC,cAAc;YACnB,IAAI,CAAC,cAAc;YACnB,IAAI,CAAC,cAAc;YACnB,IAAI,CAAC,cAAc;SACtB,CAAC;IACN,CAAC;IAEO,mBAAmB;QACvB,OAAO,CACH,IAAI,CAAC,YAAY,KAAK,IAAI;YAC1B,IAAI,CAAC,eAAe,KAAK,IAAI;YAC7B,IAAI,CAAC,YAAY,KAAK,IAAI;YAC1B,IAAI,CAAC,aAAa,KAAK,GAAG;YAC1B,IAAI,CAAC,gBAAgB;YACrB,CAAC,IAAI,CAAC,KAAK;YACX,CAAC,IAAI,CAAC,SAAS;YACf,CAAC,IAAI,CAAC,SAAS;YACf,CAAC,IAAI,CAAC,SAAS;YACf,CAAC,IAAI,CAAC,SAAS;YACf,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YAC/C,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YACjD,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YAC9C,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YAC/C,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YAClD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YACvC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YAC3C,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YAC3C,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC;YAC3C,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAC9C,CAAC;IACN,CAAC;IAEO,UAAU;QACd,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC7B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;IAC9B,CAAC;IAED,yBAAyB;IAEzB;;;;OAIG;IACH,IAAW,SAAS;QAChB,OAAO,IAAI,CAAC,UAAU,CAAC;IAC3B,CAAC;IACD,IAAW,SAAS,CAAC,KAAa;QAC9B,IAAI,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE,CAAC;YAC5B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;YACxB,IAAI,CAAC,UAAU,EAAE,CAAC;QACtB,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACI,cAAc,CAAC,SAAiB,EAAE,OAAgD;QACrF,IAAI,IAAI,CAAC,UAAU,KAAK,CAAC,EAAE,CAAC;YACxB,MAAM,CAAC,KAAK,CAAC,qGAAqG,CAAC,CAAC;YACpH,OAAO;QACX,CAAC;QACD,IAAI,OAAO,CAAC,WAAW,KAAK,SAAS,EAAE,CAAC;YACpC,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,WAAW,CAAC;QAC7D,CAAC;QACD,IAAI,OAAO,CAAC,cAAc,KAAK,SAAS,EAAE,CAAC;YACvC,IAAI,CAAC,oBAAoB,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,cAAc,CAAC;QAClE,CAAC;QACD,IAAI,OAAO,CAAC,WAAW,KAAK,SAAS,EAAE,CAAC;YACpC,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,WAAW,CAAC;QAC5D,CAAC;QACD,IAAI,OAAO,CAAC,YAAY,KAAK,SAAS,EAAE,CAAC;YACrC,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,YAAY,CAAC;QAC9D,CAAC;QACD,IAAI,OAAO,CAAC,eAAe,KAAK,SAAS,EAAE,CAAC;YACxC,IAAI,CAAC,qBAAqB,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,eAAe,CAAC;QACpE,CAAC;QACD,IAAI,OAAO,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YAC7B,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC;QAC9C,CAAC;QACD,IAAI,OAAO,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YACjC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,QAAQ,CAAC;QACtD,CAAC;QACD,IAAI,OAAO,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YACjC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,QAAQ,CAAC;QACtD,CAAC;QACD,IAAI,OAAO,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YACjC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,QAAQ,CAAC;QACtD,CAAC;QACD,IAAI,OAAO,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YACjC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,OAAO,CAAC,QAAQ,CAAC;QACtD,CAAC;QACD,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;;OAIG;IACI,gBAAgB,CAAC,SAAiB;QACrC,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QAC1C,IAAI,CAAC,oBAAoB,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QAC5C,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QACzC,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QAC1C,IAAI,CAAC,qBAAqB,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QAC7C,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QAClC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QACtC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QACtC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QACtC,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;QACtC,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;;;OAKG;IACI,gBAAgB,CAAC,YAAoB;QACxC,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACjC,IAAI,YAAY,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC;gBAC5B,GAAG,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC;YAChC,CAAC;QACL,CAAC;QACD,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;;OAIG;IACH,IAAW,WAAW;QAClB,OAAO,IAAI,CAAC,YAAY,CAAC;IAC7B,CAAC;IACD,IAAW,WAAW,CAAC,KAA+C;QAClE,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC1B,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;OAGG;IACH,IAAW,cAAc;QACrB,OAAO,IAAI,CAAC,eAAe,CAAC;IAChC,CAAC;IACD,IAAW,cAAc,CAAC,KAA6C;QACnE,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;QAC7B,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;;OAIG;IACH,IAAW,WAAW;QAClB,OAAO,IAAI,CAAC,YAAY,CAAC;IAC7B,CAAC;IACD,IAAW,WAAW,CAAC,KAA6C;QAChE,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC1B,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;OAGG;IACH,IAAW,YAAY;QACnB,OAAO,IAAI,CAAC,aAAa,CAAC;IAC9B,CAAC;IACD,IAAW,YAAY,CAAC,KAAa;QACjC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED;;;OAGG;IACH,IAAW,eAAe;QACtB,OAAO,IAAI,CAAC,gBAAgB,CAAC;IACjC,CAAC;IACD,IAAW,eAAe,CAAC,KAAc;QACrC,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;QAC9B,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED,qEAAqE;IACrE,IAAW,IAAI;QACX,OAAO,IAAI,CAAC,KAAK,CAAC;IACtB,CAAC;IACD,IAAW,IAAI,CAAC,KAAc;QAC1B,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED,qDAAqD;IACrD,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IACD,IAAW,QAAQ,CAAC,KAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED,qDAAqD;IACrD,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IACD,IAAW,QAAQ,CAAC,KAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED,qDAAqD;IACrD,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IACD,IAAW,QAAQ,CAAC,KAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED,qDAAqD;IACrD,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IACD,IAAW,QAAQ,CAAC,KAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,UAAU,EAAE,CAAC;IACtB,CAAC;IAED,+BAA+B;IAE/B;;;OAGG;IACa,WAAW,CAAC,QAAkB;QAC1C,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IACjC,CAAC;IAED,6CAA6C;IAC7B,YAAY;QACxB,OAAO,sCAAsC,CAAC;IAClD,CAAC;IAED;;;OAGG;IACa,YAAY,CAAC,cAA8B;QACvD,QAAQ,cAAc,EAAE,CAAC;YACrB,iCAAyB;YACzB;gBACI,OAAO,IAAI,CAAC;YAChB;gBACI,OAAO,KAAK,CAAC;QACrB,CAAC;IACL,CAAC;IAED;;;;;;;OAOG;IACa,iBAAiB,CAAC,QAAyB,EAAE,MAAa,EAAE,OAAuB,EAAE,QAAiB;QAClH,OAAO,IAAI,CAAC;IAChB,CAAC;IAED;;;;;;OAMG;IACa,cAAc,CAAC,OAAsC;QACjE,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QACpD,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC,YAAY,KAAK,IAAI,IAAI,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC5G,OAAO,CAAC,mBAAmB,GAAG,IAAI,CAAC,eAAe,KAAK,IAAI,IAAI,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACzH,OAAO,CAAC,gBAAgB,GAAG,IAAI,CAAC,YAAY,KAAK,IAAI,IAAI,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAChH,OAAO,CAAC,oBAAoB,GAAG,IAAI,CAAC,aAAa,KAAK,GAAG,IAAI,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACrH,OAAO,CAAC,uBAAuB,GAAG,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACtH,OAAO,CAAC,YAAY,GAAG,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACtF,OAAO,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAClG,OAAO,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAClG,OAAO,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAClG,OAAO,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACtG,CAAC;IAED;;;;;OAKG;IACa,aAAa,CAAC,UAAkB,EAAE,cAAc,8BAAsB;QAClF,IAAI,cAAc,gCAAwB,EAAE,CAAC;YACzC,OAAO,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QAC/C,CAAC;QACD,OAAO,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;IAC/C,CAAC;IAEO,kBAAkB,CAAC,UAAkB;QACzC,IAAI,UAAU,KAAK,QAAQ,EAAE,CAAC;YAC1B,OAAO;gBACH,yBAAyB,EAAE;;;;;;;;;;;;;;;;;;;;CAoB1C;gBACe,oBAAoB,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAqErC;aACY,CAAC;QACN,CAAC;aAAM,IAAI,UAAU,KAAK,UAAU,EAAE,CAAC;YACnC,OAAO;gBACH,2BAA2B,EAAE;;;;;CAK5C;gBACe,gCAAgC,EAAE;;;;;;CAMjD;aACY,CAAC;QACN,CAAC;QACD,OAAO,IAAI,CAAC;IAChB,CAAC;IAEO,kBAAkB,CAAC,UAAkB;QACzC,IAAI,UAAU,KAAK,QAAQ,EAAE,CAAC;YAC1B,OAAO;gBACH,yBAAyB,EAAE;;;;;;;;;;;;;;;;;;;;CAoB1C;gBACe,oBAAoB,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAqErC;aACY,CAAC;QACN,CAAC;aAAM,IAAI,UAAU,KAAK,UAAU,EAAE,CAAC;YACnC,OAAO;gBACH,2BAA2B,EAAE;;;;;CAK5C;gBACe,gCAAgC,EAAE;;;;;;;;;;;;;;CAcjD;aACY,CAAC;QACN,CAAC;QACD,OAAO,IAAI,CAAC;IAChB,CAAC;IAED;;;;OAIG;IACa,WAAW;QAMvB,OAAO;YACH,gBAAgB,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,eAAe,EAAE,eAAe,EAAE,YAAY,EAAE,YAAY,EAAE,iBAAiB,CAAC;SAClI,CAAC;IACN,CAAC;IAEO,iBAAiB,CAAC,SAAiB;QACvC,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC;QACxC,MAAM,IAAI,GAAG,IAAI,YAAY,CAAC,YAAY,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;QACpD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,YAAY,EAAE,CAAC,EAAE,EAAE,CAAC;YACpC,MAAM,EAAE,GAAG,CAAC,GAAG,CAAC,CAAC;YACjB,MAAM,EAAE,GAAG,CAAC,YAAY,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC;YAClC,MAAM,EAAE,GAAG,CAAC,YAAY,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC;YACtC,MAAM,EAAE,GAAG,CAAC,YAAY,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC;YACtC,MAAM,EAAE,GAAG,CAAC,YAAY,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC;YAEtC,6EAA6E;YAC7E,MAAM,GAAG,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC;YAClG,IAAI,GAAG,EAAE,CAAC;gBACN,IAAI,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;gBACrB,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;gBACzB,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;gBACzB,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;gBACzB,IAAI,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;gBACrB,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;YAC7B,CAAC;iBAAM,CAAC;gBACJ,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC;gBAChB,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC;gBACpB,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC;gBACpB,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC;gBACnB,IAAI,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC;gBACf,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC;YACvB,CAAC;YACD,MAAM,MAAM,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC;YAC7G,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YACzC,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YAEzC,yDAAyD;YACzD,MAAM,MAAM,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC;YACpG,IAAI,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YACrC,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YACzC,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC;YACvG,MAAM,GAAG,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC;YAC7G,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YAE/B,4CAA4C;YAC5C,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YACzF,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YACzG,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YACzG,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;YAEzG,2BAA2B;YAC3B,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;QACzG,CAAC;QACD,OAAO,IAAI,CAAC;IAChB,CAAC;IAEO,sBAAsB,CAAC,KAAY;QACvC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;YACtB,IAAI,CAAC,aAAa,GAAG,gCAAgC,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC;QAC7E,CAAC;QACD,MAAM,IAAI,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACrD,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC5B,IAAI,CAAC,mBAAmB,GAAG,UAAU,CAAC,iBAAiB,CACnD,IAAI,EACJ,IAAI,CAAC,aAAa,EAClB,CAAC,EACD,KAAK,EACL,KAAK,EACL,KAAK,EACL,SAAS,CAAC,4BAA4B,EACtC,SAAS,CAAC,iBAAiB,CAC9B,CAAC;QACN,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAC1C,CAAC;QACD,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;IAC/B,CAAC;IAED;;;;;;;OAOG;IACa,cAAc,CAAC,cAA6B,EAAE,MAAa,EAAE,OAAuB,EAAE,OAAgB;QAClH,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;QAC9B,IAAI,CAAC,MAAM,EAAE,CAAC;YACV,OAAO;QACX,CAAC;QAED,IAAI,IAAI,CAAC,UAAU,GAAG,CAAC,IAAI,IAAI,CAAC,mBAAmB,EAAE,EAAE,CAAC;YACpD,wFAAwF;YACxF,IAAI,IAAI,CAAC,aAAa,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBAClD,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,CAAC;YACxC,CAAC;YACD,IAAI,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBAC3B,MAAM,CAAC,UAAU,CAAC,aAAa,EAAE,IAAI,CAAC,mBAAmB,CAAC,CAAC;YAC/D,CAAC;QACL,CAAC;aAAM,IAAI,IAAI,CAAC,UAAU,KAAK,CAAC,EAAE,CAAC;YAC/B,uCAAuC;YACvC,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM,CAAC,UAAU,CAAC,YAAY,EAAE,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;gBACvD,MAAM,CAAC,UAAU,CAAC,YAAY,EAAE,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;YAC3D,CAAC;YACD,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,MAAM,CAAC,QAAQ,CAAC,eAAe,EAAE,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;gBAC3D,MAAM,CAAC,QAAQ,CAAC,eAAe,EAAE,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;YAC/D,CAAC;YACD,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;gBACpB,MAAM,CAAC,QAAQ,CAAC,YAAY,EAAE,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;gBACrD,MAAM,CAAC,QAAQ,CAAC,YAAY,EAAE,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;YACzD,CAAC;YACD,IAAI,IAAI,CAAC,aAAa,KAAK,GAAG,EAAE,CAAC;gBAC7B,MAAM,CAAC,QAAQ,CAAC,iBAAiB,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;YAC3D,CAAC;QACL,CAAC;IACL,CAAC;IAED;;;OAGG;IACa,OAAO,CAAC,qBAA+B;QACnD,IAAI,CAAC,mBAAmB,EAAE,OAAO,EAAE,CAAC;QACpC,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;IACpC,CAAC;CACJ;AAED,IAAI,WAAW,GAAG,KAAK,CAAC;AACxB;;;GAGG;AACH,MAAM,UAAU,4CAA4C;IACxD,IAAI,WAAW,EAAE,CAAC;QACd,OAAO;IACX,CAAC;IACD,WAAW,GAAG,IAAI,CAAC;IAEnB,aAAa,CAAC,8CAA8C,EAAE,oCAAoC,CAAC,CAAC;AACxG,CAAC","sourcesContent":["/** This file must only contain pure code and pure imports */\n\nimport { type Nullable } from \"../../types\";\nimport { type Scene } from \"../../scene\";\nimport { type AbstractEngine } from \"../../Engines/abstractEngine\";\nimport { type SubMesh } from \"../../Meshes/subMesh\";\nimport { type UniformBuffer } from \"../uniformBuffer\";\nimport { MaterialDefines } from \"../materialDefines\";\nimport { MaterialPluginBase } from \"../materialPluginBase.pure\";\nimport { ShaderLanguage } from \"../shaderLanguage\";\nimport { RegisterClass } from \"../../Misc/typeStore\";\nimport { type Vector3 } from \"../../Maths/math.vector\";\nimport { Logger } from \"../../Misc/logger\";\nimport { type GaussianSplattingMaterial, GetGaussianSplattingMaxPartCount } from \"./gaussianSplattingMaterial.pure\";\nimport { RawTexture } from \"../Textures/rawTexture\";\nimport { Constants } from \"../../Engines/constants\";\n\n/** @internal */\nclass GaussianSplattingDebugDefines extends MaterialDefines {\n /** Defines whether any debug feature is active */\n GS_DBG_ENABLED: boolean = false;\n /** Defines whether world-space clipping box is enabled (0: off, 1: on) */\n GS_DBG_CLIP: number = 0;\n /** Defines whether opacity culling is enabled (0: off, 1: on) */\n GS_DBG_CULL_OPACITY: number = 0;\n /** Defines whether size culling is enabled (0: off, 1: on) */\n GS_DBG_CULL_SIZE: number = 0;\n /** Defines whether per-splat opacity scaling is enabled (0: off, 1: on) */\n GS_DBG_OPACITY_SCALE: number = 0;\n /** Defines whether opacity saturation (flat disk) is enabled (0: off, 1: on) */\n GS_DBG_OPACITY_SATURATE: number = 0;\n /** Defines whether the DC (base) SH color is included (0: off, 1: on) */\n GS_DBG_SH_DC: number = 1;\n /** Defines whether SH band 1 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER1: number = 1;\n /** Defines whether SH band 2 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER2: number = 1;\n /** Defines whether SH band 3 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER3: number = 1;\n /** Defines whether SH band 4 contribution is included (0: off, 1: on) */\n GS_DBG_SH_ORDER4: number = 1;\n}\n\n/**\n * Per-part debug options for compound Gaussian splat meshes.\n * Each field is optional; unset fields fall back to the global setting on the plugin,\n * and if that is also unset, a neutral value is used (no culling, full SH, etc.).\n */\nexport interface IGaussianSplattingDebugOptions {\n /** World-space axis-aligned clipping box, or null to disable. */\n clippingBox: Nullable<{ min: Vector3; max: Vector3 }>;\n /** Opacity culling range [0..1], or null to disable. */\n opacityCulling: Nullable<{ min: number; max: number }>;\n /** Size culling range, or null to disable. */\n sizeCulling: Nullable<{ min: number; max: number }>;\n /** Scalar opacity multiplier. 1.0 = no change. */\n opacityScale: number;\n /** When true, replaces Gaussian falloff with flat disk opacity. */\n opacitySaturate: boolean;\n /** Include the DC (base) SH color. */\n shDc: boolean;\n /** Include SH band 1 contribution. */\n shOrder1: boolean;\n /** Include SH band 2 contribution. */\n shOrder2: boolean;\n /** Include SH band 3 contribution. */\n shOrder3: boolean;\n /** Include SH band 4 contribution. */\n shOrder4: boolean;\n}\n\n/**\n * Debug plugin for GaussianSplattingMaterial.\n * Provides runtime controls for clipping, opacity/size culling, opacity scaling,\n * opacity saturation, and per-SH-order toggling. All features are gated behind\n * the GS_DBG_ENABLED shader define — when every option is at its default value\n * the define is absent and the shader compiles to identical code as without the plugin.\n *\n * In compound mode (partCount \\> 0), per-part overrides can be set via setPartOptions().\n * Global settings act as defaults; per-part settings override them for that part index.\n */\nexport class GaussianSplattingDebugMaterialPlugin extends MaterialPluginBase {\n private _clippingBox: Nullable<{ min: Vector3; max: Vector3 }> = null;\n private _opacityCulling: Nullable<{ min: number; max: number }> = null;\n private _sizeCulling: Nullable<{ min: number; max: number }> = null;\n private _opacityScale: number = 1.0;\n private _opacitySaturate: boolean = false;\n private _shDc: boolean = true;\n private _shOrder1: boolean = true;\n private _shOrder2: boolean = true;\n private _shOrder3: boolean = true;\n private _shOrder4: boolean = true;\n\n // Per-part state (only populated when partCount > 0)\n private _partCount: number = 0;\n private _partClippingBoxes: Array<Nullable<{ min: Vector3; max: Vector3 }>> = [];\n private _partOpacityCullings: Array<Nullable<{ min: number; max: number }>> = [];\n private _partSizeCullings: Array<Nullable<{ min: number; max: number }>> = [];\n private _partOpacityScales: Array<Nullable<number>> = [];\n private _partOpacitySaturates: Array<Nullable<boolean>> = [];\n private _partShDcs: Array<Nullable<boolean>> = [];\n private _partShOrder1s: Array<Nullable<boolean>> = [];\n private _partShOrder2s: Array<Nullable<boolean>> = [];\n private _partShOrder3s: Array<Nullable<boolean>> = [];\n private _partShOrder4s: Array<Nullable<boolean>> = [];\n // Single registry of every _part* array — shiftPartOptions iterates this instead of\n // listing each array by name, so adding a new per-part field only requires adding it here.\n private readonly _partArrays: Array<Array<unknown>>;\n\n // Per-part debug LUT: MAX_PART_COUNT × 5 RGBA float texture (dbgPartData)\n private _dbgPartDataTexture: Nullable<RawTexture> = null;\n private _textureDirty: boolean = false;\n private _maxPartCount: number = 0;\n\n /**\n * Creates a new GaussianSplattingDebugMaterialPlugin.\n * @param material The GaussianSplattingMaterial to attach the plugin to.\n */\n constructor(material: GaussianSplattingMaterial) {\n super(material, \"GaussianSplattingDebug\", 200, new GaussianSplattingDebugDefines(), true, true);\n this._partArrays = [\n this._partClippingBoxes,\n this._partOpacityCullings,\n this._partSizeCullings,\n this._partOpacityScales,\n this._partOpacitySaturates,\n this._partShDcs,\n this._partShOrder1s,\n this._partShOrder2s,\n this._partShOrder3s,\n this._partShOrder4s,\n ];\n }\n\n private _isAnyFeatureActive(): boolean {\n return (\n this._clippingBox !== null ||\n this._opacityCulling !== null ||\n this._sizeCulling !== null ||\n this._opacityScale !== 1.0 ||\n this._opacitySaturate ||\n !this._shDc ||\n !this._shOrder1 ||\n !this._shOrder2 ||\n !this._shOrder3 ||\n !this._shOrder4 ||\n this._partClippingBoxes.some((b) => b !== null) ||\n this._partOpacityCullings.some((b) => b !== null) ||\n this._partSizeCullings.some((b) => b !== null) ||\n this._partOpacityScales.some((s) => s !== null) ||\n this._partOpacitySaturates.some((s) => s !== null) ||\n this._partShDcs.some((s) => s !== null) ||\n this._partShOrder1s.some((s) => s !== null) ||\n this._partShOrder2s.some((s) => s !== null) ||\n this._partShOrder3s.some((s) => s !== null) ||\n this._partShOrder4s.some((s) => s !== null)\n );\n }\n\n private _markDirty(): void {\n this.markAllDefinesAsDirty();\n this._textureDirty = true;\n }\n\n // ----- Public API -----\n\n /**\n * Number of parts in compound mode. Set automatically by GaussianSplattingDebugger.addMesh().\n * When 0 (non-compound), setPartOptions() logs an error.\n * @returns the part count\n */\n public get partCount(): number {\n return this._partCount;\n }\n public set partCount(count: number) {\n if (this._partCount !== count) {\n this._partCount = count;\n this._markDirty();\n }\n }\n\n /**\n * Sets per-part debug overrides for the given part index.\n * Only valid on compound meshes (partCount \\> 0); logs an error otherwise.\n * @param partIndex The zero-based part index.\n * @param options Partial set of debug options to override for this part.\n */\n public setPartOptions(partIndex: number, options: Partial<IGaussianSplattingDebugOptions>): void {\n if (this._partCount === 0) {\n Logger.Error(\"GaussianSplattingDebugMaterialPlugin.setPartOptions: called on a non-compound mesh. partCount is 0.\");\n return;\n }\n if (options.clippingBox !== undefined) {\n this._partClippingBoxes[partIndex] = options.clippingBox;\n }\n if (options.opacityCulling !== undefined) {\n this._partOpacityCullings[partIndex] = options.opacityCulling;\n }\n if (options.sizeCulling !== undefined) {\n this._partSizeCullings[partIndex] = options.sizeCulling;\n }\n if (options.opacityScale !== undefined) {\n this._partOpacityScales[partIndex] = options.opacityScale;\n }\n if (options.opacitySaturate !== undefined) {\n this._partOpacitySaturates[partIndex] = options.opacitySaturate;\n }\n if (options.shDc !== undefined) {\n this._partShDcs[partIndex] = options.shDc;\n }\n if (options.shOrder1 !== undefined) {\n this._partShOrder1s[partIndex] = options.shOrder1;\n }\n if (options.shOrder2 !== undefined) {\n this._partShOrder2s[partIndex] = options.shOrder2;\n }\n if (options.shOrder3 !== undefined) {\n this._partShOrder3s[partIndex] = options.shOrder3;\n }\n if (options.shOrder4 !== undefined) {\n this._partShOrder4s[partIndex] = options.shOrder4;\n }\n this._markDirty();\n }\n\n /**\n * Clears all per-part debug overrides for the given part index,\n * falling back to global settings.\n * @param partIndex The zero-based part index.\n */\n public clearPartOptions(partIndex: number): void {\n this._partClippingBoxes[partIndex] = null;\n this._partOpacityCullings[partIndex] = null;\n this._partSizeCullings[partIndex] = null;\n this._partOpacityScales[partIndex] = null;\n this._partOpacitySaturates[partIndex] = null;\n this._partShDcs[partIndex] = null;\n this._partShOrder1s[partIndex] = null;\n this._partShOrder2s[partIndex] = null;\n this._partShOrder3s[partIndex] = null;\n this._partShOrder4s[partIndex] = null;\n this._markDirty();\n }\n\n /**\n * Removes the per-part override slot at `removedIndex` and shifts all higher-indexed\n * slots down by one, keeping the arrays aligned with the compound mesh's new part layout.\n * @param removedIndex The original (pre-removal) part index.\n * @internal\n */\n public shiftPartOptions(removedIndex: number): void {\n for (const arr of this._partArrays) {\n if (removedIndex < arr.length) {\n arr.splice(removedIndex, 1);\n }\n }\n this._markDirty();\n }\n\n /**\n * World-space axis-aligned clipping box. Splats outside are not rendered.\n * Set to null to disable clipping.\n * Example: `{ min: new Vector3(-2,-2,-2), max: new Vector3(2,2,2) }`\n */\n public get clippingBox(): Nullable<{ min: Vector3; max: Vector3 }> {\n return this._clippingBox;\n }\n public set clippingBox(value: Nullable<{ min: Vector3; max: Vector3 }>) {\n this._clippingBox = value;\n this._markDirty();\n }\n\n /**\n * Opacity culling range [0..1]. Splats whose stored opacity falls outside this\n * range are not rendered. Set to null to disable.\n */\n public get opacityCulling(): Nullable<{ min: number; max: number }> {\n return this._opacityCulling;\n }\n public set opacityCulling(value: Nullable<{ min: number; max: number }>) {\n this._opacityCulling = value;\n this._markDirty();\n }\n\n /**\n * Size culling range. Size is pow(|det(Σ)|, 1/6) of the 3D covariance matrix,\n * equal to the geometric mean of the principal radii. Use GaussianSplattingMeshBase.splatSizeRange\n * to find the asset's range. Set to null to disable.\n */\n public get sizeCulling(): Nullable<{ min: number; max: number }> {\n return this._sizeCulling;\n }\n public set sizeCulling(value: Nullable<{ min: number; max: number }>) {\n this._sizeCulling = value;\n this._markDirty();\n }\n\n /**\n * Scalar multiplier applied to every splat's opacity after all other modifiers.\n * 1.0 (default) = no change.\n */\n public get opacityScale(): number {\n return this._opacityScale;\n }\n public set opacityScale(value: number) {\n this._opacityScale = value;\n this._markDirty();\n }\n\n /**\n * When true, replaces the Gaussian spatial falloff with a flat uniform opacity,\n * making each splat appear as a solid disk with its raw alpha value.\n */\n public get opacitySaturate(): boolean {\n return this._opacitySaturate;\n }\n public set opacitySaturate(value: boolean) {\n this._opacitySaturate = value;\n this._markDirty();\n }\n\n /** Include the DC (base) color from colorsTexture. Default: true. */\n public get shDc(): boolean {\n return this._shDc;\n }\n public set shDc(value: boolean) {\n this._shDc = value;\n this._markDirty();\n }\n\n /** Include SH band 1 contribution. Default: true. */\n public get shOrder1(): boolean {\n return this._shOrder1;\n }\n public set shOrder1(value: boolean) {\n this._shOrder1 = value;\n this._markDirty();\n }\n\n /** Include SH band 2 contribution. Default: true. */\n public get shOrder2(): boolean {\n return this._shOrder2;\n }\n public set shOrder2(value: boolean) {\n this._shOrder2 = value;\n this._markDirty();\n }\n\n /** Include SH band 3 contribution. Default: true. */\n public get shOrder3(): boolean {\n return this._shOrder3;\n }\n public set shOrder3(value: boolean) {\n this._shOrder3 = value;\n this._markDirty();\n }\n\n /** Include SH band 4 contribution. Default: true. */\n public get shOrder4(): boolean {\n return this._shOrder4;\n }\n public set shOrder4(value: boolean) {\n this._shOrder4 = value;\n this._markDirty();\n }\n\n // ----- Plugin overrides -----\n\n /**\n * Adds the per-part debug data texture name to the sampler list so the effect can bind it.\n * @param samplers the sampler list to populate\n */\n public override getSamplers(samplers: string[]): void {\n samplers.push(\"dbgPartData\");\n }\n\n /** @returns the class name of this plugin */\n public override getClassName(): string {\n return \"GaussianSplattingDebugMaterialPlugin\";\n }\n\n /**\n * @param shaderLanguage the shader language to check\n * @returns true for GLSL and WGSL\n */\n public override isCompatible(shaderLanguage: ShaderLanguage): boolean {\n switch (shaderLanguage) {\n case ShaderLanguage.GLSL:\n case ShaderLanguage.WGSL:\n return true;\n default:\n return false;\n }\n }\n\n /**\n * Always ready — no async resources.\n * @param _defines unused\n * @param _scene unused\n * @param _engine unused\n * @param _subMesh unused\n * @returns true\n */\n public override isReadyForSubMesh(_defines: MaterialDefines, _scene: Scene, _engine: AbstractEngine, _subMesh: SubMesh): boolean {\n return true;\n }\n\n /**\n * Sets shader defines from current property state. GS_DBG_ENABLED is set to true\n * only when at least one feature is non-default, ensuring zero overhead otherwise.\n * Sub-flags also check per-part arrays so compound-only overrides activate the correct\n * code paths even when the global setting is at its default.\n * @param defines the defines object\n */\n public override prepareDefines(defines: GaussianSplattingDebugDefines): void {\n defines.GS_DBG_ENABLED = this._isAnyFeatureActive();\n defines.GS_DBG_CLIP = this._clippingBox !== null || this._partClippingBoxes.some((b) => b !== null) ? 1 : 0;\n defines.GS_DBG_CULL_OPACITY = this._opacityCulling !== null || this._partOpacityCullings.some((b) => b !== null) ? 1 : 0;\n defines.GS_DBG_CULL_SIZE = this._sizeCulling !== null || this._partSizeCullings.some((b) => b !== null) ? 1 : 0;\n defines.GS_DBG_OPACITY_SCALE = this._opacityScale !== 1.0 || this._partOpacityScales.some((s) => s !== null) ? 1 : 0;\n defines.GS_DBG_OPACITY_SATURATE = this._opacitySaturate || this._partOpacitySaturates.some((s) => s !== null) ? 1 : 0;\n defines.GS_DBG_SH_DC = !this._shDc || this._partShDcs.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER1 = !this._shOrder1 || this._partShOrder1s.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER2 = !this._shOrder2 || this._partShOrder2s.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER3 = !this._shOrder3 || this._partShOrder3s.some((s) => s !== null) ? 0 : 1;\n defines.GS_DBG_SH_ORDER4 = !this._shOrder4 || this._partShOrder4s.some((s) => s !== null) ? 0 : 1;\n }\n\n /**\n * Returns shader code injections for the debug features.\n * @param shaderType \"vertex\" or \"fragment\"\n * @param shaderLanguage GLSL or WGSL\n * @returns map of injection-point name to injected code, or null\n */\n public override getCustomCode(shaderType: string, shaderLanguage = ShaderLanguage.GLSL): Nullable<{ [pointName: string]: string }> {\n if (shaderLanguage === ShaderLanguage.WGSL) {\n return this._getCustomCodeWGSL(shaderType);\n }\n return this._getCustomCodeGLSL(shaderType);\n }\n\n private _getCustomCodeGLSL(shaderType: string): Nullable<{ [pointName: string]: string }> {\n if (shaderType === \"vertex\") {\n return {\n CUSTOM_VERTEX_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\nuniform vec3 dbgClipMin;\nuniform vec3 dbgClipMax;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\nuniform float dbgMinOpacity;\nuniform float dbgMaxOpacity;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\nuniform float dbgMinSize;\nuniform float dbgMaxSize;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\nuniform float dbgOpacityScale;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nuniform sampler2D dbgPartData;\nvarying float vPartIndex;\n#endif\n`,\n CUSTOM_VERTEX_UPDATE: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\n if (worldPos.x < dbgClipMin.x || worldPos.x > dbgClipMax.x ||\n worldPos.y < dbgClipMin.y || worldPos.y > dbgClipMax.y ||\n worldPos.z < dbgClipMin.z || worldPos.z > dbgClipMax.z) {\n scale = vec2(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\n if (splat.color.w < dbgMinOpacity || splat.color.w > dbgMaxOpacity) {\n scale = vec2(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\n {\n float _d0 = splat.covA.x; float _d1 = splat.covA.y; float _d2 = splat.covA.z;\n float _d3 = splat.covA.w; float _d4 = splat.covB.x; float _d5 = splat.covB.y;\n float _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n float _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < dbgMinSize || _sz > dbgMaxSize) {\n scale = vec2(0.0);\n }\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\n vColor.w *= dbgOpacityScale;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n {\n int _pIdx = int(splat.partIndex);\n vPartIndex = float(splat.partIndex);\n vec4 _row0 = texelFetch(dbgPartData, ivec2(_pIdx, 0), 0);\n vec4 _row1 = texelFetch(dbgPartData, ivec2(_pIdx, 1), 0);\n vec4 _row2 = texelFetch(dbgPartData, ivec2(_pIdx, 2), 0);\n vec3 _clipMin = _row0.xyz;\n vec3 _clipMax = vec3(_row0.w, _row1.xy);\n float _minOp = _row1.z;\n float _maxOp = _row1.w;\n float _minSz = _row2.x;\n float _maxSz = _row2.y;\n float _opSc = _row2.z;\n #if GS_DBG_CLIP == 1\n if (worldPos.x < _clipMin.x || worldPos.x > _clipMax.x ||\n worldPos.y < _clipMin.y || worldPos.y > _clipMax.y ||\n worldPos.z < _clipMin.z || worldPos.z > _clipMax.z) {\n scale = vec2(0.0);\n }\n #endif\n #if GS_DBG_CULL_OPACITY == 1\n if (splat.color.w < _minOp || splat.color.w > _maxOp) {\n scale = vec2(0.0);\n }\n #endif\n #if GS_DBG_CULL_SIZE == 1\n {\n float _d0 = splat.covA.x; float _d1 = splat.covA.y; float _d2 = splat.covA.z;\n float _d3 = splat.covA.w; float _d4 = splat.covB.x; float _d5 = splat.covB.y;\n float _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n float _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < _minSz || _sz > _maxSz) {\n scale = vec2(0.0);\n }\n }\n #endif\n #if GS_DBG_OPACITY_SCALE == 1\n vColor.w *= _opSc;\n #endif\n }\n#endif\n`,\n };\n } else if (shaderType === \"fragment\") {\n return {\n CUSTOM_FRAGMENT_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nuniform sampler2D dbgPartData;\nvarying float vPartIndex;\n#endif\n`,\n CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n if (texelFetch(dbgPartData, ivec2(int(vPartIndex + 0.5), 2), 0).w > 0.5) { finalColor.a = vColor.a; }\n#elif defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SATURATE == 1\n finalColor.a = vColor.a;\n#endif\n`,\n };\n }\n return null;\n }\n\n private _getCustomCodeWGSL(shaderType: string): Nullable<{ [pointName: string]: string }> {\n if (shaderType === \"vertex\") {\n return {\n CUSTOM_VERTEX_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\nuniform dbgClipMin: vec3f;\nuniform dbgClipMax: vec3f;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\nuniform dbgMinOpacity: f32;\nuniform dbgMaxOpacity: f32;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\nuniform dbgMinSize: f32;\nuniform dbgMaxSize: f32;\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\nuniform dbgOpacityScale: f32;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nvar dbgPartData: texture_2d<f32>;\nvarying vPartIndex: f32;\n#endif\n`,\n CUSTOM_VERTEX_UPDATE: `\n#if defined(GS_DBG_ENABLED) && GS_DBG_CLIP == 1 && !defined(IS_COMPOUND)\n if (worldPos.x < uniforms.dbgClipMin.x || worldPos.x > uniforms.dbgClipMax.x ||\n worldPos.y < uniforms.dbgClipMin.y || worldPos.y > uniforms.dbgClipMax.y ||\n worldPos.z < uniforms.dbgClipMin.z || worldPos.z > uniforms.dbgClipMax.z) {\n scale = vec2f(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_OPACITY == 1 && !defined(IS_COMPOUND)\n if (splat.color.w < uniforms.dbgMinOpacity || splat.color.w > uniforms.dbgMaxOpacity) {\n scale = vec2f(0.0);\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_CULL_SIZE == 1 && !defined(IS_COMPOUND)\n {\n let _d0 = splat.covA.x; let _d1 = splat.covA.y; let _d2 = splat.covA.z;\n let _d3 = splat.covA.w; let _d4 = splat.covB.x; let _d5 = splat.covB.y;\n let _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n let _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < uniforms.dbgMinSize || _sz > uniforms.dbgMaxSize) {\n scale = vec2f(0.0);\n }\n }\n#endif\n#if defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SCALE == 1 && !defined(IS_COMPOUND)\n vertexOutputs.vColor.w *= uniforms.dbgOpacityScale;\n#endif\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n {\n let _pIdx = i32(splat.partIndex);\n vertexOutputs.vPartIndex = f32(splat.partIndex);\n let _row0 = textureLoad(dbgPartData, vec2i(_pIdx, 0), 0);\n let _row1 = textureLoad(dbgPartData, vec2i(_pIdx, 1), 0);\n let _row2 = textureLoad(dbgPartData, vec2i(_pIdx, 2), 0);\n let _clipMin = _row0.xyz;\n let _clipMax = vec3f(_row0.w, _row1.xy);\n let _minOp = _row1.z;\n let _maxOp = _row1.w;\n let _minSz = _row2.x;\n let _maxSz = _row2.y;\n let _opSc = _row2.z;\n #if GS_DBG_CLIP == 1\n if (worldPos.x < _clipMin.x || worldPos.x > _clipMax.x ||\n worldPos.y < _clipMin.y || worldPos.y > _clipMax.y ||\n worldPos.z < _clipMin.z || worldPos.z > _clipMax.z) {\n scale = vec2f(0.0);\n }\n #endif\n #if GS_DBG_CULL_OPACITY == 1\n if (splat.color.w < _minOp || splat.color.w > _maxOp) {\n scale = vec2f(0.0);\n }\n #endif\n #if GS_DBG_CULL_SIZE == 1\n {\n let _d0 = splat.covA.x; let _d1 = splat.covA.y; let _d2 = splat.covA.z;\n let _d3 = splat.covA.w; let _d4 = splat.covB.x; let _d5 = splat.covB.y;\n let _det = _d0*(_d3*_d5 - _d4*_d4) - _d1*(_d1*_d5 - _d4*_d2) + _d2*(_d1*_d4 - _d3*_d2);\n let _sz = pow(abs(_det), 1.0/6.0);\n if (_sz < _minSz || _sz > _maxSz) {\n scale = vec2f(0.0);\n }\n }\n #endif\n #if GS_DBG_OPACITY_SCALE == 1\n vertexOutputs.vColor.w *= _opSc;\n #endif\n }\n#endif\n`,\n };\n } else if (shaderType === \"fragment\") {\n return {\n CUSTOM_FRAGMENT_DEFINITIONS: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\nvar dbgPartData: texture_2d<f32>;\nvarying vPartIndex: f32;\n#endif\n`,\n CUSTOM_FRAGMENT_BEFORE_FRAGCOLOR: `\n#if defined(GS_DBG_ENABLED) && IS_COMPOUND\n if (textureLoad(dbgPartData, vec2i(i32(fragmentInputs.vPartIndex + 0.5), 2), 0).w > 0.5) {\n let _gsdbgA: f32 = -dot(fragmentInputs.vPosition, fragmentInputs.vPosition);\n if (_gsdbgA > -4.0) { finalColor.a = fragmentInputs.vColor.a; }\n }\n#elif defined(GS_DBG_ENABLED) && GS_DBG_OPACITY_SATURATE == 1\n {\n let _gsdbgA: f32 = -dot(fragmentInputs.vPosition, fragmentInputs.vPosition);\n if (_gsdbgA > -4.0) {\n finalColor.a = fragmentInputs.vColor.a;\n }\n }\n#endif\n`,\n };\n }\n return null;\n }\n\n /**\n * Declares the non-compound scalar debug uniform names as external so the Effect can\n * resolve their locations. WGSL uniforms are declared inline in getCustomCode() injections.\n * @returns uniform descriptor with externalUniforms list\n */\n public override getUniforms(): {\n ubo?: Array<{ name: string; size?: number; type?: string; arraySize?: number }>;\n vertex?: string;\n fragment?: string;\n externalUniforms?: string[];\n } {\n return {\n externalUniforms: [\"dbgClipMin\", \"dbgClipMax\", \"dbgMinOpacity\", \"dbgMaxOpacity\", \"dbgMinSize\", \"dbgMaxSize\", \"dbgOpacityScale\"],\n };\n }\n\n private _buildTextureData(partCount: number): Float32Array {\n const maxPartCount = this._maxPartCount;\n const data = new Float32Array(maxPartCount * 5 * 4);\n for (let i = 0; i < maxPartCount; i++) {\n const r0 = i * 4;\n const r1 = (maxPartCount + i) * 4;\n const r2 = (maxPartCount * 2 + i) * 4;\n const r3 = (maxPartCount * 3 + i) * 4;\n const r4 = (maxPartCount * 4 + i) * 4;\n\n // Row 0: clipMin.xyz, clipMax.x — Row 1: clipMax.yz, minOpacity, maxOpacity\n const box = i < partCount ? (this._partClippingBoxes[i] ?? this._clippingBox) : this._clippingBox;\n if (box) {\n data[r0] = box.min.x;\n data[r0 + 1] = box.min.y;\n data[r0 + 2] = box.min.z;\n data[r0 + 3] = box.max.x;\n data[r1] = box.max.y;\n data[r1 + 1] = box.max.z;\n } else {\n data[r0] = -1e9;\n data[r0 + 1] = -1e9;\n data[r0 + 2] = -1e9;\n data[r0 + 3] = 1e9;\n data[r1] = 1e9;\n data[r1 + 1] = 1e9;\n }\n const opCull = i < partCount ? (this._partOpacityCullings[i] ?? this._opacityCulling) : this._opacityCulling;\n data[r1 + 2] = opCull ? opCull.min : 0.0;\n data[r1 + 3] = opCull ? opCull.max : 1.0;\n\n // Row 2: minSize, maxSize, opacityScale, opacitySaturate\n const szCull = i < partCount ? (this._partSizeCullings[i] ?? this._sizeCulling) : this._sizeCulling;\n data[r2] = szCull ? szCull.min : 0.0;\n data[r2 + 1] = szCull ? szCull.max : 1e9;\n data[r2 + 2] = i < partCount ? (this._partOpacityScales[i] ?? this._opacityScale) : this._opacityScale;\n const sat = i < partCount ? (this._partOpacitySaturates[i] ?? this._opacitySaturate) : this._opacitySaturate;\n data[r2 + 3] = sat ? 1.0 : 0.0;\n\n // Row 3: shDc, shOrder1, shOrder2, shOrder3\n data[r3] = (i < partCount ? (this._partShDcs[i] ?? this._shDc) : this._shDc) ? 1.0 : 0.0;\n data[r3 + 1] = (i < partCount ? (this._partShOrder1s[i] ?? this._shOrder1) : this._shOrder1) ? 1.0 : 0.0;\n data[r3 + 2] = (i < partCount ? (this._partShOrder2s[i] ?? this._shOrder2) : this._shOrder2) ? 1.0 : 0.0;\n data[r3 + 3] = (i < partCount ? (this._partShOrder3s[i] ?? this._shOrder3) : this._shOrder3) ? 1.0 : 0.0;\n\n // Row 4: shOrder4, padding\n data[r4] = (i < partCount ? (this._partShOrder4s[i] ?? this._shOrder4) : this._shOrder4) ? 1.0 : 0.0;\n }\n return data;\n }\n\n private _updateOrCreateTexture(scene: Scene): void {\n if (!this._maxPartCount) {\n this._maxPartCount = GetGaussianSplattingMaxPartCount(scene.getEngine());\n }\n const data = this._buildTextureData(this._partCount);\n if (!this._dbgPartDataTexture) {\n this._dbgPartDataTexture = RawTexture.CreateRGBATexture(\n data,\n this._maxPartCount,\n 5,\n scene,\n false,\n false,\n Constants.TEXTURE_NEAREST_SAMPLINGMODE,\n Constants.TEXTURETYPE_FLOAT\n );\n } else {\n this._dbgPartDataTexture.update(data);\n }\n this._textureDirty = false;\n }\n\n /**\n * Binds uniform values each frame. Scalar uniforms are uploaded for non-compound mode;\n * the per-part data texture is updated and bound for compound mode.\n * @param _uniformBuffer unused\n * @param _scene the current scene\n * @param _engine unused\n * @param subMesh the submesh being rendered\n */\n public override bindForSubMesh(_uniformBuffer: UniformBuffer, _scene: Scene, _engine: AbstractEngine, subMesh: SubMesh): void {\n const effect = subMesh.effect;\n if (!effect) {\n return;\n }\n\n if (this._partCount > 0 && this._isAnyFeatureActive()) {\n // Compound mode: pack per-part data into a texture (only when the debug path is active)\n if (this._textureDirty || !this._dbgPartDataTexture) {\n this._updateOrCreateTexture(_scene);\n }\n if (this._dbgPartDataTexture) {\n effect.setTexture(\"dbgPartData\", this._dbgPartDataTexture);\n }\n } else if (this._partCount === 0) {\n // Non-compound: upload scalar uniforms\n if (this._clippingBox) {\n effect.setVector3(\"dbgClipMin\", this._clippingBox.min);\n effect.setVector3(\"dbgClipMax\", this._clippingBox.max);\n }\n if (this._opacityCulling) {\n effect.setFloat(\"dbgMinOpacity\", this._opacityCulling.min);\n effect.setFloat(\"dbgMaxOpacity\", this._opacityCulling.max);\n }\n if (this._sizeCulling) {\n effect.setFloat(\"dbgMinSize\", this._sizeCulling.min);\n effect.setFloat(\"dbgMaxSize\", this._sizeCulling.max);\n }\n if (this._opacityScale !== 1.0) {\n effect.setFloat(\"dbgOpacityScale\", this._opacityScale);\n }\n }\n }\n\n /**\n * Disposes the per-part data texture.\n * @param _forceDisposeTextures unused; the LUT texture is always disposed as it is owned by this plugin\n */\n public override dispose(_forceDisposeTextures?: boolean): void {\n this._dbgPartDataTexture?.dispose();\n this._dbgPartDataTexture = null;\n }\n}\n\nlet _Registered = false;\n/**\n * Register side effects for GaussianSplattingDebugMaterialPlugin.\n * Safe to call multiple times; only the first call has an effect.\n */\nexport function RegisterGaussianSplattingDebugMaterialPlugin(): void {\n if (_Registered) {\n return;\n }\n _Registered = true;\n\n RegisterClass(\"BABYLON.GaussianSplattingDebugMaterialPlugin\", GaussianSplattingDebugMaterialPlugin);\n}\n"]}
@@ -31,6 +31,8 @@ export declare class GaussianSplattingDebugger {
31
31
  private _plugins;
32
32
  private _meshes;
33
33
  private _disposeObservers;
34
+ private _partCountObservers;
35
+ private _partRemovedObservers;
34
36
  private _clippingBox;
35
37
  private _opacityCulling;
36
38
  private _sizeCulling;
@@ -30,6 +30,9 @@ export class GaussianSplattingDebugger {
30
30
  this._plugins = [];
31
31
  this._meshes = [];
32
32
  this._disposeObservers = [];
33
+ // observable.add() returns Nullable<Observer<T>>; Observable.remove() accepts null safely.
34
+ this._partCountObservers = [];
35
+ this._partRemovedObservers = [];
33
36
  // Cached option state so newly added meshes inherit current settings
34
37
  this._clippingBox = null;
35
38
  this._opacityCulling = null;
@@ -63,6 +66,12 @@ export class GaussianSplattingDebugger {
63
66
  this._meshes.push(mesh);
64
67
  this._plugins.push(plugin);
65
68
  this._disposeObservers.push(mesh.onDisposeObservable.add(() => this.removeMesh(mesh)));
69
+ this._partCountObservers.push(mesh.onPartCountChangedObservable.add((count) => {
70
+ plugin.partCount = count;
71
+ }));
72
+ this._partRemovedObservers.push(mesh.onPartRemovedObservable.add((removedIndex) => {
73
+ plugin.shiftPartOptions(removedIndex);
74
+ }));
66
75
  }
67
76
  /**
68
77
  * Removes a mesh and disposes its debug plugin.
@@ -74,20 +83,28 @@ export class GaussianSplattingDebugger {
74
83
  return;
75
84
  }
76
85
  mesh.onDisposeObservable.remove(this._disposeObservers[idx]);
86
+ mesh.onPartCountChangedObservable.remove(this._partCountObservers[idx]);
87
+ mesh.onPartRemovedObservable.remove(this._partRemovedObservers[idx]);
77
88
  this._plugins[idx].dispose();
78
89
  this._meshes.splice(idx, 1);
79
90
  this._plugins.splice(idx, 1);
80
91
  this._disposeObservers.splice(idx, 1);
92
+ this._partCountObservers.splice(idx, 1);
93
+ this._partRemovedObservers.splice(idx, 1);
81
94
  }
82
95
  /** Disposes all debug plugins and clears the mesh list. */
83
96
  dispose() {
84
97
  for (let i = 0; i < this._meshes.length; i++) {
85
98
  this._meshes[i].onDisposeObservable.remove(this._disposeObservers[i]);
99
+ this._meshes[i].onPartCountChangedObservable.remove(this._partCountObservers[i]);
100
+ this._meshes[i].onPartRemovedObservable.remove(this._partRemovedObservers[i]);
86
101
  this._plugins[i].dispose();
87
102
  }
88
103
  this._meshes.length = 0;
89
104
  this._plugins.length = 0;
90
105
  this._disposeObservers.length = 0;
106
+ this._partCountObservers.length = 0;
107
+ this._partRemovedObservers.length = 0;
91
108
  }
92
109
  /**
93
110
  * Returns the min/max size range of splats in a mesh.