#[cfg(feature = "arbitrary")] use quickcheck::{Arbitrary, Gen}; #[cfg(feature = "arbitrary")] use core::storage::Owned; use num_complex::Complex; use alga::general::Real; use core::{DefaultAllocator, MatrixN}; use core::dimension::{Dim, Dynamic, U2}; use core::allocator::Allocator; use geometry::UnitComplex; /// A random orthogonal matrix. #[derive(Clone, Debug)] pub struct RandomOrthogonal where DefaultAllocator: Allocator, { m: MatrixN, } impl RandomOrthogonal where DefaultAllocator: Allocator, { /// Retrieve the generated matrix. pub fn unwrap(self) -> MatrixN { self.m } /// Creates a new random orthogonal matrix from its dimension and a random reals generators. pub fn new N>(dim: D, mut rand: Rand) -> Self { let mut res = MatrixN::identity_generic(dim, dim); // Create an orthogonal matrix by compositing planar 2D rotations. for i in 0..dim.value() - 1 { let c = Complex::new(rand(), rand()); let rot: UnitComplex = UnitComplex::from_complex(c); rot.rotate(&mut res.fixed_rows_mut::(i)); } RandomOrthogonal { m: res } } } #[cfg(feature = "arbitrary")] impl Arbitrary for RandomOrthogonal where DefaultAllocator: Allocator, Owned: Clone + Send, { fn arbitrary(g: &mut G) -> Self { let dim = D::try_to_usize().unwrap_or(g.gen_range(1, 50)); Self::new(D::from_usize(dim), || N::arbitrary(g)) } }