time/format_description/modifier.rs
1//! Various modifiers for components.
2
3use core::num::NonZeroU16;
4
5// region: date modifiers
6/// Day of the month.
7#[non_exhaustive]
8#[derive(Debug, Clone, Copy, PartialEq, Eq)]
9pub struct Day {
10 /// The padding to obtain the minimum width.
11 pub padding: Padding,
12}
13
14/// The representation of a month.
15#[non_exhaustive]
16#[derive(Debug, Clone, Copy, PartialEq, Eq)]
17pub enum MonthRepr {
18 /// The number of the month (January is 1, December is 12).
19 Numerical,
20 /// The long form of the month name (e.g. "January").
21 Long,
22 /// The short form of the month name (e.g. "Jan").
23 Short,
24}
25
26/// Month of the year.
27#[non_exhaustive]
28#[derive(Debug, Clone, Copy, PartialEq, Eq)]
29pub struct Month {
30 /// The padding to obtain the minimum width.
31 pub padding: Padding,
32 /// What form of representation should be used?
33 pub repr: MonthRepr,
34 /// Is the value case sensitive when parsing?
35 pub case_sensitive: bool,
36}
37
38/// Ordinal day of the year.
39#[non_exhaustive]
40#[derive(Debug, Clone, Copy, PartialEq, Eq)]
41pub struct Ordinal {
42 /// The padding to obtain the minimum width.
43 pub padding: Padding,
44}
45
46/// The representation used for the day of the week.
47#[non_exhaustive]
48#[derive(Debug, Clone, Copy, PartialEq, Eq)]
49pub enum WeekdayRepr {
50 /// The short form of the weekday (e.g. "Mon").
51 Short,
52 /// The long form of the weekday (e.g. "Monday").
53 Long,
54 /// A numerical representation using Sunday as the first day of the week.
55 ///
56 /// Sunday is either 0 or 1, depending on the other modifier's value.
57 Sunday,
58 /// A numerical representation using Monday as the first day of the week.
59 ///
60 /// Monday is either 0 or 1, depending on the other modifier's value.
61 Monday,
62}
63
64/// Day of the week.
65#[non_exhaustive]
66#[derive(Debug, Clone, Copy, PartialEq, Eq)]
67pub struct Weekday {
68 /// What form of representation should be used?
69 pub repr: WeekdayRepr,
70 /// When using a numerical representation, should it be zero or one-indexed?
71 pub one_indexed: bool,
72 /// Is the value case sensitive when parsing?
73 pub case_sensitive: bool,
74}
75
76/// The representation used for the week number.
77#[non_exhaustive]
78#[derive(Debug, Clone, Copy, PartialEq, Eq)]
79pub enum WeekNumberRepr {
80 /// Week 1 is the week that contains January 4.
81 Iso,
82 /// Week 1 begins on the first Sunday of the calendar year.
83 Sunday,
84 /// Week 1 begins on the first Monday of the calendar year.
85 Monday,
86}
87
88/// Week within the year.
89#[non_exhaustive]
90#[derive(Debug, Clone, Copy, PartialEq, Eq)]
91pub struct WeekNumber {
92 /// The padding to obtain the minimum width.
93 pub padding: Padding,
94 /// What kind of representation should be used?
95 pub repr: WeekNumberRepr,
96}
97
98/// The representation used for a year value.
99#[non_exhaustive]
100#[derive(Debug, Clone, Copy, PartialEq, Eq)]
101pub enum YearRepr {
102 /// The full value of the year.
103 Full,
104 /// All digits except the last two. Includes the sign, if any.
105 Century,
106 /// Only the last two digits of the year.
107 LastTwo,
108}
109
110/// Year of the date.
111#[non_exhaustive]
112#[derive(Debug, Clone, Copy, PartialEq, Eq)]
113pub struct Year {
114 /// The padding to obtain the minimum width.
115 pub padding: Padding,
116 /// What kind of representation should be used?
117 pub repr: YearRepr,
118 /// Whether the value is based on the ISO week number or the Gregorian calendar.
119 pub iso_week_based: bool,
120 /// Whether the `+` sign is present when a positive year contains fewer than five digits.
121 pub sign_is_mandatory: bool,
122}
123// endregion date modifiers
124
125// region: time modifiers
126/// Hour of the day.
127#[non_exhaustive]
128#[derive(Debug, Clone, Copy, PartialEq, Eq)]
129pub struct Hour {
130 /// The padding to obtain the minimum width.
131 pub padding: Padding,
132 /// Is the hour displayed using a 12 or 24-hour clock?
133 pub is_12_hour_clock: bool,
134}
135
136/// Minute within the hour.
137#[non_exhaustive]
138#[derive(Debug, Clone, Copy, PartialEq, Eq)]
139pub struct Minute {
140 /// The padding to obtain the minimum width.
141 pub padding: Padding,
142}
143
144/// AM/PM part of the time.
145#[non_exhaustive]
146#[derive(Debug, Clone, Copy, PartialEq, Eq)]
147pub struct Period {
148 /// Is the period uppercase or lowercase?
149 pub is_uppercase: bool,
150 /// Is the value case sensitive when parsing?
151 ///
152 /// Note that when `false`, the `is_uppercase` field has no effect on parsing behavior.
153 pub case_sensitive: bool,
154}
155
156/// Second within the minute.
157#[non_exhaustive]
158#[derive(Debug, Clone, Copy, PartialEq, Eq)]
159pub struct Second {
160 /// The padding to obtain the minimum width.
161 pub padding: Padding,
162}
163
164/// The number of digits present in a subsecond representation.
165#[non_exhaustive]
166#[derive(Debug, Clone, Copy, PartialEq, Eq)]
167pub enum SubsecondDigits {
168 /// Exactly one digit.
169 One,
170 /// Exactly two digits.
171 Two,
172 /// Exactly three digits.
173 Three,
174 /// Exactly four digits.
175 Four,
176 /// Exactly five digits.
177 Five,
178 /// Exactly six digits.
179 Six,
180 /// Exactly seven digits.
181 Seven,
182 /// Exactly eight digits.
183 Eight,
184 /// Exactly nine digits.
185 Nine,
186 /// Any number of digits (up to nine) that is at least one. When formatting, the minimum digits
187 /// necessary will be used.
188 OneOrMore,
189}
190
191/// Subsecond within the second.
192#[non_exhaustive]
193#[derive(Debug, Clone, Copy, PartialEq, Eq)]
194pub struct Subsecond {
195 /// How many digits are present in the component?
196 pub digits: SubsecondDigits,
197}
198// endregion time modifiers
199
200// region: offset modifiers
201/// Hour of the UTC offset.
202#[non_exhaustive]
203#[derive(Debug, Clone, Copy, PartialEq, Eq)]
204pub struct OffsetHour {
205 /// Whether the `+` sign is present on positive values.
206 pub sign_is_mandatory: bool,
207 /// The padding to obtain the minimum width.
208 pub padding: Padding,
209}
210
211/// Minute within the hour of the UTC offset.
212#[non_exhaustive]
213#[derive(Debug, Clone, Copy, PartialEq, Eq)]
214pub struct OffsetMinute {
215 /// The padding to obtain the minimum width.
216 pub padding: Padding,
217}
218
219/// Second within the minute of the UTC offset.
220#[non_exhaustive]
221#[derive(Debug, Clone, Copy, PartialEq, Eq)]
222pub struct OffsetSecond {
223 /// The padding to obtain the minimum width.
224 pub padding: Padding,
225}
226// endregion offset modifiers
227
228/// Type of padding to ensure a minimum width.
229#[non_exhaustive]
230#[derive(Debug, Clone, Copy, PartialEq, Eq)]
231pub enum Padding {
232 /// A space character (` `) should be used as padding.
233 Space,
234 /// A zero character (`0`) should be used as padding.
235 Zero,
236 /// There is no padding. This can result in a width below the otherwise minimum number of
237 /// characters.
238 None,
239}
240
241/// Ignore some number of bytes.
242///
243/// This has no effect when formatting.
244#[non_exhaustive]
245#[derive(Debug, Clone, Copy, PartialEq, Eq)]
246pub struct Ignore {
247 /// The number of bytes to ignore.
248 pub count: NonZeroU16,
249}
250
251// Needed as `Default` is deliberately not implemented for `Ignore`. The number of bytes to ignore
252// must be explicitly provided.
253impl Ignore {
254 /// Create an instance of `Ignore` with the provided number of bytes to ignore.
255 pub const fn count(count: NonZeroU16) -> Self {
256 Self { count }
257 }
258}
259
260/// The precision of a Unix timestamp.
261#[non_exhaustive]
262#[derive(Debug, Clone, Copy, PartialEq, Eq)]
263pub enum UnixTimestampPrecision {
264 /// Seconds since the Unix epoch.
265 Second,
266 /// Milliseconds since the Unix epoch.
267 Millisecond,
268 /// Microseconds since the Unix epoch.
269 Microsecond,
270 /// Nanoseconds since the Unix epoch.
271 Nanosecond,
272}
273
274/// A Unix timestamp.
275#[non_exhaustive]
276#[derive(Debug, Clone, Copy, PartialEq, Eq)]
277pub struct UnixTimestamp {
278 /// The precision of the timestamp.
279 pub precision: UnixTimestampPrecision,
280 /// Whether the `+` sign must be present for a non-negative timestamp.
281 pub sign_is_mandatory: bool,
282}
283
284/// The end of input.
285///
286/// There is currently not customization for this modifier.
287#[non_exhaustive]
288#[derive(Debug, Clone, Copy, PartialEq, Eq)]
289pub struct End;
290
291/// Generate the provided code if and only if `pub` is present.
292macro_rules! if_pub {
293 (pub $(#[$attr:meta])*; $($x:tt)*) => {
294 $(#[$attr])*
295 ///
296 /// This function exists since [`Default::default()`] cannot be used in a `const` context.
297 /// It may be removed once that becomes possible. As the [`Default`] trait is in the
298 /// prelude, removing this function in the future will not cause any resolution failures for
299 /// the overwhelming majority of users; only users who use `#![no_implicit_prelude]` will be
300 /// affected. As such it will not be considered a breaking change.
301 $($x)*
302 };
303 ($($_:tt)*) => {};
304}
305
306/// Implement `Default` for the given type. This also generates an inherent implementation of a
307/// `default` method that is `const fn`, permitting the default value to be used in const contexts.
308// Every modifier should use this macro rather than a derived `Default`.
309macro_rules! impl_const_default {
310 ($($(#[$doc:meta])* $(@$pub:ident)? $type:ty => $default:expr;)*) => {$(
311 impl $type {
312 if_pub! {
313 $($pub)?
314 $(#[$doc])*;
315 pub const fn default() -> Self {
316 $default
317 }
318 }
319 }
320
321 $(#[$doc])*
322 impl Default for $type {
323 fn default() -> Self {
324 $default
325 }
326 }
327 )*};
328}
329
330impl_const_default! {
331 /// Creates a modifier that indicates the value is [padded with zeroes](Padding::Zero).
332 @pub Day => Self { padding: Padding::Zero };
333 /// Creates a modifier that indicates the value uses the
334 /// [`Numerical`](Self::Numerical) representation.
335 MonthRepr => Self::Numerical;
336 /// Creates an instance of this type that indicates the value uses the
337 /// [`Numerical`](MonthRepr::Numerical) representation, is [padded with zeroes](Padding::Zero),
338 /// and is case-sensitive when parsing.
339 @pub Month => Self {
340 padding: Padding::Zero,
341 repr: MonthRepr::Numerical,
342 case_sensitive: true,
343 };
344 /// Creates a modifier that indicates the value is [padded with zeroes](Padding::Zero).
345 @pub Ordinal => Self { padding: Padding::Zero };
346 /// Creates a modifier that indicates the value uses the [`Long`](Self::Long) representation.
347 WeekdayRepr => Self::Long;
348 /// Creates a modifier that indicates the value uses the [`Long`](WeekdayRepr::Long)
349 /// representation and is case-sensitive when parsing. If the representation is changed to a
350 /// numerical one, the instance defaults to one-based indexing.
351 @pub Weekday => Self {
352 repr: WeekdayRepr::Long,
353 one_indexed: true,
354 case_sensitive: true,
355 };
356 /// Creates a modifier that indicates that the value uses the [`Iso`](Self::Iso) representation.
357 WeekNumberRepr => Self::Iso;
358 /// Creates a modifier that indicates that the value is [padded with zeroes](Padding::Zero)
359 /// and uses the [`Iso`](WeekNumberRepr::Iso) representation.
360 @pub WeekNumber => Self {
361 padding: Padding::Zero,
362 repr: WeekNumberRepr::Iso,
363 };
364 /// Creates a modifier that indicates the value uses the [`Full`](Self::Full) representation.
365 YearRepr => Self::Full;
366 /// Creates a modifier that indicates the value uses the [`Full`](YearRepr::Full)
367 /// representation, is [padded with zeroes](Padding::Zero), uses the Gregorian calendar as its
368 /// base, and only includes the year's sign if necessary.
369 @pub Year => Self {
370 padding: Padding::Zero,
371 repr: YearRepr::Full,
372 iso_week_based: false,
373 sign_is_mandatory: false,
374 };
375 /// Creates a modifier that indicates the value is [padded with zeroes](Padding::Zero) and
376 /// has the 24-hour representation.
377 @pub Hour => Self {
378 padding: Padding::Zero,
379 is_12_hour_clock: false,
380 };
381 /// Creates a modifier that indicates the value is [padded with zeroes](Padding::Zero).
382 @pub Minute => Self { padding: Padding::Zero };
383 /// Creates a modifier that indicates the value uses the upper-case representation and is
384 /// case-sensitive when parsing.
385 @pub Period => Self {
386 is_uppercase: true,
387 case_sensitive: true,
388 };
389 /// Creates a modifier that indicates the value is [padded with zeroes](Padding::Zero).
390 @pub Second => Self { padding: Padding::Zero };
391 /// Creates a modifier that indicates the stringified value contains [one or more
392 /// digits](Self::OneOrMore).
393 SubsecondDigits => Self::OneOrMore;
394 /// Creates a modifier that indicates the stringified value contains [one or more
395 /// digits](SubsecondDigits::OneOrMore).
396 @pub Subsecond => Self { digits: SubsecondDigits::OneOrMore };
397 /// Creates a modifier that indicates the value only uses a sign for negative values and is
398 /// [padded with zeroes](Padding::Zero).
399 @pub OffsetHour => Self {
400 sign_is_mandatory: false,
401 padding: Padding::Zero,
402 };
403 /// Creates a modifier that indicates the value is [padded with zeroes](Padding::Zero).
404 @pub OffsetMinute => Self { padding: Padding::Zero };
405 /// Creates a modifier that indicates the value is [padded with zeroes](Padding::Zero).
406 @pub OffsetSecond => Self { padding: Padding::Zero };
407 /// Creates a modifier that indicates the value is [padded with zeroes](Self::Zero).
408 Padding => Self::Zero;
409 /// Creates a modifier that indicates the value represents the [number of seconds](Self::Second)
410 /// since the Unix epoch.
411 UnixTimestampPrecision => Self::Second;
412 /// Creates a modifier that indicates the value represents the [number of
413 /// seconds](UnixTimestampPrecision::Second) since the Unix epoch. The sign is not mandatory.
414 @pub UnixTimestamp => Self {
415 precision: UnixTimestampPrecision::Second,
416 sign_is_mandatory: false,
417 };
418 /// Creates a modifier used to represent the end of input.
419 @pub End => End;
420}