Fixed perspective transformation.

This commit is contained in:
Daan Vanden Bosch 2020-02-01 15:19:16 +01:00
parent c180b6997e
commit 4eaff297d5
4 changed files with 74 additions and 26 deletions

View File

@ -7,7 +7,9 @@ export enum Projection {
}
export class Camera {
// Only applicable in perspective mode.
/**
* Only applicable in perspective mode.
*/
private readonly fov = deg_to_rad(75);
private readonly position: Vec3 = new Vec3(0, 0, 0);
private readonly look_at: Vec3 = new Vec3(0, 0, 0);
@ -16,16 +18,16 @@ export class Camera {
private z_rot: number = 0;
private _zoom: number = 1;
readonly view_mat4 = Mat4.identity();
readonly projection_mat4 = Mat4.identity();
/**
* Effective field of view in radians. Only applicable in perspective mode.
*/
get effective_fov(): number {
private get effective_fov(): number {
return 2 * Math.atan(Math.tan(0.5 * this.fov) / this._zoom);
}
readonly view_mat4 = Mat4.identity();
readonly projection_mat4 = Mat4.identity();
constructor(
private viewport_width: number,
private viewport_height: number,
@ -43,8 +45,8 @@ export class Camera {
{
const w = width;
const h = height;
const n = -1000;
const f = 1000;
const n = 0;
const f = 100;
// prettier-ignore
this.projection_mat4.set_all(
@ -60,19 +62,18 @@ export class Camera {
{
const aspect = width / height;
const n = 0.1;
const f = 2000;
const t = n * Math.tan(0.5 * this.fov);
const b = -t;
const r = aspect * t;
const l = -r;
const n /* near */ = 0.1;
const f /* far */ = 2000;
const t /* top */ = (n * Math.tan(0.5 * this.fov)) / this._zoom;
const h /* height */ = 2 * t;
const w /* width */ = 2 * aspect * t;
// prettier-ignore
this.projection_mat4.set_all(
2*n / (r-l), 0, (l+r) / (l-r), 0,
0, 2*n / (t-b), (b+t) / (b-t), 0,
0, 0, (f+n) / (n-f), (2*f*n) / (f-n),
0, 0, 1, 0,
2*n / w, 0, 0, 0,
0, 2*n / h, 0, 0,
0, 0, (n+f) / (n-f), 2*n*f / (n-f),
0, 0, -1, 0,
);
}
break;
@ -80,9 +81,21 @@ export class Camera {
}
pan(x: number, y: number, z: number): this {
const pan_factor =
(3 * vec3_dist(this.position, this.look_at) * Math.tan(0.5 * this.effective_fov)) /
this.viewport_width;
let pan_factor: number;
switch (this.projection) {
case Projection.Orthographic:
pan_factor = 1;
break;
case Projection.Perspective:
pan_factor =
(3 *
vec3_dist(this.position, this.look_at) *
Math.tan(0.5 * this.effective_fov)) /
this.viewport_width;
break;
}
x *= pan_factor;
y *= pan_factor;
@ -128,9 +141,9 @@ export class Camera {
}
private update_matrix(): void {
this.view_mat4.data[12] = this._zoom * -this.position.x;
this.view_mat4.data[13] = this._zoom * -this.position.y;
this.view_mat4.data[14] = this._zoom * -this.position.z;
this.view_mat4.data[12] = -this.position.x;
this.view_mat4.data[13] = -this.position.y;
this.view_mat4.data[14] = -this.position.z;
this.view_mat4.data[0] = this._zoom;
this.view_mat4.data[5] = this._zoom;
this.view_mat4.data[10] = this._zoom;

View File

@ -111,9 +111,9 @@ export abstract class GfxRenderer implements Renderer {
case Projection.Perspective:
if (evt.deltaY < 0) {
this.camera.pan(0, 0, 5);
} else {
this.camera.pan(0, 0, -5);
} else {
this.camera.pan(0, 0, 5);
}
break;
}

View File

@ -12,9 +12,42 @@ export class ModelGfxRenderer implements Renderer {
constructor(private readonly store: ModelStore, private readonly renderer: GfxRenderer) {
this.canvas_element = renderer.canvas_element;
renderer.camera.pan(0, 0, -50);
renderer.camera.pan(0, 0, 50);
this.disposer.add_all(store.current_nj_object.observe(this.nj_object_or_xvm_changed));
// TODO: remove
// const cube = cube_mesh();
// cube.upload(this.renderer.gfx);
//
// this.renderer.scene.root_node.add_child(
// new SceneNode(
// undefined,
// Mat4.identity(),
// new SceneNode(
// cube,
// Mat4.compose(
// new Vec3(-3, 0, 0),
// quat_product(
// Quat.euler_angles(Math.PI / 6, 0, 0, EulerOrder.ZYX),
// Quat.euler_angles(0, -Math.PI / 6, 0, EulerOrder.ZYX),
// ),
// new Vec3(1, 1, 1),
// ),
// ),
// new SceneNode(
// cube,
// Mat4.compose(
// new Vec3(3, 0, 0),
// quat_product(
// Quat.euler_angles(-Math.PI / 6, 0, 0, EulerOrder.ZYX),
// Quat.euler_angles(0, Math.PI / 6, 0, EulerOrder.ZYX),
// ),
// new Vec3(1, 1, 1),
// ),
// ),
// ),
// );
}
dispose(): void {

View File

@ -20,6 +20,8 @@ export class TextureRenderer implements Renderer {
constructor(ctrl: TextureController, private readonly renderer: GfxRenderer) {
this.canvas_element = renderer.canvas_element;
renderer.camera.pan(0, 0, 10);
this.disposer.add_all(
ctrl.textures.observe(({ value: textures }) => {
renderer.destroy_scene();