import { computed, observable } from 'mobx'; import { Object3D } from 'three'; import { ArrayBufferCursor } from '../bin-data/ArrayBufferCursor'; import { DatNpc, DatObject, DatUnknown } from '../bin-data/parsing/dat'; import { NpcType } from './NpcType'; import { ObjectType } from './ObjectType'; import { enumValues } from '../enums'; export { NpcType } from './NpcType'; export { ObjectType } from './ObjectType'; export const RARE_ENEMY_PROB = 1 / 512; export const KONDRIEU_PROB = 1 / 512; export enum Server { Ephinea = 'Ephinea' } export const Servers: Server[] = enumValues(Server); export enum SectionId { Viridia = 'Viridia', Greenill = 'Greenill', Skyly = 'Skyly', Bluefull = 'Bluefull', Purplenum = 'Purplenum', Pinkal = 'Pinkal', Redria = 'Redria', Oran = 'Oran', Yellowboze = 'Yellowboze', Whitill = 'Whitill', } export const SectionIds: SectionId[] = enumValues(SectionId); export enum Difficulty { Normal = 'Normal', Hard = 'Hard', VHard = 'VHard', Ultimate = 'Ultimate' } export const Difficulties: Difficulty[] = enumValues(Difficulty); export class Vec3 { x: number; y: number; z: number; constructor(x?: number, y?: number, z?: number) { this.x = x || 0; this.y = y || 0; this.z = z || 0; } add(v: Vec3): Vec3 { this.x += v.x; this.y += v.y; this.z += v.z; return this; } clone(x?: number, y?: number, z?: number) { return new Vec3( typeof x === 'number' ? x : this.x, typeof y === 'number' ? y : this.y, typeof z === 'number' ? z : this.z); } }; export class Section { id: number; @observable position: Vec3; @observable yAxisRotation: number; @computed get sinYAxisRotation(): number { return Math.sin(this.yAxisRotation); } @computed get cosYAxisRotation(): number { return Math.cos(this.yAxisRotation); } constructor( id: number, position: Vec3, yAxisRotation: number ) { if (!Number.isInteger(id) || id < -1) throw new Error(`Expected id to be an integer greater than or equal to -1, got ${id}.`); if (!position) throw new Error('position is required.'); if (typeof yAxisRotation !== 'number') throw new Error('yAxisRotation is required.'); this.id = id; this.position = position; this.yAxisRotation = yAxisRotation; } } export class Quest { @observable name: string; @observable shortDescription: string; @observable longDescription: string; @observable questNo?: number; @observable episode: number; @observable areaVariants: AreaVariant[]; @observable objects: QuestObject[]; @observable npcs: QuestNpc[]; /** * (Partial) raw DAT data that can't be parsed yet by Phantasmal. */ datUnkowns: DatUnknown[]; /** * (Partial) raw BIN data that can't be parsed yet by Phantasmal. */ binData: ArrayBufferCursor; constructor( name: string, shortDescription: string, longDescription: string, questNo: number | undefined, episode: number, areaVariants: AreaVariant[], objects: QuestObject[], npcs: QuestNpc[], datUnknowns: DatUnknown[], binData: ArrayBufferCursor ) { if (questNo != null && (!Number.isInteger(questNo) || questNo < 0)) throw new Error('questNo should be null or a non-negative integer.'); if (episode !== 1 && episode !== 2 && episode !== 4) throw new Error('episode should be 1, 2 or 4.'); if (!objects || !(objects instanceof Array)) throw new Error('objs is required.'); if (!npcs || !(npcs instanceof Array)) throw new Error('npcs is required.'); this.name = name; this.shortDescription = shortDescription; this.longDescription = longDescription; this.questNo = questNo; this.episode = episode; this.areaVariants = areaVariants; this.objects = objects; this.npcs = npcs; this.datUnkowns = datUnknowns; this.binData = binData; } } /** * Abstract class from which QuestNpc and QuestObject derive. */ export class QuestEntity { @observable areaId: number; private _sectionId: number; @computed get sectionId(): number { return this.section ? this.section.id : this._sectionId; } @observable section?: Section; /** * World position */ @observable position: Vec3; @observable rotation: Vec3; /** * Section-relative position */ @computed get sectionPosition(): Vec3 { let { x, y, z } = this.position; if (this.section) { const relX = x - this.section.position.x; const relY = y - this.section.position.y; const relZ = z - this.section.position.z; const sin = -this.section.sinYAxisRotation; const cos = this.section.cosYAxisRotation; const rotX = cos * relX + sin * relZ; const rotZ = -sin * relX + cos * relZ; x = rotX; y = relY; z = rotZ; } return new Vec3(x, y, z); } set sectionPosition(sectPos: Vec3) { let { x: relX, y: relY, z: relZ } = sectPos; if (this.section) { const sin = -this.section.sinYAxisRotation; const cos = this.section.cosYAxisRotation; const rotX = cos * relX - sin * relZ; const rotZ = sin * relX + cos * relZ; const x = rotX + this.section.position.x; const y = relY + this.section.position.y; const z = rotZ + this.section.position.z; this.position = new Vec3(x, y, z); } } object3d?: Object3D; constructor( areaId: number, sectionId: number, position: Vec3, rotation: Vec3 ) { if (Object.getPrototypeOf(this) === Object.getPrototypeOf(QuestEntity)) throw new Error('Abstract class should not be instantiated directly.'); if (!Number.isInteger(areaId) || areaId < 0) throw new Error(`Expected areaId to be a non-negative integer, got ${areaId}.`); if (!Number.isInteger(sectionId) || sectionId < 0) throw new Error(`Expected sectionId to be a non-negative integer, got ${sectionId}.`); if (!position) throw new Error('position is required.'); if (!rotation) throw new Error('rotation is required.'); this.areaId = areaId; this._sectionId = sectionId; this.position = position; this.rotation = rotation; } } export class QuestObject extends QuestEntity { @observable type: ObjectType; /** * The raw data from a DAT file. */ dat: DatObject; constructor( areaId: number, sectionId: number, position: Vec3, rotation: Vec3, type: ObjectType, dat: DatObject ) { super(areaId, sectionId, position, rotation); if (!type) throw new Error('type is required.'); this.type = type; this.dat = dat; } } export class QuestNpc extends QuestEntity { @observable type: NpcType; /** * The raw data from a DAT file. */ dat: DatNpc; constructor( areaId: number, sectionId: number, position: Vec3, rotation: Vec3, type: NpcType, dat: DatNpc ) { super(areaId, sectionId, position, rotation); if (!type) throw new Error('type is required.'); this.type = type; this.dat = dat; } } export class Area { id: number; name: string; order: number; areaVariants: AreaVariant[]; constructor(id: number, name: string, order: number, areaVariants: AreaVariant[]) { if (!Number.isInteger(id) || id < 0) throw new Error(`Expected id to be a non-negative integer, got ${id}.`); if (!name) throw new Error('name is required.'); if (!areaVariants) throw new Error('areaVariants is required.'); this.id = id; this.name = name; this.order = order; this.areaVariants = areaVariants; } } export class AreaVariant { @observable sections: Section[] = []; constructor(public id: number, public area: Area) { if (!Number.isInteger(id) || id < 0) throw new Error(`Expected id to be a non-negative integer, got ${id}.`); } } export class Item { constructor(public name: string) { } } type ItemDrop = { item: Item, anythingRate: number, rareRate: number } export class EnemyDrop implements ItemDrop { rate: number; constructor( public item: Item, public anythingRate: number, public rareRate: number ) { this.rate = anythingRate * rareRate; } } export class HuntMethod { constructor( /** * The time it takes to complete the quest in hours. */ public time: number, public name: string, public quest: SimpleQuest ) { } } export class SimpleQuest { enemies: SimpleNpc[]; constructor( public name: string, public npcs: SimpleNpc[] ) { if (!name) throw new Error('name is required.'); if (!npcs) throw new Error('npcs is required.'); this.enemies = npcs.filter(npc => npc.type.enemy); } } export class SimpleNpc { constructor( public type: NpcType ) { if (!type) throw new Error('type is required.'); } }