1
use crate::{Denominator, NotOne, Numerator, Q};
2
use core::fmt;
3
use core::ops::{Div, Rem};
4
use typenum::{Bit, Mod, NInt, PInt, Quot, UInt, UTerm, Unsigned, B0, B1, Z0};
5

            
6
macro_rules! impl_fmt {
7
    ($fmt:ident, $Fmt:path, $fmt_type:literal, $prefix:literal, $Ux:ty, $WIDTH:literal) => {
8
        mod $fmt {
9
            use super::*;
10

            
11
            pub trait PrivateFmtHelperUInt {
12
                type IsZero: Bit;
13

            
14
                fn fmt(f: &mut fmt::Formatter) -> fmt::Result;
15
            }
16

            
17
            impl PrivateFmtHelperUInt for UTerm {
18
                type IsZero = B1;
19

            
20
65
                fn fmt(f: &mut fmt::Formatter) -> fmt::Result {
21
65
                    let _ = f;
22
65
                    Ok(())
23
65
                }
24
            }
25

            
26
            impl<U, B> PrivateFmtHelperUInt for UInt<U, B>
27
            where
28
                Self: Div<$Ux> + Rem<$Ux>,
29
                Quot<Self, $Ux>: PrivateFmtHelperUInt,
30
                Mod<Self, $Ux>: Unsigned,
31
            {
32
                type IsZero = B0;
33

            
34
70
                fn fmt(f: &mut fmt::Formatter) -> fmt::Result {
35
70
                    Quot::<Self, $Ux>::fmt(f)?;
36

            
37
70
                    if <Quot<Self, $Ux> as PrivateFmtHelperUInt>::IsZero::BOOL {
38
65
                        write!(f, concat!("{:", $fmt_type, "}"), Mod::<Self, $Ux>::U64)
39
                    } else {
40
5
                        write!(
41
5
                            f,
42
5
                            concat!("{:0", $WIDTH, $fmt_type, "}"),
43
5
                            Mod::<Self, $Ux>::U64
44
5
                        )
45
                    }
46
70
                }
47
            }
48

            
49
            pub trait PrivateFmtHelperInt {
50
                type IsNegative: Bit;
51

            
52
                fn fmt_abs(f: &mut fmt::Formatter) -> fmt::Result;
53
            }
54

            
55
            impl PrivateFmtHelperInt for Z0 {
56
                type IsNegative = B0;
57

            
58
5
                fn fmt_abs(f: &mut fmt::Formatter) -> fmt::Result {
59
5
                    f.write_str("0")
60
5
                }
61
            }
62

            
63
            impl<U> PrivateFmtHelperInt for PInt<U>
64
            where
65
                U: Unsigned + typenum::NonZero + PrivateFmtHelperUInt,
66
            {
67
                type IsNegative = B0;
68

            
69
50
                fn fmt_abs(f: &mut fmt::Formatter) -> fmt::Result {
70
50
                    U::fmt(f)
71
50
                }
72
            }
73

            
74
            impl<U> PrivateFmtHelperInt for NInt<U>
75
            where
76
                U: Unsigned + typenum::NonZero + PrivateFmtHelperUInt,
77
            {
78
                type IsNegative = B1;
79

            
80
15
                fn fmt_abs(f: &mut fmt::Formatter) -> fmt::Result {
81
15
                    U::fmt(f)
82
15
                }
83
            }
84

            
85
            pub trait PrivateFmtHelper {
86
                fn fmt(f: &mut fmt::Formatter) -> fmt::Result;
87
            }
88

            
89
            impl<N> PrivateFmtHelper for Q<N>
90
            where
91
                N: Numerator + PrivateFmtHelperInt,
92
            {
93
30
                fn fmt(f: &mut fmt::Formatter) -> fmt::Result {
94
30
                    if N::IsNegative::BOOL {
95
10
                        f.write_str("-")?;
96
20
                    } else if f.sign_plus() {
97
14
                        f.write_str("+")?;
98
6
                    }
99

            
100
30
                    if f.alternate() {
101
10
                        f.write_str($prefix)?;
102
20
                    }
103

            
104
30
                    N::fmt_abs(f)
105
30
                }
106
            }
107

            
108
            impl<N, D> PrivateFmtHelper for Q<N, D>
109
            where
110
                N: Numerator<D> + PrivateFmtHelperInt,
111
                D: Denominator + NotOne + PrivateFmtHelperInt,
112
            {
113
20
                fn fmt(f: &mut fmt::Formatter) -> fmt::Result {
114
20
                    if N::IsNegative::BOOL {
115
5
                        f.write_str("-")?;
116
15
                    } else if f.sign_plus() {
117
5
                        f.write_str("+")?;
118
10
                    }
119

            
120
20
                    if f.alternate() {
121
5
                        f.write_str($prefix)?;
122
15
                    }
123

            
124
20
                    N::fmt_abs(f)?;
125

            
126
20
                    f.write_str("/")?;
127

            
128
20
                    if f.alternate() {
129
5
                        f.write_str($prefix)?;
130
15
                    }
131

            
132
20
                    D::fmt_abs(f)
133
20
                }
134
            }
135

            
136
            impl<N, D> $Fmt for Q<N, D>
137
            where
138
                Self: PrivateFmtHelper,
139
                N: Numerator<D>,
140
                D: Denominator,
141
            {
142
50
                fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
143
50
                    <Self as PrivateFmtHelper>::fmt(f)
144
50
                }
145
            }
146
        }
147
    };
148
}
149

            
150
impl_fmt!(
151
    display,
152
    fmt::Display,
153
    "",
154
    "",
155
    // The largest power of 10 smaller than u64::MAX
156
    typenum::consts::U10000000000000000000,
157
    19
158
);
159

            
160
impl_fmt!(
161
    binary,
162
    fmt::Binary,
163
    "b",
164
    "0b",
165
    // 2 ** 64
166
    UInt<typenum::consts::U9223372036854775808, B0>,
167
    64
168
);
169

            
170
impl_fmt!(
171
    octal,
172
    fmt::Octal,
173
    "o",
174
    "0o",
175
    // 2 ** 63
176
    typenum::consts::U9223372036854775808,
177
    21
178
);
179

            
180
impl_fmt!(
181
    lower_hex,
182
    fmt::LowerHex,
183
    "x",
184
    "0x",
185
    // 2 ** 64
186
    UInt<typenum::consts::U9223372036854775808, B0>,
187
    16
188
);
189

            
190
impl_fmt!(
191
    upper_hex,
192
    fmt::UpperHex,
193
    "X",
194
    "0x",
195
    // 2 ** 64
196
    UInt<typenum::consts::U9223372036854775808, B0>,
197
    16
198
);
199

            
200
#[cfg(test)]
201
mod tests {
202
    use crate::*;
203
    use char_buf::CharBuf;
204
    use core::fmt::Write;
205
    use typenum::{PInt, UInt};
206

            
207
    macro_rules! format {
208
        ($($arg:tt)*) => {
209
            {
210
                let mut w = CharBuf::<1024>::new();
211
                write!(w, $($arg)*).unwrap();
212
                w
213
            }
214
        };
215
    }
216

            
217
    #[test]
218
1
    fn test_display() {
219
1
        assert_eq!(format!("{:+#}", Q::<Z0>::new()), "+0");
220
1
        assert_eq!(format!("{:}", Q::<P1>::new()), "1");
221
1
        assert_eq!(format!("{:}", Q::<N1>::new()), "-1");
222
1
        assert_eq!(format!("{:+}", Q::<P2>::new()), "+2");
223
1
        assert_eq!(format!("{:}", Q::<N2>::new()), "-2");
224
1
        assert_eq!(format!("{:}", Q::<P1, P2>::new()), "1/2");
225
1
        assert_eq!(format!("{:+}", Q::<P1, P2>::new()), "+1/2");
226
1
        assert_eq!(format!("{:}", Q::<N1, P2>::new()), "-1/2");
227
1
        assert_eq!(format!("{:#}", Q::<P1, P2>::new()), "1/2");
228
1
        assert_eq!(
229
1
            format!("{:#}", Q::<PInt<UInt<U10000000000000000000, B1>>>::new()),
230
1
            "20000000000000000001"
231
1
        );
232
1
    }
233

            
234
    #[test]
235
1
    fn test_binary() {
236
1
        assert_eq!(format!("{:+#b}", Q::<Z0>::new()), "+0b0");
237
1
        assert_eq!(format!("{:b}", Q::<P1>::new()), "1");
238
1
        assert_eq!(format!("{:b}", Q::<N1>::new()), "-1");
239
1
        assert_eq!(format!("{:+b}", Q::<P2>::new()), "+10");
240
1
        assert_eq!(format!("{:b}", Q::<N2>::new()), "-10");
241
1
        assert_eq!(format!("{:b}", Q::<P1, P2>::new()), "1/10");
242
1
        assert_eq!(format!("{:+b}", Q::<P1, P2>::new()), "+1/10");
243
1
        assert_eq!(format!("{:b}", Q::<N1, P2>::new()), "-1/10");
244
1
        assert_eq!(format!("{:#b}", Q::<P1, P2>::new()), "0b1/0b10");
245
1
        assert_eq!(
246
1
            format!("{:+#b}", Q::<PInt<UInt<U9223372036854775808, B1>>>::new()),
247
1
            "+0b10000000000000000000000000000000000000000000000000000000000000001"
248
1
        );
249
1
    }
250

            
251
    #[test]
252
1
    fn test_octal() {
253
1
        assert_eq!(format!("{:+#o}", Q::<Z0>::new()), "+0o0");
254
1
        assert_eq!(format!("{:o}", Q::<P1>::new()), "1");
255
1
        assert_eq!(format!("{:o}", Q::<N1>::new()), "-1");
256
1
        assert_eq!(format!("{:+o}", Q::<P2>::new()), "+2");
257
1
        assert_eq!(format!("{:o}", Q::<N2>::new()), "-2");
258
1
        assert_eq!(format!("{:o}", Q::<P1, P2>::new()), "1/2");
259
1
        assert_eq!(format!("{:+o}", Q::<P1, P2>::new()), "+1/2");
260
1
        assert_eq!(format!("{:o}", Q::<N1, P2>::new()), "-1/2");
261
1
        assert_eq!(format!("{:#o}", Q::<P1, P2>::new()), "0o1/0o2");
262
1
        assert_eq!(
263
1
            format!("{:+#o}", Q::<PInt<UInt<U9223372036854775808, B1>>>::new()),
264
1
            "+0o2000000000000000000001"
265
1
        );
266
1
    }
267

            
268
    #[test]
269
1
    fn test_lower_hex() {
270
1
        assert_eq!(format!("{:+#x}", Q::<Z0>::new()), "+0x0");
271
1
        assert_eq!(format!("{:x}", Q::<P1>::new()), "1");
272
1
        assert_eq!(format!("{:x}", Q::<N1>::new()), "-1");
273
1
        assert_eq!(format!("{:+x}", Q::<P2>::new()), "+2");
274
1
        assert_eq!(format!("{:x}", Q::<N2>::new()), "-2");
275
1
        assert_eq!(format!("{:x}", Q::<P1, P10>::new()), "1/a");
276
1
        assert_eq!(format!("{:+x}", Q::<P1, P10>::new()), "+1/a");
277
1
        assert_eq!(format!("{:x}", Q::<N1, P10>::new()), "-1/a");
278
1
        assert_eq!(format!("{:#x}", Q::<P1, P10>::new()), "0x1/0xa");
279
1
        assert_eq!(
280
1
            format!("{:+#x}", Q::<PInt<UInt<U9223372036854775808, B1>>>::new()),
281
1
            "+0x10000000000000001"
282
1
        );
283
1
    }
284

            
285
    #[test]
286
1
    fn test_upper_hex() {
287
1
        assert_eq!(format!("{:+#X}", Q::<Z0>::new()), "+0x0");
288
1
        assert_eq!(format!("{:X}", Q::<P1>::new()), "1");
289
1
        assert_eq!(format!("{:X}", Q::<N1>::new()), "-1");
290
1
        assert_eq!(format!("{:+X}", Q::<P2>::new()), "+2");
291
1
        assert_eq!(format!("{:X}", Q::<N2>::new()), "-2");
292
1
        assert_eq!(format!("{:X}", Q::<P1, P10>::new()), "1/A");
293
1
        assert_eq!(format!("{:+X}", Q::<P1, P10>::new()), "+1/A");
294
1
        assert_eq!(format!("{:X}", Q::<N1, P10>::new()), "-1/A");
295
1
        assert_eq!(format!("{:#X}", Q::<P1, P10>::new()), "0x1/0xA");
296
1
        assert_eq!(
297
1
            format!("{:+#X}", Q::<PInt<UInt<U9223372036854775808, B1>>>::new()),
298
1
            "+0x10000000000000001"
299
1
        );
300
1
    }
301
}