/* * * Computer-graphics specific implementations. * Currently, it is mostly implemented for homogeneous matrices in 2- and 3-space. * */ use num::One; use core::{Scalar, SquareMatrix, OwnedSquareMatrix, ColumnVector, Unit}; use core::dimension::{DimName, DimNameSub, DimNameDiff, U1, U2, U3, U4}; use core::storage::{Storage, StorageMut, OwnedStorage}; use core::allocator::{Allocator, OwnedAllocator}; use geometry::{PointBase, OrthographicBase, PerspectiveBase, IsometryBase, OwnedRotation, OwnedPoint}; use alga::general::{Real, Field}; use alga::linear::Transformation; impl SquareMatrix where N: Scalar + Field, S: OwnedStorage, S::Alloc: OwnedAllocator { /// Creates a new homogeneous matrix that applies the same scaling factor on each dimension. #[inline] pub fn new_scaling(scaling: N) -> Self { let mut res = Self::from_diagonal_element(scaling); res[(D::dim() - 1, D::dim() - 1)] = N::one(); res } /// Creates a new homogeneous matrix that applies a distinct scaling factor for each dimension. #[inline] pub fn new_nonuniform_scaling(scaling: &ColumnVector, SB>) -> Self where D: DimNameSub, SB: Storage, U1> { let mut res = Self::one(); for i in 0 .. scaling.len() { res[(i, i)] = scaling[i]; } res } /// Creates a new homogeneous matrix that applies a pure translation. #[inline] pub fn new_translation(translation: &ColumnVector, SB>) -> Self where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator, U1> { let mut res = Self::one(); res.fixed_slice_mut::, U1>(0, D::dim() - 1).copy_from(translation); res } } impl SquareMatrix where N: Real, S: OwnedStorage, S::Alloc: OwnedAllocator { /// Builds a 2 dimensional homogeneous rotation matrix from an angle in radian. #[inline] pub fn new_rotation(angle: N) -> Self where S::Alloc: Allocator { OwnedRotation::::new(angle).to_homogeneous() } } impl SquareMatrix where N: Real, S: OwnedStorage, S::Alloc: OwnedAllocator { /// Builds a 3D homogeneous rotation matrix from an axis and an angle (multiplied together). /// /// Returns the identity matrix if the given argument is zero. #[inline] pub fn new_rotation(axisangle: ColumnVector) -> Self where SB: Storage, S::Alloc: Allocator { OwnedRotation::::new(axisangle).to_homogeneous() } /// Builds a 3D homogeneous rotation matrix from an axis and an angle (multiplied together). /// /// Returns the identity matrix if the given argument is zero. #[inline] pub fn new_rotation_wrt_point(axisangle: ColumnVector, pt: OwnedPoint) -> Self where SB: Storage, S::Alloc: Allocator + Allocator + Allocator { let rot = OwnedRotation::::from_scaled_axis(axisangle); IsometryBase::rotation_wrt_point(rot, pt).to_homogeneous() } /// Builds a 3D homogeneous rotation matrix from an axis and an angle (multiplied together). /// /// Returns the identity matrix if the given argument is zero. /// This is identical to `Self::new_rotation`. #[inline] pub fn from_scaled_axis(axisangle: ColumnVector) -> Self where SB: Storage, S::Alloc: Allocator { OwnedRotation::::from_scaled_axis(axisangle).to_homogeneous() } /// Creates a new rotation from Euler angles. /// /// The primitive rotations are applied in order: 1 roll − 2 pitch − 3 yaw. pub fn from_euler_angles(roll: N, pitch: N, yaw: N) -> Self where S::Alloc: Allocator { OwnedRotation::::from_euler_angles(roll, pitch, yaw).to_homogeneous() } /// Builds a 3D homogeneous rotation matrix from an axis and a rotation angle. pub fn from_axis_angle(axis: &Unit>, angle: N) -> Self where SB: Storage, S::Alloc: Allocator { OwnedRotation::::from_axis_angle(axis, angle).to_homogeneous() } /// Creates a new homogeneous matrix for an orthographic projection. #[inline] pub fn new_orthographic(left: N, right: N, bottom: N, top: N, znear: N, zfar: N) -> Self { OrthographicBase::new(left, right, bottom, top, znear, zfar).unwrap() } /// Creates a new homogeneous matrix for a perspective projection. #[inline] pub fn new_perspective(aspect: N, fovy: N, znear: N, zfar: N) -> Self { PerspectiveBase::new(aspect, fovy, znear, zfar).unwrap() } /// Creates an isometry that corresponds to the local frame of an observer standing at the /// point `eye` and looking toward `target`. /// /// It maps the view direction `target - eye` to the positive `z` axis and the origin to the /// `eye`. #[inline] pub fn new_observer_frame(eye: &PointBase, target: &PointBase, up: &ColumnVector) -> Self where SB: OwnedStorage, SB::Alloc: OwnedAllocator + Allocator + Allocator { IsometryBase::> ::new_observer_frame(eye, target, up).to_homogeneous() } /// Builds a right-handed look-at view matrix. #[inline] pub fn look_at_rh(eye: &PointBase, target: &PointBase, up: &ColumnVector) -> Self where SB: OwnedStorage, SB::Alloc: OwnedAllocator + Allocator + Allocator { IsometryBase::> ::look_at_rh(eye, target, up).to_homogeneous() } /// Builds a left-handed look-at view matrix. #[inline] pub fn look_at_lh(eye: &PointBase, target: &PointBase, up: &ColumnVector) -> Self where SB: OwnedStorage, SB::Alloc: OwnedAllocator + Allocator + Allocator { IsometryBase::> ::look_at_lh(eye, target, up).to_homogeneous() } } impl SquareMatrix where N: Scalar + Field, S: Storage { /// Computes the transformation equal to `self` followed by an uniform scaling factor. #[inline] pub fn append_scaling(&self, scaling: N) -> OwnedSquareMatrix where D: DimNameSub, S::Alloc: Allocator, D> { let mut res = self.clone_owned(); res.append_scaling_mut(scaling); res } /// Computes the transformation equal to an uniform scaling factor followed by `self`. #[inline] pub fn prepend_scaling(&self, scaling: N) -> OwnedSquareMatrix where D: DimNameSub, S::Alloc: Allocator> { let mut res = self.clone_owned(); res.prepend_scaling_mut(scaling); res } /// Computes the transformation equal to `self` followed by a non-uniform scaling factor. #[inline] pub fn append_nonuniform_scaling(&self, scaling: &ColumnVector, SB>) -> OwnedSquareMatrix where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator { let mut res = self.clone_owned(); res.append_nonuniform_scaling_mut(scaling); res } /// Computes the transformation equal to a non-uniform scaling factor followed by `self`. #[inline] pub fn prepend_nonuniform_scaling(&self, scaling: &ColumnVector, SB>) -> OwnedSquareMatrix where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator { let mut res = self.clone_owned(); res.prepend_nonuniform_scaling_mut(scaling); res } /// Computes the transformation equal to `self` followed by a translation. #[inline] pub fn append_translation(&self, shift: &ColumnVector, SB>) -> OwnedSquareMatrix where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator, U1> { let mut res = self.clone_owned(); res.append_translation_mut(shift); res } /// Computes the transformation equal to a translation followed by `self`. #[inline] pub fn prepend_translation(&self, shift: &ColumnVector, SB>) -> OwnedSquareMatrix where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator, U1> + Allocator, DimNameDiff> + Allocator> { let mut res = self.clone_owned(); res.prepend_translation_mut(shift); res } } impl SquareMatrix where N: Scalar + Field, S: StorageMut { /// Computes in-place the transformation equal to `self` followed by an uniform scaling factor. #[inline] pub fn append_scaling_mut(&mut self, scaling: N) where D: DimNameSub, S::Alloc: Allocator, D> { let mut to_scale = self.fixed_rows_mut::>(0); to_scale *= scaling; } /// Computes in-place the transformation equal to an uniform scaling factor followed by `self`. #[inline] pub fn prepend_scaling_mut(&mut self, scaling: N) where D: DimNameSub, S::Alloc: Allocator> { let mut to_scale = self.fixed_columns_mut::>(0); to_scale *= scaling; } /// Computes in-place the transformation equal to `self` followed by a non-uniform scaling factor. #[inline] pub fn append_nonuniform_scaling_mut(&mut self, scaling: &ColumnVector, SB>) where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator { for i in 0 .. scaling.len() { let mut to_scale = self.fixed_rows_mut::(i); to_scale *= scaling[i]; } } /// Computes in-place the transformation equal to a non-uniform scaling factor followed by `self`. #[inline] pub fn prepend_nonuniform_scaling_mut(&mut self, scaling: &ColumnVector, SB>) where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator { for i in 0 .. scaling.len() { let mut to_scale = self.fixed_columns_mut::(i); to_scale *= scaling[i]; } } /// Computes the transformation equal to `self` followed by a translation. #[inline] pub fn append_translation_mut(&mut self, shift: &ColumnVector, SB>) where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator, U1> { for i in 0 .. D::dim() { for j in 0 .. D::dim() - 1 { self[(j, i)] += shift[j] * self[(D::dim() - 1, i)]; } } } /// Computes the transformation equal to a translation followed by `self`. #[inline] pub fn prepend_translation_mut(&mut self, shift: &ColumnVector, SB>) where D: DimNameSub, SB: Storage, U1>, S::Alloc: Allocator, U1> + Allocator, DimNameDiff> + Allocator> { let scale = self.fixed_slice::>(D::dim() - 1, 0).tr_dot(&shift); let post_translation = self.fixed_slice::, DimNameDiff>(0, 0) * shift; self[(D::dim() - 1, D::dim() - 1)] += scale; let mut translation = self.fixed_slice_mut::, U1>(0, D::dim() - 1); translation += post_translation; } } impl Transformation, SB>> for SquareMatrix where N: Real, D: DimNameSub, SA: OwnedStorage, SB: OwnedStorage, U1, Alloc = SA::Alloc>, SA::Alloc: OwnedAllocator + Allocator, DimNameDiff> + Allocator, U1> + Allocator>, SB::Alloc: OwnedAllocator, U1, SB> { #[inline] fn transform_vector(&self, v: &ColumnVector, SB>) -> ColumnVector, SB> { let transform = self.fixed_slice::, DimNameDiff>(0, 0); let normalizer = self.fixed_slice::>(D::dim() - 1, 0); let n = normalizer.tr_dot(&v); if !n.is_zero() { return transform * (v / n); } transform * v } #[inline] fn transform_point(&self, pt: &PointBase, SB>) -> PointBase, SB> { let transform = self.fixed_slice::, DimNameDiff>(0, 0); let translation = self.fixed_slice::, U1>(0, D::dim() - 1); let normalizer = self.fixed_slice::>(D::dim() - 1, 0); let n = normalizer.tr_dot(&pt.coords) + unsafe { *self.get_unchecked(D::dim() - 1, D::dim() - 1) }; if !n.is_zero() { return transform * (pt / n) + translation; } transform * pt + translation } }