964 lines
27 KiB
TypeScript
964 lines
27 KiB
TypeScript
import { CommandHistory } from "../commands/command-history";
|
|
import type { CommandContext, EditorCommand } from "../commands/command";
|
|
import {
|
|
areEditorSelectionsEqual,
|
|
getSelectionDefaultActiveId,
|
|
normalizeSelectionForWhiteboxSelectionMode,
|
|
type EditorSelection
|
|
} from "../core/selection";
|
|
import type { ToolMode } from "../core/tool-mode";
|
|
import { type WhiteboxSelectionMode } from "../core/whitebox-selection-mode";
|
|
import {
|
|
areTransformSessionsEqual,
|
|
cloneTransformSession,
|
|
createInactiveTransformSession,
|
|
type TransformAxis,
|
|
type TransformAxisSpace,
|
|
type TransformSessionState
|
|
} from "../core/transform-session";
|
|
import {
|
|
applyStarterEnvironmentAssetsToProjectDocument
|
|
} from "../assets/starter-environment-assets";
|
|
import {
|
|
applySceneDocumentToProject,
|
|
createDefaultSceneEditorPreferences,
|
|
createEmptyProjectDocument,
|
|
createEmptyProjectScene,
|
|
cloneSceneEditorPreferences,
|
|
createProjectDocumentFromSceneDocument,
|
|
createSceneDocumentFromProject,
|
|
type SceneEditorPanelPreferences,
|
|
type SceneEditorPreferences,
|
|
type ProjectDocument,
|
|
type SceneDocument
|
|
} from "../document/scene-document";
|
|
import {
|
|
DEFAULT_SCENE_DRAFT_STORAGE_KEY,
|
|
type LoadSceneDocumentDraftResult,
|
|
loadSceneDocumentDraft,
|
|
type KeyValueStorage,
|
|
type SaveSceneDocumentDraftResult,
|
|
saveSceneDocumentDraft
|
|
} from "../serialization/local-draft-storage";
|
|
import {
|
|
parseProjectDocumentJson,
|
|
serializeProjectDocument
|
|
} from "../serialization/scene-document-json";
|
|
import type { ViewportViewMode } from "../viewport-three/viewport-view-modes";
|
|
import {
|
|
areViewportToolPreviewsEqual,
|
|
cloneViewportToolPreview,
|
|
createDefaultViewportTransientState,
|
|
isViewportToolPreviewCompatible,
|
|
type ViewportToolPreview,
|
|
type ViewportTransientState
|
|
} from "../viewport-three/viewport-transient-state";
|
|
import {
|
|
areViewportPanelCameraStatesEqual,
|
|
cloneViewportLayoutState,
|
|
cloneViewportPanelCameraState,
|
|
createDefaultViewportLayoutState,
|
|
type ViewportDisplayMode,
|
|
type ViewportLayoutState,
|
|
type ViewportLayoutMode,
|
|
type ViewportPanelId,
|
|
type ViewportPanelState,
|
|
type ViewportPanelCameraState,
|
|
type ViewportQuadSplit
|
|
} from "../viewport-three/viewport-layout";
|
|
import {
|
|
summarizeUpdateLoopCameraState,
|
|
summarizeUpdateLoopCameraStateDeltas,
|
|
traceUpdateLoopEvent
|
|
} from "../debug/update-loop-trace";
|
|
|
|
export interface EditorStoreState {
|
|
projectDocument: ProjectDocument;
|
|
activeSceneId: string;
|
|
document: SceneDocument;
|
|
selection: EditorSelection;
|
|
activeSelectionId: string | null;
|
|
whiteboxSelectionMode: WhiteboxSelectionMode;
|
|
whiteboxSnapEnabled: boolean;
|
|
whiteboxSnapStep: number;
|
|
toolMode: ToolMode;
|
|
viewportGridVisible: boolean;
|
|
viewportLayoutMode: ViewportLayoutMode;
|
|
activeViewportPanelId: ViewportPanelId;
|
|
viewportPanels: Record<ViewportPanelId, ViewportPanelState>;
|
|
viewportQuadSplit: ViewportQuadSplit;
|
|
viewportTransientState: ViewportTransientState;
|
|
canUndo: boolean;
|
|
canRedo: boolean;
|
|
lastCommandLabel: string | null;
|
|
storageAvailable: boolean;
|
|
}
|
|
|
|
interface EditorStoreOptions {
|
|
initialDocument?: SceneDocument;
|
|
initialProjectDocument?: ProjectDocument;
|
|
initialViewportLayoutState?: ViewportLayoutState;
|
|
storage?: KeyValueStorage | null;
|
|
storageKey?: string;
|
|
}
|
|
|
|
type EditorStoreListener = () => void;
|
|
|
|
export type EditorDraftSaveResult = SaveSceneDocumentDraftResult;
|
|
export type EditorDraftLoadResult = LoadSceneDocumentDraftResult;
|
|
|
|
function getSceneEditorPreferences(
|
|
projectDocument: ProjectDocument,
|
|
sceneId = projectDocument.activeSceneId
|
|
): SceneEditorPreferences {
|
|
const scene =
|
|
projectDocument.scenes[sceneId] ??
|
|
createEmptyProjectScene({
|
|
id: sceneId
|
|
});
|
|
const defaults = createDefaultSceneEditorPreferences();
|
|
const preferences =
|
|
(scene as ProjectDocument["scenes"][string] & {
|
|
editorPreferences?: Partial<SceneEditorPreferences>;
|
|
}).editorPreferences ?? defaults;
|
|
|
|
return cloneSceneEditorPreferences({
|
|
whiteboxSelectionMode:
|
|
preferences.whiteboxSelectionMode ?? defaults.whiteboxSelectionMode,
|
|
whiteboxSnapEnabled:
|
|
preferences.whiteboxSnapEnabled ?? defaults.whiteboxSnapEnabled,
|
|
whiteboxSnapStep: preferences.whiteboxSnapStep ?? defaults.whiteboxSnapStep,
|
|
viewportGridVisible:
|
|
preferences.viewportGridVisible ?? defaults.viewportGridVisible,
|
|
viewportLayoutMode:
|
|
preferences.viewportLayoutMode ?? defaults.viewportLayoutMode,
|
|
activeViewportPanelId:
|
|
preferences.activeViewportPanelId ?? defaults.activeViewportPanelId,
|
|
viewportQuadSplit: {
|
|
x: preferences.viewportQuadSplit?.x ?? defaults.viewportQuadSplit.x,
|
|
y: preferences.viewportQuadSplit?.y ?? defaults.viewportQuadSplit.y
|
|
},
|
|
viewportPanels: {
|
|
topLeft: {
|
|
...defaults.viewportPanels.topLeft,
|
|
...preferences.viewportPanels?.topLeft
|
|
},
|
|
topRight: {
|
|
...defaults.viewportPanels.topRight,
|
|
...preferences.viewportPanels?.topRight
|
|
},
|
|
bottomLeft: {
|
|
...defaults.viewportPanels.bottomLeft,
|
|
...preferences.viewportPanels?.bottomLeft
|
|
},
|
|
bottomRight: {
|
|
...defaults.viewportPanels.bottomRight,
|
|
...preferences.viewportPanels?.bottomRight
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
function createSceneEditorPanelPreferencesFromViewportPanels(
|
|
viewportPanels: Record<ViewportPanelId, ViewportPanelState>
|
|
): Record<ViewportPanelId, SceneEditorPanelPreferences> {
|
|
return {
|
|
topLeft: {
|
|
viewMode: viewportPanels.topLeft.viewMode,
|
|
displayMode: viewportPanels.topLeft.displayMode
|
|
},
|
|
topRight: {
|
|
viewMode: viewportPanels.topRight.viewMode,
|
|
displayMode: viewportPanels.topRight.displayMode
|
|
},
|
|
bottomLeft: {
|
|
viewMode: viewportPanels.bottomLeft.viewMode,
|
|
displayMode: viewportPanels.bottomLeft.displayMode
|
|
},
|
|
bottomRight: {
|
|
viewMode: viewportPanels.bottomRight.viewMode,
|
|
displayMode: viewportPanels.bottomRight.displayMode
|
|
}
|
|
};
|
|
}
|
|
|
|
function applySceneEditorPreferencesToViewportPanels(
|
|
preferences: SceneEditorPreferences,
|
|
currentViewportPanels?: Record<ViewportPanelId, ViewportPanelState>
|
|
): Record<ViewportPanelId, ViewportPanelState> {
|
|
const fallbackPanels =
|
|
currentViewportPanels ?? createDefaultViewportLayoutState().panels;
|
|
|
|
return {
|
|
topLeft: {
|
|
...fallbackPanels.topLeft,
|
|
viewMode: preferences.viewportPanels.topLeft.viewMode,
|
|
displayMode: preferences.viewportPanels.topLeft.displayMode
|
|
},
|
|
topRight: {
|
|
...fallbackPanels.topRight,
|
|
viewMode: preferences.viewportPanels.topRight.viewMode,
|
|
displayMode: preferences.viewportPanels.topRight.displayMode
|
|
},
|
|
bottomLeft: {
|
|
...fallbackPanels.bottomLeft,
|
|
viewMode: preferences.viewportPanels.bottomLeft.viewMode,
|
|
displayMode: preferences.viewportPanels.bottomLeft.displayMode
|
|
},
|
|
bottomRight: {
|
|
...fallbackPanels.bottomRight,
|
|
viewMode: preferences.viewportPanels.bottomRight.viewMode,
|
|
displayMode: preferences.viewportPanels.bottomRight.displayMode
|
|
}
|
|
};
|
|
}
|
|
|
|
export class EditorStore {
|
|
private projectDocument: ProjectDocument;
|
|
private document: SceneDocument;
|
|
private commandTargetSceneId: string | null = null;
|
|
private selection: EditorSelection = { kind: "none" };
|
|
private activeSelectionId: string | null = null;
|
|
private whiteboxSelectionMode: WhiteboxSelectionMode = "object";
|
|
private whiteboxSnapEnabled = true;
|
|
private whiteboxSnapStep = 1;
|
|
private toolMode: ToolMode = "select";
|
|
private viewportGridVisible = true;
|
|
private viewportLayoutMode: ViewportLayoutMode;
|
|
private activeViewportPanelId: ViewportPanelId;
|
|
private viewportPanels: Record<ViewportPanelId, ViewportPanelState>;
|
|
private viewportQuadSplit: ViewportQuadSplit;
|
|
private viewportTransientState = createDefaultViewportTransientState();
|
|
private previousEditingToolMode: Exclude<ToolMode, "play"> = "select";
|
|
private readonly history = new CommandHistory();
|
|
private readonly listeners = new Set<EditorStoreListener>();
|
|
private readonly storage: KeyValueStorage | null;
|
|
private readonly storageKey: string;
|
|
private lastCommandLabel: string | null = null;
|
|
private snapshot: EditorStoreState;
|
|
|
|
private readonly commandContext: CommandContext = {
|
|
getDocument: () =>
|
|
createSceneDocumentFromProject(
|
|
this.projectDocument,
|
|
this.resolveCommandSceneId()
|
|
),
|
|
setDocument: (document) => {
|
|
this.projectDocument = applySceneDocumentToProject(
|
|
this.projectDocument,
|
|
this.resolveCommandSceneId(),
|
|
document
|
|
);
|
|
this.syncActiveSceneDocument();
|
|
},
|
|
getProjectDocument: () => this.projectDocument,
|
|
setProjectDocument: (document) => {
|
|
this.projectDocument = document;
|
|
this.syncActiveSceneDocument();
|
|
},
|
|
getSelection: () => this.selection,
|
|
setSelection: (selection) => {
|
|
this.selection = selection;
|
|
this.activeSelectionId = getSelectionDefaultActiveId(selection);
|
|
},
|
|
getToolMode: () => this.toolMode,
|
|
setToolMode: (toolMode) => {
|
|
this.toolMode = toolMode;
|
|
}
|
|
};
|
|
|
|
constructor(options: EditorStoreOptions = {}) {
|
|
const initialViewportLayoutState = cloneViewportLayoutState(
|
|
options.initialViewportLayoutState ?? createDefaultViewportLayoutState()
|
|
);
|
|
|
|
this.projectDocument = applyStarterEnvironmentAssetsToProjectDocument(
|
|
options.initialProjectDocument ??
|
|
(options.initialDocument !== undefined
|
|
? createProjectDocumentFromSceneDocument(options.initialDocument)
|
|
: createEmptyProjectDocument())
|
|
);
|
|
this.document = createSceneDocumentFromProject(this.projectDocument);
|
|
this.viewportLayoutMode = initialViewportLayoutState.layoutMode;
|
|
this.activeViewportPanelId = initialViewportLayoutState.activePanelId;
|
|
this.viewportPanels = initialViewportLayoutState.panels;
|
|
this.viewportQuadSplit = initialViewportLayoutState.viewportQuadSplit;
|
|
this.applyActiveSceneEditorPreferences();
|
|
this.storage = options.storage ?? null;
|
|
this.storageKey = options.storageKey ?? DEFAULT_SCENE_DRAFT_STORAGE_KEY;
|
|
this.snapshot = this.createSnapshot();
|
|
}
|
|
|
|
subscribe = (listener: EditorStoreListener) => {
|
|
this.listeners.add(listener);
|
|
|
|
return () => {
|
|
this.listeners.delete(listener);
|
|
};
|
|
};
|
|
|
|
getState = (): EditorStoreState => this.snapshot;
|
|
|
|
setToolMode(toolMode: ToolMode) {
|
|
if (this.toolMode === toolMode) {
|
|
return;
|
|
}
|
|
|
|
if (toolMode !== "play") {
|
|
this.previousEditingToolMode = toolMode;
|
|
}
|
|
|
|
this.toolMode = toolMode;
|
|
|
|
if (
|
|
!isViewportToolPreviewCompatible(
|
|
toolMode,
|
|
this.viewportTransientState.toolPreview
|
|
)
|
|
) {
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
toolPreview: createDefaultViewportTransientState().toolPreview
|
|
};
|
|
}
|
|
|
|
if (
|
|
toolMode !== "select" &&
|
|
this.viewportTransientState.transformSession.kind !== "none"
|
|
) {
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: createInactiveTransformSession()
|
|
};
|
|
}
|
|
|
|
this.emit();
|
|
}
|
|
|
|
setViewportLayoutMode(viewportLayoutMode: ViewportLayoutMode) {
|
|
if (this.viewportLayoutMode === viewportLayoutMode) {
|
|
return;
|
|
}
|
|
|
|
this.viewportLayoutMode = viewportLayoutMode;
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
viewportLayoutMode
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setActiveViewportPanel(panelId: ViewportPanelId) {
|
|
if (this.activeViewportPanelId === panelId) {
|
|
return;
|
|
}
|
|
|
|
this.activeViewportPanelId = panelId;
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
activeViewportPanelId: panelId
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setViewportPanelViewMode(
|
|
panelId: ViewportPanelId,
|
|
viewMode: ViewportViewMode
|
|
) {
|
|
if (this.viewportPanels[panelId].viewMode === viewMode) {
|
|
return;
|
|
}
|
|
|
|
this.viewportPanels = {
|
|
...this.viewportPanels,
|
|
[panelId]: {
|
|
...this.viewportPanels[panelId],
|
|
viewMode
|
|
}
|
|
};
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
viewportPanels: createSceneEditorPanelPreferencesFromViewportPanels(
|
|
this.viewportPanels
|
|
)
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setViewportPanelDisplayMode(
|
|
panelId: ViewportPanelId,
|
|
displayMode: ViewportDisplayMode
|
|
) {
|
|
if (this.viewportPanels[panelId].displayMode === displayMode) {
|
|
return;
|
|
}
|
|
|
|
this.viewportPanels = {
|
|
...this.viewportPanels,
|
|
[panelId]: {
|
|
...this.viewportPanels[panelId],
|
|
displayMode
|
|
}
|
|
};
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
viewportPanels: createSceneEditorPanelPreferencesFromViewportPanels(
|
|
this.viewportPanels
|
|
)
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setViewportPanelCameraState(
|
|
panelId: ViewportPanelId,
|
|
cameraState: ViewportPanelCameraState
|
|
) {
|
|
const previousCameraState = this.viewportPanels[panelId].cameraState;
|
|
const cameraStatesEqual = areViewportPanelCameraStatesEqual(
|
|
previousCameraState,
|
|
cameraState
|
|
);
|
|
|
|
traceUpdateLoopEvent("EditorStore.setViewportPanelCameraState", {
|
|
panelId,
|
|
previousCameraState:
|
|
summarizeUpdateLoopCameraState(previousCameraState),
|
|
nextCameraState: summarizeUpdateLoopCameraState(cameraState),
|
|
equalityGuardConsideredDifferent: !cameraStatesEqual,
|
|
deltas: summarizeUpdateLoopCameraStateDeltas(
|
|
previousCameraState,
|
|
cameraState
|
|
)
|
|
});
|
|
|
|
if (cameraStatesEqual) {
|
|
return;
|
|
}
|
|
|
|
this.viewportPanels = {
|
|
...this.viewportPanels,
|
|
[panelId]: {
|
|
...this.viewportPanels[panelId],
|
|
cameraState: cloneViewportPanelCameraState(cameraState)
|
|
}
|
|
};
|
|
this.emit("setViewportPanelCameraState");
|
|
}
|
|
|
|
setViewportQuadSplit(viewportQuadSplit: ViewportQuadSplit) {
|
|
if (
|
|
this.viewportQuadSplit.x === viewportQuadSplit.x &&
|
|
this.viewportQuadSplit.y === viewportQuadSplit.y
|
|
) {
|
|
return;
|
|
}
|
|
|
|
this.viewportQuadSplit = {
|
|
x: viewportQuadSplit.x,
|
|
y: viewportQuadSplit.y
|
|
};
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
viewportQuadSplit: {
|
|
...this.viewportQuadSplit
|
|
}
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setWhiteboxSnapEnabled(enabled: boolean) {
|
|
if (this.whiteboxSnapEnabled === enabled) {
|
|
return;
|
|
}
|
|
|
|
this.whiteboxSnapEnabled = enabled;
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
whiteboxSnapEnabled: enabled
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setWhiteboxSnapStep(step: number) {
|
|
if (!Number.isFinite(step) || step <= 0 || this.whiteboxSnapStep === step) {
|
|
return;
|
|
}
|
|
|
|
this.whiteboxSnapStep = step;
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
whiteboxSnapStep: step
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setViewportGridVisible(visible: boolean) {
|
|
if (this.viewportGridVisible === visible) {
|
|
return;
|
|
}
|
|
|
|
this.viewportGridVisible = visible;
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
viewportGridVisible: visible
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
setViewportToolPreview(toolPreview: ViewportToolPreview) {
|
|
const nextToolPreview = cloneViewportToolPreview(toolPreview);
|
|
|
|
if (
|
|
areViewportToolPreviewsEqual(
|
|
this.viewportTransientState.toolPreview,
|
|
nextToolPreview
|
|
)
|
|
) {
|
|
return;
|
|
}
|
|
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
toolPreview: nextToolPreview
|
|
};
|
|
this.emit();
|
|
}
|
|
|
|
clearViewportToolPreview(sourcePanelId?: ViewportPanelId) {
|
|
const currentToolPreview = this.viewportTransientState.toolPreview;
|
|
|
|
if (currentToolPreview.kind === "none") {
|
|
return;
|
|
}
|
|
|
|
if (
|
|
sourcePanelId !== undefined &&
|
|
currentToolPreview.sourcePanelId !== sourcePanelId
|
|
) {
|
|
return;
|
|
}
|
|
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
toolPreview: createDefaultViewportTransientState().toolPreview
|
|
};
|
|
this.emit();
|
|
}
|
|
|
|
setTransformSession(transformSession: TransformSessionState) {
|
|
const nextTransformSession = cloneTransformSession(transformSession);
|
|
|
|
if (
|
|
areTransformSessionsEqual(
|
|
this.viewportTransientState.transformSession,
|
|
nextTransformSession
|
|
)
|
|
) {
|
|
return;
|
|
}
|
|
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: nextTransformSession
|
|
};
|
|
this.emit();
|
|
}
|
|
|
|
clearTransformSession() {
|
|
if (this.viewportTransientState.transformSession.kind === "none") {
|
|
return;
|
|
}
|
|
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: createInactiveTransformSession()
|
|
};
|
|
this.emit();
|
|
}
|
|
|
|
setTransformAxisConstraint(
|
|
axisConstraint: TransformAxis | null,
|
|
axisConstraintSpace: TransformAxisSpace = "world"
|
|
) {
|
|
if (this.viewportTransientState.transformSession.kind !== "active") {
|
|
return;
|
|
}
|
|
|
|
if (
|
|
this.viewportTransientState.transformSession.axisConstraint ===
|
|
axisConstraint &&
|
|
this.viewportTransientState.transformSession.axisConstraintSpace ===
|
|
axisConstraintSpace
|
|
) {
|
|
return;
|
|
}
|
|
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: {
|
|
...(cloneTransformSession(
|
|
this.viewportTransientState.transformSession
|
|
) as Extract<TransformSessionState, { kind: "active" }>),
|
|
axisConstraint,
|
|
axisConstraintSpace
|
|
}
|
|
};
|
|
this.emit();
|
|
}
|
|
|
|
setViewportViewMode(viewportViewMode: ViewportViewMode) {
|
|
this.setViewportPanelViewMode(this.activeViewportPanelId, viewportViewMode);
|
|
}
|
|
|
|
enterPlayMode() {
|
|
if (this.toolMode === "play") {
|
|
return;
|
|
}
|
|
|
|
this.previousEditingToolMode = this.toolMode;
|
|
this.toolMode = "play";
|
|
|
|
if (this.viewportTransientState.toolPreview.kind !== "none") {
|
|
this.viewportTransientState = createDefaultViewportTransientState();
|
|
}
|
|
|
|
this.emit();
|
|
}
|
|
|
|
exitPlayMode() {
|
|
if (this.toolMode !== "play") {
|
|
return;
|
|
}
|
|
|
|
this.toolMode = this.previousEditingToolMode;
|
|
this.emit();
|
|
}
|
|
|
|
setSelection(selection: EditorSelection) {
|
|
if (
|
|
this.viewportTransientState.transformSession.kind === "active" &&
|
|
!areEditorSelectionsEqual(this.selection, selection)
|
|
) {
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: createInactiveTransformSession()
|
|
};
|
|
}
|
|
|
|
this.selection = selection;
|
|
this.activeSelectionId = getSelectionDefaultActiveId(selection);
|
|
this.emit();
|
|
}
|
|
|
|
setWhiteboxSelectionMode(mode: WhiteboxSelectionMode) {
|
|
if (this.whiteboxSelectionMode === mode) {
|
|
return;
|
|
}
|
|
|
|
if (this.viewportTransientState.transformSession.kind !== "none") {
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: createInactiveTransformSession()
|
|
};
|
|
}
|
|
|
|
this.whiteboxSelectionMode = mode;
|
|
this.selection = normalizeSelectionForWhiteboxSelectionMode(
|
|
this.selection,
|
|
mode
|
|
);
|
|
this.activeSelectionId = getSelectionDefaultActiveId(this.selection);
|
|
this.updateActiveSceneEditorPreferences((preferences) => ({
|
|
...preferences,
|
|
whiteboxSelectionMode: mode
|
|
}));
|
|
this.emit();
|
|
}
|
|
|
|
executeCommand(command: EditorCommand) {
|
|
if (this.viewportTransientState.transformSession.kind !== "none") {
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: createInactiveTransformSession()
|
|
};
|
|
}
|
|
|
|
this.history.execute(this.createScopedCommand(command), this.commandContext);
|
|
this.lastCommandLabel = command.label;
|
|
this.emit();
|
|
}
|
|
|
|
undo(): boolean {
|
|
let clearedTransformSession = false;
|
|
|
|
if (this.viewportTransientState.transformSession.kind !== "none") {
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: createInactiveTransformSession()
|
|
};
|
|
clearedTransformSession = true;
|
|
}
|
|
|
|
const command = this.history.undo(this.commandContext);
|
|
|
|
if (command === null) {
|
|
if (clearedTransformSession) {
|
|
this.emit();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
this.lastCommandLabel = `Undid ${command.label}`;
|
|
this.emit();
|
|
return true;
|
|
}
|
|
|
|
redo(): boolean {
|
|
let clearedTransformSession = false;
|
|
|
|
if (this.viewportTransientState.transformSession.kind !== "none") {
|
|
this.viewportTransientState = {
|
|
...this.viewportTransientState,
|
|
transformSession: createInactiveTransformSession()
|
|
};
|
|
clearedTransformSession = true;
|
|
}
|
|
|
|
const command = this.history.redo(this.commandContext);
|
|
|
|
if (command === null) {
|
|
if (clearedTransformSession) {
|
|
this.emit();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
this.lastCommandLabel = `Redid ${command.label}`;
|
|
this.emit();
|
|
return true;
|
|
}
|
|
|
|
replaceDocument(
|
|
document: SceneDocument | ProjectDocument,
|
|
resetHistory = true
|
|
) {
|
|
this.projectDocument = isProjectDocument(document)
|
|
? document
|
|
: createProjectDocumentFromSceneDocument(document);
|
|
this.syncActiveSceneDocument();
|
|
this.commandTargetSceneId = null;
|
|
this.selection = { kind: "none" };
|
|
this.activeSelectionId = null;
|
|
this.toolMode = "select";
|
|
this.previousEditingToolMode = "select";
|
|
this.viewportTransientState = createDefaultViewportTransientState();
|
|
|
|
if (resetHistory) {
|
|
this.history.clear();
|
|
this.lastCommandLabel = null;
|
|
}
|
|
|
|
this.emit();
|
|
}
|
|
|
|
saveDraft(): EditorDraftSaveResult {
|
|
if (this.storage === null) {
|
|
return {
|
|
status: "error",
|
|
message: "Browser local storage is unavailable."
|
|
};
|
|
}
|
|
|
|
return saveSceneDocumentDraft(
|
|
this.storage,
|
|
this.projectDocument,
|
|
this.createViewportLayoutState(),
|
|
this.storageKey
|
|
);
|
|
}
|
|
|
|
loadDraft(): EditorDraftLoadResult {
|
|
if (this.storage === null) {
|
|
return {
|
|
status: "error",
|
|
message: "Browser local storage is unavailable."
|
|
};
|
|
}
|
|
|
|
const draftResult = loadSceneDocumentDraft(this.storage, this.storageKey);
|
|
|
|
if (draftResult.status !== "loaded") {
|
|
return draftResult;
|
|
}
|
|
|
|
this.replaceDocument(draftResult.document);
|
|
|
|
if (draftResult.viewportLayoutState !== null) {
|
|
this.applyViewportLayoutState(draftResult.viewportLayoutState);
|
|
this.emit();
|
|
}
|
|
|
|
return draftResult;
|
|
}
|
|
|
|
getProjectDocumentDraftSnapshot(): ProjectDocument {
|
|
return this.projectDocument;
|
|
}
|
|
|
|
getViewportLayoutDraftSnapshot(): ViewportLayoutState {
|
|
return this.createViewportLayoutState();
|
|
}
|
|
|
|
exportDocumentJson(): string {
|
|
return serializeProjectDocument(this.projectDocument);
|
|
}
|
|
|
|
importDocumentJson(source: string): ProjectDocument {
|
|
const document = parseProjectDocumentJson(source);
|
|
this.replaceDocument(document);
|
|
return document;
|
|
}
|
|
|
|
private emit(reason?: string) {
|
|
traceUpdateLoopEvent("EditorStore.emit", {
|
|
reason: reason ?? "unspecified",
|
|
listenerCount: this.listeners.size
|
|
});
|
|
|
|
this.snapshot = this.createSnapshot();
|
|
|
|
for (const listener of this.listeners) {
|
|
listener();
|
|
}
|
|
}
|
|
|
|
private createViewportLayoutState(): ViewportLayoutState {
|
|
return cloneViewportLayoutState({
|
|
layoutMode: this.viewportLayoutMode,
|
|
activePanelId: this.activeViewportPanelId,
|
|
panels: this.viewportPanels,
|
|
viewportQuadSplit: this.viewportQuadSplit
|
|
});
|
|
}
|
|
|
|
private applyViewportLayoutState(viewportLayoutState: ViewportLayoutState) {
|
|
const nextViewportLayoutState =
|
|
cloneViewportLayoutState(viewportLayoutState);
|
|
|
|
this.viewportLayoutMode = nextViewportLayoutState.layoutMode;
|
|
this.activeViewportPanelId = nextViewportLayoutState.activePanelId;
|
|
this.viewportPanels = nextViewportLayoutState.panels;
|
|
this.viewportQuadSplit = nextViewportLayoutState.viewportQuadSplit;
|
|
}
|
|
|
|
private createSnapshot(): EditorStoreState {
|
|
return {
|
|
projectDocument: this.projectDocument,
|
|
activeSceneId: this.projectDocument.activeSceneId,
|
|
document: this.document,
|
|
selection: this.selection,
|
|
activeSelectionId: this.activeSelectionId,
|
|
whiteboxSelectionMode: this.whiteboxSelectionMode,
|
|
whiteboxSnapEnabled: this.whiteboxSnapEnabled,
|
|
whiteboxSnapStep: this.whiteboxSnapStep,
|
|
toolMode: this.toolMode,
|
|
viewportGridVisible: this.viewportGridVisible,
|
|
viewportLayoutMode: this.viewportLayoutMode,
|
|
activeViewportPanelId: this.activeViewportPanelId,
|
|
viewportPanels: this.viewportPanels,
|
|
viewportQuadSplit: this.viewportQuadSplit,
|
|
viewportTransientState: this.viewportTransientState,
|
|
canUndo: this.history.canUndo(),
|
|
canRedo: this.history.canRedo(),
|
|
lastCommandLabel: this.lastCommandLabel,
|
|
storageAvailable: this.storage !== null
|
|
};
|
|
}
|
|
|
|
private syncActiveSceneDocument() {
|
|
this.document = createSceneDocumentFromProject(this.projectDocument);
|
|
this.applyActiveSceneEditorPreferences();
|
|
}
|
|
|
|
private applyActiveSceneEditorPreferences() {
|
|
const preferences = getSceneEditorPreferences(this.projectDocument);
|
|
|
|
this.whiteboxSelectionMode = preferences.whiteboxSelectionMode;
|
|
this.whiteboxSnapEnabled = preferences.whiteboxSnapEnabled;
|
|
this.whiteboxSnapStep = preferences.whiteboxSnapStep;
|
|
this.viewportGridVisible = preferences.viewportGridVisible;
|
|
this.viewportLayoutMode = preferences.viewportLayoutMode;
|
|
this.activeViewportPanelId = preferences.activeViewportPanelId;
|
|
this.viewportQuadSplit = {
|
|
...preferences.viewportQuadSplit
|
|
};
|
|
this.viewportPanels = applySceneEditorPreferencesToViewportPanels(
|
|
preferences,
|
|
this.viewportPanels
|
|
);
|
|
}
|
|
|
|
private updateActiveSceneEditorPreferences(
|
|
updater: (preferences: SceneEditorPreferences) => SceneEditorPreferences
|
|
) {
|
|
const sceneId = this.projectDocument.activeSceneId;
|
|
const scene = this.projectDocument.scenes[sceneId];
|
|
|
|
if (scene === undefined) {
|
|
return;
|
|
}
|
|
|
|
this.projectDocument = {
|
|
...this.projectDocument,
|
|
scenes: {
|
|
...this.projectDocument.scenes,
|
|
[sceneId]: {
|
|
...scene,
|
|
editorPreferences: cloneSceneEditorPreferences(
|
|
updater(getSceneEditorPreferences(this.projectDocument, sceneId))
|
|
)
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
private resolveCommandSceneId(): string {
|
|
return this.commandTargetSceneId ?? this.projectDocument.activeSceneId;
|
|
}
|
|
|
|
private withCommandSceneScope<T>(sceneId: string, run: () => T): T {
|
|
const previousSceneId = this.commandTargetSceneId;
|
|
this.commandTargetSceneId = sceneId;
|
|
|
|
try {
|
|
return run();
|
|
} finally {
|
|
this.commandTargetSceneId = previousSceneId;
|
|
}
|
|
}
|
|
|
|
private createScopedCommand(command: EditorCommand): EditorCommand {
|
|
const sceneId = this.projectDocument.activeSceneId;
|
|
|
|
return {
|
|
...command,
|
|
execute: (context) =>
|
|
this.withCommandSceneScope(sceneId, () => command.execute(context)),
|
|
undo: (context) =>
|
|
this.withCommandSceneScope(sceneId, () => command.undo(context))
|
|
};
|
|
}
|
|
}
|
|
|
|
export function createEditorStore(options?: EditorStoreOptions): EditorStore {
|
|
return new EditorStore(options);
|
|
}
|
|
|
|
function isProjectDocument(
|
|
document: SceneDocument | ProjectDocument
|
|
): document is ProjectDocument {
|
|
return "scenes" in document && "activeSceneId" in document;
|
|
}
|