use super::Complex; use num_traits::{AsPrimitive, FromPrimitive, Num, NumCast, ToPrimitive}; macro_rules! impl_to_primitive { ($ty:ty, $to:ident) => { #[inline] fn $to(&self) -> Option<$ty> { if self.im.is_zero() { self.re.$to() } else { None } } }; } // impl_to_primitive // Returns None if Complex part is non-zero impl ToPrimitive for Complex { impl_to_primitive!(usize, to_usize); impl_to_primitive!(isize, to_isize); impl_to_primitive!(u8, to_u8); impl_to_primitive!(u16, to_u16); impl_to_primitive!(u32, to_u32); impl_to_primitive!(u64, to_u64); impl_to_primitive!(i8, to_i8); impl_to_primitive!(i16, to_i16); impl_to_primitive!(i32, to_i32); impl_to_primitive!(i64, to_i64); impl_to_primitive!(u128, to_u128); impl_to_primitive!(i128, to_i128); impl_to_primitive!(f32, to_f32); impl_to_primitive!(f64, to_f64); } macro_rules! impl_from_primitive { ($ty:ty, $from_xx:ident) => { #[inline] fn $from_xx(n: $ty) -> Option { Some(Complex { re: T::$from_xx(n)?, im: T::zero(), }) } }; } // impl_from_primitive impl FromPrimitive for Complex { impl_from_primitive!(usize, from_usize); impl_from_primitive!(isize, from_isize); impl_from_primitive!(u8, from_u8); impl_from_primitive!(u16, from_u16); impl_from_primitive!(u32, from_u32); impl_from_primitive!(u64, from_u64); impl_from_primitive!(i8, from_i8); impl_from_primitive!(i16, from_i16); impl_from_primitive!(i32, from_i32); impl_from_primitive!(i64, from_i64); impl_from_primitive!(u128, from_u128); impl_from_primitive!(i128, from_i128); impl_from_primitive!(f32, from_f32); impl_from_primitive!(f64, from_f64); } impl NumCast for Complex { fn from(n: U) -> Option { Some(Complex { re: T::from(n)?, im: T::zero(), }) } } impl AsPrimitive for Complex where T: AsPrimitive, U: 'static + Copy, { fn as_(self) -> U { self.re.as_() } } #[cfg(test)] mod test { use super::*; #[test] fn test_to_primitive() { let a: Complex = Complex { re: 3, im: 0 }; assert_eq!(a.to_i32(), Some(3_i32)); let b: Complex = Complex { re: 3, im: 1 }; assert_eq!(b.to_i32(), None); let x: Complex = Complex { re: 1.0, im: 0.1 }; assert_eq!(x.to_f32(), None); let y: Complex = Complex { re: 1.0, im: 0.0 }; assert_eq!(y.to_f32(), Some(1.0)); let z: Complex = Complex { re: 1.0, im: 0.0 }; assert_eq!(z.to_i32(), Some(1)); } #[test] fn test_from_primitive() { let a: Complex = FromPrimitive::from_i32(2).unwrap(); assert_eq!(a, Complex { re: 2.0, im: 0.0 }); } #[test] fn test_num_cast() { let a: Complex = NumCast::from(2_i32).unwrap(); assert_eq!(a, Complex { re: 2.0, im: 0.0 }); } #[test] fn test_as_primitive() { let a: Complex = Complex { re: 2.0, im: 0.2 }; let a_: i32 = a.as_(); assert_eq!(a_, 2_i32); } }