2016-12-05 05:44:42 +08:00
|
|
|
|
#[cfg(feature = "arbitrary")]
|
2021-08-03 00:41:46 +08:00
|
|
|
|
use crate::base::storage::Owned;
|
2018-05-23 05:58:14 +08:00
|
|
|
|
#[cfg(feature = "arbitrary")]
|
|
|
|
|
use quickcheck::{Arbitrary, Gen};
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2018-05-23 05:58:14 +08:00
|
|
|
|
use num::Zero;
|
2021-03-02 19:25:12 +08:00
|
|
|
|
|
|
|
|
|
#[cfg(feature = "rand-no-std")]
|
|
|
|
|
use rand::{
|
2021-04-11 20:09:01 +08:00
|
|
|
|
distributions::{uniform::SampleUniform, Distribution, OpenClosed01, Standard, Uniform},
|
2021-03-02 19:25:12 +08:00
|
|
|
|
Rng,
|
|
|
|
|
};
|
|
|
|
|
|
2020-03-21 19:16:46 +08:00
|
|
|
|
use simba::scalar::RealField;
|
2020-03-22 06:22:55 +08:00
|
|
|
|
use simba::simd::{SimdBool, SimdRealField};
|
2018-05-23 05:58:14 +08:00
|
|
|
|
use std::ops::Neg;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
|
use crate::base::dimension::{U1, U2, U3};
|
|
|
|
|
use crate::base::storage::Storage;
|
2022-07-27 15:45:54 +08:00
|
|
|
|
use crate::base::{
|
|
|
|
|
Matrix2, Matrix3, SMatrix, SVector, Unit, UnitVector3, Vector, Vector1, Vector2, Vector3,
|
|
|
|
|
};
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
|
use crate::geometry::{Rotation2, Rotation3, UnitComplex, UnitQuaternion};
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
*
|
2017-08-03 01:37:44 +08:00
|
|
|
|
* 2D Rotation matrix.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
*
|
|
|
|
|
*/
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Construction from a 2D rotation angle
|
2021-04-11 17:00:38 +08:00
|
|
|
|
impl<T: SimdRealField> Rotation2<T> {
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// Builds a 2 dimensional rotation matrix from an angle in radian.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use std::f32;
|
2018-11-05 17:34:58 +08:00
|
|
|
|
/// # use nalgebra::{Rotation2, Point2};
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// let rot = Rotation2::new(f32::consts::FRAC_PI_2);
|
|
|
|
|
///
|
|
|
|
|
/// assert_relative_eq!(rot * Point2::new(3.0, 4.0), Point2::new(-4.0, 3.0));
|
|
|
|
|
/// ```
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn new(angle: T) -> Self {
|
2020-03-22 06:22:55 +08:00
|
|
|
|
let (sia, coa) = angle.simd_sin_cos();
|
2021-08-04 23:34:25 +08:00
|
|
|
|
Self::from_matrix_unchecked(Matrix2::new(coa.clone(), -sia.clone(), sia, coa))
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Builds a 2 dimensional rotation matrix from an angle in radian wrapped in a 1-dimensional vector.
|
|
|
|
|
///
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// This is generally used in the context of generic programming. Using
|
|
|
|
|
/// the `::new(angle)` method instead is more common.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_scaled_axis<SB: Storage<T, U1>>(axisangle: Vector<T, U1, SB>) -> Self {
|
2021-08-04 23:34:25 +08:00
|
|
|
|
Self::new(axisangle[0].clone())
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Construction from an existing 2D matrix or rotations
|
2021-04-11 17:00:38 +08:00
|
|
|
|
impl<T: SimdRealField> Rotation2<T> {
|
2021-02-22 21:26:25 +08:00
|
|
|
|
/// Builds a rotation from a basis assumed to be orthonormal.
|
|
|
|
|
///
|
2021-08-26 09:54:14 +08:00
|
|
|
|
/// In order to get a valid rotation matrix, the input must be an
|
2021-02-22 21:26:25 +08:00
|
|
|
|
/// orthonormal basis, i.e., all vectors are normalized, and the are
|
|
|
|
|
/// all orthogonal to each other. These invariants are not checked
|
|
|
|
|
/// by this method.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_basis_unchecked(basis: &[Vector2<T>; 2]) -> Self {
|
2021-02-22 21:26:25 +08:00
|
|
|
|
let mat = Matrix2::from_columns(&basis[..]);
|
|
|
|
|
Self::from_matrix_unchecked(mat)
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-19 05:41:46 +08:00
|
|
|
|
/// Builds a rotation matrix by extracting the rotation part of the given transformation `m`.
|
|
|
|
|
///
|
|
|
|
|
/// This is an iterative method. See `.from_matrix_eps` to provide mover
|
|
|
|
|
/// convergence parameters and starting solution.
|
|
|
|
|
/// This implements "A Robust Method to Extract the Rotational Part of Deformations" by Müller et al.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_matrix(m: &Matrix2<T>) -> Self
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-04-06 00:49:48 +08:00
|
|
|
|
{
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Self::from_matrix_eps(m, T::default_epsilon(), 0, Self::identity())
|
2019-02-19 05:41:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Builds a rotation matrix by extracting the rotation part of the given transformation `m`.
|
|
|
|
|
///
|
|
|
|
|
/// This implements "A Robust Method to Extract the Rotational Part of Deformations" by Müller et al.
|
|
|
|
|
///
|
|
|
|
|
/// # Parameters
|
|
|
|
|
///
|
|
|
|
|
/// * `m`: the matrix from which the rotational part is to be extracted.
|
|
|
|
|
/// * `eps`: the angular errors tolerated between the current rotation and the optimal one.
|
|
|
|
|
/// * `max_iter`: the maximum number of iterations. Loops indefinitely until convergence if set to `0`.
|
|
|
|
|
/// * `guess`: an estimate of the solution. Convergence will be significantly faster if an initial solution close
|
|
|
|
|
/// to the actual solution is provided. Can be set to `Rotation2::identity()` if no other
|
|
|
|
|
/// guesses come to mind.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_matrix_eps(m: &Matrix2<T>, eps: T, mut max_iter: usize, guess: Self) -> Self
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-04-06 00:49:48 +08:00
|
|
|
|
{
|
2019-02-19 05:41:46 +08:00
|
|
|
|
if max_iter == 0 {
|
|
|
|
|
max_iter = usize::max_value();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let mut rot = guess.into_inner();
|
|
|
|
|
|
|
|
|
|
for _ in 0..max_iter {
|
2020-03-21 19:16:46 +08:00
|
|
|
|
let axis = rot.column(0).perp(&m.column(0)) + rot.column(1).perp(&m.column(1));
|
|
|
|
|
let denom = rot.column(0).dot(&m.column(0)) + rot.column(1).dot(&m.column(1));
|
2019-02-19 05:41:46 +08:00
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
let angle = axis / (denom.abs() + T::default_epsilon());
|
2021-08-04 23:34:25 +08:00
|
|
|
|
if angle.clone().abs() > eps {
|
2019-02-19 05:41:46 +08:00
|
|
|
|
rot = Self::new(angle) * rot;
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Self::from_matrix_unchecked(rot)
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-24 12:48:42 +08:00
|
|
|
|
/// The rotation matrix required to align `a` and `b` but with its angle.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
///
|
|
|
|
|
/// This is the rotation `R` such that `(R * a).angle(b) == 0 && (R * a).dot(b).is_positive()`.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Vector2, Rotation2};
|
|
|
|
|
/// let a = Vector2::new(1.0, 2.0);
|
|
|
|
|
/// let b = Vector2::new(2.0, 1.0);
|
|
|
|
|
/// let rot = Rotation2::rotation_between(&a, &b);
|
|
|
|
|
/// assert_relative_eq!(rot * a, b);
|
|
|
|
|
/// assert_relative_eq!(rot.inverse() * b, a);
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn rotation_between<SB, SC>(a: &Vector<T, U2, SB>, b: &Vector<T, U2, SC>) -> Self
|
2018-02-02 19:26:35 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
|
|
|
|
SB: Storage<T, U2>,
|
|
|
|
|
SC: Storage<T, U2>,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2019-03-23 21:29:07 +08:00
|
|
|
|
crate::convert(UnitComplex::rotation_between(a, b).to_rotation_matrix())
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The smallest rotation needed to make `a` and `b` collinear and point toward the same
|
|
|
|
|
/// direction, raised to the power `s`.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Vector2, Rotation2};
|
|
|
|
|
/// let a = Vector2::new(1.0, 2.0);
|
|
|
|
|
/// let b = Vector2::new(2.0, 1.0);
|
|
|
|
|
/// let rot2 = Rotation2::scaled_rotation_between(&a, &b, 0.2);
|
|
|
|
|
/// let rot5 = Rotation2::scaled_rotation_between(&a, &b, 0.5);
|
|
|
|
|
/// assert_relative_eq!(rot2 * rot2 * rot2 * rot2 * rot2 * a, b, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(rot5 * rot5 * a, b, epsilon = 1.0e-6);
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2018-02-02 19:26:35 +08:00
|
|
|
|
pub fn scaled_rotation_between<SB, SC>(
|
2021-04-11 17:00:38 +08:00
|
|
|
|
a: &Vector<T, U2, SB>,
|
|
|
|
|
b: &Vector<T, U2, SC>,
|
|
|
|
|
s: T,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
) -> Self
|
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
|
|
|
|
SB: Storage<T, U2>,
|
|
|
|
|
SC: Storage<T, U2>,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2019-03-23 21:29:07 +08:00
|
|
|
|
crate::convert(UnitComplex::scaled_rotation_between(a, b, s).to_rotation_matrix())
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The rotation matrix needed to make `self` and `other` coincide.
|
|
|
|
|
///
|
|
|
|
|
/// The result is such that: `self.rotation_to(other) * self == other`.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::Rotation2;
|
|
|
|
|
/// let rot1 = Rotation2::new(0.1);
|
|
|
|
|
/// let rot2 = Rotation2::new(1.7);
|
|
|
|
|
/// let rot_to = rot1.rotation_to(&rot2);
|
|
|
|
|
///
|
|
|
|
|
/// assert_relative_eq!(rot_to * rot1, rot2);
|
|
|
|
|
/// assert_relative_eq!(rot_to.inverse() * rot2, rot1);
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
|
#[must_use]
|
2019-02-17 05:29:41 +08:00
|
|
|
|
pub fn rotation_to(&self, other: &Self) -> Self {
|
2016-12-05 05:44:42 +08:00
|
|
|
|
other * self.inverse()
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-19 05:41:46 +08:00
|
|
|
|
/// Ensure this rotation is an orthonormal rotation matrix. This is useful when repeated
|
|
|
|
|
/// computations might cause the matrix from progressively not being orthonormal anymore.
|
|
|
|
|
#[inline]
|
2020-03-22 06:22:55 +08:00
|
|
|
|
pub fn renormalize(&mut self)
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-04-06 00:49:48 +08:00
|
|
|
|
{
|
2021-08-04 23:34:25 +08:00
|
|
|
|
let mut c = UnitComplex::from(self.clone());
|
2019-02-19 05:41:46 +08:00
|
|
|
|
let _ = c.renormalize();
|
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
*self = Self::from_matrix_eps(self.matrix(), T::default_epsilon(), 0, c.into())
|
2019-02-19 05:41:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-08-26 09:54:14 +08:00
|
|
|
|
/// Raise the rotation to a given floating power, i.e., returns the rotation with the angle
|
2017-02-13 01:17:09 +08:00
|
|
|
|
/// of `self` multiplied by `n`.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
2018-12-18 22:44:53 +08:00
|
|
|
|
/// # #[macro_use] extern crate approx;
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// # use nalgebra::Rotation2;
|
|
|
|
|
/// let rot = Rotation2::new(0.78);
|
|
|
|
|
/// let pow = rot.powf(2.0);
|
2018-12-18 22:44:53 +08:00
|
|
|
|
/// assert_relative_eq!(pow.angle(), 2.0 * 0.78);
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// ```
|
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 powf(&self, n: T) -> Self {
|
2017-08-03 01:37:44 +08:00
|
|
|
|
Self::new(self.angle() * n)
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # 2D angle extraction
|
2021-04-11 17:00:38 +08:00
|
|
|
|
impl<T: SimdRealField> Rotation2<T> {
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// The rotation angle.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::Rotation2;
|
|
|
|
|
/// let rot = Rotation2::new(1.78);
|
|
|
|
|
/// assert_relative_eq!(rot.angle(), 1.78);
|
|
|
|
|
/// ```
|
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 angle(&self) -> T {
|
2021-08-04 23:34:25 +08:00
|
|
|
|
self.matrix()[(1, 0)]
|
|
|
|
|
.clone()
|
|
|
|
|
.simd_atan2(self.matrix()[(0, 0)].clone())
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
2020-10-25 21:00:47 +08:00
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// The rotation angle needed to make `self` and `other` coincide.
|
2020-10-25 21:00:47 +08:00
|
|
|
|
///
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Example
|
2020-10-25 21:00:47 +08:00
|
|
|
|
/// ```
|
2020-10-25 21:15:26 +08:00
|
|
|
|
/// # #[macro_use] extern crate approx;
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # use nalgebra::Rotation2;
|
|
|
|
|
/// let rot1 = Rotation2::new(0.1);
|
|
|
|
|
/// let rot2 = Rotation2::new(1.7);
|
|
|
|
|
/// assert_relative_eq!(rot1.angle_to(&rot2), 1.6);
|
2020-10-25 21:00:47 +08:00
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn angle_to(&self, other: &Self) -> T {
|
2020-11-21 18:21:47 +08:00
|
|
|
|
self.rotation_to(other).angle()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The rotation angle returned as a 1-dimensional vector.
|
|
|
|
|
///
|
|
|
|
|
/// This is generally used in the context of generic programming. Using
|
|
|
|
|
/// the `.angle()` method instead is more common.
|
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn scaled_axis(&self) -> SVector<T, 1> {
|
2020-11-21 18:21:47 +08:00
|
|
|
|
Vector1::new(self.angle())
|
2020-10-25 21:00:47 +08:00
|
|
|
|
}
|
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: SimdRealField> Distribution<Rotation2<T>> for Standard
|
2020-03-22 06:22:55 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
|
|
|
|
T: SampleUniform,
|
2018-05-23 05:58:14 +08:00
|
|
|
|
{
|
2018-07-08 07:03:08 +08:00
|
|
|
|
/// Generate a uniformly distributed random rotation.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-07-04 11:19:07 +08:00
|
|
|
|
fn sample<R: Rng + ?Sized>(&self, rng: &mut R) -> Rotation2<T> {
|
2021-04-11 17:00:38 +08:00
|
|
|
|
let twopi = Uniform::new(T::zero(), T::simd_two_pi());
|
2021-04-10 14:13:46 +08:00
|
|
|
|
Rotation2::new(rng.sample(twopi))
|
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: SimdRealField + Arbitrary> Arbitrary for Rotation2<T>
|
2020-03-22 06:22:55 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
2021-08-03 00:41:46 +08:00
|
|
|
|
Owned<T, U2, U2>: Send,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-03-01 00:52:14 +08:00
|
|
|
|
fn arbitrary(g: &mut Gen) -> Self {
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Self::new(T::arbitrary(g))
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
*
|
2017-08-03 01:37:44 +08:00
|
|
|
|
* 3D Rotation matrix.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
*
|
|
|
|
|
*/
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Construction from a 3D axis and/or angles
|
2021-04-11 17:00:38 +08:00
|
|
|
|
impl<T: SimdRealField> Rotation3<T>
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
2020-03-22 06:22:55 +08:00
|
|
|
|
{
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// Builds a 3 dimensional rotation matrix from an axis and an angle.
|
|
|
|
|
///
|
|
|
|
|
/// # Arguments
|
|
|
|
|
/// * `axisangle` - A vector representing the rotation. Its magnitude is the amount of rotation
|
|
|
|
|
/// in radian. Its direction is the axis of rotation.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use std::f32;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Point3, Vector3};
|
|
|
|
|
/// let axisangle = Vector3::y() * f32::consts::FRAC_PI_2;
|
|
|
|
|
/// // Point and vector being transformed in the tests.
|
|
|
|
|
/// let pt = Point3::new(4.0, 5.0, 6.0);
|
|
|
|
|
/// let vec = Vector3::new(4.0, 5.0, 6.0);
|
|
|
|
|
/// let rot = Rotation3::new(axisangle);
|
|
|
|
|
///
|
|
|
|
|
/// assert_relative_eq!(rot * pt, Point3::new(6.0, 5.0, -4.0), epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(rot * vec, Vector3::new(6.0, 5.0, -4.0), epsilon = 1.0e-6);
|
|
|
|
|
///
|
|
|
|
|
/// // A zero vector yields an identity.
|
|
|
|
|
/// assert_eq!(Rotation3::new(Vector3::<f32>::zeros()), Rotation3::identity());
|
|
|
|
|
/// ```
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn new<SB: Storage<T, U3>>(axisangle: Vector<T, U3, SB>) -> Self {
|
2017-08-03 01:37:44 +08:00
|
|
|
|
let axisangle = axisangle.into_owned();
|
|
|
|
|
let (axis, angle) = Unit::new_and_get(axisangle);
|
2016-12-05 05:44:42 +08:00
|
|
|
|
Self::from_axis_angle(&axis, angle)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Builds a 3D rotation matrix from an axis scaled by the rotation angle.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// This is the same as `Self::new(axisangle)`.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use std::f32;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Point3, Vector3};
|
|
|
|
|
/// let axisangle = Vector3::y() * f32::consts::FRAC_PI_2;
|
|
|
|
|
/// // Point and vector being transformed in the tests.
|
|
|
|
|
/// let pt = Point3::new(4.0, 5.0, 6.0);
|
|
|
|
|
/// let vec = Vector3::new(4.0, 5.0, 6.0);
|
|
|
|
|
/// let rot = Rotation3::new(axisangle);
|
|
|
|
|
///
|
|
|
|
|
/// assert_relative_eq!(rot * pt, Point3::new(6.0, 5.0, -4.0), epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(rot * vec, Vector3::new(6.0, 5.0, -4.0), epsilon = 1.0e-6);
|
|
|
|
|
///
|
|
|
|
|
/// // A zero vector yields an identity.
|
|
|
|
|
/// assert_eq!(Rotation3::from_scaled_axis(Vector3::<f32>::zeros()), Rotation3::identity());
|
|
|
|
|
/// ```
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_scaled_axis<SB: Storage<T, U3>>(axisangle: Vector<T, U3, SB>) -> Self {
|
2016-12-05 05:44:42 +08:00
|
|
|
|
Self::new(axisangle)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Builds a 3D rotation matrix from an axis and a rotation angle.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use std::f32;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Point3, Vector3};
|
|
|
|
|
/// let axis = Vector3::y_axis();
|
|
|
|
|
/// let angle = f32::consts::FRAC_PI_2;
|
|
|
|
|
/// // Point and vector being transformed in the tests.
|
|
|
|
|
/// let pt = Point3::new(4.0, 5.0, 6.0);
|
|
|
|
|
/// let vec = Vector3::new(4.0, 5.0, 6.0);
|
|
|
|
|
/// let rot = Rotation3::from_axis_angle(&axis, angle);
|
|
|
|
|
///
|
|
|
|
|
/// assert_eq!(rot.axis().unwrap(), axis);
|
|
|
|
|
/// assert_eq!(rot.angle(), angle);
|
|
|
|
|
/// assert_relative_eq!(rot * pt, Point3::new(6.0, 5.0, -4.0), epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(rot * vec, Vector3::new(6.0, 5.0, -4.0), epsilon = 1.0e-6);
|
|
|
|
|
///
|
|
|
|
|
/// // A zero vector yields an identity.
|
|
|
|
|
/// assert_eq!(Rotation3::from_scaled_axis(Vector3::<f32>::zeros()), Rotation3::identity());
|
|
|
|
|
/// ```
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_axis_angle<SB>(axis: &Unit<Vector<T, U3, SB>>, angle: T) -> Self
|
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-08-04 23:34:25 +08:00
|
|
|
|
angle.clone().simd_ne(T::zero()).if_else(
|
2020-03-22 06:22:55 +08:00
|
|
|
|
|| {
|
2021-08-04 23:34:25 +08:00
|
|
|
|
let ux = axis.as_ref()[0].clone();
|
|
|
|
|
let uy = axis.as_ref()[1].clone();
|
|
|
|
|
let uz = axis.as_ref()[2].clone();
|
|
|
|
|
let sqx = ux.clone() * ux.clone();
|
|
|
|
|
let sqy = uy.clone() * uy.clone();
|
|
|
|
|
let sqz = uz.clone() * uz.clone();
|
2020-03-22 06:22:55 +08:00
|
|
|
|
let (sin, cos) = angle.simd_sin_cos();
|
2021-08-04 23:34:25 +08:00
|
|
|
|
let one_m_cos = T::one() - cos.clone();
|
2020-03-22 06:22:55 +08:00
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Self::from_matrix_unchecked(SMatrix::<T, 3, 3>::new(
|
2021-08-04 23:34:25 +08:00
|
|
|
|
sqx.clone() + (T::one() - sqx) * cos.clone(),
|
|
|
|
|
ux.clone() * uy.clone() * one_m_cos.clone() - uz.clone() * sin.clone(),
|
|
|
|
|
ux.clone() * uz.clone() * one_m_cos.clone() + uy.clone() * sin.clone(),
|
|
|
|
|
ux.clone() * uy.clone() * one_m_cos.clone() + uz.clone() * sin.clone(),
|
|
|
|
|
sqy.clone() + (T::one() - sqy) * cos.clone(),
|
|
|
|
|
uy.clone() * uz.clone() * one_m_cos.clone() - ux.clone() * sin.clone(),
|
|
|
|
|
ux.clone() * uz.clone() * one_m_cos.clone() - uy.clone() * sin.clone(),
|
2020-03-22 06:22:55 +08:00
|
|
|
|
uy * uz * one_m_cos + ux * sin,
|
2021-08-04 23:34:25 +08:00
|
|
|
|
sqz.clone() + (T::one() - sqz) * cos,
|
2020-03-22 06:22:55 +08:00
|
|
|
|
))
|
|
|
|
|
},
|
2020-11-16 22:20:33 +08:00
|
|
|
|
Self::identity,
|
2020-03-22 06:22:55 +08:00
|
|
|
|
)
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Creates a new rotation from Euler angles.
|
|
|
|
|
///
|
|
|
|
|
/// The primitive rotations are applied in order: 1 roll − 2 pitch − 3 yaw.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::Rotation3;
|
|
|
|
|
/// let rot = Rotation3::from_euler_angles(0.1, 0.2, 0.3);
|
|
|
|
|
/// let euler = rot.euler_angles();
|
|
|
|
|
/// assert_relative_eq!(euler.0, 0.1, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(euler.1, 0.2, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(euler.2, 0.3, epsilon = 1.0e-6);
|
|
|
|
|
/// ```
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_euler_angles(roll: T, pitch: T, yaw: T) -> Self {
|
2020-03-22 06:22:55 +08:00
|
|
|
|
let (sr, cr) = roll.simd_sin_cos();
|
|
|
|
|
let (sp, cp) = pitch.simd_sin_cos();
|
|
|
|
|
let (sy, cy) = yaw.simd_sin_cos();
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Self::from_matrix_unchecked(SMatrix::<T, 3, 3>::new(
|
2021-08-04 23:34:25 +08:00
|
|
|
|
cy.clone() * cp.clone(),
|
|
|
|
|
cy.clone() * sp.clone() * sr.clone() - sy.clone() * cr.clone(),
|
|
|
|
|
cy.clone() * sp.clone() * cr.clone() + sy.clone() * sr.clone(),
|
|
|
|
|
sy.clone() * cp.clone(),
|
|
|
|
|
sy.clone() * sp.clone() * sr.clone() + cy.clone() * cr.clone(),
|
|
|
|
|
sy * sp.clone() * cr.clone() - cy * sr.clone(),
|
2018-02-02 19:26:35 +08:00
|
|
|
|
-sp,
|
2021-08-04 23:34:25 +08:00
|
|
|
|
cp.clone() * sr,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
cp * cr,
|
|
|
|
|
))
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Construction from a 3D eye position and target point
|
2021-04-11 17:00:38 +08:00
|
|
|
|
impl<T: SimdRealField> Rotation3<T>
|
2020-11-21 18:21:47 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
2020-11-21 18:21:47 +08:00
|
|
|
|
{
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// Creates a rotation that corresponds to the local frame of an observer standing at the
|
|
|
|
|
/// origin and looking toward `dir`.
|
|
|
|
|
///
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// It maps the `z` axis to the direction `dir`.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Arguments
|
|
|
|
|
/// * dir - The look direction, that is, direction the matrix `z` axis will be aligned with.
|
|
|
|
|
/// * up - The vertical direction. The only requirement of this parameter is to not be
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// collinear to `dir`. Non-collinearity is not checked.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use std::f32;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3};
|
|
|
|
|
/// let dir = Vector3::new(1.0, 2.0, 3.0);
|
|
|
|
|
/// let up = Vector3::y();
|
|
|
|
|
///
|
2019-01-17 05:41:25 +08:00
|
|
|
|
/// let rot = Rotation3::face_towards(&dir, &up);
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// assert_relative_eq!(rot * Vector3::z(), dir.normalize());
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn face_towards<SB, SC>(dir: &Vector<T, U3, SB>, up: &Vector<T, U3, SC>) -> Self
|
2018-02-02 19:26:35 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
SB: Storage<T, U3>,
|
|
|
|
|
SC: Storage<T, U3>,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2016-12-05 05:44:42 +08:00
|
|
|
|
let zaxis = dir.normalize();
|
|
|
|
|
let xaxis = up.cross(&zaxis).normalize();
|
|
|
|
|
let yaxis = zaxis.cross(&xaxis).normalize();
|
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Self::from_matrix_unchecked(SMatrix::<T, 3, 3>::new(
|
2021-08-04 23:34:25 +08:00
|
|
|
|
xaxis.x.clone(),
|
|
|
|
|
yaxis.x.clone(),
|
|
|
|
|
zaxis.x.clone(),
|
|
|
|
|
xaxis.y.clone(),
|
|
|
|
|
yaxis.y.clone(),
|
|
|
|
|
zaxis.y.clone(),
|
|
|
|
|
xaxis.z.clone(),
|
|
|
|
|
yaxis.z.clone(),
|
|
|
|
|
zaxis.z.clone(),
|
2018-02-02 19:26:35 +08:00
|
|
|
|
))
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-28 07:18:29 +08:00
|
|
|
|
/// Deprecated: Use [`Rotation3::face_towards`] instead.
|
2020-03-21 19:16:46 +08:00
|
|
|
|
#[deprecated(note = "renamed to `face_towards`")]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn new_observer_frames<SB, SC>(dir: &Vector<T, U3, SB>, up: &Vector<T, U3, SC>) -> Self
|
2019-01-17 17:17:00 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
SB: Storage<T, U3>,
|
|
|
|
|
SC: Storage<T, U3>,
|
2019-01-17 17:17:00 +08:00
|
|
|
|
{
|
|
|
|
|
Self::face_towards(dir, up)
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// Builds a right-handed look-at view matrix without translation.
|
|
|
|
|
///
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// It maps the view direction `dir` to the **negative** `z` axis.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// This conforms to the common notion of right handed look-at matrix from the computer
|
|
|
|
|
/// graphics community.
|
|
|
|
|
///
|
|
|
|
|
/// # Arguments
|
2018-10-20 17:37:22 +08:00
|
|
|
|
/// * dir - The direction toward which the camera looks.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// * up - A vector approximately aligned with required the vertical axis. The only
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// requirement of this parameter is to not be collinear to `dir`.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use std::f32;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3};
|
|
|
|
|
/// let dir = Vector3::new(1.0, 2.0, 3.0);
|
|
|
|
|
/// let up = Vector3::y();
|
|
|
|
|
///
|
|
|
|
|
/// let rot = Rotation3::look_at_rh(&dir, &up);
|
|
|
|
|
/// assert_relative_eq!(rot * dir.normalize(), -Vector3::z());
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn look_at_rh<SB, SC>(dir: &Vector<T, U3, SB>, up: &Vector<T, U3, SC>) -> Self
|
2018-02-02 19:26:35 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
SB: Storage<T, U3>,
|
|
|
|
|
SC: Storage<T, U3>,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2019-01-17 05:41:25 +08:00
|
|
|
|
Self::face_towards(&dir.neg(), up).inverse()
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Builds a left-handed look-at view matrix without translation.
|
|
|
|
|
///
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// It maps the view direction `dir` to the **positive** `z` axis.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// This conforms to the common notion of left handed look-at matrix from the computer
|
|
|
|
|
/// graphics community.
|
|
|
|
|
///
|
|
|
|
|
/// # Arguments
|
2018-10-20 17:37:22 +08:00
|
|
|
|
/// * dir - The direction toward which the camera looks.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// * up - A vector approximately aligned with required the vertical axis. The only
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// requirement of this parameter is to not be collinear to `dir`.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use std::f32;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3};
|
|
|
|
|
/// let dir = Vector3::new(1.0, 2.0, 3.0);
|
|
|
|
|
/// let up = Vector3::y();
|
|
|
|
|
///
|
|
|
|
|
/// let rot = Rotation3::look_at_lh(&dir, &up);
|
|
|
|
|
/// assert_relative_eq!(rot * dir.normalize(), Vector3::z());
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn look_at_lh<SB, SC>(dir: &Vector<T, U3, SB>, up: &Vector<T, U3, SC>) -> Self
|
2018-02-02 19:26:35 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
SB: Storage<T, U3>,
|
|
|
|
|
SC: Storage<T, U3>,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2019-01-17 05:41:25 +08:00
|
|
|
|
Self::face_towards(dir, up).inverse()
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Construction from an existing 3D matrix or rotations
|
2021-04-11 17:00:38 +08:00
|
|
|
|
impl<T: SimdRealField> Rotation3<T>
|
2020-11-21 18:21:47 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
2020-11-21 18:21:47 +08:00
|
|
|
|
{
|
2018-09-24 12:48:42 +08:00
|
|
|
|
/// The rotation matrix required to align `a` and `b` but with its angle.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
///
|
|
|
|
|
/// This is the rotation `R` such that `(R * a).angle(b) == 0 && (R * a).dot(b).is_positive()`.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Vector3, Rotation3};
|
|
|
|
|
/// let a = Vector3::new(1.0, 2.0, 3.0);
|
|
|
|
|
/// let b = Vector3::new(3.0, 1.0, 2.0);
|
|
|
|
|
/// let rot = Rotation3::rotation_between(&a, &b).unwrap();
|
|
|
|
|
/// assert_relative_eq!(rot * a, b, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(rot.inverse() * b, a, epsilon = 1.0e-6);
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn rotation_between<SB, SC>(a: &Vector<T, U3, SB>, b: &Vector<T, U3, SC>) -> Option<Self>
|
2018-02-02 19:26:35 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
|
|
|
|
SB: Storage<T, U3>,
|
|
|
|
|
SC: Storage<T, U3>,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Self::scaled_rotation_between(a, b, T::one())
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The smallest rotation needed to make `a` and `b` collinear and point toward the same
|
|
|
|
|
/// direction, raised to the power `s`.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Vector3, Rotation3};
|
|
|
|
|
/// let a = Vector3::new(1.0, 2.0, 3.0);
|
|
|
|
|
/// let b = Vector3::new(3.0, 1.0, 2.0);
|
|
|
|
|
/// let rot2 = Rotation3::scaled_rotation_between(&a, &b, 0.2).unwrap();
|
|
|
|
|
/// let rot5 = Rotation3::scaled_rotation_between(&a, &b, 0.5).unwrap();
|
|
|
|
|
/// assert_relative_eq!(rot2 * rot2 * rot2 * rot2 * rot2 * a, b, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(rot5 * rot5 * a, b, epsilon = 1.0e-6);
|
|
|
|
|
/// ```
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2018-02-02 19:26:35 +08:00
|
|
|
|
pub fn scaled_rotation_between<SB, SC>(
|
2021-04-11 17:00:38 +08:00
|
|
|
|
a: &Vector<T, U3, SB>,
|
|
|
|
|
b: &Vector<T, U3, SC>,
|
|
|
|
|
n: T,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
) -> Option<Self>
|
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
|
|
|
|
SB: Storage<T, U3>,
|
|
|
|
|
SC: Storage<T, U3>,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2020-11-15 23:57:49 +08:00
|
|
|
|
// TODO: code duplication with Rotation.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
if let (Some(na), Some(nb)) = (a.try_normalize(T::zero()), b.try_normalize(T::zero())) {
|
2016-12-05 05:44:42 +08:00
|
|
|
|
let c = na.cross(&nb);
|
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
if let Some(axis) = Unit::try_new(c, T::default_epsilon()) {
|
2018-02-02 19:26:35 +08:00
|
|
|
|
return Some(Self::from_axis_angle(&axis, na.dot(&nb).acos() * n));
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Zero or PI.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
if na.dot(&nb) < T::zero() {
|
2016-12-05 05:44:42 +08:00
|
|
|
|
// PI
|
|
|
|
|
//
|
|
|
|
|
// The rotation axis is undefined but the angle not zero. This is not a
|
|
|
|
|
// simple rotation.
|
|
|
|
|
return None;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Some(Self::identity())
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// The rotation matrix needed to make `self` and `other` coincide.
|
|
|
|
|
///
|
|
|
|
|
/// The result is such that: `self.rotation_to(other) * self == other`.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3};
|
|
|
|
|
/// let rot1 = Rotation3::from_axis_angle(&Vector3::y_axis(), 1.0);
|
|
|
|
|
/// let rot2 = Rotation3::from_axis_angle(&Vector3::x_axis(), 0.1);
|
|
|
|
|
/// let rot_to = rot1.rotation_to(&rot2);
|
|
|
|
|
/// assert_relative_eq!(rot_to * rot1, rot2, epsilon = 1.0e-6);
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
|
#[must_use]
|
2020-11-21 18:21:47 +08:00
|
|
|
|
pub fn rotation_to(&self, other: &Self) -> Self {
|
|
|
|
|
other * self.inverse()
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-26 09:54:14 +08:00
|
|
|
|
/// Raise the rotation to a given floating power, i.e., returns the rotation with the same
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// axis as `self` and an angle equal to `self.angle()` multiplied by `n`.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3, Unit};
|
|
|
|
|
/// let axis = Unit::new_normalize(Vector3::new(1.0, 2.0, 3.0));
|
|
|
|
|
/// let angle = 1.2;
|
|
|
|
|
/// let rot = Rotation3::from_axis_angle(&axis, angle);
|
|
|
|
|
/// let pow = rot.powf(2.0);
|
|
|
|
|
/// assert_relative_eq!(pow.axis().unwrap(), axis, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_eq!(pow.angle(), 2.4);
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn powf(&self, n: T) -> Self
|
2020-11-21 18:21:47 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-11-21 18:21:47 +08:00
|
|
|
|
{
|
|
|
|
|
if let Some(axis) = self.axis() {
|
|
|
|
|
Self::from_axis_angle(&axis, self.angle() * n)
|
2021-04-11 17:00:38 +08:00
|
|
|
|
} else if self.matrix()[(0, 0)] < T::zero() {
|
|
|
|
|
let minus_id = SMatrix::<T, 3, 3>::from_diagonal_element(-T::one());
|
2020-11-21 18:21:47 +08:00
|
|
|
|
Self::from_matrix_unchecked(minus_id)
|
|
|
|
|
} else {
|
|
|
|
|
Self::identity()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-22 21:26:25 +08:00
|
|
|
|
/// Builds a rotation from a basis assumed to be orthonormal.
|
|
|
|
|
///
|
2021-08-26 09:54:14 +08:00
|
|
|
|
/// In order to get a valid rotation matrix, the input must be an
|
2021-02-22 21:26:25 +08:00
|
|
|
|
/// orthonormal basis, i.e., all vectors are normalized, and the are
|
|
|
|
|
/// all orthogonal to each other. These invariants are not checked
|
|
|
|
|
/// by this method.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_basis_unchecked(basis: &[Vector3<T>; 3]) -> Self {
|
2021-02-22 21:26:25 +08:00
|
|
|
|
let mat = Matrix3::from_columns(&basis[..]);
|
|
|
|
|
Self::from_matrix_unchecked(mat)
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// Builds a rotation matrix by extracting the rotation part of the given transformation `m`.
|
|
|
|
|
///
|
|
|
|
|
/// This is an iterative method. See `.from_matrix_eps` to provide mover
|
|
|
|
|
/// convergence parameters and starting solution.
|
|
|
|
|
/// This implements "A Robust Method to Extract the Rotational Part of Deformations" by Müller et al.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_matrix(m: &Matrix3<T>) -> Self
|
2020-11-21 18:21:47 +08:00
|
|
|
|
where
|
2022-04-01 20:30:20 +08:00
|
|
|
|
T: RealField,
|
2020-11-21 18:21:47 +08:00
|
|
|
|
{
|
2022-07-27 15:45:54 +08:00
|
|
|
|
Self::from_matrix_eps(m, T::default_epsilon(), 0, Self::identity())
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Builds a rotation matrix by extracting the rotation part of the given transformation `m`.
|
|
|
|
|
///
|
|
|
|
|
/// This implements "A Robust Method to Extract the Rotational Part of Deformations" by Müller et al.
|
|
|
|
|
///
|
|
|
|
|
/// # Parameters
|
|
|
|
|
///
|
|
|
|
|
/// * `m`: the matrix from which the rotational part is to be extracted.
|
|
|
|
|
/// * `eps`: the angular errors tolerated between the current rotation and the optimal one.
|
|
|
|
|
/// * `max_iter`: the maximum number of iterations. Loops indefinitely until convergence if set to `0`.
|
|
|
|
|
/// * `guess`: a guess of the solution. Convergence will be significantly faster if an initial solution close
|
|
|
|
|
/// to the actual solution is provided. Can be set to `Rotation3::identity()` if no other
|
|
|
|
|
/// guesses come to mind.
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn from_matrix_eps(m: &Matrix3<T>, eps: T, mut max_iter: usize, guess: Self) -> Self
|
2020-11-21 18:21:47 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-11-21 18:21:47 +08:00
|
|
|
|
{
|
|
|
|
|
if max_iter == 0 {
|
2022-03-31 23:28:31 +08:00
|
|
|
|
max_iter = usize::MAX;
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-27 15:45:54 +08:00
|
|
|
|
let mut perturbation_axes = Vector3::x_axis();
|
2020-11-21 18:21:47 +08:00
|
|
|
|
let mut rot = guess.into_inner();
|
|
|
|
|
|
|
|
|
|
for _ in 0..max_iter {
|
|
|
|
|
let axis = rot.column(0).cross(&m.column(0))
|
|
|
|
|
+ rot.column(1).cross(&m.column(1))
|
|
|
|
|
+ rot.column(2).cross(&m.column(2));
|
|
|
|
|
let denom = rot.column(0).dot(&m.column(0))
|
|
|
|
|
+ rot.column(1).dot(&m.column(1))
|
|
|
|
|
+ rot.column(2).dot(&m.column(2));
|
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
let axisangle = axis / (denom.abs() + T::default_epsilon());
|
2020-11-21 18:21:47 +08:00
|
|
|
|
|
2021-08-04 23:34:25 +08:00
|
|
|
|
if let Some((axis, angle)) = Unit::try_new_and_get(axisangle, eps.clone()) {
|
2020-11-21 18:21:47 +08:00
|
|
|
|
rot = Rotation3::from_axis_angle(&axis, angle) * rot;
|
|
|
|
|
} else {
|
2022-04-01 20:30:20 +08:00
|
|
|
|
// Check if stuck in a maximum w.r.t. the norm (m - rot).norm()
|
|
|
|
|
let mut perturbed = rot.clone();
|
|
|
|
|
let norm_squared = (m - &rot).norm_squared();
|
|
|
|
|
let mut new_norm_squared: T;
|
2022-07-27 15:45:54 +08:00
|
|
|
|
|
2022-04-01 20:30:20 +08:00
|
|
|
|
// Perturb until the new norm is significantly different
|
|
|
|
|
loop {
|
|
|
|
|
perturbed *= Rotation3::from_axis_angle(&perturbation_axes, T::frac_pi_8());
|
|
|
|
|
new_norm_squared = (m - &perturbed).norm_squared();
|
2022-07-27 15:45:54 +08:00
|
|
|
|
|
2022-04-01 20:30:20 +08:00
|
|
|
|
if relative_ne!(norm_squared, new_norm_squared) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-27 15:45:54 +08:00
|
|
|
|
|
2022-04-01 20:30:20 +08:00
|
|
|
|
// If new norm is larger, it's a minimum
|
|
|
|
|
if norm_squared < new_norm_squared {
|
|
|
|
|
break;
|
|
|
|
|
}
|
2022-07-27 15:45:54 +08:00
|
|
|
|
|
2022-04-01 20:30:20 +08:00
|
|
|
|
// If not, continue from perturbed rotation, but use a different axes for the next perturbation
|
2022-07-27 15:45:54 +08:00
|
|
|
|
perturbation_axes = UnitVector3::new_unchecked(perturbation_axes.yzx());
|
2022-04-01 20:30:20 +08:00
|
|
|
|
rot = perturbed;
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Self::from_matrix_unchecked(rot)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Ensure this rotation is an orthonormal rotation matrix. This is useful when repeated
|
|
|
|
|
/// computations might cause the matrix from progressively not being orthonormal anymore.
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn renormalize(&mut self)
|
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-11-21 18:21:47 +08:00
|
|
|
|
{
|
2021-08-04 23:34:25 +08:00
|
|
|
|
let mut c = UnitQuaternion::from(self.clone());
|
2020-11-21 18:21:47 +08:00
|
|
|
|
let _ = c.renormalize();
|
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
*self = Self::from_matrix_eps(self.matrix(), T::default_epsilon(), 0, c.into())
|
2020-11-21 18:21:47 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// # 3D axis and angle extraction
|
2021-04-11 17:00:38 +08:00
|
|
|
|
impl<T: SimdRealField> Rotation3<T> {
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// The rotation angle in [0; pi].
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Unit, Rotation3, Vector3};
|
|
|
|
|
/// let axis = Unit::new_normalize(Vector3::new(1.0, 2.0, 3.0));
|
|
|
|
|
/// let rot = Rotation3::from_axis_angle(&axis, 1.78);
|
2018-12-18 22:46:29 +08:00
|
|
|
|
/// assert_relative_eq!(rot.angle(), 1.78);
|
2018-11-04 15:47:17 +08:00
|
|
|
|
/// ```
|
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 angle(&self) -> T {
|
2021-08-04 23:34:25 +08:00
|
|
|
|
((self.matrix()[(0, 0)].clone()
|
|
|
|
|
+ self.matrix()[(1, 1)].clone()
|
|
|
|
|
+ self.matrix()[(2, 2)].clone()
|
|
|
|
|
- T::one())
|
2019-03-23 21:29:07 +08:00
|
|
|
|
/ crate::convert(2.0))
|
2020-03-22 06:22:55 +08:00
|
|
|
|
.simd_acos()
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The rotation axis. Returns `None` if the rotation angle is zero or PI.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3, Unit};
|
|
|
|
|
/// let axis = Unit::new_normalize(Vector3::new(1.0, 2.0, 3.0));
|
|
|
|
|
/// let angle = 1.2;
|
|
|
|
|
/// let rot = Rotation3::from_axis_angle(&axis, angle);
|
|
|
|
|
/// assert_relative_eq!(rot.axis().unwrap(), axis);
|
|
|
|
|
///
|
|
|
|
|
/// // Case with a zero angle.
|
|
|
|
|
/// let rot = Rotation3::from_axis_angle(&axis, 0.0);
|
|
|
|
|
/// assert!(rot.axis().is_none());
|
|
|
|
|
/// ```
|
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 axis(&self) -> Option<Unit<Vector3<T>>>
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-04-06 00:49:48 +08:00
|
|
|
|
{
|
2021-08-04 23:34:25 +08:00
|
|
|
|
let rotmat = self.matrix();
|
2021-04-11 17:00:38 +08:00
|
|
|
|
let axis = SVector::<T, 3>::new(
|
2021-08-04 23:34:25 +08:00
|
|
|
|
rotmat[(2, 1)].clone() - rotmat[(1, 2)].clone(),
|
|
|
|
|
rotmat[(0, 2)].clone() - rotmat[(2, 0)].clone(),
|
|
|
|
|
rotmat[(1, 0)].clone() - rotmat[(0, 1)].clone(),
|
2018-02-02 19:26:35 +08:00
|
|
|
|
);
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Unit::try_new(axis, T::default_epsilon())
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// The rotation axis multiplied by the rotation angle.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3, Unit};
|
|
|
|
|
/// let axisangle = Vector3::new(0.1, 0.2, 0.3);
|
|
|
|
|
/// let rot = Rotation3::new(axisangle);
|
|
|
|
|
/// assert_relative_eq!(rot.scaled_axis(), axisangle, epsilon = 1.0e-6);
|
|
|
|
|
/// ```
|
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 scaled_axis(&self) -> Vector3<T>
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-04-06 00:49:48 +08:00
|
|
|
|
{
|
2016-12-05 05:44:42 +08:00
|
|
|
|
if let Some(axis) = self.axis() {
|
2018-12-10 04:08:14 +08:00
|
|
|
|
axis.into_inner() * self.angle()
|
2018-02-02 19:26:35 +08:00
|
|
|
|
} else {
|
2017-08-03 01:37:44 +08:00
|
|
|
|
Vector::zero()
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-26 09:54:14 +08:00
|
|
|
|
/// The rotation axis and angle in ]0, pi] of this rotation matrix.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// Returns `None` if the angle is zero.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3, Unit};
|
|
|
|
|
/// let axis = Unit::new_normalize(Vector3::new(1.0, 2.0, 3.0));
|
|
|
|
|
/// let angle = 1.2;
|
|
|
|
|
/// let rot = Rotation3::from_axis_angle(&axis, angle);
|
|
|
|
|
/// let axis_angle = rot.axis_angle().unwrap();
|
|
|
|
|
/// assert_relative_eq!(axis_angle.0, axis);
|
|
|
|
|
/// assert_relative_eq!(axis_angle.1, angle);
|
|
|
|
|
///
|
|
|
|
|
/// // Case with a zero angle.
|
|
|
|
|
/// let rot = Rotation3::from_axis_angle(&axis, 0.0);
|
|
|
|
|
/// assert!(rot.axis_angle().is_none());
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
2021-06-07 22:34:03 +08:00
|
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn axis_angle(&self) -> Option<(Unit<Vector3<T>>, T)>
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-04-06 00:49:48 +08:00
|
|
|
|
{
|
2021-06-18 15:45:37 +08:00
|
|
|
|
self.axis().map(|axis| (axis, self.angle()))
|
2018-11-04 15:47:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-05 05:44:42 +08:00
|
|
|
|
/// The rotation angle needed to make `self` and `other` coincide.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
/// ```
|
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::{Rotation3, Vector3};
|
|
|
|
|
/// let rot1 = Rotation3::from_axis_angle(&Vector3::y_axis(), 1.0);
|
|
|
|
|
/// let rot2 = Rotation3::from_axis_angle(&Vector3::x_axis(), 0.1);
|
|
|
|
|
/// assert_relative_eq!(rot1.angle_to(&rot2), 1.0045657, epsilon = 1.0e-6);
|
|
|
|
|
/// ```
|
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 angle_to(&self, other: &Self) -> T
|
2020-11-21 18:21:47 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
2020-11-21 18:21:47 +08:00
|
|
|
|
{
|
2016-12-05 05:44:42 +08:00
|
|
|
|
self.rotation_to(other).angle()
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// Creates Euler angles from a rotation.
|
2018-11-04 15:47:17 +08:00
|
|
|
|
///
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// The angles are produced in the form (roll, pitch, yaw).
|
|
|
|
|
#[deprecated(note = "This is renamed to use `.euler_angles()`.")]
|
2021-07-06 05:51:27 +08:00
|
|
|
|
pub fn to_euler_angles(self) -> (T, T, T)
|
2020-04-06 00:49:48 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-04-06 00:49:48 +08:00
|
|
|
|
{
|
2020-11-21 18:21:47 +08:00
|
|
|
|
self.euler_angles()
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
2020-10-25 21:00:47 +08:00
|
|
|
|
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// Euler angles corresponding to this rotation from a rotation.
|
2020-10-25 21:00:47 +08:00
|
|
|
|
///
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// The angles are produced in the form (roll, pitch, yaw).
|
2020-10-25 21:00:47 +08:00
|
|
|
|
///
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # Example
|
2020-10-25 21:00:47 +08:00
|
|
|
|
/// ```
|
2020-11-21 18:21:47 +08:00
|
|
|
|
/// # #[macro_use] extern crate approx;
|
|
|
|
|
/// # use nalgebra::Rotation3;
|
|
|
|
|
/// let rot = Rotation3::from_euler_angles(0.1, 0.2, 0.3);
|
|
|
|
|
/// let euler = rot.euler_angles();
|
|
|
|
|
/// assert_relative_eq!(euler.0, 0.1, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(euler.1, 0.2, epsilon = 1.0e-6);
|
|
|
|
|
/// assert_relative_eq!(euler.2, 0.3, epsilon = 1.0e-6);
|
2020-10-25 21:00:47 +08:00
|
|
|
|
/// ```
|
2021-06-07 22:34:03 +08:00
|
|
|
|
#[must_use]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
pub fn euler_angles(&self) -> (T, T, T)
|
2020-10-25 21:00:47 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T: RealField,
|
2020-10-25 21:00:47 +08:00
|
|
|
|
{
|
2020-11-21 18:21:47 +08:00
|
|
|
|
// Implementation informed by "Computing Euler angles from a rotation matrix", by Gregory G. Slabaugh
|
|
|
|
|
// https://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.371.6578
|
2021-08-04 23:34:25 +08:00
|
|
|
|
if self[(2, 0)].clone().abs() < T::one() {
|
|
|
|
|
let yaw = -self[(2, 0)].clone().asin();
|
|
|
|
|
let roll = (self[(2, 1)].clone() / yaw.clone().cos())
|
|
|
|
|
.atan2(self[(2, 2)].clone() / yaw.clone().cos());
|
|
|
|
|
let pitch = (self[(1, 0)].clone() / yaw.clone().cos())
|
|
|
|
|
.atan2(self[(0, 0)].clone() / yaw.clone().cos());
|
2020-11-21 18:21:47 +08:00
|
|
|
|
(roll, yaw, pitch)
|
2021-08-04 23:34:25 +08:00
|
|
|
|
} else if self[(2, 0)].clone() <= -T::one() {
|
|
|
|
|
(
|
|
|
|
|
self[(0, 1)].clone().atan2(self[(0, 2)].clone()),
|
|
|
|
|
T::frac_pi_2(),
|
|
|
|
|
T::zero(),
|
|
|
|
|
)
|
2020-11-21 18:21:47 +08:00
|
|
|
|
} else {
|
|
|
|
|
(
|
2021-08-04 23:34:25 +08:00
|
|
|
|
-self[(0, 1)].clone().atan2(-self[(0, 2)].clone()),
|
2021-04-11 17:00:38 +08:00
|
|
|
|
-T::frac_pi_2(),
|
|
|
|
|
T::zero(),
|
2020-11-21 18:21:47 +08:00
|
|
|
|
)
|
|
|
|
|
}
|
2020-10-25 21:00:47 +08:00
|
|
|
|
}
|
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: SimdRealField> Distribution<Rotation3<T>> for Standard
|
2020-03-22 06:22:55 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
|
|
|
|
OpenClosed01: Distribution<T>,
|
|
|
|
|
T: SampleUniform,
|
2018-05-23 05:58:14 +08:00
|
|
|
|
{
|
2018-07-08 07:03:08 +08:00
|
|
|
|
/// Generate a uniformly distributed random rotation.
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-04-11 17:00:38 +08:00
|
|
|
|
fn sample<'a, R: Rng + ?Sized>(&self, rng: &mut R) -> Rotation3<T> {
|
2018-07-04 10:06:03 +08:00
|
|
|
|
// James Arvo.
|
|
|
|
|
// Fast random rotation matrices.
|
|
|
|
|
// In D. Kirk, editor, Graphics Gems III, pages 117-120. Academic, New York, 1992.
|
|
|
|
|
|
|
|
|
|
// Compute a random rotation around Z
|
2021-04-11 17:00:38 +08:00
|
|
|
|
let twopi = Uniform::new(T::zero(), T::simd_two_pi());
|
2021-04-10 14:13:46 +08:00
|
|
|
|
let theta = rng.sample(&twopi);
|
2020-03-22 06:22:55 +08:00
|
|
|
|
let (ts, tc) = theta.simd_sin_cos();
|
2021-04-11 17:00:38 +08:00
|
|
|
|
let a = SMatrix::<T, 3, 3>::new(
|
2021-08-04 23:34:25 +08:00
|
|
|
|
tc.clone(),
|
|
|
|
|
ts.clone(),
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::zero(),
|
2018-10-22 13:00:10 +08:00
|
|
|
|
-ts,
|
|
|
|
|
tc,
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::zero(),
|
|
|
|
|
T::zero(),
|
|
|
|
|
T::zero(),
|
|
|
|
|
T::one(),
|
2018-07-04 10:06:03 +08:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Compute a random rotation *of* Z
|
2021-04-10 14:13:46 +08:00
|
|
|
|
let phi = rng.sample(&twopi);
|
2018-07-04 10:06:03 +08:00
|
|
|
|
let z = rng.sample(OpenClosed01);
|
2020-03-22 06:22:55 +08:00
|
|
|
|
let (ps, pc) = phi.simd_sin_cos();
|
2021-08-04 23:34:25 +08:00
|
|
|
|
let sqrt_z = z.clone().simd_sqrt();
|
|
|
|
|
let v = Vector3::new(pc * sqrt_z.clone(), ps * sqrt_z, (T::one() - z).simd_sqrt());
|
|
|
|
|
let mut b = v.clone() * v.transpose();
|
|
|
|
|
b += b.clone();
|
2021-04-11 17:00:38 +08:00
|
|
|
|
b -= SMatrix::<T, 3, 3>::identity();
|
2018-07-04 10:06:03 +08:00
|
|
|
|
|
|
|
|
|
Rotation3::from_matrix_unchecked(b * a)
|
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: SimdRealField + Arbitrary> Arbitrary for Rotation3<T>
|
2018-02-02 19:26:35 +08:00
|
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
|
T::Element: SimdRealField,
|
2021-08-03 00:41:46 +08:00
|
|
|
|
Owned<T, U3, U3>: Send,
|
|
|
|
|
Owned<T, U3>: Send,
|
2018-02-02 19:26:35 +08:00
|
|
|
|
{
|
2016-12-05 05:44:42 +08:00
|
|
|
|
#[inline]
|
2021-03-01 00:52:14 +08:00
|
|
|
|
fn arbitrary(g: &mut Gen) -> Self {
|
2021-04-11 17:00:38 +08:00
|
|
|
|
Self::new(SVector::arbitrary(g))
|
2016-12-05 05:44:42 +08:00
|
|
|
|
}
|
|
|
|
|
}
|