2019-07-03 00:08:06 +08:00
|
|
|
import {
|
|
|
|
Intersection,
|
|
|
|
Mesh,
|
|
|
|
MeshLambertMaterial,
|
|
|
|
Object3D,
|
|
|
|
Plane,
|
|
|
|
Raycaster,
|
|
|
|
Vector2,
|
|
|
|
Vector3,
|
|
|
|
} from "three";
|
2019-07-02 23:00:24 +08:00
|
|
|
import { Area, Quest, QuestEntity, QuestNpc, QuestObject, Section } from "../domain";
|
|
|
|
import { Vec3 } from "../data_formats/Vec3";
|
2019-07-01 14:53:16 +08:00
|
|
|
import { area_store } from "../stores/AreaStore";
|
2019-07-01 01:55:30 +08:00
|
|
|
import { quest_editor_store } from "../stores/QuestEditorStore";
|
2019-07-03 00:08:06 +08:00
|
|
|
import {
|
|
|
|
NPC_COLOR,
|
|
|
|
NPC_HOVER_COLOR,
|
|
|
|
NPC_SELECTED_COLOR,
|
|
|
|
OBJECT_COLOR,
|
|
|
|
OBJECT_HOVER_COLOR,
|
|
|
|
OBJECT_SELECTED_COLOR,
|
|
|
|
} from "./entities";
|
2019-07-01 01:55:30 +08:00
|
|
|
import { Renderer } from "./Renderer";
|
|
|
|
|
|
|
|
let renderer: QuestRenderer | undefined;
|
|
|
|
|
|
|
|
export function get_quest_renderer(): QuestRenderer {
|
|
|
|
if (!renderer) renderer = new QuestRenderer();
|
|
|
|
return renderer;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface PickEntityResult {
|
|
|
|
object: Mesh;
|
|
|
|
entity: QuestEntity;
|
|
|
|
grab_offset: Vector3;
|
|
|
|
drag_adjust: Vector3;
|
|
|
|
drag_y: number;
|
|
|
|
manipulating: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
export class QuestRenderer extends Renderer {
|
|
|
|
private raycaster = new Raycaster();
|
|
|
|
|
|
|
|
private quest?: Quest;
|
|
|
|
private quest_entities_loaded = false;
|
|
|
|
private area?: Area;
|
|
|
|
private objs: Map<number, QuestObject[]> = new Map(); // Objs grouped by area id
|
|
|
|
private npcs: Map<number, QuestNpc[]> = new Map(); // Npcs grouped by area id
|
|
|
|
|
|
|
|
private collision_geometry = new Object3D();
|
|
|
|
private render_geometry = new Object3D();
|
|
|
|
private obj_geometry = new Object3D();
|
|
|
|
private npc_geometry = new Object3D();
|
|
|
|
|
|
|
|
private hovered_data?: PickEntityResult;
|
|
|
|
private selected_data?: PickEntityResult;
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
|
2019-07-03 00:08:06 +08:00
|
|
|
this.renderer.domElement.addEventListener("mousedown", this.on_mouse_down);
|
|
|
|
this.renderer.domElement.addEventListener("mouseup", this.on_mouse_up);
|
|
|
|
this.renderer.domElement.addEventListener("mousemove", this.on_mouse_move);
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
this.scene.add(this.obj_geometry);
|
|
|
|
this.scene.add(this.npc_geometry);
|
|
|
|
}
|
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
set_quest_and_area(quest?: Quest, area?: Area): void {
|
2019-07-01 01:55:30 +08:00
|
|
|
let update = false;
|
|
|
|
|
|
|
|
if (this.area !== area) {
|
|
|
|
this.area = area;
|
|
|
|
update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.quest !== quest) {
|
|
|
|
this.quest = quest;
|
|
|
|
|
|
|
|
this.objs.clear();
|
|
|
|
this.npcs.clear();
|
|
|
|
|
|
|
|
if (quest) {
|
|
|
|
for (const obj of quest.objects) {
|
|
|
|
const array = this.objs.get(obj.area_id) || [];
|
|
|
|
array.push(obj);
|
|
|
|
this.objs.set(obj.area_id, array);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const npc of quest.npcs) {
|
|
|
|
const array = this.npcs.get(npc.area_id) || [];
|
|
|
|
array.push(npc);
|
|
|
|
this.npcs.set(npc.area_id, array);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
update = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (update) {
|
|
|
|
this.update_geometry();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
protected render(): void {
|
2019-07-01 01:55:30 +08:00
|
|
|
this.controls.update();
|
|
|
|
this.add_loaded_entities();
|
|
|
|
this.renderer.render(this.scene, this.camera);
|
|
|
|
}
|
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
private async update_geometry(): Promise<void> {
|
2019-07-01 01:55:30 +08:00
|
|
|
this.scene.remove(this.obj_geometry);
|
|
|
|
this.scene.remove(this.npc_geometry);
|
|
|
|
this.obj_geometry = new Object3D();
|
|
|
|
this.npc_geometry = new Object3D();
|
|
|
|
this.scene.add(this.obj_geometry);
|
|
|
|
this.scene.add(this.npc_geometry);
|
|
|
|
this.quest_entities_loaded = false;
|
|
|
|
|
|
|
|
this.scene.remove(this.collision_geometry);
|
|
|
|
|
|
|
|
if (this.quest && this.area) {
|
|
|
|
const episode = this.quest.episode;
|
|
|
|
const area_id = this.area.id;
|
|
|
|
const variant = this.quest.area_variants.find(v => v.area.id === area_id);
|
|
|
|
const variant_id = (variant && variant.id) || 0;
|
|
|
|
|
2019-07-03 00:08:06 +08:00
|
|
|
const collision_geometry = await area_store.get_area_collision_geometry(
|
|
|
|
episode,
|
|
|
|
area_id,
|
|
|
|
variant_id
|
|
|
|
);
|
2019-07-01 01:55:30 +08:00
|
|
|
|
2019-07-01 14:53:16 +08:00
|
|
|
if (this.quest && this.area) {
|
|
|
|
this.scene.remove(this.collision_geometry);
|
2019-07-01 01:55:30 +08:00
|
|
|
|
2019-07-01 14:53:16 +08:00
|
|
|
this.reset_camera(new Vector3(0, 800, 700), new Vector3(0, 0, 0));
|
2019-07-01 01:55:30 +08:00
|
|
|
|
2019-07-01 14:53:16 +08:00
|
|
|
this.collision_geometry = collision_geometry;
|
|
|
|
this.scene.add(collision_geometry);
|
|
|
|
}
|
|
|
|
|
2019-07-03 00:08:06 +08:00
|
|
|
const render_geometry = await area_store.get_area_render_geometry(
|
|
|
|
episode,
|
|
|
|
area_id,
|
|
|
|
variant_id
|
|
|
|
);
|
2019-07-01 14:53:16 +08:00
|
|
|
|
|
|
|
if (this.quest && this.area) {
|
|
|
|
this.render_geometry = render_geometry;
|
|
|
|
}
|
2019-07-01 01:55:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
private add_loaded_entities(): void {
|
2019-07-01 01:55:30 +08:00
|
|
|
if (this.quest && this.area && !this.quest_entities_loaded) {
|
|
|
|
let loaded = true;
|
|
|
|
|
|
|
|
for (const object of this.quest.objects) {
|
|
|
|
if (object.area_id === this.area.id) {
|
|
|
|
if (object.object_3d) {
|
|
|
|
this.obj_geometry.add(object.object_3d);
|
|
|
|
} else {
|
|
|
|
loaded = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const npc of this.quest.npcs) {
|
|
|
|
if (npc.area_id === this.area.id) {
|
|
|
|
if (npc.object_3d) {
|
|
|
|
this.npc_geometry.add(npc.object_3d);
|
|
|
|
} else {
|
|
|
|
loaded = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.quest_entities_loaded = loaded;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private on_mouse_down = (e: MouseEvent) => {
|
|
|
|
const old_selected_data = this.selected_data;
|
2019-07-03 00:08:06 +08:00
|
|
|
const data = this.pick_entity(this.pointer_pos_to_device_coords(e));
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
// Did we pick a different object than the previously hovered over 3D object?
|
|
|
|
if (this.hovered_data && (!data || data.object !== this.hovered_data.object)) {
|
|
|
|
(this.hovered_data.object.material as MeshLambertMaterial).color.set(
|
2019-07-03 00:08:06 +08:00
|
|
|
this.get_color(this.hovered_data.entity, "normal")
|
2019-07-01 01:55:30 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Did we pick a different object than the previously selected 3D object?
|
|
|
|
if (this.selected_data && (!data || data.object !== this.selected_data.object)) {
|
|
|
|
(this.selected_data.object.material as MeshLambertMaterial).color.set(
|
2019-07-03 00:08:06 +08:00
|
|
|
this.get_color(this.selected_data.entity, "normal")
|
2019-07-01 01:55:30 +08:00
|
|
|
);
|
|
|
|
this.selected_data.manipulating = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data) {
|
|
|
|
// User selected an entity.
|
|
|
|
(data.object.material as MeshLambertMaterial).color.set(
|
2019-07-03 00:08:06 +08:00
|
|
|
this.get_color(data.entity, "selected")
|
2019-07-01 01:55:30 +08:00
|
|
|
);
|
|
|
|
data.manipulating = true;
|
|
|
|
this.hovered_data = data;
|
|
|
|
this.selected_data = data;
|
|
|
|
this.controls.enabled = false;
|
|
|
|
} else {
|
|
|
|
// User clicked on terrain or outside of area.
|
|
|
|
this.hovered_data = undefined;
|
|
|
|
this.selected_data = undefined;
|
|
|
|
this.controls.enabled = true;
|
|
|
|
}
|
|
|
|
|
2019-07-03 00:08:06 +08:00
|
|
|
const selection_changed =
|
|
|
|
old_selected_data && data
|
|
|
|
? old_selected_data.object !== data.object
|
|
|
|
: old_selected_data !== data;
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
if (selection_changed) {
|
|
|
|
quest_editor_store.set_selected_entity(data && data.entity);
|
|
|
|
}
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
private on_mouse_up = () => {
|
|
|
|
if (this.selected_data) {
|
|
|
|
this.selected_data.manipulating = false;
|
|
|
|
this.controls.enabled = true;
|
|
|
|
}
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
private on_mouse_move = (e: MouseEvent) => {
|
|
|
|
const pointer_pos = this.pointer_pos_to_device_coords(e);
|
|
|
|
|
|
|
|
if (this.selected_data && this.selected_data.manipulating) {
|
|
|
|
if (e.buttons === 1) {
|
|
|
|
// User is dragging a selected entity.
|
|
|
|
const data = this.selected_data;
|
|
|
|
|
|
|
|
if (e.shiftKey) {
|
|
|
|
// Vertical movement.
|
|
|
|
// We intersect with a plane that's oriented toward the camera and that's coplanar with the point where the entity was grabbed.
|
|
|
|
this.raycaster.setFromCamera(pointer_pos, this.camera);
|
|
|
|
const ray = this.raycaster.ray;
|
2019-07-03 00:08:06 +08:00
|
|
|
const negative_world_dir = this.camera
|
|
|
|
.getWorldDirection(new Vector3())
|
|
|
|
.negate();
|
2019-07-01 01:55:30 +08:00
|
|
|
const plane = new Plane().setFromNormalAndCoplanarPoint(
|
|
|
|
new Vector3(negative_world_dir.x, 0, negative_world_dir.z).normalize(),
|
|
|
|
data.object.position.sub(data.grab_offset)
|
|
|
|
);
|
|
|
|
const intersection_point = new Vector3();
|
|
|
|
|
|
|
|
if (ray.intersectPlane(plane, intersection_point)) {
|
|
|
|
const y = intersection_point.y + data.grab_offset.y;
|
|
|
|
const y_delta = y - data.entity.position.y;
|
|
|
|
data.drag_y += y_delta;
|
|
|
|
data.drag_adjust.y -= y_delta;
|
|
|
|
data.entity.position = new Vec3(
|
|
|
|
data.entity.position.x,
|
|
|
|
y,
|
|
|
|
data.entity.position.z
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Horizontal movement accross terrain.
|
|
|
|
// Cast ray adjusted for dragging entities.
|
|
|
|
const { intersection, section } = this.pick_terrain(pointer_pos, data);
|
|
|
|
|
|
|
|
if (intersection) {
|
|
|
|
data.entity.position = new Vec3(
|
|
|
|
intersection.point.x,
|
|
|
|
intersection.point.y + data.drag_y,
|
|
|
|
intersection.point.z
|
|
|
|
);
|
|
|
|
data.entity.section = section;
|
|
|
|
} else {
|
|
|
|
// 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_pos, this.camera);
|
|
|
|
const ray = this.raycaster.ray;
|
|
|
|
// ray.origin.add(data.dragAdjust);
|
|
|
|
const plane = new Plane(
|
|
|
|
new Vector3(0, 1, 0),
|
2019-07-03 00:08:06 +08:00
|
|
|
-data.entity.position.y + data.grab_offset.y
|
|
|
|
);
|
2019-07-01 01:55:30 +08:00
|
|
|
const intersection_point = new Vector3();
|
|
|
|
|
|
|
|
if (ray.intersectPlane(plane, intersection_point)) {
|
|
|
|
data.entity.position = new Vec3(
|
|
|
|
intersection_point.x + data.grab_offset.x,
|
|
|
|
data.entity.position.y,
|
|
|
|
intersection_point.z + data.grab_offset.z
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// User is hovering.
|
|
|
|
const old_data = this.hovered_data;
|
|
|
|
const data = this.pick_entity(pointer_pos);
|
|
|
|
|
|
|
|
if (old_data && (!data || data.object !== old_data.object)) {
|
|
|
|
if (!this.selected_data || old_data.object !== this.selected_data.object) {
|
|
|
|
(old_data.object.material as MeshLambertMaterial).color.set(
|
2019-07-03 00:08:06 +08:00
|
|
|
this.get_color(old_data.entity, "normal")
|
2019-07-01 01:55:30 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.hovered_data = undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data && (!old_data || data.object !== old_data.object)) {
|
|
|
|
if (!this.selected_data || data.object !== this.selected_data.object) {
|
|
|
|
(data.object.material as MeshLambertMaterial).color.set(
|
2019-07-03 00:08:06 +08:00
|
|
|
this.get_color(data.entity, "hover")
|
2019-07-01 01:55:30 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.hovered_data = data;
|
|
|
|
}
|
|
|
|
}
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-07-01 01:55:30 +08:00
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
private pointer_pos_to_device_coords(e: MouseEvent): Vector2 {
|
2019-07-01 01:55:30 +08:00
|
|
|
const coords = new Vector2();
|
|
|
|
this.renderer.getSize(coords);
|
2019-07-03 00:08:06 +08:00
|
|
|
coords.width = (e.offsetX / coords.width) * 2 - 1;
|
|
|
|
coords.height = (e.offsetY / coords.height) * -2 + 1;
|
2019-07-01 01:55:30 +08:00
|
|
|
return coords;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param pointer_pos - pointer coordinates in normalized device space
|
|
|
|
*/
|
|
|
|
private pick_entity(pointer_pos: Vector2): PickEntityResult | undefined {
|
|
|
|
// Find the nearest object and NPC under the pointer.
|
|
|
|
this.raycaster.setFromCamera(pointer_pos, this.camera);
|
2019-07-03 00:08:06 +08:00
|
|
|
const [nearest_object] = this.raycaster.intersectObjects(this.obj_geometry.children);
|
|
|
|
const [nearest_npc] = this.raycaster.intersectObjects(this.npc_geometry.children);
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
if (!nearest_object && !nearest_npc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const object_dist = nearest_object ? nearest_object.distance : Infinity;
|
|
|
|
const npc_dist = nearest_npc ? nearest_npc.distance : Infinity;
|
|
|
|
const intersection = object_dist < npc_dist ? nearest_object : nearest_npc;
|
|
|
|
|
|
|
|
const entity = intersection.object.userData.entity;
|
|
|
|
// Vector that points from the grabbing point to the model's origin.
|
2019-07-03 00:08:06 +08:00
|
|
|
const grab_offset = intersection.object.position.clone().sub(intersection.point);
|
2019-07-01 01:55:30 +08:00
|
|
|
// 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.
|
2019-07-03 00:08:06 +08:00
|
|
|
this.raycaster.set(intersection.object.position, new Vector3(0, -1, 0));
|
|
|
|
const [terrain] = this.raycaster.intersectObjects(this.collision_geometry.children, true);
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
if (terrain) {
|
|
|
|
drag_adjust.sub(new Vector3(0, terrain.distance, 0));
|
|
|
|
drag_y += terrain.distance;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
object: intersection.object as Mesh,
|
|
|
|
entity,
|
|
|
|
grab_offset,
|
|
|
|
drag_adjust,
|
|
|
|
drag_y,
|
2019-07-03 00:08:06 +08:00
|
|
|
manipulating: false,
|
2019-07-01 01:55:30 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param pointer_pos - pointer coordinates in normalized device space
|
|
|
|
*/
|
|
|
|
private pick_terrain(
|
|
|
|
pointer_pos: Vector2,
|
|
|
|
data: PickEntityResult
|
|
|
|
): {
|
2019-07-03 00:08:06 +08:00
|
|
|
intersection?: Intersection;
|
|
|
|
section?: Section;
|
2019-07-01 01:55:30 +08:00
|
|
|
} {
|
|
|
|
this.raycaster.setFromCamera(pointer_pos, this.camera);
|
|
|
|
this.raycaster.ray.origin.add(data.drag_adjust);
|
2019-07-03 00:08:06 +08:00
|
|
|
const terrains = this.raycaster.intersectObjects(this.collision_geometry.children, true);
|
2019-07-01 01:55:30 +08:00
|
|
|
|
|
|
|
// Don't allow entities to be placed on very steep terrain.
|
|
|
|
// E.g. walls.
|
|
|
|
// TODO: make use of the flags field in the collision data.
|
|
|
|
for (const terrain of terrains) {
|
|
|
|
if (terrain.face!.normal.y > 0.75) {
|
|
|
|
// Find section ID.
|
2019-07-03 00:08:06 +08:00
|
|
|
this.raycaster.set(terrain.point.clone().setY(1000), new Vector3(0, -1, 0));
|
2019-07-01 01:55:30 +08:00
|
|
|
const render_terrains = this.raycaster
|
|
|
|
.intersectObjects(this.render_geometry.children, true)
|
|
|
|
.filter(rt => rt.object.userData.section.id >= 0);
|
|
|
|
|
|
|
|
return {
|
|
|
|
intersection: terrain,
|
2019-07-03 00:08:06 +08:00
|
|
|
section: render_terrains[0] && render_terrains[0].object.userData.section,
|
2019-07-01 01:55:30 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
private get_color(entity: QuestEntity, type: "normal" | "hover" | "selected"): number {
|
2019-07-01 01:55:30 +08:00
|
|
|
const is_npc = entity instanceof QuestNpc;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
default:
|
2019-07-03 00:08:06 +08:00
|
|
|
case "normal":
|
|
|
|
return is_npc ? NPC_COLOR : OBJECT_COLOR;
|
|
|
|
case "hover":
|
|
|
|
return is_npc ? NPC_HOVER_COLOR : OBJECT_HOVER_COLOR;
|
|
|
|
case "selected":
|
|
|
|
return is_npc ? NPC_SELECTED_COLOR : OBJECT_SELECTED_COLOR;
|
2019-07-01 01:55:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|