1
use crate::{Denominator, Numerator, Q};
2
use core::ops::{Add, Mul};
3
use typenum::{Gcd, Gcf, PartialDiv, Prod, Sum};
4

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

            
9
impl<Nl, Dl, Nr, Dr, No, Do> PrivateAddHelper<Q<Nr, Dr>> for Q<Nl, Dl>
10
where
11
    Nl: Numerator<Dl> + Mul<Dr>,
12
    Dl: Denominator + Mul<Dr>,
13
    Nr: Numerator<Dr> + Mul<Dl>,
14
    Dr: Denominator,
15
    No: Numerator<Do>,
16
    Do: Denominator,
17
    Prod<Nl, Dr>: Add<Prod<Nr, Dl>>,
18
    Sum<Prod<Nl, Dr>, Prod<Nr, Dl>>: Gcd<Prod<Dl, Dr>>
19
        + PartialDiv<Gcf<Sum<Prod<Nl, Dr>, Prod<Nr, Dl>>, Prod<Dl, Dr>>, Output = No>,
20
    Prod<Dl, Dr>: PartialDiv<Gcf<Sum<Prod<Nl, Dr>, Prod<Nr, Dl>>, Prod<Dl, Dr>>, Output = Do>,
21
{
22
    type Output = Q<No, Do>;
23
}
24

            
25
impl<Nl, Dl, Nr, Dr, No, Do> Add<Q<Nr, Dr>> for Q<Nl, Dl>
26
where
27
    Self: PrivateAddHelper<Q<Nr, Dr>, Output = Q<No, Do>>,
28
    Nl: Numerator<Dl>,
29
    Dl: Denominator,
30
    Nr: Numerator<Dr>,
31
    Dr: Denominator,
32
    No: Numerator<Do>,
33
    Do: Denominator,
34
{
35
    type Output = Q<No, Do>;
36

            
37
9
    fn add(self, rhs: Q<Nr, Dr>) -> Self::Output {
38
9
        let _ = rhs;
39
9
        Self::Output::new()
40
9
    }
41
}
42

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

            
48
    #[test]
49
1
    fn test_add() {
50
1
        // zero + zero
51
1
        assert!(Q::<Z0>::new() + Q::<Z0>::new() == Q::<Z0>::new());
52

            
53
        // zero + positive
54
1
        assert!(Q::<Z0>::new() + Q::<P1, P2>::new() == Q::<P1, P2>::new());
55

            
56
        // zero + negative
57
1
        assert!(Q::<Z0>::new() + Q::<N1, P2>::new() == Q::<N1, P2>::new());
58

            
59
        // positive + zero
60
1
        assert!(Q::<P1, P2>::new() + Q::<Z0>::new() == Q::<P1, P2>::new());
61

            
62
        // positive + positive
63
1
        assert!(Q::<P1, P2>::new() + Q::<P1, P3>::new() == Q::<P5, P6>::new());
64

            
65
        // positive + negative
66
1
        assert!(Q::<P1, P2>::new() + Q::<N1, P3>::new() == Q::<P1, P6>::new());
67

            
68
        // negative + zero
69
1
        assert!(Q::<N1, P2>::new() + Q::<Z0>::new() == Q::<N1, P2>::new());
70

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

            
74
        // negative + negative
75
1
        assert!(Q::<N1, P2>::new() + Q::<N1, P3>::new() == Q::<N5, P6>::new());
76
1
    }
77
}