import { Persister } from "../../core/Persister"; import { throttle } from "lodash"; import { ComponentConfig, ItemConfigType } from "golden-layout"; const LAYOUT_CONFIG_KEY = "QuestEditorUiPersister.layout_config"; export class QuestEditorUiPersister extends Persister { persist_layout_config = throttle( (config: ItemConfigType[]) => { this.persist(LAYOUT_CONFIG_KEY, this.to_persisted_item_config(config)); }, 500, { leading: false, trailing: true }, ); async load_layout_config( default_config: ItemConfigType[], ): Promise { const config = await this.load(LAYOUT_CONFIG_KEY); if (config) { const components = this.extract_components(default_config); const verified_config = this.sanitized_layout_config( this.from_persisted_item_config(config), components, ); if (verified_config) { return verified_config; } } return undefined; } private sanitized_layout_config( config: ItemConfigType[], components: Map, ): ItemConfigType[] | undefined { const found = new Set(); const sanitized_config = config.map(child => this.sanitize_layout_child(child, components, found), ); if (found.size !== components.size) { // A component was added, use default layout instead of persisted layout. return undefined; } // Filter out removed components. return sanitized_config.filter(item => item) as ItemConfigType[]; } /** * Removed old components and adds titles and ids to current components. * Modifies the given ItemConfigType object. */ private sanitize_layout_child( config: ItemConfigType, components: Map, found: Set, ): ItemConfigType | undefined { if (!config) { return undefined; } if (config.type === "component" && "componentName" in config) { const component = components.get(config.componentName); // Remove deprecated components. if (!component) { return undefined; } found.add(config.componentName); config.id = component.id; config.title = component.title; } if (config.content) { config.content = config.content .map(child => this.sanitize_layout_child(child, components, found)) .filter(item => item) as ItemConfigType[]; } return config; } private extract_components( config: ItemConfigType[], map: Map = new Map(), ): Map { for (const child of config) { if ("componentName" in child) { map.set(child.componentName, child); } if (child.content) { this.extract_components(child.content, map); } } return map; } private to_persisted_item_config(config: ItemConfigType[]): PersistedItemConfig[] { return config.map(item => ({ id: item.id, type: item.type, componentName: "componentName" in item ? item.componentName : undefined, width: item.width, height: item.height, content: item.content && this.to_persisted_item_config(item.content), })); } /** * This simply makes a copy to ensure legacy properties are removed. */ private from_persisted_item_config(config: PersistedItemConfig[]): ItemConfigType[] { return config.map(item => ({ id: item.id, type: item.type, componentName: item.componentName, width: item.width, height: item.height, content: item.content && this.from_persisted_item_config(item.content), isClosable: false, })); } } type PersistedItemConfig = { id?: string | string[]; type: string; componentName?: string; width?: number; height?: number; content?: PersistedItemConfig[]; };