mujoco-react 9.2.0 → 9.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -165,11 +165,24 @@ visual scenarios as data, pass the scenario directly; the component resolves the
165
165
  splat asset and paired MJCF collision proxy metadata for you.
166
166
 
167
167
  ```tsx
168
- import { SplatEnvironment, withSplatEnvironment } from "mujoco-react";
168
+ import { MujocoCanvas, SplatEnvironment, useSplatSceneConfig } from "mujoco-react";
169
169
 
170
- <SplatEnvironment scenario={scenario} renderer="custom" />;
170
+ const splat = useSplatSceneConfig({ sceneConfig, scenario });
171
+
172
+ <MujocoCanvas config={splat.sceneConfig}>
173
+ {splat.environment ? (
174
+ <SplatEnvironment environment={splat.environment} renderer="custom">
175
+ <MySplatRenderer src={splat.environment.splat.src} />
176
+ </SplatEnvironment>
177
+ ) : null}
178
+ </MujocoCanvas>;
171
179
  ```
172
180
 
181
+ Use `splat.readiness` or `getSplatEnvironmentReadiness(scenario)` to gate
182
+ authoring and import flows. The status distinguishes disabled scenarios,
183
+ missing splat assets, missing MJCF collision proxies, unsupported Spark formats,
184
+ and ready paired environments.
185
+
173
186
  For MuJoCo + 3DGS composition, derive the collision environment from the same
174
187
  splat metadata and pass the resulting config to `<MujocoCanvas>`:
175
188
 
@@ -192,16 +205,18 @@ npm install @sparkjsdev/spark
192
205
  ```tsx
193
206
  import {
194
207
  SparkSplatEnvironment,
195
- useSparkSplatLifecycle,
208
+ useSparkSplatEnvironment,
196
209
  } from "mujoco-react/spark";
197
210
 
198
211
  function Scene() {
199
- const splat = useSparkSplatLifecycle();
212
+ const splat = useSparkSplatEnvironment({ sceneConfig, scenario });
200
213
 
201
214
  return (
202
- <MujocoCanvas config={sceneConfig} gl={{ preserveDrawingBuffer: true }}>
203
- <SparkSplatEnvironment scenario={scenario} hideGroundMeshes {...splat.props} />
204
- <StatusBadge status={splat.status} error={splat.error} />
215
+ <MujocoCanvas config={splat.sceneConfig} gl={{ preserveDrawingBuffer: true }}>
216
+ {splat.environment ? (
217
+ <SparkSplatEnvironment hideGroundMeshes {...splat.props} />
218
+ ) : null}
219
+ <StatusBadge status={splat.lifecycle.status} error={splat.lifecycle.error} />
205
220
  </MujocoCanvas>
206
221
  );
207
222
  }
@@ -1009,11 +1024,82 @@ Use `useFrameCapture()` or the standalone `captureFrame()` helpers when you own
1009
1024
  the canvas or want to capture a custom container.
1010
1025
 
1011
1026
  Use `captureCameraFrame()` / `captureCameraFrameBlob()` when dataset generation
1012
- needs a fixed offscreen camera pose or resolution without moving the user's
1013
- interactive viewport.
1027
+ needs an offscreen camera render at a stable resolution without moving the
1028
+ user's interactive viewport. Pass `cameraName`, `siteName`, or `bodyName` to
1029
+ record true MuJoCo-mounted camera frames; the returned image includes
1030
+ `source.kind` so dataset pipelines can reject fallback or synthetic fixed poses.
1014
1031
 
1015
1032
  Use `recordCameraSequence()` / `useCameraSequenceRecorder()` to step policy
1016
- rollouts and capture synchronized frames from one or more fixed camera configs.
1033
+ rollouts and capture synchronized per-camera frames from one or more MuJoCo
1034
+ camera configs. Sequence recording requires mounted MuJoCo camera, site, or
1035
+ body selectors by default; use still capture APIs for synthetic debug poses.
1036
+
1037
+ For LeRobot-style datasets, prefer the named-camera wrapper. It resolves task
1038
+ camera keys to MuJoCo cameras/sites/bodies, records the sequence, and returns
1039
+ the plan and readiness summary alongside frame provenance:
1040
+
1041
+ ```tsx
1042
+ const sequence = await recordMountedCameraFrameSequence(api, {
1043
+ cameraKeys: ["head", "left_wrist", "right_wrist"],
1044
+ aliases: {
1045
+ head: [{ siteName: "head_camera_rgb_optical_frame" }],
1046
+ left_wrist: [{ siteName: "left_wrist_camera_optical_frame" }],
1047
+ right_wrist: [{ siteName: "right_wrist_camera_optical_frame" }],
1048
+ },
1049
+ defaults: {
1050
+ width: 640,
1051
+ height: 480,
1052
+ type: "image/png",
1053
+ fov: 45,
1054
+ near: 0.01,
1055
+ far: 100,
1056
+ },
1057
+ frames: 16,
1058
+ stepsPerFrame: 1,
1059
+ retainFrames: false,
1060
+ requireMountedSources: true,
1061
+ onFrame: ({ frameIndex, cameras }) => {
1062
+ queueLeRobotImages(frameIndex, cameras);
1063
+ },
1064
+ });
1065
+
1066
+ sequence.readiness.ready; // true when every requested stream resolved
1067
+ sequence.plan.missingKeys; // unresolved task cameras, if requireAll is false
1068
+ sequence.cameraSummaries.head.source; // mounted source provenance
1069
+ ```
1070
+
1071
+ `recordMountedCameraFrameSequence()` requires all requested `cameraKeys` by
1072
+ default so dataset recording cannot silently omit a camera stream. Set
1073
+ `requireAll: false` only for exploratory tooling that can tolerate partial
1074
+ camera coverage.
1075
+
1076
+ Inside `<MujocoCanvas>` children, `useMountedCameraSequenceRecorder()` exposes
1077
+ the same planning and recording surface with React status/error state.
1078
+
1079
+ Use `resolveMountedCameraFrameSource()` when dataset feature names need to map
1080
+ to named MuJoCo cameras, sites, or bodies before recording. The helper accepts
1081
+ the model resource lists plus app-level aliases and returns both the capture
1082
+ selector and the mounted-source provenance that should be stored beside the
1083
+ dataset:
1084
+
1085
+ ```tsx
1086
+ const resolved = resolveMountedCameraFrameSource("head", {
1087
+ cameras: api.getCameras(),
1088
+ sites: api.getSites(),
1089
+ bodies: api.getBodies(),
1090
+ aliases: {
1091
+ head: [{ siteName: "head_camera_rgb_optical_frame" }],
1092
+ },
1093
+ });
1094
+
1095
+ if (!resolved) throw new Error("head does not resolve to a MuJoCo source");
1096
+
1097
+ await api.recordCameraSequence({
1098
+ frames: 16,
1099
+ requireMountedSources: true,
1100
+ cameras: [{ key: "head", width: 640, height: 480, ...resolved.selector }],
1101
+ });
1102
+ ```
1017
1103
 
1018
1104
  ### `useCtrlNoise(config)`
1019
1105
 
@@ -3,7 +3,88 @@ import { useEffect, useMemo } from 'react';
3
3
  import * as THREE from 'three';
4
4
  import { jsxs, Fragment, jsx } from 'react/jsx-runtime';
5
5
 
6
- // src/components/VisualScenario.tsx
6
+ // src/types.ts
7
+ var runtimeRobotResources = {};
8
+ var REGISTER_RESOURCE_KEYS = ["actuators", "sensors", "bodies", "joints", "sites", "geoms", "keyframes", "cameras"];
9
+ function createEmptyRuntimeResources() {
10
+ return {
11
+ actuators: {},
12
+ sensors: {},
13
+ bodies: {},
14
+ joints: {},
15
+ sites: {},
16
+ geoms: {},
17
+ keyframes: {},
18
+ cameras: {}
19
+ };
20
+ }
21
+ function registerRobotResources(resources) {
22
+ for (const [robot, robotResources] of Object.entries(resources)) {
23
+ const existing = runtimeRobotResources[robot] ?? createEmptyRuntimeResources();
24
+ for (const key of REGISTER_RESOURCE_KEYS) {
25
+ existing[key] = { ...existing[key], ...robotResources[key] ?? {} };
26
+ }
27
+ runtimeRobotResources[robot] = existing;
28
+ }
29
+ }
30
+ function createResourceCategory(key) {
31
+ return new Proxy({}, {
32
+ get(_target, robot) {
33
+ if (typeof robot !== "string") return void 0;
34
+ return runtimeRobotResources[robot]?.[key] ?? {};
35
+ },
36
+ ownKeys() {
37
+ return Reflect.ownKeys(runtimeRobotResources);
38
+ },
39
+ getOwnPropertyDescriptor(_target, robot) {
40
+ if (typeof robot !== "string" || !(robot in runtimeRobotResources)) return void 0;
41
+ return { enumerable: true, configurable: true };
42
+ }
43
+ });
44
+ }
45
+ var RobotResources = new Proxy(runtimeRobotResources, {
46
+ get(target, robot) {
47
+ if (typeof robot !== "string") return void 0;
48
+ return target[robot] ?? createEmptyRuntimeResources();
49
+ },
50
+ ownKeys(target) {
51
+ return Reflect.ownKeys(target);
52
+ },
53
+ getOwnPropertyDescriptor(target, robot) {
54
+ if (typeof robot !== "string" || !(robot in target)) return void 0;
55
+ return { enumerable: true, configurable: true };
56
+ }
57
+ });
58
+ var RobotActuators = createResourceCategory("actuators");
59
+ var RobotSensors = createResourceCategory("sensors");
60
+ var RobotBodies = createResourceCategory("bodies");
61
+ var RobotJoints = createResourceCategory("joints");
62
+ var RobotSites = createResourceCategory("sites");
63
+ var RobotGeoms = createResourceCategory("geoms");
64
+ var RobotKeyframes = createResourceCategory("keyframes");
65
+ var RobotCameras = createResourceCategory("cameras");
66
+ function getContact(contacts, i) {
67
+ try {
68
+ return contacts.get(i);
69
+ } catch {
70
+ return void 0;
71
+ }
72
+ }
73
+ function withContacts(data, read) {
74
+ const contacts = data.contact;
75
+ try {
76
+ return read(contacts);
77
+ } finally {
78
+ contacts.delete?.();
79
+ }
80
+ }
81
+ var SplatEnvironmentReadinessStatus = {
82
+ Disabled: "disabled",
83
+ MissingSplat: "missing-splat",
84
+ MissingCollisionProxy: "missing-collision-proxy",
85
+ UnsupportedFormat: "unsupported-format",
86
+ Ready: "ready"
87
+ };
7
88
  var DEFAULT_BACKGROUND = "#181a1f";
8
89
  function ScenarioLighting({
9
90
  preset = "studio",
@@ -205,21 +286,151 @@ function useSplatEnvironment({
205
286
  const resolvedSrc = src ?? scenarioEnvironment?.splat.src ?? scenario?.splat?.src;
206
287
  const resolvedFormat = format ?? scenarioEnvironment?.splat.format ?? scenario?.splat?.format ?? "spz";
207
288
  const resolvedCollisionProxy = collisionProxy ?? scenarioEnvironment?.collisionProxy ?? scenario?.splat?.collisionProxy ?? void 0;
289
+ const readiness = useMemo(
290
+ () => getSplatEnvironmentReadiness({
291
+ environment: scenarioEnvironment,
292
+ scenario,
293
+ renderer,
294
+ src: resolvedSrc,
295
+ format: resolvedFormat,
296
+ collisionProxy: resolvedCollisionProxy
297
+ }),
298
+ [
299
+ collisionProxy,
300
+ renderer,
301
+ resolvedCollisionProxy,
302
+ resolvedFormat,
303
+ resolvedSrc,
304
+ scenario,
305
+ scenarioEnvironment
306
+ ]
307
+ );
208
308
  return useMemo(
209
309
  () => ({
210
310
  src: resolvedSrc,
211
311
  format: resolvedFormat,
212
312
  collisionProxy: resolvedCollisionProxy,
313
+ readiness,
213
314
  userData: createSplatEnvironmentUserData({
214
315
  environment: scenarioEnvironment,
215
316
  src: resolvedSrc,
216
317
  format: resolvedFormat,
217
- collisionProxy: resolvedCollisionProxy
318
+ collisionProxy: resolvedCollisionProxy,
319
+ readiness
218
320
  })
219
321
  }),
220
- [scenarioEnvironment, resolvedSrc, resolvedFormat, resolvedCollisionProxy]
322
+ [
323
+ scenarioEnvironment,
324
+ resolvedSrc,
325
+ resolvedFormat,
326
+ resolvedCollisionProxy,
327
+ readiness
328
+ ]
329
+ );
330
+ }
331
+ function useSplatSceneConfig({
332
+ sceneConfig,
333
+ scenario,
334
+ environment,
335
+ enabled = true,
336
+ renderer
337
+ }) {
338
+ const resolvedEnvironment = useMemo(
339
+ () => enabled ? environment ?? (scenario ? createPairedSplatEnvironment(scenario, { renderer }) : void 0) : void 0,
340
+ [enabled, environment, renderer, scenario]
341
+ );
342
+ const readiness = useMemo(
343
+ () => getSplatEnvironmentReadiness({
344
+ environment: resolvedEnvironment,
345
+ scenario,
346
+ renderer,
347
+ enabled
348
+ }),
349
+ [enabled, renderer, resolvedEnvironment, scenario]
350
+ );
351
+ const resolvedSceneConfig = useMemo(
352
+ () => resolvedEnvironment ? withSplatEnvironment(sceneConfig, resolvedEnvironment) : sceneConfig,
353
+ [resolvedEnvironment, sceneConfig]
354
+ );
355
+ return useMemo(
356
+ () => ({
357
+ environment: resolvedEnvironment,
358
+ sceneConfig: resolvedSceneConfig,
359
+ enabled: enabled && readiness.status !== SplatEnvironmentReadinessStatus.Disabled,
360
+ readiness
361
+ }),
362
+ [enabled, readiness, resolvedEnvironment, resolvedSceneConfig]
221
363
  );
222
364
  }
365
+ function getSplatEnvironmentReadiness({
366
+ environment,
367
+ scenario,
368
+ renderer,
369
+ src,
370
+ format,
371
+ collisionProxy,
372
+ enabled = true
373
+ }) {
374
+ const splat = scenario?.splat;
375
+ const resolvedSrc = src ?? environment?.splat.src ?? splat?.src;
376
+ const resolvedFormat = format ?? environment?.splat.format ?? splat?.format ?? "spz";
377
+ const resolvedRenderer = renderer ?? environment?.splat.renderer;
378
+ const resolvedCollisionProxy = collisionProxy ?? environment?.collisionProxy ?? splat?.collisionProxy ?? void 0;
379
+ const requiresCollisionProxy = splat?.requiresCollisionProxy ?? true;
380
+ if (!enabled || splat && splat.enabled === false && !environment) {
381
+ return {
382
+ status: SplatEnvironmentReadinessStatus.Disabled,
383
+ ready: false,
384
+ requiresCollisionProxy,
385
+ missing: [],
386
+ format: resolvedFormat,
387
+ renderer: resolvedRenderer,
388
+ message: "Splat environment is disabled."
389
+ };
390
+ }
391
+ if (!resolvedSrc) {
392
+ return {
393
+ status: SplatEnvironmentReadinessStatus.MissingSplat,
394
+ ready: false,
395
+ requiresCollisionProxy,
396
+ missing: ["splat"],
397
+ format: resolvedFormat,
398
+ renderer: resolvedRenderer,
399
+ message: "Splat environment is missing a visual asset source."
400
+ };
401
+ }
402
+ if (resolvedRenderer === "spark" && resolvedFormat !== "spz") {
403
+ return {
404
+ status: SplatEnvironmentReadinessStatus.UnsupportedFormat,
405
+ ready: false,
406
+ requiresCollisionProxy,
407
+ missing: [],
408
+ format: resolvedFormat,
409
+ renderer: resolvedRenderer,
410
+ message: `Spark splat rendering requires .spz assets; received ${resolvedFormat}.`
411
+ };
412
+ }
413
+ if (requiresCollisionProxy && !resolvedCollisionProxy?.xmlPath) {
414
+ return {
415
+ status: SplatEnvironmentReadinessStatus.MissingCollisionProxy,
416
+ ready: false,
417
+ requiresCollisionProxy,
418
+ missing: ["collisionProxy"],
419
+ format: resolvedFormat,
420
+ renderer: resolvedRenderer,
421
+ message: "Splat environment is missing paired MJCF collision proxy XML."
422
+ };
423
+ }
424
+ return {
425
+ status: SplatEnvironmentReadinessStatus.Ready,
426
+ ready: true,
427
+ requiresCollisionProxy,
428
+ missing: [],
429
+ format: resolvedFormat,
430
+ renderer: resolvedRenderer,
431
+ message: requiresCollisionProxy ? "Splat environment has visual asset and collision proxy metadata." : "Splat environment has a visual asset and does not require collision proxy metadata."
432
+ };
433
+ }
223
434
  function createPairedSplatEnvironment(scenario, options = {}) {
224
435
  const splat = scenario.splat;
225
436
  const collisionProxy = splat?.collisionProxy;
@@ -277,7 +488,8 @@ function createSplatEnvironmentUserData({
277
488
  environment,
278
489
  src,
279
490
  format = "spz",
280
- collisionProxy
491
+ collisionProxy,
492
+ readiness
281
493
  }) {
282
494
  return {
283
495
  role: "splat-environment",
@@ -288,7 +500,9 @@ function createSplatEnvironmentUserData({
288
500
  splatRenderer: environment?.splat.renderer,
289
501
  collisionProxyStatus: collisionProxy?.status ?? "missing",
290
502
  collisionProxyXmlPath: collisionProxy?.xmlPath,
291
- collisionProxyPrimitives: collisionProxy?.primitives ?? []
503
+ collisionProxyPrimitives: collisionProxy?.primitives ?? [],
504
+ readinessStatus: readiness?.status,
505
+ readinessMessage: readiness?.message
292
506
  };
293
507
  }
294
508
  function createSparkSplatViewerUrl({
@@ -395,6 +609,6 @@ function clamp01(value) {
395
609
  * SPDX-License-Identifier: Apache-2.0
396
610
  */
397
611
 
398
- export { ScenarioLighting, SplatEnvironment, VisualScenarioEffects, createPairedSplatEnvironment, createSparkSplatViewerUrl, createSplatEnvironmentUserData, getScenarioBackground, getScenarioCameraPosition, useSplatEnvironment, useVisualScenarioEffects, withSplatEnvironment };
399
- //# sourceMappingURL=chunk-33CV6HSV.js.map
400
- //# sourceMappingURL=chunk-33CV6HSV.js.map
612
+ export { RobotActuators, RobotBodies, RobotCameras, RobotGeoms, RobotJoints, RobotKeyframes, RobotResources, RobotSensors, RobotSites, ScenarioLighting, SplatEnvironment, SplatEnvironmentReadinessStatus, VisualScenarioEffects, createPairedSplatEnvironment, createSparkSplatViewerUrl, createSplatEnvironmentUserData, getContact, getScenarioBackground, getScenarioCameraPosition, getSplatEnvironmentReadiness, registerRobotResources, useSplatEnvironment, useSplatSceneConfig, useVisualScenarioEffects, withContacts, withSplatEnvironment };
613
+ //# sourceMappingURL=chunk-T3GVZJ4F.js.map
614
+ //# sourceMappingURL=chunk-T3GVZJ4F.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/types.ts","../src/components/VisualScenario.tsx"],"names":[],"mappings":"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","file":"chunk-T3GVZJ4F.js","sourcesContent":["/**\n * @license\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport type React from 'react';\nimport type { ReactNode } from 'react';\nimport type { CanvasProps, ThreeElements } from '@react-three/fiber';\nimport * as THREE from 'three';\n\n// ---- Register (type-safe named resources) ----\n\n/**\n * Module augmentation interface for type-safe resource names.\n *\n * Declare your model's resource names via module augmentation:\n * ```ts\n * declare module 'mujoco-react' {\n * interface Register {\n * robots: {\n * panda: {\n * actuators: 'joint1' | 'joint2' | 'gripper';\n * sensors: 'force_sensor' | 'torque_sensor';\n * bodies: 'link0' | 'link1' | 'hand';\n * };\n * };\n * actuators: 'joint1' | 'joint2' | 'gripper';\n * sensors: 'force_sensor' | 'torque_sensor';\n * bodies: 'link0' | 'link1' | 'hand';\n * }\n * }\n * ```\n *\n * When no augmentation is declared, all names fall back to `string`.\n */\nexport interface Register {}\n\nexport type RegisteredRobotMap = Register extends { robots: infer T extends Record<string, Record<string, string>> }\n ? T\n : never;\nexport type Robots = [RegisteredRobotMap] extends [never] ? string : Extract<keyof RegisteredRobotMap, string>;\nexport type RobotResource<TRobot extends string, TKey extends string> =\n [RegisteredRobotMap] extends [never]\n ? string\n : TRobot extends keyof RegisteredRobotMap\n ? TKey extends keyof RegisteredRobotMap[TRobot]\n ? RegisteredRobotMap[TRobot][TKey]\n : string\n : never;\nexport type RobotActuators<TRobot extends string> = RobotResource<TRobot, 'actuators'>;\nexport type RobotSensors<TRobot extends string> = RobotResource<TRobot, 'sensors'>;\nexport type RobotBodies<TRobot extends string> = RobotResource<TRobot, 'bodies'>;\nexport type RobotJoints<TRobot extends string> = RobotResource<TRobot, 'joints'>;\nexport type RobotSites<TRobot extends string> = RobotResource<TRobot, 'sites'>;\nexport type RobotGeoms<TRobot extends string> = RobotResource<TRobot, 'geoms'>;\nexport type RobotKeyframes<TRobot extends string> = RobotResource<TRobot, 'keyframes'>;\nexport type RobotCameras<TRobot extends string> = RobotResource<TRobot, 'cameras'>;\n\nexport type RegisterResourceKey = 'actuators' | 'sensors' | 'bodies' | 'joints' | 'sites' | 'geoms' | 'keyframes' | 'cameras';\nexport type RobotResourceObject<TRobot extends string, TKey extends RegisterResourceKey> =\n string extends RobotResource<TRobot, TKey>\n ? Record<string, string>\n : { readonly [K in RobotResource<TRobot, TKey>]: K };\nexport type RobotResourceCategory<TKey extends RegisterResourceKey> =\n string extends Robots\n ? Record<string, Record<string, string>>\n : { readonly [TRobot in Robots]: RobotResourceObject<TRobot, TKey> };\nexport type RobotResourceRegistry =\n string extends Robots\n ? Record<string, Record<RegisterResourceKey, Record<string, string>>>\n : { readonly [TRobot in Robots]: { readonly [TKey in RegisterResourceKey]: RobotResourceObject<TRobot, TKey> } };\n\ntype RuntimeRobotResources = Record<string, Record<RegisterResourceKey, Record<string, string>>>;\ntype RuntimeRobotResourceRegistration = Readonly<Record<string, Readonly<Record<RegisterResourceKey, Readonly<Record<string, string>>>>>>;\n\nconst runtimeRobotResources: RuntimeRobotResources = {};\nconst REGISTER_RESOURCE_KEYS: RegisterResourceKey[] = ['actuators', 'sensors', 'bodies', 'joints', 'sites', 'geoms', 'keyframes', 'cameras'];\n\nfunction createEmptyRuntimeResources(): Record<RegisterResourceKey, Record<string, string>> {\n return {\n actuators: {},\n sensors: {},\n bodies: {},\n joints: {},\n sites: {},\n geoms: {},\n keyframes: {},\n cameras: {},\n };\n}\n\nexport function registerRobotResources(resources: RuntimeRobotResourceRegistration): void {\n for (const [robot, robotResources] of Object.entries(resources)) {\n const existing = runtimeRobotResources[robot] ?? createEmptyRuntimeResources();\n for (const key of REGISTER_RESOURCE_KEYS) {\n existing[key] = { ...existing[key], ...(robotResources[key] ?? {}) };\n }\n runtimeRobotResources[robot] = existing;\n }\n}\n\nfunction createResourceCategory<TKey extends RegisterResourceKey>(key: TKey): RobotResourceCategory<TKey> {\n return new Proxy({}, {\n get(_target, robot) {\n if (typeof robot !== 'string') return undefined;\n return runtimeRobotResources[robot]?.[key] ?? {};\n },\n ownKeys() {\n return Reflect.ownKeys(runtimeRobotResources);\n },\n getOwnPropertyDescriptor(_target, robot) {\n if (typeof robot !== 'string' || !(robot in runtimeRobotResources)) return undefined;\n return { enumerable: true, configurable: true };\n },\n }) as RobotResourceCategory<TKey>;\n}\n\nexport const RobotResources: RobotResourceRegistry = new Proxy(runtimeRobotResources, {\n get(target, robot) {\n if (typeof robot !== 'string') return undefined;\n return target[robot] ?? createEmptyRuntimeResources();\n },\n ownKeys(target) {\n return Reflect.ownKeys(target);\n },\n getOwnPropertyDescriptor(target, robot) {\n if (typeof robot !== 'string' || !(robot in target)) return undefined;\n return { enumerable: true, configurable: true };\n },\n}) as RobotResourceRegistry;\n\nexport const RobotActuators: RobotResourceCategory<'actuators'> = createResourceCategory('actuators');\nexport const RobotSensors: RobotResourceCategory<'sensors'> = createResourceCategory('sensors');\nexport const RobotBodies: RobotResourceCategory<'bodies'> = createResourceCategory('bodies');\nexport const RobotJoints: RobotResourceCategory<'joints'> = createResourceCategory('joints');\nexport const RobotSites: RobotResourceCategory<'sites'> = createResourceCategory('sites');\nexport const RobotGeoms: RobotResourceCategory<'geoms'> = createResourceCategory('geoms');\nexport const RobotKeyframes: RobotResourceCategory<'keyframes'> = createResourceCategory('keyframes');\nexport const RobotCameras: RobotResourceCategory<'cameras'> = createResourceCategory('cameras');\n\nexport type Actuators = Register extends { actuators: infer T extends string } ? T : string;\nexport type Sensors = Register extends { sensors: infer T extends string } ? T : string;\nexport type Bodies = Register extends { bodies: infer T extends string } ? T : string;\nexport type Joints = Register extends { joints: infer T extends string } ? T : string;\nexport type Sites = Register extends { sites: infer T extends string } ? T : string;\nexport type Geoms = Register extends { geoms: infer T extends string } ? T : string;\nexport type Keyframes = Register extends { keyframes: infer T extends string } ? T : string;\nexport type Cameras = Register extends { cameras: infer T extends string } ? T : string;\n\n// ---- MuJoCo WASM Types ----\n\n/**\n * A single MuJoCo contact from the WASM module.\n * Accessed via `data.contact.get(i)`.\n */\nexport interface MujocoContact {\n geom1: number;\n geom2: number;\n pos: Float64Array;\n frame: Float64Array;\n dist: number;\n}\n\n/**\n * WASM contact array — supports indexed access via `.get(i)`.\n */\nexport interface MujocoContactArray {\n get(i: number): MujocoContact | undefined;\n delete?: () => void;\n}\n\n/**\n * Read a single contact from an already-acquired WASM contact array.\n * Returns undefined if the access fails (WASM heap issue, bad index, etc.).\n */\nexport function getContact(contacts: MujocoContactArray, i: number): MujocoContact | undefined {\n try {\n return contacts.get(i);\n } catch {\n return undefined;\n }\n}\n\n/**\n * Access the current contact vector and release the copied WASM handle afterwards.\n */\nexport function withContacts<T>(data: MujocoData, read: (contacts: MujocoContactArray) => T): T {\n const contacts = data.contact;\n try {\n return read(contacts);\n } finally {\n contacts.delete?.();\n }\n}\n\n/**\n * Minimal interface for MuJoCo Model to avoid 'any'.\n */\nexport interface MujocoModel {\n // Counts\n nbody: number;\n ngeom: number;\n nsite: number;\n nu: number;\n njnt: number;\n nq: number;\n nv: number;\n nkey: number;\n nsensor: number;\n nsensordata: number;\n nlight: number;\n ntendon: number;\n nflex: number;\n nmesh: number;\n nmat: number;\n ncam?: number;\n\n // Name tables\n names: Int8Array;\n name_bodyadr: Int32Array;\n name_jntadr: Int32Array;\n name_geomadr: Int32Array;\n name_siteadr: Int32Array;\n name_actuatoradr: Int32Array;\n name_keyadr: Int32Array;\n name_sensoradr: Int32Array;\n name_tendonadr: Int32Array;\n name_camadr?: Int32Array;\n\n // Body\n body_mass: Float64Array;\n body_parentid: Int32Array;\n body_jntnum: Int32Array;\n body_jntadr: Int32Array;\n body_pos: Float64Array;\n body_quat: Float64Array;\n body_geomnum: Int32Array;\n body_geomadr: Int32Array;\n body_inertia: Float64Array;\n\n // Default configuration\n qpos0: Float64Array;\n\n // Joint\n jnt_qposadr: Int32Array;\n jnt_dofadr: Int32Array;\n jnt_type: Int32Array;\n jnt_range: Float64Array;\n jnt_bodyid: Int32Array;\n jnt_pos: Float64Array;\n jnt_axis: Float64Array;\n jnt_limited: Uint8Array;\n\n // Geom\n geom_group: Int32Array;\n geom_type: Int32Array;\n geom_size: Float64Array;\n geom_pos: Float64Array;\n geom_quat: Float64Array;\n geom_matid: Int32Array;\n geom_rgba: Float32Array;\n geom_dataid: Int32Array;\n geom_bodyid: Int32Array;\n geom_contype: Int32Array;\n geom_conaffinity: Int32Array;\n geom_friction: Float64Array;\n\n // Material\n mat_rgba: Float32Array;\n\n // Mesh\n mesh_vertadr: Int32Array;\n mesh_vertnum: Int32Array;\n mesh_faceadr: Int32Array;\n mesh_facenum: Int32Array;\n mesh_vert: Float32Array;\n mesh_face: Int32Array;\n mesh_normal: Float32Array;\n\n // Site\n site_bodyid: Int32Array;\n\n // Actuator\n actuator_trnid: Int32Array;\n actuator_ctrlrange: Float64Array;\n actuator_trntype: Int32Array;\n actuator_gainprm: Float64Array;\n actuator_biasprm: Float64Array;\n\n // Sensor\n sensor_type: Int32Array;\n sensor_dim: Int32Array;\n sensor_adr: Int32Array;\n sensor_objtype: Int32Array;\n sensor_objid: Int32Array;\n\n // Keyframe\n key_qpos: Float64Array;\n key_ctrl: Float64Array;\n key_time: Float64Array;\n key_qvel: Float64Array;\n\n // Light\n light_pos: Float64Array;\n light_dir: Float64Array;\n light_diffuse: Float32Array;\n light_specular: Float32Array;\n light_type: Int32Array;\n light_active: Uint8Array;\n light_castshadow: Uint8Array;\n light_attenuation: Float32Array;\n light_cutoff: Float32Array;\n light_exponent: Float32Array;\n light_intensity: Float32Array;\n\n // Camera\n cam_bodyid?: Int32Array;\n cam_pos?: Float64Array;\n cam_quat?: Float64Array;\n cam_fovy?: Float64Array;\n\n // Tendon\n ten_wrapadr: Int32Array;\n ten_wrapnum: Int32Array;\n ten_range: Float64Array;\n ten_rgba: Float32Array;\n ten_width: Float64Array;\n\n // Flex\n flex_vertadr: Int32Array;\n flex_vertnum: Int32Array;\n flex_faceadr: Int32Array;\n flex_facenum: Int32Array;\n flex_face: Int32Array;\n flex_rgba: Float32Array;\n\n // Model options\n opt: {\n timestep: number;\n gravity: Float64Array;\n integrator: number;\n [key: string]: unknown;\n };\n\n delete: () => void;\n [key: string]: unknown;\n}\n\n/**\n * Minimal interface for MuJoCo Data to avoid 'any'.\n */\nexport interface MujocoData {\n time: number;\n qpos: Float64Array;\n qvel: Float64Array;\n ctrl: Float64Array;\n act: Float64Array;\n xpos: Float64Array;\n xquat: Float64Array;\n xfrc_applied: Float64Array;\n qfrc_applied: Float64Array;\n qfrc_bias: Float64Array;\n site_xpos: Float64Array;\n site_xmat: Float64Array;\n cam_xpos?: Float64Array;\n cam_xmat?: Float64Array;\n xmat?: Float64Array;\n sensordata: Float64Array;\n ncon: number;\n contact: MujocoContactArray;\n cvel: Float64Array;\n cfrc_ext: Float64Array;\n ten_length: Float64Array;\n wrap_xpos: Float64Array;\n ten_wrapadr: Int32Array;\n flexvert_xpos: Float64Array;\n geom_xpos: Float64Array;\n geom_xmat: Float64Array;\n delete: () => void;\n [key: string]: unknown;\n}\n\n/**\n * Minimal interface for the MuJoCo WASM Module.\n */\nexport interface MujocoModule {\n MjModel: {\n from_xml_path?: (path: string) => MujocoModel;\n from_xml_string?: (xml: string, vfs?: unknown) => MujocoModel;\n loadFromXML?: (path: string) => MujocoModel;\n [key: string]: unknown;\n };\n MjData: new (model: MujocoModel) => MujocoData;\n MjvOption: new () => { delete: () => void; [key: string]: unknown };\n mj_forward: (m: MujocoModel, d: MujocoData) => void;\n mj_step: (m: MujocoModel, d: MujocoData) => void;\n mj_resetData: (m: MujocoModel, d: MujocoData) => void;\n mj_step1: (m: MujocoModel, d: MujocoData) => void;\n mj_step2: (m: MujocoModel, d: MujocoData) => void;\n mj_applyFT: (\n model: MujocoModel,\n data: MujocoData,\n force: Float64Array,\n torque: Float64Array,\n point: Float64Array,\n bodyId: number,\n qfrc_target: Float64Array\n ) => void;\n mj_ray: (\n model: MujocoModel,\n data: MujocoData,\n pnt: Float64Array,\n vec: Float64Array,\n geomgroup: Uint8Array | null,\n flg_static: number,\n bodyexclude: number,\n geomid: Int32Array\n ) => number;\n mj_name2id: (model: MujocoModel, type: number, name: string) => number;\n mjtObj: Record<string, number>;\n mjtGeom: Record<string, number | {value: number}>;\n mjtJoint: Record<string, number | {value: number}>;\n mjtSensor: Record<string, number | {value: number}>;\n FS: {\n writeFile: (path: string, content: string | Uint8Array) => void;\n readFile: (path: string, opts?: { encoding: string }) => string | Uint8Array;\n mkdir: (path: string) => void;\n unmount: (path: string) => void;\n };\n [key: string]: unknown;\n}\n\n// ---- Scene Configuration ----\n\nexport interface SceneObject {\n name: string;\n type: 'box' | 'sphere' | 'cylinder';\n size: [number, number, number];\n position: [number, number, number];\n rgba: [number, number, number, number];\n mass?: number;\n freejoint?: boolean;\n friction?: string;\n solref?: string;\n solimp?: string;\n condim?: number;\n /** MuJoCo geom group. Group 3 is conventionally used for collision-only helper geoms. */\n group?: number;\n}\n\nexport interface XmlPatch {\n target: string;\n inject?: string;\n injectAfter?: string;\n replace?: [string, string];\n}\n\nexport type LocalMujocoFile = File;\n\nexport interface LoadFromFilesOptions {\n /** Entry MJCF/URDF file. Inferred from scene.xml, model.xml, robot.xml, or the first XML/URDF file when omitted. */\n sceneFile?: string;\n /** Additional MJCF environment XML files merged into the entry scene before MuJoCo compilation. */\n environmentFiles?: string[];\n homeJoints?: number[];\n xmlPatches?: XmlPatch[];\n sceneObjects?: SceneObject[];\n onReset?: (input: ResetCallbackInput) => void;\n}\n\nexport interface SceneConfig {\n /** Base URL for fetching model files. The loader fetches `src + sceneFile` and follows dependencies. */\n src: string;\n /** Entry MJCF XML or URDF file name, e.g. 'scene.xml' or 'robot.urdf'. */\n sceneFile: string;\n /** Browser-selected files for local MJCF/URDF loading. Preserves webkitRelativePath when available. */\n files?: readonly LocalMujocoFile[];\n /**\n * Additional MJCF environment XML files merged into the entry scene before compilation.\n *\n * Use this for static collision/physics layers such as a Gaussian-splat\n * environment's proxy `scene.xml`; render the splat itself as a separate\n * visual layer.\n */\n environmentFiles?: string[];\n sceneObjects?: SceneObject[];\n homeJoints?: number[];\n xmlPatches?: XmlPatch[];\n onReset?: (input: ResetCallbackInput) => void;\n}\n\n// ---- IK Controller Config ----\n\nexport type ResourceSelector<TInfo, TName extends string = string> =\n | TName\n | readonly TName[]\n | RegExp\n | ((info: TInfo) => boolean);\n\nexport interface IkConfig {\n /** MuJoCo site name for IK target. */\n siteName: Sites;\n /**\n * Explicit joints for IK. When omitted, the controller infers scalar hinge/slide\n * joints by walking from the site body to the model root.\n */\n joints?: ResourceSelector<JointInfo, Joints>;\n /** Explicit actuators for IK control output. */\n actuators?: ResourceSelector<ActuatorInfo, Actuators>;\n /**\n * Number of joints to solve for, assuming legacy contiguous qpos/ctrl layout\n * starting at index 0. Prefer inferred IK or `joints`/`actuators`.\n */\n numJoints?: number;\n /** Custom IK solver. When omitted, uses built-in Damped Least-Squares solver. */\n ikSolveFn?: IKSolveFn;\n /** DLS damping. Default: 0.01. */\n damping?: number;\n /** Max solver iterations. Default: 50. */\n maxIterations?: number;\n}\n\nexport interface IkContextValue {\n ikEnabledRef: React.RefObject<boolean>;\n ikCalculatingRef: React.RefObject<boolean>;\n ikTargetRef: React.RefObject<THREE.Group>;\n siteIdRef: React.RefObject<number>;\n setIkEnabled: (enabled: boolean) => void;\n moveTarget: (pos: THREE.Vector3, duration?: number) => void;\n syncTargetToSite: () => void;\n solveIK: (input: IkSolveInput) => number[] | null;\n getGizmoStats: () => { pos: THREE.Vector3; rot: THREE.Euler } | null;\n}\n\nexport interface SceneMarker {\n id: number;\n position: THREE.Vector3;\n label: string;\n}\n\n// ---- Physics Config (spec 1.1) ----\n\nexport interface PhysicsConfig {\n gravity?: [number, number, number];\n timestep?: number;\n substeps?: number;\n paused?: boolean;\n speed?: number;\n}\n\n// ---- IK ----\n\nexport type IKSolveFn = (\n input: IkSolveInput\n) => number[] | null;\n\nexport interface IkSolveInput {\n position: THREE.Vector3;\n quaternion: THREE.Quaternion;\n currentQ: number[];\n context?: IKSolveContext;\n}\n\nexport interface IKSolveContext {\n model: MujocoModel;\n data: MujocoData;\n siteId: number;\n controlGroup: ControlGroupInfo;\n}\n\n// ---- Callbacks ----\n\nexport interface PhysicsStepInput {\n model: MujocoModel;\n data: MujocoData;\n}\n\nexport interface ResetCallbackInput extends PhysicsStepInput {}\n\nexport interface ReadyCallbackInput {\n api: MujocoSimAPI;\n}\n\nexport interface StepCallbackInput {\n time: number;\n model: MujocoModel;\n data: MujocoData;\n}\n\nexport interface SelectionCallbackInput {\n bodyId: number;\n name: string;\n}\n\nexport type PhysicsStepCallback = (input: PhysicsStepInput) => void;\n\n// ---- State Management (spec 4.1) ----\n\nexport interface StateSnapshot {\n time: number;\n qpos: Float64Array;\n qvel: Float64Array;\n ctrl: Float64Array;\n act: Float64Array;\n qfrc_applied: Float64Array;\n}\n\n// ---- Model Introspection (spec 5.1) ----\n\nexport interface BodyInfo {\n id: number;\n name: string;\n mass: number;\n parentId: number;\n}\n\nexport interface JointInfo {\n id: number;\n name: string;\n type: number;\n typeName: string;\n range: [number, number];\n limited: boolean;\n bodyId: number;\n qposAdr: number;\n dofAdr: number;\n}\n\nexport interface GeomInfo {\n id: number;\n name: string;\n type: number;\n typeName: string;\n size: [number, number, number];\n bodyId: number;\n}\n\nexport interface SiteInfo {\n id: number;\n name: string;\n bodyId: number;\n}\n\nexport interface ActuatorInfo {\n id: number;\n name: string;\n range: [number, number];\n}\n\nexport interface ActuatedJointInfo extends JointInfo {\n actuatorId: number;\n actuatorName: string;\n ctrlAdr: number;\n ctrlRange: [number, number];\n}\n\nexport interface ControlJointInfo extends JointInfo {\n actuatorId: number | null;\n actuatorName: string | null;\n ctrlAdr: number | null;\n ctrlRange: [number, number] | null;\n}\n\nexport interface ControlGroupSelector {\n /** Infer a kinematic chain from a MuJoCo site. */\n siteName?: Sites;\n /** Infer a kinematic chain from a body. */\n bodyName?: Bodies;\n /** Select joints by name, names, regex, or predicate. */\n joints?: ResourceSelector<JointInfo, Joints>;\n /** Select actuators by name, names, regex, or predicate. */\n actuators?: ResourceSelector<ActuatorInfo, Actuators>;\n}\n\nexport interface ControlGroupInfo {\n /** Joints in solve/control order. */\n joints: ControlJointInfo[];\n /** Actuators in control output order. */\n actuators: ActuatorInfo[];\n /** qpos addresses for scalar hinge/slide joints. */\n qposAdr: number[];\n /** dof addresses for scalar hinge/slide joints. */\n dofAdr: number[];\n /** ctrl addresses matching writable actuators. */\n ctrlAdr: number[];\n readQpos(data: MujocoData): Float64Array;\n readCtrl(data: MujocoData): Float64Array;\n writeQpos(data: MujocoData, values: ArrayLike<number>): void;\n writeCtrl(data: MujocoData, values: ArrayLike<number>): void;\n}\n\nexport interface SensorInfo {\n id: number;\n name: string;\n type: number;\n typeName: string;\n dim: number;\n adr: number;\n}\n\nexport interface CameraInfo {\n id: number;\n name: string;\n bodyId: number;\n fov: number | null;\n position: [number, number, number] | null;\n quaternion: [number, number, number, number] | null;\n}\n\n// ---- Contacts (spec 2.4, 2.5) ----\n\nexport interface ContactInfo {\n geom1: number;\n geom1Name: string;\n geom2: number;\n geom2Name: string;\n pos: [number, number, number];\n depth: number;\n}\n\n// ---- Raycast (spec 7.1) ----\n\nexport interface RayHit {\n point: THREE.Vector3;\n bodyId: number;\n geomId: number;\n distance: number;\n}\n\n// ---- Model Options (spec 5.3) ----\n\nexport interface ModelOptions {\n timestep: number;\n gravity: [number, number, number];\n integrator: number;\n}\n\n// ---- Trajectory (spec 13.1, 13.2) ----\n\nexport interface TrajectoryFrame {\n time: number;\n qpos: Float64Array;\n qvel?: Float64Array;\n ctrl?: Float64Array;\n sensordata?: Float64Array;\n}\n\nexport interface TrajectoryData {\n frames: TrajectoryFrame[];\n fps: number;\n}\n\nexport type PlaybackState = 'idle' | 'playing' | 'paused' | 'completed';\n\n// ---- Keyboard Teleop (spec 12.1) ----\n\nexport interface KeyBinding {\n actuator: Actuators;\n delta?: number;\n toggle?: [number, number];\n set?: number;\n}\n\nexport interface KeyboardTeleopConfig {\n bindings: Record<string, KeyBinding>;\n enabled?: boolean;\n}\n\n// ---- Policy (spec 10.1) ----\n\nexport type PolicyVector = Float32Array | Float64Array | number[];\n\nexport interface PolicyObservationInput {\n model: MujocoModel;\n data: MujocoData;\n}\n\nexport interface PolicyInferenceInput extends PolicyObservationInput {\n observation: PolicyVector;\n}\n\nexport interface PolicyActionInput extends PolicyInferenceInput {\n action: PolicyVector;\n}\n\nexport interface PolicyConfig {\n frequency: number;\n enabled?: boolean;\n onObservation: (input: PolicyObservationInput) => PolicyVector;\n /** Run policy inference. Omit to pass observations directly to `onAction` for custom inline controllers. */\n infer?: (input: PolicyInferenceInput) => PolicyVector;\n onAction: (input: PolicyActionInput) => void;\n}\n\n// ---- Observation Builder ----\n\nexport type ObservationOutput = 'float32' | 'float64';\n\nexport interface ObservationConfig {\n /** Include scalar simulation time. */\n time?: boolean;\n /** Include all qpos values. */\n qpos?: boolean;\n /** Include all qvel values. */\n qvel?: boolean;\n /** Include all ctrl values. */\n ctrl?: boolean;\n /** Include all actuator activation values. */\n act?: boolean;\n /** Include all raw sensordata values. */\n sensordata?: boolean;\n /** Include named sensor values in the configured order. */\n sensors?: readonly Sensors[];\n /** Include named site world positions in the configured order. */\n sites?: readonly Sites[];\n /** Include world gravity projected into each named body's local frame. */\n projectedGravity?: Bodies | readonly Bodies[];\n /** Output array type. Defaults to Float32Array. */\n output?: ObservationOutput;\n}\n\nexport interface ObservationLayoutItem {\n name: string;\n start: number;\n size: number;\n}\n\nexport interface ObservationResult {\n values: Float32Array | Float64Array;\n layout: ObservationLayoutItem[];\n}\n\nexport interface ObservationHandle {\n /** Read a fresh observation from the current live MuJoCo model/data refs. */\n read(): ObservationResult;\n /** Read just the vector values for policy inference. */\n readValues(): Float32Array | Float64Array;\n}\n\n// ---- Debug Component (spec 6.1) ----\n\nexport interface DebugProps {\n showGeoms?: boolean;\n showSites?: boolean;\n showJoints?: boolean;\n showContacts?: boolean;\n showCOM?: boolean;\n showInertia?: boolean;\n showTendons?: boolean;\n}\n\n// ---- Component Props ----\n\nexport interface IkGizmoProps {\n controller: IkContextValue;\n siteName?: string;\n scale?: number;\n onDrag?: (input: IkGizmoDragInput) => void;\n}\n\nexport interface IkGizmoDragInput {\n position: THREE.Vector3;\n quaternion: THREE.Quaternion;\n}\n\nexport interface DragInteractionProps {\n stiffness?: number;\n showArrow?: boolean;\n}\n\nexport interface SceneLightsProps {\n /** Override intensity for all MJCF lights. Default: 1.0. */\n intensity?: number;\n}\n\n// ---- Visual scenarios / 3DGS composition ----\n\nexport type ScenarioLightingPreset = 'studio' | 'warehouse' | 'low-light' | 'splat';\nexport type SplatFormat = 'spz' | 'ply' | 'splat';\nexport type SplatRendererKind = 'spark' | 'custom';\nexport type SplatCollisionPrimitive = 'plane' | 'box' | 'sphere' | 'capsule' | 'mesh';\n\nexport interface ScenarioCameraConfig {\n jitter?: number;\n exposure?: number;\n noise?: number;\n blur?: number;\n}\n\nexport interface ScenarioMaterialConfig {\n randomizeObjectColors?: boolean;\n randomizeTableMaterial?: boolean;\n roughness?: number;\n metalness?: number;\n}\n\nexport interface SplatAssetConfig {\n src: string;\n /** Common browser-friendly splat format. Renderer-specific loaders may accept more. */\n format?: SplatFormat;\n /** Optional renderer hint. The library does not import renderer-specific code. */\n renderer?: SplatRendererKind;\n}\n\nexport interface SplatScenarioConfig {\n enabled: boolean;\n /** Common browser-friendly splat format. Renderer-specific loaders may accept more. */\n format?: SplatFormat;\n src?: string;\n requiresCollisionProxy?: boolean;\n collisionProxy?: SplatCollisionProxyConfig | null;\n}\n\nexport interface SplatCollisionProxyConfig {\n /** MJCF/XML file or artifact path that provides physics collision for the visual splat. */\n xmlPath?: string;\n /** Human-readable status for authoring and validation flows. */\n status?: 'missing' | 'planned' | 'generated' | 'validated';\n /** Primitive proxy shapes expected in the MJCF collision proxy. */\n primitives?: SplatCollisionPrimitive[];\n /** Optional notes that should travel with scene variants and rollout metadata. */\n notes?: string[];\n}\n\nexport interface PairedSplatEnvironmentConfig {\n id: string;\n label: string;\n description?: string;\n /** Visual-only Gaussian splat asset. */\n splat: SplatAssetConfig;\n /** MJCF/XML contact geometry paired with the visual splat. */\n collisionProxy: SplatCollisionProxyConfig & { xmlPath: string };\n}\n\nexport const SplatEnvironmentReadinessStatus = {\n Disabled: 'disabled',\n MissingSplat: 'missing-splat',\n MissingCollisionProxy: 'missing-collision-proxy',\n UnsupportedFormat: 'unsupported-format',\n Ready: 'ready',\n} as const;\n\nexport type SplatEnvironmentReadinessStatus =\n (typeof SplatEnvironmentReadinessStatus)[keyof typeof SplatEnvironmentReadinessStatus];\n\nexport interface SplatEnvironmentReadiness {\n status: SplatEnvironmentReadinessStatus;\n ready: boolean;\n requiresCollisionProxy: boolean;\n missing: Array<'splat' | 'collisionProxy'>;\n format?: SplatFormat;\n renderer?: SplatRendererKind;\n message: string;\n}\n\nexport interface SplatEnvironmentMetadataInput {\n environment?: PairedSplatEnvironmentConfig;\n scenario?: VisualScenarioConfig;\n renderer?: SplatRendererKind;\n src?: string;\n format?: SplatFormat;\n collisionProxy?: SplatCollisionProxyConfig;\n}\n\nexport interface SplatEnvironmentMetadata {\n src?: string;\n format: SplatFormat;\n collisionProxy?: SplatCollisionProxyConfig;\n readiness: SplatEnvironmentReadiness;\n userData: Record<string, unknown>;\n}\n\nexport type SplatSceneInput =\n | PairedSplatEnvironmentConfig\n | VisualScenarioConfig\n | undefined\n | null;\n\nexport interface SplatSceneConfigInput {\n sceneConfig: SceneConfig;\n scenario?: VisualScenarioConfig;\n environment?: PairedSplatEnvironmentConfig;\n enabled?: boolean;\n renderer?: SplatRendererKind;\n}\n\nexport interface SplatSceneConfigState {\n environment: PairedSplatEnvironmentConfig | undefined;\n sceneConfig: SceneConfig;\n enabled: boolean;\n readiness: SplatEnvironmentReadiness;\n}\n\nexport interface VisualScenarioConfig {\n id?: string;\n label?: string;\n seed?: number;\n lighting?: ScenarioLightingPreset;\n environment?: string;\n camera?: ScenarioCameraConfig;\n materials?: ScenarioMaterialConfig;\n splat?: SplatScenarioConfig | null;\n}\n\nexport interface ScenarioLightingProps {\n preset?: ScenarioLightingPreset;\n intensity?: number;\n castShadow?: boolean;\n}\n\nexport interface SplatEnvironmentProps extends Omit<ThreeElements['group'], 'ref'> {\n environment?: PairedSplatEnvironmentConfig;\n scenario?: VisualScenarioConfig;\n renderer?: SplatRendererKind;\n src?: string;\n format?: SplatFormat;\n collisionProxy?: ReactNode;\n collisionProxyMetadata?: SplatCollisionProxyConfig;\n showPlaceholder?: boolean;\n}\n\nexport interface VisualScenarioEffectsProps {\n scenario?: VisualScenarioConfig;\n enabled?: boolean;\n applyBackground?: boolean;\n applyFog?: boolean;\n applyRenderer?: boolean;\n applyMaterials?: boolean;\n background?: THREE.ColorRepresentation;\n fogNear?: number;\n fogFar?: number;\n materialFilter?: (input: VisualScenarioMaterialFilterInput) => boolean;\n}\n\nexport interface VisualScenarioMaterialFilterInput {\n object: THREE.Object3D;\n material: THREE.Material;\n}\n\nexport type TrajectoryInput = TrajectoryFrame[] | number[][];\n\nexport interface TrajectoryPlayerProps {\n trajectory: TrajectoryInput;\n fps?: number;\n speed?: number;\n loop?: boolean;\n playing?: boolean;\n mode?: 'kinematic' | 'physics';\n onFrame?: (input: TrajectoryFrameCallbackInput) => void;\n onComplete?: () => void;\n onStateChange?: (input: TrajectoryStateChangeInput) => void;\n}\n\nexport interface TrajectoryFrameCallbackInput {\n frameIndex: number;\n frame: TrajectoryFrame | number[] | undefined;\n}\n\nexport interface TrajectoryStateChangeInput {\n state: PlaybackState;\n}\n\nexport interface SelectionHighlightProps {\n bodyId: number | null;\n color?: string;\n emissiveIntensity?: number;\n}\n\nexport interface ContactListenerProps {\n body: Bodies;\n onContactEnter?: (info: ContactInfo) => void;\n onContactExit?: (info: ContactInfo) => void;\n}\n\nexport interface BodyProps {\n name: Bodies;\n type: 'box' | 'sphere' | 'cylinder';\n size: [number, number, number];\n position?: [number, number, number];\n rgba?: [number, number, number, number];\n mass?: number;\n freejoint?: boolean;\n friction?: string;\n solref?: string;\n solimp?: string;\n condim?: number;\n /** MuJoCo geom group. Group 3 is conventionally used for collision-only helper geoms. */\n group?: number;\n children?: ReactNode;\n}\n\n// ---- Public API (spec: full surface) ----\n\nexport interface MujocoSimAPI {\n // State\n readonly status: 'loading' | 'ready' | 'error';\n readonly config: SceneConfig;\n\n // Simulation control (spec 1.1, 1.2, 1.3)\n reset(): void;\n setSpeed(multiplier: number): void;\n togglePause(): boolean;\n setPaused(paused: boolean): void;\n step(n?: number): void;\n getTime(): number;\n getTimestep(): number;\n applyKeyframe(nameOrIndex: Keyframes | number): void;\n\n // State management (spec 4.1, 4.2, 4.3)\n saveState(): StateSnapshot;\n restoreState(snapshot: StateSnapshot): void;\n setQpos(values: Float64Array | number[]): void;\n setQvel(values: Float64Array | number[]): void;\n getQpos(): Float64Array;\n getQvel(): Float64Array;\n\n // Actuator / control (spec 3.1)\n setCtrl(nameOrValues: Actuators | Record<Actuators, number>, value?: number): void;\n getCtrl(): Float64Array;\n getControlMap(): ControlGroupInfo;\n getActuatedJoints(): ActuatedJointInfo[];\n resolveControlGroup(selector: ControlGroupSelector): ControlGroupInfo | null;\n\n // Force application (spec 8.1)\n applyForce(bodyName: Bodies, force: THREE.Vector3, point?: THREE.Vector3): void;\n applyTorque(bodyName: Bodies, torque: THREE.Vector3): void;\n setExternalForce(bodyName: Bodies, force: THREE.Vector3, torque: THREE.Vector3): void;\n applyGeneralizedForce(values: Float64Array | number[]): void;\n\n // Sensors (spec 2.1)\n getSensorData(name: Sensors): Float64Array | null;\n\n // Contacts (spec 2.4)\n getContacts(): ContactInfo[];\n\n // Model introspection (spec 5.1, 5.2)\n getBodies(): BodyInfo[];\n getJoints(): JointInfo[];\n getGeoms(): GeomInfo[];\n getSites(): SiteInfo[];\n getActuators(): ActuatorInfo[];\n getSensors(): SensorInfo[];\n getCameras(): CameraInfo[];\n\n // Model parameters (spec 5.3)\n getModelOption(): ModelOptions;\n setGravity(g: [number, number, number]): void;\n setTimestep(dt: number): void;\n\n // Raycasting (spec 7.1)\n raycast(origin: THREE.Vector3, direction: THREE.Vector3, maxDist?: number): RayHit | null;\n\n // Keyframes (spec 4.2)\n getKeyframeNames(): string[];\n getKeyframeCount(): number;\n\n // Model loading (spec 9.1)\n loadScene(newConfig: SceneConfig): Promise<void>;\n loadFromFiles(files: FileList | readonly LocalMujocoFile[], options?: LoadFromFilesOptions): Promise<void>;\n addBody(body: SceneObject): Promise<void>;\n removeBody(name: Bodies): Promise<void>;\n recompile(patches?: XmlPatch[]): Promise<void>;\n\n // Canvas\n getCanvas(): HTMLCanvasElement | null;\n getCanvasSnapshot(width?: number, height?: number, mimeType?: string): string;\n captureFrame(options?: MujocoFrameCaptureOptions): Promise<FrameCaptureResult>;\n captureFrameBlob(options?: MujocoFrameCaptureOptions): Promise<FrameCaptureBlobResult>;\n captureCameraFrame(options?: CameraFrameCaptureOptions): Promise<CameraFrameCaptureResult>;\n captureCameraFrameBlob(options?: CameraFrameCaptureOptions): Promise<CameraFrameCaptureBlobResult>;\n recordCameraSequence(options: CameraFrameSequenceOptions): Promise<CameraFrameSequenceResult>;\n project2DTo3D(\n x: number,\n y: number,\n cameraPos: THREE.Vector3,\n lookAt: THREE.Vector3\n ): { point: THREE.Vector3; bodyId: number; geomId: number } | null;\n\n // Domain randomization (spec 10.3)\n setBodyMass(name: Bodies, mass: number): void;\n setGeomFriction(name: Geoms, friction: [number, number, number]): void;\n setGeomSize(name: Geoms, size: [number, number, number]): void;\n\n // Internal refs for advanced use\n readonly mjModelRef: React.RefObject<MujocoModel | null>;\n readonly mjDataRef: React.RefObject<MujocoData | null>;\n}\n\nexport type FrameCaptureStatus = 'idle' | 'capturing' | 'captured' | 'error';\n\nexport type FrameCaptureTarget =\n | HTMLCanvasElement\n | HTMLElement\n | null\n | undefined;\n\nexport type FrameCaptureTargetRef =\n React.RefObject<HTMLCanvasElement | HTMLElement | null>;\n\nexport interface FrameCaptureOptions {\n target?: FrameCaptureTarget | FrameCaptureTargetRef;\n type?: string;\n quality?: number;\n waitForAnimationFrame?: boolean;\n}\n\nexport type MujocoFrameCaptureOptions = Omit<FrameCaptureOptions, 'target'>;\n\nexport interface FrameCaptureResult {\n canvas: HTMLCanvasElement;\n dataUrl: string;\n type: string;\n}\n\nexport interface FrameCaptureBlobResult {\n canvas: HTMLCanvasElement;\n blob: Blob;\n type: string;\n}\n\nexport interface FrameCaptureAPI {\n status: FrameCaptureStatus;\n error: Error | null;\n isCapturing: boolean;\n capture: (options?: FrameCaptureOptions) => Promise<FrameCaptureResult>;\n captureBlob: (\n options?: FrameCaptureOptions\n ) => Promise<FrameCaptureBlobResult>;\n reset: () => void;\n}\n\nexport type CameraFrameCaptureVector3 =\n | THREE.Vector3\n | readonly [number, number, number];\n\nexport type CameraFrameCaptureQuaternion =\n | THREE.Quaternion\n | readonly [number, number, number, number];\n\nexport interface CameraFrameCaptureOptions {\n /** Existing Three camera to clone before applying pose overrides. */\n camera?: THREE.Camera;\n /** Named MuJoCo `<camera>` to render from when available in the loaded model. */\n cameraName?: Cameras;\n /** Named MuJoCo site to use as the rendered camera pose. Useful for robot-mounted optical frames. */\n siteName?: Sites;\n /** Named MuJoCo body to use as the rendered camera pose. */\n bodyName?: Bodies;\n position?: CameraFrameCaptureVector3;\n lookAt?: CameraFrameCaptureVector3;\n quaternion?: CameraFrameCaptureQuaternion;\n up?: CameraFrameCaptureVector3;\n width?: number;\n height?: number;\n type?: string;\n quality?: number;\n fov?: number;\n near?: number;\n far?: number;\n /** Provenance for the camera pose used by the capture. Usually set by the MuJoCo provider. */\n source?: CameraFrameCaptureSource;\n}\n\nexport type CameraFrameCaptureSource =\n | { kind: 'mujoco-camera'; cameraName: Cameras }\n | { kind: 'mujoco-site'; siteName: Sites }\n | { kind: 'mujoco-body'; bodyName: Bodies }\n | { kind: 'custom-camera' }\n | { kind: 'explicit-pose' }\n | { kind: 'fallback-camera' };\n\nexport interface CameraFrameCaptureResult {\n canvas: HTMLCanvasElement;\n camera: THREE.Camera;\n dataUrl: string;\n type: string;\n width: number;\n height: number;\n source: CameraFrameCaptureSource;\n}\n\nexport interface CameraFrameCaptureBlobResult {\n canvas: HTMLCanvasElement;\n camera: THREE.Camera;\n blob: Blob;\n type: string;\n width: number;\n height: number;\n source: CameraFrameCaptureSource;\n}\n\nexport interface CameraFrameCaptureAPI {\n status: FrameCaptureStatus;\n error: Error | null;\n isCapturing: boolean;\n capture: (\n options?: CameraFrameCaptureOptions\n ) => Promise<CameraFrameCaptureResult>;\n captureBlob: (\n options?: CameraFrameCaptureOptions\n ) => Promise<CameraFrameCaptureBlobResult>;\n reset: () => void;\n}\n\nexport interface CameraFrameSequenceCamera extends CameraFrameCaptureOptions {\n key: string;\n}\n\nexport interface CameraFrameSequenceFrame {\n frameIndex: number;\n time: number;\n cameras: Record<string, CameraFrameCaptureResult>;\n}\n\nexport interface CameraFrameSequenceCameraSummary {\n key: string;\n width: number;\n height: number;\n source: CameraFrameCaptureSource;\n frameCount: number;\n firstFrameIndex: number | null;\n lastFrameIndex: number | null;\n firstTimestamp: number | null;\n lastTimestamp: number | null;\n}\n\nexport interface CameraFrameSequenceSampleInput extends PhysicsStepInput {\n frameIndex: number;\n time: number;\n}\n\nexport interface CameraFrameSequenceStepInput extends PhysicsStepInput {\n frameIndex: number;\n stepIndex: number;\n time: number;\n}\n\nexport interface CameraFrameSequenceOptions {\n cameras: readonly CameraFrameSequenceCamera[];\n frames: number;\n /** Number of MuJoCo steps between captured frames. Use 0 for static camera provenance captures. */\n stepsPerFrame?: number;\n reset?: boolean;\n captureInitialFrame?: boolean;\n retainFrames?: boolean;\n /**\n * Require each recorded stream to resolve from exactly one mounted MuJoCo\n * camera/site/body selector. Defaults to true because sequence recording is\n * intended for dataset/policy camera streams.\n */\n requireMountedSources?: boolean;\n signal?: AbortSignal;\n /** Called after stepping and before image capture for this frame. Use this to record synchronized state/action rows. */\n onSample?: (input: CameraFrameSequenceSampleInput) => void | Promise<void>;\n /** Called before each MuJoCo step inside sequence recording. Use this to apply policy/control actions. */\n onBeforeStep?: (input: CameraFrameSequenceStepInput) => void | Promise<void>;\n /** Called after each MuJoCo step inside sequence recording. Use this for step-level telemetry. */\n onAfterStep?: (input: CameraFrameSequenceStepInput) => void | Promise<void>;\n onFrame?: (frame: CameraFrameSequenceFrame) => void | Promise<void>;\n}\n\nexport interface CameraFrameSequenceResult {\n frames: CameraFrameSequenceFrame[];\n cameraKeys: string[];\n cameraSummaries: Record<string, CameraFrameSequenceCameraSummary>;\n frameCount: number;\n}\n\nexport interface CameraFrameSequenceRecorderAPI {\n status: FrameCaptureStatus;\n error: Error | null;\n isRecording: boolean;\n record: (options: CameraFrameSequenceOptions) => Promise<CameraFrameSequenceResult>;\n reset: () => void;\n}\n\n// ---- Canvas Props ----\n\nexport type MujocoCanvasProps = Omit<CanvasProps, 'onError'> & {\n config: SceneConfig;\n /** R3F content rendered while the MuJoCo WASM module is still loading. */\n loadingFallback?: ReactNode;\n onReady?: (input: ReadyCallbackInput) => void;\n onError?: (error: Error) => void;\n onStep?: (input: StepCallbackInput) => void;\n onSelection?: (input: SelectionCallbackInput) => void;\n // Declarative physics config (spec 1.1)\n gravity?: [number, number, number];\n timestep?: number;\n substeps?: number;\n paused?: boolean;\n speed?: number;\n interpolate?: boolean;\n};\n\n// ---- Hook Return Types ----\n\nexport interface SitePositionResult {\n position: React.RefObject<THREE.Vector3>;\n quaternion: React.RefObject<THREE.Quaternion>;\n}\n\nexport interface MujocoContextValue {\n mujoco: MujocoModule | null;\n status: 'loading' | 'ready' | 'error';\n error: string | null;\n}\n\n/** @deprecated Use `SensorHandle` instead. */\nexport interface SensorResult {\n value: React.RefObject<Float64Array>;\n size: number;\n}\n\nexport interface CtrlHandle {\n /** Read the current ctrl value. */\n read(): number;\n /** Write a ctrl value (goes directly to data.ctrl). */\n write(value: number): void;\n /** Actuator name. */\n name: Actuators;\n /** Actuator control range [min, max]. */\n range: [number, number];\n}\n\nexport interface SensorHandle {\n /** Read the current sensor data. */\n read(): Float64Array;\n /** Sensor dimensionality. */\n dim: number;\n /** Sensor name. */\n name: Sensors;\n}\n\nexport interface BodyStateResult {\n position: React.RefObject<THREE.Vector3>;\n quaternion: React.RefObject<THREE.Quaternion>;\n linearVelocity: React.RefObject<THREE.Vector3>;\n angularVelocity: React.RefObject<THREE.Vector3>;\n}\n\nexport interface JointStateResult {\n position: React.RefObject<number | Float64Array>;\n velocity: React.RefObject<number | Float64Array>;\n}\n","/**\n * @license\n * SPDX-License-Identifier: Apache-2.0\n */\n\nimport { useThree } from '@react-three/fiber';\nimport type { ThreeElements } from '@react-three/fiber';\nimport type { ReactNode } from 'react';\nimport { useEffect, useMemo } from 'react';\nimport * as THREE from 'three';\nimport { SplatEnvironmentReadinessStatus } from '../types';\nimport type {\n PairedSplatEnvironmentConfig,\n ScenarioMaterialConfig,\n SceneConfig,\n SplatCollisionProxyConfig,\n SplatEnvironmentReadiness,\n SplatEnvironmentMetadata,\n SplatEnvironmentMetadataInput,\n SplatFormat,\n SplatRendererKind,\n SplatSceneConfigInput,\n SplatSceneConfigState,\n SplatSceneInput,\n ScenarioLightingPreset,\n ScenarioLightingProps,\n SplatEnvironmentProps,\n VisualScenarioConfig,\n VisualScenarioEffectsProps,\n} from '../types';\n\nconst DEFAULT_BACKGROUND = '#181a1f';\n\nexport function ScenarioLighting({\n preset = 'studio',\n castShadow = true,\n intensity = 1,\n}: ScenarioLightingProps) {\n if (preset === 'warehouse') {\n return (\n <>\n <ambientLight intensity={0.18 * intensity} />\n <directionalLight\n position={[3.5, -2, 5]}\n intensity={2.2 * intensity}\n castShadow={castShadow}\n />\n <directionalLight position={[-2, 1.5, 2.5]} intensity={0.25 * intensity} />\n </>\n );\n }\n\n if (preset === 'low-light') {\n return (\n <>\n <ambientLight intensity={0.08 * intensity} />\n <directionalLight\n position={[2, -2, 3]}\n intensity={0.75 * intensity}\n castShadow={castShadow}\n />\n <pointLight position={[-0.5, -0.8, 1.3]} intensity={0.6 * intensity} />\n </>\n );\n }\n\n if (preset === 'splat') {\n return (\n <>\n <ambientLight intensity={0.42 * intensity} />\n <directionalLight\n position={[1.8, -2.4, 3.5]}\n intensity={1.2 * intensity}\n castShadow={castShadow}\n />\n <pointLight position={[0.4, 0.2, 1.4]} intensity={0.35 * intensity} />\n </>\n );\n }\n\n return (\n <>\n <ambientLight intensity={0.35 * intensity} />\n <directionalLight\n position={[2.5, -3, 4]}\n intensity={1.6 * intensity}\n castShadow={castShadow}\n />\n </>\n );\n}\n\nexport function getScenarioBackground(\n preset: ScenarioLightingPreset | undefined,\n fallback = DEFAULT_BACKGROUND\n) {\n if (preset === 'warehouse') return '#20242b';\n if (preset === 'low-light') return '#0f1115';\n if (preset === 'splat') return '#1b1f24';\n return fallback;\n}\n\nexport function getScenarioCameraPosition(\n basePosition: readonly [number, number, number],\n scenario?: Pick<VisualScenarioConfig, 'camera'>\n): [number, number, number] {\n const [x, y, z] = basePosition;\n const jitter = scenario?.camera?.jitter ?? 0;\n\n return [\n Number((x + jitter * 0.6).toFixed(3)),\n Number((y - jitter * 0.4).toFixed(3)),\n Number((z + jitter * 0.25).toFixed(3)),\n ];\n}\n\nexport function VisualScenarioEffects(props: VisualScenarioEffectsProps) {\n useVisualScenarioEffects(props);\n return null;\n}\n\nexport function useVisualScenarioEffects({\n scenario,\n enabled = true,\n applyBackground = true,\n applyFog = true,\n applyRenderer = true,\n applyMaterials = true,\n background,\n fogNear,\n fogFar,\n materialFilter,\n}: VisualScenarioEffectsProps) {\n const { gl, scene, invalidate } = useThree();\n\n useEffect(() => {\n if (!enabled || !scenario) {\n return undefined;\n }\n\n const previousExposure = gl.toneMappingExposure;\n const previousBackground = scene.background;\n const previousFog = scene.fog;\n const materialSnapshots = new Map<\n THREE.Material,\n {\n color?: THREE.Color;\n roughness?: number;\n metalness?: number;\n }\n >();\n\n if (applyRenderer) {\n gl.toneMappingExposure = scenario.camera?.exposure ?? 1;\n }\n\n if (applyBackground) {\n scene.background = new THREE.Color(\n background ?? getScenarioBackground(scenario.lighting)\n );\n }\n\n if (applyFog) {\n scene.fog = createScenarioFog(scenario, background, fogNear, fogFar);\n }\n\n if (applyMaterials && scenario.materials) {\n applyScenarioMaterials(scene, scenario, materialSnapshots, materialFilter);\n }\n\n invalidate();\n\n return () => {\n gl.toneMappingExposure = previousExposure;\n scene.background = previousBackground;\n scene.fog = previousFog;\n\n for (const [material, snapshot] of materialSnapshots) {\n const mutable = getMutableScenarioMaterial(material);\n if (!mutable) continue;\n if (snapshot.color) mutable.color.copy(snapshot.color);\n if (typeof snapshot.roughness === 'number') {\n mutable.roughness = snapshot.roughness;\n }\n if (typeof snapshot.metalness === 'number') {\n mutable.metalness = snapshot.metalness;\n }\n mutable.needsUpdate = true;\n }\n\n invalidate();\n };\n }, [\n applyBackground,\n applyFog,\n applyMaterials,\n applyRenderer,\n background,\n enabled,\n fogFar,\n fogNear,\n gl,\n invalidate,\n materialFilter,\n scenario,\n scene,\n ]);\n}\n\n/**\n * Renderer-agnostic Gaussian splat environment boundary.\n *\n * This component intentionally does not import a specific 3DGS renderer. Pass a\n * Spark/GaussianSplats3D object as `children` once the app chooses a renderer,\n * and pass MuJoCo/MJCF collision proxy visuals via `collisionProxy`.\n */\nexport function SplatEnvironment({\n environment,\n scenario,\n renderer,\n src,\n format,\n collisionProxy,\n collisionProxyMetadata,\n children,\n showPlaceholder = true,\n ...groupProps\n}: SplatEnvironmentProps) {\n const metadata = useSplatEnvironment({\n environment,\n scenario,\n renderer,\n src,\n format,\n collisionProxy: collisionProxyMetadata,\n });\n const existingUserData =\n typeof groupProps.userData === 'object' && groupProps.userData !== null\n ? groupProps.userData\n : {};\n\n return (\n <group\n {...groupProps}\n userData={{\n ...existingUserData,\n ...metadata.userData,\n }}\n >\n {children}\n {children || !showPlaceholder ? null : <SplatPlaceholder />}\n {collisionProxy}\n </group>\n );\n}\n\nexport function useSplatEnvironment({\n environment,\n scenario,\n renderer,\n src,\n format,\n collisionProxy,\n}: SplatEnvironmentMetadataInput): SplatEnvironmentMetadata {\n const scenarioEnvironment = useMemo(\n () =>\n environment ??\n (scenario\n ? createPairedSplatEnvironment(scenario, { renderer })\n : undefined),\n [environment, renderer, scenario]\n );\n const resolvedSrc = src ?? scenarioEnvironment?.splat.src ?? scenario?.splat?.src;\n const resolvedFormat =\n format ??\n scenarioEnvironment?.splat.format ??\n scenario?.splat?.format ??\n 'spz';\n const resolvedCollisionProxy =\n collisionProxy ??\n scenarioEnvironment?.collisionProxy ??\n scenario?.splat?.collisionProxy ??\n undefined;\n const readiness = useMemo(\n () =>\n getSplatEnvironmentReadiness({\n environment: scenarioEnvironment,\n scenario,\n renderer,\n src: resolvedSrc,\n format: resolvedFormat,\n collisionProxy: resolvedCollisionProxy,\n }),\n [\n collisionProxy,\n renderer,\n resolvedCollisionProxy,\n resolvedFormat,\n resolvedSrc,\n scenario,\n scenarioEnvironment,\n ]\n );\n\n return useMemo(\n () => ({\n src: resolvedSrc,\n format: resolvedFormat,\n collisionProxy: resolvedCollisionProxy,\n readiness,\n userData: createSplatEnvironmentUserData({\n environment: scenarioEnvironment,\n src: resolvedSrc,\n format: resolvedFormat,\n collisionProxy: resolvedCollisionProxy,\n readiness,\n }),\n }),\n [\n scenarioEnvironment,\n resolvedSrc,\n resolvedFormat,\n resolvedCollisionProxy,\n readiness,\n ]\n );\n}\n\n/**\n * Resolve a visual scenario's paired splat environment and compose its MJCF\n * collision proxy into a MuJoCo scene config.\n *\n * This hook is renderer-agnostic: apps can use it with Spark, another 3DGS\n * renderer, or their own Three scene objects while keeping physics collision\n * files paired with the visual splat metadata.\n */\nexport function useSplatSceneConfig({\n sceneConfig,\n scenario,\n environment,\n enabled = true,\n renderer,\n}: SplatSceneConfigInput): SplatSceneConfigState {\n const resolvedEnvironment = useMemo(\n () =>\n enabled\n ? environment ??\n (scenario\n ? createPairedSplatEnvironment(scenario, { renderer })\n : undefined)\n : undefined,\n [enabled, environment, renderer, scenario]\n );\n const readiness = useMemo(\n () =>\n getSplatEnvironmentReadiness({\n environment: resolvedEnvironment,\n scenario,\n renderer,\n enabled,\n }),\n [enabled, renderer, resolvedEnvironment, scenario]\n );\n const resolvedSceneConfig = useMemo(\n () =>\n resolvedEnvironment\n ? withSplatEnvironment(sceneConfig, resolvedEnvironment)\n : sceneConfig,\n [resolvedEnvironment, sceneConfig]\n );\n\n return useMemo(\n () => ({\n environment: resolvedEnvironment,\n sceneConfig: resolvedSceneConfig,\n enabled: enabled && readiness.status !== SplatEnvironmentReadinessStatus.Disabled,\n readiness,\n }),\n [enabled, readiness, resolvedEnvironment, resolvedSceneConfig]\n );\n}\n\nexport function getSplatEnvironmentReadiness({\n environment,\n scenario,\n renderer,\n src,\n format,\n collisionProxy,\n enabled = true,\n}: {\n environment?: PairedSplatEnvironmentConfig;\n scenario?: Pick<VisualScenarioConfig, 'splat'>;\n renderer?: SplatRendererKind;\n src?: string;\n format?: SplatFormat;\n collisionProxy?: SplatCollisionProxyConfig;\n enabled?: boolean;\n}): SplatEnvironmentReadiness {\n const splat = scenario?.splat;\n const resolvedSrc = src ?? environment?.splat.src ?? splat?.src;\n const resolvedFormat =\n format ?? environment?.splat.format ?? splat?.format ?? 'spz';\n const resolvedRenderer = renderer ?? environment?.splat.renderer;\n const resolvedCollisionProxy =\n collisionProxy ?? environment?.collisionProxy ?? splat?.collisionProxy ?? undefined;\n const requiresCollisionProxy = splat?.requiresCollisionProxy ?? true;\n\n if (!enabled || (splat && splat.enabled === false && !environment)) {\n return {\n status: SplatEnvironmentReadinessStatus.Disabled,\n ready: false,\n requiresCollisionProxy,\n missing: [],\n format: resolvedFormat,\n renderer: resolvedRenderer,\n message: 'Splat environment is disabled.',\n };\n }\n\n if (!resolvedSrc) {\n return {\n status: SplatEnvironmentReadinessStatus.MissingSplat,\n ready: false,\n requiresCollisionProxy,\n missing: ['splat'],\n format: resolvedFormat,\n renderer: resolvedRenderer,\n message: 'Splat environment is missing a visual asset source.',\n };\n }\n\n if (resolvedRenderer === 'spark' && resolvedFormat !== 'spz') {\n return {\n status: SplatEnvironmentReadinessStatus.UnsupportedFormat,\n ready: false,\n requiresCollisionProxy,\n missing: [],\n format: resolvedFormat,\n renderer: resolvedRenderer,\n message: `Spark splat rendering requires .spz assets; received ${resolvedFormat}.`,\n };\n }\n\n if (requiresCollisionProxy && !resolvedCollisionProxy?.xmlPath) {\n return {\n status: SplatEnvironmentReadinessStatus.MissingCollisionProxy,\n ready: false,\n requiresCollisionProxy,\n missing: ['collisionProxy'],\n format: resolvedFormat,\n renderer: resolvedRenderer,\n message: 'Splat environment is missing paired MJCF collision proxy XML.',\n };\n }\n\n return {\n status: SplatEnvironmentReadinessStatus.Ready,\n ready: true,\n requiresCollisionProxy,\n missing: [],\n format: resolvedFormat,\n renderer: resolvedRenderer,\n message: requiresCollisionProxy\n ? 'Splat environment has visual asset and collision proxy metadata.'\n : 'Splat environment has a visual asset and does not require collision proxy metadata.',\n };\n}\n\n/**\n * Convert a generic visual scenario splat block into a paired visual/physics\n * environment config. Returns undefined until both the splat asset and MJCF\n * collision proxy are present.\n */\nexport function createPairedSplatEnvironment(\n scenario: Pick<VisualScenarioConfig, 'id' | 'label' | 'environment' | 'splat'>,\n options: {\n id?: string;\n label?: string;\n description?: string;\n renderer?: SplatRendererKind;\n } = {}\n): PairedSplatEnvironmentConfig | undefined {\n const splat = scenario.splat;\n const collisionProxy = splat?.collisionProxy;\n\n if (!splat?.enabled || !splat.src || !collisionProxy?.xmlPath) {\n return undefined;\n }\n\n return {\n id: options.id ?? scenario.id ?? 'splat-environment',\n label: options.label ?? scenario.label ?? 'Gaussian splat environment',\n description:\n options.description ??\n (scenario.environment\n ? `Visual ${scenario.environment} splat paired with MJCF collision proxy.`\n : undefined),\n splat: {\n src: splat.src,\n format: splat.format ?? 'spz',\n renderer: options.renderer,\n },\n collisionProxy: {\n ...collisionProxy,\n xmlPath: collisionProxy.xmlPath,\n },\n };\n}\n\nfunction isPairedSplatEnvironment(input: SplatSceneInput): input is PairedSplatEnvironmentConfig {\n return !!input && 'collisionProxy' in input && 'splat' in input;\n}\n\nfunction sceneRelativePath(sceneConfig: SceneConfig, path: string): string {\n const src = sceneConfig.src;\n if (!src) return path;\n\n const base = src.endsWith('/') ? src : src + '/';\n if (path.startsWith(base)) return path.slice(base.length);\n return path;\n}\n\nfunction uniquePaths(paths: readonly string[]): string[] {\n const seen = new Set<string>();\n const result: string[] = [];\n for (const path of paths) {\n if (seen.has(path)) continue;\n seen.add(path);\n result.push(path);\n }\n return result;\n}\n\n/**\n * Compose a MuJoCo scene config with a paired splat collision proxy.\n *\n * This keeps the common hybrid setup declarative:\n * robot XML remains `sceneFile`, the `.spz` remains a visual-only layer, and\n * the paired MJCF collision proxy is added to `environmentFiles`.\n */\nexport function withSplatEnvironment(\n sceneConfig: SceneConfig,\n input: SplatSceneInput,\n options: { renderer?: SplatRendererKind } = {}\n): SceneConfig {\n const environment = isPairedSplatEnvironment(input)\n ? input\n : input\n ? createPairedSplatEnvironment(input, options)\n : undefined;\n const xmlPath = environment?.collisionProxy.xmlPath;\n if (!xmlPath) return sceneConfig;\n\n return {\n ...sceneConfig,\n environmentFiles: uniquePaths([\n ...(sceneConfig.environmentFiles ?? []),\n sceneRelativePath(sceneConfig, xmlPath),\n ]),\n };\n}\n\nexport function createSplatEnvironmentUserData({\n environment,\n src,\n format = 'spz',\n collisionProxy,\n readiness,\n}: {\n environment?: PairedSplatEnvironmentConfig;\n src?: string;\n format?: SplatFormat;\n collisionProxy?: SplatCollisionProxyConfig;\n readiness?: SplatEnvironmentReadiness;\n}) {\n return {\n role: 'splat-environment',\n environmentId: environment?.id,\n environmentLabel: environment?.label,\n splatSrc: src,\n splatFormat: format,\n splatRenderer: environment?.splat.renderer,\n collisionProxyStatus: collisionProxy?.status ?? 'missing',\n collisionProxyXmlPath: collisionProxy?.xmlPath,\n collisionProxyPrimitives: collisionProxy?.primitives ?? [],\n readinessStatus: readiness?.status,\n readinessMessage: readiness?.message,\n };\n}\n\nexport function createSparkSplatViewerUrl({\n viewerUrl,\n splatSrc,\n}: {\n viewerUrl: string;\n splatSrc: string;\n}) {\n const url = new URL(viewerUrl, 'http://mujoco-react.local');\n url.searchParams.set('splat', splatSrc);\n return viewerUrl.startsWith('http') ? url.toString() : `${url.pathname}${url.search}`;\n}\n\nfunction SplatPlaceholder() {\n return (\n <group>\n <mesh position={[0, 0, 1.2]}>\n <boxGeometry args={[2.4, 2.4, 2.4]} />\n <meshBasicMaterial\n color=\"#8b8b8b\"\n transparent\n opacity={0.06}\n wireframe\n side={THREE.DoubleSide}\n />\n </mesh>\n </group>\n );\n}\n\nfunction createScenarioFog(\n scenario: VisualScenarioConfig,\n background: THREE.ColorRepresentation | undefined,\n fogNear: number | undefined,\n fogFar: number | undefined\n) {\n if (scenario.lighting === 'low-light') {\n return new THREE.Fog(\n background ?? getScenarioBackground(scenario.lighting),\n fogNear ?? 2.5,\n fogFar ?? 9\n );\n }\n\n if (scenario.lighting === 'warehouse') {\n return new THREE.Fog(\n background ?? getScenarioBackground(scenario.lighting),\n fogNear ?? 5,\n fogFar ?? 16\n );\n }\n\n return null;\n}\n\nfunction applyScenarioMaterials(\n scene: THREE.Scene,\n scenario: VisualScenarioConfig,\n snapshots: Map<\n THREE.Material,\n {\n color?: THREE.Color;\n roughness?: number;\n metalness?: number;\n }\n >,\n materialFilter: VisualScenarioEffectsProps['materialFilter']\n) {\n const materials = scenario.materials;\n if (!materials) return;\n\n scene.traverse((object) => {\n if (!(object instanceof THREE.Mesh)) {\n return;\n }\n\n for (const material of normalizeMaterials(object.material)) {\n const mutable = getMutableScenarioMaterial(material);\n if (!mutable) continue;\n if (materialFilter && !materialFilter({ object, material })) continue;\n\n if (!snapshots.has(material)) {\n snapshots.set(material, {\n color: mutable.color.clone(),\n roughness: mutable.roughness,\n metalness: mutable.metalness,\n });\n }\n\n applyScenarioMaterial(mutable, object, scenario, materials);\n }\n });\n}\n\nfunction applyScenarioMaterial(\n material: THREE.MeshStandardMaterial | THREE.MeshPhysicalMaterial,\n object: THREE.Object3D,\n scenario: VisualScenarioConfig,\n materials: ScenarioMaterialConfig\n) {\n const seed = scenario.seed ?? 0;\n const objectKey = `${scenario.id ?? 'scenario'}:${object.name}:${material.name}:${seed}`;\n const variation = hashToUnitInterval(objectKey);\n\n if (materials.randomizeObjectColors) {\n material.color.setHSL(variation, 0.38, 0.42);\n }\n\n if (materials.randomizeTableMaterial) {\n material.roughness = clamp01(\n materials.roughness ?? 0.35 + variation * 0.45\n );\n material.metalness = clamp01(\n materials.metalness ?? variation * 0.12\n );\n }\n\n material.needsUpdate = true;\n}\n\nfunction normalizeMaterials(\n material: THREE.Material | THREE.Material[]\n): THREE.Material[] {\n return Array.isArray(material) ? material : [material];\n}\n\nfunction getMutableScenarioMaterial(\n material: THREE.Material\n): THREE.MeshStandardMaterial | THREE.MeshPhysicalMaterial | null {\n if (\n material instanceof THREE.MeshStandardMaterial ||\n material instanceof THREE.MeshPhysicalMaterial\n ) {\n return material;\n }\n\n return null;\n}\n\nfunction hashToUnitInterval(value: string) {\n let hash = 2166136261;\n for (let index = 0; index < value.length; index += 1) {\n hash ^= value.charCodeAt(index);\n hash = Math.imul(hash, 16777619);\n }\n return (hash >>> 0) / 4294967295;\n}\n\nfunction clamp01(value: number) {\n return Math.max(0, Math.min(1, value));\n}\n\nexport type SplatCollisionProxy = ReactNode | ThreeElements['group'];\n"]}