mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 23:38:30 +08:00
Refactored events model and added a basic view for it behind a feature flag.
This commit is contained in:
parent
c287fdeb2f
commit
ff8f02fe5b
@ -55,14 +55,14 @@ export enum DatEventActionType {
|
|||||||
SpawnNpcs = 0x8,
|
SpawnNpcs = 0x8,
|
||||||
Unlock = 0xa,
|
Unlock = 0xa,
|
||||||
Lock = 0xb,
|
Lock = 0xb,
|
||||||
SpawnWave = 0xc,
|
TriggerEvent = 0xc,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type DatEventAction =
|
export type DatEventAction =
|
||||||
| DatEventActionSpawnNpcs
|
| DatEventActionSpawnNpcs
|
||||||
| DatEventActionUnlock
|
| DatEventActionUnlock
|
||||||
| DatEventActionLock
|
| DatEventActionLock
|
||||||
| DatEventActionSpawnWave;
|
| DatEventActionTriggerEvent;
|
||||||
|
|
||||||
export type DatEventActionSpawnNpcs = {
|
export type DatEventActionSpawnNpcs = {
|
||||||
readonly type: DatEventActionType.SpawnNpcs;
|
readonly type: DatEventActionType.SpawnNpcs;
|
||||||
@ -80,9 +80,9 @@ export type DatEventActionLock = {
|
|||||||
readonly door_id: number;
|
readonly door_id: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DatEventActionSpawnWave = {
|
export type DatEventActionTriggerEvent = {
|
||||||
readonly type: DatEventActionType.SpawnWave;
|
readonly type: DatEventActionType.TriggerEvent;
|
||||||
readonly wave_id: number;
|
readonly event_id: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DatUnknown = {
|
export type DatUnknown = {
|
||||||
@ -344,10 +344,10 @@ function parse_wave_actions(cursor: Cursor): DatEventAction[] {
|
|||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case DatEventActionType.SpawnWave:
|
case DatEventActionType.TriggerEvent:
|
||||||
actions.push({
|
actions.push({
|
||||||
type: DatEventActionType.SpawnWave,
|
type: DatEventActionType.TriggerEvent,
|
||||||
wave_id: cursor.u32(),
|
event_id: cursor.u32(),
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -519,8 +519,8 @@ function write_waves(cursor: WritableCursor, waves: readonly DatEvent[]): void {
|
|||||||
cursor.write_u16(action.door_id);
|
cursor.write_u16(action.door_id);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case DatEventActionType.SpawnWave:
|
case DatEventActionType.TriggerEvent:
|
||||||
cursor.write_u32(action.wave_id);
|
cursor.write_u32(action.event_id);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -32,7 +32,7 @@ export type Quest = {
|
|||||||
readonly episode: Episode;
|
readonly episode: Episode;
|
||||||
readonly objects: readonly QuestObject[];
|
readonly objects: readonly QuestObject[];
|
||||||
readonly npcs: readonly QuestNpc[];
|
readonly npcs: readonly QuestNpc[];
|
||||||
readonly waves: readonly QuestEvent[];
|
readonly events: readonly QuestEvent[];
|
||||||
/**
|
/**
|
||||||
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
* (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,
|
episode,
|
||||||
objects,
|
objects,
|
||||||
npcs: parse_npc_data(episode, dat.npcs),
|
npcs: parse_npc_data(episode, dat.npcs),
|
||||||
waves: dat.waves,
|
events: dat.waves,
|
||||||
dat_unknowns: dat.unknowns,
|
dat_unknowns: dat.unknowns,
|
||||||
object_code: bin.object_code,
|
object_code: bin.object_code,
|
||||||
shop_items: bin.shop_items,
|
shop_items: bin.shop_items,
|
||||||
@ -139,7 +139,7 @@ export function write_quest_qst(quest: Quest, file_name: string): ArrayBuffer {
|
|||||||
const dat = write_dat({
|
const dat = write_dat({
|
||||||
objs: objects_to_dat_data(quest.objects),
|
objs: objects_to_dat_data(quest.objects),
|
||||||
npcs: npcs_to_dat_data(quest.npcs),
|
npcs: npcs_to_dat_data(quest.npcs),
|
||||||
waves: quest.waves,
|
waves: quest.events,
|
||||||
unknowns: quest.dat_unknowns,
|
unknowns: quest.dat_unknowns,
|
||||||
});
|
});
|
||||||
const bin = write_bin({
|
const bin = write_bin({
|
||||||
|
@ -82,6 +82,7 @@ export abstract class Widget implements Disposable {
|
|||||||
protected finalize_construction(proto: any): void {
|
protected finalize_construction(proto: any): void {
|
||||||
if (Object.getPrototypeOf(this) !== proto) return;
|
if (Object.getPrototypeOf(this) !== proto) return;
|
||||||
|
|
||||||
|
// At this point we know `this.element` is initialized.
|
||||||
if (this.options.class) {
|
if (this.options.class) {
|
||||||
this.element.classList.add(this.options.class);
|
this.element.classList.add(this.options.class);
|
||||||
}
|
}
|
||||||
|
@ -103,24 +103,28 @@ export function create_element<T extends HTMLElement>(
|
|||||||
const element = document.createElement(tag_name) as any;
|
const element = document.createElement(tag_name) as any;
|
||||||
|
|
||||||
if (attributes) {
|
if (attributes) {
|
||||||
if (attributes.class != undefined) element.className = attributes.class;
|
if (attributes instanceof HTMLElement) {
|
||||||
if (attributes.text != undefined) element.textContent = attributes.text;
|
element.append(attributes);
|
||||||
if (attributes.title != undefined) element.title = attributes.title;
|
} else {
|
||||||
if (attributes.href != undefined) element.href = attributes.href;
|
if (attributes.class != undefined) element.className = attributes.class;
|
||||||
if (attributes.src != undefined) element.src = attributes.src;
|
if (attributes.text != undefined) element.textContent = attributes.text;
|
||||||
if (attributes.width != undefined) element.width = attributes.width;
|
if (attributes.title != undefined) element.title = attributes.title;
|
||||||
if (attributes.height != undefined) element.height = attributes.height;
|
if (attributes.href != undefined) element.href = attributes.href;
|
||||||
if (attributes.alt != undefined) element.alt = attributes.alt;
|
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) {
|
if (attributes.data) {
|
||||||
for (const [key, val] of Object.entries(attributes.data)) {
|
for (const [key, val] of Object.entries(attributes.data)) {
|
||||||
element.dataset[key] = val;
|
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);
|
element.append(...children);
|
||||||
@ -137,39 +141,41 @@ export function bind_hidden(element: HTMLElement, observable: Observable<boolean
|
|||||||
}
|
}
|
||||||
|
|
||||||
export enum Icon {
|
export enum Icon {
|
||||||
|
ArrowDown,
|
||||||
File,
|
File,
|
||||||
|
GitHub,
|
||||||
NewFile,
|
NewFile,
|
||||||
Save,
|
Play,
|
||||||
TriangleUp,
|
Plus,
|
||||||
TriangleDown,
|
|
||||||
Undo,
|
|
||||||
Redo,
|
Redo,
|
||||||
Remove,
|
Remove,
|
||||||
GitHub,
|
Save,
|
||||||
Play,
|
TriangleDown,
|
||||||
|
TriangleUp,
|
||||||
|
Undo,
|
||||||
}
|
}
|
||||||
|
|
||||||
export function icon(icon: Icon): HTMLElement {
|
export function icon(icon: Icon): HTMLElement {
|
||||||
let icon_str!: string;
|
let icon_str!: string;
|
||||||
|
|
||||||
switch (icon) {
|
switch (icon) {
|
||||||
|
case Icon.ArrowDown:
|
||||||
|
icon_str = "fas fa-arrow-down";
|
||||||
|
break;
|
||||||
case Icon.File:
|
case Icon.File:
|
||||||
icon_str = "fas fa-file";
|
icon_str = "fas fa-file";
|
||||||
break;
|
break;
|
||||||
|
case Icon.GitHub:
|
||||||
|
icon_str = "fab fa-github";
|
||||||
|
break;
|
||||||
case Icon.NewFile:
|
case Icon.NewFile:
|
||||||
icon_str = "fas fa-file-medical";
|
icon_str = "fas fa-file-medical";
|
||||||
break;
|
break;
|
||||||
case Icon.Save:
|
case Icon.Play:
|
||||||
icon_str = "fas fa-save";
|
icon_str = "fas fa-play";
|
||||||
break;
|
break;
|
||||||
case Icon.TriangleUp:
|
case Icon.Plus:
|
||||||
icon_str = "fas fa-caret-up";
|
icon_str = "fas fa-plus";
|
||||||
break;
|
|
||||||
case Icon.TriangleDown:
|
|
||||||
icon_str = "fas fa-caret-down";
|
|
||||||
break;
|
|
||||||
case Icon.Undo:
|
|
||||||
icon_str = "fas fa-undo";
|
|
||||||
break;
|
break;
|
||||||
case Icon.Redo:
|
case Icon.Redo:
|
||||||
icon_str = "fas fa-redo";
|
icon_str = "fas fa-redo";
|
||||||
@ -177,11 +183,17 @@ export function icon(icon: Icon): HTMLElement {
|
|||||||
case Icon.Remove:
|
case Icon.Remove:
|
||||||
icon_str = "fas fa-trash-alt";
|
icon_str = "fas fa-trash-alt";
|
||||||
break;
|
break;
|
||||||
case Icon.GitHub:
|
case Icon.Save:
|
||||||
icon_str = "fab fa-github";
|
icon_str = "fas fa-save";
|
||||||
break;
|
break;
|
||||||
case Icon.Play:
|
case Icon.TriangleDown:
|
||||||
icon_str = "fas fa-play";
|
icon_str = "fas fa-caret-down";
|
||||||
|
break;
|
||||||
|
case Icon.TriangleUp:
|
||||||
|
icon_str = "fas fa-caret-up";
|
||||||
|
break;
|
||||||
|
case Icon.Undo:
|
||||||
|
icon_str = "fas fa-undo";
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -32,6 +32,10 @@
|
|||||||
border-bottom-color: var(--bg-color);
|
border-bottom-color: var(--bg-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#root .lm_header .lm_controls > li {
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
|
||||||
#root .lm_content {
|
#root .lm_content {
|
||||||
overflow: visible;
|
overflow: visible;
|
||||||
}
|
}
|
||||||
|
33
src/quest_editor/gui/EventsView.css
Normal file
33
src/quest_editor/gui/EventsView.css
Normal file
@ -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;
|
||||||
|
}
|
71
src/quest_editor/gui/EventsView.ts
Normal file
71
src/quest_editor/gui/EventsView.ts
Normal file
@ -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];
|
||||||
|
};
|
||||||
|
}
|
@ -14,6 +14,7 @@ import { gui_store, GuiTool } from "../../core/stores/GuiStore";
|
|||||||
import { quest_editor_store } from "../stores/QuestEditorStore";
|
import { quest_editor_store } from "../stores/QuestEditorStore";
|
||||||
import { NpcListView } from "./NpcListView";
|
import { NpcListView } from "./NpcListView";
|
||||||
import { ObjectListView } from "./ObjectListView";
|
import { ObjectListView } from "./ObjectListView";
|
||||||
|
import { EventsView } from "./EventsView";
|
||||||
import Logger = require("js-logger");
|
import Logger = require("js-logger");
|
||||||
|
|
||||||
const logger = Logger.get("quest_editor/gui/QuestEditorView");
|
const logger = Logger.get("quest_editor/gui/QuestEditorView");
|
||||||
@ -29,6 +30,10 @@ const VIEW_TO_NAME = new Map<new () => ResizableWidget, string>([
|
|||||||
[ObjectListView, "object_list_view"],
|
[ObjectListView, "object_list_view"],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
if (gui_store.feature_active("events")) {
|
||||||
|
VIEW_TO_NAME.set(EventsView, "events_view");
|
||||||
|
}
|
||||||
|
|
||||||
const DEFAULT_LAYOUT_CONFIG = {
|
const DEFAULT_LAYOUT_CONFIG = {
|
||||||
settings: {
|
settings: {
|
||||||
showPopoutIcon: false,
|
showPopoutIcon: false,
|
||||||
@ -50,19 +55,30 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [
|
|||||||
type: "row",
|
type: "row",
|
||||||
content: [
|
content: [
|
||||||
{
|
{
|
||||||
type: "stack",
|
type: "column",
|
||||||
width: 3,
|
width: 2,
|
||||||
content: [
|
content: [
|
||||||
{
|
{
|
||||||
title: "Info",
|
type: "stack",
|
||||||
type: "component",
|
content: [
|
||||||
componentName: VIEW_TO_NAME.get(QuestInfoView),
|
{
|
||||||
isClosable: false,
|
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",
|
type: "component",
|
||||||
componentName: VIEW_TO_NAME.get(NpcCountsView),
|
componentName: VIEW_TO_NAME.get(EntityInfoView),
|
||||||
isClosable: false,
|
isClosable: false,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@ -89,12 +105,6 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [
|
|||||||
type: "stack",
|
type: "stack",
|
||||||
width: 2,
|
width: 2,
|
||||||
content: [
|
content: [
|
||||||
{
|
|
||||||
title: "Entity",
|
|
||||||
type: "component",
|
|
||||||
componentName: VIEW_TO_NAME.get(EntityInfoView),
|
|
||||||
isClosable: false,
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
title: "NPCs",
|
title: "NPCs",
|
||||||
type: "component",
|
type: "component",
|
||||||
@ -107,6 +117,16 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [
|
|||||||
componentName: VIEW_TO_NAME.get(ObjectListView),
|
componentName: VIEW_TO_NAME.get(ObjectListView),
|
||||||
isClosable: false,
|
isClosable: false,
|
||||||
},
|
},
|
||||||
|
...(gui_store.feature_active("events")
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
title: "Events",
|
||||||
|
type: "component",
|
||||||
|
componentName: VIEW_TO_NAME.get(EventsView),
|
||||||
|
isClosable: false,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: []),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -31,13 +31,3 @@ export class QuestEventActionLockModel extends QuestEventActionModel {
|
|||||||
this.door_id = door_id;
|
this.door_id = door_id;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class QuestEventActionSpawnWaveModel extends QuestEventActionModel {
|
|
||||||
readonly wave_id: number;
|
|
||||||
|
|
||||||
constructor(wave_id: number) {
|
|
||||||
super();
|
|
||||||
|
|
||||||
this.wave_id = wave_id;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
15
src/quest_editor/model/QuestEventChainModel.ts
Normal file
15
src/quest_editor/model/QuestEventChainModel.ts
Normal file
@ -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<QuestEventModel>;
|
||||||
|
|
||||||
|
readonly events: ListProperty<QuestEventModel>;
|
||||||
|
|
||||||
|
constructor(events: QuestEventModel[]) {
|
||||||
|
this._events = list_property(undefined, ...events);
|
||||||
|
this.events = this._events;
|
||||||
|
}
|
||||||
|
}
|
@ -13,7 +13,7 @@ import { ListProperty } from "../../core/observable/property/list/ListProperty";
|
|||||||
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
|
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
|
||||||
import { QuestEntityModel } from "./QuestEntityModel";
|
import { QuestEntityModel } from "./QuestEntityModel";
|
||||||
import { entity_type_to_string } from "../../core/data_formats/parsing/quest/entities";
|
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");
|
const logger = Logger.get("quest_editor/model/QuestModel");
|
||||||
|
|
||||||
@ -27,7 +27,7 @@ export class QuestModel {
|
|||||||
private readonly _area_variants: WritableListProperty<AreaVariantModel> = list_property();
|
private readonly _area_variants: WritableListProperty<AreaVariantModel> = list_property();
|
||||||
private readonly _objects: WritableListProperty<QuestObjectModel>;
|
private readonly _objects: WritableListProperty<QuestObjectModel>;
|
||||||
private readonly _npcs: WritableListProperty<QuestNpcModel>;
|
private readonly _npcs: WritableListProperty<QuestNpcModel>;
|
||||||
private readonly _waves: WritableListProperty<QuestEventModel>;
|
private readonly _event_chains: WritableListProperty<QuestEventChainModel>;
|
||||||
|
|
||||||
readonly id: Property<number> = this._id;
|
readonly id: Property<number> = this._id;
|
||||||
|
|
||||||
@ -60,7 +60,7 @@ export class QuestModel {
|
|||||||
|
|
||||||
readonly npcs: ListProperty<QuestNpcModel>;
|
readonly npcs: ListProperty<QuestNpcModel>;
|
||||||
|
|
||||||
readonly waves: ListProperty<QuestEventModel>;
|
readonly event_chains: ListProperty<QuestEventChainModel>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
||||||
@ -81,7 +81,7 @@ export class QuestModel {
|
|||||||
map_designations: Map<number, number>,
|
map_designations: Map<number, number>,
|
||||||
objects: readonly QuestObjectModel[],
|
objects: readonly QuestObjectModel[],
|
||||||
npcs: readonly QuestNpcModel[],
|
npcs: readonly QuestNpcModel[],
|
||||||
waves: readonly QuestEventModel[],
|
event_chains: readonly QuestEventChainModel[],
|
||||||
dat_unknowns: readonly DatUnknown[],
|
dat_unknowns: readonly DatUnknown[],
|
||||||
object_code: readonly Segment[],
|
object_code: readonly Segment[],
|
||||||
shop_items: readonly number[],
|
shop_items: readonly number[],
|
||||||
@ -90,7 +90,7 @@ export class QuestModel {
|
|||||||
if (!map_designations) throw new Error("map_designations is required.");
|
if (!map_designations) throw new Error("map_designations is required.");
|
||||||
if (!Array.isArray(objects)) throw new Error("objs 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(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(dat_unknowns)) throw new Error("dat_unknowns is required.");
|
||||||
if (!Array.isArray(object_code)) throw new Error("object_code 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.");
|
if (!Array.isArray(shop_items)) throw new Error("shop_items is required.");
|
||||||
@ -107,8 +107,8 @@ export class QuestModel {
|
|||||||
this.objects = this._objects;
|
this.objects = this._objects;
|
||||||
this._npcs = list_property(undefined, ...npcs);
|
this._npcs = list_property(undefined, ...npcs);
|
||||||
this.npcs = this._npcs;
|
this.npcs = this._npcs;
|
||||||
this._waves = list_property(undefined, ...waves);
|
this._event_chains = list_property(undefined, ...event_chains);
|
||||||
this.waves = this._waves;
|
this.event_chains = this._event_chains;
|
||||||
this.dat_unknowns = dat_unknowns;
|
this.dat_unknowns = dat_unknowns;
|
||||||
this.object_code = object_code;
|
this.object_code = object_code;
|
||||||
this.shop_items = shop_items;
|
this.shop_items = shop_items;
|
||||||
|
@ -26,17 +26,9 @@ import { create_new_quest } from "./quest_creation";
|
|||||||
import { CreateEntityAction } from "../actions/CreateEntityAction";
|
import { CreateEntityAction } from "../actions/CreateEntityAction";
|
||||||
import { RemoveEntityAction } from "../actions/RemoveEntityAction";
|
import { RemoveEntityAction } from "../actions/RemoveEntityAction";
|
||||||
import { Euler, Vector3 } from "three";
|
import { Euler, Vector3 } from "three";
|
||||||
import { vec3_to_threejs } from "../../core/rendering/conversion";
|
|
||||||
import { RotateEntityAction } from "../actions/RotateEntityAction";
|
import { RotateEntityAction } from "../actions/RotateEntityAction";
|
||||||
import { ExecutionResult, VirtualMachine } from "../scripting/vm";
|
import { ExecutionResult, VirtualMachine } from "../scripting/vm";
|
||||||
import { QuestEventModel } from "../model/QuestEventModel";
|
import { convert_quest_from_model, convert_quest_to_model } from "./model_conversion";
|
||||||
import { DatEventActionType } from "../../core/data_formats/parsing/quest/dat";
|
|
||||||
import {
|
|
||||||
QuestEventActionLockModel,
|
|
||||||
QuestEventActionSpawnNpcsModel,
|
|
||||||
QuestEventActionSpawnWaveModel,
|
|
||||||
QuestEventActionUnlockModel,
|
|
||||||
} from "../model/QuestEventActionModel";
|
|
||||||
import Logger = require("js-logger");
|
import Logger = require("js-logger");
|
||||||
|
|
||||||
const logger = Logger.get("quest_editor/gui/QuestEditorStore");
|
const logger = Logger.get("quest_editor/gui/QuestEditorStore");
|
||||||
@ -118,92 +110,7 @@ export class QuestEditorStore implements Disposable {
|
|||||||
try {
|
try {
|
||||||
const buffer = await read_file(file);
|
const buffer = await read_file(file);
|
||||||
const quest = parse_quest(new ArrayBufferCursor(buffer, Endianness.Little));
|
const quest = parse_quest(new ArrayBufferCursor(buffer, Endianness.Little));
|
||||||
this.set_quest(
|
this.set_quest(quest && convert_quest_to_model(quest), file.name);
|
||||||
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,
|
|
||||||
);
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
logger.error("Couldn't read file.", 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);
|
let file_name = prompt("File name:", default_file_name);
|
||||||
if (!file_name) return;
|
if (!file_name) return;
|
||||||
|
|
||||||
const buffer = write_quest_qst(
|
const buffer = write_quest_qst(convert_quest_from_model(quest), file_name);
|
||||||
{
|
|
||||||
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,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!file_name.endsWith(".qst")) {
|
if (!file_name.endsWith(".qst")) {
|
||||||
file_name += ".qst";
|
file_name += ".qst";
|
||||||
|
205
src/quest_editor/stores/model_conversion.ts
Normal file
205
src/quest_editor/stores/model_conversion.ts
Normal file
@ -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,
|
||||||
|
};
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user