1
use crate::NonZeroUnsigned;
2
use typenum::{NInt, PInt, Z0};
3

            
4
pub trait Sealed {
5
    const F64: f64;
6
}
7

            
8
impl Sealed for Z0 {
9
    const F64: f64 = 0_f64;
10
}
11

            
12
impl<U> Sealed for PInt<U>
13
where
14
    U: NonZeroUnsigned,
15
{
16
    const F64: f64 = U::F64;
17
}
18

            
19
impl<U> Sealed for NInt<U>
20
where
21
    U: NonZeroUnsigned,
22
{
23
    const F64: f64 = -U::F64;
24
}
25

            
26
/// Type-level integers.
27
///
28
/// # Examples
29
///
30
/// This trait is basically implemented for types that implement [typenum]'s `Integer`.
31
///
32
/// ```
33
/// use typerat::*;
34
/// use typenum::{PInt, UInt, UTerm};
35
///
36
/// fn to_i8_typenum<I: typenum::Integer>() -> i8 {
37
///     I::to_i8()
38
/// }
39
///
40
/// fn to_i8_typerat<I: Integer>() -> i8 {
41
///     I::to_i8()
42
/// }
43
///
44
/// assert_eq!(to_i8_typenum::<Z0>(), 0_i8);
45
/// assert_eq!(to_i8_typerat::<Z0>(), 0_i8);
46
///
47
/// assert_eq!(to_i8_typenum::<P1>(), 1_i8);
48
/// assert_eq!(to_i8_typerat::<P1>(), 1_i8);
49
///
50
/// assert_eq!(to_i8_typenum::<N1>(), -1_i8);
51
/// assert_eq!(to_i8_typerat::<N1>(), -1_i8);
52
/// ```
53
///
54
/// Unlike [typenum], compile time unsigned integers with leading zeros are not allowed.
55
///
56
/// ```
57
/// # use typerat::*;
58
/// # use typenum::{PInt, UInt, UTerm};
59
/// #
60
/// # fn to_i8_typenum<I: typenum::Integer>() -> i8 {
61
/// #     I::to_i8()
62
/// # }
63
/// #
64
/// assert_eq!(to_i8_typenum::<PInt<UInt<UTerm, B0>>>(), 0_i8);
65
/// ```
66
///
67
/// ```compile_fail
68
/// # use typerat::*;
69
/// # use typenum::{PInt, UInt, UTerm};
70
/// #
71
/// # fn to_i8_typerat<I: Integer>() -> i8 {
72
/// #     I::to_i8()
73
/// # }
74
/// #
75
/// assert_eq!(to_i8_typerat::<PInt<UInt<UTerm, B0>>>(), 0_i8);
76
/// ```
77
///
78
/// [typenum]: https://crates.io/crates/typenum
79
pub trait Integer: typenum::Integer + Sealed {}
80

            
81
impl Integer for Z0 {}
82

            
83
impl<U> Integer for PInt<U> where U: NonZeroUnsigned {}
84

            
85
impl<U> Integer for NInt<U> where U: NonZeroUnsigned {}
86

            
87
#[cfg(test)]
88
mod tests {
89
    use super::*;
90
    use typenum::consts::*;
91
    use typenum::UInt;
92

            
93
    #[test]
94
1
    fn test_f64() {
95
1
        assert_eq!(Z0::F64, 0_f64);
96
1
        assert_eq!(P1::F64, 1_f64);
97
1
        assert_eq!(N1::F64, -1_f64);
98
1
        assert_eq!(P16::F64, 16_f64);
99
1
        assert_eq!(N16::F64, -16_f64);
100
1
        assert_eq!(
101
1
            PInt::<UInt<U9223372036854775808, B1>>::F64,
102
1
            18446744073709551617_f64
103
1
        );
104
1
        assert_eq!(
105
1
            NInt::<UInt<U9223372036854775808, B1>>::F64,
106
1
            -18446744073709551617_f64
107
1
        );
108
1
    }
109
}