icu_locid/extensions/private/mod.rs
1// This file is part of ICU4X. For terms of use, please see the file
2// called LICENSE at the top level of the ICU4X source tree
3// (online at: https://github.com/unicode-org/icu4x/blob/main/LICENSE ).
4
5//! Private Use Extensions is a list of extensions intended for
6//! private use.
7//!
8//! Those extensions are treated as a pass-through, and no Unicode related
9//! behavior depends on them.
10//!
11//! The main struct for this extension is [`Private`] which is a list of [`Subtag`]s.
12//!
13//! # Examples
14//!
15//! ```
16//! use icu::locid::extensions::private::subtag;
17//! use icu::locid::{locale, Locale};
18//!
19//! let mut loc: Locale = "en-US-x-foo-faa".parse().expect("Parsing failed.");
20//!
21//! assert!(loc.extensions.private.contains(&subtag!("foo")));
22//! assert_eq!(loc.extensions.private.iter().next(), Some(&subtag!("foo")));
23//!
24//! loc.extensions.private.clear();
25//!
26//! assert!(loc.extensions.private.is_empty());
27//! assert_eq!(loc, locale!("en-US"));
28//! ```
29
30mod other;
31
32use alloc::vec::Vec;
33use core::ops::Deref;
34
35#[doc(inline)]
36pub use other::{subtag, Subtag};
37
38use crate::parser::ParserError;
39use crate::parser::SubtagIterator;
40use crate::shortvec::ShortBoxSlice;
41
42/// A list of [`Private Use Extensions`] as defined in [`Unicode Locale
43/// Identifier`] specification.
44///
45/// Those extensions are treated as a pass-through, and no Unicode related
46/// behavior depends on them.
47///
48/// # Examples
49///
50/// ```
51/// use icu::locid::extensions::private::{Private, Subtag};
52///
53/// let subtag1: Subtag = "foo".parse().expect("Failed to parse a Subtag.");
54/// let subtag2: Subtag = "bar".parse().expect("Failed to parse a Subtag.");
55///
56/// let private = Private::from_vec_unchecked(vec![subtag1, subtag2]);
57/// assert_eq!(&private.to_string(), "x-foo-bar");
58/// ```
59///
60/// [`Private Use Extensions`]: https://unicode.org/reports/tr35/#pu_extensions
61/// [`Unicode Locale Identifier`]: https://unicode.org/reports/tr35/#Unicode_locale_identifier
62#[derive(Clone, PartialEq, Eq, Debug, Default, Hash, PartialOrd, Ord)]
63pub struct Private(ShortBoxSlice<Subtag>);
64
65impl Private {
66 /// Returns a new empty list of private-use extensions. Same as [`default()`](Default::default()), but is `const`.
67 ///
68 /// # Examples
69 ///
70 /// ```
71 /// use icu::locid::extensions::private::Private;
72 ///
73 /// assert_eq!(Private::new(), Private::default());
74 /// ```
75 #[inline]
76 pub const fn new() -> Self {
77 Self(ShortBoxSlice::new())
78 }
79
80 /// A constructor which takes a pre-sorted list of [`Subtag`].
81 ///
82 /// # Examples
83 ///
84 /// ```
85 /// use icu::locid::extensions::private::{Private, Subtag};
86 ///
87 /// let subtag1: Subtag = "foo".parse().expect("Failed to parse a Subtag.");
88 /// let subtag2: Subtag = "bar".parse().expect("Failed to parse a Subtag.");
89 ///
90 /// let private = Private::from_vec_unchecked(vec![subtag1, subtag2]);
91 /// assert_eq!(&private.to_string(), "x-foo-bar");
92 /// ```
93 pub fn from_vec_unchecked(input: Vec<Subtag>) -> Self {
94 Self(input.into())
95 }
96
97 /// A constructor which takes a single [`Subtag`].
98 ///
99 /// # Examples
100 ///
101 /// ```
102 /// use icu::locid::extensions::private::{Private, Subtag};
103 ///
104 /// let subtag: Subtag = "foo".parse().expect("Failed to parse a Subtag.");
105 ///
106 /// let private = Private::new_single(subtag);
107 /// assert_eq!(&private.to_string(), "x-foo");
108 /// ```
109 pub const fn new_single(input: Subtag) -> Self {
110 Self(ShortBoxSlice::new_single(input))
111 }
112
113 /// Empties the [`Private`] list.
114 ///
115 /// # Examples
116 ///
117 /// ```
118 /// use icu::locid::extensions::private::{Private, Subtag};
119 ///
120 /// let subtag1: Subtag = "foo".parse().expect("Failed to parse a Subtag.");
121 /// let subtag2: Subtag = "bar".parse().expect("Failed to parse a Subtag.");
122 /// let mut private = Private::from_vec_unchecked(vec![subtag1, subtag2]);
123 ///
124 /// assert_eq!(&private.to_string(), "x-foo-bar");
125 ///
126 /// private.clear();
127 ///
128 /// assert_eq!(private, Private::new());
129 /// ```
130 pub fn clear(&mut self) {
131 self.0.clear();
132 }
133
134 pub(crate) fn try_from_iter(iter: &mut SubtagIterator) -> Result<Self, ParserError> {
135 let keys = iter
136 .map(Subtag::try_from_bytes)
137 .collect::<Result<ShortBoxSlice<_>, _>>()?;
138
139 Ok(Self(keys))
140 }
141
142 pub(crate) fn for_each_subtag_str<E, F>(&self, f: &mut F) -> Result<(), E>
143 where
144 F: FnMut(&str) -> Result<(), E>,
145 {
146 if self.is_empty() {
147 return Ok(());
148 }
149 f("x")?;
150 self.deref().iter().map(|t| t.as_str()).try_for_each(f)
151 }
152}
153
154writeable::impl_display_with_writeable!(Private);
155
156impl writeable::Writeable for Private {
157 fn write_to<W: core::fmt::Write + ?Sized>(&self, sink: &mut W) -> core::fmt::Result {
158 if self.is_empty() {
159 return Ok(());
160 }
161 sink.write_str("x")?;
162 for key in self.iter() {
163 sink.write_char('-')?;
164 writeable::Writeable::write_to(key, sink)?;
165 }
166 Ok(())
167 }
168
169 fn writeable_length_hint(&self) -> writeable::LengthHint {
170 if self.is_empty() {
171 return writeable::LengthHint::exact(0);
172 }
173 let mut result = writeable::LengthHint::exact(1);
174 for key in self.iter() {
175 result += writeable::Writeable::writeable_length_hint(key) + 1;
176 }
177 result
178 }
179}
180
181impl Deref for Private {
182 type Target = [Subtag];
183
184 fn deref(&self) -> &Self::Target {
185 self.0.deref()
186 }
187}