mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Basic entity creation via drag and drop.
This commit is contained in:
parent
4e55f1b9fd
commit
7ae4ad428c
@ -24,7 +24,7 @@ export type QuestNpc = {
|
||||
readonly pso_type_id: number;
|
||||
readonly npc_id: number;
|
||||
readonly script_label: number;
|
||||
readonly roaming: number;
|
||||
readonly pso_roaming: number;
|
||||
};
|
||||
|
||||
export type QuestObject = {
|
||||
@ -56,6 +56,10 @@ export function entity_type_to_string(type: EntityType): string {
|
||||
return (NpcType as any)[type] || (ObjectType as any)[type];
|
||||
}
|
||||
|
||||
export function is_npc_type(entity_type: EntityType): entity_type is NpcType {
|
||||
return NpcType[entity_type] != undefined;
|
||||
}
|
||||
|
||||
export function entity_data(type: EntityType): EntityTypeData {
|
||||
return npc_data(type as NpcType) || object_data(type as ObjectType);
|
||||
}
|
||||
|
@ -19,7 +19,7 @@ import { QuestNpc, QuestObject } from "./entities";
|
||||
import { Episode } from "./Episode";
|
||||
import { object_data, ObjectType, pso_id_to_object_type } from "./object_types";
|
||||
import { parse_qst, QstContainedFile, write_qst } from "./qst";
|
||||
import { NpcType } from "./npc_types";
|
||||
import { npc_data, NpcType } from "./npc_types";
|
||||
|
||||
const logger = Logger.get("core/data_formats/parsing/quest");
|
||||
|
||||
@ -284,7 +284,7 @@ function parse_npc_data(episode: number, npcs: DatNpc[]): QuestNpc[] {
|
||||
pso_type_id: npc_data.type_id,
|
||||
npc_id: npc_data.npc_id,
|
||||
script_label: Math.round(npc_data.script_label),
|
||||
roaming: npc_data.roaming,
|
||||
pso_roaming: npc_data.roaming,
|
||||
};
|
||||
});
|
||||
}
|
||||
@ -583,307 +583,29 @@ function npcs_to_dat_data(npcs: QuestNpc[]): DatNpc[] {
|
||||
const dv = new DataView(new ArrayBuffer(4));
|
||||
|
||||
return npcs.map(npc => {
|
||||
const type_data = npc_type_to_dat_data(npc.type) || {
|
||||
type_id: npc.pso_type_id,
|
||||
roaming: npc.roaming,
|
||||
regular: true,
|
||||
};
|
||||
const type_data = npc_data(npc.type);
|
||||
const type_id =
|
||||
type_data.pso_type_id == undefined ? npc.pso_type_id : type_data.pso_type_id;
|
||||
const roaming = type_data.pso_roaming == undefined ? npc.pso_roaming : type_data.pso_roaming;
|
||||
const regular = type_data.pso_regular == undefined ? true : type_data.pso_regular;
|
||||
|
||||
dv.setFloat32(0, npc.scale.y);
|
||||
dv.setUint32(0, (dv.getUint32(0) & ~0x800000) | (type_data.regular ? 0 : 0x800000));
|
||||
dv.setUint32(0, (dv.getUint32(0) & ~0x800000) | (regular ? 0 : 0x800000));
|
||||
const scale_y = dv.getFloat32(0);
|
||||
|
||||
let scale = new Vec3(npc.scale.x, scale_y, npc.scale.z);
|
||||
|
||||
return {
|
||||
type_id: type_data.type_id,
|
||||
type_id,
|
||||
section_id: npc.section_id,
|
||||
position: npc.position,
|
||||
rotation: npc.rotation,
|
||||
scale,
|
||||
npc_id: npc.npc_id,
|
||||
script_label: npc.script_label,
|
||||
roaming: type_data.roaming,
|
||||
roaming,
|
||||
area_id: npc.area_id,
|
||||
unknown: npc.unknown,
|
||||
};
|
||||
});
|
||||
}
|
||||
|
||||
function npc_type_to_dat_data(
|
||||
type: NpcType,
|
||||
): { type_id: number; roaming: number; regular: boolean } | undefined {
|
||||
switch (type) {
|
||||
default:
|
||||
throw new Error(`Unexpected type ${NpcType[type]}.`);
|
||||
|
||||
case NpcType.Unknown:
|
||||
return undefined;
|
||||
|
||||
case NpcType.FemaleFat:
|
||||
return { type_id: 0x004, roaming: 0, regular: true };
|
||||
case NpcType.FemaleMacho:
|
||||
return { type_id: 0x005, roaming: 0, regular: true };
|
||||
case NpcType.FemaleTall:
|
||||
return { type_id: 0x007, roaming: 0, regular: true };
|
||||
case NpcType.MaleDwarf:
|
||||
return { type_id: 0x00a, roaming: 0, regular: true };
|
||||
case NpcType.MaleFat:
|
||||
return { type_id: 0x00b, roaming: 0, regular: true };
|
||||
case NpcType.MaleMacho:
|
||||
return { type_id: 0x00c, roaming: 0, regular: true };
|
||||
case NpcType.MaleOld:
|
||||
return { type_id: 0x00d, roaming: 0, regular: true };
|
||||
case NpcType.BlueSoldier:
|
||||
return { type_id: 0x019, roaming: 0, regular: true };
|
||||
case NpcType.RedSoldier:
|
||||
return { type_id: 0x01a, roaming: 0, regular: true };
|
||||
case NpcType.Principal:
|
||||
return { type_id: 0x01b, roaming: 0, regular: true };
|
||||
case NpcType.Tekker:
|
||||
return { type_id: 0x01c, roaming: 0, regular: true };
|
||||
case NpcType.GuildLady:
|
||||
return { type_id: 0x01d, roaming: 0, regular: true };
|
||||
case NpcType.Scientist:
|
||||
return { type_id: 0x01e, roaming: 0, regular: true };
|
||||
case NpcType.Nurse:
|
||||
return { type_id: 0x01f, roaming: 0, regular: true };
|
||||
case NpcType.Irene:
|
||||
return { type_id: 0x020, roaming: 0, regular: true };
|
||||
case NpcType.ItemShop:
|
||||
return { type_id: 0x0f1, roaming: 0, regular: true };
|
||||
case NpcType.Nurse2:
|
||||
return { type_id: 0x0fe, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.Hildebear:
|
||||
return { type_id: 0x040, roaming: 0, regular: true };
|
||||
case NpcType.Hildeblue:
|
||||
return { type_id: 0x040, roaming: 1, regular: true };
|
||||
case NpcType.RagRappy:
|
||||
return { type_id: 0x041, roaming: 0, regular: true };
|
||||
case NpcType.AlRappy:
|
||||
return { type_id: 0x041, roaming: 1, regular: true };
|
||||
case NpcType.Monest:
|
||||
return { type_id: 0x042, roaming: 0, regular: true };
|
||||
case NpcType.SavageWolf:
|
||||
return { type_id: 0x043, roaming: 0, regular: true };
|
||||
case NpcType.BarbarousWolf:
|
||||
return { type_id: 0x043, roaming: 0, regular: false };
|
||||
case NpcType.Booma:
|
||||
return { type_id: 0x044, roaming: 0, regular: true };
|
||||
case NpcType.Gobooma:
|
||||
return { type_id: 0x044, roaming: 1, regular: true };
|
||||
case NpcType.Gigobooma:
|
||||
return { type_id: 0x044, roaming: 2, regular: true };
|
||||
case NpcType.Dragon:
|
||||
return { type_id: 0x0c0, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.GrassAssassin:
|
||||
return { type_id: 0x060, roaming: 0, regular: true };
|
||||
case NpcType.PoisonLily:
|
||||
return { type_id: 0x061, roaming: 0, regular: true };
|
||||
case NpcType.NarLily:
|
||||
return { type_id: 0x061, roaming: 1, regular: true };
|
||||
case NpcType.NanoDragon:
|
||||
return { type_id: 0x062, roaming: 0, regular: true };
|
||||
case NpcType.EvilShark:
|
||||
return { type_id: 0x063, roaming: 0, regular: true };
|
||||
case NpcType.PalShark:
|
||||
return { type_id: 0x063, roaming: 1, regular: true };
|
||||
case NpcType.GuilShark:
|
||||
return { type_id: 0x063, roaming: 2, regular: true };
|
||||
case NpcType.PofuillySlime:
|
||||
return { type_id: 0x064, roaming: 0, regular: true };
|
||||
case NpcType.PouillySlime:
|
||||
return { type_id: 0x064, roaming: 0, regular: false };
|
||||
case NpcType.PanArms:
|
||||
return { type_id: 0x065, roaming: 0, regular: true };
|
||||
case NpcType.DeRolLe:
|
||||
return { type_id: 0x0c1, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.Dubchic:
|
||||
return { type_id: 0x080, roaming: 0, regular: true };
|
||||
case NpcType.Gilchic:
|
||||
return { type_id: 0x080, roaming: 1, regular: true };
|
||||
case NpcType.Garanz:
|
||||
return { type_id: 0x081, roaming: 0, regular: true };
|
||||
case NpcType.SinowBeat:
|
||||
return { type_id: 0x082, roaming: 0, regular: true };
|
||||
case NpcType.SinowGold:
|
||||
return { type_id: 0x082, roaming: 0, regular: false };
|
||||
case NpcType.Canadine:
|
||||
return { type_id: 0x083, roaming: 0, regular: true };
|
||||
case NpcType.Canane:
|
||||
return { type_id: 0x084, roaming: 0, regular: true };
|
||||
case NpcType.Dubswitch:
|
||||
return { type_id: 0x085, roaming: 0, regular: true };
|
||||
case NpcType.VolOpt:
|
||||
return { type_id: 0x0c5, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.Delsaber:
|
||||
return { type_id: 0x0a0, roaming: 0, regular: true };
|
||||
case NpcType.ChaosSorcerer:
|
||||
return { type_id: 0x0a1, roaming: 0, regular: true };
|
||||
case NpcType.DarkGunner:
|
||||
return { type_id: 0x0a2, roaming: 0, regular: true };
|
||||
case NpcType.ChaosBringer:
|
||||
return { type_id: 0x0a4, roaming: 0, regular: true };
|
||||
case NpcType.DarkBelra:
|
||||
return { type_id: 0x0a5, roaming: 0, regular: true };
|
||||
case NpcType.Dimenian:
|
||||
return { type_id: 0x0a6, roaming: 0, regular: true };
|
||||
case NpcType.LaDimenian:
|
||||
return { type_id: 0x0a6, roaming: 1, regular: true };
|
||||
case NpcType.SoDimenian:
|
||||
return { type_id: 0x0a6, roaming: 2, regular: true };
|
||||
case NpcType.Bulclaw:
|
||||
return { type_id: 0x0a7, roaming: 0, regular: true };
|
||||
case NpcType.Claw:
|
||||
return { type_id: 0x0a8, roaming: 0, regular: true };
|
||||
case NpcType.DarkFalz:
|
||||
return { type_id: 0x0c8, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.Hildebear2:
|
||||
return { type_id: 0x040, roaming: 0, regular: true };
|
||||
case NpcType.Hildeblue2:
|
||||
return { type_id: 0x040, roaming: 1, regular: true };
|
||||
case NpcType.RagRappy2:
|
||||
return { type_id: 0x041, roaming: 0, regular: true };
|
||||
case NpcType.LoveRappy:
|
||||
return { type_id: 0x041, roaming: 1, regular: true };
|
||||
case NpcType.Monest2:
|
||||
return { type_id: 0x042, roaming: 0, regular: true };
|
||||
case NpcType.PoisonLily2:
|
||||
return { type_id: 0x061, roaming: 0, regular: true };
|
||||
case NpcType.NarLily2:
|
||||
return { type_id: 0x061, roaming: 1, regular: true };
|
||||
case NpcType.GrassAssassin2:
|
||||
return { type_id: 0x060, roaming: 0, regular: true };
|
||||
case NpcType.Dimenian2:
|
||||
return { type_id: 0x0a6, roaming: 0, regular: true };
|
||||
case NpcType.LaDimenian2:
|
||||
return { type_id: 0x0a6, roaming: 1, regular: true };
|
||||
case NpcType.SoDimenian2:
|
||||
return { type_id: 0x0a6, roaming: 2, regular: true };
|
||||
case NpcType.DarkBelra2:
|
||||
return { type_id: 0x0a5, roaming: 0, regular: true };
|
||||
case NpcType.BarbaRay:
|
||||
return { type_id: 0x0cb, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.SavageWolf2:
|
||||
return { type_id: 0x043, roaming: 0, regular: true };
|
||||
case NpcType.BarbarousWolf2:
|
||||
return { type_id: 0x043, roaming: 0, regular: false };
|
||||
case NpcType.PanArms2:
|
||||
return { type_id: 0x065, roaming: 0, regular: true };
|
||||
case NpcType.Dubchic2:
|
||||
return { type_id: 0x080, roaming: 0, regular: true };
|
||||
case NpcType.Gilchic2:
|
||||
return { type_id: 0x080, roaming: 1, regular: true };
|
||||
case NpcType.Garanz2:
|
||||
return { type_id: 0x081, roaming: 0, regular: true };
|
||||
case NpcType.Dubswitch2:
|
||||
return { type_id: 0x085, roaming: 0, regular: true };
|
||||
case NpcType.Delsaber2:
|
||||
return { type_id: 0x0a0, roaming: 0, regular: true };
|
||||
case NpcType.ChaosSorcerer2:
|
||||
return { type_id: 0x0a1, roaming: 0, regular: true };
|
||||
case NpcType.GolDragon:
|
||||
return { type_id: 0x0cc, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.SinowBerill:
|
||||
return { type_id: 0x0d4, roaming: 0, regular: true };
|
||||
case NpcType.SinowSpigell:
|
||||
return { type_id: 0x0d4, roaming: 1, regular: true };
|
||||
case NpcType.Merillia:
|
||||
return { type_id: 0x0d5, roaming: 0, regular: true };
|
||||
case NpcType.Meriltas:
|
||||
return { type_id: 0x0d5, roaming: 1, regular: true };
|
||||
case NpcType.Mericarol:
|
||||
return { type_id: 0x0d6, roaming: 0, regular: true };
|
||||
case NpcType.Mericus:
|
||||
return { type_id: 0x0d6, roaming: 1, regular: true };
|
||||
case NpcType.Merikle:
|
||||
return { type_id: 0x0d6, roaming: 2, regular: true };
|
||||
case NpcType.UlGibbon:
|
||||
return { type_id: 0x0d7, roaming: 0, regular: true };
|
||||
case NpcType.ZolGibbon:
|
||||
return { type_id: 0x0d7, roaming: 1, regular: true };
|
||||
case NpcType.Gibbles:
|
||||
return { type_id: 0x0d8, roaming: 0, regular: true };
|
||||
case NpcType.Gee:
|
||||
return { type_id: 0x0d9, roaming: 0, regular: true };
|
||||
case NpcType.GiGue:
|
||||
return { type_id: 0x0da, roaming: 0, regular: true };
|
||||
case NpcType.GalGryphon:
|
||||
return { type_id: 0x0c0, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.Deldepth:
|
||||
return { type_id: 0x0db, roaming: 0, regular: true };
|
||||
case NpcType.Delbiter:
|
||||
return { type_id: 0x0dc, roaming: 0, regular: true };
|
||||
case NpcType.Dolmolm:
|
||||
return { type_id: 0x0dd, roaming: 0, regular: true };
|
||||
case NpcType.Dolmdarl:
|
||||
return { type_id: 0x0dd, roaming: 1, regular: true };
|
||||
case NpcType.Morfos:
|
||||
return { type_id: 0x0de, roaming: 0, regular: true };
|
||||
case NpcType.Recobox:
|
||||
return { type_id: 0x0df, roaming: 0, regular: true };
|
||||
case NpcType.Epsilon:
|
||||
return { type_id: 0x0e0, roaming: 0, regular: true };
|
||||
case NpcType.SinowZoa:
|
||||
return { type_id: 0x0e0, roaming: 0, regular: true };
|
||||
case NpcType.SinowZele:
|
||||
return { type_id: 0x0e0, roaming: 1, regular: true };
|
||||
case NpcType.IllGill:
|
||||
return { type_id: 0x0e1, roaming: 0, regular: true };
|
||||
case NpcType.DelLily:
|
||||
return { type_id: 0x061, roaming: 0, regular: true };
|
||||
case NpcType.OlgaFlow:
|
||||
return { type_id: 0x0ca, roaming: 0, regular: true };
|
||||
|
||||
case NpcType.SandRappy:
|
||||
return { type_id: 0x041, roaming: 0, regular: true };
|
||||
case NpcType.DelRappy:
|
||||
return { type_id: 0x041, roaming: 1, regular: true };
|
||||
case NpcType.Astark:
|
||||
return { type_id: 0x110, roaming: 0, regular: true };
|
||||
case NpcType.SatelliteLizard:
|
||||
return { type_id: 0x111, roaming: 0, regular: true };
|
||||
case NpcType.Yowie:
|
||||
return { type_id: 0x111, roaming: 0, regular: false };
|
||||
case NpcType.MerissaA:
|
||||
return { type_id: 0x112, roaming: 0, regular: true };
|
||||
case NpcType.MerissaAA:
|
||||
return { type_id: 0x112, roaming: 1, regular: true };
|
||||
case NpcType.Girtablulu:
|
||||
return { type_id: 0x113, roaming: 0, regular: true };
|
||||
case NpcType.Zu:
|
||||
return { type_id: 0x114, roaming: 0, regular: true };
|
||||
case NpcType.Pazuzu:
|
||||
return { type_id: 0x114, roaming: 1, regular: true };
|
||||
case NpcType.Boota:
|
||||
return { type_id: 0x115, roaming: 0, regular: true };
|
||||
case NpcType.ZeBoota:
|
||||
return { type_id: 0x115, roaming: 1, regular: true };
|
||||
case NpcType.BaBoota:
|
||||
return { type_id: 0x115, roaming: 2, regular: true };
|
||||
case NpcType.Dorphon:
|
||||
return { type_id: 0x116, roaming: 0, regular: true };
|
||||
case NpcType.DorphonEclair:
|
||||
return { type_id: 0x116, roaming: 1, regular: true };
|
||||
case NpcType.Goran:
|
||||
return { type_id: 0x117, roaming: 0, regular: true };
|
||||
case NpcType.PyroGoran:
|
||||
return { type_id: 0x117, roaming: 1, regular: true };
|
||||
case NpcType.GoranDetonator:
|
||||
return { type_id: 0x117, roaming: 2, regular: true };
|
||||
case NpcType.SaintMilion:
|
||||
return { type_id: 0x119, roaming: 0, regular: true };
|
||||
case NpcType.Shambertin:
|
||||
return { type_id: 0x119, roaming: 1, regular: true };
|
||||
case NpcType.Kondrieu:
|
||||
return { type_id: 0x119, roaming: 0, regular: false };
|
||||
}
|
||||
}
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -29,10 +29,7 @@ export interface ListProperty<T> extends Property<T[]> {
|
||||
|
||||
get(index: number): T;
|
||||
|
||||
observe_list(
|
||||
observer: (change: ListPropertyChangeEvent<T>) => void,
|
||||
options?: { call_now?: boolean },
|
||||
): Disposable;
|
||||
observe_list(observer: (change: ListPropertyChangeEvent<T>) => void): Disposable;
|
||||
}
|
||||
|
||||
export function is_list_property<T>(observable: Observable<T[]>): observable is ListProperty<T> {
|
||||
|
@ -4,7 +4,7 @@ import { WritableProperty } from "../WritableProperty";
|
||||
import { Observable } from "../../Observable";
|
||||
import { property } from "../../index";
|
||||
import { AbstractProperty } from "../AbstractProperty";
|
||||
import { Property } from "../Property";
|
||||
import { is_property, Property } from "../Property";
|
||||
import { is_list_property, ListChangeType, ListPropertyChangeEvent } from "./ListProperty";
|
||||
import Logger from "js-logger";
|
||||
|
||||
@ -97,12 +97,18 @@ export class SimpleListProperty<T> extends AbstractProperty<T[]>
|
||||
|
||||
bind_to(observable: Observable<T[]>): Disposable {
|
||||
if (is_list_property(observable)) {
|
||||
this.val = observable.val;
|
||||
|
||||
return observable.observe_list(change => {
|
||||
if (change.type === ListChangeType.ListChange) {
|
||||
this.splice(change.index, change.removed.length, ...change.inserted);
|
||||
}
|
||||
});
|
||||
} else {
|
||||
if (is_property(observable)) {
|
||||
this.val = observable.val;
|
||||
}
|
||||
|
||||
return observable.observe(({ value }) => this.set_val(value));
|
||||
}
|
||||
}
|
||||
|
@ -72,10 +72,10 @@ export abstract class Renderer implements Disposable {
|
||||
this.schedule_render();
|
||||
}
|
||||
|
||||
pointer_pos_to_device_coords(e: MouseEvent): Vector2 {
|
||||
pointer_pos_to_device_coords(v: Vector2): Vector2 {
|
||||
const coords = this.renderer.getSize(new Vector2());
|
||||
coords.width = (e.offsetX / coords.width) * 2 - 1;
|
||||
coords.height = (e.offsetY / coords.height) * -2 + 1;
|
||||
coords.width = (v.x / coords.width) * 2 - 1;
|
||||
coords.height = (v.y / coords.height) * -2 + 1;
|
||||
return coords;
|
||||
}
|
||||
|
||||
|
22
src/quest_editor/gui/EntityListView.css
Normal file
22
src/quest_editor/gui/EntityListView.css
Normal file
@ -0,0 +1,22 @@
|
||||
.quest_editor_EntityListView {
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
.quest_editor_EntityListView_entity_list {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fill, 100px);
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.quest_editor_EntityListView_entity {
|
||||
box-sizing: border-box;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
text-align: center;
|
||||
height: 100px;
|
||||
padding: 5px;
|
||||
border: solid 2px olivedrab;
|
||||
background-color: darkolivegreen;
|
||||
color: greenyellow;
|
||||
}
|
161
src/quest_editor/gui/EntityListView.ts
Normal file
161
src/quest_editor/gui/EntityListView.ts
Normal file
@ -0,0 +1,161 @@
|
||||
import { ResizableWidget } from "../../core/gui/ResizableWidget";
|
||||
import { bind_children_to, el } from "../../core/gui/dom";
|
||||
import "./EntityListView.css";
|
||||
import { entity_data, EntityType } from "../../core/data_formats/parsing/quest/entities";
|
||||
import { ListProperty } from "../../core/observable/property/list/ListProperty";
|
||||
import { Vec2 } from "../../core/data_formats/vector";
|
||||
import { Disposable } from "../../core/observable/Disposable";
|
||||
|
||||
export abstract class EntityListView<T extends EntityType> extends ResizableWidget {
|
||||
readonly element: HTMLElement;
|
||||
|
||||
protected constructor(private readonly class_name: string, entities: ListProperty<T>) {
|
||||
super();
|
||||
|
||||
const list_element = el.div({ class: "quest_editor_EntityListView_entity_list" });
|
||||
|
||||
this.element = el.div({ class: `${class_name} quest_editor_EntityListView` }, list_element);
|
||||
|
||||
this.disposables(
|
||||
bind_children_to(list_element, entities, this.create_entity_element),
|
||||
|
||||
make_draggable(list_element, target => {
|
||||
if (target !== list_element) {
|
||||
const drag_element = target.cloneNode(true) as HTMLElement;
|
||||
drag_element.style.width = "100px";
|
||||
return [drag_element, entities.get(parseInt(target.dataset.index!, 10))];
|
||||
} else {
|
||||
return undefined;
|
||||
}
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
private create_entity_element = (entity: T, index: number): HTMLElement => {
|
||||
const div = el.div({
|
||||
class: "quest_editor_EntityListView_entity",
|
||||
text: entity_data(entity).name,
|
||||
data: { index: index.toString() },
|
||||
});
|
||||
|
||||
div.draggable = true;
|
||||
|
||||
return div;
|
||||
};
|
||||
}
|
||||
|
||||
export type EntityDrag = {
|
||||
readonly offset_x: number;
|
||||
readonly offset_y: number;
|
||||
readonly data_transfer: DataTransfer;
|
||||
readonly drag_element: HTMLElement;
|
||||
readonly entity_type: EntityType;
|
||||
};
|
||||
|
||||
function make_draggable(
|
||||
element: HTMLElement,
|
||||
start: (target: HTMLElement) => [HTMLElement, any] | undefined,
|
||||
): Disposable {
|
||||
let detail: { drag_element: HTMLElement; entity_type: EntityType } | undefined;
|
||||
const grab_point = new Vec2(0, 0);
|
||||
|
||||
function clear(): void {
|
||||
if (detail) {
|
||||
detail.drag_element.remove();
|
||||
detail = undefined;
|
||||
}
|
||||
}
|
||||
|
||||
function redispatch(e: DragEvent): void {
|
||||
if (e.target instanceof HTMLElement && detail && e.dataTransfer) {
|
||||
e.target.dispatchEvent(
|
||||
new CustomEvent<EntityDrag>(`phantasmal-${e.type}`, {
|
||||
detail: {
|
||||
...detail,
|
||||
data_transfer: e.dataTransfer,
|
||||
offset_x: e.offsetX,
|
||||
offset_y: e.offsetY,
|
||||
},
|
||||
bubbles: true,
|
||||
}),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
function dragstart(e: DragEvent): void {
|
||||
if (e.target instanceof HTMLElement) {
|
||||
clear();
|
||||
|
||||
const result = start(e.target);
|
||||
|
||||
if (result) {
|
||||
grab_point.set(e.offsetX + 2, e.offsetY + 2);
|
||||
|
||||
detail = {
|
||||
drag_element: result[0],
|
||||
entity_type: result[1],
|
||||
};
|
||||
|
||||
detail.drag_element.style.position = "fixed";
|
||||
detail.drag_element.style.pointerEvents = "none";
|
||||
detail.drag_element.style.zIndex = "500";
|
||||
detail.drag_element.style.top = "0";
|
||||
detail.drag_element.style.left = "0";
|
||||
detail.drag_element.style.transform = `translate(${e.clientX -
|
||||
grab_point.x}px, ${e.clientY - grab_point.y}px)`;
|
||||
document.body.append(detail.drag_element);
|
||||
|
||||
e.dataTransfer!.setDragImage(el.div(), 0, 0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function dragenter(e: DragEvent): void {
|
||||
redispatch(e);
|
||||
}
|
||||
|
||||
function dragover(e: DragEvent): void {
|
||||
if (e.target instanceof HTMLElement && detail) {
|
||||
detail.drag_element.style.transform = `translate(${e.clientX -
|
||||
grab_point.x}px, ${e.clientY - grab_point.y}px)`;
|
||||
|
||||
redispatch(e);
|
||||
}
|
||||
}
|
||||
|
||||
function dragleave(e: DragEvent): void {
|
||||
redispatch(e);
|
||||
}
|
||||
|
||||
function dragend(): void {
|
||||
clear();
|
||||
}
|
||||
|
||||
function drop(e: DragEvent): void {
|
||||
try {
|
||||
redispatch(e);
|
||||
} finally {
|
||||
clear();
|
||||
}
|
||||
}
|
||||
|
||||
element.addEventListener("dragstart", dragstart);
|
||||
document.addEventListener("dragenter", dragenter);
|
||||
document.addEventListener("dragover", dragover);
|
||||
document.addEventListener("dragleave", dragleave);
|
||||
document.addEventListener("dragend", dragend);
|
||||
document.addEventListener("drop", drop);
|
||||
|
||||
return {
|
||||
dispose(): void {
|
||||
element.removeEventListener("dragstart", dragstart);
|
||||
document.removeEventListener("dragenter", dragenter);
|
||||
document.removeEventListener("dragover", dragover);
|
||||
document.removeEventListener("dragleave", dragleave);
|
||||
document.removeEventListener("dragend", dragend);
|
||||
document.removeEventListener("drop", drop);
|
||||
|
||||
clear();
|
||||
},
|
||||
};
|
||||
}
|
@ -2,9 +2,10 @@ import { ResizableWidget } from "../../core/gui/ResizableWidget";
|
||||
import { el } from "../../core/gui/dom";
|
||||
import { quest_editor_store } from "../stores/QuestEditorStore";
|
||||
import { npc_data, NpcType } from "../../core/data_formats/parsing/quest/npc_types";
|
||||
import { QuestModel } from "../model/QuestModel";
|
||||
import "./NpcCountsView.css";
|
||||
import { DisabledView } from "./DisabledView";
|
||||
import { property } from "../../core/observable";
|
||||
import { QuestNpcModel } from "../model/QuestNpcModel";
|
||||
|
||||
export class NpcCountsView extends ResizableWidget {
|
||||
readonly element = el.div({ class: "quest_editor_NpcCountsView" });
|
||||
@ -26,24 +27,24 @@ export class NpcCountsView extends ResizableWidget {
|
||||
this.disposables(
|
||||
this.no_quest_view.visible.bind_to(no_quest),
|
||||
|
||||
quest.observe(({ value }) => this.update_view(value), {
|
||||
call_now: true,
|
||||
}),
|
||||
quest
|
||||
.flat_map(quest => (quest ? quest.npcs : property([])))
|
||||
.observe(({ value: npcs }) => this.update_view(npcs), {
|
||||
call_now: true,
|
||||
}),
|
||||
);
|
||||
|
||||
this.finalize_construction(NpcCountsView.prototype);
|
||||
}
|
||||
|
||||
private update_view(quest?: QuestModel): void {
|
||||
private update_view(npcs: QuestNpcModel[]): void {
|
||||
const frag = document.createDocumentFragment();
|
||||
|
||||
const npc_counts = new Map<NpcType, number>();
|
||||
|
||||
if (quest) {
|
||||
for (const npc of quest.npcs.val) {
|
||||
const val = npc_counts.get(npc.type) || 0;
|
||||
npc_counts.set(npc.type, val + 1);
|
||||
}
|
||||
for (const npc of npcs) {
|
||||
const val = npc_counts.get(npc.type) || 0;
|
||||
npc_counts.set(npc.type, val + 1);
|
||||
}
|
||||
|
||||
const extra_canadines = (npc_counts.get(NpcType.Canane) || 0) * 8;
|
||||
|
11
src/quest_editor/gui/NpcListView.ts
Normal file
11
src/quest_editor/gui/NpcListView.ts
Normal file
@ -0,0 +1,11 @@
|
||||
import { EntityListView } from "./EntityListView";
|
||||
import { NPC_TYPES, NpcType } from "../../core/data_formats/parsing/quest/npc_types";
|
||||
import { list_property } from "../../core/observable";
|
||||
|
||||
export class NpcListView extends EntityListView<NpcType> {
|
||||
constructor() {
|
||||
super("quest_editor_NpcListView", list_property(undefined, ...NPC_TYPES));
|
||||
|
||||
this.finalize_construction(NpcListView.prototype);
|
||||
}
|
||||
}
|
@ -12,6 +12,7 @@ import { AsmEditorView } from "./AsmEditorView";
|
||||
import { EntityInfoView } from "./EntityInfoView";
|
||||
import { gui_store, GuiTool } from "../../core/stores/GuiStore";
|
||||
import { quest_editor_store } from "../stores/QuestEditorStore";
|
||||
import { NpcListView } from "./NpcListView";
|
||||
import Logger = require("js-logger");
|
||||
|
||||
const logger = Logger.get("quest_editor/gui/QuestEditorView");
|
||||
@ -23,7 +24,7 @@ const VIEW_TO_NAME = new Map<new () => ResizableWidget, string>([
|
||||
[QuestRendererView, "quest_renderer"],
|
||||
[AsmEditorView, "asm_editor"],
|
||||
[EntityInfoView, "entity_info"],
|
||||
// [AddObjectView, "add_object"],
|
||||
[NpcListView, "npc_list_view"],
|
||||
]);
|
||||
|
||||
const DEFAULT_LAYOUT_CONFIG = {
|
||||
@ -83,11 +84,22 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [
|
||||
],
|
||||
},
|
||||
{
|
||||
title: "Entity",
|
||||
type: "component",
|
||||
componentName: VIEW_TO_NAME.get(EntityInfoView),
|
||||
isClosable: false,
|
||||
type: "stack",
|
||||
width: 2,
|
||||
content: [
|
||||
{
|
||||
title: "Entity",
|
||||
type: "component",
|
||||
componentName: VIEW_TO_NAME.get(EntityInfoView),
|
||||
isClosable: false,
|
||||
},
|
||||
{
|
||||
title: "NPCs",
|
||||
type: "component",
|
||||
componentName: VIEW_TO_NAME.get(NpcListView),
|
||||
isClosable: false,
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
|
@ -11,6 +11,7 @@ import {
|
||||
area_collision_geometry_to_object_3d,
|
||||
area_geometry_to_sections_and_object_3d,
|
||||
} from "../rendering/conversion/areas";
|
||||
import { AreaVariantModel } from "../model/AreaVariantModel";
|
||||
|
||||
const render_geometry_cache = new LoadingCache<
|
||||
string,
|
||||
@ -20,46 +21,47 @@ const collision_geometry_cache = new LoadingCache<string, Promise<Object3D>>();
|
||||
|
||||
export async function load_area_sections(
|
||||
episode: Episode,
|
||||
area_id: number,
|
||||
area_variant: number,
|
||||
area_variant: AreaVariantModel,
|
||||
): Promise<SectionModel[]> {
|
||||
return render_geometry_cache.get_or_set(`${episode}-${area_id}-${area_variant}`, () =>
|
||||
load_area_sections_and_render_geometry(episode, area_id, area_variant),
|
||||
return render_geometry_cache.get_or_set(
|
||||
`${episode}-${area_variant.area.id}-${area_variant.id}`,
|
||||
() => load_area_sections_and_render_geometry(episode, area_variant),
|
||||
).sections;
|
||||
}
|
||||
|
||||
export async function load_area_render_geometry(
|
||||
episode: Episode,
|
||||
area_id: number,
|
||||
area_variant: number,
|
||||
area_variant: AreaVariantModel,
|
||||
): Promise<Object3D> {
|
||||
return render_geometry_cache.get_or_set(`${episode}-${area_id}-${area_variant}`, () =>
|
||||
load_area_sections_and_render_geometry(episode, area_id, area_variant),
|
||||
return render_geometry_cache.get_or_set(
|
||||
`${episode}-${area_variant.area.id}-${area_variant.id}`,
|
||||
() => load_area_sections_and_render_geometry(episode, area_variant),
|
||||
).geometry;
|
||||
}
|
||||
|
||||
export async function load_area_collision_geometry(
|
||||
episode: Episode,
|
||||
area_id: number,
|
||||
area_variant: number,
|
||||
area_variant: AreaVariantModel,
|
||||
): Promise<Object3D> {
|
||||
return collision_geometry_cache.get_or_set(`${episode}-${area_id}-${area_variant}`, () =>
|
||||
get_area_asset(episode, area_id, area_variant, "collision").then(buffer =>
|
||||
area_collision_geometry_to_object_3d(
|
||||
parse_area_collision_geometry(new ArrayBufferCursor(buffer, Endianness.Little)),
|
||||
return collision_geometry_cache.get_or_set(
|
||||
`${episode}-${area_variant.area.id}-${area_variant.id}`,
|
||||
() =>
|
||||
get_area_asset(episode, area_variant, "collision").then(buffer =>
|
||||
area_collision_geometry_to_object_3d(
|
||||
parse_area_collision_geometry(new ArrayBufferCursor(buffer, Endianness.Little)),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
function load_area_sections_and_render_geometry(
|
||||
episode: Episode,
|
||||
area_id: number,
|
||||
area_variant: number,
|
||||
area_variant: AreaVariantModel,
|
||||
): { geometry: Promise<Object3D>; sections: Promise<SectionModel[]> } {
|
||||
const promise = get_area_asset(episode, area_id, area_variant, "render").then(buffer =>
|
||||
const promise = get_area_asset(episode, area_variant, "render").then(buffer =>
|
||||
area_geometry_to_sections_and_object_3d(
|
||||
parse_area_geometry(new ArrayBufferCursor(buffer, Endianness.Little)),
|
||||
area_variant,
|
||||
),
|
||||
);
|
||||
|
||||
@ -126,21 +128,23 @@ const area_base_names = [
|
||||
|
||||
async function get_area_asset(
|
||||
episode: Episode,
|
||||
area_id: number,
|
||||
area_variant: number,
|
||||
area_variant: AreaVariantModel,
|
||||
type: "render" | "collision",
|
||||
): Promise<ArrayBuffer> {
|
||||
const base_url = area_version_to_base_url(episode, area_id, area_variant);
|
||||
const base_url = area_version_to_base_url(episode, area_variant);
|
||||
const suffix = type === "render" ? "n.rel" : "c.rel";
|
||||
return load_array_buffer(base_url + suffix);
|
||||
}
|
||||
|
||||
function area_version_to_base_url(episode: Episode, area_id: number, area_variant: number): string {
|
||||
// Exception for Seaside area at night variant 1.
|
||||
function area_version_to_base_url(episode: Episode, area_variant: AreaVariantModel): string {
|
||||
let area_id = area_variant.area.id;
|
||||
let area_variant_id = area_variant.id;
|
||||
|
||||
// Exception for Seaside area at night, variant 1.
|
||||
// Phantasmal World 4 and Lost heart breaker use this to have two tower maps.
|
||||
if (area_id === 16 && area_variant === 1) {
|
||||
if (area_id === 16 && area_variant_id === 1) {
|
||||
area_id = 17;
|
||||
area_variant = 1;
|
||||
area_variant_id = 1;
|
||||
}
|
||||
|
||||
const episode_base_names = area_base_names[episode - 1];
|
||||
@ -148,20 +152,20 @@ function area_version_to_base_url(episode: Episode, area_id: number, area_varian
|
||||
if (0 <= area_id && area_id < episode_base_names.length) {
|
||||
const [base_name, variants] = episode_base_names[area_id];
|
||||
|
||||
if (0 <= area_variant && area_variant < variants) {
|
||||
if (0 <= area_variant_id && area_variant_id < variants) {
|
||||
let variant: string;
|
||||
|
||||
if (variants === 1) {
|
||||
variant = "";
|
||||
} else {
|
||||
variant = String(area_variant);
|
||||
while (variant.length < 2) variant = "0" + variant;
|
||||
variant = String(area_variant_id);
|
||||
variant = variant.padStart(2, "0");
|
||||
}
|
||||
|
||||
return `/maps/map_${base_name}${variant}`;
|
||||
} else {
|
||||
throw new Error(
|
||||
`Unknown variant ${area_variant} of area ${area_id} in episode ${episode}.`,
|
||||
`Unknown variant ${area_variant_id} of area ${area_id} in episode ${episode}.`,
|
||||
);
|
||||
}
|
||||
} else {
|
||||
|
@ -5,11 +5,10 @@ import { AreaModel } from "./AreaModel";
|
||||
import { SectionModel } from "./SectionModel";
|
||||
|
||||
export class AreaVariantModel {
|
||||
readonly id: number;
|
||||
|
||||
readonly area: AreaModel;
|
||||
|
||||
private readonly _sections: WritableListProperty<SectionModel> = list_property();
|
||||
|
||||
readonly id: number;
|
||||
readonly area: AreaModel;
|
||||
readonly sections: ListProperty<SectionModel> = this._sections;
|
||||
|
||||
constructor(id: number, area: AreaModel) {
|
||||
|
@ -73,6 +73,12 @@ export abstract class QuestEntityModel<Type extends EntityType = EntityType> {
|
||||
position: Vec3,
|
||||
rotation: Vec3,
|
||||
) {
|
||||
if (type == undefined) throw new Error("type is required.");
|
||||
if (!Number.isInteger(area_id)) throw new Error("area_id should be an integer.");
|
||||
if (!Number.isInteger(section_id)) throw new Error("section_id should be an integer.");
|
||||
if (!position) throw new Error("position is required.");
|
||||
if (!rotation) throw new Error("rotation is required.");
|
||||
|
||||
this.type = type;
|
||||
this.area_id = area_id;
|
||||
this.section = this._section;
|
||||
|
@ -11,6 +11,7 @@ import { AreaVariantModel } from "./AreaVariantModel";
|
||||
import { area_store } from "../stores/AreaStore";
|
||||
import { ListProperty } from "../../core/observable/property/list/ListProperty";
|
||||
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
|
||||
import { QuestEntityModel } from "./QuestEntityModel";
|
||||
|
||||
const logger = Logger.get("quest_editor/model/QuestModel");
|
||||
|
||||
@ -110,6 +111,7 @@ export class QuestModel {
|
||||
private readonly _long_description: WritableProperty<string> = property("");
|
||||
private readonly _map_designations: WritableProperty<Map<number, number>>;
|
||||
private readonly _area_variants: WritableListProperty<AreaVariantModel> = list_property();
|
||||
private readonly _npcs: WritableListProperty<QuestNpcModel>;
|
||||
|
||||
constructor(
|
||||
id: number,
|
||||
@ -149,7 +151,8 @@ export class QuestModel {
|
||||
this._map_designations = property(map_designations);
|
||||
this.map_designations = this._map_designations;
|
||||
this.objects = list_property(undefined, ...objects);
|
||||
this.npcs = list_property(undefined, ...npcs);
|
||||
this._npcs = list_property(undefined, ...npcs);
|
||||
this.npcs = this._npcs;
|
||||
this.dat_unknowns = dat_unknowns;
|
||||
this.object_code = object_code;
|
||||
this.shop_items = shop_items;
|
||||
@ -176,6 +179,18 @@ export class QuestModel {
|
||||
this.map_designations.observe(this.update_area_variants);
|
||||
}
|
||||
|
||||
add_npc(npc: QuestNpcModel): void {
|
||||
this._npcs.push(npc);
|
||||
}
|
||||
|
||||
remove_entity(entity: QuestEntityModel): void {
|
||||
if (entity instanceof QuestNpcModel) {
|
||||
this._npcs.remove(entity);
|
||||
} else {
|
||||
// TODO: objects
|
||||
}
|
||||
}
|
||||
|
||||
private update_area_variants = (): void => {
|
||||
const variants = new Map<number, AreaVariantModel>();
|
||||
|
||||
|
@ -6,7 +6,7 @@ export class QuestNpcModel extends QuestEntityModel<NpcType> {
|
||||
readonly pso_type_id: number;
|
||||
readonly npc_id: number;
|
||||
readonly script_label: number;
|
||||
readonly roaming: number;
|
||||
readonly pso_roaming: number;
|
||||
readonly scale: Vec3;
|
||||
/**
|
||||
* Data of which the purpose hasn't been discovered yet.
|
||||
@ -18,7 +18,7 @@ export class QuestNpcModel extends QuestEntityModel<NpcType> {
|
||||
pso_type_id: number,
|
||||
npc_id: number,
|
||||
script_label: number,
|
||||
roaming: number,
|
||||
pso_roaming: number,
|
||||
area_id: number,
|
||||
section_id: number,
|
||||
position: Vec3,
|
||||
@ -26,12 +26,27 @@ export class QuestNpcModel extends QuestEntityModel<NpcType> {
|
||||
scale: Vec3,
|
||||
unknown: number[][],
|
||||
) {
|
||||
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.isInteger(script_label)) throw new Error("script_label should be an integer.");
|
||||
if (!Number.isInteger(pso_roaming)) throw new Error("pso_roaming should be an integer.");
|
||||
if (!scale) throw new Error("scale is required.");
|
||||
if (!unknown) throw new Error("unknown is required.");
|
||||
if (unknown.length !== 3)
|
||||
throw new Error(`unknown should be of length 3, was ${unknown.length}.`);
|
||||
if (unknown[0].length !== 10)
|
||||
throw new Error(`unknown[0] should be of length 10, was ${unknown[0].length}`);
|
||||
if (unknown[1].length !== 6)
|
||||
throw new Error(`unknown[1] should be of length 6, was ${unknown[1].length}`);
|
||||
if (unknown[2].length !== 4)
|
||||
throw new Error(`unknown[2] should be of length 4, was ${unknown[2].length}`);
|
||||
|
||||
super(type, area_id, section_id, position, rotation);
|
||||
|
||||
this.pso_type_id = pso_type_id;
|
||||
this.npc_id = npc_id;
|
||||
this.script_label = script_label;
|
||||
this.roaming = roaming;
|
||||
this.pso_roaming = pso_roaming;
|
||||
this.unknown = unknown;
|
||||
this.scale = scale;
|
||||
}
|
||||
|
@ -1,4 +1,5 @@
|
||||
import { Vec3 } from "../../core/data_formats/vector";
|
||||
import { AreaVariantModel } from "./AreaVariantModel";
|
||||
|
||||
export class SectionModel {
|
||||
readonly id: number;
|
||||
@ -6,17 +7,25 @@ export class SectionModel {
|
||||
readonly y_axis_rotation: number;
|
||||
readonly sin_y_axis_rotation: number;
|
||||
readonly cos_y_axis_rotation: number;
|
||||
readonly area_variant: AreaVariantModel;
|
||||
|
||||
constructor(id: number, position: Vec3, y_axis_rotation: number) {
|
||||
constructor(
|
||||
id: number,
|
||||
position: Vec3,
|
||||
y_axis_rotation: number,
|
||||
area_variant: AreaVariantModel,
|
||||
) {
|
||||
if (!Number.isInteger(id) || id < -1)
|
||||
throw new Error(`Expected id to be an integer greater than or equal to -1, got ${id}.`);
|
||||
if (!position) throw new Error("position is required.");
|
||||
if (!Number.isFinite(y_axis_rotation)) throw new Error("y_axis_rotation is required.");
|
||||
if (!area_variant) throw new Error("area_variant is required.");
|
||||
|
||||
this.id = id;
|
||||
this.position = position;
|
||||
this.y_axis_rotation = y_axis_rotation;
|
||||
this.sin_y_axis_rotation = Math.sin(this.y_axis_rotation);
|
||||
this.cos_y_axis_rotation = Math.cos(this.y_axis_rotation);
|
||||
this.area_variant = area_variant;
|
||||
}
|
||||
}
|
||||
|
@ -9,6 +9,9 @@ import { AreaUserData } from "./conversion/areas";
|
||||
import { SectionModel } from "../model/SectionModel";
|
||||
import { Disposable } from "../../core/observable/Disposable";
|
||||
import { Disposer } from "../../core/observable/Disposer";
|
||||
import { EntityDrag } from "../gui/EntityListView";
|
||||
import { is_npc_type } from "../../core/data_formats/parsing/quest/entities";
|
||||
import { npc_data } from "../../core/data_formats/parsing/quest/npc_types";
|
||||
|
||||
const DOWN_VECTOR = new Vector3(0, -1, 0);
|
||||
|
||||
@ -18,10 +21,28 @@ type Highlighted = {
|
||||
};
|
||||
|
||||
type Pick = {
|
||||
/**
|
||||
* Whether we picked an entity that is being created or one that has existed before.
|
||||
*/
|
||||
creating: boolean;
|
||||
|
||||
initial_section?: SectionModel;
|
||||
|
||||
initial_position: Vec3;
|
||||
|
||||
/**
|
||||
* Vector that points from the grabbing point to the model's origin.
|
||||
*/
|
||||
grab_offset: Vector3;
|
||||
|
||||
/**
|
||||
* Vector that points from the grabbing point to the terrain point directly under the model's origin.
|
||||
*/
|
||||
drag_adjust: Vector3;
|
||||
|
||||
/**
|
||||
* Distance to terrain.
|
||||
*/
|
||||
drag_y: number;
|
||||
};
|
||||
|
||||
@ -57,6 +78,13 @@ export class QuestEntityControls implements Disposable {
|
||||
}
|
||||
}),
|
||||
);
|
||||
|
||||
renderer.dom_element.addEventListener("mousedown", this.mousedown);
|
||||
renderer.dom_element.addEventListener("mousemove", this.mousemove);
|
||||
renderer.dom_element.addEventListener("mouseup", this.mouseup);
|
||||
renderer.dom_element.addEventListener("phantasmal-dragenter", this.dragenter);
|
||||
renderer.dom_element.addEventListener("phantasmal-dragover", this.dragover);
|
||||
renderer.dom_element.addEventListener("phantasmal-dragleave", this.dragleave);
|
||||
}
|
||||
|
||||
dispose(): void {
|
||||
@ -71,14 +99,22 @@ export class QuestEntityControls implements Disposable {
|
||||
|
||||
if (mesh) {
|
||||
this.select({ entity, mesh });
|
||||
} else {
|
||||
if (this.selected) {
|
||||
set_color(this.selected, ColorType.Normal);
|
||||
}
|
||||
|
||||
this.selected = undefined;
|
||||
}
|
||||
};
|
||||
|
||||
on_mouse_down = (e: MouseEvent) => {
|
||||
private mousedown = (e: MouseEvent) => {
|
||||
this.process_event(e);
|
||||
this.stop_transforming();
|
||||
|
||||
const new_pick = this.pick_entity(this.renderer.pointer_pos_to_device_coords(e));
|
||||
const new_pick = this.pick_entity(
|
||||
this.renderer.pointer_pos_to_device_coords(new Vector2(e.offsetX, e.offsetY)),
|
||||
);
|
||||
|
||||
if (new_pick) {
|
||||
// Disable camera controls while the user is transforming an entity.
|
||||
@ -93,25 +129,12 @@ export class QuestEntityControls implements Disposable {
|
||||
this.renderer.schedule_render();
|
||||
};
|
||||
|
||||
on_mouse_up = (e: MouseEvent) => {
|
||||
private mousemove = (e: MouseEvent) => {
|
||||
this.process_event(e);
|
||||
|
||||
if (!this.moved_since_last_mouse_down && !this.pick) {
|
||||
// If the user clicks on nothing, deselect the currently selected entity.
|
||||
this.deselect();
|
||||
}
|
||||
|
||||
this.stop_transforming();
|
||||
// Enable camera controls again after transforming an entity.
|
||||
this.renderer.controls.enabled = true;
|
||||
|
||||
this.renderer.schedule_render();
|
||||
};
|
||||
|
||||
on_mouse_move = (e: MouseEvent) => {
|
||||
this.process_event(e);
|
||||
|
||||
const pointer_device_pos = this.renderer.pointer_pos_to_device_coords(e);
|
||||
const pointer_device_pos = this.renderer.pointer_pos_to_device_coords(
|
||||
new Vector2(e.offsetX, e.offsetY),
|
||||
);
|
||||
|
||||
if (this.selected && this.pick) {
|
||||
if (this.moved_since_last_mouse_down) {
|
||||
@ -139,19 +162,168 @@ export class QuestEntityControls implements Disposable {
|
||||
}
|
||||
};
|
||||
|
||||
private process_event(e: MouseEvent): void {
|
||||
if (e.type === "mousedown") {
|
||||
private mouseup = (e: MouseEvent) => {
|
||||
this.process_event(e);
|
||||
|
||||
if (!this.moved_since_last_mouse_down && !this.pick) {
|
||||
// If the user clicks on nothing, deselect the currently selected entity.
|
||||
this.deselect();
|
||||
}
|
||||
|
||||
this.stop_transforming();
|
||||
// Enable camera controls again after transforming an entity.
|
||||
this.renderer.controls.enabled = true;
|
||||
|
||||
this.renderer.schedule_render();
|
||||
};
|
||||
|
||||
private dragenter = (e: Event) => {
|
||||
const area = quest_editor_store.current_area.val;
|
||||
if (!area) return;
|
||||
|
||||
const detail = (e as CustomEvent<EntityDrag>).detail;
|
||||
|
||||
const pointer_position = this.renderer.pointer_pos_to_device_coords(
|
||||
new Vector2(detail.offset_x, detail.offset_y),
|
||||
);
|
||||
|
||||
const { intersection, section } = this.pick_terrain(pointer_position, new Vector3());
|
||||
|
||||
let position: Vec3 | undefined;
|
||||
|
||||
if (intersection) {
|
||||
position = new Vec3(intersection.point.x, intersection.point.y, intersection.point.z);
|
||||
} else {
|
||||
// If the cursor is not over any terrain, we translate the entity across the horizontal plane in which the origin lies.
|
||||
this.raycaster.setFromCamera(pointer_position, this.renderer.camera);
|
||||
const ray = this.raycaster.ray;
|
||||
const plane = new Plane(new Vector3(0, 1, 0), 0);
|
||||
const intersection_point = new Vector3();
|
||||
|
||||
if (ray.intersectPlane(plane, intersection_point)) {
|
||||
position = new Vec3(intersection_point.x, 0, intersection_point.z);
|
||||
}
|
||||
}
|
||||
|
||||
const quest = quest_editor_store.current_quest.val;
|
||||
|
||||
if (quest && position) {
|
||||
if (is_npc_type(detail.entity_type)) {
|
||||
const data = npc_data(detail.entity_type);
|
||||
|
||||
if (data.pso_type_id !== undefined && data.pso_roaming != undefined) {
|
||||
detail.drag_element.style.display = "none";
|
||||
detail.data_transfer.dropEffect = "copy";
|
||||
|
||||
const npc = new QuestNpcModel(
|
||||
detail.entity_type,
|
||||
data.pso_type_id,
|
||||
0,
|
||||
0,
|
||||
data.pso_roaming,
|
||||
section ? section.area_variant.area.id : area.id,
|
||||
section ? section.id : 0,
|
||||
new Vec3(0, 0, 0),
|
||||
new Vec3(0, 0, 0),
|
||||
new Vec3(1, 1, 1),
|
||||
// TODO: do the following values make sense?
|
||||
[[0, 0, 0, 0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0], [0, 0, 0, 0]],
|
||||
);
|
||||
npc.set_world_position(position);
|
||||
quest.add_npc(npc);
|
||||
|
||||
quest_editor_store.set_selected_entity(npc);
|
||||
|
||||
this.pick = {
|
||||
creating: true,
|
||||
initial_section: section,
|
||||
initial_position: position,
|
||||
grab_offset: new Vector3(0, 0, 0),
|
||||
drag_adjust: new Vector3(0, 0, 0),
|
||||
drag_y: 0,
|
||||
};
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
private dragover = (e: Event) => {
|
||||
if (!quest_editor_store.current_area.val) return;
|
||||
|
||||
const detail = (e as CustomEvent<EntityDrag>).detail;
|
||||
detail.data_transfer.dropEffect = "copy";
|
||||
|
||||
if (this.selected && this.pick) {
|
||||
const pointer_device_pos = this.renderer.pointer_pos_to_device_coords(
|
||||
new Vector2(detail.offset_x, detail.offset_y),
|
||||
);
|
||||
|
||||
// TODO:
|
||||
// if (e.buttons === 1) {
|
||||
// User is transforming selected entity.
|
||||
// User is dragging selected entity.
|
||||
// if (e.shiftKey) {
|
||||
// Vertical movement.
|
||||
// this.translate_vertically(this.selected, this.pick, pointer_device_pos);
|
||||
// } else {
|
||||
// Horizontal movement across terrain.
|
||||
this.translate_horizontally(this.selected, this.pick, pointer_device_pos);
|
||||
// }
|
||||
// }
|
||||
|
||||
this.renderer.schedule_render();
|
||||
}
|
||||
};
|
||||
|
||||
private dragleave = (e: Event) => {
|
||||
if (!quest_editor_store.current_area.val) return;
|
||||
|
||||
const detail = (e as CustomEvent<EntityDrag>).detail;
|
||||
if (detail.drag_element) detail.drag_element.style.display = "flex";
|
||||
|
||||
const quest = quest_editor_store.current_quest.val;
|
||||
|
||||
if (quest && this.selected && this.selected.entity.type == detail.entity_type) {
|
||||
quest.remove_entity(this.selected.entity);
|
||||
}
|
||||
};
|
||||
|
||||
private drop = () => {
|
||||
// TODO: push onto undo stack.
|
||||
};
|
||||
|
||||
private process_event(e: Event): void {
|
||||
let offset_x: number;
|
||||
let offset_y: number;
|
||||
|
||||
if (e instanceof MouseEvent) {
|
||||
offset_x = e.offsetX;
|
||||
offset_y = e.offsetY;
|
||||
} else if (e instanceof CustomEvent) {
|
||||
const detail = (e as CustomEvent<EntityDrag>).detail;
|
||||
|
||||
if (!("offset_x" in detail && "offset_y" in detail)) {
|
||||
return;
|
||||
}
|
||||
|
||||
offset_x = detail.offset_x;
|
||||
offset_y = detail.offset_y;
|
||||
} else {
|
||||
return;
|
||||
}
|
||||
|
||||
if (e.type === "mousedown" || e.type === "phantasmal-dragenter") {
|
||||
this.moved_since_last_mouse_down = false;
|
||||
} else {
|
||||
if (
|
||||
e.offsetX !== this.last_pointer_position.x ||
|
||||
e.offsetY !== this.last_pointer_position.y
|
||||
offset_x !== this.last_pointer_position.x ||
|
||||
offset_y !== this.last_pointer_position.y
|
||||
) {
|
||||
this.moved_since_last_mouse_down = true;
|
||||
}
|
||||
}
|
||||
|
||||
this.last_pointer_position.set(e.offsetX, e.offsetY);
|
||||
this.last_pointer_position.set(offset_x, offset_y);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -246,7 +418,7 @@ export class QuestEntityControls implements Disposable {
|
||||
pointer_position: Vector2,
|
||||
): void {
|
||||
// Cast ray adjusted for dragging entities.
|
||||
const { intersection, section } = this.pick_terrain(pointer_position, pick);
|
||||
const { intersection, section } = this.pick_terrain(pointer_position, pick.drag_adjust);
|
||||
|
||||
if (intersection) {
|
||||
selection.entity.set_world_position(
|
||||
@ -284,7 +456,7 @@ export class QuestEntityControls implements Disposable {
|
||||
}
|
||||
|
||||
private stop_transforming = () => {
|
||||
if (this.moved_since_last_mouse_down && this.selected && this.pick) {
|
||||
if (this.moved_since_last_mouse_down && this.selected && this.pick && !this.pick.creating) {
|
||||
const entity = this.selected.entity;
|
||||
quest_editor_store.push_translate_entity_action(
|
||||
entity,
|
||||
@ -314,11 +486,8 @@ export class QuestEntityControls implements Disposable {
|
||||
}
|
||||
|
||||
const entity = (intersection.object.userData as EntityUserData).entity;
|
||||
// Vector that points from the grabbing point to the model's origin.
|
||||
const grab_offset = intersection.object.position.clone().sub(intersection.point);
|
||||
// Vector that points from the grabbing point to the terrain point directly under the model's origin.
|
||||
const drag_adjust = grab_offset.clone();
|
||||
// Distance to terrain.
|
||||
let drag_y = 0;
|
||||
|
||||
// Find vertical distance to terrain.
|
||||
@ -334,6 +503,7 @@ export class QuestEntityControls implements Disposable {
|
||||
}
|
||||
|
||||
return {
|
||||
creating: false,
|
||||
mesh: intersection.object as Mesh,
|
||||
entity,
|
||||
initial_section: entity.section.val,
|
||||
@ -346,17 +516,17 @@ export class QuestEntityControls implements Disposable {
|
||||
|
||||
/**
|
||||
* @param pointer_pos - pointer coordinates in normalized device space
|
||||
* @param data - entity picking data
|
||||
* @param drag_adjust - vector from origin of entity to grabbing point
|
||||
*/
|
||||
private pick_terrain(
|
||||
pointer_pos: Vector2,
|
||||
data: Pick,
|
||||
drag_adjust: Vector3,
|
||||
): {
|
||||
intersection?: Intersection;
|
||||
section?: SectionModel;
|
||||
} {
|
||||
this.raycaster.setFromCamera(pointer_pos, this.renderer.camera);
|
||||
this.raycaster.ray.origin.add(data.drag_adjust);
|
||||
this.raycaster.ray.origin.add(drag_adjust);
|
||||
const intersections = this.raycaster.intersectObjects(
|
||||
this.renderer.collision_geometry.children,
|
||||
true,
|
||||
|
@ -13,10 +13,16 @@ import { QuestEntityModel } from "../model/QuestEntityModel";
|
||||
import { Disposer } from "../../core/observable/Disposer";
|
||||
import { Disposable } from "../../core/observable/Disposable";
|
||||
import { AreaModel } from "../model/AreaModel";
|
||||
import { AreaVariantModel } from "../model/AreaVariantModel";
|
||||
import { area_store } from "../stores/AreaStore";
|
||||
import { create_npc_mesh, create_object_mesh } from "./conversion/entities";
|
||||
import { AreaUserData } from "./conversion/areas";
|
||||
import { quest_editor_store } from "../stores/QuestEditorStore";
|
||||
import {
|
||||
ListChangeType,
|
||||
ListPropertyChangeEvent,
|
||||
} from "../../core/observable/property/list/ListProperty";
|
||||
import { QuestNpcModel } from "../model/QuestNpcModel";
|
||||
import { QuestObjectModel } from "../model/QuestObjectModel";
|
||||
import { entity_type_to_string } from "../../core/data_formats/parsing/quest/entities";
|
||||
|
||||
const logger = Logger.get("quest_editor/rendering/QuestModelManager");
|
||||
|
||||
@ -25,122 +31,153 @@ const CAMERA_LOOK_AT = new Vector3(0, 0, 0);
|
||||
const DUMMY_OBJECT = new Object3D();
|
||||
|
||||
export class QuestModelManager implements Disposable {
|
||||
private quest?: QuestModel;
|
||||
private area?: AreaModel;
|
||||
private area_variant?: AreaVariantModel;
|
||||
private disposer = new Disposer();
|
||||
private readonly disposer = new Disposer();
|
||||
private readonly quest_disposer = this.disposer.add(new Disposer());
|
||||
private readonly area_model_manager: AreaModelManager;
|
||||
private readonly npc_model_manager: EntityModelManager;
|
||||
private readonly object_model_manager: EntityModelManager;
|
||||
|
||||
constructor(private renderer: QuestRenderer) {}
|
||||
constructor(private readonly renderer: QuestRenderer) {
|
||||
this.area_model_manager = new AreaModelManager(this.renderer);
|
||||
this.npc_model_manager = new EntityModelManager(this.renderer);
|
||||
this.object_model_manager = new EntityModelManager(this.renderer);
|
||||
|
||||
async load_models(quest?: QuestModel, area?: AreaModel): Promise<void> {
|
||||
let area_variant: AreaVariantModel | undefined;
|
||||
this.disposer.add_all(
|
||||
quest_editor_store.current_quest.observe(this.quest_or_area_changed),
|
||||
|
||||
if (quest && area) {
|
||||
area_variant =
|
||||
quest.area_variants.val.find(v => v.area.id === area.id) ||
|
||||
area_store.get_variant(quest.episode, area.id, 0);
|
||||
}
|
||||
|
||||
if (this.quest === quest && this.area_variant === area_variant) {
|
||||
return;
|
||||
}
|
||||
|
||||
this.quest = quest;
|
||||
this.area = area;
|
||||
this.area_variant = area_variant;
|
||||
|
||||
this.disposer.dispose_all();
|
||||
|
||||
if (quest && area) {
|
||||
try {
|
||||
// Load necessary area geometry.
|
||||
const episode = quest.episode;
|
||||
const area_id = area.id;
|
||||
const variant_id = area_variant ? area_variant.id : 0;
|
||||
|
||||
const collision_geometry = await load_area_collision_geometry(
|
||||
episode,
|
||||
area_id,
|
||||
variant_id,
|
||||
);
|
||||
|
||||
const render_geometry = await load_area_render_geometry(
|
||||
episode,
|
||||
area_id,
|
||||
variant_id,
|
||||
);
|
||||
|
||||
this.add_sections_to_collision_geometry(collision_geometry, render_geometry);
|
||||
|
||||
if (this.quest !== quest || this.area_variant !== area_variant) return;
|
||||
|
||||
this.renderer.collision_geometry = collision_geometry;
|
||||
this.renderer.render_geometry = render_geometry;
|
||||
|
||||
this.renderer.reset_camera(CAMERA_POSITION, CAMERA_LOOK_AT);
|
||||
|
||||
// Load entity models.
|
||||
this.renderer.reset_entity_models();
|
||||
|
||||
for (const npc of quest.npcs.val) {
|
||||
if (npc.area_id === area.id) {
|
||||
const npc_geom = await load_npc_geometry(npc.type);
|
||||
const npc_tex = await load_npc_textures(npc.type);
|
||||
|
||||
if (this.quest !== quest || this.area_variant !== area_variant) return;
|
||||
|
||||
const model = create_npc_mesh(npc, npc_geom, npc_tex);
|
||||
this.update_entity_geometry(npc, model);
|
||||
}
|
||||
}
|
||||
|
||||
for (const object of quest.objects.val) {
|
||||
if (object.area_id === area.id) {
|
||||
const object_geom = await load_object_geometry(object.type);
|
||||
const object_tex = await load_object_textures(object.type);
|
||||
|
||||
if (this.quest !== quest || this.area_variant !== area_variant) return;
|
||||
|
||||
const model = create_object_mesh(object, object_geom, object_tex);
|
||||
this.update_entity_geometry(object, model);
|
||||
}
|
||||
}
|
||||
} catch (e) {
|
||||
logger.error(`Couldn't load models for quest ${quest.id}, ${area.name}.`, e);
|
||||
this.renderer.collision_geometry = DUMMY_OBJECT;
|
||||
this.renderer.render_geometry = DUMMY_OBJECT;
|
||||
this.renderer.reset_entity_models();
|
||||
}
|
||||
} else {
|
||||
this.renderer.collision_geometry = DUMMY_OBJECT;
|
||||
this.renderer.render_geometry = DUMMY_OBJECT;
|
||||
this.renderer.reset_entity_models();
|
||||
}
|
||||
quest_editor_store.current_area.observe(this.quest_or_area_changed),
|
||||
);
|
||||
}
|
||||
|
||||
dispose(): void {
|
||||
this.disposer.dispose();
|
||||
}
|
||||
|
||||
private quest_or_area_changed = async () => {
|
||||
this.quest_disposer.dispose_all();
|
||||
this.npc_model_manager.remove_all();
|
||||
this.object_model_manager.remove_all();
|
||||
this.renderer.reset_entity_models();
|
||||
|
||||
const quest = quest_editor_store.current_quest.val;
|
||||
const area = quest_editor_store.current_area.val;
|
||||
|
||||
// Load area model.
|
||||
await this.area_model_manager.load(quest, area);
|
||||
|
||||
if (
|
||||
quest !== quest_editor_store.current_quest.val ||
|
||||
area !== quest_editor_store.current_area.val
|
||||
) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Load entity models.
|
||||
if (quest && area) {
|
||||
this.npc_model_manager.add(quest.npcs.val.filter(entity => entity.area_id === area.id));
|
||||
this.object_model_manager.add(
|
||||
quest.objects.val.filter(entity => entity.area_id === area.id),
|
||||
);
|
||||
this.quest_disposer.add_all(
|
||||
quest.npcs.observe_list(this.npcs_changed),
|
||||
quest.objects.observe_list(this.objects_changed),
|
||||
);
|
||||
}
|
||||
};
|
||||
|
||||
private npcs_changed = (change: ListPropertyChangeEvent<QuestNpcModel>): void => {
|
||||
const area = quest_editor_store.current_area.val;
|
||||
|
||||
if (change.type === ListChangeType.ListChange && area) {
|
||||
this.npc_model_manager.remove(change.removed);
|
||||
|
||||
this.npc_model_manager.add(
|
||||
change.inserted.filter(entity => entity.area_id === area.id),
|
||||
);
|
||||
}
|
||||
};
|
||||
|
||||
private objects_changed = (change: ListPropertyChangeEvent<QuestObjectModel>): void => {
|
||||
const area = quest_editor_store.current_area.val;
|
||||
|
||||
if (change.type === ListChangeType.ListChange && area) {
|
||||
this.object_model_manager.remove(change.removed);
|
||||
|
||||
this.object_model_manager.add(
|
||||
change.inserted.filter(entity => entity.area_id === area.id),
|
||||
);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
class AreaModelManager {
|
||||
private readonly raycaster = new Raycaster();
|
||||
private readonly origin = new Vector3();
|
||||
private readonly down = new Vector3(0, -1, 0);
|
||||
private readonly up = new Vector3(0, 1, 0);
|
||||
private quest?: QuestModel;
|
||||
private area?: AreaModel;
|
||||
|
||||
constructor(private readonly renderer: QuestRenderer) {}
|
||||
|
||||
async load(quest?: QuestModel, area?: AreaModel): Promise<void> {
|
||||
this.quest = quest;
|
||||
this.area = area;
|
||||
|
||||
if (!quest || !area) {
|
||||
this.renderer.collision_geometry = DUMMY_OBJECT;
|
||||
this.renderer.render_geometry = DUMMY_OBJECT;
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
const area_variant =
|
||||
quest.area_variants.val.find(v => v.area.id === area.id) || area.area_variants[0];
|
||||
|
||||
// Load necessary area geometry.
|
||||
const episode = quest.episode;
|
||||
|
||||
const collision_geometry = await load_area_collision_geometry(episode, area_variant);
|
||||
if (this.should_cancel(quest, area)) return;
|
||||
|
||||
const render_geometry = await load_area_render_geometry(episode, area_variant);
|
||||
if (this.should_cancel(quest, area)) return;
|
||||
|
||||
this.add_sections_to_collision_geometry(collision_geometry, render_geometry);
|
||||
|
||||
this.renderer.collision_geometry = collision_geometry;
|
||||
this.renderer.render_geometry = render_geometry;
|
||||
|
||||
this.renderer.reset_camera(CAMERA_POSITION, CAMERA_LOOK_AT);
|
||||
} catch (e) {
|
||||
logger.error(`Couldn't load area models for quest ${quest.id}, ${area.name}.`, e);
|
||||
|
||||
this.renderer.collision_geometry = DUMMY_OBJECT;
|
||||
this.renderer.render_geometry = DUMMY_OBJECT;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Ensures that {@link load} is reentrant.
|
||||
*/
|
||||
private should_cancel(quest: QuestModel, area: AreaModel): boolean {
|
||||
return this.quest !== quest || this.area !== area;
|
||||
}
|
||||
|
||||
private add_sections_to_collision_geometry(
|
||||
collision_geom: Object3D,
|
||||
render_geom: Object3D,
|
||||
): void {
|
||||
const raycaster = new Raycaster();
|
||||
const origin = new Vector3();
|
||||
const down = new Vector3(0, -1, 0);
|
||||
const up = new Vector3(0, 1, 0);
|
||||
|
||||
for (const collision_area of collision_geom.children) {
|
||||
(collision_area as Mesh).geometry.boundingBox.getCenter(origin);
|
||||
(collision_area as Mesh).geometry.boundingBox.getCenter(this.origin);
|
||||
|
||||
raycaster.set(origin, down);
|
||||
const intersection1 = raycaster
|
||||
this.raycaster.set(this.origin, this.down);
|
||||
const intersection1 = this.raycaster
|
||||
.intersectObject(render_geom, true)
|
||||
.find(i => (i.object.userData as AreaUserData).section != undefined);
|
||||
|
||||
raycaster.set(origin, up);
|
||||
const intersection2 = raycaster
|
||||
this.raycaster.set(this.origin, this.up);
|
||||
const intersection2 = this.raycaster
|
||||
.intersectObject(render_geom, true)
|
||||
.find(i => (i.object.userData as AreaUserData).section != undefined);
|
||||
|
||||
@ -162,20 +199,114 @@ export class QuestModelManager implements Disposable {
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class EntityModelManager {
|
||||
private readonly queue: QuestEntityModel[] = [];
|
||||
private readonly loaded_entities: {
|
||||
entity: QuestEntityModel;
|
||||
disposer: Disposer;
|
||||
}[] = [];
|
||||
private loading = false;
|
||||
|
||||
constructor(private readonly renderer: QuestRenderer) {}
|
||||
|
||||
async add(entities: QuestEntityModel[]): Promise<void> {
|
||||
this.queue.push(...entities);
|
||||
|
||||
if (!this.loading) {
|
||||
try {
|
||||
this.loading = true;
|
||||
|
||||
while (this.queue.length) {
|
||||
const entity = this.queue[0];
|
||||
|
||||
try {
|
||||
await this.load(entity);
|
||||
} catch (e) {
|
||||
logger.error(
|
||||
`Couldn't load model for entity ${entity_type_to_string(entity.type)}.`,
|
||||
e,
|
||||
);
|
||||
} finally {
|
||||
const index = this.queue.indexOf(entity);
|
||||
|
||||
if (index !== -1) {
|
||||
this.queue.splice(index, 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
} finally {
|
||||
this.loading = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
remove(entities: QuestEntityModel[]): void {
|
||||
for (const entity of entities) {
|
||||
const queue_index = this.queue.indexOf(entity);
|
||||
|
||||
if (queue_index !== -1) {
|
||||
this.queue.splice(queue_index, 1);
|
||||
}
|
||||
|
||||
const loaded_index = this.loaded_entities.findIndex(loaded => loaded.entity === entity);
|
||||
|
||||
if (loaded_index !== -1) {
|
||||
const loaded = this.loaded_entities.splice(loaded_index, 1)[0];
|
||||
|
||||
this.renderer.remove_entity_model(loaded.entity);
|
||||
loaded.disposer.dispose();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
remove_all(): void {
|
||||
for (const { disposer } of this.loaded_entities) {
|
||||
disposer.dispose();
|
||||
}
|
||||
|
||||
this.loaded_entities.splice(0, Infinity);
|
||||
this.queue.splice(0, Infinity);
|
||||
}
|
||||
|
||||
private async load(entity: QuestEntityModel): Promise<void> {
|
||||
let model: Mesh;
|
||||
|
||||
if (entity instanceof QuestNpcModel) {
|
||||
const npc_geom = await load_npc_geometry(entity.type);
|
||||
const npc_tex = await load_npc_textures(entity.type);
|
||||
model = create_npc_mesh(entity, npc_geom, npc_tex);
|
||||
} else if (entity instanceof QuestObjectModel) {
|
||||
const object_geom = await load_object_geometry(entity.type);
|
||||
const object_tex = await load_object_textures(entity.type);
|
||||
model = create_object_mesh(entity, object_geom, object_tex);
|
||||
} else {
|
||||
throw new Error(`Unknown entity type ${entity.type}.`);
|
||||
}
|
||||
|
||||
// The model load might be cancelled by now.
|
||||
if (this.queue.includes(entity)) {
|
||||
this.update_entity_geometry(entity, model);
|
||||
}
|
||||
}
|
||||
|
||||
private update_entity_geometry(entity: QuestEntityModel, model: Mesh): void {
|
||||
this.renderer.add_entity_model(model);
|
||||
|
||||
this.disposer.add_all(
|
||||
entity.world_position.observe(({ value: { x, y, z } }) => {
|
||||
model.position.set(x, y, z);
|
||||
this.renderer.schedule_render();
|
||||
}),
|
||||
this.loaded_entities.push({
|
||||
entity,
|
||||
disposer: new Disposer(
|
||||
entity.world_position.observe(({ value: { x, y, z } }) => {
|
||||
model.position.set(x, y, z);
|
||||
this.renderer.schedule_render();
|
||||
}),
|
||||
|
||||
entity.rotation.observe(({ value: { x, y, z } }) => {
|
||||
model.rotation.set(x, y, z);
|
||||
this.renderer.schedule_render();
|
||||
}),
|
||||
);
|
||||
entity.rotation.observe(({ value: { x, y, z } }) => {
|
||||
model.rotation.set(x, y, z);
|
||||
this.renderer.schedule_render();
|
||||
}),
|
||||
),
|
||||
});
|
||||
}
|
||||
}
|
||||
|
@ -34,10 +34,6 @@ export class QuestRenderer extends Renderer {
|
||||
|
||||
private _render_geometry = new Object3D();
|
||||
|
||||
get render_geometry(): Object3D {
|
||||
return this._render_geometry;
|
||||
}
|
||||
|
||||
set render_geometry(render_geometry: Object3D) {
|
||||
this.scene.remove(this._render_geometry);
|
||||
this._render_geometry = render_geometry;
|
||||
@ -54,7 +50,6 @@ export class QuestRenderer extends Renderer {
|
||||
private readonly disposer = new Disposer();
|
||||
private readonly perspective_camera: PerspectiveCamera;
|
||||
private readonly entity_to_mesh = new Map<QuestEntityModel, Mesh>();
|
||||
private readonly model_manager = this.disposer.add(new QuestModelManager(this));
|
||||
private readonly entity_controls = this.disposer.add(new QuestEntityControls(this));
|
||||
|
||||
constructor() {
|
||||
@ -63,14 +58,10 @@ export class QuestRenderer extends Renderer {
|
||||
this.perspective_camera = this.camera as PerspectiveCamera;
|
||||
|
||||
this.disposer.add_all(
|
||||
quest_editor_store.current_quest.observe(this.load_models),
|
||||
quest_editor_store.current_area.observe(this.load_models),
|
||||
new QuestModelManager(this),
|
||||
|
||||
quest_editor_store.debug.observe(({ value }) => (this.debug = value)),
|
||||
);
|
||||
|
||||
this.dom_element.addEventListener("mousedown", this.entity_controls.on_mouse_down);
|
||||
this.dom_element.addEventListener("mouseup", this.entity_controls.on_mouse_up);
|
||||
this.dom_element.addEventListener("mousemove", this.entity_controls.on_mouse_move);
|
||||
}
|
||||
|
||||
dispose(): void {
|
||||
@ -89,6 +80,7 @@ export class QuestRenderer extends Renderer {
|
||||
this._entity_models = new Group();
|
||||
this.scene.add(this._entity_models);
|
||||
this.entity_to_mesh.clear();
|
||||
this.schedule_render();
|
||||
}
|
||||
|
||||
add_entity_model(model: Mesh): void {
|
||||
@ -103,14 +95,16 @@ export class QuestRenderer extends Renderer {
|
||||
this.schedule_render();
|
||||
}
|
||||
|
||||
remove_entity_model(entity: QuestEntityModel): void {
|
||||
const mesh = this.entity_to_mesh.get(entity);
|
||||
|
||||
if (mesh) {
|
||||
this._entity_models.remove(mesh);
|
||||
this.schedule_render();
|
||||
}
|
||||
}
|
||||
|
||||
get_entity_mesh(entity: QuestEntityModel): Mesh | undefined {
|
||||
return this.entity_to_mesh.get(entity);
|
||||
}
|
||||
|
||||
private load_models = () => {
|
||||
this.model_manager.load_models(
|
||||
quest_editor_store.current_quest.val,
|
||||
quest_editor_store.current_area.val,
|
||||
);
|
||||
};
|
||||
}
|
||||
|
@ -15,6 +15,7 @@ import { RenderObject } from "../../../core/data_formats/parsing/area_geometry";
|
||||
import { GeometryBuilder } from "../../../core/rendering/conversion/GeometryBuilder";
|
||||
import { ninja_object_to_geometry_builder } from "../../../core/rendering/conversion/ninja_geometry";
|
||||
import { SectionModel } from "../../model/SectionModel";
|
||||
import { AreaVariantModel } from "../../model/AreaVariantModel";
|
||||
|
||||
const materials = [
|
||||
// Wall
|
||||
@ -116,6 +117,7 @@ export function area_collision_geometry_to_object_3d(object: CollisionObject): O
|
||||
|
||||
export function area_geometry_to_sections_and_object_3d(
|
||||
object: RenderObject,
|
||||
area_variant: AreaVariantModel,
|
||||
): [SectionModel[], Object3D] {
|
||||
const sections: SectionModel[] = [];
|
||||
const group = new Group();
|
||||
@ -144,7 +146,12 @@ export function area_geometry_to_sections_and_object_3d(
|
||||
mesh.updateMatrixWorld();
|
||||
|
||||
if (section.id >= 0) {
|
||||
const sec = new SectionModel(section.id, section.position, section.rotation.y);
|
||||
const sec = new SectionModel(
|
||||
section.id,
|
||||
section.position,
|
||||
section.rotation.y,
|
||||
area_variant,
|
||||
);
|
||||
sections.push(sec);
|
||||
(mesh.userData as AreaUserData).section = sec;
|
||||
}
|
||||
|
@ -46,12 +46,8 @@ class AreaStore {
|
||||
return area_variant;
|
||||
};
|
||||
|
||||
get_area_sections = (
|
||||
episode: Episode,
|
||||
area_id: number,
|
||||
variant_id: number,
|
||||
): Promise<SectionModel[]> => {
|
||||
return load_area_sections(episode, area_id, variant_id);
|
||||
get_area_sections = (episode: Episode, variant: AreaVariantModel): Promise<SectionModel[]> => {
|
||||
return load_area_sections(episode, variant);
|
||||
};
|
||||
}
|
||||
|
||||
|
@ -29,26 +29,21 @@ import Logger = require("js-logger");
|
||||
const logger = Logger.get("quest_editor/gui/QuestEditorStore");
|
||||
|
||||
export class QuestEditorStore implements Disposable {
|
||||
readonly debug: WritableProperty<boolean> = property(false);
|
||||
readonly undo = new UndoStack();
|
||||
readonly current_quest_filename: Property<string | undefined>;
|
||||
readonly current_quest: Property<QuestModel | undefined>;
|
||||
readonly current_area: Property<AreaModel | undefined>;
|
||||
readonly selected_entity: Property<QuestEntityModel | undefined>;
|
||||
|
||||
private readonly disposer = new Disposer();
|
||||
private readonly _current_quest_filename = property<string | undefined>(undefined);
|
||||
private readonly _current_quest = property<QuestModel | undefined>(undefined);
|
||||
private readonly _current_area = property<AreaModel | undefined>(undefined);
|
||||
private readonly _selected_entity = property<QuestEntityModel | undefined>(undefined);
|
||||
|
||||
constructor() {
|
||||
this.current_quest_filename = this._current_quest_filename;
|
||||
this.current_quest = this._current_quest;
|
||||
this.current_area = this._current_area;
|
||||
this.selected_entity = this._selected_entity;
|
||||
readonly debug: WritableProperty<boolean> = property(false);
|
||||
readonly undo = new UndoStack();
|
||||
readonly current_quest_filename: Property<string | undefined> = this._current_quest_filename;
|
||||
readonly current_quest: Property<QuestModel | undefined> = this._current_quest;
|
||||
readonly current_area: Property<AreaModel | undefined> = this._current_area;
|
||||
readonly selected_entity: Property<QuestEntityModel | undefined> = this._selected_entity;
|
||||
|
||||
this.disposer.add(
|
||||
constructor() {
|
||||
this.disposer.add_all(
|
||||
gui_store.tool.observe(
|
||||
({ value: tool }) => {
|
||||
if (tool === GuiTool.QuestEditor) {
|
||||
@ -57,6 +52,26 @@ export class QuestEditorStore implements Disposable {
|
||||
},
|
||||
{ call_now: true },
|
||||
),
|
||||
|
||||
this.current_quest
|
||||
.flat_map(quest => (quest ? quest.npcs : property([])))
|
||||
.observe(({ value: npcs }) => {
|
||||
const selected = this.selected_entity.val;
|
||||
|
||||
if (selected instanceof QuestNpcModel && !npcs.includes(selected)) {
|
||||
this.set_selected_entity(undefined);
|
||||
}
|
||||
}),
|
||||
|
||||
this.current_quest
|
||||
.flat_map(quest => (quest ? quest.objects : property([])))
|
||||
.observe(({ value: objects }) => {
|
||||
const selected = this.selected_entity.val;
|
||||
|
||||
if (selected instanceof QuestObjectModel && !objects.includes(selected)) {
|
||||
this.set_selected_entity(undefined);
|
||||
}
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
@ -121,7 +136,7 @@ export class QuestEditorStore implements Disposable {
|
||||
npc.pso_type_id,
|
||||
npc.npc_id,
|
||||
npc.script_label,
|
||||
npc.roaming,
|
||||
npc.pso_roaming,
|
||||
npc.area_id,
|
||||
npc.section_id,
|
||||
npc.position,
|
||||
@ -145,7 +160,14 @@ export class QuestEditorStore implements Disposable {
|
||||
const quest = this.current_quest.val;
|
||||
if (!quest) return;
|
||||
|
||||
let file_name = prompt("File name:");
|
||||
let default_file_name = this.current_quest_filename.val;
|
||||
|
||||
if (default_file_name) {
|
||||
const ext_start = default_file_name.lastIndexOf(".");
|
||||
if (ext_start !== -1) default_file_name = default_file_name.slice(0, ext_start);
|
||||
}
|
||||
|
||||
let file_name = prompt("File name:", default_file_name);
|
||||
if (!file_name) return;
|
||||
|
||||
const buffer = write_quest_qst(
|
||||
@ -178,7 +200,7 @@ export class QuestEditorStore implements Disposable {
|
||||
pso_type_id: npc.pso_type_id,
|
||||
npc_id: npc.npc_id,
|
||||
script_label: npc.script_label,
|
||||
roaming: npc.roaming,
|
||||
pso_roaming: npc.pso_roaming,
|
||||
})),
|
||||
dat_unknowns: quest.dat_unknowns,
|
||||
object_code: quest.object_code,
|
||||
@ -261,11 +283,7 @@ export class QuestEditorStore implements Disposable {
|
||||
|
||||
// Load section data.
|
||||
for (const variant of quest.area_variants.val) {
|
||||
const sections = await area_store.get_area_sections(
|
||||
quest.episode,
|
||||
variant.area.id,
|
||||
variant.id,
|
||||
);
|
||||
const sections = await area_store.get_area_sections(quest.episode, variant);
|
||||
variant.sections.val.splice(0, Infinity, ...sections);
|
||||
|
||||
for (const object of quest.objects.val.filter(o => o.area_id === variant.area.id)) {
|
||||
|
Loading…
Reference in New Issue
Block a user