blobt
4 years ago
4 changed files with 544 additions and 2 deletions
-
5src/render/common.ts
-
2src/render/core/Application.ts
-
367src/render/core/Camera.ts
-
170src/render/core/Frustum.ts
@ -0,0 +1,367 @@ |
|||||
|
import { vec3, vec4, mat4, Util } from "../math"; |
||||
|
import { Frustum } from "./Frustum"; |
||||
|
import { ECameraType } from "../common"; |
||||
|
|
||||
|
export class Camera { |
||||
|
/** |
||||
|
* @var 摄像机类型 |
||||
|
*/ |
||||
|
private _type: ECameraType = ECameraType.FLYCAMERA; |
||||
|
|
||||
|
/** |
||||
|
* @var 摄像机位置 |
||||
|
*/ |
||||
|
private _position: vec3 = new vec3(); |
||||
|
|
||||
|
/** |
||||
|
* @var x轴 |
||||
|
*/ |
||||
|
private _xAxis: vec3 = new vec3([1, 0, 0]); |
||||
|
|
||||
|
/** |
||||
|
* @var y轴 |
||||
|
*/ |
||||
|
private _yAxis: vec3 = new vec3([0, 1, 0]); |
||||
|
|
||||
|
/** |
||||
|
* @var z轴 |
||||
|
*/ |
||||
|
private _zAxis: vec3 = new vec3([0, 0, 1]); |
||||
|
|
||||
|
/** |
||||
|
* @var 近剪裁面 |
||||
|
*/ |
||||
|
private _near: number; |
||||
|
|
||||
|
/** |
||||
|
* @var 远剪裁面 |
||||
|
*/ |
||||
|
private _far: number; |
||||
|
|
||||
|
/** |
||||
|
* @var |
||||
|
*/ |
||||
|
private _left: number; |
||||
|
|
||||
|
|
||||
|
/** |
||||
|
* @var |
||||
|
*/ |
||||
|
private _right: number; |
||||
|
|
||||
|
/** |
||||
|
* @var |
||||
|
*/ |
||||
|
private _bottom: number; |
||||
|
|
||||
|
/** |
||||
|
* @var |
||||
|
*/ |
||||
|
private _top: number; |
||||
|
|
||||
|
/** |
||||
|
* @var fov角度 |
||||
|
*/ |
||||
|
private _fovY: number; |
||||
|
|
||||
|
/** |
||||
|
* @var 宽高比 |
||||
|
*/ |
||||
|
private _aspectRatio: number; |
||||
|
|
||||
|
/** |
||||
|
* @var 投影矩阵 |
||||
|
*/ |
||||
|
private _projectionMatrix: mat4; |
||||
|
|
||||
|
/** |
||||
|
* @var 视口矩阵 |
||||
|
*/ |
||||
|
private _viewMatrix: mat4; |
||||
|
|
||||
|
/** |
||||
|
* @var 视口矩阵逆矩阵 |
||||
|
*/ |
||||
|
private _invViewMatrix: mat4; |
||||
|
|
||||
|
/** |
||||
|
* @var 视口投影矩阵 |
||||
|
*/ |
||||
|
private _viewProjMatrix: mat4; |
||||
|
|
||||
|
/** |
||||
|
* @var 视口投影矩阵逆矩阵 |
||||
|
*/ |
||||
|
private _invViewProjMatrix: mat4; |
||||
|
|
||||
|
/** |
||||
|
* @var 视锥体 |
||||
|
*/ |
||||
|
private _frustum: Frustum; |
||||
|
|
||||
|
|
||||
|
/** |
||||
|
* @var 渲染上下文 |
||||
|
*/ |
||||
|
public gl: WebGLRenderingContext; |
||||
|
|
||||
|
/** |
||||
|
* @var |
||||
|
*/ |
||||
|
public controlByMouse: boolean; |
||||
|
|
||||
|
public constructor(gl: WebGLRenderingContext, width: number, height: number, fovY: number = 45.0, zNear: number = 1, zFar: number = 1000) { |
||||
|
this.gl = gl; |
||||
|
this._aspectRatio = width / height; |
||||
|
this._fovY = Util.toRadian(fovY); |
||||
|
|
||||
|
this._near = zNear; |
||||
|
this._far = zFar; |
||||
|
|
||||
|
this._top = this._near * Math.tan(this._fovY * 0.5); |
||||
|
this._right = this._top * this._aspectRatio; |
||||
|
this._bottom = -this._top; |
||||
|
this._left = -this._right; |
||||
|
this._frustum = new Frustum(); |
||||
|
|
||||
|
this._projectionMatrix = new mat4(); |
||||
|
this._viewMatrix = new mat4(); |
||||
|
this._invViewMatrix = new mat4(); |
||||
|
this._viewProjMatrix = new mat4(); |
||||
|
this._invViewProjMatrix = new mat4(); |
||||
|
|
||||
|
this.controlByMouse = false; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取fov |
||||
|
*/ |
||||
|
public get fovY(): number { |
||||
|
return this._fovY; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置fov |
||||
|
*/ |
||||
|
public set fovY(value: number) { |
||||
|
this._fovY = value; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取近剪裁面 |
||||
|
*/ |
||||
|
public get near(): number { |
||||
|
return this._near |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置近剪裁面 |
||||
|
*/ |
||||
|
public set near(value: number) { |
||||
|
this._near = value; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取远剪裁面 |
||||
|
*/ |
||||
|
public get far(): number { |
||||
|
return this._far |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置远剪裁面 |
||||
|
*/ |
||||
|
public set far(value: number) { |
||||
|
this._far = value; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取宽高比 |
||||
|
*/ |
||||
|
public get aspectRatio(): number { |
||||
|
return this._aspectRatio; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置宽高比 |
||||
|
*/ |
||||
|
public set aspectRatio(value: number) { |
||||
|
this._aspectRatio = value; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取相机位置 |
||||
|
*/ |
||||
|
public get position(): vec3 { |
||||
|
return this._position; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置相机位置 |
||||
|
*/ |
||||
|
public set position(value: vec3) { |
||||
|
this._position = value; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置视口 |
||||
|
* @param x |
||||
|
* @param y |
||||
|
* @param width |
||||
|
* @param height |
||||
|
*/ |
||||
|
public setViewport(x: number, y: number, width: number, height: number): void { |
||||
|
this.gl.viewport(x, y, width, height); |
||||
|
this.gl.scissor(x, y, width, height); |
||||
|
this.aspectRatio = width / height; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取视口 |
||||
|
* @returns |
||||
|
*/ |
||||
|
public getViewport(): Int32Array { |
||||
|
return this.gl.getParameter(this.gl.VIEWPORT); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置相机位置 |
||||
|
*/ |
||||
|
public set x(value: number) { |
||||
|
this._position.x = value; |
||||
|
} |
||||
|
|
||||
|
public set y(value: number) { |
||||
|
this._position.y = value; |
||||
|
} |
||||
|
|
||||
|
public set z(value: number) { |
||||
|
this._position.z = value; |
||||
|
} |
||||
|
|
||||
|
public get x(): number { |
||||
|
return this._position.x; |
||||
|
} |
||||
|
|
||||
|
public get y(): number { |
||||
|
return this._position.y; |
||||
|
} |
||||
|
|
||||
|
public get z(): number { |
||||
|
return this._position.z; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取x轴 |
||||
|
*/ |
||||
|
public get xAxis(): vec3 { |
||||
|
return this._xAxis; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取y轴 |
||||
|
*/ |
||||
|
public get yAxis(): vec3 { |
||||
|
return this._yAxis; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取z轴 |
||||
|
*/ |
||||
|
public get zAxis(): vec3 { |
||||
|
return this._zAxis; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取相机类型 |
||||
|
*/ |
||||
|
public get type(): ECameraType { |
||||
|
return this._type; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 设置相机类型 |
||||
|
*/ |
||||
|
public set type(value: ECameraType) { |
||||
|
this._type = value; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取相机 近剪裁面的 left right bottom top |
||||
|
*/ |
||||
|
public get left(): number { |
||||
|
return this._left; |
||||
|
} |
||||
|
|
||||
|
public get right(): number { |
||||
|
return this._right; |
||||
|
} |
||||
|
|
||||
|
public get bottom(): number { |
||||
|
return this._bottom; |
||||
|
} |
||||
|
|
||||
|
public get top(): number { |
||||
|
return this._top; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 获取世界坐标逆矩阵 |
||||
|
*/ |
||||
|
public get invViewMatrix(): mat4 { |
||||
|
return this._invViewMatrix; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 更新相机状态 |
||||
|
* @param intervaleSec 更新间隔 |
||||
|
*/ |
||||
|
public update(intervaleSec: number): void { |
||||
|
this._projectionMatrix = mat4.perspective(this.fovY, this.aspectRatio, this.near, this.far); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 从当前轴以及postion合成view矩阵 |
||||
|
*/ |
||||
|
private _calcViewMatrix(): void { |
||||
|
// 固定forward方向
|
||||
|
this._zAxis.normalize(); |
||||
|
|
||||
|
// 获取up方向
|
||||
|
vec3.cross(this._zAxis, this._xAxis, this._yAxis); |
||||
|
this._yAxis.normalize(); |
||||
|
|
||||
|
// 获取right方向
|
||||
|
vec3.cross(this._yAxis, this._zAxis, this._xAxis); |
||||
|
this._xAxis.normalize(); |
||||
|
|
||||
|
let x: number = -vec3.dot(this._xAxis, this._position); |
||||
|
let y: number = -vec3.dot(this._yAxis, this._position); |
||||
|
let z: number = -vec3.dot(this._zAxis, this._position); |
||||
|
|
||||
|
this._viewMatrix.values[0] = this._xAxis.x; |
||||
|
this._viewMatrix.values[1] = this._yAxis.x; |
||||
|
this._viewMatrix.values[2] = this._zAxis.x; |
||||
|
this._viewMatrix.values[3] = 0.0; |
||||
|
|
||||
|
this._viewMatrix.values[4] = this._xAxis.y; |
||||
|
this._viewMatrix.values[5] = this._yAxis.y; |
||||
|
this._viewMatrix.values[6] = this._zAxis.y; |
||||
|
this._viewMatrix.values[7] = 0.0; |
||||
|
|
||||
|
this._viewMatrix.values[8] = this._xAxis.z; |
||||
|
this._viewMatrix.values[9] = this._yAxis.z; |
||||
|
this._viewMatrix.values[10] = this._zAxis.z; |
||||
|
this._viewMatrix.values[11] = 0.0; |
||||
|
|
||||
|
this._viewMatrix.values[12] = x; |
||||
|
this._viewMatrix.values[13] = y; |
||||
|
this._viewMatrix.values[14] = z; |
||||
|
this._viewMatrix.values[15] = 1.0; |
||||
|
|
||||
|
//生成view的逆矩阵,其实就是世界矩阵
|
||||
|
this._viewMatrix.inverse(this._invViewMatrix); |
||||
|
|
||||
|
//TODO this._frustum.buildFromCamera(this);
|
||||
|
} |
||||
|
} |
@ -0,0 +1,170 @@ |
|||||
|
import { vec3, vec4, Util } from "../math"; |
||||
|
import { Camera } from "./Camera"; |
||||
|
|
||||
|
export class Frustum { |
||||
|
/** |
||||
|
* @var 原点桌标 |
||||
|
*/ |
||||
|
private _origin: vec3; |
||||
|
|
||||
|
/** |
||||
|
* @var 视锥体的8个顶点, 0~3是近剪裁面, 4~7是远剪裁面 |
||||
|
*/ |
||||
|
private _points: vec3[]; |
||||
|
|
||||
|
/** |
||||
|
* @var 视锥体的上,下,左,右,远,近 6个平面,其法向量都是朝视锥体内部 |
||||
|
*/ |
||||
|
private _planes: vec4[]; |
||||
|
|
||||
|
public constructor(origin: vec3 | null = null, points8: vec3[] | null = null) { |
||||
|
//预先内存分配8个点
|
||||
|
if (origin !== null) { |
||||
|
this._origin = origin; |
||||
|
} else { |
||||
|
this._origin = new vec3(); |
||||
|
} |
||||
|
if (points8 !== null && points8.length === 8) { |
||||
|
this._points = points8; |
||||
|
} else { |
||||
|
this._points = new Array(8); |
||||
|
for (let i = 0; i < this._points.length; i++) { |
||||
|
this._points[i] = new vec3(); |
||||
|
} |
||||
|
} |
||||
|
this._planes = new Array(6); |
||||
|
for (let i = 0; i < this._planes.length; i++) { |
||||
|
this._planes[i] = new vec4(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 从摄像机参数建立视锥体 |
||||
|
* @param camera |
||||
|
*/ |
||||
|
public buildFromCamera(camera: Camera): void { |
||||
|
|
||||
|
let left: number = camera.left * camera.far / camera.near; |
||||
|
let right: number = camera.right * camera.far / camera.near; |
||||
|
let bottom: number = camera.bottom * camera.far / camera.near; |
||||
|
let top: number = camera.top * camera.far / camera.near; |
||||
|
|
||||
|
//计算近剪裁面的4个点
|
||||
|
this.points[0].x = left; |
||||
|
this.points[0].y = bottom; |
||||
|
this.points[0].z = -camera.near; |
||||
|
this.points[1].x = right; |
||||
|
this.points[1].y = bottom; |
||||
|
this.points[1].z = -camera.near; |
||||
|
this.points[2].x = right; |
||||
|
this.points[2].y = top; |
||||
|
this.points[2].z = -camera.near; |
||||
|
this.points[3].x = left; |
||||
|
this.points[3].y = top; |
||||
|
this.points[3].z = -camera.near; |
||||
|
|
||||
|
//计算远剪裁面的4个点
|
||||
|
this.points[4].x = left; |
||||
|
this.points[4].y = bottom; |
||||
|
this.points[4].z = -camera.far; |
||||
|
this.points[5].x = right; |
||||
|
this.points[5].y = bottom; |
||||
|
this.points[5].z = -camera.far; |
||||
|
this.points[6].x = right; |
||||
|
this.points[6].y = top; |
||||
|
this.points[6].z = -camera.far; |
||||
|
this.points[7].x = left; |
||||
|
this.points[7].y = top; |
||||
|
this.points[7].z = -camera.far; |
||||
|
|
||||
|
//将摄像机的原点变换到世界坐标系
|
||||
|
this._origin.x = this._origin.y = this._origin.z = 0; |
||||
|
this._origin = camera.invViewMatrix.multiplyVec3(this.origin); |
||||
|
|
||||
|
//将8个顶点的坐标系变换都世界坐标系
|
||||
|
for (let i = 0; i < this._points.length; i++) { |
||||
|
this._points[i] = camera.invViewMatrix.multiplyVec3(this.points[i]); |
||||
|
} |
||||
|
|
||||
|
//构建世界坐标系表示的6个平面,法线朝内
|
||||
|
Util.planeFromPoints(this._origin, this._points[0], this._points[3], this._planes[0]); |
||||
|
Util.planeFromPoints(this._origin, this._points[2], this._points[1], this._planes[1]); |
||||
|
Util.planeFromPoints(this._origin, this._points[3], this._points[2], this._planes[2]); |
||||
|
Util.planeFromPoints(this._origin, this._points[1], this._points[0], this._planes[3]); |
||||
|
Util.planeFromPoints(this._points[0], this._points[2], this._points[1], this._planes[4]); |
||||
|
Util.planeFromPoints(this._points[5], this._points[7], this._points[4], this._planes[5]); |
||||
|
//平面单位化
|
||||
|
for (let i: number = 0; i < this._planes.length; i++) { |
||||
|
Util.planeNormalize(this._planes[i]); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 判定6个平面与中心点距离???? |
||||
|
* @param center |
||||
|
* @param radius |
||||
|
* @returns |
||||
|
*/ |
||||
|
public isSphereVisible(center: vec3, radius: number): boolean { |
||||
|
radius = -radius; |
||||
|
for (let i: number = 0; i < this._planes.length; i++) { |
||||
|
if (Util.planeDistanceFromPoint(this._planes[i], center) < radius) { |
||||
|
return false |
||||
|
} |
||||
|
} |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* ????? |
||||
|
* @param mins |
||||
|
* @param maxs |
||||
|
* @returns |
||||
|
*/ |
||||
|
public isBoundBoxVisible(mins: vec3, maxs: vec3): boolean { |
||||
|
for (let i: number = 0; i < this._planes.length; i++) { |
||||
|
let current: vec4 = this._planes[i]; |
||||
|
vec3.v0.x = (current.x > 0.0) ? maxs.x : mins.x; |
||||
|
vec3.v0.y = (current.y > 0.0) ? maxs.y : mins.x; |
||||
|
vec3.v0.z = (current.z > 0.0) ? maxs.z : mins.z; |
||||
|
if (Util.planeDistanceFromPoint(current, vec3.v0) < 0.0) { |
||||
|
return false; |
||||
|
} |
||||
|
} |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* ????? |
||||
|
* @param a |
||||
|
* @param b |
||||
|
* @param c |
||||
|
* @returns |
||||
|
*/ |
||||
|
public isTriangleVisible(a: vec3, b: vec3, c: vec3): boolean { |
||||
|
for (let i: number = 0; i < this._planes.length; i++) { |
||||
|
let current: vec4 = this._planes[i]; |
||||
|
if (Util.planeDistanceFromPoint(current, a) >= 0.0) { |
||||
|
continue; |
||||
|
} |
||||
|
if (Util.planeDistanceFromPoint(current, b) >= 0.0) { |
||||
|
continue; |
||||
|
} |
||||
|
if (Util.planeDistanceFromPoint(current, c) >= 0.0) { |
||||
|
continue; |
||||
|
} |
||||
|
} |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
|
||||
|
public get origin(): vec3 { |
||||
|
return this._origin; |
||||
|
} |
||||
|
public get points(): vec3[] { |
||||
|
return this._points; |
||||
|
} |
||||
|
public get planes(): vec4[] { |
||||
|
return this._planes; |
||||
|
} |
||||
|
} |
Reference in new issue
xxxxxxxxxx