import Logger from 'js-logger'; import { action, observable } from 'mobx'; import { AnimationClip, AnimationMixer, Object3D } from 'three'; import { BufferCursor } from '../bin_data/BufferCursor'; import { get_area_sections } from '../bin_data/loading/areas'; import { get_npc_geometry, get_object_geometry } from '../bin_data/loading/entities'; import { parse_nj, parse_xj } from '../bin_data/parsing/ninja'; import { parse_njm_4 } from '../bin_data/parsing/ninja/motion'; import { parse_quest, write_quest_qst } from '../bin_data/parsing/quest'; import { Area, Quest, QuestEntity, Section, Vec3 } from '../domain'; import { create_animation_clip } from '../rendering/animation'; import { create_npc_mesh, create_object_mesh } from '../rendering/entities'; import { create_model_mesh } from '../rendering/models'; const logger = Logger.get('stores/QuestEditorStore'); class QuestEditorStore { @observable current_quest?: Quest; @observable current_area?: Area; @observable selected_entity?: QuestEntity; @observable.ref current_model?: Object3D; @observable.ref animation_mixer?: AnimationMixer; set_quest = action('set_quest', (quest?: Quest) => { this.reset_model_and_quest_state(); this.current_quest = quest; if (quest && quest.area_variants.length) { this.current_area = quest.area_variants[0].area; } }) set_model = action('set_model', (model?: Object3D) => { this.reset_model_and_quest_state(); this.current_model = model; }) add_animation = action('add_animation', (clip: AnimationClip) => { if (!this.current_model) return; if (this.animation_mixer) { this.animation_mixer.stopAllAction(); this.animation_mixer.uncacheRoot(this.current_model); } else { this.animation_mixer = new AnimationMixer(this.current_model); } const action = this.animation_mixer.clipAction(clip); action.play(); }) private reset_model_and_quest_state() { this.current_quest = undefined; this.current_area = undefined; this.selected_entity = undefined; if (this.current_model && this.animation_mixer) { this.animation_mixer.uncacheRoot(this.current_model); } this.current_model = undefined; this.animation_mixer = undefined; } setSelectedEntity = (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)) { logger.error('Couldn\'t read file.'); return; } if (file.name.endsWith('.nj')) { this.set_model(create_model_mesh(parse_nj(new BufferCursor(reader.result, true)))); } else if (file.name.endsWith('.xj')) { this.set_model(create_model_mesh(parse_xj(new BufferCursor(reader.result, true)))); } else if (file.name.endsWith('.njm')) { this.add_animation( create_animation_clip(parse_njm_4(new BufferCursor(reader.result, true))) ); } else { 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) { const sections = await 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 { const geometry = await get_object_geometry(object.type); this.set_section_on_visible_quest_entity(object, sections); object.object3d = create_object_mesh(object, geometry); } catch (e) { logger.error(e); } } // Generate NPC geometry. for (const npc of quest.npcs.filter(npc => npc.area_id === variant.area.id)) { try { const geometry = await get_npc_geometry(npc.type); this.set_section_on_visible_quest_entity(npc, sections); npc.object3d = create_npc_mesh(npc, geometry); } 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; 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 { 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();