2019-07-03 00:08:06 +08:00
|
|
|
import * as THREE from "three";
|
|
|
|
import {
|
2019-07-11 23:30:23 +08:00
|
|
|
Camera,
|
2019-07-03 00:08:06 +08:00
|
|
|
Color,
|
2019-07-11 23:30:23 +08:00
|
|
|
Group,
|
2019-07-03 00:08:06 +08:00
|
|
|
HemisphereLight,
|
|
|
|
MOUSE,
|
|
|
|
Scene,
|
2019-07-11 23:30:23 +08:00
|
|
|
Vector2,
|
2019-07-03 00:08:06 +08:00
|
|
|
Vector3,
|
|
|
|
WebGLRenderer,
|
|
|
|
} from "three";
|
|
|
|
import OrbitControlsCreator from "three-orbit-controls";
|
2019-05-29 00:40:29 +08:00
|
|
|
|
|
|
|
const OrbitControls = OrbitControlsCreator(THREE);
|
|
|
|
|
2019-07-11 23:30:23 +08:00
|
|
|
export class Renderer<C extends Camera> {
|
2019-07-20 03:49:59 +08:00
|
|
|
protected _debug = false;
|
|
|
|
|
|
|
|
get debug(): boolean {
|
|
|
|
return this._debug;
|
|
|
|
}
|
|
|
|
|
|
|
|
set debug(debug: boolean) {
|
|
|
|
this._debug = debug;
|
|
|
|
}
|
|
|
|
|
2019-07-18 01:37:48 +08:00
|
|
|
readonly camera: C;
|
|
|
|
readonly controls: any;
|
|
|
|
readonly scene = new Scene();
|
|
|
|
readonly light_holder = new Group();
|
2019-05-29 20:46:29 +08:00
|
|
|
|
2019-07-03 06:43:17 +08:00
|
|
|
private renderer = new WebGLRenderer({ antialias: true });
|
|
|
|
private render_scheduled = false;
|
2019-07-14 03:48:18 +08:00
|
|
|
private light = new HemisphereLight(0xffffff, 0x505050, 1.2);
|
2019-07-03 06:43:17 +08:00
|
|
|
|
2019-07-11 23:30:23 +08:00
|
|
|
constructor(camera: C) {
|
|
|
|
this.camera = camera;
|
2019-07-03 06:43:17 +08:00
|
|
|
|
2019-07-18 01:37:48 +08:00
|
|
|
this.controls = new OrbitControls(camera, this.dom_element);
|
2019-05-29 20:46:29 +08:00
|
|
|
this.controls.mouseButtons.ORBIT = MOUSE.RIGHT;
|
|
|
|
this.controls.mouseButtons.PAN = MOUSE.LEFT;
|
2019-07-03 06:43:17 +08:00
|
|
|
this.controls.addEventListener("change", this.schedule_render);
|
2019-05-29 20:46:29 +08:00
|
|
|
|
2019-07-24 00:39:47 +08:00
|
|
|
this.scene.background = new Color(0x181818);
|
2019-07-03 16:53:22 +08:00
|
|
|
this.light_holder.add(this.light);
|
|
|
|
this.scene.add(this.light_holder);
|
2019-07-18 01:37:48 +08:00
|
|
|
|
|
|
|
this.renderer.setPixelRatio(window.devicePixelRatio);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-07-01 01:55:30 +08:00
|
|
|
get dom_element(): HTMLElement {
|
2019-05-29 20:46:29 +08:00
|
|
|
return this.renderer.domElement;
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
set_size(width: number, height: number): void {
|
2019-05-29 20:46:29 +08:00
|
|
|
this.renderer.setSize(width, height);
|
2019-07-03 19:29:07 +08:00
|
|
|
this.schedule_render();
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-07-18 03:59:41 +08:00
|
|
|
pointer_pos_to_device_coords(e: MouseEvent): Vector2 {
|
|
|
|
const coords = this.renderer.getSize(new Vector2());
|
2019-07-18 01:37:48 +08:00
|
|
|
coords.width = (e.offsetX / coords.width) * 2 - 1;
|
|
|
|
coords.height = (e.offsetY / coords.height) * -2 + 1;
|
2019-07-18 03:59:41 +08:00
|
|
|
return coords;
|
2019-07-18 01:37:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
schedule_render = () => {
|
2019-07-03 06:43:17 +08:00
|
|
|
if (!this.render_scheduled) {
|
|
|
|
this.render_scheduled = true;
|
|
|
|
requestAnimationFrame(this.call_render);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-07-18 01:37:48 +08:00
|
|
|
reset_camera(position: Vector3, look_at: Vector3): void {
|
2019-05-29 20:46:29 +08:00
|
|
|
this.controls.reset();
|
2019-07-01 01:55:30 +08:00
|
|
|
this.camera.position.copy(position);
|
|
|
|
this.camera.lookAt(look_at);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-07-18 01:37:48 +08:00
|
|
|
protected render(): void {
|
|
|
|
this.renderer.render(this.scene, this.camera);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-07-03 06:43:17 +08:00
|
|
|
private call_render = () => {
|
|
|
|
this.render_scheduled = false;
|
2019-07-01 01:55:30 +08:00
|
|
|
this.render();
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|