mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Area variants now change automatically when bb_map_designate instructions are added, deleted or changed. The 3D view also updates automatically.
This commit is contained in:
parent
2d551a1951
commit
ecbab0637d
@ -31,7 +31,8 @@ Features that are in ***bold italics*** are planned and not yet implemented.
|
|||||||
## Area Selection
|
## Area Selection
|
||||||
|
|
||||||
- Dropdown menu to switch area
|
- Dropdown menu to switch area
|
||||||
- Add new area
|
- Change area variant by editing assembly
|
||||||
|
- Update 3D view automatically
|
||||||
|
|
||||||
## Simple Quest Properties
|
## Simple Quest Properties
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@ export class BinFile {
|
|||||||
readonly short_description: string,
|
readonly short_description: string,
|
||||||
readonly long_description: string,
|
readonly long_description: string,
|
||||||
readonly object_code: Segment[],
|
readonly object_code: Segment[],
|
||||||
readonly shop_items: number[]
|
readonly shop_items: number[],
|
||||||
) {}
|
) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -41,7 +41,7 @@ SEGMENT_PRIORITY[SegmentType.Data] = 0;
|
|||||||
export function parse_bin(
|
export function parse_bin(
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
entry_labels: number[] = [0],
|
entry_labels: number[] = [0],
|
||||||
lenient: boolean = false
|
lenient: boolean = false,
|
||||||
): BinFile {
|
): BinFile {
|
||||||
const object_code_offset = cursor.u32(); // Always 4652
|
const object_code_offset = cursor.u32(); // Always 4652
|
||||||
const label_offset_table_offset = cursor.u32(); // Relative offsets
|
const label_offset_table_offset = cursor.u32(); // Relative offsets
|
||||||
@ -80,7 +80,7 @@ export function parse_bin(
|
|||||||
short_description,
|
short_description,
|
||||||
long_description,
|
long_description,
|
||||||
segments,
|
segments,
|
||||||
shop_items
|
shop_items,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -179,7 +179,7 @@ class LabelHolder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get_info(
|
get_info(
|
||||||
label: number
|
label: number,
|
||||||
): { offset: number; next?: { label: number; offset: number } } | undefined {
|
): { offset: number; next?: { label: number; offset: number } } | undefined {
|
||||||
const offset_and_index = this.label_map.get(label);
|
const offset_and_index = this.label_map.get(label);
|
||||||
|
|
||||||
@ -212,7 +212,7 @@ function parse_object_code(
|
|||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
label_holder: LabelHolder,
|
label_holder: LabelHolder,
|
||||||
entry_labels: number[],
|
entry_labels: number[],
|
||||||
lenient: boolean
|
lenient: boolean,
|
||||||
): Segment[] {
|
): Segment[] {
|
||||||
const offset_to_segment = new Map<number, Segment>();
|
const offset_to_segment = new Map<number, Segment>();
|
||||||
|
|
||||||
@ -221,7 +221,7 @@ function parse_object_code(
|
|||||||
label_holder,
|
label_holder,
|
||||||
entry_labels.reduce((m, l) => m.set(l, SegmentType.Instructions), new Map()),
|
entry_labels.reduce((m, l) => m.set(l, SegmentType.Instructions), new Map()),
|
||||||
offset_to_segment,
|
offset_to_segment,
|
||||||
lenient
|
lenient,
|
||||||
);
|
);
|
||||||
|
|
||||||
const segments: Segment[] = [];
|
const segments: Segment[] = [];
|
||||||
@ -259,7 +259,7 @@ function parse_object_code(
|
|||||||
// Should never happen.
|
// Should never happen.
|
||||||
if (end_offset <= offset) {
|
if (end_offset <= offset) {
|
||||||
logger.error(
|
logger.error(
|
||||||
`Next offset ${end_offset} was smaller than or equal to current offset ${offset}.`
|
`Next offset ${end_offset} was smaller than or equal to current offset ${offset}.`,
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -325,8 +325,8 @@ function find_and_parse_segments(
|
|||||||
label_holder: LabelHolder,
|
label_holder: LabelHolder,
|
||||||
labels: Map<number, SegmentType>,
|
labels: Map<number, SegmentType>,
|
||||||
offset_to_segment: Map<number, Segment>,
|
offset_to_segment: Map<number, Segment>,
|
||||||
lenient: boolean
|
lenient: boolean,
|
||||||
) {
|
): void {
|
||||||
let start_segment_count: number;
|
let start_segment_count: number;
|
||||||
|
|
||||||
// Iteratively parse segments from label references.
|
// Iteratively parse segments from label references.
|
||||||
@ -359,7 +359,7 @@ function find_and_parse_segments(
|
|||||||
labels,
|
labels,
|
||||||
instruction,
|
instruction,
|
||||||
i,
|
i,
|
||||||
SegmentType.Instructions
|
SegmentType.Instructions,
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
case Kind.ILabelVar:
|
case Kind.ILabelVar:
|
||||||
@ -377,6 +377,7 @@ function find_and_parse_segments(
|
|||||||
get_arg_label_values(cfg, labels, instruction, i, SegmentType.String);
|
get_arg_label_values(cfg, labels, instruction, i, SegmentType.String);
|
||||||
break;
|
break;
|
||||||
case Kind.RegTupRef:
|
case Kind.RegTupRef:
|
||||||
|
{
|
||||||
// Never on the stack.
|
// Never on the stack.
|
||||||
const arg = instruction.args[i];
|
const arg = instruction.args[i];
|
||||||
|
|
||||||
@ -387,7 +388,7 @@ function find_and_parse_segments(
|
|||||||
const label_values = register_value(
|
const label_values = register_value(
|
||||||
cfg,
|
cfg,
|
||||||
instruction,
|
instruction,
|
||||||
arg.value + j
|
arg.value + j,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (label_values.size() <= 10) {
|
if (label_values.size() <= 10) {
|
||||||
@ -397,7 +398,7 @@ function find_and_parse_segments(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -414,13 +415,13 @@ function get_arg_label_values(
|
|||||||
labels: Map<number, SegmentType>,
|
labels: Map<number, SegmentType>,
|
||||||
instruction: Instruction,
|
instruction: Instruction,
|
||||||
param_idx: number,
|
param_idx: number,
|
||||||
segment_type: SegmentType
|
segment_type: SegmentType,
|
||||||
): void {
|
): void {
|
||||||
if (instruction.opcode.stack === StackInteraction.Pop) {
|
if (instruction.opcode.stack === StackInteraction.Pop) {
|
||||||
const stack_values = stack_value(
|
const stack_values = stack_value(
|
||||||
cfg,
|
cfg,
|
||||||
instruction,
|
instruction,
|
||||||
instruction.opcode.params.length - param_idx - 1
|
instruction.opcode.params.length - param_idx - 1,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (stack_values.size() <= 10) {
|
if (stack_values.size() <= 10) {
|
||||||
@ -451,8 +452,8 @@ function parse_segment(
|
|||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
label: number,
|
label: number,
|
||||||
type: SegmentType,
|
type: SegmentType,
|
||||||
lenient: boolean
|
lenient: boolean,
|
||||||
) {
|
): void {
|
||||||
try {
|
try {
|
||||||
const info = label_holder.get_info(label);
|
const info = label_holder.get_info(label);
|
||||||
|
|
||||||
@ -492,7 +493,7 @@ function parse_segment(
|
|||||||
end_offset,
|
end_offset,
|
||||||
labels,
|
labels,
|
||||||
info.next && info.next.label,
|
info.next && info.next.label,
|
||||||
lenient
|
lenient,
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
case SegmentType.Data:
|
case SegmentType.Data:
|
||||||
@ -506,7 +507,7 @@ function parse_segment(
|
|||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (lenient) {
|
if (lenient) {
|
||||||
logger.error("Couldn't fully parse object code.", e);
|
logger.error("Couldn't fully parse object code segment.", e);
|
||||||
} else {
|
} else {
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
@ -520,8 +521,8 @@ function parse_instructions_segment(
|
|||||||
end_offset: number,
|
end_offset: number,
|
||||||
labels: number[],
|
labels: number[],
|
||||||
next_label: number | undefined,
|
next_label: number | undefined,
|
||||||
lenient: boolean
|
lenient: boolean,
|
||||||
) {
|
): void {
|
||||||
const instructions: Instruction[] = [];
|
const instructions: Instruction[] = [];
|
||||||
|
|
||||||
const segment: InstructionSegment = {
|
const segment: InstructionSegment = {
|
||||||
@ -556,7 +557,7 @@ function parse_instructions_segment(
|
|||||||
if (lenient) {
|
if (lenient) {
|
||||||
logger.error(
|
logger.error(
|
||||||
`Exception occurred while parsing arguments for instruction ${opcode.mnemonic}.`,
|
`Exception occurred while parsing arguments for instruction ${opcode.mnemonic}.`,
|
||||||
e
|
e,
|
||||||
);
|
);
|
||||||
instructions.push(new Instruction(opcode, []));
|
instructions.push(new Instruction(opcode, []));
|
||||||
} else {
|
} else {
|
||||||
@ -586,7 +587,7 @@ function parse_instructions_segment(
|
|||||||
cursor,
|
cursor,
|
||||||
next_label,
|
next_label,
|
||||||
SegmentType.Instructions,
|
SegmentType.Instructions,
|
||||||
lenient
|
lenient,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -596,8 +597,8 @@ function parse_data_segment(
|
|||||||
offset_to_segment: Map<number, Segment>,
|
offset_to_segment: Map<number, Segment>,
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
end_offset: number,
|
end_offset: number,
|
||||||
labels: number[]
|
labels: number[],
|
||||||
) {
|
): void {
|
||||||
const start_offset = cursor.position;
|
const start_offset = cursor.position;
|
||||||
const segment: DataSegment = {
|
const segment: DataSegment = {
|
||||||
type: SegmentType.Data,
|
type: SegmentType.Data,
|
||||||
@ -611,8 +612,8 @@ function parse_string_segment(
|
|||||||
offset_to_segment: Map<number, Segment>,
|
offset_to_segment: Map<number, Segment>,
|
||||||
cursor: Cursor,
|
cursor: Cursor,
|
||||||
end_offset: number,
|
end_offset: number,
|
||||||
labels: number[]
|
labels: number[],
|
||||||
) {
|
): void {
|
||||||
const start_offset = cursor.position;
|
const start_offset = cursor.position;
|
||||||
const segment: StringSegment = {
|
const segment: StringSegment = {
|
||||||
type: SegmentType.String,
|
type: SegmentType.String,
|
||||||
@ -653,7 +654,7 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
|||||||
value: cursor.string_utf16(
|
value: cursor.string_utf16(
|
||||||
Math.min(4096, cursor.bytes_left),
|
Math.min(4096, cursor.bytes_left),
|
||||||
true,
|
true,
|
||||||
false
|
false,
|
||||||
),
|
),
|
||||||
size: cursor.position - start_pos,
|
size: cursor.position - start_pos,
|
||||||
});
|
});
|
||||||
@ -686,7 +687,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: 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.
|
||||||
@ -762,7 +763,7 @@ function write_object_code(
|
|||||||
default:
|
default:
|
||||||
// TYPE_ANY, TYPE_VALUE and TYPE_POINTER cannot be serialized.
|
// TYPE_ANY, TYPE_VALUE and TYPE_POINTER cannot be serialized.
|
||||||
throw new Error(
|
throw new Error(
|
||||||
`Parameter type ${Kind[param.type.kind]} not implemented.`
|
`Parameter type ${Kind[param.type.kind]} not implemented.`,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,7 @@ import { Endianness } from "../../Endianness";
|
|||||||
import { walk_qst_files } from "../../../../test/src/utils";
|
import { walk_qst_files } from "../../../../test/src/utils";
|
||||||
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
||||||
import { BufferCursor } from "../../cursor/BufferCursor";
|
import { BufferCursor } from "../../cursor/BufferCursor";
|
||||||
import { parse_quest, Quest, write_quest_qst } from "./index";
|
import { parse_quest, write_quest_qst } from "./index";
|
||||||
import { ObjectType } from "./object_types";
|
import { ObjectType } from "./object_types";
|
||||||
|
|
||||||
test("parse Towards the Future", () => {
|
test("parse Towards the Future", () => {
|
||||||
@ -21,7 +21,8 @@ test("parse Towards the Future", () => {
|
|||||||
expect(quest.objects[0].type).toBe(ObjectType.MenuActivation);
|
expect(quest.objects[0].type).toBe(ObjectType.MenuActivation);
|
||||||
expect(quest.objects[4].type).toBe(ObjectType.PlayerSet);
|
expect(quest.objects[4].type).toBe(ObjectType.PlayerSet);
|
||||||
expect(quest.npcs.length).toBe(216);
|
expect(quest.npcs.length).toBe(216);
|
||||||
expect(testable_area_variants(quest)).toEqual([
|
expect(quest.map_designations).toEqual(
|
||||||
|
new Map([
|
||||||
[0, 0],
|
[0, 0],
|
||||||
[2, 0],
|
[2, 0],
|
||||||
[11, 0],
|
[11, 0],
|
||||||
@ -32,7 +33,8 @@ test("parse Towards the Future", () => {
|
|||||||
[8, 4],
|
[8, 4],
|
||||||
[10, 4],
|
[10, 4],
|
||||||
[14, 0],
|
[14, 0],
|
||||||
]);
|
]),
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -86,14 +88,7 @@ function round_trip_test(path: string, file_name: string, contents: Buffer): voi
|
|||||||
expect(test_npc.type).toBe(orig_npc.type);
|
expect(test_npc.type).toBe(orig_npc.type);
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(test_quest.area_variants.length).toBe(orig_quest.area_variants.length);
|
expect(test_quest.map_designations).toEqual(orig_quest.map_designations);
|
||||||
|
|
||||||
for (let i = 0; i < orig_quest.area_variants.length; i++) {
|
|
||||||
const orig_area_variant = orig_quest.area_variants[i];
|
|
||||||
const test_area_variant = test_quest.area_variants[i];
|
|
||||||
expect(test_area_variant.id).toBe(orig_area_variant.id);
|
|
||||||
expect(test_area_variant.area.id).toBe(orig_area_variant.area.id);
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(test_quest.object_code.length).toBe(orig_quest.object_code.length);
|
expect(test_quest.object_code.length).toBe(orig_quest.object_code.length);
|
||||||
|
|
||||||
@ -102,7 +97,3 @@ function round_trip_test(path: string, file_name: string, contents: Buffer): voi
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function testable_area_variants(quest: Quest): any[][] {
|
|
||||||
return quest.area_variants.map(av => [av.area.id, av.id]);
|
|
||||||
}
|
|
||||||
|
@ -13,7 +13,6 @@ import { Cursor } from "../../cursor/Cursor";
|
|||||||
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { Endianness } from "../../Endianness";
|
import { Endianness } from "../../Endianness";
|
||||||
import { Vec3 } from "../../vector";
|
import { Vec3 } from "../../vector";
|
||||||
import { AreaVariant, get_area_variant } from "./areas";
|
|
||||||
import { BinFile, parse_bin, write_bin } from "./bin";
|
import { BinFile, 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 } from "./entities";
|
||||||
@ -39,7 +38,7 @@ export type Quest = {
|
|||||||
readonly dat_unknowns: DatUnknown[];
|
readonly dat_unknowns: DatUnknown[];
|
||||||
readonly object_code: Segment[];
|
readonly object_code: Segment[];
|
||||||
readonly shop_items: number[];
|
readonly shop_items: number[];
|
||||||
readonly area_variants: AreaVariant[];
|
readonly map_designations: Map<number, number>;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -86,7 +85,7 @@ export function parse_quest(cursor: Cursor, lenient: boolean = false): Quest | u
|
|||||||
);
|
);
|
||||||
const bin = parse_bin(bin_decompressed, [0], lenient);
|
const bin = parse_bin(bin_decompressed, [0], lenient);
|
||||||
let episode = Episode.I;
|
let episode = Episode.I;
|
||||||
let area_variants: AreaVariant[] = [];
|
let map_designations: Map<number, number> = new Map();
|
||||||
|
|
||||||
if (bin.object_code.length) {
|
if (bin.object_code.length) {
|
||||||
let label_0_segment: InstructionSegment | undefined;
|
let label_0_segment: InstructionSegment | undefined;
|
||||||
@ -100,12 +99,7 @@ export function parse_quest(cursor: Cursor, lenient: boolean = false): Quest | u
|
|||||||
|
|
||||||
if (label_0_segment) {
|
if (label_0_segment) {
|
||||||
episode = get_episode(label_0_segment.instructions);
|
episode = get_episode(label_0_segment.instructions);
|
||||||
area_variants = extract_area_variants(
|
map_designations = extract_map_designations(dat, episode, label_0_segment.instructions);
|
||||||
dat,
|
|
||||||
episode,
|
|
||||||
label_0_segment.instructions,
|
|
||||||
lenient,
|
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
logger.warn(`No instruction for label 0 found.`);
|
logger.warn(`No instruction for label 0 found.`);
|
||||||
}
|
}
|
||||||
@ -125,7 +119,7 @@ export function parse_quest(cursor: Cursor, lenient: boolean = false): Quest | u
|
|||||||
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,
|
||||||
area_variants,
|
map_designations,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -192,49 +186,20 @@ function get_episode(func_0_instructions: Instruction[]): Episode {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function extract_area_variants(
|
function extract_map_designations(
|
||||||
dat: DatFile,
|
dat: DatFile,
|
||||||
episode: Episode,
|
episode: Episode,
|
||||||
func_0_instructions: Instruction[],
|
func_0_instructions: Instruction[],
|
||||||
lenient: boolean,
|
): Map<number, number> {
|
||||||
): AreaVariant[] {
|
const map_designations = new Map<number, number>();
|
||||||
// Add area variants that have npcs or objects even if there are no BB_Map_Designate instructions for them.
|
|
||||||
const area_variants = new Map<number, number>();
|
|
||||||
|
|
||||||
for (const npc of dat.npcs) {
|
for (const inst of func_0_instructions) {
|
||||||
area_variants.set(npc.area_id, 0);
|
if (inst.opcode === Opcode.BB_MAP_DESIGNATE) {
|
||||||
}
|
map_designations.set(inst.args[0].value, inst.args[2].value);
|
||||||
|
|
||||||
for (const obj of dat.objs) {
|
|
||||||
area_variants.set(obj.area_id, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
const bb_maps = func_0_instructions.filter(
|
|
||||||
instruction => instruction.opcode === Opcode.BB_MAP_DESIGNATE,
|
|
||||||
);
|
|
||||||
|
|
||||||
for (const bb_map of bb_maps) {
|
|
||||||
const area_id: number = bb_map.args[0].value;
|
|
||||||
const variant_id: number = bb_map.args[2].value;
|
|
||||||
area_variants.set(area_id, variant_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
const area_variants_array: AreaVariant[] = [];
|
|
||||||
|
|
||||||
for (const [area_id, variant_id] of area_variants.entries()) {
|
|
||||||
try {
|
|
||||||
area_variants_array.push(get_area_variant(episode, area_id, variant_id));
|
|
||||||
} catch (e) {
|
|
||||||
if (lenient) {
|
|
||||||
logger.error(`Unknown area variant.`, e);
|
|
||||||
} else {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sort by area order and then variant id.
|
return map_designations;
|
||||||
return area_variants_array.sort((a, b) => a.area.order - b.area.order || a.id - b.id);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function parse_obj_data(objs: DatObject[]): QuestObject[] {
|
function parse_obj_data(objs: DatObject[]): QuestObject[] {
|
||||||
|
23
src/domain/ObservableArea.ts
Normal file
23
src/domain/ObservableArea.ts
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
import { ObservableAreaVariant } from "./ObservableAreaVariant";
|
||||||
|
|
||||||
|
export class ObservableArea {
|
||||||
|
/**
|
||||||
|
* Matches the PSO ID.
|
||||||
|
*/
|
||||||
|
readonly id: number;
|
||||||
|
readonly name: string;
|
||||||
|
readonly order: number;
|
||||||
|
readonly area_variants: ObservableAreaVariant[];
|
||||||
|
|
||||||
|
constructor(id: number, name: string, order: number, area_variants: ObservableAreaVariant[]) {
|
||||||
|
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 (!area_variants) throw new Error("area_variants is required.");
|
||||||
|
|
||||||
|
this.id = id;
|
||||||
|
this.name = name;
|
||||||
|
this.order = order;
|
||||||
|
this.area_variants = area_variants;
|
||||||
|
}
|
||||||
|
}
|
17
src/domain/ObservableAreaVariant.ts
Normal file
17
src/domain/ObservableAreaVariant.ts
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import { ObservableArea } from "./ObservableArea";
|
||||||
|
import { IObservableArray, observable } from "mobx";
|
||||||
|
import { Section } from "./index";
|
||||||
|
|
||||||
|
export class ObservableAreaVariant {
|
||||||
|
readonly id: number;
|
||||||
|
readonly area: ObservableArea;
|
||||||
|
@observable.shallow readonly sections: IObservableArray<Section> = observable.array();
|
||||||
|
|
||||||
|
constructor(id: number, area: ObservableArea) {
|
||||||
|
if (!Number.isInteger(id) || id < 0)
|
||||||
|
throw new Error(`Expected id to be a non-negative integer, got ${id}.`);
|
||||||
|
|
||||||
|
this.id = id;
|
||||||
|
this.area = area;
|
||||||
|
}
|
||||||
|
}
|
170
src/domain/ObservableQuest.ts
Normal file
170
src/domain/ObservableQuest.ts
Normal file
@ -0,0 +1,170 @@
|
|||||||
|
import { action, computed, observable } from "mobx";
|
||||||
|
import { check_episode, Episode } from "../data_formats/parsing/quest/Episode";
|
||||||
|
import { ObservableAreaVariant } from "./ObservableAreaVariant";
|
||||||
|
import { area_store } from "../stores/AreaStore";
|
||||||
|
import { DatUnknown } from "../data_formats/parsing/quest/dat";
|
||||||
|
import { Segment } from "../scripting/instructions";
|
||||||
|
import { ObservableQuestNpc, ObservableQuestObject } from "./index";
|
||||||
|
import Logger from "js-logger";
|
||||||
|
|
||||||
|
const logger = Logger.get("domain/ObservableQuest");
|
||||||
|
|
||||||
|
export class ObservableQuest {
|
||||||
|
@observable private _id!: number;
|
||||||
|
|
||||||
|
get id(): number {
|
||||||
|
return this._id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
set_id(id: number): void {
|
||||||
|
if (!Number.isInteger(id) || id < 0 || id > 4294967295)
|
||||||
|
throw new Error("id must be an integer greater than 0 and less than 4294967295.");
|
||||||
|
this._id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@observable private _language!: number;
|
||||||
|
|
||||||
|
get language(): number {
|
||||||
|
return this._language;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
set_language(language: number): void {
|
||||||
|
if (!Number.isInteger(language)) throw new Error("language must be an integer.");
|
||||||
|
this._language = language;
|
||||||
|
}
|
||||||
|
|
||||||
|
@observable private _name!: string;
|
||||||
|
|
||||||
|
get name(): string {
|
||||||
|
return this._name;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
set_name(name: string): void {
|
||||||
|
if (name.length > 32) throw new Error("name can't be longer than 32 characters.");
|
||||||
|
this._name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
@observable private _short_description!: string;
|
||||||
|
|
||||||
|
get short_description(): string {
|
||||||
|
return this._short_description;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
set_short_description(short_description: string): void {
|
||||||
|
if (short_description.length > 128)
|
||||||
|
throw new Error("short_description can't be longer than 128 characters.");
|
||||||
|
this._short_description = short_description;
|
||||||
|
}
|
||||||
|
|
||||||
|
@observable private _long_description!: string;
|
||||||
|
|
||||||
|
get long_description(): string {
|
||||||
|
return this._long_description;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
set_long_description(long_description: string): void {
|
||||||
|
if (long_description.length > 288)
|
||||||
|
throw new Error("long_description can't be longer than 288 characters.");
|
||||||
|
this._long_description = long_description;
|
||||||
|
}
|
||||||
|
|
||||||
|
readonly episode: Episode;
|
||||||
|
|
||||||
|
@observable readonly objects: ObservableQuestObject[];
|
||||||
|
@observable readonly npcs: ObservableQuestNpc[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map of area IDs to entity counts.
|
||||||
|
*/
|
||||||
|
@computed get entities_per_area(): Map<number, number> {
|
||||||
|
const map = new Map<number, number>();
|
||||||
|
|
||||||
|
for (const npc of this.npcs) {
|
||||||
|
map.set(npc.area_id, (map.get(npc.area_id) || 0) + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const obj of this.objects) {
|
||||||
|
map.set(obj.area_id, (map.get(obj.area_id) || 0) + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map of area IDs to area variant IDs. One designation per area.
|
||||||
|
*/
|
||||||
|
@observable map_designations: Map<number, number>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* One variant per area.
|
||||||
|
*/
|
||||||
|
@computed get area_variants(): ObservableAreaVariant[] {
|
||||||
|
const variants: ObservableAreaVariant[] = [];
|
||||||
|
|
||||||
|
for (const area_id of this.entities_per_area.keys()) {
|
||||||
|
try {
|
||||||
|
variants.push(area_store.get_variant(this.episode, area_id, 0));
|
||||||
|
} catch (e) {
|
||||||
|
logger.warn(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const [area_id, variant_id] of this.map_designations) {
|
||||||
|
try {
|
||||||
|
variants.push(area_store.get_variant(this.episode, area_id, variant_id));
|
||||||
|
} catch (e) {
|
||||||
|
logger.warn(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return variants;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
||||||
|
*/
|
||||||
|
readonly dat_unknowns: DatUnknown[];
|
||||||
|
readonly object_code: Segment[];
|
||||||
|
readonly shop_items: number[];
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
id: number,
|
||||||
|
language: number,
|
||||||
|
name: string,
|
||||||
|
short_description: string,
|
||||||
|
long_description: string,
|
||||||
|
episode: Episode,
|
||||||
|
map_designations: Map<number, number>,
|
||||||
|
objects: ObservableQuestObject[],
|
||||||
|
npcs: ObservableQuestNpc[],
|
||||||
|
dat_unknowns: DatUnknown[],
|
||||||
|
object_code: Segment[],
|
||||||
|
shop_items: number[],
|
||||||
|
) {
|
||||||
|
check_episode(episode);
|
||||||
|
if (!map_designations) throw new Error("map_designations is required.");
|
||||||
|
if (!Array.isArray(objects)) throw new Error("objs is required.");
|
||||||
|
if (!Array.isArray(npcs)) throw new Error("npcs is required.");
|
||||||
|
if (!dat_unknowns) throw new Error("dat_unknowns is required.");
|
||||||
|
if (!object_code) throw new Error("object_code is required.");
|
||||||
|
if (!shop_items) throw new Error("shop_items is required.");
|
||||||
|
|
||||||
|
this.set_id(id);
|
||||||
|
this.set_language(language);
|
||||||
|
this.set_name(name);
|
||||||
|
this.set_short_description(short_description);
|
||||||
|
this.set_long_description(long_description);
|
||||||
|
this.episode = episode;
|
||||||
|
this.map_designations = map_designations;
|
||||||
|
this.objects = objects;
|
||||||
|
this.npcs = npcs;
|
||||||
|
this.dat_unknowns = dat_unknowns;
|
||||||
|
this.object_code = object_code;
|
||||||
|
this.shop_items = shop_items;
|
||||||
|
}
|
||||||
|
}
|
@ -1,10 +1,8 @@
|
|||||||
import { action, computed, IObservableArray, observable } from "mobx";
|
import { action, computed, observable } from "mobx";
|
||||||
import { DatUnknown } from "../data_formats/parsing/quest/dat";
|
|
||||||
import { EntityType } from "../data_formats/parsing/quest/entities";
|
import { EntityType } from "../data_formats/parsing/quest/entities";
|
||||||
import { check_episode, Episode } from "../data_formats/parsing/quest/Episode";
|
import { Episode } from "../data_formats/parsing/quest/Episode";
|
||||||
import { Vec3 } from "../data_formats/vector";
|
import { Vec3 } from "../data_formats/vector";
|
||||||
import { enum_values } from "../enums";
|
import { enum_values } from "../enums";
|
||||||
import { Segment } from "../scripting/instructions";
|
|
||||||
import { ItemType } from "./items";
|
import { ItemType } from "./items";
|
||||||
import { ObjectType } from "../data_formats/parsing/quest/object_types";
|
import { ObjectType } from "../data_formats/parsing/quest/object_types";
|
||||||
import { NpcType } from "../data_formats/parsing/quest/npc_types";
|
import { NpcType } from "../data_formats/parsing/quest/npc_types";
|
||||||
@ -65,122 +63,6 @@ export class Section {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ObservableQuest {
|
|
||||||
@observable private _id!: number;
|
|
||||||
|
|
||||||
get id(): number {
|
|
||||||
return this._id;
|
|
||||||
}
|
|
||||||
|
|
||||||
@action
|
|
||||||
set_id(id: number): void {
|
|
||||||
if (!Number.isInteger(id) || id < 0 || id > 4294967295)
|
|
||||||
throw new Error("id must be an integer greater than 0 and less than 4294967295.");
|
|
||||||
this._id = id;
|
|
||||||
}
|
|
||||||
|
|
||||||
@observable private _language!: number;
|
|
||||||
|
|
||||||
get language(): number {
|
|
||||||
return this._language;
|
|
||||||
}
|
|
||||||
|
|
||||||
@action
|
|
||||||
set_language(language: number): void {
|
|
||||||
if (!Number.isInteger(language)) throw new Error("language must be an integer.");
|
|
||||||
this._language = language;
|
|
||||||
}
|
|
||||||
|
|
||||||
@observable private _name!: string;
|
|
||||||
|
|
||||||
get name(): string {
|
|
||||||
return this._name;
|
|
||||||
}
|
|
||||||
|
|
||||||
@action
|
|
||||||
set_name(name: string): void {
|
|
||||||
if (name.length > 32) throw new Error("name can't be longer than 32 characters.");
|
|
||||||
this._name = name;
|
|
||||||
}
|
|
||||||
|
|
||||||
@observable private _short_description!: string;
|
|
||||||
|
|
||||||
get short_description(): string {
|
|
||||||
return this._short_description;
|
|
||||||
}
|
|
||||||
|
|
||||||
@action
|
|
||||||
set_short_description(short_description: string): void {
|
|
||||||
if (short_description.length > 128)
|
|
||||||
throw new Error("short_description can't be longer than 128 characters.");
|
|
||||||
this._short_description = short_description;
|
|
||||||
}
|
|
||||||
|
|
||||||
@observable _long_description!: string;
|
|
||||||
|
|
||||||
get long_description(): string {
|
|
||||||
return this._long_description;
|
|
||||||
}
|
|
||||||
|
|
||||||
@action
|
|
||||||
set_long_description(long_description: string): void {
|
|
||||||
if (long_description.length > 288)
|
|
||||||
throw new Error("long_description can't be longer than 288 characters.");
|
|
||||||
this._long_description = long_description;
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly episode: Episode;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* One variant per area.
|
|
||||||
*/
|
|
||||||
@observable readonly area_variants: ObservableAreaVariant[];
|
|
||||||
@observable readonly objects: ObservableQuestObject[];
|
|
||||||
@observable readonly npcs: ObservableQuestNpc[];
|
|
||||||
/**
|
|
||||||
* (Partial) raw DAT data that can't be parsed yet by Phantasmal.
|
|
||||||
*/
|
|
||||||
readonly dat_unknowns: DatUnknown[];
|
|
||||||
readonly object_code: Segment[];
|
|
||||||
readonly shop_items: number[];
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
id: number,
|
|
||||||
language: number,
|
|
||||||
name: string,
|
|
||||||
short_description: string,
|
|
||||||
long_description: string,
|
|
||||||
episode: Episode,
|
|
||||||
area_variants: ObservableAreaVariant[],
|
|
||||||
objects: ObservableQuestObject[],
|
|
||||||
npcs: ObservableQuestNpc[],
|
|
||||||
dat_unknowns: DatUnknown[],
|
|
||||||
object_code: Segment[],
|
|
||||||
shop_items: number[],
|
|
||||||
) {
|
|
||||||
check_episode(episode);
|
|
||||||
if (!area_variants) throw new Error("area_variants is required.");
|
|
||||||
if (!Array.isArray(objects)) throw new Error("objs is required.");
|
|
||||||
if (!Array.isArray(npcs)) throw new Error("npcs is required.");
|
|
||||||
if (!dat_unknowns) throw new Error("dat_unknowns is required.");
|
|
||||||
if (!object_code) throw new Error("object_code is required.");
|
|
||||||
if (!shop_items) throw new Error("shop_items is required.");
|
|
||||||
|
|
||||||
this.set_id(id);
|
|
||||||
this.set_language(language);
|
|
||||||
this.set_name(name);
|
|
||||||
this.set_short_description(short_description);
|
|
||||||
this.set_long_description(long_description);
|
|
||||||
this.episode = episode;
|
|
||||||
this.area_variants = area_variants;
|
|
||||||
this.objects = objects;
|
|
||||||
this.npcs = npcs;
|
|
||||||
this.dat_unknowns = dat_unknowns;
|
|
||||||
this.object_code = object_code;
|
|
||||||
this.shop_items = shop_items;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract class from which ObservableQuestNpc and ObservableQuestObject derive.
|
* Abstract class from which ObservableQuestNpc and ObservableQuestObject derive.
|
||||||
*/
|
*/
|
||||||
@ -198,7 +80,7 @@ export abstract class ObservableQuestEntity<Type extends EntityType = EntityType
|
|||||||
@observable.ref section?: Section;
|
@observable.ref section?: Section;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* World position
|
* Section-relative position
|
||||||
*/
|
*/
|
||||||
@observable.ref position: Vec3;
|
@observable.ref position: Vec3;
|
||||||
|
|
||||||
@ -207,10 +89,27 @@ export abstract class ObservableQuestEntity<Type extends EntityType = EntityType
|
|||||||
@observable.ref scale: Vec3;
|
@observable.ref scale: Vec3;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Section-relative position
|
* World position
|
||||||
*/
|
*/
|
||||||
@computed get section_position(): Vec3 {
|
@computed get world_position(): Vec3 {
|
||||||
let { x, y, z } = this.position;
|
if (this.section) {
|
||||||
|
let { x: rel_x, y: rel_y, z: rel_z } = this.position;
|
||||||
|
|
||||||
|
const sin = -this.section.sin_y_axis_rotation;
|
||||||
|
const cos = this.section.cos_y_axis_rotation;
|
||||||
|
const rot_x = cos * rel_x - sin * rel_z;
|
||||||
|
const rot_z = sin * rel_x + cos * rel_z;
|
||||||
|
const x = rot_x + this.section.position.x;
|
||||||
|
const y = rel_y + this.section.position.y;
|
||||||
|
const z = rot_z + this.section.position.z;
|
||||||
|
return new Vec3(x, y, z);
|
||||||
|
} else {
|
||||||
|
return this.position;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
set world_position(pos: Vec3) {
|
||||||
|
let { x, y, z } = pos;
|
||||||
|
|
||||||
if (this.section) {
|
if (this.section) {
|
||||||
const rel_x = x - this.section.position.x;
|
const rel_x = x - this.section.position.x;
|
||||||
@ -225,23 +124,8 @@ export abstract class ObservableQuestEntity<Type extends EntityType = EntityType
|
|||||||
z = rot_z;
|
z = rot_z;
|
||||||
}
|
}
|
||||||
|
|
||||||
return new Vec3(x, y, z);
|
|
||||||
}
|
|
||||||
|
|
||||||
set section_position(sec_pos: Vec3) {
|
|
||||||
let { x: rel_x, y: rel_y, z: rel_z } = sec_pos;
|
|
||||||
|
|
||||||
if (this.section) {
|
|
||||||
const sin = -this.section.sin_y_axis_rotation;
|
|
||||||
const cos = this.section.cos_y_axis_rotation;
|
|
||||||
const rot_x = cos * rel_x - sin * rel_z;
|
|
||||||
const rot_z = sin * rel_x + cos * rel_z;
|
|
||||||
const x = rot_x + this.section.position.x;
|
|
||||||
const y = rel_y + this.section.position.y;
|
|
||||||
const z = rot_z + this.section.position.z;
|
|
||||||
this.position = new Vec3(x, y, z);
|
this.position = new Vec3(x, y, z);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
protected constructor(
|
protected constructor(
|
||||||
type: Type,
|
type: Type,
|
||||||
@ -269,8 +153,8 @@ export abstract class ObservableQuestEntity<Type extends EntityType = EntityType
|
|||||||
}
|
}
|
||||||
|
|
||||||
@action
|
@action
|
||||||
set_position_and_section(position: Vec3, section?: Section): void {
|
set_world_position_and_section(world_position: Vec3, section?: Section): void {
|
||||||
this.position = position;
|
this.world_position = world_position;
|
||||||
this.section = section;
|
this.section = section;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -327,42 +211,6 @@ export class ObservableQuestNpc extends ObservableQuestEntity<NpcType> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ObservableArea {
|
|
||||||
/**
|
|
||||||
* Matches the PSO ID.
|
|
||||||
*/
|
|
||||||
readonly id: number;
|
|
||||||
readonly name: string;
|
|
||||||
readonly order: number;
|
|
||||||
readonly area_variants: ObservableAreaVariant[];
|
|
||||||
|
|
||||||
constructor(id: number, name: string, order: number, area_variants: ObservableAreaVariant[]) {
|
|
||||||
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 (!area_variants) throw new Error("area_variants is required.");
|
|
||||||
|
|
||||||
this.id = id;
|
|
||||||
this.name = name;
|
|
||||||
this.order = order;
|
|
||||||
this.area_variants = area_variants;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export class ObservableAreaVariant {
|
|
||||||
readonly id: number;
|
|
||||||
readonly area: ObservableArea;
|
|
||||||
@observable.shallow readonly sections: IObservableArray<Section> = observable.array();
|
|
||||||
|
|
||||||
constructor(id: number, area: ObservableArea) {
|
|
||||||
if (!Number.isInteger(id) || id < 0)
|
|
||||||
throw new Error(`Expected id to be a non-negative integer, got ${id}.`);
|
|
||||||
|
|
||||||
this.id = id;
|
|
||||||
this.area = area;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type ItemDrop = {
|
type ItemDrop = {
|
||||||
item_type: ItemType;
|
item_type: ItemType;
|
||||||
anything_rate: number;
|
anything_rate: number;
|
||||||
|
@ -108,13 +108,13 @@ export class QuestEntityControls {
|
|||||||
if (this.selected && this.pick) {
|
if (this.selected && this.pick) {
|
||||||
if (this.moved_since_last_mouse_down) {
|
if (this.moved_since_last_mouse_down) {
|
||||||
if (e.buttons === 1) {
|
if (e.buttons === 1) {
|
||||||
// User is tranforming selected entity.
|
// User is transforming selected entity.
|
||||||
// User is dragging selected entity.
|
// User is dragging selected entity.
|
||||||
if (e.shiftKey) {
|
if (e.shiftKey) {
|
||||||
// Vertical movement.
|
// Vertical movement.
|
||||||
this.translate_vertically(this.selected, this.pick, pointer_device_pos);
|
this.translate_vertically(this.selected, this.pick, pointer_device_pos);
|
||||||
} else {
|
} else {
|
||||||
// Horizontal movement accross terrain.
|
// Horizontal movement across terrain.
|
||||||
this.translate_horizontally(this.selected, this.pick, pointer_device_pos);
|
this.translate_horizontally(this.selected, this.pick, pointer_device_pos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -219,13 +219,13 @@ export class QuestEntityControls {
|
|||||||
|
|
||||||
if (ray.intersectPlane(plane, intersection_point)) {
|
if (ray.intersectPlane(plane, intersection_point)) {
|
||||||
const y = intersection_point.y + pick.grab_offset.y;
|
const y = intersection_point.y + pick.grab_offset.y;
|
||||||
const y_delta = y - selection.entity.position.y;
|
const y_delta = y - selection.entity.world_position.y;
|
||||||
pick.drag_y += y_delta;
|
pick.drag_y += y_delta;
|
||||||
pick.drag_adjust.y -= y_delta;
|
pick.drag_adjust.y -= y_delta;
|
||||||
selection.entity.position = new Vec3(
|
selection.entity.world_position = new Vec3(
|
||||||
selection.entity.position.x,
|
selection.entity.world_position.x,
|
||||||
y,
|
y,
|
||||||
selection.entity.position.z,
|
selection.entity.world_position.z,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -239,7 +239,7 @@ export class QuestEntityControls {
|
|||||||
const { intersection, section } = this.pick_terrain(pointer_position, pick);
|
const { intersection, section } = this.pick_terrain(pointer_position, pick);
|
||||||
|
|
||||||
if (intersection) {
|
if (intersection) {
|
||||||
selection.entity.set_position_and_section(
|
selection.entity.set_world_position_and_section(
|
||||||
new Vec3(
|
new Vec3(
|
||||||
intersection.point.x,
|
intersection.point.x,
|
||||||
intersection.point.y + pick.drag_y,
|
intersection.point.y + pick.drag_y,
|
||||||
@ -254,14 +254,14 @@ export class QuestEntityControls {
|
|||||||
// ray.origin.add(data.dragAdjust);
|
// ray.origin.add(data.dragAdjust);
|
||||||
const plane = new Plane(
|
const plane = new Plane(
|
||||||
new Vector3(0, 1, 0),
|
new Vector3(0, 1, 0),
|
||||||
-selection.entity.position.y + pick.grab_offset.y,
|
-selection.entity.world_position.y + pick.grab_offset.y,
|
||||||
);
|
);
|
||||||
const intersection_point = new Vector3();
|
const intersection_point = new Vector3();
|
||||||
|
|
||||||
if (ray.intersectPlane(plane, intersection_point)) {
|
if (ray.intersectPlane(plane, intersection_point)) {
|
||||||
selection.entity.position = new Vec3(
|
selection.entity.world_position = new Vec3(
|
||||||
intersection_point.x + pick.grab_offset.x,
|
intersection_point.x + pick.grab_offset.x,
|
||||||
selection.entity.position.y,
|
selection.entity.world_position.y,
|
||||||
intersection_point.z + pick.grab_offset.z,
|
intersection_point.z + pick.grab_offset.z,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -318,7 +318,7 @@ export class QuestEntityControls {
|
|||||||
return {
|
return {
|
||||||
mesh: intersection.object as Mesh,
|
mesh: intersection.object as Mesh,
|
||||||
entity,
|
entity,
|
||||||
initial_position: entity.position,
|
initial_position: entity.world_position,
|
||||||
grab_offset,
|
grab_offset,
|
||||||
drag_adjust,
|
drag_adjust,
|
||||||
drag_y,
|
drag_y,
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import Logger from "js-logger";
|
import Logger from "js-logger";
|
||||||
import { autorun, IReactionDisposer } from "mobx";
|
import { autorun, IReactionDisposer } from "mobx";
|
||||||
import { Mesh, Object3D, Vector3, Raycaster, Intersection } from "three";
|
import { Intersection, Mesh, Object3D, Raycaster, Vector3 } from "three";
|
||||||
import { load_area_collision_geometry, load_area_render_geometry } from "../loading/areas";
|
import { load_area_collision_geometry, load_area_render_geometry } from "../loading/areas";
|
||||||
import {
|
import {
|
||||||
load_npc_geometry,
|
load_npc_geometry,
|
||||||
@ -11,7 +11,10 @@ import {
|
|||||||
import { create_npc_mesh, create_object_mesh } from "./conversion/entities";
|
import { create_npc_mesh, create_object_mesh } from "./conversion/entities";
|
||||||
import { QuestRenderer } from "./QuestRenderer";
|
import { QuestRenderer } from "./QuestRenderer";
|
||||||
import { AreaUserData } from "./conversion/areas";
|
import { AreaUserData } from "./conversion/areas";
|
||||||
import { ObservableArea, ObservableQuest, ObservableQuestEntity } from "../domain";
|
import { ObservableQuestEntity } from "../domain";
|
||||||
|
import { ObservableQuest } from "../domain/ObservableQuest";
|
||||||
|
import { ObservableArea } from "../domain/ObservableArea";
|
||||||
|
import { ObservableAreaVariant } from "../domain/ObservableAreaVariant";
|
||||||
|
|
||||||
const logger = Logger.get("rendering/QuestModelManager");
|
const logger = Logger.get("rendering/QuestModelManager");
|
||||||
|
|
||||||
@ -22,17 +25,25 @@ const DUMMY_OBJECT = new Object3D();
|
|||||||
export class QuestModelManager {
|
export class QuestModelManager {
|
||||||
private quest?: ObservableQuest;
|
private quest?: ObservableQuest;
|
||||||
private area?: ObservableArea;
|
private area?: ObservableArea;
|
||||||
|
private area_variant?: ObservableAreaVariant;
|
||||||
private entity_reaction_disposers: IReactionDisposer[] = [];
|
private entity_reaction_disposers: IReactionDisposer[] = [];
|
||||||
|
|
||||||
constructor(private renderer: QuestRenderer) {}
|
constructor(private renderer: QuestRenderer) {}
|
||||||
|
|
||||||
async load_models(quest?: ObservableQuest, area?: ObservableArea): Promise<void> {
|
async load_models(quest?: ObservableQuest, area?: ObservableArea): Promise<void> {
|
||||||
if (this.quest === quest && this.area === area) {
|
let area_variant: ObservableAreaVariant | undefined;
|
||||||
|
|
||||||
|
if (quest && area) {
|
||||||
|
area_variant = quest.area_variants.find(v => v.area.id === area.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.quest === quest && this.area_variant === area_variant) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.quest = quest;
|
this.quest = quest;
|
||||||
this.area = area;
|
this.area = area;
|
||||||
|
this.area_variant = area_variant;
|
||||||
|
|
||||||
this.dispose_entity_reactions();
|
this.dispose_entity_reactions();
|
||||||
|
|
||||||
@ -41,8 +52,7 @@ export class QuestModelManager {
|
|||||||
// Load necessary area geometry.
|
// Load necessary area geometry.
|
||||||
const episode = quest.episode;
|
const episode = quest.episode;
|
||||||
const area_id = area.id;
|
const area_id = area.id;
|
||||||
const variant = quest.area_variants.find(v => v.area.id === area_id);
|
const variant_id = area_variant ? area_variant.id : 0;
|
||||||
const variant_id = (variant && variant.id) || 0;
|
|
||||||
|
|
||||||
const collision_geometry = await load_area_collision_geometry(
|
const collision_geometry = await load_area_collision_geometry(
|
||||||
episode,
|
episode,
|
||||||
@ -58,7 +68,7 @@ export class QuestModelManager {
|
|||||||
|
|
||||||
this.add_sections_to_collision_geometry(collision_geometry, render_geometry);
|
this.add_sections_to_collision_geometry(collision_geometry, render_geometry);
|
||||||
|
|
||||||
if (this.quest !== quest || this.area !== area) return;
|
if (this.quest !== quest || this.area_variant !== area_variant) return;
|
||||||
|
|
||||||
this.renderer.collision_geometry = collision_geometry;
|
this.renderer.collision_geometry = collision_geometry;
|
||||||
this.renderer.render_geometry = render_geometry;
|
this.renderer.render_geometry = render_geometry;
|
||||||
@ -73,7 +83,7 @@ export class QuestModelManager {
|
|||||||
const npc_geom = await load_npc_geometry(npc.type);
|
const npc_geom = await load_npc_geometry(npc.type);
|
||||||
const npc_tex = await load_npc_textures(npc.type);
|
const npc_tex = await load_npc_textures(npc.type);
|
||||||
|
|
||||||
if (this.quest !== quest || this.area !== area) return;
|
if (this.quest !== quest || this.area_variant !== area_variant) return;
|
||||||
|
|
||||||
const model = create_npc_mesh(npc, npc_geom, npc_tex);
|
const model = create_npc_mesh(npc, npc_geom, npc_tex);
|
||||||
this.update_entity_geometry(npc, model);
|
this.update_entity_geometry(npc, model);
|
||||||
@ -85,7 +95,7 @@ export class QuestModelManager {
|
|||||||
const object_geom = await load_object_geometry(object.type);
|
const object_geom = await load_object_geometry(object.type);
|
||||||
const object_tex = await load_object_textures(object.type);
|
const object_tex = await load_object_textures(object.type);
|
||||||
|
|
||||||
if (this.quest !== quest || this.area !== area) return;
|
if (this.quest !== quest || this.area_variant !== area_variant) return;
|
||||||
|
|
||||||
const model = create_object_mesh(object, object_geom, object_tex);
|
const model = create_object_mesh(object, object_geom, object_tex);
|
||||||
this.update_entity_geometry(object, model);
|
this.update_entity_geometry(object, model);
|
||||||
@ -150,7 +160,7 @@ export class QuestModelManager {
|
|||||||
|
|
||||||
this.entity_reaction_disposers.push(
|
this.entity_reaction_disposers.push(
|
||||||
autorun(() => {
|
autorun(() => {
|
||||||
const { x, y, z } = entity.position;
|
const { x, y, z } = entity.world_position;
|
||||||
model.position.set(x, y, z);
|
model.position.set(x, y, z);
|
||||||
const rot = entity.rotation;
|
const rot = entity.rotation;
|
||||||
model.rotation.set(rot.x, rot.y, rot.z);
|
model.rotation.set(rot.x, rot.y, rot.z);
|
||||||
|
@ -94,8 +94,8 @@ export function area_collision_geometry_to_object_3d(object: CollisionObject): O
|
|||||||
indices[2],
|
indices[2],
|
||||||
new Vector3(normal.x, normal.y, normal.z),
|
new Vector3(normal.x, normal.y, normal.z),
|
||||||
undefined,
|
undefined,
|
||||||
color_index
|
color_index,
|
||||||
)
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -115,7 +115,7 @@ export function area_collision_geometry_to_object_3d(object: CollisionObject): O
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function area_geometry_to_sections_and_object_3d(
|
export function area_geometry_to_sections_and_object_3d(
|
||||||
object: RenderObject
|
object: RenderObject,
|
||||||
): [Section[], Object3D] {
|
): [Section[], Object3D] {
|
||||||
const sections: Section[] = [];
|
const sections: Section[] = [];
|
||||||
const group = new Group();
|
const group = new Group();
|
||||||
@ -135,7 +135,7 @@ export function area_geometry_to_sections_and_object_3d(
|
|||||||
transparent: true,
|
transparent: true,
|
||||||
opacity: 0.25,
|
opacity: 0.25,
|
||||||
side: DoubleSide,
|
side: DoubleSide,
|
||||||
})
|
}),
|
||||||
);
|
);
|
||||||
group.add(mesh);
|
group.add(mesh);
|
||||||
|
|
||||||
|
@ -76,7 +76,7 @@ function create(
|
|||||||
mesh.name = name;
|
mesh.name = name;
|
||||||
(mesh.userData as EntityUserData).entity = entity;
|
(mesh.userData as EntityUserData).entity = entity;
|
||||||
|
|
||||||
const { x, y, z } = entity.position;
|
const { x, y, z } = entity.world_position;
|
||||||
mesh.position.set(x, y, z);
|
mesh.position.set(x, y, z);
|
||||||
const rot = entity.rotation;
|
const rot = entity.rotation;
|
||||||
mesh.rotation.set(rot.x, rot.y, rot.z);
|
mesh.rotation.set(rot.x, rot.y, rot.z);
|
||||||
|
@ -1,24 +1,29 @@
|
|||||||
import { observable } from "mobx";
|
import { observable } from "mobx";
|
||||||
import { editor } from "monaco-editor";
|
import { editor } from "monaco-editor";
|
||||||
import AssemblyWorker from "worker-loader!./assembly_worker";
|
import AssemblyWorker from "worker-loader!./assembly_worker";
|
||||||
import { AssemblyChangeInput, NewAssemblyInput, ScriptWorkerOutput } from "./assembler_messages";
|
import {
|
||||||
import { AssemblyError } from "./assembly";
|
AssemblyChangeInput,
|
||||||
|
AssemblyWorkerOutput,
|
||||||
|
NewAssemblyInput,
|
||||||
|
} from "./assembly_worker_messages";
|
||||||
|
import { AssemblyError, AssemblyWarning } from "./assembly";
|
||||||
import { disassemble } from "./disassembly";
|
import { disassemble } from "./disassembly";
|
||||||
import { Segment } from "./instructions";
|
import { ObservableQuest } from "../domain/ObservableQuest";
|
||||||
|
|
||||||
export class AssemblyAnalyser {
|
export class AssemblyAnalyser {
|
||||||
|
@observable warnings: AssemblyWarning[] = [];
|
||||||
@observable errors: AssemblyError[] = [];
|
@observable errors: AssemblyError[] = [];
|
||||||
|
|
||||||
private worker = new AssemblyWorker();
|
private worker = new AssemblyWorker();
|
||||||
private object_code: Segment[] = [];
|
private quest?: ObservableQuest;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.worker.onmessage = this.process_worker_message;
|
this.worker.onmessage = this.process_worker_message;
|
||||||
}
|
}
|
||||||
|
|
||||||
disassemble(object_code: Segment[]): string[] {
|
disassemble(quest: ObservableQuest): string[] {
|
||||||
this.object_code = object_code;
|
this.quest = quest;
|
||||||
const assembly = disassemble(object_code);
|
const assembly = disassemble(quest.object_code);
|
||||||
const message: NewAssemblyInput = { type: "new_assembly_input", assembly };
|
const message: NewAssemblyInput = { type: "new_assembly_input", assembly };
|
||||||
this.worker.postMessage(message);
|
this.worker.postMessage(message);
|
||||||
return assembly;
|
return assembly;
|
||||||
@ -34,10 +39,12 @@ export class AssemblyAnalyser {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private process_worker_message = (e: MessageEvent): void => {
|
private process_worker_message = (e: MessageEvent): void => {
|
||||||
const message: ScriptWorkerOutput = e.data;
|
const message: AssemblyWorkerOutput = e.data;
|
||||||
|
|
||||||
if (message.type === "new_object_code_output") {
|
if (message.type === "new_object_code_output" && this.quest) {
|
||||||
this.object_code.splice(0, this.object_code.length, ...message.object_code);
|
this.quest.object_code.splice(0, this.quest.object_code.length, ...message.object_code);
|
||||||
|
this.quest.map_designations = message.map_designations;
|
||||||
|
this.warnings = message.warnings;
|
||||||
this.errors = message.errors;
|
this.errors = message.errors;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -12,7 +12,7 @@ export enum TokenType {
|
|||||||
UnterminatedString,
|
UnterminatedString,
|
||||||
Ident,
|
Ident,
|
||||||
InvalidIdent,
|
InvalidIdent,
|
||||||
ArgSeperator,
|
ArgSeparator,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Token =
|
export type Token =
|
||||||
@ -29,7 +29,7 @@ export type Token =
|
|||||||
| UnterminatedStringToken
|
| UnterminatedStringToken
|
||||||
| IdentToken
|
| IdentToken
|
||||||
| InvalidIdentToken
|
| InvalidIdentToken
|
||||||
| ArgSeperatorToken;
|
| ArgSeparatorToken;
|
||||||
|
|
||||||
export type IntToken = {
|
export type IntToken = {
|
||||||
type: TokenType.Int;
|
type: TokenType.Int;
|
||||||
@ -116,8 +116,8 @@ export type InvalidIdentToken = {
|
|||||||
len: number;
|
len: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type ArgSeperatorToken = {
|
export type ArgSeparatorToken = {
|
||||||
type: TokenType.ArgSeperator;
|
type: TokenType.ArgSeparator;
|
||||||
col: number;
|
col: number;
|
||||||
len: number;
|
len: number;
|
||||||
};
|
};
|
||||||
@ -159,7 +159,7 @@ export class AssemblyLexer {
|
|||||||
} else if (/[-\d]/.test(char)) {
|
} else if (/[-\d]/.test(char)) {
|
||||||
token = this.tokenize_number_or_label();
|
token = this.tokenize_number_or_label();
|
||||||
} else if ("," === char) {
|
} else if ("," === char) {
|
||||||
token = { type: TokenType.ArgSeperator, col: this.col, len: 1 };
|
token = { type: TokenType.ArgSeparator, col: this.col, len: 1 };
|
||||||
this.skip();
|
this.skip();
|
||||||
} else if ("." === char) {
|
} else if ("." === char) {
|
||||||
token = this.tokenize_section();
|
token = this.tokenize_section();
|
||||||
|
@ -396,7 +396,7 @@ class Assembler {
|
|||||||
let arg_count = 0;
|
let arg_count = 0;
|
||||||
|
|
||||||
for (const token of this.tokens) {
|
for (const token of this.tokens) {
|
||||||
if (token.type !== TokenType.ArgSeperator) {
|
if (token.type !== TokenType.ArgSeparator) {
|
||||||
arg_count++;
|
arg_count++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -510,7 +510,7 @@ class Assembler {
|
|||||||
const token = this.tokens[i];
|
const token = this.tokens[i];
|
||||||
const param = params[param_i];
|
const param = params[param_i];
|
||||||
|
|
||||||
if (token.type === TokenType.ArgSeperator) {
|
if (token.type === TokenType.ArgSeparator) {
|
||||||
if (should_be_arg) {
|
if (should_be_arg) {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
col: token.col,
|
col: token.col,
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
import { NewObjectCodeOutput, ScriptWorkerInput } from "./assembler_messages";
|
import { AssemblyWorkerInput, NewObjectCodeOutput } from "./assembly_worker_messages";
|
||||||
import { assemble } from "./assembly";
|
import { assemble } from "./assembly";
|
||||||
import Logger from "js-logger";
|
import Logger from "js-logger";
|
||||||
|
import { SegmentType } from "./instructions";
|
||||||
|
import { Opcode } from "./opcodes";
|
||||||
|
|
||||||
Logger.useDefaults({
|
Logger.useDefaults({
|
||||||
defaultLevel: (Logger as any)[process.env["LOG_LEVEL"] || "OFF"],
|
defaultLevel: (Logger as any)[process.env["LOG_LEVEL"] || "OFF"],
|
||||||
@ -9,7 +11,7 @@ Logger.useDefaults({
|
|||||||
const ctx: Worker = self as any;
|
const ctx: Worker = self as any;
|
||||||
|
|
||||||
let lines: string[] = [];
|
let lines: string[] = [];
|
||||||
const messages: ScriptWorkerInput[] = [];
|
const messages: AssemblyWorkerInput[] = [];
|
||||||
let timeout: any;
|
let timeout: any;
|
||||||
|
|
||||||
ctx.onmessage = (e: MessageEvent) => {
|
ctx.onmessage = (e: MessageEvent) => {
|
||||||
@ -45,7 +47,7 @@ function process_messages(): void {
|
|||||||
endLineNumber,
|
endLineNumber,
|
||||||
startColumn,
|
startColumn,
|
||||||
endColumn,
|
endColumn,
|
||||||
new_lines[0]
|
new_lines[0],
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
// Keep the left part of the first changed line.
|
// Keep the left part of the first changed line.
|
||||||
@ -55,7 +57,7 @@ function process_messages(): void {
|
|||||||
replace_line_part_left(
|
replace_line_part_left(
|
||||||
endLineNumber,
|
endLineNumber,
|
||||||
endColumn,
|
endColumn,
|
||||||
new_lines[new_lines.length - 1]
|
new_lines[new_lines.length - 1],
|
||||||
);
|
);
|
||||||
|
|
||||||
// Replace all the lines in between.
|
// Replace all the lines in between.
|
||||||
@ -63,16 +65,34 @@ function process_messages(): void {
|
|||||||
replace_lines(
|
replace_lines(
|
||||||
startLineNumber + 1,
|
startLineNumber + 1,
|
||||||
endLineNumber - 1,
|
endLineNumber - 1,
|
||||||
new_lines.slice(1, new_lines.length - 1)
|
new_lines.slice(1, new_lines.length - 1),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const assembler_result = assemble(lines);
|
||||||
|
const map_designations = new Map<number, number>();
|
||||||
|
|
||||||
|
for (const segment of assembler_result.object_code) {
|
||||||
|
if (segment.labels.includes(0)) {
|
||||||
|
if (segment.type === SegmentType.Instructions) {
|
||||||
|
for (const inst of segment.instructions) {
|
||||||
|
if (inst.opcode === Opcode.BB_MAP_DESIGNATE) {
|
||||||
|
map_designations.set(inst.args[0].value, inst.args[2].value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const response: NewObjectCodeOutput = {
|
const response: NewObjectCodeOutput = {
|
||||||
type: "new_object_code_output",
|
type: "new_object_code_output",
|
||||||
...assemble(lines),
|
map_designations,
|
||||||
|
...assembler_result,
|
||||||
};
|
};
|
||||||
ctx.postMessage(response);
|
ctx.postMessage(response);
|
||||||
}
|
}
|
||||||
@ -81,7 +101,7 @@ function replace_line_part(
|
|||||||
line_no: number,
|
line_no: number,
|
||||||
start_col: number,
|
start_col: number,
|
||||||
end_col: number,
|
end_col: number,
|
||||||
new_line_parts: string[]
|
new_line_parts: string[],
|
||||||
): void {
|
): void {
|
||||||
const line = lines[line_no - 1];
|
const line = lines[line_no - 1];
|
||||||
// We keep the parts of the line that weren't affected by the edit.
|
// We keep the parts of the line that weren't affected by the edit.
|
||||||
@ -96,7 +116,7 @@ function replace_line_part(
|
|||||||
1,
|
1,
|
||||||
line_start + new_line_parts[0],
|
line_start + new_line_parts[0],
|
||||||
...new_line_parts.slice(1, new_line_parts.length - 1),
|
...new_line_parts.slice(1, new_line_parts.length - 1),
|
||||||
new_line_parts[new_line_parts.length - 1] + line_end
|
new_line_parts[new_line_parts.length - 1] + line_end,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -118,7 +138,7 @@ function replace_lines_and_merge_line_parts(
|
|||||||
end_line_no: number,
|
end_line_no: number,
|
||||||
start_col: number,
|
start_col: number,
|
||||||
end_col: number,
|
end_col: number,
|
||||||
new_line_part: string
|
new_line_part: string,
|
||||||
): void {
|
): void {
|
||||||
const start_line = lines[start_line_no - 1];
|
const start_line = lines[start_line_no - 1];
|
||||||
const end_line = lines[end_line_no - 1];
|
const end_line = lines[end_line_no - 1];
|
||||||
@ -129,6 +149,6 @@ function replace_lines_and_merge_line_parts(
|
|||||||
lines.splice(
|
lines.splice(
|
||||||
start_line_no - 1,
|
start_line_no - 1,
|
||||||
end_line_no - start_line_no + 1,
|
end_line_no - start_line_no + 1,
|
||||||
start_line_start + new_line_part + end_line_end
|
start_line_start + new_line_part + end_line_end,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import { editor } from "monaco-editor";
|
import { editor } from "monaco-editor";
|
||||||
import { AssemblyError } from "./assembly";
|
import { AssemblyError, AssemblyWarning } from "./assembly";
|
||||||
import { Segment } from "./instructions";
|
import { Segment } from "./instructions";
|
||||||
|
|
||||||
export type ScriptWorkerInput = NewAssemblyInput | AssemblyChangeInput;
|
export type AssemblyWorkerInput = NewAssemblyInput | AssemblyChangeInput;
|
||||||
|
|
||||||
export type NewAssemblyInput = {
|
export type NewAssemblyInput = {
|
||||||
readonly type: "new_assembly_input";
|
readonly type: "new_assembly_input";
|
||||||
@ -14,10 +14,12 @@ export type AssemblyChangeInput = {
|
|||||||
readonly changes: editor.IModelContentChange[];
|
readonly changes: editor.IModelContentChange[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type ScriptWorkerOutput = NewObjectCodeOutput;
|
export type AssemblyWorkerOutput = NewObjectCodeOutput;
|
||||||
|
|
||||||
export type NewObjectCodeOutput = {
|
export type NewObjectCodeOutput = {
|
||||||
readonly type: "new_object_code_output";
|
readonly type: "new_object_code_output";
|
||||||
readonly object_code: Segment[];
|
readonly object_code: Segment[];
|
||||||
|
readonly map_designations: Map<number, number>;
|
||||||
|
readonly warnings: AssemblyWarning[];
|
||||||
readonly errors: AssemblyError[];
|
readonly errors: AssemblyError[];
|
||||||
};
|
};
|
@ -1,7 +1,9 @@
|
|||||||
import { ObservableArea, ObservableAreaVariant, Section } from "../domain";
|
import { Section } from "../domain";
|
||||||
import { load_area_sections } from "../loading/areas";
|
import { load_area_sections } from "../loading/areas";
|
||||||
import { Episode, EPISODES } from "../data_formats/parsing/quest/Episode";
|
import { Episode, EPISODES } from "../data_formats/parsing/quest/Episode";
|
||||||
import { get_areas_for_episode } from "../data_formats/parsing/quest/areas";
|
import { get_areas_for_episode } from "../data_formats/parsing/quest/areas";
|
||||||
|
import { ObservableAreaVariant } from "../domain/ObservableAreaVariant";
|
||||||
|
import { ObservableArea } from "../domain/ObservableArea";
|
||||||
|
|
||||||
class AreaStore {
|
class AreaStore {
|
||||||
private readonly areas: ObservableArea[][] = [];
|
private readonly areas: ObservableArea[][] = [];
|
||||||
|
@ -5,20 +5,20 @@ import { ArrayBufferCursor } from "../data_formats/cursor/ArrayBufferCursor";
|
|||||||
import { parse_quest, write_quest_qst } from "../data_formats/parsing/quest";
|
import { parse_quest, write_quest_qst } from "../data_formats/parsing/quest";
|
||||||
import { Vec3 } from "../data_formats/vector";
|
import { Vec3 } from "../data_formats/vector";
|
||||||
import {
|
import {
|
||||||
ObservableArea,
|
|
||||||
ObservableQuest,
|
|
||||||
ObservableQuestEntity,
|
ObservableQuestEntity,
|
||||||
Section,
|
|
||||||
ObservableQuestObject,
|
|
||||||
ObservableQuestNpc,
|
ObservableQuestNpc,
|
||||||
|
ObservableQuestObject,
|
||||||
|
Section,
|
||||||
} from "../domain";
|
} from "../domain";
|
||||||
import { read_file } from "../read_file";
|
import { read_file } from "../read_file";
|
||||||
import { SimpleUndo, UndoStack, undo_manager } from "../undo";
|
import { SimpleUndo, undo_manager, UndoStack } from "../undo";
|
||||||
import { application_store } from "./ApplicationStore";
|
import { application_store } from "./ApplicationStore";
|
||||||
import { area_store } from "./AreaStore";
|
import { area_store } from "./AreaStore";
|
||||||
import { create_new_quest } from "./quest_creation";
|
import { create_new_quest } from "./quest_creation";
|
||||||
import { Episode } from "../data_formats/parsing/quest/Episode";
|
import { Episode } from "../data_formats/parsing/quest/Episode";
|
||||||
import { entity_data } from "../data_formats/parsing/quest/entities";
|
import { entity_data } from "../data_formats/parsing/quest/entities";
|
||||||
|
import { ObservableQuest } from "../domain/ObservableQuest";
|
||||||
|
import { ObservableArea } from "../domain/ObservableArea";
|
||||||
|
|
||||||
const logger = Logger.get("stores/QuestEditorStore");
|
const logger = Logger.get("stores/QuestEditorStore");
|
||||||
|
|
||||||
@ -71,7 +71,7 @@ class QuestEditorStore {
|
|||||||
set_current_area_id = (area_id?: number) => {
|
set_current_area_id = (area_id?: number) => {
|
||||||
this.selected_entity = undefined;
|
this.selected_entity = undefined;
|
||||||
|
|
||||||
if (area_id == null) {
|
if (area_id == undefined) {
|
||||||
this.current_area = undefined;
|
this.current_area = undefined;
|
||||||
} else if (this.current_quest) {
|
} else if (this.current_quest) {
|
||||||
this.current_area = area_store.get_area(this.current_quest.episode, area_id);
|
this.current_area = area_store.get_area(this.current_quest.episode, area_id);
|
||||||
@ -97,9 +97,7 @@ class QuestEditorStore {
|
|||||||
quest.short_description,
|
quest.short_description,
|
||||||
quest.long_description,
|
quest.long_description,
|
||||||
quest.episode,
|
quest.episode,
|
||||||
quest.area_variants.map(variant =>
|
quest.map_designations,
|
||||||
area_store.get_variant(quest.episode, variant.area.id, variant.id),
|
|
||||||
),
|
|
||||||
quest.objects.map(
|
quest.objects.map(
|
||||||
obj =>
|
obj =>
|
||||||
new ObservableQuestObject(
|
new ObservableQuestObject(
|
||||||
@ -174,7 +172,7 @@ class QuestEditorStore {
|
|||||||
type: obj.type,
|
type: obj.type,
|
||||||
area_id: obj.area_id,
|
area_id: obj.area_id,
|
||||||
section_id: obj.section_id,
|
section_id: obj.section_id,
|
||||||
position: obj.section_position,
|
position: obj.position,
|
||||||
rotation: obj.rotation,
|
rotation: obj.rotation,
|
||||||
scale: obj.scale,
|
scale: obj.scale,
|
||||||
unknown: obj.unknown,
|
unknown: obj.unknown,
|
||||||
@ -183,7 +181,7 @@ class QuestEditorStore {
|
|||||||
type: npc.type,
|
type: npc.type,
|
||||||
area_id: npc.area_id,
|
area_id: npc.area_id,
|
||||||
section_id: npc.section_id,
|
section_id: npc.section_id,
|
||||||
position: npc.section_position,
|
position: npc.position,
|
||||||
rotation: npc.rotation,
|
rotation: npc.rotation,
|
||||||
scale: npc.scale,
|
scale: npc.scale,
|
||||||
unknown: npc.unknown,
|
unknown: npc.unknown,
|
||||||
@ -193,7 +191,7 @@ class QuestEditorStore {
|
|||||||
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,
|
||||||
area_variants: quest.area_variants,
|
map_designations: quest.map_designations,
|
||||||
},
|
},
|
||||||
file_name,
|
file_name,
|
||||||
);
|
);
|
||||||
@ -223,11 +221,11 @@ class QuestEditorStore {
|
|||||||
this.undo.push_action(
|
this.undo.push_action(
|
||||||
`Move ${entity_data(entity.type).name}`,
|
`Move ${entity_data(entity.type).name}`,
|
||||||
() => {
|
() => {
|
||||||
entity.position = initial_position;
|
entity.world_position = initial_position;
|
||||||
quest_editor_store.set_selected_entity(entity);
|
quest_editor_store.set_selected_entity(entity);
|
||||||
},
|
},
|
||||||
() => {
|
() => {
|
||||||
entity.position = new_position;
|
entity.world_position = new_position;
|
||||||
quest_editor_store.set_selected_entity(entity);
|
quest_editor_store.set_selected_entity(entity);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@ -286,22 +284,13 @@ class QuestEditorStore {
|
|||||||
});
|
});
|
||||||
|
|
||||||
private set_section_on_quest_entity = (entity: ObservableQuestEntity, sections: Section[]) => {
|
private set_section_on_quest_entity = (entity: ObservableQuestEntity, sections: Section[]) => {
|
||||||
let { x, y, z } = entity.position;
|
|
||||||
|
|
||||||
const section = sections.find(s => s.id === entity.section_id);
|
const section = sections.find(s => s.id === entity.section_id);
|
||||||
|
|
||||||
if (section) {
|
if (section) {
|
||||||
const { x: sec_x, y: sec_y, z: sec_z } = section.position;
|
entity.section = section;
|
||||||
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 {
|
} else {
|
||||||
logger.warn(`Section ${entity.section_id} not found.`);
|
logger.warn(`Section ${entity.section_id} not found.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
entity.set_position_and_section(new Vec3(x, y, z), section);
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
import { Vec3 } from "../data_formats/vector";
|
import { Vec3 } from "../data_formats/vector";
|
||||||
import { ObservableQuest, ObservableQuestNpc, ObservableQuestObject } from "../domain";
|
import { ObservableQuestNpc, ObservableQuestObject } from "../domain";
|
||||||
import { area_store } from "./AreaStore";
|
import { Instruction, SegmentType } from "../scripting/instructions";
|
||||||
import { SegmentType, Instruction } from "../scripting/instructions";
|
|
||||||
import { Opcode } from "../scripting/opcodes";
|
import { Opcode } from "../scripting/opcodes";
|
||||||
import { Episode } from "../data_formats/parsing/quest/Episode";
|
import { Episode } from "../data_formats/parsing/quest/Episode";
|
||||||
import { ObjectType } from "../data_formats/parsing/quest/object_types";
|
import { ObjectType } from "../data_formats/parsing/quest/object_types";
|
||||||
import { NpcType } from "../data_formats/parsing/quest/npc_types";
|
import { NpcType } from "../data_formats/parsing/quest/npc_types";
|
||||||
|
import { ObservableQuest } from "../domain/ObservableQuest";
|
||||||
|
|
||||||
export function create_new_quest(episode: Episode): ObservableQuest {
|
export function create_new_quest(episode: Episode): ObservableQuest {
|
||||||
if (episode === Episode.II) throw new Error("Episode II not yet supported.");
|
if (episode === Episode.II) throw new Error("Episode II not yet supported.");
|
||||||
@ -18,7 +18,7 @@ export function create_new_quest(episode: Episode): ObservableQuest {
|
|||||||
"Created with phantasmal.world.",
|
"Created with phantasmal.world.",
|
||||||
"Created with phantasmal.world.",
|
"Created with phantasmal.world.",
|
||||||
episode,
|
episode,
|
||||||
[area_store.get_variant(episode, 0, 0)],
|
new Map().set(0, 0),
|
||||||
create_default_objects(),
|
create_default_objects(),
|
||||||
create_default_npcs(),
|
create_default_npcs(),
|
||||||
[],
|
[],
|
||||||
|
@ -181,7 +181,7 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
this.disposers.push(
|
this.disposers.push(
|
||||||
this.dispose,
|
this.dispose,
|
||||||
autorun(this.update_model),
|
autorun(this.update_model),
|
||||||
autorun(this.update_model_markers)
|
autorun(this.update_model_markers),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -209,7 +209,7 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
const quest = quest_editor_store.current_quest;
|
const quest = quest_editor_store.current_quest;
|
||||||
|
|
||||||
if (quest && this.editor && this.assembly_analyser) {
|
if (quest && this.editor && this.assembly_analyser) {
|
||||||
const assembly = this.assembly_analyser.disassemble(quest.object_code);
|
const assembly = this.assembly_analyser.disassemble(quest);
|
||||||
const model = editor.createModel(assembly.join("\n"), "psoasm");
|
const model = editor.createModel(assembly.join("\n"), "psoasm");
|
||||||
|
|
||||||
quest_editor_store.script_undo.action = new Action(
|
quest_editor_store.script_undo.action = new Action(
|
||||||
@ -223,7 +223,7 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
if (this.editor) {
|
if (this.editor) {
|
||||||
this.editor.trigger("redo stack", "redo", undefined);
|
this.editor.trigger("redo stack", "redo", undefined);
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
let initial_version = model.getAlternativeVersionId();
|
let initial_version = model.getAlternativeVersionId();
|
||||||
@ -290,7 +290,7 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
endLineNumber: error.line_no,
|
endLineNumber: error.line_no,
|
||||||
startColumn: error.col,
|
startColumn: error.col,
|
||||||
endColumn: error.col + error.length,
|
endColumn: error.col + error.length,
|
||||||
}))
|
})),
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -30,17 +30,17 @@ export class EntityInfoComponent extends Component {
|
|||||||
<td>{section_id}</td>
|
<td>{section_id}</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th colSpan={2}>World position:</th>
|
<th colSpan={2}>Section position:</th>
|
||||||
</tr>
|
</tr>
|
||||||
<CoordRow entity={entity} position_type="position" coord="x" />
|
<CoordRow entity={entity} position_type="position" coord="x" />
|
||||||
<CoordRow entity={entity} position_type="position" coord="y" />
|
<CoordRow entity={entity} position_type="position" coord="y" />
|
||||||
<CoordRow entity={entity} position_type="position" coord="z" />
|
<CoordRow entity={entity} position_type="position" coord="z" />
|
||||||
<tr>
|
<tr>
|
||||||
<th colSpan={2}>Section position:</th>
|
<th colSpan={2}>World position:</th>
|
||||||
</tr>
|
</tr>
|
||||||
<CoordRow entity={entity} position_type="section_position" coord="x" />
|
<CoordRow entity={entity} position_type="world_position" coord="x" />
|
||||||
<CoordRow entity={entity} position_type="section_position" coord="y" />
|
<CoordRow entity={entity} position_type="world_position" coord="y" />
|
||||||
<CoordRow entity={entity} position_type="section_position" coord="z" />
|
<CoordRow entity={entity} position_type="world_position" coord="z" />
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
);
|
);
|
||||||
@ -58,7 +58,7 @@ export class EntityInfoComponent extends Component {
|
|||||||
|
|
||||||
type CoordProps = {
|
type CoordProps = {
|
||||||
entity: ObservableQuestEntity;
|
entity: ObservableQuestEntity;
|
||||||
position_type: "position" | "section_position";
|
position_type: "position" | "world_position";
|
||||||
coord: "x" | "y" | "z";
|
coord: "x" | "y" | "z";
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -127,15 +127,15 @@ class CoordInput extends Component<CoordProps, { value: number; initial_position
|
|||||||
}
|
}
|
||||||
|
|
||||||
private focus = () => {
|
private focus = () => {
|
||||||
this.setState({ initial_position: this.props.entity.position });
|
this.setState({ initial_position: this.props.entity.world_position });
|
||||||
};
|
};
|
||||||
|
|
||||||
private blur = () => {
|
private blur = () => {
|
||||||
if (!this.state.initial_position.equals(this.props.entity.position)) {
|
if (!this.state.initial_position.equals(this.props.entity.world_position)) {
|
||||||
quest_editor_store.push_entity_move_action(
|
quest_editor_store.push_entity_move_action(
|
||||||
this.props.entity,
|
this.props.entity,
|
||||||
this.state.initial_position,
|
this.state.initial_position,
|
||||||
this.props.entity.position,
|
this.props.entity.world_position,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -4,5 +4,5 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.main > * {
|
.main > * {
|
||||||
margin: 0 3px;
|
margin: 0 3px !important;
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import { Button, Dropdown, Form, Icon, Input, Menu, Modal, Select, Upload } from "antd";
|
import { Button, Dropdown, Form, Icon, Input, Menu, Modal, Select, Upload } from "antd";
|
||||||
import { ClickParam } from "antd/lib/menu";
|
import { ClickParam } from "antd/lib/menu";
|
||||||
import { UploadChangeParam, UploadFile } from "antd/lib/upload/interface";
|
import { UploadChangeParam, UploadFile } from "antd/lib/upload/interface";
|
||||||
import { computed } from "mobx";
|
|
||||||
import { observer } from "mobx-react";
|
import { observer } from "mobx-react";
|
||||||
import React, { ChangeEvent, Component, ReactNode } from "react";
|
import React, { ChangeEvent, Component, ReactNode } from "react";
|
||||||
import { area_store } from "../../stores/AreaStore";
|
import { area_store } from "../../stores/AreaStore";
|
||||||
@ -93,23 +92,6 @@ export class Toolbar extends Component {
|
|||||||
|
|
||||||
@observer
|
@observer
|
||||||
class AreaComponent extends Component {
|
class AreaComponent extends Component {
|
||||||
@computed private get entities_per_area(): Map<number, number> {
|
|
||||||
const quest = quest_editor_store.current_quest;
|
|
||||||
const map = new Map<number, number>();
|
|
||||||
|
|
||||||
if (quest) {
|
|
||||||
for (const npc of quest.npcs) {
|
|
||||||
map.set(npc.area_id, (map.get(npc.area_id) || 0) + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const obj of quest.objects) {
|
|
||||||
map.set(obj.area_id, (map.get(obj.area_id) || 0) + 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return map;
|
|
||||||
}
|
|
||||||
|
|
||||||
render(): ReactNode {
|
render(): ReactNode {
|
||||||
const quest = quest_editor_store.current_quest;
|
const quest = quest_editor_store.current_quest;
|
||||||
const areas = quest ? area_store.get_areas_for_episode(quest.episode) : [];
|
const areas = quest ? area_store.get_areas_for_episode(quest.episode) : [];
|
||||||
@ -123,7 +105,7 @@ class AreaComponent extends Component {
|
|||||||
disabled={!quest}
|
disabled={!quest}
|
||||||
>
|
>
|
||||||
{areas.map(area => {
|
{areas.map(area => {
|
||||||
const entity_count = quest && this.entities_per_area.get(area.id);
|
const entity_count = quest && quest.entities_per_area.get(area.id);
|
||||||
return (
|
return (
|
||||||
<Select.Option key={area.id} value={area.id}>
|
<Select.Option key={area.id} value={area.id}>
|
||||||
{area.name}
|
{area.name}
|
||||||
|
Loading…
Reference in New Issue
Block a user