phantasmal-world/src/stores/QuestEditorStore.ts

146 lines
5.1 KiB
TypeScript
Raw Normal View History

import Logger from 'js-logger';
import { action, observable } from 'mobx';
2019-07-01 14:53:16 +08:00
import { BufferCursor } from '../data_formats/BufferCursor';
import { parse_quest, write_quest_qst } from '../data_formats/parsing/quest';
2019-07-02 23:00:24 +08:00
import { Area, Quest, QuestEntity, Section } from '../domain';
import { Vec3 } from "../data_formats/Vec3";
2019-06-30 05:18:52 +08:00
import { create_npc_mesh as create_npc_object_3d, create_object_mesh as create_object_object_3d } from '../rendering/entities';
2019-07-01 14:53:16 +08:00
import { area_store } from './AreaStore';
import { entity_store } from './EntityStore';
2019-06-22 02:06:55 +08:00
const logger = Logger.get('stores/QuestEditorStore');
class QuestEditorStore {
@observable current_quest?: Quest;
@observable current_area?: Area;
@observable selected_entity?: QuestEntity;
set_quest = action('set_quest', (quest?: Quest) => {
this.reset_quest_state();
this.current_quest = quest;
2019-06-26 23:47:53 +08:00
if (quest && quest.area_variants.length) {
this.current_area = quest.area_variants[0].area;
}
})
private reset_quest_state() {
this.current_quest = undefined;
this.current_area = undefined;
this.selected_entity = undefined;
}
2019-06-30 05:18:52 +08:00
set_selected_entity = (entity?: QuestEntity) => {
this.selected_entity = entity;
}
set_current_area_id = action('set_current_area_id', (area_id?: number) => {
this.selected_entity = undefined;
if (area_id == null) {
this.current_area = undefined;
} else if (this.current_quest) {
const area_variant = this.current_quest.area_variants.find(
variant => variant.area.id === area_id
);
this.current_area = area_variant && area_variant.area;
}
})
load_file = (file: File) => {
const reader = new FileReader();
reader.addEventListener('loadend', () => { this.loadend(file, reader) });
reader.readAsArrayBuffer(file);
}
// TODO: notify user of problems.
private loadend = async (file: File, reader: FileReader) => {
if (!(reader.result instanceof ArrayBuffer)) {
2019-06-22 02:06:55 +08:00
logger.error('Couldn\'t read file.');
return;
}
const quest = parse_quest(new BufferCursor(reader.result, true));
this.set_quest(quest);
if (quest) {
// Load section data.
for (const variant of quest.area_variants) {
2019-07-01 14:53:16 +08:00
const sections = await area_store.get_area_sections(
quest.episode,
variant.area.id,
variant.id
);
variant.sections = sections;
// Generate object geometry.
for (const object of quest.objects.filter(o => o.area_id === variant.area.id)) {
try {
2019-07-01 14:53:16 +08:00
const object_geom = await entity_store.get_object_geometry(object.type);
this.set_section_on_visible_quest_entity(object, sections);
object.object_3d = create_object_object_3d(object, object_geom);
} catch (e) {
logger.error(e);
}
}
// Generate NPC geometry.
for (const npc of quest.npcs.filter(npc => npc.area_id === variant.area.id)) {
try {
2019-07-01 14:53:16 +08:00
const npc_geom = await entity_store.get_npc_geometry(npc.type);
this.set_section_on_visible_quest_entity(npc, sections);
npc.object_3d = create_npc_object_3d(npc, npc_geom);
} catch (e) {
logger.error(e);
}
}
}
} else {
logger.error('Couldn\'t parse quest file.');
}
}
private set_section_on_visible_quest_entity = async (
entity: QuestEntity,
sections: Section[]
) => {
let { x, y, z } = entity.position;
2019-06-26 23:47:53 +08:00
const section = sections.find(s => s.id === entity.section_id);
entity.section = section;
if (section) {
const { x: sec_x, y: sec_y, z: sec_z } = section.position;
const rot_x = section.cos_y_axis_rotation * x + section.sin_y_axis_rotation * z;
const rot_z = -section.sin_y_axis_rotation * x + section.cos_y_axis_rotation * z;
x = rot_x + sec_x;
y += sec_y;
z = rot_z + sec_z;
} else {
2019-06-26 23:47:53 +08:00
logger.warn(`Section ${entity.section_id} not found.`);
}
entity.position = new Vec3(x, y, z);
}
save_current_quest_to_file = (file_name: string) => {
if (this.current_quest) {
const cursor = write_quest_qst(this.current_quest, file_name);
if (!file_name.endsWith('.qst')) {
file_name += '.qst';
}
const a = document.createElement('a');
a.href = URL.createObjectURL(new Blob([cursor.buffer]));
a.download = file_name;
document.body.appendChild(a);
a.click();
URL.revokeObjectURL(a.href);
document.body.removeChild(a);
}
}
}
export const quest_editor_store = new QuestEditorStore();