2014-10-13 02:21:06 +08:00
|
|
|
use std::num;
|
2014-11-15 22:47:59 +08:00
|
|
|
use traits::structure::BaseFloat;
|
2014-10-13 02:21:06 +08:00
|
|
|
use structs::{Pnt3, Vec3, Mat4};
|
|
|
|
|
|
|
|
/// A 3D orthographic projection stored without any matrix.
|
|
|
|
///
|
|
|
|
/// Reading or modifying its individual properties is cheap but applying the transformation is costly.
|
|
|
|
#[deriving(Eq, PartialEq, Encodable, Decodable, Clone, Show)]
|
|
|
|
pub struct Ortho3<N> {
|
|
|
|
width: N,
|
|
|
|
height: N,
|
|
|
|
znear: N,
|
|
|
|
zfar: N
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A 3D orthographic projection stored as a 4D matrix.
|
|
|
|
///
|
|
|
|
/// Reading or modifying its individual properties is costly but applying the transformation is cheap.
|
|
|
|
#[deriving(Eq, PartialEq, Encodable, Decodable, Clone, Show)]
|
|
|
|
pub struct OrthoMat3<N> {
|
|
|
|
mat: Mat4<N>
|
|
|
|
}
|
|
|
|
|
2014-11-15 22:47:59 +08:00
|
|
|
impl<N: BaseFloat> Ortho3<N> {
|
2014-10-13 02:21:06 +08:00
|
|
|
/// Creates a new 3D orthographic projection.
|
|
|
|
pub fn new(width: N, height: N, znear: N, zfar: N) -> Ortho3<N> {
|
2014-11-16 21:04:15 +08:00
|
|
|
assert!(!::is_zero(&(zfar - znear)));
|
|
|
|
assert!(!::is_zero(&width));
|
|
|
|
assert!(!::is_zero(&height));
|
2014-10-13 02:21:06 +08:00
|
|
|
|
|
|
|
Ortho3 {
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
znear: znear,
|
|
|
|
zfar: zfar
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Builds a 4D projection matrix (using homogeneous coordinates) for this projection.
|
|
|
|
pub fn to_mat(&self) -> Mat4<N> {
|
|
|
|
self.to_persp_mat().mat
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Build a `OrthoMat3` representing this projection.
|
|
|
|
pub fn to_persp_mat(&self) -> OrthoMat3<N> {
|
|
|
|
OrthoMat3::new(self.width, self.height, self.znear, self.zfar)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-15 22:47:59 +08:00
|
|
|
impl<N: BaseFloat + Clone> Ortho3<N> {
|
2014-10-13 02:21:06 +08:00
|
|
|
/// The width of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn width(&self) -> N {
|
|
|
|
self.width.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The height of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn height(&self) -> N {
|
|
|
|
self.height.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The near plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn znear(&self) -> N {
|
|
|
|
self.znear.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The far plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn zfar(&self) -> N {
|
|
|
|
self.zfar.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the width of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_width(&mut self, width: N) {
|
|
|
|
self.width = width
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the height of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_height(&mut self, height: N) {
|
|
|
|
self.height = height
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the near plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_znear(&mut self, znear: N) {
|
|
|
|
self.znear = znear
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the far plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_zfar(&mut self, zfar: N) {
|
|
|
|
self.zfar = zfar
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Projects a point.
|
|
|
|
#[inline]
|
|
|
|
pub fn project_pnt(&self, p: &Pnt3<N>) -> Pnt3<N> {
|
|
|
|
// FIXME: optimize that
|
|
|
|
self.to_persp_mat().project_pnt(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Projects a vector.
|
|
|
|
#[inline]
|
|
|
|
pub fn project_vec(&self, p: &Vec3<N>) -> Vec3<N> {
|
|
|
|
// FIXME: optimize that
|
|
|
|
self.to_persp_mat().project_vec(p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-15 22:47:59 +08:00
|
|
|
impl<N: BaseFloat> OrthoMat3<N> {
|
2014-10-13 02:21:06 +08:00
|
|
|
/// Creates a new orthographic projection matrix from the width, heihgt, znear and zfar planes of the view cuboid.
|
|
|
|
pub fn new(width: N, height: N, znear: N, zfar: N) -> OrthoMat3<N> {
|
2014-11-16 21:04:15 +08:00
|
|
|
assert!(!::is_zero(&(zfar - znear)));
|
|
|
|
assert!(!::is_zero(&width));
|
|
|
|
assert!(!::is_zero(&height));
|
2014-10-13 02:21:06 +08:00
|
|
|
|
2014-11-16 21:04:15 +08:00
|
|
|
let mat: Mat4<N> = ::one();
|
2014-10-13 02:21:06 +08:00
|
|
|
|
|
|
|
let mut res = OrthoMat3 { mat: mat };
|
|
|
|
res.set_width(width);
|
|
|
|
res.set_height(height);
|
|
|
|
res.set_znear_and_zfar(znear, zfar);
|
|
|
|
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a new orthographic matrix from a 4D matrix.
|
|
|
|
///
|
|
|
|
/// This is unsafe because the input matrix is not checked to be a orthographic projection.
|
|
|
|
#[inline]
|
|
|
|
pub unsafe fn new_with_mat(mat: Mat4<N>) -> OrthoMat3<N> {
|
|
|
|
OrthoMat3 {
|
|
|
|
mat: mat
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a reference to the 4D matrix (using homogeneous coordinates) of this projection.
|
|
|
|
#[inline]
|
|
|
|
pub fn as_mat<'a>(&'a self) -> &'a Mat4<N> {
|
|
|
|
&self.mat
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The width of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn width(&self) -> N {
|
|
|
|
num::cast::<f64, N>(2.0).unwrap() / self.mat.m11
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The height of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn height(&self) -> N {
|
|
|
|
num::cast::<f64, N>(2.0).unwrap() / self.mat.m22
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The near plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn znear(&self) -> N {
|
2014-11-16 21:04:15 +08:00
|
|
|
(self.mat.m34 + ::one()) / self.mat.m33
|
2014-10-13 02:21:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// The far plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn zfar(&self) -> N {
|
2014-11-16 21:04:15 +08:00
|
|
|
(self.mat.m34 - ::one()) / self.mat.m33
|
2014-10-13 02:21:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the width of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_width(&mut self, width: N) {
|
2014-11-16 21:04:15 +08:00
|
|
|
assert!(!::is_zero(&width));
|
2014-10-13 02:21:06 +08:00
|
|
|
self.mat.m11 = num::cast::<f64, N>(2.0).unwrap() / width;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the height of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_height(&mut self, height: N) {
|
2014-11-16 21:04:15 +08:00
|
|
|
assert!(!::is_zero(&height));
|
2014-10-13 02:21:06 +08:00
|
|
|
self.mat.m22 = num::cast::<f64, N>(2.0).unwrap() / height;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the near plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_znear(&mut self, znear: N) {
|
|
|
|
let zfar = self.zfar();
|
|
|
|
self.set_znear_and_zfar(znear, zfar);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the far plane offset of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_zfar(&mut self, zfar: N) {
|
|
|
|
let znear = self.znear();
|
|
|
|
self.set_znear_and_zfar(znear, zfar);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the near and far plane offsets of the view cuboid.
|
|
|
|
#[inline]
|
|
|
|
pub fn set_znear_and_zfar(&mut self, znear: N, zfar: N) {
|
2014-11-16 21:04:15 +08:00
|
|
|
assert!(!::is_zero(&(zfar - znear)));
|
2014-10-13 02:21:06 +08:00
|
|
|
self.mat.m33 = -num::cast::<f64, N>(2.0).unwrap() / (zfar - znear);
|
|
|
|
self.mat.m34 = -(zfar + znear) / (zfar - znear);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Projects a point.
|
|
|
|
#[inline]
|
|
|
|
pub fn project_pnt(&self, p: &Pnt3<N>) -> Pnt3<N> {
|
|
|
|
Pnt3::new(
|
|
|
|
self.mat.m11 * p.x,
|
|
|
|
self.mat.m22 * p.y,
|
|
|
|
self.mat.m33 * p.z + self.mat.m34
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Projects a vector.
|
|
|
|
#[inline]
|
|
|
|
pub fn project_vec(&self, p: &Vec3<N>) -> Vec3<N> {
|
|
|
|
Vec3::new(
|
|
|
|
self.mat.m11 * p.x,
|
|
|
|
self.mat.m22 * p.y,
|
|
|
|
self.mat.m33 * p.z
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-15 22:47:59 +08:00
|
|
|
impl<N: BaseFloat + Clone> OrthoMat3<N> {
|
2014-10-13 02:21:06 +08:00
|
|
|
/// Returns the 4D matrix (using homogeneous coordinates) of this projection.
|
|
|
|
#[inline]
|
|
|
|
pub fn to_mat<'a>(&'a self) -> Mat4<N> {
|
|
|
|
self.mat.clone()
|
|
|
|
}
|
|
|
|
}
|