2019-05-29 04:24:28 +08:00
|
|
|
import { BufferGeometry, DoubleSide, Mesh, MeshLambertMaterial } from 'three';
|
2019-05-29 00:40:29 +08:00
|
|
|
import { autorun } from 'mobx';
|
|
|
|
import { Vec3, VisibleQuestEntity, QuestNpc, QuestObject, Section } from '../domain';
|
|
|
|
|
|
|
|
export const OBJECT_COLOR = 0xFFFF00;
|
|
|
|
export const OBJECT_HOVER_COLOR = 0xFFDF3F;
|
|
|
|
export const OBJECT_SELECTED_COLOR = 0xFFAA00;
|
|
|
|
export const NPC_COLOR = 0xFF0000;
|
|
|
|
export const NPC_HOVER_COLOR = 0xFF3F5F;
|
|
|
|
export const NPC_SELECTED_COLOR = 0xFF0054;
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
export function createObjectMesh(object: QuestObject, sections: Section[], geometry: BufferGeometry): Mesh {
|
|
|
|
return createMesh(object, sections, geometry, OBJECT_COLOR, 'Object');
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
export function createNpcMesh(npc: QuestNpc, sections: Section[], geometry: BufferGeometry): Mesh {
|
|
|
|
return createMesh(npc, sections, geometry, NPC_COLOR, 'NPC');
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
function createMesh(
|
2019-05-29 00:40:29 +08:00
|
|
|
entity: VisibleQuestEntity,
|
|
|
|
sections: Section[],
|
|
|
|
geometry: BufferGeometry,
|
|
|
|
color: number,
|
|
|
|
type: string
|
|
|
|
): Mesh {
|
|
|
|
let {x, y, z} = entity.position;
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
const section = sections.find(s => s.id === entity.sectionId);
|
2019-05-29 00:40:29 +08:00
|
|
|
entity.section = section;
|
|
|
|
|
|
|
|
if (section) {
|
2019-05-29 07:37:00 +08:00
|
|
|
const {x: secX, y: secY, z: secZ} = section.position;
|
|
|
|
const rotX = section.cosYAxisRotation * x + section.sinYAxisRotation * z;
|
|
|
|
const rotZ = -section.sinYAxisRotation * x + section.cosYAxisRotation * z;
|
|
|
|
x = rotX + secX;
|
|
|
|
y += secY;
|
|
|
|
z = rotZ + secZ;
|
2019-05-29 00:40:29 +08:00
|
|
|
} else {
|
2019-05-29 07:37:00 +08:00
|
|
|
console.warn(`Section ${entity.sectionId} not found.`);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
const object3d = new Mesh(
|
2019-05-29 04:24:28 +08:00
|
|
|
geometry,
|
2019-05-29 00:40:29 +08:00
|
|
|
new MeshLambertMaterial({
|
|
|
|
color,
|
|
|
|
side: DoubleSide
|
|
|
|
})
|
|
|
|
);
|
2019-05-29 07:37:00 +08:00
|
|
|
object3d.name = type;
|
|
|
|
object3d.userData.entity = entity;
|
2019-05-29 00:40:29 +08:00
|
|
|
|
|
|
|
// TODO: dispose autorun?
|
|
|
|
autorun(() => {
|
|
|
|
const {x, y, z} = entity.position;
|
2019-05-29 07:37:00 +08:00
|
|
|
object3d.position.set(x, y, z);
|
2019-05-29 00:40:29 +08:00
|
|
|
const rot = entity.rotation;
|
2019-05-29 07:37:00 +08:00
|
|
|
object3d.rotation.set(rot.x, rot.y, rot.z);
|
2019-05-29 00:40:29 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
entity.position = new Vec3(x, y, z);
|
|
|
|
|
2019-05-29 07:37:00 +08:00
|
|
|
return object3d;
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|