NPCs in the NPC list are now filtered according to the selected area. Objects can now be added via drag and drop from the object list.

This commit is contained in:
Daan Vanden Bosch 2019-09-22 15:18:46 +02:00
parent f0d474ad40
commit f40b1fb168
8 changed files with 331 additions and 39 deletions

View File

@ -27,7 +27,7 @@ export function get_area_variant(
const variant = area.area_variants[variant_id];
if (!variant) throw new Error(`No area variant with id ${variant_id}.`);
return variant;
return Object.freeze(variant);
}
const AREAS: { [episode: number]: Area[] } = [];
@ -39,7 +39,7 @@ function create_area(id: number, name: string, order: number, variants: number):
area.area_variants.push({ id, area });
}
return area;
return Object.freeze(area);
}
// The IDs match the PSO IDs for areas.

File diff suppressed because it is too large Load Diff

View File

@ -8,6 +8,7 @@
grid-column-gap: 6px;
grid-row-gap: 6px;
justify-content: center;
margin: 6px;
}
.quest_editor_EntityListView_entity {

View File

@ -2,14 +2,17 @@ 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 { entity_dnd_source } from "./entity_dnd";
import { render_entity_to_image } from "../rendering/render_entity_to_image";
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
import { list_property } from "../../core/observable";
export abstract class EntityListView<T extends EntityType> extends ResizableWidget {
readonly element: HTMLElement;
protected constructor(private readonly class_name: string, entities: ListProperty<T>) {
protected readonly entities: WritableListProperty<T> = list_property();
protected constructor(class_name: string) {
super();
const list_element = el.div({ class: "quest_editor_EntityListView_entity_list" });
@ -17,7 +20,7 @@ export abstract class EntityListView<T extends EntityType> extends ResizableWidg
this.element = el.div({ class: `${class_name} quest_editor_EntityListView` }, list_element);
this.disposables(
bind_children_to(list_element, entities, this.create_entity_element),
bind_children_to(list_element, this.entities, this.create_entity_element),
entity_dnd_source(list_element, target => {
let element: HTMLElement | null = target;
@ -28,7 +31,7 @@ export abstract class EntityListView<T extends EntityType> extends ResizableWidg
if (index != undefined) {
return [
element.querySelector("img")!.cloneNode(true) as HTMLElement,
entities.get(parseInt(index, 10)),
this.entities.get(parseInt(index, 10)),
];
}

View File

@ -1,11 +1,34 @@
import { EntityListView } from "./EntityListView";
import { NPC_TYPES, NpcType } from "../../core/data_formats/parsing/quest/npc_types";
import { list_property } from "../../core/observable";
import { npc_data, NPC_TYPES, NpcType } from "../../core/data_formats/parsing/quest/npc_types";
import { quest_editor_store } from "../stores/QuestEditorStore";
import { Episode } from "../../core/data_formats/parsing/quest/Episode";
export class NpcListView extends EntityListView<NpcType> {
constructor() {
super("quest_editor_NpcListView", list_property(undefined, ...NPC_TYPES));
super("quest_editor_NpcListView");
this.disposables(
quest_editor_store.current_quest.observe(this.filter_npcs),
quest_editor_store.current_area.observe(this.filter_npcs),
);
this.filter_npcs();
this.finalize_construction(NpcListView.prototype);
}
private filter_npcs = (): void => {
const quest = quest_editor_store.current_quest.val;
const area = quest_editor_store.current_area.val;
const episode = quest ? quest.episode : Episode.I;
const area_id = area ? area.id : 0;
this.entities.val = NPC_TYPES.filter(npc => {
const data = npc_data(npc);
return (
(data.episode == undefined || data.episode === episode) &&
data.area_ids.includes(area_id)
);
});
};
}

View File

@ -0,0 +1,12 @@
import { EntityListView } from "./EntityListView";
import { OBJECT_TYPES, ObjectType } from "../../core/data_formats/parsing/quest/object_types";
export class ObjectListView extends EntityListView<ObjectType> {
constructor() {
super("quest_editor_ObjectListView");
this.entities.val = OBJECT_TYPES;
this.finalize_construction(ObjectListView.prototype);
}
}

View File

@ -13,6 +13,7 @@ import { EntityInfoView } from "./EntityInfoView";
import { gui_store, GuiTool } from "../../core/stores/GuiStore";
import { quest_editor_store } from "../stores/QuestEditorStore";
import { NpcListView } from "./NpcListView";
import { ObjectListView } from "./ObjectListView";
import Logger = require("js-logger");
const logger = Logger.get("quest_editor/gui/QuestEditorView");
@ -25,6 +26,7 @@ const VIEW_TO_NAME = new Map<new () => ResizableWidget, string>([
[AsmEditorView, "asm_editor"],
[EntityInfoView, "entity_info"],
[NpcListView, "npc_list_view"],
[ObjectListView, "object_list_view"],
]);
const DEFAULT_LAYOUT_CONFIG = {
@ -99,6 +101,12 @@ const DEFAULT_LAYOUT_CONTENT: ItemConfigType[] = [
componentName: VIEW_TO_NAME.get(NpcListView),
isClosable: false,
},
{
title: "Objects",
type: "component",
componentName: VIEW_TO_NAME.get(ObjectListView),
isClosable: false,
},
],
},
],

View File

@ -17,6 +17,7 @@ import {
remove_entity_dnd_listener,
} from "../gui/entity_dnd";
import { vec3_to_threejs } from "../../core/rendering/conversion";
import { QuestObjectModel } from "../model/QuestObjectModel";
const UP_VECTOR = Object.freeze(new Vector3(0, 1, 0));
const DOWN_VECTOR = Object.freeze(new Vector3(0, -1, 0));
@ -198,18 +199,14 @@ export class QuestEntityControls implements Disposable {
if (!area || !quest) return;
let entity: QuestEntityModel;
if (is_npc_type(e.entity_type)) {
const data = npc_data(e.entity_type);
if (data.pso_type_id == undefined || data.pso_roaming == undefined) return;
e.drag_element.style.display = "none";
if (e.event.dataTransfer) {
e.event.dataTransfer.dropEffect = "copy";
}
const npc = new QuestNpcModel(
entity = new QuestNpcModel(
e.entity_type,
data.pso_type_id,
0,
@ -223,21 +220,42 @@ export class QuestEntityControls implements Disposable {
// 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]],
);
const grab_offset = new Vector3(0, 0, 0);
const drag_adjust = new Vector3(0, 0, 0);
this.translate_entity_horizontally(npc, grab_offset, drag_adjust);
quest.add_npc(npc);
quest_editor_store.set_selected_entity(npc);
this.pick = {
mode: PickMode.Creating,
initial_section: npc.section.val,
initial_position: npc.world_position.val,
grab_offset,
drag_adjust,
};
} else {
entity = new QuestObjectModel(
e.entity_type,
0,
0,
area.id,
0,
new Vec3(0, 0, 0),
new Vec3(0, 0, 0),
// TODO: which default properties?
new Map(),
// TODO: do the following values make sense?
[[0, 0, 0, 0, 0, 0], [0, 0]],
);
}
e.drag_element.style.display = "none";
if (e.event.dataTransfer) {
e.event.dataTransfer.dropEffect = "copy";
}
const grab_offset = new Vector3(0, 0, 0);
const drag_adjust = new Vector3(0, 0, 0);
this.translate_entity_horizontally(entity, grab_offset, drag_adjust);
quest.add_entity(entity);
quest_editor_store.set_selected_entity(entity);
this.pick = {
mode: PickMode.Creating,
initial_section: entity.section.val,
initial_position: entity.world_position.val,
grab_offset,
drag_adjust,
};
};
private dragover = (e: EntityDragEvent) => {