1//! A format item with borrowed data.
23#[cfg(feature = "alloc")]
4use alloc::string::String;
5#[cfg(feature = "alloc")]
6use core::fmt;
78use crate::error;
9use crate::format_description::Component;
1011/// A complete description of how to format and parse a type.
12#[non_exhaustive]
13#[cfg_attr(not(feature = "alloc"), derive(Debug))]
14#[derive(Clone, PartialEq, Eq)]
15pub enum BorrowedFormatItem<'a> {
16/// Bytes that are formatted as-is.
17 ///
18 /// **Note**: These bytes **should** be UTF-8, but are not required to be. The value is passed
19 /// through `String::from_utf8_lossy` when necessary.
20Literal(&'a [u8]),
21/// A minimal representation of a single non-literal item.
22Component(Component),
23/// A series of literals or components that collectively form a partial or complete
24 /// description.
25Compound(&'a [Self]),
26/// A `FormatItem` that may or may not be present when parsing. If parsing fails, there
27 /// will be no effect on the resulting `struct`.
28 ///
29 /// This variant has no effect on formatting, as the value is guaranteed to be present.
30Optional(&'a Self),
31/// A series of `FormatItem`s where, when parsing, the first successful parse is used. When
32 /// formatting, the first element of the slice is used. An empty slice is a no-op when
33 /// formatting or parsing.
34First(&'a [Self]),
35}
3637#[cfg(feature = "alloc")]
38impl fmt::Debug for BorrowedFormatItem<'_> {
39fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
40match self {
41Self::Literal(literal) => f.write_str(&String::from_utf8_lossy(literal)),
42Self::Component(component) => component.fmt(f),
43Self::Compound(compound) => compound.fmt(f),
44Self::Optional(item) => f.debug_tuple("Optional").field(item).finish(),
45Self::First(items) => f.debug_tuple("First").field(items).finish(),
46 }
47 }
48}
4950impl From<Component> for BorrowedFormatItem<'_> {
51fn from(component: Component) -> Self {
52Self::Component(component)
53 }
54}
5556impl TryFrom<BorrowedFormatItem<'_>> for Component {
57type Error = error::DifferentVariant;
5859fn try_from(value: BorrowedFormatItem<'_>) -> Result<Self, Self::Error> {
60match value {
61 BorrowedFormatItem::Component(component) => Ok(component),
62_ => Err(error::DifferentVariant),
63 }
64 }
65}
6667impl<'a> From<&'a [BorrowedFormatItem<'_>]> for BorrowedFormatItem<'a> {
68fn from(items: &'a [BorrowedFormatItem<'_>]) -> Self {
69Self::Compound(items)
70 }
71}
7273impl<'a> TryFrom<BorrowedFormatItem<'a>> for &[BorrowedFormatItem<'a>] {
74type Error = error::DifferentVariant;
7576fn try_from(value: BorrowedFormatItem<'a>) -> Result<Self, Self::Error> {
77match value {
78 BorrowedFormatItem::Compound(items) => Ok(items),
79_ => Err(error::DifferentVariant),
80 }
81 }
82}
8384impl PartialEq<Component> for BorrowedFormatItem<'_> {
85fn eq(&self, rhs: &Component) -> bool {
86matches!(self, Self::Component(component) if component == rhs)
87 }
88}
8990impl PartialEq<BorrowedFormatItem<'_>> for Component {
91fn eq(&self, rhs: &BorrowedFormatItem<'_>) -> bool {
92 rhs == self
93}
94}
9596impl PartialEq<&[Self]> for BorrowedFormatItem<'_> {
97fn eq(&self, rhs: &&[Self]) -> bool {
98matches!(self, Self::Compound(compound) if compound == rhs)
99 }
100}
101102impl PartialEq<BorrowedFormatItem<'_>> for &[BorrowedFormatItem<'_>] {
103fn eq(&self, rhs: &BorrowedFormatItem<'_>) -> bool {
104 rhs == self
105}
106}