Entity-specific properties are now shown in the entity info view for supported entities.

This commit is contained in:
Daan Vanden Bosch 2020-09-23 21:37:20 +02:00
parent 9ea2faa826
commit 033cbf2436
17 changed files with 2294 additions and 889 deletions

View File

@ -155,7 +155,8 @@ Features that are in ***bold italics*** are planned but not yet implemented.
## Bugs ## Bugs
- [Load Quest](#load-quest): Can't parse quest 125 White Day - [Load Quest](#load-quest): Can't parse quest 125 White Day
- [Script Assembly Editor](#script-assembly-editor): Go to definition doesn't work in RT (#231) and PW2 (#234) - [Script Assembly Editor](#script-assembly-editor): Go to definition doesn't work in RT (#231), PW2
(#234) and Magnitude of Metal (#1, can jump to label 207 from line 433, but not from line 465)
- When a modal dialog is open, global keybindings should be disabled - When a modal dialog is open, global keybindings should be disabled
- Entities with rendering issues: - Entities with rendering issues:
- Caves 4 Button door - Caves 4 Button door
@ -174,3 +175,4 @@ Features that are in ***bold italics*** are planned but not yet implemented.
- Merissa A - Merissa A
- Merissa AA - Merissa AA
- All "Sonic" objects, even the ones that aren't actually Sonic, are rendered as Sonic - All "Sonic" objects, even the ones that aren't actually Sonic, are rendered as Sonic
- [Event Actions](#Event Actions): Editing event actions can't be undone

View File

@ -71,7 +71,7 @@ re-run whenever a file is changed. The testing framework used is Jest.
### Code Style, Linting and Formatting ### Code Style, Linting and Formatting
Class names are in `PascalCase` and all other identifiers are in `snake_case`. Class/interface/type names are in `PascalCase` and all other identifiers are in `snake_case`.
ESLint and Prettier are used for linting and formatting. Run with `yarn lint` and/or configure your ESLint and Prettier are used for linting and formatting. Run with `yarn lint` and/or configure your
editor to use the ESLint/Prettier configuration. editor to use the ESLint/Prettier configuration.
@ -84,4 +84,6 @@ Create an optimized production build with `yarn build`.
### prs-rs ### prs-rs
Provides faster PRS routines using WebAssembly. Build for WebPack with `yarn build_prs_rs_browser`. Build for Jest with `yarn build_prs_rs_testing`. Building requires [wasm-pack](https://github.com/rustwasm/wasm-pack). Provides faster PRS routines using WebAssembly. Build for WebPack with `yarn build_prs_rs_browser`.
Build for Jest with `yarn build_prs_rs_testing`. Building requires
[wasm-pack](https://github.com/rustwasm/wasm-pack).

View File

@ -1,8 +1,9 @@
import { Cursor } from "../block/cursor/Cursor"; import { Cursor } from "../block/cursor/Cursor";
import { Vec3 } from "../vector"; import { Vec3 } from "../vector";
import { ANGLE_TO_RAD, NjObject, parse_xj_object } from "./ninja"; import { NjObject, parse_xj_object } from "./ninja";
import { XjModel } from "./ninja/xj"; import { XjModel } from "./ninja/xj";
import { parse_rel } from "./rel"; import { parse_rel } from "./rel";
import { angle_to_rad } from "./ninja/angle";
export type RenderObject = { export type RenderObject = {
sections: RenderSection[]; sections: RenderSection[];
@ -34,9 +35,9 @@ export function parse_area_geometry(cursor: Cursor): RenderObject {
const section_id = cursor.i32(); const section_id = cursor.i32();
const section_position = cursor.vec3_f32(); const section_position = cursor.vec3_f32();
const section_rotation = { const section_rotation = {
x: cursor.u32() * ANGLE_TO_RAD, x: angle_to_rad(cursor.u32()),
y: cursor.u32() * ANGLE_TO_RAD, y: angle_to_rad(cursor.u32()),
z: cursor.u32() * ANGLE_TO_RAD, z: angle_to_rad(cursor.u32()),
}; };
cursor.seek(4); cursor.seek(4);

View File

@ -0,0 +1,10 @@
const ANGLE_TO_RAD = (2 * Math.PI) / 0x10000;
const RAD_TO_ANGLE = 0x10000 / (2 * Math.PI);
export function angle_to_rad(angle: number): number {
return angle * ANGLE_TO_RAD;
}
export function rad_to_angle(rad: number): number {
return Math.round(rad * RAD_TO_ANGLE);
}

View File

@ -4,8 +4,7 @@ import { parse_iff } from "../iff";
import { NjcmModel, parse_njcm_model } from "./njcm"; import { NjcmModel, parse_njcm_model } from "./njcm";
import { parse_xj_model, XjModel } from "./xj"; import { parse_xj_model, XjModel } from "./xj";
import { Result, success } from "../../../Result"; import { Result, success } from "../../../Result";
import { angle_to_rad } from "./angle";
export const ANGLE_TO_RAD = (2 * Math.PI) / 0xffff;
const NJCM = 0x4d434a4e; const NJCM = 0x4d434a4e;
@ -159,9 +158,9 @@ function parse_sibling_objects<M extends NjModel>(
const model_offset = cursor.u32(); const model_offset = cursor.u32();
const pos = cursor.vec3_f32(); const pos = cursor.vec3_f32();
const rotation = { const rotation = {
x: cursor.i32() * ANGLE_TO_RAD, x: angle_to_rad(cursor.i32()),
y: cursor.i32() * ANGLE_TO_RAD, y: angle_to_rad(cursor.i32()),
z: cursor.i32() * ANGLE_TO_RAD, z: angle_to_rad(cursor.i32()),
}; };
const scale = cursor.vec3_f32(); const scale = cursor.vec3_f32();
const child_offset = cursor.u32(); const child_offset = cursor.u32();

View File

@ -1,6 +1,6 @@
import { ANGLE_TO_RAD } from "./index";
import { Cursor } from "../../block/cursor/Cursor"; import { Cursor } from "../../block/cursor/Cursor";
import { Vec3 } from "../../vector"; import { Vec3 } from "../../vector";
import { angle_to_rad } from "./angle";
const NMDM = 0x4d444d4e; const NMDM = 0x4d444d4e;
@ -210,9 +210,9 @@ function parse_motion_data_a(
frames.push({ frames.push({
frame: cursor.u16(), frame: cursor.u16(),
value: { value: {
x: cursor.u16() * ANGLE_TO_RAD, x: angle_to_rad(cursor.u16()),
y: cursor.u16() * ANGLE_TO_RAD, y: angle_to_rad(cursor.u16()),
z: cursor.u16() * ANGLE_TO_RAD, z: angle_to_rad(cursor.u16()),
}, },
}); });
} }
@ -238,9 +238,9 @@ function parse_motion_data_a_wide(cursor: Cursor, keyframe_count: number): NjKey
frames.push({ frames.push({
frame: cursor.u32(), frame: cursor.u32(),
value: { value: {
x: cursor.i32() * ANGLE_TO_RAD, x: angle_to_rad(cursor.i32()),
y: cursor.i32() * ANGLE_TO_RAD, y: angle_to_rad(cursor.i32()),
z: cursor.i32() * ANGLE_TO_RAD, z: angle_to_rad(cursor.i32()),
}, },
}); });
} }

View File

@ -1,10 +1,13 @@
import { npc_data, NpcType, NpcTypeData } from "./npc_types"; import { npc_data, NpcType, NpcTypeData } from "./npc_types";
import { object_data, ObjectType, ObjectTypeData } from "./object_types"; import { object_data, ObjectType, ObjectTypeData } from "./object_types";
import { DatEvent, DatUnknown } from "./dat"; import { DatEvent, DatUnknown, NPC_BYTE_SIZE } from "./dat";
import { Episode } from "./Episode"; import { Episode } from "./Episode";
import { Segment } from "../../asm/instructions"; import { Segment } from "../../asm/instructions";
import { QuestNpc } from "./QuestNpc"; import { get_npc_type, QuestNpc } from "./QuestNpc";
import { QuestObject } from "./QuestObject"; import { get_object_type, QuestObject } from "./QuestObject";
import { EntityProp, EntityPropType } from "./properties";
import { angle_to_rad, rad_to_angle } from "../ninja/angle";
import { assert, require_finite, require_integer } from "../../../util";
export type Quest = { export type Quest = {
id: number; id: number;
@ -48,3 +51,72 @@ export function is_object_type(entity_type: EntityType): entity_type is ObjectTy
export function entity_data(type: EntityType): EntityTypeData { export function entity_data(type: EntityType): EntityTypeData {
return npc_data(type as NpcType) ?? object_data(type as ObjectType); return npc_data(type as NpcType) ?? object_data(type as ObjectType);
} }
export function get_entity_type(entity: QuestEntity): EntityType {
return entity.data.byteLength === NPC_BYTE_SIZE
? get_npc_type(entity as QuestNpc)
: get_object_type(entity as QuestObject);
}
export function get_entity_prop_value(entity: QuestEntity, prop: EntityProp): number {
switch (prop.type) {
case EntityPropType.U8:
return entity.view.getUint8(prop.offset);
case EntityPropType.U16:
return entity.view.getUint16(prop.offset, true);
case EntityPropType.U32:
return entity.view.getUint32(prop.offset, true);
case EntityPropType.I8:
return entity.view.getInt8(prop.offset);
case EntityPropType.I16:
return entity.view.getInt16(prop.offset, true);
case EntityPropType.I32:
return entity.view.getInt32(prop.offset, true);
case EntityPropType.F32:
return entity.view.getFloat32(prop.offset, true);
case EntityPropType.Angle:
return angle_to_rad(entity.view.getInt32(prop.offset, true));
}
}
export function set_entity_prop_value(entity: QuestEntity, prop: EntityProp, value: number): void {
switch (prop.type) {
case EntityPropType.U8:
require_in_bounds(value, 0, 0xff);
entity.view.setUint8(prop.offset, value);
break;
case EntityPropType.U16:
require_in_bounds(value, 0, 0xffff);
entity.view.setUint16(prop.offset, value, true);
break;
case EntityPropType.U32:
require_in_bounds(value, 0, 0xffffffff);
entity.view.setUint32(prop.offset, value, true);
break;
case EntityPropType.I8:
require_in_bounds(value, -0x80, 0x7f);
entity.view.setInt8(prop.offset, value);
break;
case EntityPropType.I16:
require_in_bounds(value, -0x8000, 0x7fff);
entity.view.setInt16(prop.offset, value, true);
break;
case EntityPropType.I32:
require_in_bounds(value, -0x80000000, 0x7fffffff);
entity.view.setInt32(prop.offset, value, true);
break;
case EntityPropType.F32:
entity.view.setFloat32(prop.offset, value, true);
break;
case EntityPropType.Angle:
require_finite(value, "value");
entity.view.setInt32(prop.offset, rad_to_angle(value), true);
break;
}
}
function require_in_bounds(value: unknown, min: number, max: number): asserts value is number {
require_integer(value, "value");
assert(value >= min, () => `value should be greater than or equal to ${min} but was ${value}.`);
assert(value <= max, () => `value should be less than or equal to ${max} but was ${value}.`);
}

View File

@ -3,6 +3,7 @@ import { Vec3 } from "../../vector";
import { Episode } from "./Episode"; import { Episode } from "./Episode";
import { NPC_BYTE_SIZE } from "./dat"; import { NPC_BYTE_SIZE } from "./dat";
import { assert } from "../../../util"; import { assert } from "../../../util";
import { angle_to_rad, rad_to_angle } from "../ninja/angle";
const DEFAULT_SCALE: Vec3 = Object.freeze({ x: 1, y: 1, z: 1 }); const DEFAULT_SCALE: Vec3 = Object.freeze({ x: 1, y: 1, z: 1 });
@ -103,16 +104,16 @@ export function set_npc_position(npc: QuestNpc, position: Vec3): void {
export function get_npc_rotation(npc: QuestNpc): Vec3 { export function get_npc_rotation(npc: QuestNpc): Vec3 {
return { return {
x: (npc.view.getInt32(32, true) / 0xffff) * 2 * Math.PI, x: angle_to_rad(npc.view.getInt32(32, true)),
y: (npc.view.getInt32(36, true) / 0xffff) * 2 * Math.PI, y: angle_to_rad(npc.view.getInt32(36, true)),
z: (npc.view.getInt32(40, true) / 0xffff) * 2 * Math.PI, z: angle_to_rad(npc.view.getInt32(40, true)),
}; };
} }
export function set_npc_rotation(npc: QuestNpc, rotation: Vec3): void { export function set_npc_rotation(npc: QuestNpc, rotation: Vec3): void {
npc.view.setInt32(32, Math.round((rotation.x / (2 * Math.PI)) * 0xffff), true); npc.view.setInt32(32, rad_to_angle(rotation.x), true);
npc.view.setInt32(36, Math.round((rotation.y / (2 * Math.PI)) * 0xffff), true); npc.view.setInt32(36, rad_to_angle(rotation.y), true);
npc.view.setInt32(40, Math.round((rotation.z / (2 * Math.PI)) * 0xffff), true); npc.view.setInt32(40, rad_to_angle(rotation.z), true);
} }
/** /**

View File

@ -2,6 +2,7 @@ import { id_to_object_type, object_data, ObjectType } from "./object_types";
import { Vec3 } from "../../vector"; import { Vec3 } from "../../vector";
import { OBJECT_BYTE_SIZE } from "./dat"; import { OBJECT_BYTE_SIZE } from "./dat";
import { assert } from "../../../util"; import { assert } from "../../../util";
import { angle_to_rad, rad_to_angle } from "../ninja/angle";
export type QuestObject = { export type QuestObject = {
area_id: number; area_id: number;
@ -82,20 +83,20 @@ export function set_object_position(object: QuestObject, position: Vec3): void {
export function get_object_rotation(object: QuestObject): Vec3 { export function get_object_rotation(object: QuestObject): Vec3 {
return { return {
x: (object.view.getInt32(28, true) / 0xffff) * 2 * Math.PI, x: angle_to_rad(object.view.getInt32(28, true)),
y: (object.view.getInt32(32, true) / 0xffff) * 2 * Math.PI, y: angle_to_rad(object.view.getInt32(32, true)),
z: (object.view.getInt32(36, true) / 0xffff) * 2 * Math.PI, z: angle_to_rad(object.view.getInt32(36, true)),
}; };
} }
export function set_object_rotation(object: QuestObject, rotation: Vec3): void { export function set_object_rotation(object: QuestObject, rotation: Vec3): void {
object.view.setInt32(28, Math.round((rotation.x / (2 * Math.PI)) * 0xffff), true); object.view.setInt32(28, rad_to_angle(rotation.x), true);
object.view.setInt32(32, Math.round((rotation.y / (2 * Math.PI)) * 0xffff), true); object.view.setInt32(32, rad_to_angle(rotation.y), true);
object.view.setInt32(36, Math.round((rotation.z / (2 * Math.PI)) * 0xffff), true); object.view.setInt32(36, rad_to_angle(rotation.z), true);
} }
// //
// Complex properties that use multiple parts of the data block and possible other properties. // Complex properties that use multiple parts of the data block and possibly other properties.
// //
export function get_object_type(object: QuestObject): ObjectType { export function get_object_type(object: QuestObject): ObjectType {

View File

@ -1,4 +1,5 @@
import { check_episode, Episode } from "./Episode"; import { check_episode, Episode } from "./Episode";
import { EntityProp } from "./properties";
// Make sure ObjectType does not overlap NpcType. // Make sure ObjectType does not overlap NpcType.
export enum NpcType { export enum NpcType {
@ -222,6 +223,10 @@ export type NpcTypeData = {
* Slime). * Slime).
*/ */
readonly regular?: boolean; readonly regular?: boolean;
/**
* Default object-specific properties.
*/
readonly properties: readonly EntityProp[];
}; };
export const NPC_TYPES: NpcType[] = []; export const NPC_TYPES: NpcType[] = [];
@ -279,6 +284,7 @@ function define_npc_type_data(
type_id, type_id,
skin, skin,
regular, regular,
properties: [],
}); });
if (episode) { if (episode) {

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,23 @@
/**
* Represents a configurable property for accessing parts of entity data of which the use is not
* fully understood or ambiguous.
*/
export type EntityProp = {
readonly name: string;
readonly offset: number;
readonly type: EntityPropType;
};
export enum EntityPropType {
U8,
U16,
U32,
I8,
I16,
I32,
F32,
/**
* Signed 32-bit integer that represents an angle. 0x10000 is 360°.
*/
Angle,
}

View File

@ -104,17 +104,24 @@ export function defined<T>(value: T | undefined | null, name: string): asserts v
assert(value != undefined, () => `${name} should not be null or undefined (was ${value}).`); assert(value != undefined, () => `${name} should not be null or undefined (was ${value}).`);
} }
export function require_finite(value: number, name: string): void { export function require_finite(value: unknown, name: string): asserts value is number {
assert(Number.isFinite(value), () => `${name} should be a finite number (was ${value}).`); assert(Number.isFinite(value), () => `${name} should be a finite number (was ${value}).`);
} }
export function require_integer(value: number, name: string): void { export function require_number(value: unknown, name: string): asserts value is number {
assert(typeof value === "number", () => `${name} should be a number (was ${value}).`);
}
export function require_integer(value: unknown, name: string): asserts value is number {
assert(Number.isInteger(value), () => `${name} should be an integer (was ${value}).`); assert(Number.isInteger(value), () => `${name} should be an integer (was ${value}).`);
} }
export function require_non_negative_integer(value: number, name: string): void { export function require_non_negative_integer(
value: unknown,
name: string,
): asserts value is number {
assert( assert(
Number.isInteger(value) && value >= 0, Number.isInteger(value) && (value as any) >= 0,
() => `${name} should be a non-negative integer (was ${value}).`, () => `${name} should be a non-negative integer (was ${value}).`,
); );
} }

View File

@ -2,13 +2,15 @@ import { Controller } from "../../core/controllers/Controller";
import { QuestEditorStore } from "../stores/QuestEditorStore"; import { QuestEditorStore } from "../stores/QuestEditorStore";
import { Property } from "../../core/observable/property/Property"; import { Property } from "../../core/observable/property/Property";
import { QuestNpcModel } from "../model/QuestNpcModel"; import { QuestNpcModel } from "../model/QuestNpcModel";
import { property } from "../../core/observable"; import { flat_map_to_list, list_property, property } from "../../core/observable";
import { Euler, Vector3 } from "three"; import { Euler, Vector3 } from "three";
import { deg_to_rad } from "../../core/math"; import { deg_to_rad } from "../../core/math";
import { TranslateEntityAction } from "../actions/TranslateEntityAction"; import { TranslateEntityAction } from "../actions/TranslateEntityAction";
import { RotateEntityAction } from "../actions/RotateEntityAction"; import { RotateEntityAction } from "../actions/RotateEntityAction";
import { euler } from "../model/euler"; import { euler } from "../model/euler";
import { entity_data } from "../../core/data_formats/parsing/quest/Quest"; import { entity_data } from "../../core/data_formats/parsing/quest/Quest";
import { ListProperty } from "../../core/observable/property/list/ListProperty";
import { QuestEntityPropModel } from "../model/QuestEntityPropModel";
const DUMMY_VECTOR = Object.freeze(new Vector3()); const DUMMY_VECTOR = Object.freeze(new Vector3());
const DUMMY_EULER = Object.freeze(new Euler()); const DUMMY_EULER = Object.freeze(new Euler());
@ -23,6 +25,7 @@ export class EntityInfoController extends Controller {
readonly wave_hidden: Property<boolean>; readonly wave_hidden: Property<boolean>;
readonly position: Property<Vector3>; readonly position: Property<Vector3>;
readonly rotation: Property<Euler>; readonly rotation: Property<Euler>;
readonly props: ListProperty<QuestEntityPropModel>;
constructor(private readonly store: QuestEditorStore) { constructor(private readonly store: QuestEditorStore) {
super(); super();
@ -44,6 +47,7 @@ export class EntityInfoController extends Controller {
this.wave_hidden = entity.map(e => !(e instanceof QuestNpcModel)); this.wave_hidden = entity.map(e => !(e instanceof QuestNpcModel));
this.position = entity.flat_map(e => e?.position ?? property(DUMMY_VECTOR)); this.position = entity.flat_map(e => e?.position ?? property(DUMMY_VECTOR));
this.rotation = entity.flat_map(e => e?.rotation ?? property(DUMMY_EULER)); this.rotation = entity.flat_map(e => e?.rotation ?? property(DUMMY_EULER));
this.props = flat_map_to_list(e => e?.props ?? list_property(), entity);
} }
focused = (): void => { focused = (): void => {

View File

@ -1,17 +1,22 @@
import { bind_attr, div, table, td, th, tr } from "../../core/gui/dom"; import { bind_attr, bind_children_to, div, table, td, th, tr } from "../../core/gui/dom";
import { UnavailableView } from "./UnavailableView"; import { UnavailableView } from "./UnavailableView";
import "./EntityInfoView.css"; import "./EntityInfoView.css";
import { NumberInput } from "../../core/gui/NumberInput"; import { NumberInput } from "../../core/gui/NumberInput";
import { rad_to_deg } from "../../core/math"; import { rad_to_deg } from "../../core/math";
import { EntityInfoController } from "../controllers/EntityInfoController"; import { EntityInfoController } from "../controllers/EntityInfoController";
import { ResizableView } from "../../core/gui/ResizableView"; import { ResizableView } from "../../core/gui/ResizableView";
import { QuestEntityPropModel } from "../model/QuestEntityPropModel";
import { Disposable } from "../../core/observable/Disposable";
import { Disposer } from "../../core/observable/Disposer";
import { EntityPropType } from "../../core/data_formats/parsing/quest/properties";
export class EntityInfoView extends ResizableView { export class EntityInfoView extends ResizableView {
readonly element = div({ className: "quest_editor_EntityInfoView", tabIndex: -1 }); readonly element = div({ className: "quest_editor_EntityInfoView", tabIndex: -1 });
private readonly no_entity_view = new UnavailableView("No entity selected."); private readonly no_entity_view = new UnavailableView("No entity selected.");
private readonly table_element = table(); private readonly standard_props_element = table();
private readonly specific_props_element = table();
private readonly type_element: HTMLTableCellElement; private readonly type_element: HTMLTableCellElement;
private readonly name_element: HTMLTableCellElement; private readonly name_element: HTMLTableCellElement;
@ -30,7 +35,7 @@ export class EntityInfoView extends ResizableView {
const coord_class = "quest_editor_EntityInfoView_coord"; const coord_class = "quest_editor_EntityInfoView_coord";
this.table_element.append( this.standard_props_element.append(
tr(th("Type:"), (this.type_element = td())), tr(th("Type:"), (this.type_element = td())),
tr(th("Name:"), (this.name_element = td())), tr(th("Name:"), (this.name_element = td())),
tr(th("Section:"), (this.section_id_element = td())), tr(th("Section:"), (this.section_id_element = td())),
@ -45,12 +50,18 @@ export class EntityInfoView extends ResizableView {
tr(th({ className: coord_class }, "Z:"), td(this.rot_z_element.element)), tr(th({ className: coord_class }, "Z:"), td(this.rot_z_element.element)),
); );
this.element.append(this.table_element, this.no_entity_view.element); bind_children_to(this.specific_props_element, ctrl.props, this.create_prop_row);
this.element.append(
this.standard_props_element,
this.specific_props_element,
this.no_entity_view.element,
);
this.element.addEventListener("focus", ctrl.focused, true); this.element.addEventListener("focus", ctrl.focused, true);
this.disposables( this.disposables(
bind_attr(this.table_element, "hidden", ctrl.unavailable), bind_attr(this.standard_props_element, "hidden", ctrl.unavailable),
this.no_entity_view.visible.bind_to(ctrl.unavailable), this.no_entity_view.visible.bind_to(ctrl.unavailable),
bind_attr(this.type_element, "textContent", ctrl.type), bind_attr(this.type_element, "textContent", ctrl.type),
@ -102,4 +113,60 @@ export class EntityInfoView extends ResizableView {
this.rot_y_element.enabled.val = enabled; this.rot_y_element.enabled.val = enabled;
this.rot_z_element.enabled.val = enabled; this.rot_z_element.enabled.val = enabled;
} }
private create_prop_row(prop: QuestEntityPropModel): [HTMLTableRowElement, Disposable] {
const disposer = new Disposer();
let min: number | undefined;
let max: number | undefined;
switch (prop.type) {
case EntityPropType.U8:
min = 0;
max = 0xff;
break;
case EntityPropType.U16:
min = 0;
max = 0xffff;
break;
case EntityPropType.U32:
min = 0;
max = 0xffffffff;
break;
case EntityPropType.I8:
min = -0x80;
max = 0x7f;
break;
case EntityPropType.I16:
min = -0x8000;
max = 0x7fff;
break;
case EntityPropType.I32:
min = -0x80000000;
max = 0x7fffffff;
break;
case EntityPropType.Angle:
min = -2 * Math.PI;
max = 2 * Math.PI;
break;
}
const round_to =
prop.type === EntityPropType.F32 || prop.type === EntityPropType.Angle ? 3 : 1;
const value_input = disposer.add(
new NumberInput(prop.value.val, {
min,
max,
round_to,
enabled: false,
}),
);
disposer.add_all(value_input.value.bind_to(prop.value));
const element = tr(th(`${prop.name}:`), td(value_input.element));
return [element, disposer];
}
} }

View File

@ -1,6 +1,11 @@
import { EntityType, QuestEntity } from "../../core/data_formats/parsing/quest/Quest"; import {
entity_data,
EntityType,
get_entity_type,
QuestEntity,
} from "../../core/data_formats/parsing/quest/Quest";
import { Property } from "../../core/observable/property/Property"; import { Property } from "../../core/observable/property/Property";
import { property } from "../../core/observable"; import { list_property, property } from "../../core/observable";
import { WritableProperty } from "../../core/observable/property/WritableProperty"; import { WritableProperty } from "../../core/observable/property/WritableProperty";
import { SectionModel } from "./SectionModel"; import { SectionModel } from "./SectionModel";
import { Euler, Quaternion, Vector3 } from "three"; import { Euler, Quaternion, Vector3 } from "three";
@ -8,6 +13,9 @@ import { floor_mod } from "../../core/math";
import { euler, euler_from_quat } from "./euler"; import { euler, euler_from_quat } from "./euler";
import { vec3_to_threejs } from "../../core/rendering/conversion"; import { vec3_to_threejs } from "../../core/rendering/conversion";
import { Vec3 } from "../../core/data_formats/vector"; import { Vec3 } from "../../core/data_formats/vector";
import { WritableListProperty } from "../../core/observable/property/list/WritableListProperty";
import { QuestEntityPropModel } from "./QuestEntityPropModel";
import { ListProperty } from "../../core/observable/property/list/ListProperty";
// These quaternions are used as temporary variables to avoid memory allocation. // These quaternions are used as temporary variables to avoid memory allocation.
const q1 = new Quaternion(); const q1 = new Quaternion();
@ -23,6 +31,7 @@ export abstract class QuestEntityModel<
private readonly _world_position: WritableProperty<Vector3>; private readonly _world_position: WritableProperty<Vector3>;
private readonly _rotation: WritableProperty<Euler>; private readonly _rotation: WritableProperty<Euler>;
private readonly _world_rotation: WritableProperty<Euler>; private readonly _world_rotation: WritableProperty<Euler>;
private readonly _props: WritableListProperty<QuestEntityPropModel>;
/** /**
* Many modifications done to the underlying entity directly will not be reflected in this * Many modifications done to the underlying entity directly will not be reflected in this
@ -54,6 +63,8 @@ export abstract class QuestEntityModel<
readonly world_rotation: Property<Euler>; readonly world_rotation: Property<Euler>;
readonly props: ListProperty<QuestEntityPropModel>;
protected constructor(entity: Entity) { protected constructor(entity: Entity) {
this.entity = entity; this.entity = entity;
@ -78,6 +89,14 @@ export abstract class QuestEntityModel<
this._world_rotation = property(rotation); this._world_rotation = property(rotation);
this.world_rotation = this._world_rotation; this.world_rotation = this._world_rotation;
this._props = list_property(
undefined,
...entity_data(get_entity_type(entity)).properties.map(
p => new QuestEntityPropModel(entity, p),
),
);
this.props = this._props;
} }
set_section(section: SectionModel): this { set_section(section: SectionModel): this {

View File

@ -0,0 +1,36 @@
import { WritableProperty } from "../../core/observable/property/WritableProperty";
import { Property } from "../../core/observable/property/Property";
import { EntityProp, EntityPropType } from "../../core/data_formats/parsing/quest/properties";
import { property } from "../../core/observable";
import {
get_entity_prop_value,
QuestEntity,
set_entity_prop_value,
} from "../../core/data_formats/parsing/quest/Quest";
export class QuestEntityPropModel {
private readonly entity: QuestEntity;
private readonly prop: EntityProp;
private readonly _value: WritableProperty<number>;
readonly name: string;
readonly type: EntityPropType;
readonly value: Property<number>;
constructor(quest_entity: QuestEntity, entity_prop: EntityProp) {
this.entity = quest_entity;
this.prop = entity_prop;
this.name = entity_prop.name;
this.type = entity_prop.type;
this._value = property(get_entity_prop_value(quest_entity, entity_prop));
this.value = this._value;
}
set_value(value: number): void {
set_entity_prop_value(this.entity, this.prop, value);
this._value.val = value;
}
}