1
use typenum::{Bit, UInt, UTerm, Unsigned, B1};
2

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

            
7
impl Sealed for UInt<UTerm, B1> {
8
    const F64: f64 = 1_f64;
9
}
10

            
11
impl<U, B> Sealed for UInt<U, B>
12
where
13
    U: NonZeroUnsigned,
14
    B: Bit,
15
{
16
    const F64: f64 = <U as Sealed>::F64 * 2_f64 + B::U8 as f64;
17
}
18

            
19
/// Type-level non-zero unsigned integers.
20
///
21
/// # Examples
22
///
23
/// This trait is basically implemented for types that implement [typenum]'s `Unsigned` and `NonZero`.
24
///
25
/// ```
26
/// use typerat::*;
27
/// use typenum::{UInt, UTerm};
28
///
29
/// fn to_u8_typenum<U: typenum::Unsigned + typenum::NonZero>() -> u8 {
30
///     U::to_u8()
31
/// }
32
///
33
/// fn to_u8_typerat<U: NonZeroUnsigned>() -> u8 {
34
///     U::to_u8()
35
/// }
36
///
37
/// assert_eq!(to_u8_typenum::<U1>(), 1_u8);
38
/// assert_eq!(to_u8_typerat::<U1>(), 1_u8);
39
/// ```
40
///
41
/// Unlike [typenum], compile time unsigned integers with leading zeros are not allowed.
42
///
43
/// ```
44
/// # use typerat::*;
45
/// # use typenum::{UInt, UTerm};
46
/// #
47
/// # fn to_u8_typenum<U: typenum::Unsigned + typenum::NonZero>() -> u8 {
48
/// #     U::to_u8()
49
/// # }
50
/// #
51
/// assert_eq!(to_u8_typenum::<UInt<UTerm, B0>>(), 0_u8);
52
/// ```
53
///
54
/// ```compile_fail
55
/// # use typerat::*;
56
/// # use typenum::{UInt, UTerm};
57
/// #
58
/// # fn to_u8_typerat<U: NonZeroUnsigned>() -> u8 {
59
/// #     U::to_u8()
60
/// # }
61
/// #
62
/// assert_eq!(to_u8_typerat::<UInt<UTerm, B0>>(), 0_u8);
63
/// ```
64
///
65
/// [typenum]: https://crates.io/crates/typenum
66
pub trait NonZeroUnsigned: Unsigned + typenum::NonZero + Sealed {}
67

            
68
impl NonZeroUnsigned for UInt<UTerm, B1> {}
69

            
70
impl<U, B> NonZeroUnsigned for UInt<U, B>
71
where
72
    U: NonZeroUnsigned,
73
    B: Bit,
74
{
75
}
76

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

            
82
    #[test]
83
1
    fn test_f64() {
84
1
        assert_eq!(U1::F64, 1_f64);
85
1
        assert_eq!(U2::F64, 2_f64);
86
1
        assert_eq!(U16::F64, 16_f64);
87
1
        assert_eq!(
88
1
            UInt::<U9223372036854775808, B1>::F64,
89
1
            18446744073709551617_f64
90
1
        );
91
1
    }
92
}