2026-04-11 03:46:30 +02:00
|
|
|
import { createOpaqueId } from "../core/ids";
|
2026-03-31 02:02:50 +02:00
|
|
|
import type { Brush } from "./brushes";
|
2026-03-31 17:31:48 +02:00
|
|
|
import type { ModelInstance } from "../assets/model-instances";
|
|
|
|
|
import type { ProjectAssetRecord } from "../assets/project-assets";
|
2026-03-31 03:00:38 +02:00
|
|
|
import type { EntityInstance } from "../entities/entity-instances";
|
2026-03-31 06:15:41 +02:00
|
|
|
import type { InteractionLink } from "../interactions/interaction-links";
|
2026-04-11 04:19:50 +02:00
|
|
|
import {
|
|
|
|
|
cloneMaterialRegistry,
|
|
|
|
|
createStarterMaterialRegistry,
|
|
|
|
|
type MaterialDef
|
|
|
|
|
} from "../materials/starter-material-library";
|
|
|
|
|
import {
|
|
|
|
|
createDefaultWorldSettings,
|
|
|
|
|
type WorldSettings
|
|
|
|
|
} from "./world-settings";
|
2026-03-31 01:29:35 +02:00
|
|
|
|
2026-04-11 04:13:45 +02:00
|
|
|
export const SCENE_DOCUMENT_VERSION = 23 as const;
|
2026-04-11 03:46:30 +02:00
|
|
|
export const MULTI_SCENE_FOUNDATION_SCENE_DOCUMENT_VERSION = 22 as const;
|
2026-04-07 07:10:29 +02:00
|
|
|
export const WATER_SURFACE_DISPLACEMENT_SCENE_DOCUMENT_VERSION = 21 as const;
|
2026-04-06 08:18:03 +02:00
|
|
|
export const WHITEBOX_BOX_VOLUME_SCENE_DOCUMENT_VERSION = 20 as const;
|
2026-04-05 02:27:13 +02:00
|
|
|
export const WHITEBOX_GEOMETRY_SCENE_DOCUMENT_VERSION = 19 as const;
|
2026-04-04 19:25:43 +02:00
|
|
|
export const WHITEBOX_FLOAT_TRANSFORM_SCENE_DOCUMENT_VERSION = 18 as const;
|
2026-04-11 04:19:50 +02:00
|
|
|
export const PLAYER_START_COLLIDER_SETTINGS_SCENE_DOCUMENT_VERSION =
|
|
|
|
|
17 as const;
|
2026-04-04 07:51:38 +02:00
|
|
|
export const IMPORTED_MODEL_COLLIDERS_SCENE_DOCUMENT_VERSION = 16 as const;
|
2026-04-03 01:02:53 +02:00
|
|
|
export const ENTITY_NAMES_SCENE_DOCUMENT_VERSION = 15 as const;
|
2026-04-02 19:36:28 +02:00
|
|
|
export const SPATIAL_AUDIO_SCENE_DOCUMENT_VERSION = 13 as const;
|
2026-04-01 00:09:49 +02:00
|
|
|
export const ANIMATION_PLAYBACK_SCENE_DOCUMENT_VERSION = 12 as const;
|
2026-03-31 19:57:31 +02:00
|
|
|
export const LOCAL_LIGHTS_AND_SKYBOX_SCENE_DOCUMENT_VERSION = 10 as const;
|
2026-03-31 17:31:48 +02:00
|
|
|
export const MODEL_ASSET_PIPELINE_SCENE_DOCUMENT_VERSION = 9 as const;
|
2026-03-31 02:02:50 +02:00
|
|
|
export const FOUNDATION_SCENE_DOCUMENT_VERSION = 1 as const;
|
2026-03-31 02:33:18 +02:00
|
|
|
export const BOX_BRUSH_SCENE_DOCUMENT_VERSION = 2 as const;
|
2026-03-31 03:00:38 +02:00
|
|
|
export const FACE_MATERIALS_SCENE_DOCUMENT_VERSION = 3 as const;
|
2026-03-31 04:22:46 +02:00
|
|
|
export const RUNNER_V1_SCENE_DOCUMENT_VERSION = 4 as const;
|
2026-03-31 05:09:25 +02:00
|
|
|
export const FIRST_ROOM_POLISH_SCENE_DOCUMENT_VERSION = 5 as const;
|
2026-03-31 05:50:35 +02:00
|
|
|
export const WORLD_ENVIRONMENT_SCENE_DOCUMENT_VERSION = 6 as const;
|
2026-03-31 06:15:41 +02:00
|
|
|
export const ENTITY_SYSTEM_FOUNDATION_SCENE_DOCUMENT_VERSION = 7 as const;
|
2026-04-11 04:19:50 +02:00
|
|
|
export const TRIGGER_ACTION_TARGET_FOUNDATION_SCENE_DOCUMENT_VERSION =
|
|
|
|
|
8 as const;
|
2026-04-11 04:07:43 +02:00
|
|
|
export const RUNNER_LOADING_SCREEN_SCENE_DOCUMENT_VERSION = 23 as const;
|
2026-03-31 01:29:35 +02:00
|
|
|
|
2026-04-11 03:46:30 +02:00
|
|
|
export const DEFAULT_PROJECT_SCENE_ID = "scene-main" as const;
|
|
|
|
|
|
2026-04-11 04:07:43 +02:00
|
|
|
export interface SceneLoadingScreenSettings {
|
|
|
|
|
colorHex: string;
|
|
|
|
|
headline: string | null;
|
|
|
|
|
description: string | null;
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-11 03:46:30 +02:00
|
|
|
export interface ProjectScene {
|
|
|
|
|
id: string;
|
|
|
|
|
name: string;
|
2026-04-11 04:07:43 +02:00
|
|
|
loadingScreen: SceneLoadingScreenSettings;
|
2026-04-11 03:46:30 +02:00
|
|
|
world: WorldSettings;
|
|
|
|
|
brushes: Record<string, Brush>;
|
|
|
|
|
modelInstances: Record<string, ModelInstance>;
|
|
|
|
|
entities: Record<string, EntityInstance>;
|
|
|
|
|
interactionLinks: Record<string, InteractionLink>;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface ProjectDocument {
|
|
|
|
|
version: typeof SCENE_DOCUMENT_VERSION;
|
|
|
|
|
activeSceneId: string;
|
|
|
|
|
scenes: Record<string, ProjectScene>;
|
|
|
|
|
materials: Record<string, MaterialDef>;
|
|
|
|
|
textures: Record<string, never>;
|
|
|
|
|
assets: Record<string, ProjectAssetRecord>;
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-31 01:29:35 +02:00
|
|
|
export interface SceneDocument {
|
|
|
|
|
version: typeof SCENE_DOCUMENT_VERSION;
|
|
|
|
|
name: string;
|
|
|
|
|
world: WorldSettings;
|
2026-03-31 02:33:18 +02:00
|
|
|
materials: Record<string, MaterialDef>;
|
2026-03-31 01:29:35 +02:00
|
|
|
textures: Record<string, never>;
|
2026-03-31 17:31:48 +02:00
|
|
|
assets: Record<string, ProjectAssetRecord>;
|
2026-03-31 02:02:50 +02:00
|
|
|
brushes: Record<string, Brush>;
|
2026-03-31 17:31:48 +02:00
|
|
|
modelInstances: Record<string, ModelInstance>;
|
2026-03-31 03:00:38 +02:00
|
|
|
entities: Record<string, EntityInstance>;
|
2026-03-31 06:15:41 +02:00
|
|
|
interactionLinks: Record<string, InteractionLink>;
|
2026-03-31 01:29:35 +02:00
|
|
|
}
|
|
|
|
|
|
2026-04-11 04:19:50 +02:00
|
|
|
export function createEmptySceneDocument(
|
|
|
|
|
overrides: Partial<Pick<SceneDocument, "name" | "world" | "materials">> = {}
|
|
|
|
|
): SceneDocument {
|
2026-03-31 01:29:35 +02:00
|
|
|
return {
|
|
|
|
|
version: SCENE_DOCUMENT_VERSION,
|
|
|
|
|
name: overrides.name ?? "Untitled Scene",
|
|
|
|
|
world: overrides.world ?? createDefaultWorldSettings(),
|
2026-04-11 04:19:50 +02:00
|
|
|
materials: cloneMaterialRegistry(
|
|
|
|
|
overrides.materials ?? createStarterMaterialRegistry()
|
|
|
|
|
),
|
2026-03-31 01:29:35 +02:00
|
|
|
textures: {},
|
|
|
|
|
assets: {},
|
|
|
|
|
brushes: {},
|
|
|
|
|
modelInstances: {},
|
|
|
|
|
entities: {},
|
|
|
|
|
interactionLinks: {}
|
|
|
|
|
};
|
|
|
|
|
}
|
2026-04-11 03:46:30 +02:00
|
|
|
|
|
|
|
|
export function createEmptyProjectScene(
|
2026-04-11 04:19:50 +02:00
|
|
|
overrides: Partial<
|
|
|
|
|
Pick<ProjectScene, "id" | "name" | "loadingScreen" | "world">
|
|
|
|
|
> = {}
|
2026-04-11 03:46:30 +02:00
|
|
|
): ProjectScene {
|
|
|
|
|
return {
|
|
|
|
|
id: overrides.id ?? createOpaqueId("scene"),
|
|
|
|
|
name: overrides.name ?? "Untitled Scene",
|
2026-04-11 04:13:45 +02:00
|
|
|
loadingScreen: cloneSceneLoadingScreenSettings(
|
|
|
|
|
overrides.loadingScreen ?? createDefaultSceneLoadingScreenSettings()
|
|
|
|
|
),
|
2026-04-11 03:46:30 +02:00
|
|
|
world: overrides.world ?? createDefaultWorldSettings(),
|
|
|
|
|
brushes: {},
|
|
|
|
|
modelInstances: {},
|
|
|
|
|
entities: {},
|
|
|
|
|
interactionLinks: {}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function createEmptyProjectDocument(
|
|
|
|
|
overrides: Partial<
|
|
|
|
|
Pick<ProjectDocument, "activeSceneId" | "materials" | "textures" | "assets">
|
|
|
|
|
> & {
|
|
|
|
|
sceneId?: string;
|
|
|
|
|
sceneName?: string;
|
|
|
|
|
world?: WorldSettings;
|
|
|
|
|
} = {}
|
|
|
|
|
): ProjectDocument {
|
|
|
|
|
const initialScene = createEmptyProjectScene({
|
2026-04-11 04:19:50 +02:00
|
|
|
id:
|
|
|
|
|
overrides.sceneId ?? overrides.activeSceneId ?? DEFAULT_PROJECT_SCENE_ID,
|
2026-04-11 03:46:30 +02:00
|
|
|
name: overrides.sceneName,
|
|
|
|
|
world: overrides.world
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
version: SCENE_DOCUMENT_VERSION,
|
|
|
|
|
activeSceneId: initialScene.id,
|
|
|
|
|
scenes: {
|
|
|
|
|
[initialScene.id]: initialScene
|
|
|
|
|
},
|
|
|
|
|
materials: cloneMaterialRegistry(
|
|
|
|
|
overrides.materials ?? createStarterMaterialRegistry()
|
|
|
|
|
),
|
|
|
|
|
textures: overrides.textures ?? {},
|
|
|
|
|
assets: overrides.assets ?? {}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function getProjectScene(
|
|
|
|
|
projectDocument: ProjectDocument,
|
|
|
|
|
sceneId = projectDocument.activeSceneId
|
|
|
|
|
): ProjectScene {
|
|
|
|
|
const scene = projectDocument.scenes[sceneId];
|
|
|
|
|
|
|
|
|
|
if (scene === undefined) {
|
|
|
|
|
throw new Error(`Project scene ${sceneId} does not exist.`);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return scene;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function createSceneDocumentFromProject(
|
|
|
|
|
projectDocument: ProjectDocument,
|
|
|
|
|
sceneId = projectDocument.activeSceneId
|
|
|
|
|
): SceneDocument {
|
|
|
|
|
const scene = getProjectScene(projectDocument, sceneId);
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
version: projectDocument.version,
|
|
|
|
|
name: scene.name,
|
|
|
|
|
world: scene.world,
|
|
|
|
|
materials: projectDocument.materials,
|
|
|
|
|
textures: projectDocument.textures,
|
|
|
|
|
assets: projectDocument.assets,
|
|
|
|
|
brushes: scene.brushes,
|
|
|
|
|
modelInstances: scene.modelInstances,
|
|
|
|
|
entities: scene.entities,
|
|
|
|
|
interactionLinks: scene.interactionLinks
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function createProjectDocumentFromSceneDocument(
|
|
|
|
|
sceneDocument: SceneDocument,
|
|
|
|
|
sceneId = DEFAULT_PROJECT_SCENE_ID
|
|
|
|
|
): ProjectDocument {
|
|
|
|
|
return {
|
|
|
|
|
version: SCENE_DOCUMENT_VERSION,
|
|
|
|
|
activeSceneId: sceneId,
|
|
|
|
|
scenes: {
|
|
|
|
|
[sceneId]: {
|
|
|
|
|
id: sceneId,
|
|
|
|
|
name: sceneDocument.name,
|
2026-04-11 04:07:43 +02:00
|
|
|
loadingScreen: createDefaultSceneLoadingScreenSettings(),
|
2026-04-11 03:46:30 +02:00
|
|
|
world: sceneDocument.world,
|
|
|
|
|
brushes: sceneDocument.brushes,
|
|
|
|
|
modelInstances: sceneDocument.modelInstances,
|
|
|
|
|
entities: sceneDocument.entities,
|
|
|
|
|
interactionLinks: sceneDocument.interactionLinks
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
materials: sceneDocument.materials,
|
|
|
|
|
textures: sceneDocument.textures,
|
|
|
|
|
assets: sceneDocument.assets
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function applySceneDocumentToProject(
|
|
|
|
|
projectDocument: ProjectDocument,
|
|
|
|
|
sceneId: string,
|
|
|
|
|
sceneDocument: SceneDocument
|
|
|
|
|
): ProjectDocument {
|
|
|
|
|
const previousScene = getProjectScene(projectDocument, sceneId);
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
...projectDocument,
|
|
|
|
|
version: SCENE_DOCUMENT_VERSION,
|
|
|
|
|
materials: sceneDocument.materials,
|
|
|
|
|
textures: sceneDocument.textures,
|
|
|
|
|
assets: sceneDocument.assets,
|
|
|
|
|
scenes: {
|
|
|
|
|
...projectDocument.scenes,
|
|
|
|
|
[sceneId]: {
|
|
|
|
|
...previousScene,
|
|
|
|
|
name: sceneDocument.name,
|
|
|
|
|
world: sceneDocument.world,
|
|
|
|
|
brushes: sceneDocument.brushes,
|
|
|
|
|
modelInstances: sceneDocument.modelInstances,
|
|
|
|
|
entities: sceneDocument.entities,
|
|
|
|
|
interactionLinks: sceneDocument.interactionLinks
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
2026-04-11 04:07:43 +02:00
|
|
|
|
|
|
|
|
export function createDefaultSceneLoadingScreenSettings(): SceneLoadingScreenSettings {
|
|
|
|
|
return {
|
|
|
|
|
colorHex: "#0d1117",
|
|
|
|
|
headline: null,
|
|
|
|
|
description: null
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function cloneSceneLoadingScreenSettings(
|
|
|
|
|
settings: SceneLoadingScreenSettings
|
|
|
|
|
): SceneLoadingScreenSettings {
|
|
|
|
|
return {
|
|
|
|
|
colorHex: settings.colorHex,
|
|
|
|
|
headline: settings.headline,
|
|
|
|
|
description: settings.description
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function areSceneLoadingScreenSettingsEqual(
|
|
|
|
|
left: SceneLoadingScreenSettings,
|
|
|
|
|
right: SceneLoadingScreenSettings
|
|
|
|
|
): boolean {
|
|
|
|
|
return (
|
|
|
|
|
left.colorHex === right.colorHex &&
|
|
|
|
|
left.headline === right.headline &&
|
|
|
|
|
left.description === right.description
|
|
|
|
|
);
|
|
|
|
|
}
|