2018-02-02 19:26:35 +08:00
|
|
|
#[cfg(feature = "arbitrary")]
|
2016-12-05 05:44:42 +08:00
|
|
|
use quickcheck::{Arbitrary, Gen};
|
2021-03-02 19:25:12 +08:00
|
|
|
#[cfg(feature = "rand-no-std")]
|
|
|
|
use rand::{
|
|
|
|
distributions::{Distribution, Standard},
|
|
|
|
Rng,
|
|
|
|
};
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2021-04-12 18:14:16 +08:00
|
|
|
#[cfg(feature = "serde-serialize-no-std")]
|
2018-10-13 16:25:34 +08:00
|
|
|
use serde::{Deserialize, Deserializer, Serialize, Serializer};
|
2017-08-03 01:37:44 +08:00
|
|
|
use std::fmt;
|
2018-09-22 22:02:55 +08:00
|
|
|
use std::mem;
|
2017-05-04 10:02:30 +08:00
|
|
|
|
2020-03-21 19:16:46 +08:00
|
|
|
use simba::scalar::RealField;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
use crate::base::dimension::U3;
|
|
|
|
use crate::base::storage::Storage;
|
|
|
|
use crate::base::{Matrix4, Scalar, Vector, Vector3};
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
use crate::geometry::{Point3, Projective3};
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2020-02-23 23:30:11 +08:00
|
|
|
/// A 3D perspective projection stored as a homogeneous 4x4 matrix.
|
2021-04-11 17:00:38 +08:00
|
|
|
pub struct Perspective3<T: Scalar> {
|
|
|
|
matrix: Matrix4<T>,
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField> Copy for Perspective3<T> {}
|
2017-08-03 01:37:44 +08:00
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField> Clone for Perspective3<T> {
|
2017-08-03 01:37:44 +08:00
|
|
|
#[inline]
|
|
|
|
fn clone(&self) -> Self {
|
2020-11-19 19:55:15 +08:00
|
|
|
Self::from_matrix_unchecked(self.matrix)
|
2017-05-04 10:02:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField> fmt::Debug for Perspective3<T> {
|
2017-08-03 01:37:44 +08:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
|
|
self.matrix.fmt(f)
|
2017-05-04 10:02:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField> PartialEq for Perspective3<T> {
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
|
|
|
fn eq(&self, right: &Self) -> bool {
|
|
|
|
self.matrix == right.matrix
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-12 18:14:16 +08:00
|
|
|
#[cfg(feature = "serde-serialize-no-std")]
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField + Serialize> Serialize for Perspective3<T> {
|
2017-08-03 01:37:44 +08:00
|
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
2020-04-06 00:49:48 +08:00
|
|
|
where
|
|
|
|
S: Serializer,
|
|
|
|
{
|
2018-02-02 19:26:35 +08:00
|
|
|
self.matrix.serialize(serializer)
|
|
|
|
}
|
2017-08-03 01:37:44 +08:00
|
|
|
}
|
|
|
|
|
2021-04-12 18:14:16 +08:00
|
|
|
#[cfg(feature = "serde-serialize-no-std")]
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<'a, T: RealField + Deserialize<'a>> Deserialize<'a> for Perspective3<T> {
|
2017-08-03 01:37:44 +08:00
|
|
|
fn deserialize<Des>(deserializer: Des) -> Result<Self, Des::Error>
|
2020-04-06 00:49:48 +08:00
|
|
|
where
|
|
|
|
Des: Deserializer<'a>,
|
|
|
|
{
|
2021-04-11 17:00:38 +08:00
|
|
|
let matrix = Matrix4::<T>::deserialize(deserializer)?;
|
2017-08-03 01:37:44 +08:00
|
|
|
|
2019-02-17 05:29:41 +08:00
|
|
|
Ok(Self::from_matrix_unchecked(matrix))
|
2018-02-02 19:26:35 +08:00
|
|
|
}
|
2017-08-03 01:37:44 +08:00
|
|
|
}
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField> Perspective3<T> {
|
2016-12-05 05:44:42 +08:00
|
|
|
/// Creates a new perspective matrix from the aspect ratio, y field of view, and near/far planes.
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn new(aspect: T, fovy: T, znear: T, zfar: T) -> Self {
|
2018-02-02 19:26:35 +08:00
|
|
|
assert!(
|
2021-04-11 17:00:38 +08:00
|
|
|
!relative_eq!(zfar - znear, T::zero()),
|
2018-02-02 19:26:35 +08:00
|
|
|
"The near-plane and far-plane must not be superimposed."
|
|
|
|
);
|
|
|
|
assert!(
|
2021-04-11 17:00:38 +08:00
|
|
|
!relative_eq!(aspect, T::zero()),
|
2020-10-11 16:41:25 +08:00
|
|
|
"The aspect ratio must not be zero."
|
2018-02-02 19:26:35 +08:00
|
|
|
);
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2017-08-03 01:37:44 +08:00
|
|
|
let matrix = Matrix4::identity();
|
2019-02-17 05:29:41 +08:00
|
|
|
let mut res = Self::from_matrix_unchecked(matrix);
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
res.set_fovy(fovy);
|
|
|
|
res.set_aspect(aspect);
|
|
|
|
res.set_znear_and_zfar(znear, zfar);
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
res.matrix[(3, 3)] = T::zero();
|
|
|
|
res.matrix[(3, 2)] = -T::one();
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Wraps the given matrix to interpret it as a 3D perspective matrix.
|
|
|
|
///
|
2020-02-23 23:30:11 +08:00
|
|
|
/// It is not checked whether or not the given matrix actually represents a perspective
|
2016-12-05 05:44:42 +08:00
|
|
|
/// projection.
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn from_matrix_unchecked(matrix: Matrix4<T>) -> Self {
|
2020-10-11 16:57:26 +08:00
|
|
|
Self { matrix }
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2017-02-13 01:17:09 +08:00
|
|
|
/// Retrieves the inverse of the underlying homogeneous matrix.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn inverse(&self) -> Matrix4<T> {
|
2017-02-13 01:17:09 +08:00
|
|
|
let mut res = self.to_homogeneous();
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
res[(0, 0)] = T::one() / self.matrix[(0, 0)];
|
|
|
|
res[(1, 1)] = T::one() / self.matrix[(1, 1)];
|
|
|
|
res[(2, 2)] = T::zero();
|
2017-02-13 01:17:09 +08:00
|
|
|
|
|
|
|
let m23 = self.matrix[(2, 3)];
|
|
|
|
let m32 = self.matrix[(3, 2)];
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
res[(2, 3)] = T::one() / m32;
|
|
|
|
res[(3, 2)] = T::one() / m23;
|
2017-02-13 01:17:09 +08:00
|
|
|
res[(3, 3)] = -self.matrix[(2, 2)] / (m23 * m32);
|
|
|
|
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
2016-12-05 05:44:42 +08:00
|
|
|
/// Computes the corresponding homogeneous matrix.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn to_homogeneous(&self) -> Matrix4<T> {
|
2016-12-05 05:44:42 +08:00
|
|
|
self.matrix.clone_owned()
|
|
|
|
}
|
|
|
|
|
2017-08-03 01:37:44 +08:00
|
|
|
/// A reference to the underlying homogeneous transformation matrix.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn as_matrix(&self) -> &Matrix4<T> {
|
2017-08-03 01:37:44 +08:00
|
|
|
&self.matrix
|
|
|
|
}
|
|
|
|
|
2018-09-22 22:02:55 +08:00
|
|
|
/// A reference to this transformation seen as a `Projective3`.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn as_projective(&self) -> &Projective3<T> {
|
2018-09-22 22:02:55 +08:00
|
|
|
unsafe { mem::transmute(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This transformation seen as a `Projective3`.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn to_projective(&self) -> Projective3<T> {
|
2018-09-22 22:02:55 +08:00
|
|
|
Projective3::from_matrix_unchecked(self.matrix)
|
|
|
|
}
|
|
|
|
|
2017-08-03 01:37:44 +08:00
|
|
|
/// Retrieves the underlying homogeneous matrix.
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn into_inner(self) -> Matrix4<T> {
|
2018-12-10 04:38:02 +08:00
|
|
|
self.matrix
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves the underlying homogeneous matrix.
|
|
|
|
/// Deprecated: Use [Perspective3::into_inner] instead.
|
2020-03-21 19:16:46 +08:00
|
|
|
#[deprecated(note = "use `.into_inner()` instead")]
|
2018-12-10 04:38:02 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn unwrap(self) -> Matrix4<T> {
|
2017-08-03 01:37:44 +08:00
|
|
|
self.matrix
|
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Gets the `width / height` aspect ratio of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn aspect(&self) -> T {
|
2016-12-05 05:44:42 +08:00
|
|
|
self.matrix[(1, 1)] / self.matrix[(0, 0)]
|
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Gets the y field of view of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn fovy(&self) -> T {
|
|
|
|
(T::one() / self.matrix[(1, 1)]).atan() * crate::convert(2.0)
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Gets the near plane offset of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn znear(&self) -> T {
|
|
|
|
let ratio = (-self.matrix[(2, 2)] + T::one()) / (-self.matrix[(2, 2)] - T::one());
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2020-03-21 19:16:46 +08:00
|
|
|
self.matrix[(2, 3)] / (ratio * crate::convert(2.0))
|
|
|
|
- self.matrix[(2, 3)] / crate::convert(2.0)
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Gets the far plane offset of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn zfar(&self) -> T {
|
|
|
|
let ratio = (-self.matrix[(2, 2)] + T::one()) / (-self.matrix[(2, 2)] - T::one());
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
(self.matrix[(2, 3)] - ratio * self.matrix[(2, 3)]) / crate::convert(2.0)
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2020-11-15 23:57:49 +08:00
|
|
|
// TODO: add a method to retrieve znear and zfar simultaneously?
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2020-11-15 23:57:49 +08:00
|
|
|
// TODO: when we get specialization, specialize the Mul impl instead.
|
2016-12-05 05:44:42 +08:00
|
|
|
/// Projects a point. Faster than matrix multiplication.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn project_point(&self, p: &Point3<T>) -> Point3<T> {
|
|
|
|
let inverse_denom = -T::one() / p[2];
|
2017-08-03 01:37:44 +08:00
|
|
|
Point3::new(
|
2018-02-02 19:26:35 +08:00
|
|
|
self.matrix[(0, 0)] * p[0] * inverse_denom,
|
|
|
|
self.matrix[(1, 1)] * p[1] * inverse_denom,
|
|
|
|
(self.matrix[(2, 2)] * p[2] + self.matrix[(2, 3)]) * inverse_denom,
|
2016-12-05 05:44:42 +08:00
|
|
|
)
|
2017-02-13 01:17:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Un-projects a point. Faster than multiplication by the matrix inverse.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn unproject_point(&self, p: &Point3<T>) -> Point3<T> {
|
2017-02-13 01:17:09 +08:00
|
|
|
let inverse_denom = self.matrix[(2, 3)] / (p[2] + self.matrix[(2, 2)]);
|
|
|
|
|
2017-08-03 01:37:44 +08:00
|
|
|
Point3::new(
|
2017-02-13 01:17:09 +08:00
|
|
|
p[0] * inverse_denom / self.matrix[(0, 0)],
|
|
|
|
p[1] * inverse_denom / self.matrix[(1, 1)],
|
2018-02-02 19:26:35 +08:00
|
|
|
-inverse_denom,
|
2017-02-13 01:17:09 +08:00
|
|
|
)
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
|
2020-11-15 23:57:49 +08:00
|
|
|
// TODO: when we get specialization, specialize the Mul impl instead.
|
2016-12-05 05:44:42 +08:00
|
|
|
/// Projects a vector. Faster than matrix multiplication.
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn project_vector<SB>(&self, p: &Vector<T, U3, SB>) -> Vector3<T>
|
2020-04-06 00:49:48 +08:00
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
SB: Storage<T, U3>,
|
2020-04-06 00:49:48 +08:00
|
|
|
{
|
2021-04-11 17:00:38 +08:00
|
|
|
let inverse_denom = -T::one() / p[2];
|
2017-08-03 01:37:44 +08:00
|
|
|
Vector3::new(
|
2016-12-05 05:44:42 +08:00
|
|
|
self.matrix[(0, 0)] * p[0] * inverse_denom,
|
|
|
|
self.matrix[(1, 1)] * p[1] * inverse_denom,
|
2018-02-02 19:26:35 +08:00
|
|
|
self.matrix[(2, 2)],
|
2016-12-05 05:44:42 +08:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Updates this perspective matrix with a new `width / height` aspect ratio of the view
|
2018-09-24 12:48:42 +08:00
|
|
|
/// frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn set_aspect(&mut self, aspect: T) {
|
2018-02-02 19:26:35 +08:00
|
|
|
assert!(
|
2021-04-11 17:00:38 +08:00
|
|
|
!relative_eq!(aspect, T::zero()),
|
2018-02-02 19:26:35 +08:00
|
|
|
"The aspect ratio must not be zero."
|
|
|
|
);
|
2016-12-05 05:44:42 +08:00
|
|
|
self.matrix[(0, 0)] = self.matrix[(1, 1)] / aspect;
|
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Updates this perspective with a new y field of view of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn set_fovy(&mut self, fovy: T) {
|
2018-02-02 19:26:35 +08:00
|
|
|
let old_m22 = self.matrix[(1, 1)];
|
2021-04-11 17:00:38 +08:00
|
|
|
self.matrix[(1, 1)] = T::one() / (fovy / crate::convert(2.0)).tan();
|
2016-12-05 05:44:42 +08:00
|
|
|
self.matrix[(0, 0)] = self.matrix[(0, 0)] * (self.matrix[(1, 1)] / old_m22);
|
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Updates this perspective matrix with a new near plane offset of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn set_znear(&mut self, znear: T) {
|
2016-12-05 05:44:42 +08:00
|
|
|
let zfar = self.zfar();
|
|
|
|
self.set_znear_and_zfar(znear, zfar);
|
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Updates this perspective matrix with a new far plane offset of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn set_zfar(&mut self, zfar: T) {
|
2016-12-05 05:44:42 +08:00
|
|
|
let znear = self.znear();
|
|
|
|
self.set_znear_and_zfar(znear, zfar);
|
|
|
|
}
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
/// Updates this perspective matrix with new near and far plane offsets of the view frustum.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn set_znear_and_zfar(&mut self, znear: T, zfar: T) {
|
2016-12-05 05:44:42 +08:00
|
|
|
self.matrix[(2, 2)] = (zfar + znear) / (znear - zfar);
|
2019-03-23 21:29:07 +08:00
|
|
|
self.matrix[(2, 3)] = zfar * znear * crate::convert(2.0) / (znear - zfar);
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-02 19:25:12 +08:00
|
|
|
#[cfg(feature = "rand-no-std")]
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField> Distribution<Perspective3<T>> for Standard
|
2020-04-06 00:49:48 +08:00
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
Standard: Distribution<T>,
|
2018-05-23 05:58:14 +08:00
|
|
|
{
|
2021-04-10 14:20:30 +08:00
|
|
|
/// Generate an arbitrary random variate for testing purposes.
|
2021-04-11 17:00:38 +08:00
|
|
|
fn sample<'a, R: Rng + ?Sized>(&self, r: &'a mut R) -> Perspective3<T> {
|
2021-03-02 19:25:12 +08:00
|
|
|
use crate::base::helper;
|
2018-05-23 05:58:14 +08:00
|
|
|
let znear = r.gen();
|
2021-04-11 17:00:38 +08:00
|
|
|
let zfar = helper::reject_rand(r, |&x: &T| !(x - znear).is_zero());
|
|
|
|
let aspect = helper::reject_rand(r, |&x: &T| !x.is_zero());
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2018-05-23 05:58:14 +08:00
|
|
|
Perspective3::new(aspect, r.gen(), znear, zfar)
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-02 19:26:35 +08:00
|
|
|
#[cfg(feature = "arbitrary")]
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField + Arbitrary> Arbitrary for Perspective3<T> {
|
2021-03-01 00:52:14 +08:00
|
|
|
fn arbitrary(g: &mut Gen) -> Self {
|
2021-03-02 19:25:12 +08:00
|
|
|
use crate::base::helper;
|
2018-02-02 19:26:35 +08:00
|
|
|
let znear = Arbitrary::arbitrary(g);
|
2021-04-11 17:00:38 +08:00
|
|
|
let zfar = helper::reject(g, |&x: &T| !(x - znear).is_zero());
|
|
|
|
let aspect = helper::reject(g, |&x: &T| !x.is_zero());
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
Self::new(aspect, Arbitrary::arbitrary(g), znear, zfar)
|
|
|
|
}
|
|
|
|
}
|
2018-10-13 16:25:34 +08:00
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: RealField> From<Perspective3<T>> for Matrix4<T> {
|
2018-10-13 16:25:34 +08:00
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
fn from(pers: Perspective3<T>) -> Self {
|
2020-10-11 17:57:43 +08:00
|
|
|
pers.into_inner()
|
2018-10-13 16:25:34 +08:00
|
|
|
}
|
|
|
|
}
|