From ff8f02fe5bf900c2d41d94217786fa62ca043111 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Thu, 10 Oct 2019 23:11:52 +0200 Subject: [PATCH] Refactored events model and added a basic view for it behind a feature flag. --- src/core/data_formats/parsing/quest/dat.ts | 20 +- src/core/data_formats/parsing/quest/index.ts | 6 +- src/core/gui/Widget.ts | 1 + src/core/gui/dom.ts | 82 ++++--- src/core/gui/golden_layout_theme.css | 4 + src/quest_editor/gui/EventsView.css | 33 +++ src/quest_editor/gui/EventsView.ts | 71 ++++++ src/quest_editor/gui/QuestEditorView.ts | 48 ++-- .../model/QuestEventActionModel.ts | 10 - .../model/QuestEventChainModel.ts | 15 ++ src/quest_editor/model/QuestModel.ts | 14 +- src/quest_editor/stores/QuestEditorStore.ts | 175 +-------------- src/quest_editor/stores/model_conversion.ts | 205 ++++++++++++++++++ 13 files changed, 433 insertions(+), 251 deletions(-) create mode 100644 src/quest_editor/gui/EventsView.css create mode 100644 src/quest_editor/gui/EventsView.ts create mode 100644 src/quest_editor/model/QuestEventChainModel.ts create mode 100644 src/quest_editor/stores/model_conversion.ts diff --git a/src/core/data_formats/parsing/quest/dat.ts b/src/core/data_formats/parsing/quest/dat.ts index c5ac7627..cfe33401 100644 --- a/src/core/data_formats/parsing/quest/dat.ts +++ b/src/core/data_formats/parsing/quest/dat.ts @@ -55,14 +55,14 @@ export enum DatEventActionType { SpawnNpcs = 0x8, Unlock = 0xa, Lock = 0xb, - SpawnWave = 0xc, + TriggerEvent = 0xc, } export type DatEventAction = | DatEventActionSpawnNpcs | DatEventActionUnlock | DatEventActionLock - | DatEventActionSpawnWave; + | DatEventActionTriggerEvent; export type DatEventActionSpawnNpcs = { readonly type: DatEventActionType.SpawnNpcs; @@ -80,9 +80,9 @@ export type DatEventActionLock = { readonly door_id: number; }; -export type DatEventActionSpawnWave = { - readonly type: DatEventActionType.SpawnWave; - readonly wave_id: number; +export type DatEventActionTriggerEvent = { + readonly type: DatEventActionType.TriggerEvent; + readonly event_id: number; }; export type DatUnknown = { @@ -344,10 +344,10 @@ function parse_wave_actions(cursor: Cursor): DatEventAction[] { }); break; - case DatEventActionType.SpawnWave: + case DatEventActionType.TriggerEvent: actions.push({ - type: DatEventActionType.SpawnWave, - wave_id: cursor.u32(), + type: DatEventActionType.TriggerEvent, + event_id: cursor.u32(), }); break; @@ -519,8 +519,8 @@ function write_waves(cursor: WritableCursor, waves: readonly DatEvent[]): void { cursor.write_u16(action.door_id); break; - case DatEventActionType.SpawnWave: - cursor.write_u32(action.wave_id); + case DatEventActionType.TriggerEvent: + cursor.write_u32(action.event_id); break; default: diff --git a/src/core/data_formats/parsing/quest/index.ts b/src/core/data_formats/parsing/quest/index.ts index 0cfda0d3..11a17ce4 100644 --- a/src/core/data_formats/parsing/quest/index.ts +++ b/src/core/data_formats/parsing/quest/index.ts @@ -32,7 +32,7 @@ export type Quest = { readonly episode: Episode; readonly objects: readonly QuestObject[]; readonly npcs: readonly QuestNpc[]; - readonly waves: readonly QuestEvent[]; + readonly events: readonly QuestEvent[]; /** * (Partial) raw DAT data that can't be parsed yet by Phantasmal. */ @@ -127,7 +127,7 @@ export function parse_quest(cursor: Cursor, lenient: boolean = false): Quest | u episode, objects, npcs: parse_npc_data(episode, dat.npcs), - waves: dat.waves, + events: dat.waves, dat_unknowns: dat.unknowns, object_code: bin.object_code, shop_items: bin.shop_items, @@ -139,7 +139,7 @@ export function write_quest_qst(quest: Quest, file_name: string): ArrayBuffer { const dat = write_dat({ objs: objects_to_dat_data(quest.objects), npcs: npcs_to_dat_data(quest.npcs), - waves: quest.waves, + waves: quest.events, unknowns: quest.dat_unknowns, }); const bin = write_bin({ diff --git a/src/core/gui/Widget.ts b/src/core/gui/Widget.ts index 2cfb3c96..8b197e2c 100644 --- a/src/core/gui/Widget.ts +++ b/src/core/gui/Widget.ts @@ -82,6 +82,7 @@ export abstract class Widget implements Disposable { protected finalize_construction(proto: any): void { if (Object.getPrototypeOf(this) !== proto) return; + // At this point we know `this.element` is initialized. if (this.options.class) { this.element.classList.add(this.options.class); } diff --git a/src/core/gui/dom.ts b/src/core/gui/dom.ts index 0b32252c..e61b7d8a 100644 --- a/src/core/gui/dom.ts +++ b/src/core/gui/dom.ts @@ -103,24 +103,28 @@ export function create_element( const element = document.createElement(tag_name) as any; if (attributes) { - if (attributes.class != undefined) element.className = attributes.class; - if (attributes.text != undefined) element.textContent = attributes.text; - if (attributes.title != undefined) element.title = attributes.title; - if (attributes.href != undefined) element.href = attributes.href; - if (attributes.src != undefined) element.src = attributes.src; - if (attributes.width != undefined) element.width = attributes.width; - if (attributes.height != undefined) element.height = attributes.height; - if (attributes.alt != undefined) element.alt = attributes.alt; + if (attributes instanceof HTMLElement) { + element.append(attributes); + } else { + if (attributes.class != undefined) element.className = attributes.class; + if (attributes.text != undefined) element.textContent = attributes.text; + if (attributes.title != undefined) element.title = attributes.title; + if (attributes.href != undefined) element.href = attributes.href; + if (attributes.src != undefined) element.src = attributes.src; + if (attributes.width != undefined) element.width = attributes.width; + if (attributes.height != undefined) element.height = attributes.height; + if (attributes.alt != undefined) element.alt = attributes.alt; - if (attributes.data) { - for (const [key, val] of Object.entries(attributes.data)) { - element.dataset[key] = val; + if (attributes.data) { + for (const [key, val] of Object.entries(attributes.data)) { + element.dataset[key] = val; + } } + + if (attributes.col_span != undefined) element.colSpan = attributes.col_span; + + if (attributes.tab_index != undefined) element.tabIndex = attributes.tab_index; } - - if (attributes.col_span != undefined) element.colSpan = attributes.col_span; - - if (attributes.tab_index != undefined) element.tabIndex = attributes.tab_index; } element.append(...children); @@ -137,39 +141,41 @@ export function bind_hidden(element: HTMLElement, observable: Observable li { + cursor: default; +} + #root .lm_content { overflow: visible; } diff --git a/src/quest_editor/gui/EventsView.css b/src/quest_editor/gui/EventsView.css new file mode 100644 index 00000000..6e47b00c --- /dev/null +++ b/src/quest_editor/gui/EventsView.css @@ -0,0 +1,33 @@ +.quest_editor_EventsView { + overflow-y: auto; + display: flex; + flex-direction: column; + align-items: center; +} + +.quest_editor_EventsView_chain { + display: flex; + flex-direction: column; + align-items: center; + margin: 3px; +} + +.quest_editor_EventsView_chain_arrow { + font-size: 18px; /* For icon */ + margin: 3px; +} + +.quest_editor_EventsView_event:last-of-type .quest_editor_EventsView_chain_arrow { + color: var(--text-color-disabled); +} + +.quest_editor_EventsView_event { + display: flex; + flex-direction: column; + align-items: center; + width: 100%; +} + +.quest_editor_EventsView_event th { + text-align: left; +} diff --git a/src/quest_editor/gui/EventsView.ts b/src/quest_editor/gui/EventsView.ts new file mode 100644 index 00000000..c0d81927 --- /dev/null +++ b/src/quest_editor/gui/EventsView.ts @@ -0,0 +1,71 @@ +import { ResizableWidget } from "../../core/gui/ResizableWidget"; +import { bind_children_to, el, icon, Icon } from "../../core/gui/dom"; +import { quest_editor_store } from "../stores/QuestEditorStore"; +import { QuestEventChainModel } from "../model/QuestEventChainModel"; +import { Disposer } from "../../core/observable/Disposer"; +import { NumberInput } from "../../core/gui/NumberInput"; +import "./EventsView.css"; +import { Button } from "../../core/gui/Button"; +import { Disposable } from "../../core/observable/Disposable"; + +export class EventsView extends ResizableWidget { + private readonly quest_disposer = this.disposable(new Disposer()); + + readonly element = el.div({ class: "quest_editor_EventsView" }); + + constructor() { + super(); + + this.disposables( + quest_editor_store.current_quest.observe(({ value: quest }) => { + this.quest_disposer.dispose_all(); + + if (quest) { + this.quest_disposer.add( + bind_children_to( + this.element, + quest.event_chains, + this.create_chain_element, + ), + ); + } + }), + ); + + this.finalize_construction(EventsView.prototype); + } + + private create_chain_element = (chain: QuestEventChainModel): [HTMLElement, Disposable] => { + const disposer = new Disposer(); + const element = el.div( + { class: "quest_editor_EventsView_chain" }, + ...chain.events.val.map(event => + el.div( + { class: "quest_editor_EventsView_event" }, + el.table( + el.tr(el.th({ text: "ID:" }), el.td({ text: event.id.toString() })), + el.tr( + el.th({ text: "Section:" }), + el.td( + disposer.add(new NumberInput(event.section_id, { enabled: false })) + .element, + ), + ), + el.tr(el.th({ text: "Wave:" }), el.td({ text: event.wave.toString() })), + el.tr( + el.th({ text: "Delay:" }), + el.td( + disposer.add(new NumberInput(event.delay, { enabled: false })) + .element, + ), + ), + ), + el.div({ class: "quest_editor_EventsView_chain_arrow" }, icon(Icon.ArrowDown)), + ), + ), + disposer.add(new Button("Add event", { icon_left: Icon.Plus, enabled: false })).element, + ); + + return [element, disposer]; + }; +} diff --git a/src/quest_editor/gui/QuestEditorView.ts b/src/quest_editor/gui/QuestEditorView.ts index 899138d9..8b289871 100644 --- a/src/quest_editor/gui/QuestEditorView.ts +++ b/src/quest_editor/gui/QuestEditorView.ts @@ -14,6 +14,7 @@ import { gui_store, GuiTool } from "../../core/stores/GuiStore"; import { quest_editor_store } from "../stores/QuestEditorStore"; import { NpcListView } from "./NpcListView"; import { ObjectListView } from "./ObjectListView"; +import { EventsView } from "./EventsView"; import Logger = require("js-logger"); const logger = Logger.get("quest_editor/gui/QuestEditorView"); @@ -29,6 +30,10 @@ const VIEW_TO_NAME = new Map ResizableWidget, string>([ [ObjectListView, "object_list_view"], ]); +if (gui_store.feature_active("events")) { + VIEW_TO_NAME.set(EventsView, "events_view"); +} + const DEFAULT_LAYOUT_CONFIG = { settings: { showPopoutIcon: false, @@ -50,19 +55,30 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [ type: "row", content: [ { - type: "stack", - width: 3, + type: "column", + width: 2, content: [ { - title: "Info", - type: "component", - componentName: VIEW_TO_NAME.get(QuestInfoView), - isClosable: false, + type: "stack", + content: [ + { + title: "Info", + type: "component", + componentName: VIEW_TO_NAME.get(QuestInfoView), + isClosable: false, + }, + { + title: "NPC Counts", + type: "component", + componentName: VIEW_TO_NAME.get(NpcCountsView), + isClosable: false, + }, + ], }, { - title: "NPC Counts", + title: "Entity", type: "component", - componentName: VIEW_TO_NAME.get(NpcCountsView), + componentName: VIEW_TO_NAME.get(EntityInfoView), isClosable: false, }, ], @@ -89,12 +105,6 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [ type: "stack", width: 2, content: [ - { - title: "Entity", - type: "component", - componentName: VIEW_TO_NAME.get(EntityInfoView), - isClosable: false, - }, { title: "NPCs", type: "component", @@ -107,6 +117,16 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [ componentName: VIEW_TO_NAME.get(ObjectListView), isClosable: false, }, + ...(gui_store.feature_active("events") + ? [ + { + title: "Events", + type: "component", + componentName: VIEW_TO_NAME.get(EventsView), + isClosable: false, + }, + ] + : []), ], }, ], diff --git a/src/quest_editor/model/QuestEventActionModel.ts b/src/quest_editor/model/QuestEventActionModel.ts index f423d2d2..c940b9ff 100644 --- a/src/quest_editor/model/QuestEventActionModel.ts +++ b/src/quest_editor/model/QuestEventActionModel.ts @@ -31,13 +31,3 @@ export class QuestEventActionLockModel extends QuestEventActionModel { this.door_id = door_id; } } - -export class QuestEventActionSpawnWaveModel extends QuestEventActionModel { - readonly wave_id: number; - - constructor(wave_id: number) { - super(); - - this.wave_id = wave_id; - } -} diff --git a/src/quest_editor/model/QuestEventChainModel.ts b/src/quest_editor/model/QuestEventChainModel.ts new file mode 100644 index 00000000..3bdaf75b --- /dev/null +++ b/src/quest_editor/model/QuestEventChainModel.ts @@ -0,0 +1,15 @@ +import { QuestEventModel } from "./QuestEventModel"; +import { ListProperty } from "../../core/observable/property/list/ListProperty"; +import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty"; +import { list_property } from "../../core/observable"; + +export class QuestEventChainModel { + private readonly _events: WritableListProperty; + + readonly events: ListProperty; + + constructor(events: QuestEventModel[]) { + this._events = list_property(undefined, ...events); + this.events = this._events; + } +} diff --git a/src/quest_editor/model/QuestModel.ts b/src/quest_editor/model/QuestModel.ts index 6bb51b47..0784b2a7 100644 --- a/src/quest_editor/model/QuestModel.ts +++ b/src/quest_editor/model/QuestModel.ts @@ -13,7 +13,7 @@ import { ListProperty } from "../../core/observable/property/list/ListProperty"; import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty"; import { QuestEntityModel } from "./QuestEntityModel"; import { entity_type_to_string } from "../../core/data_formats/parsing/quest/entities"; -import { QuestEventModel } from "./QuestEventModel"; +import { QuestEventChainModel } from "./QuestEventChainModel"; const logger = Logger.get("quest_editor/model/QuestModel"); @@ -27,7 +27,7 @@ export class QuestModel { private readonly _area_variants: WritableListProperty = list_property(); private readonly _objects: WritableListProperty; private readonly _npcs: WritableListProperty; - private readonly _waves: WritableListProperty; + private readonly _event_chains: WritableListProperty; readonly id: Property = this._id; @@ -60,7 +60,7 @@ export class QuestModel { readonly npcs: ListProperty; - readonly waves: ListProperty; + readonly event_chains: ListProperty; /** * (Partial) raw DAT data that can't be parsed yet by Phantasmal. @@ -81,7 +81,7 @@ export class QuestModel { map_designations: Map, objects: readonly QuestObjectModel[], npcs: readonly QuestNpcModel[], - waves: readonly QuestEventModel[], + event_chains: readonly QuestEventChainModel[], dat_unknowns: readonly DatUnknown[], object_code: readonly Segment[], shop_items: readonly number[], @@ -90,7 +90,7 @@ export class QuestModel { if (!map_designations) throw new Error("map_designations is required."); if (!Array.isArray(objects)) throw new Error("objs is required."); if (!Array.isArray(npcs)) throw new Error("npcs is required."); - if (!Array.isArray(waves)) throw new Error("waves is required."); + if (!Array.isArray(event_chains)) throw new Error("event_chains is required."); if (!Array.isArray(dat_unknowns)) throw new Error("dat_unknowns is required."); if (!Array.isArray(object_code)) throw new Error("object_code is required."); if (!Array.isArray(shop_items)) throw new Error("shop_items is required."); @@ -107,8 +107,8 @@ export class QuestModel { this.objects = this._objects; this._npcs = list_property(undefined, ...npcs); this.npcs = this._npcs; - this._waves = list_property(undefined, ...waves); - this.waves = this._waves; + this._event_chains = list_property(undefined, ...event_chains); + this.event_chains = this._event_chains; this.dat_unknowns = dat_unknowns; this.object_code = object_code; this.shop_items = shop_items; diff --git a/src/quest_editor/stores/QuestEditorStore.ts b/src/quest_editor/stores/QuestEditorStore.ts index 58b3d983..71c3432d 100644 --- a/src/quest_editor/stores/QuestEditorStore.ts +++ b/src/quest_editor/stores/QuestEditorStore.ts @@ -26,17 +26,9 @@ import { create_new_quest } from "./quest_creation"; import { CreateEntityAction } from "../actions/CreateEntityAction"; import { RemoveEntityAction } from "../actions/RemoveEntityAction"; import { Euler, Vector3 } from "three"; -import { vec3_to_threejs } from "../../core/rendering/conversion"; import { RotateEntityAction } from "../actions/RotateEntityAction"; import { ExecutionResult, VirtualMachine } from "../scripting/vm"; -import { QuestEventModel } from "../model/QuestEventModel"; -import { DatEventActionType } from "../../core/data_formats/parsing/quest/dat"; -import { - QuestEventActionLockModel, - QuestEventActionSpawnNpcsModel, - QuestEventActionSpawnWaveModel, - QuestEventActionUnlockModel, -} from "../model/QuestEventActionModel"; +import { convert_quest_from_model, convert_quest_to_model } from "./model_conversion"; import Logger = require("js-logger"); const logger = Logger.get("quest_editor/gui/QuestEditorStore"); @@ -118,92 +110,7 @@ export class QuestEditorStore implements Disposable { try { const buffer = await read_file(file); const quest = parse_quest(new ArrayBufferCursor(buffer, Endianness.Little)); - this.set_quest( - quest && - new QuestModel( - quest.id, - quest.language, - quest.name, - quest.short_description, - quest.long_description, - quest.episode, - quest.map_designations, - quest.objects.map( - obj => - new QuestObjectModel( - obj.type, - obj.id, - obj.group_id, - obj.area_id, - obj.section_id, - vec3_to_threejs(obj.position), - new Euler( - obj.rotation.x, - obj.rotation.y, - obj.rotation.z, - "ZXY", - ), - obj.properties, - obj.unknown, - ), - ), - quest.npcs.map( - npc => - new QuestNpcModel( - npc.type, - npc.pso_type_id, - npc.npc_id, - npc.script_label, - npc.pso_roaming, - npc.area_id, - npc.section_id, - vec3_to_threejs(npc.position), - new Euler( - npc.rotation.x, - npc.rotation.y, - npc.rotation.z, - "ZXY", - ), - vec3_to_threejs(npc.scale), - npc.unknown, - ), - ), - quest.waves.map( - wave => - new QuestEventModel( - wave.id, - wave.section_id, - wave.wave, - wave.delay, - wave.actions.map(action => { - switch (action.type) { - case DatEventActionType.SpawnNpcs: - return new QuestEventActionSpawnNpcsModel( - action.section_id, - action.appear_flag, - ); - case DatEventActionType.Unlock: - return new QuestEventActionUnlockModel( - action.door_id, - ); - case DatEventActionType.Lock: - return new QuestEventActionLockModel(action.door_id); - case DatEventActionType.SpawnWave: - return new QuestEventActionSpawnWaveModel( - action.wave_id, - ); - } - }), - wave.area_id, - wave.unknown, - ), - ), - quest.dat_unknowns, - quest.object_code, - quest.shop_items, - ), - file.name, - ); + this.set_quest(quest && convert_quest_to_model(quest), file.name); } catch (e) { logger.error("Couldn't read file.", e); } @@ -223,83 +130,7 @@ export class QuestEditorStore implements Disposable { let file_name = prompt("File name:", default_file_name); if (!file_name) return; - const buffer = write_quest_qst( - { - id: quest.id.val, - language: quest.language.val, - name: quest.name.val, - short_description: quest.short_description.val, - long_description: quest.long_description.val, - episode: quest.episode, - objects: quest.objects.val.map(obj => ({ - type: obj.type, - area_id: obj.area_id, - section_id: obj.section_id.val, - position: obj.position.val, - rotation: obj.rotation.val, - unknown: obj.unknown, - id: obj.id, - group_id: obj.group_id, - properties: obj.properties, - })), - npcs: quest.npcs.val.map(npc => ({ - type: npc.type, - area_id: npc.area_id, - section_id: npc.section_id.val, - position: npc.position.val, - rotation: npc.rotation.val, - scale: npc.scale, - unknown: npc.unknown, - pso_type_id: npc.pso_type_id, - npc_id: npc.npc_id, - script_label: npc.script_label, - pso_roaming: npc.pso_roaming, - })), - waves: quest.waves.val.map(wave => ({ - id: wave.id, - section_id: wave.section_id, - wave: wave.wave, - delay: wave.delay, - actions: wave.actions.map(action => { - if (action instanceof QuestEventActionSpawnNpcsModel) { - return { - type: DatEventActionType.SpawnNpcs, - section_id: action.section_id, - appear_flag: action.appear_flag, - }; - } else if (action instanceof QuestEventActionUnlockModel) { - return { - type: DatEventActionType.Unlock, - door_id: action.door_id, - }; - } else if (action instanceof QuestEventActionLockModel) { - return { - type: DatEventActionType.Lock, - door_id: action.door_id, - }; - } else if (action instanceof QuestEventActionSpawnWaveModel) { - return { - type: DatEventActionType.SpawnWave, - wave_id: action.wave_id, - }; - } else { - throw new Error( - `Unknown wave action type ${ - Object.getPrototypeOf(action).constructor - }`, - ); - } - }), - area_id: wave.area_id, - unknown: wave.unknown, - })), - dat_unknowns: quest.dat_unknowns, - object_code: quest.object_code, - shop_items: quest.shop_items, - map_designations: quest.map_designations.val, - }, - file_name, - ); + const buffer = write_quest_qst(convert_quest_from_model(quest), file_name); if (!file_name.endsWith(".qst")) { file_name += ".qst"; diff --git a/src/quest_editor/stores/model_conversion.ts b/src/quest_editor/stores/model_conversion.ts new file mode 100644 index 00000000..b7175773 --- /dev/null +++ b/src/quest_editor/stores/model_conversion.ts @@ -0,0 +1,205 @@ +import { Quest } from "../../core/data_formats/parsing/quest"; +import { QuestModel } from "../model/QuestModel"; +import { QuestObjectModel } from "../model/QuestObjectModel"; +import { vec3_to_threejs } from "../../core/rendering/conversion"; +import { Euler } from "three"; +import { QuestNpcModel } from "../model/QuestNpcModel"; +import { QuestEventModel } from "../model/QuestEventModel"; +import { + DatEventActionTriggerEvent, + DatEventActionType, +} from "../../core/data_formats/parsing/quest/dat"; +import { + QuestEventActionLockModel, + QuestEventActionSpawnNpcsModel, + QuestEventActionUnlockModel, +} from "../model/QuestEventActionModel"; +import { QuestEventChainModel } from "../model/QuestEventChainModel"; +import { QuestEvent } from "../../core/data_formats/parsing/quest/entities"; +import Logger from "js-logger"; + +const logger = Logger.get("quest_editor/stores/model_conversion"); + +export function convert_quest_to_model(quest: Quest): QuestModel { + // Build up event chains. + const events = quest.events.slice(); + const event_chains: QuestEventChainModel[] = []; + + while (events.length) { + let event: QuestEvent | undefined = events.shift(); + const chain_events = []; + + while (event) { + chain_events.push( + new QuestEventModel( + event.id, + event.section_id, + event.wave, + event.delay, + event.actions + .filter(action => action.type !== DatEventActionType.TriggerEvent) + .map(action => { + switch (action.type) { + case DatEventActionType.SpawnNpcs: + return new QuestEventActionSpawnNpcsModel( + action.section_id, + action.appear_flag, + ); + case DatEventActionType.Unlock: + return new QuestEventActionUnlockModel(action.door_id); + case DatEventActionType.Lock: + return new QuestEventActionLockModel(action.door_id); + case DatEventActionType.TriggerEvent: + throw new Error("Can't convert trigger event actions."); + } + }), + event.area_id, + event.unknown, + ), + ); + + const event_id = event.id; + + const trigger_event_actions = event.actions.filter( + action => action.type === DatEventActionType.TriggerEvent, + ) as DatEventActionTriggerEvent[]; + + event = undefined; + + if (trigger_event_actions.length >= 1) { + if (trigger_event_actions.length > 1) { + logger.warn(`Event ${event_id} has more than 1 trigger event action.`); + } + + const index = events.findIndex(e => e.id === trigger_event_actions[0].event_id); + + if (index !== -1) { + event = events.splice(index, 1)[0]; + } + } + } + + const chain = new QuestEventChainModel(chain_events); + event_chains.push(chain); + } + + // Create quest model. + return new QuestModel( + quest.id, + quest.language, + quest.name, + quest.short_description, + quest.long_description, + quest.episode, + quest.map_designations, + quest.objects.map( + obj => + new QuestObjectModel( + obj.type, + obj.id, + obj.group_id, + obj.area_id, + obj.section_id, + vec3_to_threejs(obj.position), + new Euler(obj.rotation.x, obj.rotation.y, obj.rotation.z, "ZXY"), + obj.properties, + obj.unknown, + ), + ), + quest.npcs.map( + npc => + new QuestNpcModel( + npc.type, + npc.pso_type_id, + npc.npc_id, + npc.script_label, + npc.pso_roaming, + npc.area_id, + npc.section_id, + vec3_to_threejs(npc.position), + new Euler(npc.rotation.x, npc.rotation.y, npc.rotation.z, "ZXY"), + vec3_to_threejs(npc.scale), + npc.unknown, + ), + ), + event_chains, + quest.dat_unknowns, + quest.object_code, + quest.shop_items, + ); +} + +export function convert_quest_from_model(quest: QuestModel): Quest { + return { + id: quest.id.val, + language: quest.language.val, + name: quest.name.val, + short_description: quest.short_description.val, + long_description: quest.long_description.val, + episode: quest.episode, + objects: quest.objects.val.map(obj => ({ + type: obj.type, + area_id: obj.area_id, + section_id: obj.section_id.val, + position: obj.position.val, + rotation: obj.rotation.val, + unknown: obj.unknown, + id: obj.id, + group_id: obj.group_id, + properties: obj.properties, + })), + npcs: quest.npcs.val.map(npc => ({ + type: npc.type, + area_id: npc.area_id, + section_id: npc.section_id.val, + position: npc.position.val, + rotation: npc.rotation.val, + scale: npc.scale, + unknown: npc.unknown, + pso_type_id: npc.pso_type_id, + npc_id: npc.npc_id, + script_label: npc.script_label, + pso_roaming: npc.pso_roaming, + })), + events: quest.waves.val.map(wave => ({ + id: wave.id, + section_id: wave.section_id, + wave: wave.wave, + delay: wave.delay, + actions: wave.actions.map(action => { + if (action instanceof QuestEventActionSpawnNpcsModel) { + return { + type: DatEventActionType.SpawnNpcs, + section_id: action.section_id, + appear_flag: action.appear_flag, + }; + } else if (action instanceof QuestEventActionUnlockModel) { + return { + type: DatEventActionType.Unlock, + door_id: action.door_id, + }; + } else if (action instanceof QuestEventActionLockModel) { + return { + type: DatEventActionType.Lock, + door_id: action.door_id, + }; + } else if (action instanceof QuestEventActionTriggerEventModel) { + return { + type: DatEventActionType.TriggerEvent, + wave_id: action.wave_id, + }; + } else { + throw new Error( + `Unknown event action type ${Object.getPrototypeOf(action).constructor}`, + ); + } + }), + area_id: wave.area_id, + unknown: wave.unknown, + })), + dat_unknowns: quest.dat_unknowns, + object_code: quest.object_code, + shop_items: quest.shop_items, + map_designations: quest.map_designations.val, + }; +}