phantasmal-world/src/core/rendering/Renderer.ts
2019-09-19 22:20:17 +02:00

134 lines
3.6 KiB
TypeScript

import CameraControls from "camera-controls";
import * as THREE from "three";
import {
Camera,
Clock,
Color,
Group,
HemisphereLight,
OrthographicCamera,
PerspectiveCamera,
Scene,
Vector2,
Vector3,
WebGLRenderer,
} from "three";
import { Disposable } from "../observable/Disposable";
CameraControls.install({
// Hack to make panning and orbiting work the way we want.
THREE: {
...THREE,
MOUSE: { ...THREE.MOUSE, LEFT: THREE.MOUSE.RIGHT, RIGHT: THREE.MOUSE.LEFT },
},
});
export abstract class Renderer implements Disposable {
private _debug = false;
get debug(): boolean {
return this._debug;
}
set debug(debug: boolean) {
this._debug = debug;
}
readonly camera: Camera;
readonly controls: CameraControls;
readonly scene = new Scene();
readonly light_holder = new Group();
private renderer = new WebGLRenderer({ antialias: true });
private render_scheduled = false;
private animation_frame_handle?: number = undefined;
private light = new HemisphereLight(0xffffff, 0x505050, 1.2);
private controls_clock = new Clock();
protected constructor(camera: PerspectiveCamera | OrthographicCamera) {
this.camera = camera;
this.dom_element.tabIndex = 0;
this.dom_element.addEventListener("mousedown", this.on_mouse_down);
this.dom_element.style.outline = "none";
this.controls = new CameraControls(camera, this.renderer.domElement);
this.controls.dampingFactor = 1;
this.controls.draggingDampingFactor = 1;
this.scene.background = new Color(0x181818);
this.light_holder.add(this.light);
this.scene.add(this.light_holder);
this.renderer.setPixelRatio(window.devicePixelRatio);
}
get dom_element(): HTMLElement {
return this.renderer.domElement;
}
set_size(width: number, height: number): void {
this.renderer.setSize(width, height);
this.schedule_render();
}
pointer_pos_to_device_coords(v: Vector2): Vector2 {
const coords = this.renderer.getSize(new Vector2());
coords.width = (v.x / coords.width) * 2 - 1;
coords.height = (v.y / coords.height) * -2 + 1;
return coords;
}
start_rendering(): void {
this.schedule_render();
this.animation_frame_handle = requestAnimationFrame(this.call_render);
}
stop_rendering(): void {
if (this.animation_frame_handle != undefined) {
cancelAnimationFrame(this.animation_frame_handle);
this.animation_frame_handle = undefined;
}
}
schedule_render = () => {
this.render_scheduled = true;
};
reset_camera(position: Vector3, look_at: Vector3): void {
this.controls.setLookAt(
position.x,
position.y,
position.z,
look_at.x,
look_at.y,
look_at.z,
);
}
dispose(): void {
this.renderer.dispose();
}
protected render(): void {
this.renderer.render(this.scene, this.camera);
}
private on_mouse_down = (e: Event) => {
if (e.currentTarget) (e.currentTarget as HTMLElement).focus();
};
private call_render = () => {
const controls_updated = this.controls.update(this.controls_clock.getDelta());
const should_render = this.render_scheduled || controls_updated;
this.render_scheduled = false;
if (should_render) {
this.render();
}
this.animation_frame_handle = requestAnimationFrame(this.call_render);
};
}