1
use crate::{Denominator, NonZero, Numerator, Q};
2
use core::ops::{Mul, Rem};
3
use typenum::{Abs, AbsVal, Gcd, Gcf, Max, Maximum, Mod, PartialDiv, Prod, P1};
4

            
5
pub trait PrivateRemHelper<Rhs> {
6
    type Output;
7
}
8

            
9
impl<Nl, Dl, Nr, Dr, No, Do> PrivateRemHelper<Q<Nr, Dr>> for Q<Nl, Dl>
10
where
11
    Q<Nr, Dr>: NonZero,
12
    Nl: Numerator<Dl> + Mul<Dr>,
13
    Dl: Denominator + Mul<Dr>,
14
    Nr: Numerator<Dr> + Mul<Dl>,
15
    Dr: Denominator,
16
    No: Numerator<Do>,
17
    Do: Denominator,
18
    Prod<Nr, Dl>: Abs,
19
    AbsVal<Prod<Nr, Dl>>: Max<P1>,
20
    Prod<Nl, Dr>: Rem<Maximum<AbsVal<Prod<Nr, Dl>>, P1>>,
21
    Mod<Prod<Nl, Dr>, Maximum<AbsVal<Prod<Nr, Dl>>, P1>>: Gcd<Prod<Dl, Dr>>
22
        + PartialDiv<
23
            Gcf<Mod<Prod<Nl, Dr>, Maximum<AbsVal<Prod<Nr, Dl>>, P1>>, Prod<Dl, Dr>>,
24
            Output = No,
25
        >,
26
    Prod<Dl, Dr>: PartialDiv<
27
        Gcf<Mod<Prod<Nl, Dr>, Maximum<AbsVal<Prod<Nr, Dl>>, P1>>, Prod<Dl, Dr>>,
28
        Output = Do,
29
    >,
30
{
31
    type Output = Q<No, Do>;
32
}
33

            
34
impl<Nl, Dl, Nr, Dr, No, Do> Rem<Q<Nr, Dr>> for Q<Nl, Dl>
35
where
36
    Self: PrivateRemHelper<Q<Nr, Dr>, Output = Q<No, Do>>,
37
    Q<Nr, Dr>: NonZero,
38
    Nl: Numerator<Dl>,
39
    Dl: Denominator,
40
    Nr: Numerator<Dr>,
41
    Dr: Denominator,
42
    No: Numerator<Do>,
43
    Do: Denominator,
44
{
45
    type Output = Q<No, Do>;
46

            
47
6
    fn rem(self, rhs: Q<Nr, Dr>) -> Self::Output {
48
6
        let _ = rhs;
49
6
        Self::Output::new()
50
6
    }
51
}
52

            
53
#[cfg(test)]
54
mod tests {
55
    use super::*;
56
    use typenum::consts::*;
57

            
58
    #[test]
59
1
    fn test_rem() {
60
1
        // zero % positive
61
1
        assert!(Q::<Z0>::new() % Q::<P1, P2>::new() == Q::<Z0>::new());
62

            
63
        // zero % negative
64
1
        assert!(Q::<Z0>::new() % Q::<N1, P2>::new() == Q::<Z0>::new());
65

            
66
        // positive % zero
67
        // let _ = Q::<P1, P2>::new() % Q::<Z0>::new();
68

            
69
        // positive % positive
70
1
        assert!(Q::<P2, P3>::new() % Q::<P1, P2>::new() == Q::<P1, P6>::new());
71

            
72
        // positive % negative
73
1
        assert!(Q::<P2, P3>::new() % Q::<N1, P2>::new() == Q::<P1, P6>::new());
74

            
75
        // negative % positive
76
1
        assert!(Q::<N2, P3>::new() % Q::<P1, P2>::new() == Q::<N1, P6>::new());
77

            
78
        // negative % negative
79
1
        assert!(Q::<N2, P3>::new() % Q::<N1, P2>::new() == Q::<N1, P6>::new());
80
1
    }
81
}