2019-12-24 10:04:18 +08:00
|
|
|
import { GuiStore, GuiTool } from "../../core/stores/GuiStore";
|
|
|
|
import { AreaStore } from "../stores/AreaStore";
|
|
|
|
import { QuestEditorStore } from "../stores/QuestEditorStore";
|
|
|
|
import { AreaModel } from "../model/AreaModel";
|
2020-01-15 04:19:07 +08:00
|
|
|
import { list_property, map, property } from "../../core/observable";
|
2019-12-24 10:04:18 +08:00
|
|
|
import { Property } from "../../core/observable/property/Property";
|
|
|
|
import { undo_manager } from "../../core/undo/UndoManager";
|
|
|
|
import { Controller } from "../../core/controllers/Controller";
|
|
|
|
import { Episode } from "../../core/data_formats/parsing/quest/Episode";
|
|
|
|
import { create_new_quest } from "../stores/quest_creation";
|
2020-01-06 06:20:53 +08:00
|
|
|
import { open_files, read_file } from "../../core/files";
|
2020-01-02 22:26:40 +08:00
|
|
|
import {
|
|
|
|
parse_bin_dat_to_quest,
|
|
|
|
parse_qst_to_quest,
|
|
|
|
Quest,
|
|
|
|
write_quest_qst,
|
|
|
|
} from "../../core/data_formats/parsing/quest";
|
2019-12-24 10:04:18 +08:00
|
|
|
import { ArrayBufferCursor } from "../../core/data_formats/cursor/ArrayBufferCursor";
|
|
|
|
import { Endianness } from "../../core/data_formats/Endianness";
|
|
|
|
import { convert_quest_from_model, convert_quest_to_model } from "../stores/model_conversion";
|
2019-12-25 07:17:02 +08:00
|
|
|
import { LogManager } from "../../core/Logger";
|
2020-01-02 22:26:40 +08:00
|
|
|
import { basename } from "../../core/util";
|
2019-12-24 10:04:18 +08:00
|
|
|
|
2019-12-25 07:17:02 +08:00
|
|
|
const logger = LogManager.get("quest_editor/controllers/QuestEditorToolBarController");
|
2019-12-24 10:04:18 +08:00
|
|
|
|
|
|
|
export type AreaAndLabel = { readonly area: AreaModel; readonly label: string };
|
|
|
|
|
|
|
|
export class QuestEditorToolBarController extends Controller {
|
2020-01-15 04:19:07 +08:00
|
|
|
private _save_as_dialog_visible = property(false);
|
|
|
|
private _filename = property("");
|
2019-12-24 10:04:18 +08:00
|
|
|
|
|
|
|
readonly vm_feature_active: boolean;
|
|
|
|
readonly areas: Property<readonly AreaAndLabel[]>;
|
|
|
|
readonly current_area: Property<AreaAndLabel>;
|
|
|
|
readonly can_save: Property<boolean>;
|
|
|
|
readonly can_undo: Property<boolean>;
|
|
|
|
readonly can_redo: Property<boolean>;
|
|
|
|
readonly can_select_area: Property<boolean>;
|
|
|
|
readonly can_debug: Property<boolean>;
|
|
|
|
readonly can_step: Property<boolean>;
|
|
|
|
readonly can_stop: Property<boolean>;
|
2020-01-15 04:19:07 +08:00
|
|
|
readonly save_as_dialog_visible: Property<boolean> = this._save_as_dialog_visible;
|
|
|
|
readonly filename: Property<string> = this._filename;
|
2019-12-24 10:04:18 +08:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
gui_store: GuiStore,
|
|
|
|
private readonly area_store: AreaStore,
|
|
|
|
private readonly quest_editor_store: QuestEditorStore,
|
|
|
|
) {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.vm_feature_active = gui_store.feature_active("vm");
|
|
|
|
|
|
|
|
// Ensure the areas list is updated when entities are added or removed (the count in the
|
|
|
|
// label should update).
|
|
|
|
this.areas = quest_editor_store.current_quest.flat_map(quest => {
|
|
|
|
if (quest) {
|
|
|
|
return quest?.entities_per_area.flat_map(entities_per_area => {
|
|
|
|
return list_property<AreaAndLabel>(
|
|
|
|
undefined,
|
|
|
|
...area_store.get_areas_for_episode(quest.episode).map(area => {
|
|
|
|
const entity_count = entities_per_area.get(area.id);
|
|
|
|
return {
|
|
|
|
area,
|
|
|
|
label: area.name + (entity_count ? ` (${entity_count})` : ""),
|
|
|
|
};
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
return list_property<AreaAndLabel>();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.current_area = map(
|
|
|
|
(areas, area) => areas.find(al => al.area == area)!,
|
|
|
|
this.areas,
|
|
|
|
quest_editor_store.current_area,
|
|
|
|
);
|
|
|
|
|
|
|
|
const quest_loaded = quest_editor_store.current_quest.map(q => q != undefined);
|
|
|
|
this.can_save = quest_loaded;
|
|
|
|
this.can_select_area = quest_loaded;
|
|
|
|
this.can_debug = quest_loaded;
|
|
|
|
|
|
|
|
this.can_undo = map(
|
|
|
|
(c, r) => c && !r,
|
|
|
|
undo_manager.can_undo,
|
|
|
|
quest_editor_store.quest_runner.running,
|
|
|
|
);
|
|
|
|
|
|
|
|
this.can_redo = map(
|
|
|
|
(c, r) => c && !r,
|
|
|
|
undo_manager.can_redo,
|
|
|
|
quest_editor_store.quest_runner.running,
|
|
|
|
);
|
|
|
|
|
|
|
|
this.can_step = quest_editor_store.quest_runner.paused;
|
|
|
|
|
|
|
|
this.can_stop = quest_editor_store.quest_runner.running;
|
|
|
|
|
|
|
|
this.disposables(
|
2020-01-15 04:19:07 +08:00
|
|
|
quest_editor_store.current_quest.observe(() => this.set_filename("")),
|
2019-12-24 10:04:18 +08:00
|
|
|
|
2020-01-06 06:20:53 +08:00
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-O", async () => {
|
|
|
|
const files = await open_files({ accept: ".bin, .dat, .qst", multiple: true });
|
|
|
|
this.parse_files(files);
|
2019-12-24 10:04:18 +08:00
|
|
|
}),
|
|
|
|
|
2020-01-15 04:19:07 +08:00
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-Shift-S", this.save_as_clicked),
|
2019-12-24 10:04:18 +08:00
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-Z", () => {
|
|
|
|
undo_manager.undo();
|
|
|
|
}),
|
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-Shift-Z", () => {
|
|
|
|
undo_manager.redo();
|
|
|
|
}),
|
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "F5", this.debug),
|
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "Shift-F5", this.stop),
|
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "F6", this.resume),
|
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "F8", this.step_over),
|
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "F7", this.step_in),
|
|
|
|
|
|
|
|
gui_store.on_global_keydown(GuiTool.QuestEditor, "Shift-F8", this.step_out),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
create_new_quest = async (episode: Episode): Promise<void> =>
|
2019-12-30 06:00:09 +08:00
|
|
|
this.quest_editor_store.set_current_quest(create_new_quest(this.area_store, episode));
|
2019-12-24 10:04:18 +08:00
|
|
|
|
|
|
|
// TODO: notify user of problems.
|
2020-01-06 06:20:53 +08:00
|
|
|
parse_files = async (files: File[]): Promise<void> => {
|
2019-12-24 10:04:18 +08:00
|
|
|
try {
|
2020-01-03 01:42:08 +08:00
|
|
|
if (files.length === 0) return;
|
|
|
|
|
2020-01-02 22:26:40 +08:00
|
|
|
let quest: Quest | undefined;
|
|
|
|
|
|
|
|
const qst = files.find(f => f.name.toLowerCase().endsWith(".qst"));
|
|
|
|
|
|
|
|
if (qst) {
|
|
|
|
const buffer = await read_file(qst);
|
|
|
|
quest = parse_qst_to_quest(new ArrayBufferCursor(buffer, Endianness.Little));
|
2020-01-15 04:19:07 +08:00
|
|
|
this.set_filename(basename(qst.name));
|
2020-01-03 01:42:08 +08:00
|
|
|
|
|
|
|
if (!quest) {
|
|
|
|
logger.error("Couldn't parse quest file.");
|
|
|
|
}
|
2020-01-02 22:26:40 +08:00
|
|
|
} else {
|
|
|
|
const bin = files.find(f => f.name.toLowerCase().endsWith(".bin"));
|
|
|
|
const dat = files.find(f => f.name.toLowerCase().endsWith(".dat"));
|
|
|
|
|
|
|
|
if (bin && dat) {
|
|
|
|
const bin_buffer = await read_file(bin);
|
|
|
|
const dat_buffer = await read_file(dat);
|
|
|
|
quest = parse_bin_dat_to_quest(
|
|
|
|
new ArrayBufferCursor(bin_buffer, Endianness.Little),
|
|
|
|
new ArrayBufferCursor(dat_buffer, Endianness.Little),
|
|
|
|
);
|
2020-01-15 04:19:07 +08:00
|
|
|
this.set_filename(basename(bin.name || dat.name));
|
2019-12-24 10:04:18 +08:00
|
|
|
|
2020-01-03 01:42:08 +08:00
|
|
|
if (!quest) {
|
|
|
|
logger.error("Couldn't parse quest file.");
|
|
|
|
}
|
|
|
|
}
|
2019-12-24 10:04:18 +08:00
|
|
|
}
|
|
|
|
|
2019-12-30 06:00:09 +08:00
|
|
|
await this.quest_editor_store.set_current_quest(
|
2019-12-24 10:04:18 +08:00
|
|
|
quest && convert_quest_to_model(this.area_store, quest),
|
|
|
|
);
|
|
|
|
} catch (e) {
|
|
|
|
logger.error("Couldn't read file.", e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
set_area = ({ area }: AreaAndLabel): void => {
|
|
|
|
this.quest_editor_store.set_current_area(area);
|
|
|
|
};
|
|
|
|
|
2020-01-15 04:19:07 +08:00
|
|
|
save_as_clicked = (): void => {
|
|
|
|
if (this.quest_editor_store.current_quest.val) {
|
|
|
|
this._save_as_dialog_visible.val = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-12-24 10:04:18 +08:00
|
|
|
save_as = (): void => {
|
|
|
|
const quest = this.quest_editor_store.current_quest.val;
|
|
|
|
if (!quest) return;
|
|
|
|
|
2020-01-15 04:19:07 +08:00
|
|
|
let filename = this.filename.val;
|
|
|
|
const buffer = write_quest_qst(convert_quest_from_model(quest), filename);
|
2019-12-24 10:04:18 +08:00
|
|
|
|
2020-01-15 04:19:07 +08:00
|
|
|
if (!filename.endsWith(".qst")) {
|
|
|
|
filename += ".qst";
|
2019-12-24 10:04:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const a = document.createElement("a");
|
|
|
|
a.href = URL.createObjectURL(new Blob([buffer], { type: "application/octet-stream" }));
|
2020-01-15 04:19:07 +08:00
|
|
|
a.download = filename;
|
2019-12-24 10:04:18 +08:00
|
|
|
document.body.appendChild(a);
|
|
|
|
a.click();
|
|
|
|
URL.revokeObjectURL(a.href);
|
|
|
|
document.body.removeChild(a);
|
2020-01-15 04:19:07 +08:00
|
|
|
|
|
|
|
this.dismiss_save_as_dialog();
|
|
|
|
};
|
|
|
|
|
|
|
|
dismiss_save_as_dialog = (): void => {
|
|
|
|
this._save_as_dialog_visible.val = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
set_filename = (filename: string): void => {
|
|
|
|
this._filename.val = filename;
|
2019-12-24 10:04:18 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
debug = (): void => {
|
|
|
|
const quest = this.quest_editor_store.current_quest.val;
|
|
|
|
|
|
|
|
if (quest) {
|
|
|
|
this.quest_editor_store.quest_runner.run(quest);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
resume = (): void => {
|
|
|
|
this.quest_editor_store.quest_runner.resume();
|
|
|
|
};
|
|
|
|
|
|
|
|
step_over = (): void => {
|
|
|
|
this.quest_editor_store.quest_runner.step_over();
|
|
|
|
};
|
|
|
|
|
|
|
|
step_in = (): void => {
|
|
|
|
this.quest_editor_store.quest_runner.step_into();
|
|
|
|
};
|
|
|
|
|
|
|
|
step_out = (): void => {
|
|
|
|
this.quest_editor_store.quest_runner.step_out();
|
|
|
|
};
|
|
|
|
|
|
|
|
stop = (): void => {
|
|
|
|
this.quest_editor_store.quest_runner.stop();
|
|
|
|
};
|
|
|
|
}
|