2019-07-07 07:16:12 +08:00
|
|
|
import {
|
2019-07-10 02:22:18 +08:00
|
|
|
BufferGeometry,
|
2019-07-07 07:16:12 +08:00
|
|
|
DoubleSide,
|
|
|
|
Face3,
|
2019-07-10 02:22:18 +08:00
|
|
|
Float32BufferAttribute,
|
2019-07-07 07:16:12 +08:00
|
|
|
Geometry,
|
|
|
|
Group,
|
2019-07-10 02:22:18 +08:00
|
|
|
Matrix4,
|
2019-07-07 07:16:12 +08:00
|
|
|
Mesh,
|
|
|
|
MeshBasicMaterial,
|
|
|
|
MeshLambertMaterial,
|
|
|
|
Object3D,
|
2019-07-10 02:22:18 +08:00
|
|
|
Uint16BufferAttribute,
|
2019-07-07 07:16:12 +08:00
|
|
|
Vector3,
|
|
|
|
} from "three";
|
|
|
|
import { CollisionObject } from "../data_formats/parsing/area_collision_geometry";
|
2019-07-10 02:22:18 +08:00
|
|
|
import { RenderObject } from "../data_formats/parsing/area_geometry";
|
|
|
|
import { Section } from "../domain";
|
|
|
|
import { xj_model_to_geometry } from "./xj_model_to_geometry";
|
2019-07-07 07:16:12 +08:00
|
|
|
|
|
|
|
const materials = [
|
|
|
|
// Wall
|
|
|
|
new MeshBasicMaterial({
|
|
|
|
color: 0x80c0d0,
|
|
|
|
transparent: true,
|
|
|
|
opacity: 0.25,
|
2019-07-14 00:58:11 +08:00
|
|
|
visible: false,
|
2019-07-07 07:16:12 +08:00
|
|
|
}),
|
|
|
|
// Ground
|
|
|
|
new MeshLambertMaterial({
|
2019-07-14 03:48:18 +08:00
|
|
|
color: 0xa0a0a0,
|
2019-07-07 07:16:12 +08:00
|
|
|
side: DoubleSide,
|
|
|
|
}),
|
|
|
|
// Vegetation
|
|
|
|
new MeshLambertMaterial({
|
|
|
|
color: 0x50b070,
|
|
|
|
side: DoubleSide,
|
|
|
|
}),
|
|
|
|
// Section transition zone
|
|
|
|
new MeshLambertMaterial({
|
|
|
|
color: 0x604080,
|
|
|
|
side: DoubleSide,
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
const wireframe_materials = [
|
|
|
|
// Wall
|
|
|
|
new MeshBasicMaterial({
|
|
|
|
color: 0x90d0e0,
|
|
|
|
wireframe: true,
|
|
|
|
transparent: true,
|
|
|
|
opacity: 0.3,
|
2019-07-14 00:58:11 +08:00
|
|
|
visible: false,
|
2019-07-07 07:16:12 +08:00
|
|
|
}),
|
|
|
|
// Ground
|
|
|
|
new MeshBasicMaterial({
|
2019-07-14 03:48:18 +08:00
|
|
|
color: 0xd0d0d0,
|
2019-07-07 07:16:12 +08:00
|
|
|
wireframe: true,
|
|
|
|
}),
|
|
|
|
// Vegetation
|
|
|
|
new MeshBasicMaterial({
|
2019-07-14 03:48:18 +08:00
|
|
|
color: 0x80e0a0,
|
2019-07-07 07:16:12 +08:00
|
|
|
wireframe: true,
|
|
|
|
}),
|
|
|
|
// Section transition zone
|
|
|
|
new MeshBasicMaterial({
|
2019-07-14 03:48:18 +08:00
|
|
|
color: 0x9070b0,
|
2019-07-07 07:16:12 +08:00
|
|
|
wireframe: true,
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
|
|
|
|
export function area_collision_geometry_to_object_3d(object: CollisionObject): Object3D {
|
|
|
|
const group = new Group();
|
|
|
|
|
|
|
|
for (const collision_mesh of object.meshes) {
|
|
|
|
// Use Geometry and not BufferGeometry for better raycaster performance.
|
|
|
|
const geom = new Geometry();
|
|
|
|
|
|
|
|
for (const { x, y, z } of collision_mesh.vertices) {
|
|
|
|
geom.vertices.push(new Vector3(x, y, z));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const { indices, flags, normal } of collision_mesh.triangles) {
|
|
|
|
const is_section_transition = flags & 0b1000000;
|
|
|
|
const is_vegetation = flags & 0b10000;
|
|
|
|
const is_ground = flags & 0b1;
|
|
|
|
const color_index = is_section_transition ? 3 : is_vegetation ? 2 : is_ground ? 1 : 0;
|
|
|
|
|
|
|
|
geom.faces.push(
|
|
|
|
new Face3(
|
|
|
|
indices[0],
|
|
|
|
indices[1],
|
|
|
|
indices[2],
|
|
|
|
new Vector3(normal.x, normal.y, normal.z),
|
|
|
|
undefined,
|
|
|
|
color_index
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const mesh = new Mesh(geom, materials);
|
|
|
|
mesh.renderOrder = 1;
|
|
|
|
group.add(mesh);
|
|
|
|
|
|
|
|
const wireframe_mesh = new Mesh(geom, wireframe_materials);
|
|
|
|
wireframe_mesh.renderOrder = 2;
|
|
|
|
group.add(wireframe_mesh);
|
|
|
|
}
|
|
|
|
|
|
|
|
return group;
|
|
|
|
}
|
2019-07-10 02:22:18 +08:00
|
|
|
|
|
|
|
export function area_geometry_to_sections_and_object_3d(
|
|
|
|
object: RenderObject
|
|
|
|
): [Section[], Object3D] {
|
|
|
|
const sections: Section[] = [];
|
|
|
|
const group = new Group();
|
|
|
|
|
|
|
|
for (const section of object.sections) {
|
|
|
|
const positions: number[] = [];
|
|
|
|
const normals: number[] = [];
|
|
|
|
const indices: number[] = [];
|
|
|
|
|
|
|
|
for (const model of section.models) {
|
2019-07-13 23:09:28 +08:00
|
|
|
xj_model_to_geometry(model, new Matrix4(), positions, normals, [], indices, []);
|
2019-07-10 02:22:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const geometry = new BufferGeometry();
|
|
|
|
geometry.addAttribute("position", new Float32BufferAttribute(positions, 3));
|
|
|
|
geometry.addAttribute("normal", new Float32BufferAttribute(normals, 3));
|
|
|
|
geometry.setIndex(new Uint16BufferAttribute(indices, 1));
|
|
|
|
|
|
|
|
const mesh = new Mesh(
|
|
|
|
geometry,
|
|
|
|
new MeshLambertMaterial({
|
|
|
|
color: 0x44aaff,
|
|
|
|
transparent: true,
|
2019-07-11 23:30:23 +08:00
|
|
|
opacity: 0.75,
|
2019-07-10 02:22:18 +08:00
|
|
|
side: DoubleSide,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
mesh.position.set(section.position.x, section.position.y, section.position.z);
|
|
|
|
mesh.rotation.set(section.rotation.x, section.rotation.y, section.rotation.z);
|
|
|
|
group.add(mesh);
|
|
|
|
|
|
|
|
const sec = new Section(section.id, section.position, section.rotation.y);
|
|
|
|
mesh.userData.section = sec;
|
|
|
|
sections.push(sec);
|
|
|
|
}
|
|
|
|
|
|
|
|
return [sections, group];
|
|
|
|
}
|