mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Huge performance improvement when translating quest entities.
This commit is contained in:
parent
f670718637
commit
15327d1478
@ -7,7 +7,13 @@ export class Vec2 {
|
|||||||
this.y = y;
|
this.y = y;
|
||||||
}
|
}
|
||||||
|
|
||||||
add(v: Vec2): Vec2 {
|
set(x: number, y: number): this {
|
||||||
|
this.x = x;
|
||||||
|
this.y = y;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
add(v: Vec2): this {
|
||||||
this.x += v.x;
|
this.x += v.x;
|
||||||
this.y += v.y;
|
this.y += v.y;
|
||||||
return this;
|
return this;
|
||||||
@ -33,7 +39,14 @@ export class Vec3 {
|
|||||||
this.z = z;
|
this.z = z;
|
||||||
}
|
}
|
||||||
|
|
||||||
add(v: Vec3): Vec3 {
|
set(x: number, y: number, z: number): this {
|
||||||
|
this.x = x;
|
||||||
|
this.y = y;
|
||||||
|
this.z = z;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
add(v: Vec3): this {
|
||||||
this.x += v.x;
|
this.x += v.x;
|
||||||
this.y += v.y;
|
this.y += v.y;
|
||||||
this.z += v.z;
|
this.z += v.z;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { Episode, check_episode } from ".";
|
import { Episode, check_episode, EntityType } from ".";
|
||||||
|
|
||||||
export class NpcType {
|
export class NpcType implements EntityType {
|
||||||
readonly id: number;
|
readonly id: number;
|
||||||
/**
|
/**
|
||||||
* Matches the constant name. E.g. the code of NpcType.Zu is "Zu".
|
* Matches the constant name. E.g. the code of NpcType.Zu is "Zu".
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -1,4 +1,4 @@
|
|||||||
import { computed, observable } from "mobx";
|
import { computed, observable, action } from "mobx";
|
||||||
import { DatNpc, DatObject, DatUnknown } from "../data_formats/parsing/quest/dat";
|
import { DatNpc, DatObject, DatUnknown } from "../data_formats/parsing/quest/dat";
|
||||||
import { Vec3 } from "../data_formats/vector";
|
import { Vec3 } from "../data_formats/vector";
|
||||||
import { enum_values } from "../enums";
|
import { enum_values } from "../enums";
|
||||||
@ -127,10 +127,18 @@ export class Quest {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface EntityType {
|
||||||
|
readonly id: number;
|
||||||
|
readonly code: string;
|
||||||
|
readonly name: string;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract class from which QuestNpc and QuestObject derive.
|
* Abstract class from which QuestNpc and QuestObject derive.
|
||||||
*/
|
*/
|
||||||
export class QuestEntity {
|
export class QuestEntity<Type extends EntityType = EntityType> {
|
||||||
|
readonly type: Type;
|
||||||
|
|
||||||
@observable area_id: number;
|
@observable area_id: number;
|
||||||
|
|
||||||
private _section_id: number;
|
private _section_id: number;
|
||||||
@ -139,14 +147,14 @@ export class QuestEntity {
|
|||||||
return this.section ? this.section.id : this._section_id;
|
return this.section ? this.section.id : this._section_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
@observable section?: Section;
|
@observable.ref section?: Section;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* World position
|
* World position
|
||||||
*/
|
*/
|
||||||
@observable position: Vec3;
|
@observable.ref position: Vec3;
|
||||||
|
|
||||||
@observable rotation: Vec3;
|
@observable.ref rotation: Vec3;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Section-relative position
|
* Section-relative position
|
||||||
@ -185,9 +193,10 @@ export class QuestEntity {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
constructor(area_id: number, section_id: number, position: Vec3, rotation: Vec3) {
|
constructor(type: Type, area_id: number, section_id: number, position: Vec3, rotation: Vec3) {
|
||||||
if (Object.getPrototypeOf(this) === Object.getPrototypeOf(QuestEntity))
|
if (Object.getPrototypeOf(this) === Object.getPrototypeOf(QuestEntity))
|
||||||
throw new Error("Abstract class should not be instantiated directly.");
|
throw new Error("Abstract class should not be instantiated directly.");
|
||||||
|
if (!type) throw new Error("type is required.");
|
||||||
if (!Number.isInteger(area_id) || area_id < 0)
|
if (!Number.isInteger(area_id) || area_id < 0)
|
||||||
throw new Error(`Expected area_id to be a non-negative integer, got ${area_id}.`);
|
throw new Error(`Expected area_id to be a non-negative integer, got ${area_id}.`);
|
||||||
if (!Number.isInteger(section_id) || section_id < 0)
|
if (!Number.isInteger(section_id) || section_id < 0)
|
||||||
@ -195,14 +204,21 @@ export class QuestEntity {
|
|||||||
if (!position) throw new Error("position is required.");
|
if (!position) throw new Error("position is required.");
|
||||||
if (!rotation) throw new Error("rotation is required.");
|
if (!rotation) throw new Error("rotation is required.");
|
||||||
|
|
||||||
|
this.type = type;
|
||||||
this.area_id = area_id;
|
this.area_id = area_id;
|
||||||
this._section_id = section_id;
|
this._section_id = section_id;
|
||||||
this.position = position;
|
this.position = position;
|
||||||
this.rotation = rotation;
|
this.rotation = rotation;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
set_position_and_section(position: Vec3, section?: Section): void {
|
||||||
|
this.position = position;
|
||||||
|
this.section = section;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class QuestObject extends QuestEntity {
|
export class QuestObject extends QuestEntity<ObjectType> {
|
||||||
@observable type: ObjectType;
|
@observable type: ObjectType;
|
||||||
/**
|
/**
|
||||||
* The raw data from a DAT file.
|
* The raw data from a DAT file.
|
||||||
@ -217,16 +233,14 @@ export class QuestObject extends QuestEntity {
|
|||||||
type: ObjectType,
|
type: ObjectType,
|
||||||
dat: DatObject
|
dat: DatObject
|
||||||
) {
|
) {
|
||||||
super(area_id, section_id, position, rotation);
|
super(type, area_id, section_id, position, rotation);
|
||||||
|
|
||||||
if (!type) throw new Error("type is required.");
|
|
||||||
|
|
||||||
this.type = type;
|
this.type = type;
|
||||||
this.dat = dat;
|
this.dat = dat;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class QuestNpc extends QuestEntity {
|
export class QuestNpc extends QuestEntity<NpcType> {
|
||||||
@observable type: NpcType;
|
@observable type: NpcType;
|
||||||
/**
|
/**
|
||||||
* The raw data from a DAT file.
|
* The raw data from a DAT file.
|
||||||
@ -241,7 +255,7 @@ export class QuestNpc extends QuestEntity {
|
|||||||
type: NpcType,
|
type: NpcType,
|
||||||
dat: DatNpc
|
dat: DatNpc
|
||||||
) {
|
) {
|
||||||
super(area_id, section_id, position, rotation);
|
super(type, area_id, section_id, position, rotation);
|
||||||
|
|
||||||
if (!type) throw new Error("type is required.");
|
if (!type) throw new Error("type is required.");
|
||||||
|
|
||||||
@ -270,7 +284,7 @@ export class Area {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export class AreaVariant {
|
export class AreaVariant {
|
||||||
@observable sections: Section[] = [];
|
@observable.shallow sections: Section[] = [];
|
||||||
|
|
||||||
constructor(public id: number, public area: Area) {
|
constructor(public id: number, public area: Area) {
|
||||||
if (!Number.isInteger(id) || id < 0)
|
if (!Number.isInteger(id) || id < 0)
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { autorun, runInAction } from "mobx";
|
import { autorun } from "mobx";
|
||||||
import { Intersection, Mesh, MeshLambertMaterial, Plane, Raycaster, Vector2, Vector3 } from "three";
|
import { Intersection, Mesh, MeshLambertMaterial, Plane, Raycaster, Vector2, Vector3 } from "three";
|
||||||
import { Vec3 } from "../data_formats/vector";
|
import { Vec3 } from "../data_formats/vector";
|
||||||
import { QuestEntity, QuestNpc, QuestObject, Section } from "../domain";
|
import { QuestEntity, QuestNpc, QuestObject, Section } from "../domain";
|
||||||
@ -239,14 +239,14 @@ export class QuestEntityControls {
|
|||||||
const { intersection, section } = this.pick_terrain(pointer_position, pick);
|
const { intersection, section } = this.pick_terrain(pointer_position, pick);
|
||||||
|
|
||||||
if (intersection) {
|
if (intersection) {
|
||||||
runInAction(() => {
|
selection.entity.set_position_and_section(
|
||||||
selection.entity.position = new Vec3(
|
new Vec3(
|
||||||
intersection.point.x,
|
intersection.point.x,
|
||||||
intersection.point.y + pick.drag_y,
|
intersection.point.y + pick.drag_y,
|
||||||
intersection.point.z
|
intersection.point.z
|
||||||
|
),
|
||||||
|
section
|
||||||
);
|
);
|
||||||
selection.entity.section = section;
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
// If the cursor is not over any terrain, we translate the entity accross the horizontal plane in which the entity's origin lies.
|
// If the cursor is not over any terrain, we translate the entity accross the horizontal plane in which the entity's origin lies.
|
||||||
this.raycaster.setFromCamera(pointer_position, this.renderer.camera);
|
this.raycaster.setFromCamera(pointer_position, this.renderer.camera);
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
import { InputNumber } from "antd";
|
import { InputNumber } from "antd";
|
||||||
import { observer } from "mobx-react";
|
import { observer } from "mobx-react";
|
||||||
import React, { ReactNode, Component } from "react";
|
import React, { ReactNode, Component, PureComponent } from "react";
|
||||||
import { QuestNpc, QuestObject, QuestEntity } from "../../domain";
|
import { QuestNpc, QuestObject, QuestEntity } from "../../domain";
|
||||||
import "./EntityInfoComponent.css";
|
import "./EntityInfoComponent.css";
|
||||||
|
import { IReactionDisposer, autorun } from "mobx";
|
||||||
|
import { Vec3 } from "../../data_formats/vector";
|
||||||
|
|
||||||
export type Props = {
|
export type Props = {
|
||||||
entity?: QuestEntity;
|
entity?: QuestEntity;
|
||||||
@ -104,28 +106,69 @@ export class EntityInfoComponent extends Component<Props> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@observer
|
type CoordProps = {
|
||||||
class CoordRow extends Component<{
|
|
||||||
entity: QuestEntity;
|
entity: QuestEntity;
|
||||||
position_type: "position" | "section_position";
|
position_type: "position" | "section_position";
|
||||||
coord: "x" | "y" | "z";
|
coord: "x" | "y" | "z";
|
||||||
}> {
|
};
|
||||||
|
|
||||||
|
class CoordRow extends PureComponent<CoordProps> {
|
||||||
render(): ReactNode {
|
render(): ReactNode {
|
||||||
const entity = this.props.entity;
|
|
||||||
const value = entity[this.props.position_type][this.props.coord];
|
|
||||||
return (
|
return (
|
||||||
<tr>
|
<tr>
|
||||||
<td>{this.props.coord.toUpperCase()}: </td>
|
<td>{this.props.coord.toUpperCase()}: </td>
|
||||||
<td>
|
<td>
|
||||||
|
<CoordInput {...this.props} />
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class CoordInput extends Component<CoordProps, { value: number }> {
|
||||||
|
private disposer?: IReactionDisposer;
|
||||||
|
|
||||||
|
state = { value: 0 };
|
||||||
|
|
||||||
|
componentDidMount(): void {
|
||||||
|
this.start_observing();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount(): void {
|
||||||
|
if (this.disposer) this.disposer();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prev_props: CoordProps): void {
|
||||||
|
if (this.props.entity !== prev_props.entity) {
|
||||||
|
this.start_observing();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render(): ReactNode {
|
||||||
|
return (
|
||||||
<InputNumber
|
<InputNumber
|
||||||
value={value}
|
value={this.state.value}
|
||||||
size="small"
|
size="small"
|
||||||
precision={3}
|
precision={3}
|
||||||
className="EntityInfoComponent-coord"
|
className="EntityInfoComponent-coord"
|
||||||
onChange={this.changed}
|
onChange={this.changed}
|
||||||
/>
|
/>
|
||||||
</td>
|
);
|
||||||
</tr>
|
}
|
||||||
|
|
||||||
|
private start_observing() {
|
||||||
|
if (this.disposer) this.disposer();
|
||||||
|
|
||||||
|
this.disposer = autorun(
|
||||||
|
() => {
|
||||||
|
this.setState({
|
||||||
|
value: this.props.entity[this.props.position_type][this.props.coord],
|
||||||
|
});
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: `${this.props.entity.type.code}.${this.props.position_type}.${this.props.coord} changed`,
|
||||||
|
delay: 50,
|
||||||
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user