1
use crate::int::Sealed as _;
2
use crate::{Denominator, Integer, NotOne, Numerator, Recip, Q};
3
use core::ops::{Div, Rem};
4
use typenum::{Mod, Quot};
5

            
6
pub trait PrivateF64Helper {
7
    const F64: f64;
8
}
9

            
10
impl<N> PrivateF64Helper for Q<N>
11
where
12
    N: Numerator,
13
{
14
    const F64: f64 = N::F64;
15
}
16

            
17
impl<N, D> PrivateF64Helper for Q<N, D>
18
where
19
    N: Numerator<D> + Div<D> + Rem<D>,
20
    D: Denominator + NotOne + Numerator,
21
    Quot<N, D>: Integer,
22
    Mod<N, D>: Numerator<D>,
23
    Q<Mod<N, D>, D>: Recip,
24
    <Q<Mod<N, D>, D> as Recip>::Output: PrivateF64Helper,
25
{
26
    const F64: f64 =
27
        Quot::<N, D>::F64 + 1_f64 / <<Q<Mod<N, D>, D> as Recip>::Output as PrivateF64Helper>::F64;
28
}
29

            
30
pub trait Sealed: PrivateF64Helper {
31
    const F64: f64 = <Self as PrivateF64Helper>::F64;
32
}
33

            
34
impl<N, D> Sealed for Q<N, D>
35
where
36
    Self: PrivateF64Helper,
37
    N: Numerator<D>,
38
    D: Denominator,
39
{
40
}
41

            
42
/// Type-level rational numbers.
43
pub trait Rational: 'static + Default + Copy + Sealed {
44
    /// Approximate 64-bit floating point number of this rational number.
45
    ///
46
    /// # Examples
47
    ///
48
    /// ```
49
    /// use typerat::*;
50
    ///
51
    /// assert_eq!(Q::<Z0>::F64, 0_f64);
52
    /// assert_eq!(Q::<P1>::F64, 1_f64);
53
    /// assert_eq!(Q::<N1>::F64, -1_f64);
54
    /// assert_eq!(Q::<N1, P2>::F64, -0.5_f64);
55
    /// assert_eq!(Q::<P1, P3>::F64, 1_f64 / 3_f64);
56
    /// assert_eq!(Q::<N5, P3>::F64, -1_f64 - 1_f64 / 1.5_f64);
57
    /// ```
58
    const F64: f64;
59

            
60
    /// Borrows an instance of this type-level rational number.
61
    fn borrow<'a>() -> &'a Self;
62
}
63

            
64
impl<N, D> Rational for Q<N, D>
65
where
66
    Self: Sealed,
67
    N: Numerator<D>,
68
    D: Denominator,
69
{
70
    const F64: f64 = <Self as PrivateF64Helper>::F64;
71

            
72
1
    fn borrow<'a>() -> &'a Self {
73
1
        &Self::SELF
74
1
    }
75
}
76

            
77
#[cfg(test)]
78
mod tests {
79
    use super::*;
80
    use typenum::consts::*;
81

            
82
1
    fn default_and_borrow<'a, T>() -> (T, &'a T)
83
1
    where
84
1
        T: Rational,
85
1
    {
86
1
        (T::default(), T::borrow())
87
1
    }
88

            
89
    #[test]
90
1
    fn test_borrow() {
91
1
        let (a, b) = default_and_borrow::<Q<P1, P2>>();
92
1

            
93
1
        assert_eq!(&a, b);
94
1
    }
95
}