From 52376193ae4dfb42cff815955a1c0344cd39ed11 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Wed, 24 Jul 2019 16:44:17 +0200 Subject: [PATCH] Quest editor layout is now persisted. --- src/persistence/HuntMethodPersister.ts | 6 +- src/persistence/HuntOptimizerPersister.ts | 6 +- src/persistence/QuestEditorUiPersister.ts | 57 +++++++ src/ui/index.less | 2 +- src/ui/quest_editor/QuestEditorComponent.tsx | 149 +++++++++++-------- src/ui/quest_editor/Toolbar.tsx | 1 - 6 files changed, 155 insertions(+), 66 deletions(-) create mode 100644 src/persistence/QuestEditorUiPersister.ts diff --git a/src/persistence/HuntMethodPersister.ts b/src/persistence/HuntMethodPersister.ts index 3c4b83e8..a26f2a6f 100644 --- a/src/persistence/HuntMethodPersister.ts +++ b/src/persistence/HuntMethodPersister.ts @@ -1,6 +1,8 @@ import { Persister } from "./Persister"; import { Server, HuntMethod } from "../domain"; +const METHOD_USER_TIMES_KEY = "HuntMethodStore.methodUserTimes"; + class HuntMethodPersister extends Persister { persist_method_user_times(hunt_methods: HuntMethod[], server: Server): void { const user_times: PersistedUserTimes = {}; @@ -11,7 +13,7 @@ class HuntMethodPersister extends Persister { } } - this.persist_for_server(server, "HuntMethodStore.methodUserTimes", user_times); + this.persist_for_server(server, METHOD_USER_TIMES_KEY, user_times); } async load_method_user_times( @@ -20,7 +22,7 @@ class HuntMethodPersister extends Persister { ): Promise { const user_times = await this.load_for_server( server, - "HuntMethodStore.methodUserTimes" + METHOD_USER_TIMES_KEY ); if (user_times) { diff --git a/src/persistence/HuntOptimizerPersister.ts b/src/persistence/HuntOptimizerPersister.ts index 90d180f0..6b5293ca 100644 --- a/src/persistence/HuntOptimizerPersister.ts +++ b/src/persistence/HuntOptimizerPersister.ts @@ -3,11 +3,13 @@ import { WantedItem } from "../stores/HuntOptimizerStore"; import { item_type_stores } from "../stores/ItemTypeStore"; import { Persister } from "./Persister"; +const WANTED_ITEMS_KEY = "HuntOptimizerStore.wantedItems"; + class HuntOptimizerPersister extends Persister { persist_wanted_items(server: Server, wanted_items: WantedItem[]): void { this.persist_for_server( server, - "HuntOptimizerStore.wantedItems", + WANTED_ITEMS_KEY, wanted_items.map( ({ item_type, amount }): PersistedWantedItem => ({ itemTypeId: item_type.id, @@ -22,7 +24,7 @@ class HuntOptimizerPersister extends Persister { const persisted_wanted_items = await this.load_for_server( server, - "HuntOptimizerStore.wantedItems" + WANTED_ITEMS_KEY ); const wanted_items: WantedItem[] = []; diff --git a/src/persistence/QuestEditorUiPersister.ts b/src/persistence/QuestEditorUiPersister.ts new file mode 100644 index 00000000..d02a6cba --- /dev/null +++ b/src/persistence/QuestEditorUiPersister.ts @@ -0,0 +1,57 @@ +import { Persister } from "./Persister"; +import { throttle } from "lodash"; + +const LAYOUT_CONFIG_KEY = "QuestEditorUiPersister.layout_config"; + +class QuestEditorUiPersister extends Persister { + persist_layout_config = throttle( + (config: any) => { + this.persist(LAYOUT_CONFIG_KEY, config); + }, + 1000, + { leading: false, trailing: true } + ); + + async load_layout_config(components: string[], default_config: any): Promise { + const config = await this.load(LAYOUT_CONFIG_KEY); + + let valid = this.verify_layout_config(config, new Set(components)); + + if (valid) { + return config; + } else { + return default_config; + } + } + + private verify_layout_config( + config: any, + components: Set, + found: Set = new Set(), + first: boolean = true + ): boolean { + if (!config) { + return false; + } + + if (config.component) { + if (!components.has(config.component)) { + return false; + } else { + found.add(config.component); + } + } + + if (config.content) { + for (const child of config.content) { + if (!this.verify_layout_config(child, components, found, false)) { + return false; + } + } + } + + return first ? components.size === found.size : true; + } +} + +export const quest_editor_ui_persister = new QuestEditorUiPersister(); diff --git a/src/ui/index.less b/src/ui/index.less index 8fcf2940..efc59d69 100644 --- a/src/ui/index.less +++ b/src/ui/index.less @@ -38,7 +38,7 @@ } body { - overflow: hidden; + overflow: hidden; // Necessary for golden layout. } // react-virtualized diff --git a/src/ui/quest_editor/QuestEditorComponent.tsx b/src/ui/quest_editor/QuestEditorComponent.tsx index b3d5c3a6..f040292c 100644 --- a/src/ui/quest_editor/QuestEditorComponent.tsx +++ b/src/ui/quest_editor/QuestEditorComponent.tsx @@ -9,6 +9,64 @@ import { QuestInfoComponent } from "./QuestInfoComponent"; import { QuestRendererComponent } from "./QuestRendererComponent"; import { ScriptEditorComponent } from "./ScriptEditorComponent"; import { Toolbar } from "./Toolbar"; +import { quest_editor_ui_persister } from "../../persistence/QuestEditorUiPersister"; +import Logger from "js-logger"; + +const logger = Logger.get("ui/quest_editor/QuestEditorComponent"); + +const DEFAULT_LAYOUT_CONFIG = { + settings: { + showPopoutIcon: false, + }, + dimensions: { + headerHeight: 28, + }, + labels: { + close: "Close", + maximise: "Maximise", + minimise: "Minimise", + popout: "Open in new window", + }, + content: [ + { + type: "row", + content: [ + { + title: "Info", + type: "react-component", + component: QuestInfoComponent.name, + isClosable: false, + width: 3, + }, + { + type: "stack", + width: 9, + content: [ + { + title: "3D View", + type: "react-component", + component: QuestRendererComponent.name, + isClosable: false, + }, + { + title: "Script", + type: "react-component", + component: ScriptEditorComponent.name, + isClosable: false, + }, + ], + }, + { + title: "Entity", + type: "react-component", + component: EntityInfoComponent.name, + isClosable: false, + width: 2, + }, + ], + }, + ], +}; @observer export class QuestEditorComponent extends Component { @@ -20,68 +78,39 @@ export class QuestEditorComponent extends Component { window.addEventListener("resize", this.resize); - setTimeout(() => { + setTimeout(async () => { if (this.layout_element.current && !this.layout) { - this.layout = new GoldenLayout( - { - settings: { - showPopoutIcon: false, - }, - dimensions: { - headerHeight: 28, - }, - labels: { - close: "Close", - maximise: "Maximise", - minimise: "Minimise", - popout: "Open in new window", - }, - content: [ - { - type: "row", - content: [ - { - title: "Info", - type: "react-component", - component: "QuestInfoComponent", - isClosable: false, - width: 3, - }, - { - type: "stack", - width: 9, - content: [ - { - title: "3D View", - type: "react-component", - component: "QuestRendererComponent", - isClosable: false, - }, - { - title: "Script", - type: "react-component", - component: "ScriptEditorComponent", - isClosable: false, - }, - ], - }, - { - title: "Entity", - type: "react-component", - component: "EntityInfoComponent", - isClosable: false, - width: 2, - }, - ], - }, - ], - }, - this.layout_element.current + const config = await quest_editor_ui_persister.load_layout_config( + [ + QuestInfoComponent.name, + QuestRendererComponent.name, + EntityInfoComponent.name, + ScriptEditorComponent.name, + ], + DEFAULT_LAYOUT_CONFIG ); - this.layout.registerComponent("QuestInfoComponent", QuestInfoComponent); - this.layout.registerComponent("QuestRendererComponent", QuestRendererComponent); - this.layout.registerComponent("EntityInfoComponent", EntityInfoComponent); - this.layout.registerComponent("ScriptEditorComponent", ScriptEditorComponent); + + try { + this.layout = new GoldenLayout(config, this.layout_element.current); + } catch (e) { + logger.warn("Couldn't initialize golden layout with persisted layout.", e); + + this.layout = new GoldenLayout( + DEFAULT_LAYOUT_CONFIG, + this.layout_element.current + ); + } + + this.layout.registerComponent(QuestInfoComponent.name, QuestInfoComponent); + this.layout.registerComponent(QuestRendererComponent.name, QuestRendererComponent); + this.layout.registerComponent(EntityInfoComponent.name, EntityInfoComponent); + this.layout.registerComponent(ScriptEditorComponent.name, ScriptEditorComponent); + this.layout.on("stateChanged", () => { + if (this.layout) { + quest_editor_ui_persister.persist_layout_config(this.layout.toConfig()); + } + }); + this.layout.init(); } }, 0); diff --git a/src/ui/quest_editor/Toolbar.tsx b/src/ui/quest_editor/Toolbar.tsx index 1ade66f2..b061046b 100644 --- a/src/ui/quest_editor/Toolbar.tsx +++ b/src/ui/quest_editor/Toolbar.tsx @@ -22,7 +22,6 @@ export class Toolbar extends Component { overlay={ Episode I - Episode II Episode IV }