1
use crate::{Denominator, NonZeroUnsigned, Numerator, Q};
2
use typenum::{NInt, PInt};
3

            
4
/// The unary reciprocal operation.
5
pub trait Recip {
6
    /// The resulting type after applying the reciprocal operation.
7
    type Output;
8

            
9
    /// Performs the reciprocal operation.
10
    fn recip(self) -> Self::Output;
11
}
12

            
13
impl<Un, Ud> Recip for Q<PInt<Un>, PInt<Ud>>
14
where
15
    Un: NonZeroUnsigned,
16
    Ud: NonZeroUnsigned,
17
    PInt<Un>: Numerator<PInt<Ud>> + Denominator,
18
    PInt<Ud>: Denominator + Numerator<PInt<Un>>,
19
{
20
    type Output = Q<PInt<Ud>, PInt<Un>>;
21

            
22
1
    fn recip(self) -> Self::Output {
23
1
        Self::Output::new()
24
1
    }
25
}
26

            
27
impl<Un, Ud> Recip for Q<NInt<Un>, PInt<Ud>>
28
where
29
    Un: NonZeroUnsigned,
30
    Ud: NonZeroUnsigned,
31
    NInt<Un>: Numerator<PInt<Ud>>,
32
    PInt<Ud>: Denominator,
33
    NInt<Ud>: Numerator<PInt<Un>>,
34
    PInt<Un>: Denominator,
35
{
36
    type Output = Q<NInt<Ud>, PInt<Un>>;
37

            
38
1
    fn recip(self) -> Self::Output {
39
1
        Self::Output::new()
40
1
    }
41
}
42

            
43
#[cfg(test)]
44
mod tests {
45
    use super::*;
46
    use typenum::consts::*;
47

            
48
    #[test]
49
1
    fn test_recip() {
50
1
        // positive
51
1
        assert!(Q::<P1, P2>::new().recip() == Q::<P2>::new());
52

            
53
        // negative
54
1
        assert!(Q::<N1, P3>::new().recip() == Q::<N3>::new());
55
1
    }
56
}