1 | use super::Complex; |
2 | use num_traits::{AsPrimitive, FromPrimitive, Num, NumCast, ToPrimitive}; |
3 | |
4 | macro_rules! impl_to_primitive { |
5 | ($ty:ty, $to:ident) => { |
6 | #[inline] |
7 | fn $to(&self) -> Option<$ty> { |
8 | if self.im.is_zero() { |
9 | self.re.$to() |
10 | } else { |
11 | None |
12 | } |
13 | } |
14 | }; |
15 | } // impl_to_primitive |
16 | |
17 | // Returns None if Complex part is non-zero |
18 | impl<T: ToPrimitive + Num> ToPrimitive for Complex<T> { |
19 | impl_to_primitive!(usize, to_usize); |
20 | impl_to_primitive!(isize, to_isize); |
21 | impl_to_primitive!(u8, to_u8); |
22 | impl_to_primitive!(u16, to_u16); |
23 | impl_to_primitive!(u32, to_u32); |
24 | impl_to_primitive!(u64, to_u64); |
25 | impl_to_primitive!(i8, to_i8); |
26 | impl_to_primitive!(i16, to_i16); |
27 | impl_to_primitive!(i32, to_i32); |
28 | impl_to_primitive!(i64, to_i64); |
29 | impl_to_primitive!(u128, to_u128); |
30 | impl_to_primitive!(i128, to_i128); |
31 | impl_to_primitive!(f32, to_f32); |
32 | impl_to_primitive!(f64, to_f64); |
33 | } |
34 | |
35 | macro_rules! impl_from_primitive { |
36 | ($ty:ty, $from_xx:ident) => { |
37 | #[inline] |
38 | fn $from_xx(n: $ty) -> Option<Self> { |
39 | Some(Complex { |
40 | re: T::$from_xx(n)?, |
41 | im: T::zero(), |
42 | }) |
43 | } |
44 | }; |
45 | } // impl_from_primitive |
46 | |
47 | impl<T: FromPrimitive + Num> FromPrimitive for Complex<T> { |
48 | impl_from_primitive!(usize, from_usize); |
49 | impl_from_primitive!(isize, from_isize); |
50 | impl_from_primitive!(u8, from_u8); |
51 | impl_from_primitive!(u16, from_u16); |
52 | impl_from_primitive!(u32, from_u32); |
53 | impl_from_primitive!(u64, from_u64); |
54 | impl_from_primitive!(i8, from_i8); |
55 | impl_from_primitive!(i16, from_i16); |
56 | impl_from_primitive!(i32, from_i32); |
57 | impl_from_primitive!(i64, from_i64); |
58 | impl_from_primitive!(u128, from_u128); |
59 | impl_from_primitive!(i128, from_i128); |
60 | impl_from_primitive!(f32, from_f32); |
61 | impl_from_primitive!(f64, from_f64); |
62 | } |
63 | |
64 | impl<T: NumCast + Num> NumCast for Complex<T> { |
65 | fn from<U: ToPrimitive>(n: U) -> Option<Self> { |
66 | Some(Complex { |
67 | re: T::from(n)?, |
68 | im: T::zero(), |
69 | }) |
70 | } |
71 | } |
72 | |
73 | impl<T, U> AsPrimitive<U> for Complex<T> |
74 | where |
75 | T: AsPrimitive<U>, |
76 | U: 'static + Copy, |
77 | { |
78 | fn as_(self) -> U { |
79 | self.re.as_() |
80 | } |
81 | } |
82 | |
83 | #[cfg (test)] |
84 | mod test { |
85 | use super::*; |
86 | |
87 | #[test ] |
88 | fn test_to_primitive() { |
89 | let a: Complex<u32> = Complex { re: 3, im: 0 }; |
90 | assert_eq!(a.to_i32(), Some(3_i32)); |
91 | let b: Complex<u32> = Complex { re: 3, im: 1 }; |
92 | assert_eq!(b.to_i32(), None); |
93 | let x: Complex<f32> = Complex { re: 1.0, im: 0.1 }; |
94 | assert_eq!(x.to_f32(), None); |
95 | let y: Complex<f32> = Complex { re: 1.0, im: 0.0 }; |
96 | assert_eq!(y.to_f32(), Some(1.0)); |
97 | let z: Complex<f32> = Complex { re: 1.0, im: 0.0 }; |
98 | assert_eq!(z.to_i32(), Some(1)); |
99 | } |
100 | |
101 | #[test ] |
102 | fn test_from_primitive() { |
103 | let a: Complex<f32> = FromPrimitive::from_i32(2).unwrap(); |
104 | assert_eq!(a, Complex { re: 2.0, im: 0.0 }); |
105 | } |
106 | |
107 | #[test ] |
108 | fn test_num_cast() { |
109 | let a: Complex<f32> = NumCast::from(2_i32).unwrap(); |
110 | assert_eq!(a, Complex { re: 2.0, im: 0.0 }); |
111 | } |
112 | |
113 | #[test ] |
114 | fn test_as_primitive() { |
115 | let a: Complex<f32> = Complex { re: 2.0, im: 0.2 }; |
116 | let a_: i32 = a.as_(); |
117 | assert_eq!(a_, 2_i32); |
118 | } |
119 | } |
120 | |