mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
209 lines
6.8 KiB
TypeScript
209 lines
6.8 KiB
TypeScript
import { ResizableWidget } from "../../core/gui/ResizableWidget";
|
|
import { create_element, el } from "../../core/gui/dom";
|
|
import { QuestEditorToolBar } from "./QuestEditorToolBar";
|
|
import GoldenLayout, { Container, ContentItem, ItemConfigType } from "golden-layout";
|
|
import { quest_editor_ui_persister } from "../persistence/QuestEditorUiPersister";
|
|
import { QuesInfoView } from "./QuesInfoView";
|
|
import "golden-layout/src/css/goldenlayout-base.css";
|
|
import "../../core/gui/golden_layout_theme.css";
|
|
import { NpcCountsView } from "./NpcCountsView";
|
|
import { QuestRendererView } from "./QuestRendererView";
|
|
import { AsmEditorView } from "./AsmEditorView";
|
|
import { EntityInfoView } from "./EntityInfoView";
|
|
import { gui_store, GuiTool } from "../../core/stores/GuiStore";
|
|
import { quest_editor_store } from "../stores/QuestEditorStore";
|
|
import Logger = require("js-logger");
|
|
|
|
const logger = Logger.get("quest_editor/gui/QuestEditorView");
|
|
|
|
// Don't change these values, as they are persisted in the user's browser.
|
|
const VIEW_TO_NAME = new Map<new () => ResizableWidget, string>([
|
|
[QuesInfoView, "quest_info"],
|
|
[NpcCountsView, "npc_counts"],
|
|
[QuestRendererView, "quest_renderer"],
|
|
[AsmEditorView, "asm_editor"],
|
|
[EntityInfoView, "entity_info"],
|
|
// [AddObjectView, "add_object"],
|
|
]);
|
|
|
|
const DEFAULT_LAYOUT_CONFIG = {
|
|
settings: {
|
|
showPopoutIcon: false,
|
|
showMaximiseIcon: false,
|
|
},
|
|
dimensions: {
|
|
headerHeight: 22,
|
|
},
|
|
labels: {
|
|
close: "Close",
|
|
maximise: "Maximise",
|
|
minimise: "Minimise",
|
|
popout: "Open in new window",
|
|
},
|
|
};
|
|
|
|
const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [
|
|
{
|
|
type: "row",
|
|
content: [
|
|
{
|
|
type: "stack",
|
|
width: 3,
|
|
content: [
|
|
{
|
|
title: "Info",
|
|
type: "component",
|
|
componentName: VIEW_TO_NAME.get(QuesInfoView),
|
|
isClosable: false,
|
|
},
|
|
{
|
|
title: "NPC Counts",
|
|
type: "component",
|
|
componentName: VIEW_TO_NAME.get(NpcCountsView),
|
|
isClosable: false,
|
|
},
|
|
],
|
|
},
|
|
{
|
|
type: "stack",
|
|
width: 9,
|
|
content: [
|
|
{
|
|
title: "3D View",
|
|
type: "component",
|
|
componentName: VIEW_TO_NAME.get(QuestRendererView),
|
|
isClosable: false,
|
|
},
|
|
{
|
|
title: "Script",
|
|
type: "component",
|
|
componentName: VIEW_TO_NAME.get(AsmEditorView),
|
|
isClosable: false,
|
|
},
|
|
],
|
|
},
|
|
{
|
|
title: "Entity",
|
|
type: "component",
|
|
componentName: VIEW_TO_NAME.get(EntityInfoView),
|
|
isClosable: false,
|
|
width: 2,
|
|
},
|
|
],
|
|
},
|
|
];
|
|
|
|
export class QuestEditorView extends ResizableWidget {
|
|
private readonly tool_bar_view = this.disposable(new QuestEditorToolBar());
|
|
|
|
private readonly layout_element = create_element("div", { class: "quest_editor_gl_container" });
|
|
private readonly layout: Promise<GoldenLayout>;
|
|
|
|
private readonly sub_views = new Map<string, ResizableWidget>();
|
|
|
|
constructor() {
|
|
super(el.div({ class: "quest_editor_QuestEditorView" }));
|
|
|
|
this.element.append(this.tool_bar_view.element, this.layout_element);
|
|
|
|
this.layout = this.init_golden_layout();
|
|
|
|
this.disposables(
|
|
gui_store.on_global_keydown(
|
|
GuiTool.QuestEditor,
|
|
"Ctrl-Alt-D",
|
|
() => (quest_editor_store.debug.val = !quest_editor_store.debug.val),
|
|
),
|
|
);
|
|
}
|
|
|
|
resize(width: number, height: number): this {
|
|
super.resize(width, height);
|
|
|
|
const layout_height = Math.max(0, height - this.tool_bar_view.height);
|
|
this.layout_element.style.width = `${width}px`;
|
|
this.layout_element.style.height = `${layout_height}px`;
|
|
this.layout.then(layout => layout.updateSize(width, layout_height));
|
|
|
|
return this;
|
|
}
|
|
|
|
dispose(): void {
|
|
super.dispose();
|
|
this.layout.then(layout => layout.destroy());
|
|
|
|
for (const view of this.sub_views.values()) {
|
|
view.dispose();
|
|
}
|
|
|
|
this.sub_views.clear();
|
|
}
|
|
|
|
private async init_golden_layout(): Promise<GoldenLayout> {
|
|
const content = await quest_editor_ui_persister.load_layout_config(
|
|
[...VIEW_TO_NAME.values()],
|
|
DEFAULT_LAYOUT_CONTENT,
|
|
);
|
|
|
|
try {
|
|
return this.attempt_gl_init({
|
|
...DEFAULT_LAYOUT_CONFIG,
|
|
content,
|
|
});
|
|
} catch (e) {
|
|
logger.warn("Couldn't instantiate golden layout with persisted layout.", e);
|
|
|
|
return this.attempt_gl_init({
|
|
...DEFAULT_LAYOUT_CONFIG,
|
|
content: DEFAULT_LAYOUT_CONTENT,
|
|
});
|
|
}
|
|
}
|
|
|
|
private attempt_gl_init(config: GoldenLayout.Config): GoldenLayout {
|
|
const layout = new GoldenLayout(config, this.layout_element);
|
|
const self = this;
|
|
|
|
try {
|
|
for (const [view_ctor, name] of VIEW_TO_NAME) {
|
|
layout.registerComponent(name, function(container: Container) {
|
|
const view = new view_ctor();
|
|
|
|
container.on("close", () => view.dispose());
|
|
container.on("resize", () =>
|
|
// Subtract 4 from height to work around bug in Golden Layout related to headerHeight.
|
|
view.resize(container.width, container.height - 4),
|
|
);
|
|
|
|
view.resize(container.width, container.height);
|
|
|
|
self.sub_views.set(name, view);
|
|
container.getElement().append(view.element);
|
|
});
|
|
}
|
|
|
|
layout.on("stateChanged", () => {
|
|
if (this.layout) {
|
|
quest_editor_ui_persister.persist_layout_config(layout.toConfig().content);
|
|
}
|
|
});
|
|
|
|
layout.on("stackCreated", (stack: ContentItem) => {
|
|
stack.on("activeContentItemChanged", (item: ContentItem) => {
|
|
if ("componentName" in item.config) {
|
|
const view = this.sub_views.get(item.config.componentName);
|
|
if (view) view.focus();
|
|
}
|
|
});
|
|
});
|
|
|
|
layout.init();
|
|
|
|
return layout;
|
|
} catch (e) {
|
|
layout.destroy();
|
|
throw e;
|
|
}
|
|
}
|
|
}
|