mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
Wave data is now parsed/written and converted to models.
This commit is contained in:
parent
dbe4f3ab78
commit
9803bfe125
@ -54,14 +54,14 @@ export abstract class AbstractWritableCursor extends AbstractCursor implements W
|
|||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
write_u8_array(array: number[]): this {
|
write_u8_array(array: readonly number[]): this {
|
||||||
this.ensure_size(array.length);
|
this.ensure_size(array.length);
|
||||||
new Uint8Array(this.backing_buffer, this.offset + this.position).set(new Uint8Array(array));
|
new Uint8Array(this.backing_buffer, this.offset + this.position).set(new Uint8Array(array));
|
||||||
this._position += array.length;
|
this._position += array.length;
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
write_u16_array(array: number[]): this {
|
write_u16_array(array: readonly number[]): this {
|
||||||
this.ensure_size(2 * array.length);
|
this.ensure_size(2 * array.length);
|
||||||
const len = array.length;
|
const len = array.length;
|
||||||
|
|
||||||
@ -72,7 +72,7 @@ export abstract class AbstractWritableCursor extends AbstractCursor implements W
|
|||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
write_u32_array(array: number[]): this {
|
write_u32_array(array: readonly number[]): this {
|
||||||
this.ensure_size(4 * array.length);
|
this.ensure_size(4 * array.length);
|
||||||
const len = array.length;
|
const len = array.length;
|
||||||
|
|
||||||
|
@ -12,7 +12,7 @@ export class ResizableBufferCursor extends AbstractWritableCursor implements Wri
|
|||||||
|
|
||||||
set size(size: number) {
|
set size(size: number) {
|
||||||
if (size > this._size) {
|
if (size > this._size) {
|
||||||
this.ensure_size(size - this._size);
|
this.ensure_size(size - this.position);
|
||||||
} else {
|
} else {
|
||||||
this._size = size;
|
this._size = size;
|
||||||
}
|
}
|
||||||
|
@ -30,17 +30,15 @@ import { ResizableBuffer } from "../../ResizableBuffer";
|
|||||||
|
|
||||||
const logger = Logger.get("core/data_formats/parsing/quest/bin");
|
const logger = Logger.get("core/data_formats/parsing/quest/bin");
|
||||||
|
|
||||||
export class BinFile {
|
export type BinFile = {
|
||||||
constructor(
|
readonly quest_id: number;
|
||||||
readonly quest_id: number,
|
readonly language: number;
|
||||||
readonly language: number,
|
readonly quest_name: string;
|
||||||
readonly quest_name: string,
|
readonly short_description: string;
|
||||||
readonly short_description: string,
|
readonly long_description: string;
|
||||||
readonly long_description: string,
|
readonly object_code: readonly Segment[];
|
||||||
readonly object_code: Segment[],
|
readonly shop_items: readonly number[];
|
||||||
readonly shop_items: number[],
|
};
|
||||||
) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
const SEGMENT_PRIORITY: number[] = [];
|
const SEGMENT_PRIORITY: number[] = [];
|
||||||
SEGMENT_PRIORITY[SegmentType.Instructions] = 2;
|
SEGMENT_PRIORITY[SegmentType.Instructions] = 2;
|
||||||
@ -82,15 +80,15 @@ export function parse_bin(
|
|||||||
|
|
||||||
const segments = parse_object_code(object_code, label_holder, entry_labels, lenient);
|
const segments = parse_object_code(object_code, label_holder, entry_labels, lenient);
|
||||||
|
|
||||||
return new BinFile(
|
return {
|
||||||
quest_id,
|
quest_id,
|
||||||
language,
|
language,
|
||||||
quest_name,
|
quest_name,
|
||||||
short_description,
|
short_description,
|
||||||
long_description,
|
long_description,
|
||||||
segments,
|
object_code: segments,
|
||||||
shop_items,
|
shop_items,
|
||||||
);
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function write_bin(bin: BinFile): ArrayBuffer {
|
export function write_bin(bin: BinFile): ArrayBuffer {
|
||||||
@ -697,7 +695,7 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
|||||||
|
|
||||||
function write_object_code(
|
function write_object_code(
|
||||||
cursor: WritableCursor,
|
cursor: WritableCursor,
|
||||||
segments: Segment[],
|
segments: readonly Segment[],
|
||||||
): { size: number; label_offsets: number[] } {
|
): { size: number; label_offsets: number[] } {
|
||||||
const start_pos = cursor.position;
|
const start_pos = cursor.position;
|
||||||
// Keep track of label offsets.
|
// Keep track of label offsets.
|
||||||
|
@ -2,7 +2,7 @@ import { Endianness } from "../../Endianness";
|
|||||||
import { prs_decompress } from "../../compression/prs/decompress";
|
import { prs_decompress } from "../../compression/prs/decompress";
|
||||||
import { BufferCursor } from "../../cursor/BufferCursor";
|
import { BufferCursor } from "../../cursor/BufferCursor";
|
||||||
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { parse_dat, write_dat } from "./dat";
|
import { DatFile, parse_dat, write_dat } from "./dat";
|
||||||
import { readFileSync } from "fs";
|
import { readFileSync } from "fs";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -37,13 +37,25 @@ test("parse, modify and write DAT", () => {
|
|||||||
const test_parsed = parse_dat(orig_dat);
|
const test_parsed = parse_dat(orig_dat);
|
||||||
orig_dat.seek_start(0);
|
orig_dat.seek_start(0);
|
||||||
|
|
||||||
test_parsed.objs[9].position = {
|
const test_updated: DatFile = {
|
||||||
x: 13,
|
...test_parsed,
|
||||||
y: 17,
|
objs: test_parsed.objs.map((obj, i) => {
|
||||||
z: 19,
|
if (i === 9) {
|
||||||
|
return {
|
||||||
|
...obj,
|
||||||
|
position: {
|
||||||
|
x: 13,
|
||||||
|
y: 17,
|
||||||
|
z: 19,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
const test_dat = new ResizableBufferCursor(write_dat(test_parsed), Endianness.Little);
|
const test_dat = new ResizableBufferCursor(write_dat(test_updated), Endianness.Little);
|
||||||
|
|
||||||
expect(test_dat.size).toBe(orig_dat.size);
|
expect(test_dat.size).toBe(orig_dat.size);
|
||||||
|
|
||||||
|
@ -5,51 +5,98 @@ import { Cursor } from "../../cursor/Cursor";
|
|||||||
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
import { Vec3 } from "../../vector";
|
import { Vec3 } from "../../vector";
|
||||||
|
import { WritableCursor } from "../../cursor/WritableCursor";
|
||||||
|
|
||||||
const logger = Logger.get("data_formats/parsing/quest/dat");
|
const logger = Logger.get("core/data_formats/parsing/quest/dat");
|
||||||
|
|
||||||
const OBJECT_SIZE = 68;
|
const OBJECT_SIZE = 68;
|
||||||
const NPC_SIZE = 72;
|
const NPC_SIZE = 72;
|
||||||
|
|
||||||
export type DatFile = {
|
export type DatFile = {
|
||||||
objs: DatObject[];
|
readonly objs: readonly DatObject[];
|
||||||
npcs: DatNpc[];
|
readonly npcs: readonly DatNpc[];
|
||||||
unknowns: DatUnknown[];
|
readonly waves: readonly DatWave[];
|
||||||
|
readonly unknowns: readonly DatUnknown[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DatEntity = {
|
export type DatEntity = {
|
||||||
type_id: number;
|
readonly type_id: number;
|
||||||
section_id: number;
|
readonly section_id: number;
|
||||||
position: Vec3;
|
readonly position: Vec3;
|
||||||
rotation: Vec3;
|
readonly rotation: Vec3;
|
||||||
area_id: number;
|
readonly area_id: number;
|
||||||
unknown: number[][];
|
readonly unknown: readonly number[][];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DatObject = DatEntity & {
|
export type DatObject = DatEntity & {
|
||||||
id: number;
|
readonly id: number;
|
||||||
group_id: number;
|
readonly group_id: number;
|
||||||
properties: number[];
|
readonly properties: readonly number[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DatNpc = DatEntity & {
|
export type DatNpc = DatEntity & {
|
||||||
scale: Vec3;
|
readonly scale: Vec3;
|
||||||
npc_id: number;
|
readonly npc_id: number;
|
||||||
script_label: number;
|
readonly script_label: number;
|
||||||
roaming: number;
|
readonly roaming: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DatWave = {
|
||||||
|
readonly id: number;
|
||||||
|
readonly section_id: number;
|
||||||
|
readonly wave: number;
|
||||||
|
readonly delay: number;
|
||||||
|
readonly actions: readonly DatWaveAction[];
|
||||||
|
readonly area_id: number;
|
||||||
|
readonly unknown: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export enum DatWaveActionType {
|
||||||
|
SpawnNpcs = 0x8,
|
||||||
|
Unlock = 0xa,
|
||||||
|
Lock = 0xb,
|
||||||
|
SpawnWave = 0xc,
|
||||||
|
}
|
||||||
|
|
||||||
|
export type DatWaveAction =
|
||||||
|
| DatWaveActionSpawnNpcs
|
||||||
|
| DatWaveActionUnlock
|
||||||
|
| DatWaveActionLock
|
||||||
|
| DatWaveActionSpawnWave;
|
||||||
|
|
||||||
|
export type DatWaveActionSpawnNpcs = {
|
||||||
|
readonly type: DatWaveActionType.SpawnNpcs;
|
||||||
|
readonly section_id: number;
|
||||||
|
readonly appear_flag: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DatWaveActionUnlock = {
|
||||||
|
readonly type: DatWaveActionType.Unlock;
|
||||||
|
readonly door_id: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DatWaveActionLock = {
|
||||||
|
readonly type: DatWaveActionType.Lock;
|
||||||
|
readonly door_id: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DatWaveActionSpawnWave = {
|
||||||
|
readonly type: DatWaveActionType.SpawnWave;
|
||||||
|
readonly wave_id: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DatUnknown = {
|
export type DatUnknown = {
|
||||||
entity_type: number;
|
readonly entity_type: number;
|
||||||
total_size: number;
|
readonly total_size: number;
|
||||||
area_id: number;
|
readonly area_id: number;
|
||||||
entities_size: number;
|
readonly entities_size: number;
|
||||||
data: number[];
|
readonly data: number[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export function parse_dat(cursor: Cursor): DatFile {
|
export function parse_dat(cursor: Cursor): DatFile {
|
||||||
const objs: DatObject[] = [];
|
const objs: DatObject[] = [];
|
||||||
const npcs: DatNpc[] = [];
|
const npcs: DatNpc[] = [];
|
||||||
|
const waves: DatWave[] = [];
|
||||||
const unknowns: DatUnknown[] = [];
|
const unknowns: DatUnknown[] = [];
|
||||||
|
|
||||||
while (cursor.bytes_left) {
|
while (cursor.bytes_left) {
|
||||||
@ -68,99 +115,16 @@ export function parse_dat(cursor: Cursor): DatFile {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const entities_cursor = cursor.take(entities_size);
|
||||||
|
|
||||||
if (entity_type === 1) {
|
if (entity_type === 1) {
|
||||||
// Objects
|
parse_objects(entities_cursor, area_id, objs);
|
||||||
const object_count = Math.floor(entities_size / OBJECT_SIZE);
|
|
||||||
const start_position = cursor.position;
|
|
||||||
|
|
||||||
for (let i = 0; i < object_count; ++i) {
|
|
||||||
const type_id = cursor.u16();
|
|
||||||
const unknown1 = cursor.u8_array(6);
|
|
||||||
const id = cursor.u16();
|
|
||||||
const group_id = cursor.u16();
|
|
||||||
const section_id = cursor.u16();
|
|
||||||
const unknown2 = cursor.u8_array(2);
|
|
||||||
const position = cursor.vec3_f32();
|
|
||||||
const rotation = {
|
|
||||||
x: (cursor.i32() / 0xffff) * 2 * Math.PI,
|
|
||||||
y: (cursor.i32() / 0xffff) * 2 * Math.PI,
|
|
||||||
z: (cursor.i32() / 0xffff) * 2 * Math.PI,
|
|
||||||
};
|
|
||||||
const properties = [
|
|
||||||
cursor.f32(),
|
|
||||||
cursor.f32(),
|
|
||||||
cursor.f32(),
|
|
||||||
cursor.u32(),
|
|
||||||
cursor.u32(),
|
|
||||||
cursor.u32(),
|
|
||||||
cursor.u32(),
|
|
||||||
];
|
|
||||||
|
|
||||||
objs.push({
|
|
||||||
type_id,
|
|
||||||
id,
|
|
||||||
group_id,
|
|
||||||
section_id,
|
|
||||||
position,
|
|
||||||
rotation,
|
|
||||||
properties,
|
|
||||||
area_id,
|
|
||||||
unknown: [unknown1, unknown2],
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const bytes_read = cursor.position - start_position;
|
|
||||||
|
|
||||||
if (bytes_read !== entities_size) {
|
|
||||||
logger.warn(
|
|
||||||
`Read ${bytes_read} bytes instead of expected ${entities_size} for entity type ${entity_type} (Object).`,
|
|
||||||
);
|
|
||||||
cursor.seek(entities_size - bytes_read);
|
|
||||||
}
|
|
||||||
} else if (entity_type === 2) {
|
} else if (entity_type === 2) {
|
||||||
// NPCs
|
parse_npcs(entities_cursor, area_id, npcs);
|
||||||
const npc_count = Math.floor(entities_size / NPC_SIZE);
|
} else if (entity_type === 3) {
|
||||||
const start_position = cursor.position;
|
parse_waves(entities_cursor, area_id, waves);
|
||||||
|
|
||||||
for (let i = 0; i < npc_count; ++i) {
|
|
||||||
const type_id = cursor.u16();
|
|
||||||
const unknown1 = cursor.u8_array(10);
|
|
||||||
const section_id = cursor.u16();
|
|
||||||
const unknown2 = cursor.u8_array(6);
|
|
||||||
const position = cursor.vec3_f32();
|
|
||||||
const rotation_x = (cursor.i32() / 0xffff) * 2 * Math.PI;
|
|
||||||
const rotation_y = (cursor.i32() / 0xffff) * 2 * Math.PI;
|
|
||||||
const rotation_z = (cursor.i32() / 0xffff) * 2 * Math.PI;
|
|
||||||
const scale = cursor.vec3_f32();
|
|
||||||
const npc_id = cursor.f32();
|
|
||||||
const script_label = cursor.f32();
|
|
||||||
const roaming = cursor.u32();
|
|
||||||
const unknown3 = cursor.u8_array(4);
|
|
||||||
|
|
||||||
npcs.push({
|
|
||||||
type_id,
|
|
||||||
section_id,
|
|
||||||
position,
|
|
||||||
rotation: { x: rotation_x, y: rotation_y, z: rotation_z },
|
|
||||||
scale,
|
|
||||||
npc_id,
|
|
||||||
script_label,
|
|
||||||
roaming,
|
|
||||||
area_id,
|
|
||||||
unknown: [unknown1, unknown2, unknown3],
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const bytes_read = cursor.position - start_position;
|
|
||||||
|
|
||||||
if (bytes_read !== entities_size) {
|
|
||||||
logger.warn(
|
|
||||||
`Read ${bytes_read} bytes instead of expected ${entities_size} for entity type ${entity_type} (NPC).`,
|
|
||||||
);
|
|
||||||
cursor.seek(entities_size - bytes_read);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// There are also waves (type 3) and unknown entity types 4 and 5.
|
// Unknown entity types 4 and 5.
|
||||||
unknowns.push({
|
unknowns.push({
|
||||||
entity_type,
|
entity_type,
|
||||||
total_size,
|
total_size,
|
||||||
@ -169,13 +133,19 @@ export function parse_dat(cursor: Cursor): DatFile {
|
|||||||
data: cursor.u8_array(entities_size),
|
data: cursor.u8_array(entities_size),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (entities_cursor.bytes_left) {
|
||||||
|
logger.warn(
|
||||||
|
`Read ${entities_cursor.position} bytes instead of expected ${entities_cursor.size} for entity type ${entity_type}.`,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return { objs, npcs, unknowns };
|
return { objs, npcs, waves, unknowns };
|
||||||
}
|
}
|
||||||
|
|
||||||
export function write_dat({ objs, npcs, unknowns }: DatFile): ResizableBuffer {
|
export function write_dat({ objs, npcs, waves, unknowns }: DatFile): ResizableBuffer {
|
||||||
const buffer = new ResizableBuffer(
|
const buffer = new ResizableBuffer(
|
||||||
objs.length * (16 + OBJECT_SIZE) +
|
objs.length * (16 + OBJECT_SIZE) +
|
||||||
npcs.length * (16 + NPC_SIZE) +
|
npcs.length * (16 + NPC_SIZE) +
|
||||||
@ -183,6 +153,214 @@ export function write_dat({ objs, npcs, unknowns }: DatFile): ResizableBuffer {
|
|||||||
);
|
);
|
||||||
const cursor = new ResizableBufferCursor(buffer, Endianness.Little);
|
const cursor = new ResizableBufferCursor(buffer, Endianness.Little);
|
||||||
|
|
||||||
|
write_objects(cursor, objs);
|
||||||
|
|
||||||
|
write_npcs(cursor, npcs);
|
||||||
|
|
||||||
|
write_waves(cursor, waves);
|
||||||
|
|
||||||
|
for (const unknown of unknowns) {
|
||||||
|
cursor.write_u32(unknown.entity_type);
|
||||||
|
cursor.write_u32(unknown.total_size);
|
||||||
|
cursor.write_u32(unknown.area_id);
|
||||||
|
cursor.write_u32(unknown.entities_size);
|
||||||
|
cursor.write_u8_array(unknown.data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Final header.
|
||||||
|
cursor.write_u32(0);
|
||||||
|
cursor.write_u32(0);
|
||||||
|
cursor.write_u32(0);
|
||||||
|
cursor.write_u32(0);
|
||||||
|
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
function parse_objects(cursor: Cursor, area_id: number, objs: DatObject[]): void {
|
||||||
|
const object_count = Math.floor(cursor.size / OBJECT_SIZE);
|
||||||
|
|
||||||
|
for (let i = 0; i < object_count; ++i) {
|
||||||
|
const type_id = cursor.u16();
|
||||||
|
const unknown1 = cursor.u8_array(6);
|
||||||
|
const id = cursor.u16();
|
||||||
|
const group_id = cursor.u16();
|
||||||
|
const section_id = cursor.u16();
|
||||||
|
const unknown2 = cursor.u8_array(2);
|
||||||
|
const position = cursor.vec3_f32();
|
||||||
|
const rotation = {
|
||||||
|
x: (cursor.i32() / 0xffff) * 2 * Math.PI,
|
||||||
|
y: (cursor.i32() / 0xffff) * 2 * Math.PI,
|
||||||
|
z: (cursor.i32() / 0xffff) * 2 * Math.PI,
|
||||||
|
};
|
||||||
|
const properties = [
|
||||||
|
cursor.f32(),
|
||||||
|
cursor.f32(),
|
||||||
|
cursor.f32(),
|
||||||
|
cursor.u32(),
|
||||||
|
cursor.u32(),
|
||||||
|
cursor.u32(),
|
||||||
|
cursor.u32(),
|
||||||
|
];
|
||||||
|
|
||||||
|
objs.push({
|
||||||
|
type_id,
|
||||||
|
id,
|
||||||
|
group_id,
|
||||||
|
section_id,
|
||||||
|
position,
|
||||||
|
rotation,
|
||||||
|
properties,
|
||||||
|
area_id,
|
||||||
|
unknown: [unknown1, unknown2],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function parse_npcs(cursor: Cursor, area_id: number, npcs: DatNpc[]): void {
|
||||||
|
const npc_count = Math.floor(cursor.size / NPC_SIZE);
|
||||||
|
|
||||||
|
for (let i = 0; i < npc_count; ++i) {
|
||||||
|
const type_id = cursor.u16();
|
||||||
|
const unknown1 = cursor.u8_array(10);
|
||||||
|
const section_id = cursor.u16();
|
||||||
|
const unknown2 = cursor.u8_array(6);
|
||||||
|
const position = cursor.vec3_f32();
|
||||||
|
const rotation_x = (cursor.i32() / 0xffff) * 2 * Math.PI;
|
||||||
|
const rotation_y = (cursor.i32() / 0xffff) * 2 * Math.PI;
|
||||||
|
const rotation_z = (cursor.i32() / 0xffff) * 2 * Math.PI;
|
||||||
|
const scale = cursor.vec3_f32();
|
||||||
|
const npc_id = cursor.f32();
|
||||||
|
const script_label = cursor.f32();
|
||||||
|
const roaming = cursor.u32();
|
||||||
|
const unknown3 = cursor.u8_array(4);
|
||||||
|
|
||||||
|
npcs.push({
|
||||||
|
type_id,
|
||||||
|
section_id,
|
||||||
|
position,
|
||||||
|
rotation: { x: rotation_x, y: rotation_y, z: rotation_z },
|
||||||
|
scale,
|
||||||
|
npc_id,
|
||||||
|
script_label,
|
||||||
|
roaming,
|
||||||
|
area_id,
|
||||||
|
unknown: [unknown1, unknown2, unknown3],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function parse_waves(cursor: Cursor, area_id: number, waves: DatWave[]): void {
|
||||||
|
const actions_offset = cursor.u32();
|
||||||
|
cursor.seek(4); // Always 0x10
|
||||||
|
const wave_count = cursor.u32();
|
||||||
|
cursor.seek(3); // Always 0
|
||||||
|
const wave_type = cursor.u8();
|
||||||
|
|
||||||
|
if (wave_type === 0x32) {
|
||||||
|
throw new Error("Can't parse challenge mode quests yet.");
|
||||||
|
}
|
||||||
|
|
||||||
|
cursor.seek_start(actions_offset);
|
||||||
|
const actions_cursor = cursor.take(cursor.bytes_left);
|
||||||
|
cursor.seek_start(16);
|
||||||
|
|
||||||
|
for (let i = 0; i < wave_count; ++i) {
|
||||||
|
const id = cursor.u32();
|
||||||
|
cursor.seek(4); // Always 0x100
|
||||||
|
const section_id = cursor.u16();
|
||||||
|
const wave = cursor.u16();
|
||||||
|
const delay = cursor.u16();
|
||||||
|
const unknown = cursor.u16(); // "wavesetting"?
|
||||||
|
const wave_actions_offset = cursor.u32();
|
||||||
|
|
||||||
|
actions_cursor.seek_start(wave_actions_offset);
|
||||||
|
const actions = parse_wave_actions(actions_cursor);
|
||||||
|
|
||||||
|
waves.push({
|
||||||
|
id,
|
||||||
|
section_id,
|
||||||
|
wave,
|
||||||
|
delay,
|
||||||
|
actions,
|
||||||
|
area_id,
|
||||||
|
unknown,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cursor.position !== actions_offset) {
|
||||||
|
logger.warn(
|
||||||
|
`Read ${cursor.position - 16} bytes of wave data instead of expected ${actions_offset -
|
||||||
|
16}.`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let last_u8 = 0xff;
|
||||||
|
|
||||||
|
while (actions_cursor.bytes_left) {
|
||||||
|
last_u8 = actions_cursor.u8();
|
||||||
|
|
||||||
|
if (last_u8 !== 0xff) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (last_u8 !== 0xff) {
|
||||||
|
actions_cursor.seek(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure the cursor position represents the amount of bytes we've consumed.
|
||||||
|
cursor.seek_start(actions_offset + actions_cursor.position);
|
||||||
|
}
|
||||||
|
|
||||||
|
function parse_wave_actions(cursor: Cursor): DatWaveAction[] {
|
||||||
|
const actions: DatWaveAction[] = [];
|
||||||
|
|
||||||
|
outer: while (cursor.bytes_left) {
|
||||||
|
const type = cursor.u8();
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case 1:
|
||||||
|
break outer;
|
||||||
|
|
||||||
|
case DatWaveActionType.SpawnNpcs:
|
||||||
|
actions.push({
|
||||||
|
type: DatWaveActionType.SpawnNpcs,
|
||||||
|
section_id: cursor.u16(),
|
||||||
|
appear_flag: cursor.u16(),
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DatWaveActionType.Unlock:
|
||||||
|
actions.push({
|
||||||
|
type: DatWaveActionType.Unlock,
|
||||||
|
door_id: cursor.u16(),
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DatWaveActionType.Lock:
|
||||||
|
actions.push({
|
||||||
|
type: DatWaveActionType.Lock,
|
||||||
|
door_id: cursor.u16(),
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DatWaveActionType.SpawnWave:
|
||||||
|
actions.push({
|
||||||
|
type: DatWaveActionType.SpawnWave,
|
||||||
|
wave_id: cursor.u32(),
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
logger.warn(`Unexpected wave action type ${type}.`);
|
||||||
|
break outer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return actions;
|
||||||
|
}
|
||||||
|
|
||||||
|
function write_objects(cursor: WritableCursor, objs: readonly DatObject[]): void {
|
||||||
const grouped_objs = groupBy(objs, obj => obj.area_id);
|
const grouped_objs = groupBy(objs, obj => obj.area_id);
|
||||||
const obj_area_ids = Object.keys(grouped_objs)
|
const obj_area_ids = Object.keys(grouped_objs)
|
||||||
.map(key => parseInt(key, 10))
|
.map(key => parseInt(key, 10))
|
||||||
@ -231,7 +409,9 @@ export function write_dat({ objs, npcs, unknowns }: DatFile): ResizableBuffer {
|
|||||||
cursor.write_u32(obj.properties[6]);
|
cursor.write_u32(obj.properties[6]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function write_npcs(cursor: WritableCursor, npcs: readonly DatNpc[]): void {
|
||||||
const grouped_npcs = groupBy(npcs, npc => npc.area_id);
|
const grouped_npcs = groupBy(npcs, npc => npc.area_id);
|
||||||
const npc_area_ids = Object.keys(grouped_npcs)
|
const npc_area_ids = Object.keys(grouped_npcs)
|
||||||
.map(key => parseInt(key, 10))
|
.map(key => parseInt(key, 10))
|
||||||
@ -276,20 +456,101 @@ export function write_dat({ objs, npcs, unknowns }: DatFile): ResizableBuffer {
|
|||||||
cursor.write_u8_array(npc.unknown[2]);
|
cursor.write_u8_array(npc.unknown[2]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
for (const unknown of unknowns) {
|
|
||||||
cursor.write_u32(unknown.entity_type);
|
function write_waves(cursor: WritableCursor, waves: readonly DatWave[]): void {
|
||||||
cursor.write_u32(unknown.total_size);
|
const grouped_waves = groupBy(waves, wave => wave.area_id);
|
||||||
cursor.write_u32(unknown.area_id);
|
const wave_area_ids = Object.keys(grouped_waves)
|
||||||
cursor.write_u32(unknown.entities_size);
|
.map(key => parseInt(key, 10))
|
||||||
cursor.write_u8_array(unknown.data);
|
.sort((a, b) => a - b);
|
||||||
}
|
|
||||||
|
for (const area_id of wave_area_ids) {
|
||||||
// Final header.
|
const area_waves = grouped_waves[area_id];
|
||||||
cursor.write_u32(0);
|
|
||||||
cursor.write_u32(0);
|
// Standard header.
|
||||||
cursor.write_u32(0);
|
cursor.write_u32(3); // Entity type
|
||||||
cursor.write_u32(0);
|
const total_size_offset = cursor.position;
|
||||||
|
cursor.write_u32(0); // Placeholder for the total size.
|
||||||
return buffer;
|
cursor.write_u32(area_id);
|
||||||
|
const entities_size_offset = cursor.position;
|
||||||
|
cursor.write_u32(0); // Placeholder for the entities size.
|
||||||
|
|
||||||
|
// Wave header.
|
||||||
|
const start_pos = cursor.position;
|
||||||
|
// TODO: actual wave size is dependent on the wave type (challenge mode).
|
||||||
|
// Absolute offset.
|
||||||
|
const actions_offset = start_pos + 16 + 20 * area_waves.length;
|
||||||
|
cursor.size = Math.max(actions_offset, cursor.size);
|
||||||
|
|
||||||
|
cursor.write_u32(actions_offset - start_pos);
|
||||||
|
cursor.write_u32(0x10);
|
||||||
|
cursor.write_u32(area_waves.length);
|
||||||
|
cursor.write_u32(0); // TODO: write wave type (challenge mode).
|
||||||
|
|
||||||
|
// Relative offset.
|
||||||
|
let wave_actions_offset = 0;
|
||||||
|
|
||||||
|
for (const wave of area_waves) {
|
||||||
|
cursor.write_u32(wave.id);
|
||||||
|
cursor.write_u32(0x10000);
|
||||||
|
cursor.write_u16(wave.section_id);
|
||||||
|
cursor.write_u16(wave.wave);
|
||||||
|
cursor.write_u16(wave.delay);
|
||||||
|
cursor.write_u16(wave.unknown);
|
||||||
|
cursor.write_u32(wave_actions_offset);
|
||||||
|
const next_wave_pos = cursor.position;
|
||||||
|
|
||||||
|
cursor.seek_start(actions_offset + wave_actions_offset);
|
||||||
|
|
||||||
|
for (const action of wave.actions) {
|
||||||
|
cursor.write_u8(action.type);
|
||||||
|
|
||||||
|
switch (action.type) {
|
||||||
|
case DatWaveActionType.SpawnNpcs:
|
||||||
|
cursor.write_u16(action.section_id);
|
||||||
|
cursor.write_u16(action.appear_flag);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DatWaveActionType.Unlock:
|
||||||
|
cursor.write_u16(action.door_id);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DatWaveActionType.Lock:
|
||||||
|
cursor.write_u16(action.door_id);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case DatWaveActionType.SpawnWave:
|
||||||
|
cursor.write_u32(action.wave_id);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
// Need to cast because TypeScript infers action to be `never`.
|
||||||
|
throw new Error(`Unknown wave action type ${(action as any).type}.`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// End of wave actions.
|
||||||
|
cursor.write_u8(1);
|
||||||
|
|
||||||
|
wave_actions_offset = cursor.position - actions_offset;
|
||||||
|
|
||||||
|
cursor.seek_start(next_wave_pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
cursor.seek_start(actions_offset + wave_actions_offset);
|
||||||
|
|
||||||
|
while ((cursor.position - actions_offset) % 4 !== 0) {
|
||||||
|
cursor.write_u8(0xff);
|
||||||
|
}
|
||||||
|
|
||||||
|
const end_pos = cursor.position;
|
||||||
|
|
||||||
|
cursor.seek_start(total_size_offset);
|
||||||
|
cursor.write_u32(end_pos - start_pos + 16);
|
||||||
|
|
||||||
|
cursor.seek_start(entities_size_offset);
|
||||||
|
cursor.write_u32(end_pos - start_pos);
|
||||||
|
|
||||||
|
cursor.seek_start(end_pos);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
import { Vec3 } from "../../vector";
|
import { Vec3 } from "../../vector";
|
||||||
import { npc_data, NpcType, NpcTypeData } from "./npc_types";
|
import { npc_data, NpcType, NpcTypeData } from "./npc_types";
|
||||||
import { object_data, ObjectType, ObjectTypeData } from "./object_types";
|
import { object_data, ObjectType, ObjectTypeData } from "./object_types";
|
||||||
|
import { DatWave } from "./dat";
|
||||||
export type QuestEntity = QuestNpc | QuestObject;
|
|
||||||
|
|
||||||
export type QuestNpc = {
|
export type QuestNpc = {
|
||||||
readonly type: NpcType;
|
readonly type: NpcType;
|
||||||
@ -20,7 +19,7 @@ export type QuestNpc = {
|
|||||||
/**
|
/**
|
||||||
* Data of which the purpose hasn't been discovered yet.
|
* Data of which the purpose hasn't been discovered yet.
|
||||||
*/
|
*/
|
||||||
readonly unknown: number[][];
|
readonly unknown: readonly number[][];
|
||||||
readonly pso_type_id: number;
|
readonly pso_type_id: number;
|
||||||
readonly npc_id: number;
|
readonly npc_id: number;
|
||||||
readonly script_label: number;
|
readonly script_label: number;
|
||||||
@ -45,9 +44,11 @@ export type QuestObject = {
|
|||||||
/**
|
/**
|
||||||
* Data of which the purpose hasn't been discovered yet.
|
* Data of which the purpose hasn't been discovered yet.
|
||||||
*/
|
*/
|
||||||
readonly unknown: number[][];
|
readonly unknown: readonly number[][];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type QuestWave = DatWave;
|
||||||
|
|
||||||
export type EntityTypeData = NpcTypeData | ObjectTypeData;
|
export type EntityTypeData = NpcTypeData | ObjectTypeData;
|
||||||
|
|
||||||
export type EntityType = NpcType | ObjectType;
|
export type EntityType = NpcType | ObjectType;
|
||||||
|
@ -12,9 +12,9 @@ import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
|||||||
import { Cursor } from "../../cursor/Cursor";
|
import { Cursor } from "../../cursor/Cursor";
|
||||||
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { Endianness } from "../../Endianness";
|
import { Endianness } from "../../Endianness";
|
||||||
import { BinFile, parse_bin, write_bin } from "./bin";
|
import { parse_bin, write_bin } from "./bin";
|
||||||
import { DatFile, DatNpc, DatObject, DatUnknown, parse_dat, write_dat } from "./dat";
|
import { DatFile, DatNpc, DatObject, DatUnknown, parse_dat, write_dat } from "./dat";
|
||||||
import { QuestNpc, QuestObject } from "./entities";
|
import { QuestNpc, QuestObject, QuestWave } from "./entities";
|
||||||
import { Episode } from "./Episode";
|
import { Episode } from "./Episode";
|
||||||
import { object_data, ObjectType, pso_id_to_object_type } from "./object_types";
|
import { object_data, ObjectType, pso_id_to_object_type } from "./object_types";
|
||||||
import { parse_qst, QstContainedFile, write_qst } from "./qst";
|
import { parse_qst, QstContainedFile, write_qst } from "./qst";
|
||||||
@ -29,14 +29,15 @@ export type Quest = {
|
|||||||
readonly short_description: string;
|
readonly short_description: string;
|
||||||
readonly long_description: string;
|
readonly long_description: string;
|
||||||
readonly episode: Episode;
|
readonly episode: Episode;
|
||||||
readonly objects: QuestObject[];
|
readonly objects: readonly QuestObject[];
|
||||||
readonly npcs: QuestNpc[];
|
readonly npcs: readonly QuestNpc[];
|
||||||
|
readonly waves: readonly QuestWave[];
|
||||||
/**
|
/**
|
||||||
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
||||||
*/
|
*/
|
||||||
readonly dat_unknowns: DatUnknown[];
|
readonly dat_unknowns: readonly DatUnknown[];
|
||||||
readonly object_code: Segment[];
|
readonly object_code: readonly Segment[];
|
||||||
readonly shop_items: number[];
|
readonly shop_items: readonly number[];
|
||||||
readonly map_designations: Map<number, number>;
|
readonly map_designations: Map<number, number>;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -125,6 +126,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,
|
||||||
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,
|
||||||
@ -136,19 +138,18 @@ 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,
|
||||||
unknowns: quest.dat_unknowns,
|
unknowns: quest.dat_unknowns,
|
||||||
});
|
});
|
||||||
const bin = write_bin(
|
const bin = write_bin({
|
||||||
new BinFile(
|
quest_id: quest.id,
|
||||||
quest.id,
|
language: quest.language,
|
||||||
quest.language,
|
quest_name: quest.name,
|
||||||
quest.name,
|
short_description: quest.short_description,
|
||||||
quest.short_description,
|
long_description: quest.long_description,
|
||||||
quest.long_description,
|
object_code: quest.object_code,
|
||||||
quest.object_code,
|
shop_items: quest.shop_items,
|
||||||
quest.shop_items,
|
});
|
||||||
),
|
|
||||||
);
|
|
||||||
const ext_start = file_name.lastIndexOf(".");
|
const ext_start = file_name.lastIndexOf(".");
|
||||||
const base_file_name =
|
const base_file_name =
|
||||||
ext_start === -1 ? file_name.slice(0, 11) : file_name.slice(0, Math.min(11, ext_start));
|
ext_start === -1 ? file_name.slice(0, 11) : file_name.slice(0, Math.min(11, ext_start));
|
||||||
@ -211,7 +212,10 @@ function extract_map_designations(
|
|||||||
return map_designations;
|
return map_designations;
|
||||||
}
|
}
|
||||||
|
|
||||||
function extract_script_entry_points(objects: QuestObject[], npcs: DatNpc[]): number[] {
|
function extract_script_entry_points(
|
||||||
|
objects: readonly QuestObject[],
|
||||||
|
npcs: readonly DatNpc[],
|
||||||
|
): number[] {
|
||||||
const entry_points = new Set([0]);
|
const entry_points = new Set([0]);
|
||||||
|
|
||||||
for (const obj of objects) {
|
for (const obj of objects) {
|
||||||
@ -235,7 +239,7 @@ function extract_script_entry_points(objects: QuestObject[], npcs: DatNpc[]): nu
|
|||||||
return [...entry_points];
|
return [...entry_points];
|
||||||
}
|
}
|
||||||
|
|
||||||
function parse_obj_data(objs: DatObject[]): QuestObject[] {
|
function parse_obj_data(objs: readonly DatObject[]): QuestObject[] {
|
||||||
return objs.map(obj_data => {
|
return objs.map(obj_data => {
|
||||||
const type = pso_id_to_object_type(obj_data.type_id);
|
const type = pso_id_to_object_type(obj_data.type_id);
|
||||||
|
|
||||||
@ -270,7 +274,7 @@ function parse_obj_data(objs: DatObject[]): QuestObject[] {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function parse_npc_data(episode: number, npcs: DatNpc[]): QuestNpc[] {
|
function parse_npc_data(episode: number, npcs: readonly DatNpc[]): QuestNpc[] {
|
||||||
return npcs.map(npc_data => {
|
return npcs.map(npc_data => {
|
||||||
return {
|
return {
|
||||||
type: get_npc_type(episode, npc_data),
|
type: get_npc_type(episode, npc_data),
|
||||||
@ -564,7 +568,7 @@ function get_npc_type(episode: number, { type_id, scale, roaming, area_id }: Dat
|
|||||||
return NpcType.Unknown;
|
return NpcType.Unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
function objects_to_dat_data(objects: QuestObject[]): DatObject[] {
|
function objects_to_dat_data(objects: readonly QuestObject[]): DatObject[] {
|
||||||
return objects.map(object => ({
|
return objects.map(object => ({
|
||||||
type_id: object_data(object.type).pso_id!,
|
type_id: object_data(object.type).pso_id!,
|
||||||
id: object.id,
|
id: object.id,
|
||||||
@ -578,7 +582,8 @@ function objects_to_dat_data(objects: QuestObject[]): DatObject[] {
|
|||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
function npcs_to_dat_data(npcs: QuestNpc[]): DatNpc[] {
|
function npcs_to_dat_data(npcs: readonly QuestNpc[]): DatNpc[] {
|
||||||
|
// TODO: use primitive reinterpretation functions instead of DataView.
|
||||||
const dv = new DataView(new ArrayBuffer(4));
|
const dv = new DataView(new ArrayBuffer(4));
|
||||||
|
|
||||||
return npcs.map(npc => {
|
return npcs.map(npc => {
|
||||||
|
@ -18,4 +18,9 @@ export class AreaVariantModel {
|
|||||||
this.id = id;
|
this.id = id;
|
||||||
this.area = area;
|
this.area = area;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
set_sections(sections: readonly SectionModel[]): this {
|
||||||
|
this._sections.val = sections;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,6 +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 { QuestWaveModel } from "./QuestWaveModel";
|
||||||
|
|
||||||
const logger = Logger.get("quest_editor/model/QuestModel");
|
const logger = Logger.get("quest_editor/model/QuestModel");
|
||||||
|
|
||||||
@ -26,6 +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<QuestWaveModel>;
|
||||||
|
|
||||||
readonly id: Property<number> = this._id;
|
readonly id: Property<number> = this._id;
|
||||||
|
|
||||||
@ -58,6 +60,8 @@ export class QuestModel {
|
|||||||
|
|
||||||
readonly npcs: ListProperty<QuestNpcModel>;
|
readonly npcs: ListProperty<QuestNpcModel>;
|
||||||
|
|
||||||
|
readonly waves: ListProperty<QuestWaveModel>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
||||||
*/
|
*/
|
||||||
@ -75,16 +79,18 @@ export class QuestModel {
|
|||||||
long_description: string,
|
long_description: string,
|
||||||
episode: Episode,
|
episode: Episode,
|
||||||
map_designations: Map<number, number>,
|
map_designations: Map<number, number>,
|
||||||
objects: QuestObjectModel[],
|
objects: readonly QuestObjectModel[],
|
||||||
npcs: QuestNpcModel[],
|
npcs: readonly QuestNpcModel[],
|
||||||
dat_unknowns: DatUnknown[],
|
waves: readonly QuestWaveModel[],
|
||||||
object_code: Segment[],
|
dat_unknowns: readonly DatUnknown[],
|
||||||
shop_items: number[],
|
object_code: readonly Segment[],
|
||||||
|
shop_items: readonly number[],
|
||||||
) {
|
) {
|
||||||
check_episode(episode);
|
check_episode(episode);
|
||||||
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(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.");
|
||||||
@ -101,6 +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.waves = this._waves;
|
||||||
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;
|
||||||
|
@ -11,7 +11,7 @@ export class QuestNpcModel extends QuestEntityModel<NpcType> {
|
|||||||
/**
|
/**
|
||||||
* Data of which the purpose hasn't been discovered yet.
|
* Data of which the purpose hasn't been discovered yet.
|
||||||
*/
|
*/
|
||||||
readonly unknown: number[][];
|
readonly unknown: readonly number[][];
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
type: NpcType,
|
type: NpcType,
|
||||||
@ -24,7 +24,7 @@ export class QuestNpcModel extends QuestEntityModel<NpcType> {
|
|||||||
position: Vector3,
|
position: Vector3,
|
||||||
rotation: Euler,
|
rotation: Euler,
|
||||||
scale: Vector3,
|
scale: Vector3,
|
||||||
unknown: number[][],
|
unknown: readonly number[][],
|
||||||
) {
|
) {
|
||||||
if (!Number.isInteger(pso_type_id)) throw new Error("pso_type_id should be an integer.");
|
if (!Number.isInteger(pso_type_id)) throw new Error("pso_type_id should be an integer.");
|
||||||
if (!Number.isFinite(npc_id)) throw new Error("npc_id should be a number.");
|
if (!Number.isFinite(npc_id)) throw new Error("npc_id should be a number.");
|
||||||
|
@ -9,7 +9,7 @@ export class QuestObjectModel extends QuestEntityModel<ObjectType> {
|
|||||||
/**
|
/**
|
||||||
* Data of which the purpose hasn't been discovered yet.
|
* Data of which the purpose hasn't been discovered yet.
|
||||||
*/
|
*/
|
||||||
readonly unknown: number[][];
|
readonly unknown: readonly number[][];
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
type: ObjectType,
|
type: ObjectType,
|
||||||
@ -20,7 +20,7 @@ export class QuestObjectModel extends QuestEntityModel<ObjectType> {
|
|||||||
position: Vector3,
|
position: Vector3,
|
||||||
rotation: Euler,
|
rotation: Euler,
|
||||||
properties: Map<string, number>,
|
properties: Map<string, number>,
|
||||||
unknown: number[][],
|
unknown: readonly number[][],
|
||||||
) {
|
) {
|
||||||
super(type, area_id, section_id, position, rotation);
|
super(type, area_id, section_id, position, rotation);
|
||||||
|
|
||||||
|
43
src/quest_editor/model/QuestWaveActionModel.ts
Normal file
43
src/quest_editor/model/QuestWaveActionModel.ts
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
export abstract class QuestWaveActionModel {}
|
||||||
|
|
||||||
|
export class QuestWaveActionSpawnNpcsModel extends QuestWaveActionModel {
|
||||||
|
readonly section_id: number;
|
||||||
|
readonly appear_flag: number;
|
||||||
|
|
||||||
|
constructor(section_id: number, appear_flag: number) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.section_id = section_id;
|
||||||
|
this.appear_flag = appear_flag;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class QuestWaveActionUnlockModel extends QuestWaveActionModel {
|
||||||
|
readonly door_id: number;
|
||||||
|
|
||||||
|
constructor(door_id: number) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.door_id = door_id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class QuestWaveActionLockModel extends QuestWaveActionModel {
|
||||||
|
readonly door_id: number;
|
||||||
|
|
||||||
|
constructor(door_id: number) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.door_id = door_id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class QuestWaveActionSpawnWaveModel extends QuestWaveActionModel {
|
||||||
|
readonly wave_id: number;
|
||||||
|
|
||||||
|
constructor(wave_id: number) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.wave_id = wave_id;
|
||||||
|
}
|
||||||
|
}
|
37
src/quest_editor/model/QuestWaveModel.ts
Normal file
37
src/quest_editor/model/QuestWaveModel.ts
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
import { QuestWaveActionModel } from "./QuestWaveActionModel";
|
||||||
|
|
||||||
|
export class QuestWaveModel {
|
||||||
|
readonly id: number;
|
||||||
|
readonly section_id: number;
|
||||||
|
readonly wave: number;
|
||||||
|
readonly delay: number;
|
||||||
|
readonly actions: readonly QuestWaveActionModel[];
|
||||||
|
readonly area_id: number;
|
||||||
|
readonly unknown: number;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
id: number,
|
||||||
|
section_id: number,
|
||||||
|
wave: number,
|
||||||
|
delay: number,
|
||||||
|
actions: readonly QuestWaveActionModel[],
|
||||||
|
area_id: number,
|
||||||
|
unknown: number,
|
||||||
|
) {
|
||||||
|
if (!Number.isInteger(id)) throw new Error("id should be an integer.");
|
||||||
|
if (!Number.isInteger(section_id)) throw new Error("section_id should be an integer.");
|
||||||
|
if (!Number.isInteger(wave)) throw new Error("wave should be an integer.");
|
||||||
|
if (!Number.isInteger(delay)) throw new Error("delay should be an integer.");
|
||||||
|
if (!Array.isArray(actions)) throw new Error("actions should be an array.");
|
||||||
|
if (!Number.isInteger(area_id)) throw new Error("area_id should be an integer.");
|
||||||
|
if (!Number.isInteger(unknown)) throw new Error("unknown should be an integer.");
|
||||||
|
|
||||||
|
this.id = id;
|
||||||
|
this.section_id = section_id;
|
||||||
|
this.wave = wave;
|
||||||
|
this.delay = delay;
|
||||||
|
this.actions = actions;
|
||||||
|
this.area_id = area_id;
|
||||||
|
this.unknown = unknown;
|
||||||
|
}
|
||||||
|
}
|
@ -117,9 +117,7 @@ test("assembling disassembled object code with manual stack management should re
|
|||||||
expect(errors).toEqual([]);
|
expect(errors).toEqual([]);
|
||||||
expect(warnings).toEqual([]);
|
expect(warnings).toEqual([]);
|
||||||
|
|
||||||
bin.object_code.splice(0, bin.object_code.length, ...object_code);
|
const test_bytes = new ArrayBufferCursor(write_bin({ ...bin, object_code }), Endianness.Little);
|
||||||
|
|
||||||
const test_bytes = new ArrayBufferCursor(write_bin(bin), Endianness.Little);
|
|
||||||
|
|
||||||
orig_bytes.seek_start(0);
|
orig_bytes.seek_start(0);
|
||||||
expect(test_bytes.size).toBe(orig_bytes.size);
|
expect(test_bytes.size).toBe(orig_bytes.size);
|
||||||
|
@ -16,7 +16,7 @@ type ArgWithType = Arg & {
|
|||||||
* @param object_code - The object code to disassemble.
|
* @param object_code - The object code to disassemble.
|
||||||
* @param manual_stack - If true, will output stack management instructions (argpush variants). Otherwise the arguments of stack management instructions will be output as arguments to the instruction that pops them from the stack.
|
* @param manual_stack - If true, will output stack management instructions (argpush variants). Otherwise the arguments of stack management instructions will be output as arguments to the instruction that pops them from the stack.
|
||||||
*/
|
*/
|
||||||
export function disassemble(object_code: Segment[], manual_stack = false): string[] {
|
export function disassemble(object_code: readonly Segment[], manual_stack = false): string[] {
|
||||||
logger.trace("disassemble start");
|
logger.trace("disassemble start");
|
||||||
|
|
||||||
const lines: string[] = [];
|
const lines: string[] = [];
|
||||||
|
@ -160,6 +160,6 @@ function segments_equal(a: Segment, b: Segment): boolean {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function segment_arrays_equal(a: Segment[], b: Segment[]): boolean {
|
export function segment_arrays_equal(a: readonly Segment[], b: readonly Segment[]): boolean {
|
||||||
return arrays_equal(a, b, segments_equal);
|
return arrays_equal(a, b, segments_equal);
|
||||||
}
|
}
|
||||||
|
@ -28,7 +28,15 @@ import { RemoveEntityAction } from "../actions/RemoveEntityAction";
|
|||||||
import { Euler, Vector3 } from "three";
|
import { Euler, Vector3 } from "three";
|
||||||
import { vec3_to_threejs } from "../../core/rendering/conversion";
|
import { vec3_to_threejs } from "../../core/rendering/conversion";
|
||||||
import { RotateEntityAction } from "../actions/RotateEntityAction";
|
import { RotateEntityAction } from "../actions/RotateEntityAction";
|
||||||
import { VirtualMachine, ExecutionResult } from "../scripting/vm";
|
import { ExecutionResult, VirtualMachine } from "../scripting/vm";
|
||||||
|
import { QuestWaveModel } from "../model/QuestWaveModel";
|
||||||
|
import { DatWaveActionType } from "../../core/data_formats/parsing/quest/dat";
|
||||||
|
import {
|
||||||
|
QuestWaveActionLockModel,
|
||||||
|
QuestWaveActionSpawnNpcsModel,
|
||||||
|
QuestWaveActionSpawnWaveModel,
|
||||||
|
QuestWaveActionUnlockModel,
|
||||||
|
} from "../model/QuestWaveActionModel";
|
||||||
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");
|
||||||
@ -160,6 +168,36 @@ export class QuestEditorStore implements Disposable {
|
|||||||
npc.unknown,
|
npc.unknown,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
quest.waves.map(
|
||||||
|
wave =>
|
||||||
|
new QuestWaveModel(
|
||||||
|
wave.id,
|
||||||
|
wave.section_id,
|
||||||
|
wave.wave,
|
||||||
|
wave.delay,
|
||||||
|
wave.actions.map(action => {
|
||||||
|
switch (action.type) {
|
||||||
|
case DatWaveActionType.SpawnNpcs:
|
||||||
|
return new QuestWaveActionSpawnNpcsModel(
|
||||||
|
action.section_id,
|
||||||
|
action.appear_flag,
|
||||||
|
);
|
||||||
|
case DatWaveActionType.Unlock:
|
||||||
|
return new QuestWaveActionUnlockModel(
|
||||||
|
action.door_id,
|
||||||
|
);
|
||||||
|
case DatWaveActionType.Lock:
|
||||||
|
return new QuestWaveActionLockModel(action.door_id);
|
||||||
|
case DatWaveActionType.SpawnWave:
|
||||||
|
return new QuestWaveActionSpawnWaveModel(
|
||||||
|
action.wave_id,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
wave.area_id,
|
||||||
|
wave.unknown,
|
||||||
|
),
|
||||||
|
),
|
||||||
quest.dat_unknowns,
|
quest.dat_unknowns,
|
||||||
quest.object_code,
|
quest.object_code,
|
||||||
quest.shop_items,
|
quest.shop_items,
|
||||||
@ -217,6 +255,44 @@ export class QuestEditorStore implements Disposable {
|
|||||||
script_label: npc.script_label,
|
script_label: npc.script_label,
|
||||||
pso_roaming: npc.pso_roaming,
|
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 QuestWaveActionSpawnNpcsModel) {
|
||||||
|
return {
|
||||||
|
type: DatWaveActionType.SpawnNpcs,
|
||||||
|
section_id: action.section_id,
|
||||||
|
appear_flag: action.appear_flag,
|
||||||
|
};
|
||||||
|
} else if (action instanceof QuestWaveActionUnlockModel) {
|
||||||
|
return {
|
||||||
|
type: DatWaveActionType.Unlock,
|
||||||
|
door_id: action.door_id,
|
||||||
|
};
|
||||||
|
} else if (action instanceof QuestWaveActionLockModel) {
|
||||||
|
return {
|
||||||
|
type: DatWaveActionType.Lock,
|
||||||
|
door_id: action.door_id,
|
||||||
|
};
|
||||||
|
} else if (action instanceof QuestWaveActionSpawnWaveModel) {
|
||||||
|
return {
|
||||||
|
type: DatWaveActionType.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,
|
dat_unknowns: quest.dat_unknowns,
|
||||||
object_code: quest.object_code,
|
object_code: quest.object_code,
|
||||||
shop_items: quest.shop_items,
|
shop_items: quest.shop_items,
|
||||||
@ -316,7 +392,7 @@ export class QuestEditorStore implements Disposable {
|
|||||||
// Load section data.
|
// Load section data.
|
||||||
for (const variant of quest.area_variants.val) {
|
for (const variant of quest.area_variants.val) {
|
||||||
const sections = await area_store.get_area_sections(quest.episode, variant);
|
const sections = await area_store.get_area_sections(quest.episode, variant);
|
||||||
variant.sections.val.splice(0, Infinity, ...sections);
|
variant.set_sections(sections);
|
||||||
|
|
||||||
for (const object of quest.objects.val.filter(o => o.area_id === variant.area.id)) {
|
for (const object of quest.objects.val.filter(o => o.area_id === variant.area.id)) {
|
||||||
try {
|
try {
|
||||||
|
@ -17,6 +17,7 @@ import {
|
|||||||
import { QuestObjectModel } from "../model/QuestObjectModel";
|
import { QuestObjectModel } from "../model/QuestObjectModel";
|
||||||
import { QuestNpcModel } from "../model/QuestNpcModel";
|
import { QuestNpcModel } from "../model/QuestNpcModel";
|
||||||
import { Euler, Vector3 } from "three";
|
import { Euler, Vector3 } from "three";
|
||||||
|
import { QuestWaveModel } from "../model/QuestWaveModel";
|
||||||
|
|
||||||
export function create_new_quest(episode: Episode): QuestModel {
|
export function create_new_quest(episode: Episode): QuestModel {
|
||||||
if (episode === Episode.II) throw new Error("Episode II not yet supported.");
|
if (episode === Episode.II) throw new Error("Episode II not yet supported.");
|
||||||
@ -32,6 +33,7 @@ export function create_new_quest(episode: Episode): QuestModel {
|
|||||||
new Map().set(0, 0),
|
new Map().set(0, 0),
|
||||||
create_default_objects(),
|
create_default_objects(),
|
||||||
create_default_npcs(),
|
create_default_npcs(),
|
||||||
|
create_default_waves(),
|
||||||
[],
|
[],
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
@ -804,3 +806,7 @@ function create_default_npcs(): QuestNpcModel[] {
|
|||||||
),
|
),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function create_default_waves(): QuestWaveModel[] {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user