icu_locid_transform/directionality.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
5use crate::provider::*;
6use crate::{LocaleExpander, LocaleTransformError};
7use icu_locid::subtags::Script;
8use icu_locid::LanguageIdentifier;
9use icu_provider::prelude::*;
10
11/// Represents the direction of a script.
12///
13/// [`LocaleDirectionality`] can be used to get this information.
14#[derive(Debug, PartialEq, Eq, Clone, Copy)]
15#[non_exhaustive]
16pub enum Direction {
17 /// The script is left-to-right.
18 LeftToRight,
19 /// The script is right-to-left.
20 RightToLeft,
21}
22
23/// Provides methods to determine the direction of a locale.
24///
25/// # Examples
26///
27/// ```
28/// use icu::locid::locale;
29/// use icu::locid_transform::{Direction, LocaleDirectionality};
30///
31/// let ld = LocaleDirectionality::new();
32///
33/// assert_eq!(ld.get(&locale!("en")), Some(Direction::LeftToRight));
34/// ```
35#[derive(Debug)]
36pub struct LocaleDirectionality {
37 script_direction: DataPayload<ScriptDirectionV1Marker>,
38 expander: LocaleExpander,
39}
40
41#[cfg(feature = "compiled_data")]
42impl Default for LocaleDirectionality {
43 fn default() -> Self {
44 Self::new()
45 }
46}
47
48impl LocaleDirectionality {
49 /// Creates a [`LocaleDirectionality`] from compiled data.
50 ///
51 /// This includes limited likely subtags data, see [`LocaleExpander::new()`].
52 #[cfg(feature = "compiled_data")]
53 pub const fn new() -> Self {
54 Self::new_with_expander(LocaleExpander::new())
55 }
56
57 // Note: This is a custom impl because the bounds on `try_new_unstable` don't suffice
58 #[doc = icu_provider::gen_any_buffer_unstable_docs!(ANY, Self::new)]
59 pub fn try_new_with_any_provider(
60 provider: &(impl AnyProvider + ?Sized),
61 ) -> Result<LocaleDirectionality, LocaleTransformError> {
62 let expander = LocaleExpander::try_new_with_any_provider(provider)?;
63 Self::try_new_with_expander_unstable(&provider.as_downcasting(), expander)
64 }
65
66 // Note: This is a custom impl because the bounds on `try_new_unstable` don't suffice
67 #[doc = icu_provider::gen_any_buffer_unstable_docs!(BUFFER, Self::new)]
68 #[cfg(feature = "serde")]
69 pub fn try_new_with_buffer_provider(
70 provider: &(impl BufferProvider + ?Sized),
71 ) -> Result<LocaleDirectionality, LocaleTransformError> {
72 let expander = LocaleExpander::try_new_with_buffer_provider(provider)?;
73 Self::try_new_with_expander_unstable(&provider.as_deserializing(), expander)
74 }
75
76 #[doc = icu_provider::gen_any_buffer_unstable_docs!(UNSTABLE, Self::new)]
77 pub fn try_new_unstable<P>(provider: &P) -> Result<LocaleDirectionality, LocaleTransformError>
78 where
79 P: DataProvider<ScriptDirectionV1Marker>
80 + DataProvider<LikelySubtagsForLanguageV1Marker>
81 + DataProvider<LikelySubtagsForScriptRegionV1Marker>
82 + ?Sized,
83 {
84 let expander = LocaleExpander::try_new_unstable(provider)?;
85 Self::try_new_with_expander_unstable(provider, expander)
86 }
87
88 /// Creates a [`LocaleDirectionality`] with a custom [`LocaleExpander`] and compiled data.
89 ///
90 /// This allows using [`LocaleExpander::new_extended()`] with data for all locales.
91 ///
92 /// # Examples
93 ///
94 /// ```
95 /// use icu::locid::locale;
96 /// use icu::locid_transform::{
97 /// Direction, LocaleDirectionality, LocaleExpander,
98 /// };
99 ///
100 /// let ld_default = LocaleDirectionality::new();
101 ///
102 /// assert_eq!(ld_default.get(&locale!("jbn")), None);
103 ///
104 /// let expander = LocaleExpander::new_extended();
105 /// let ld_extended = LocaleDirectionality::new_with_expander(expander);
106 ///
107 /// assert_eq!(
108 /// ld_extended.get(&locale!("jbn")),
109 /// Some(Direction::RightToLeft)
110 /// );
111 /// ```
112 #[cfg(feature = "compiled_data")]
113 pub const fn new_with_expander(expander: LocaleExpander) -> Self {
114 LocaleDirectionality {
115 script_direction: DataPayload::from_static_ref(
116 crate::provider::Baked::SINGLETON_LOCID_TRANSFORM_SCRIPT_DIR_V1,
117 ),
118 expander,
119 }
120 }
121
122 #[doc = icu_provider::gen_any_buffer_unstable_docs!(UNSTABLE, Self::new_with_expander)]
123 pub fn try_new_with_expander_unstable<P>(
124 provider: &P,
125 expander: LocaleExpander,
126 ) -> Result<LocaleDirectionality, LocaleTransformError>
127 where
128 P: DataProvider<ScriptDirectionV1Marker> + ?Sized,
129 {
130 let script_direction = provider.load(Default::default())?.take_payload()?;
131
132 Ok(LocaleDirectionality {
133 script_direction,
134 expander,
135 })
136 }
137
138 /// Returns the script direction of the given locale.
139 ///
140 /// Note that the direction is a property of the script of a locale, not of the language. As such,
141 /// when given a locale without an associated script tag (i.e., `locale!("en")` vs. `locale!("en-Latn")`),
142 /// this method first tries to infer the script using the language and region before returning its direction.
143 ///
144 /// If you already have a script struct and want to get its direction, you should use
145 /// `Locale::from(Some(my_script))` and call this method.
146 ///
147 /// This method will return `None` if either a locale's script cannot be determined, or there is no information
148 /// for the script.
149 ///
150 /// # Examples
151 ///
152 /// Using an existing locale:
153 ///
154 /// ```
155 /// use icu::locid::locale;
156 /// use icu::locid_transform::{Direction, LocaleDirectionality};
157 ///
158 /// let ld = LocaleDirectionality::new();
159 ///
160 /// assert_eq!(ld.get(&locale!("en-US")), Some(Direction::LeftToRight));
161 ///
162 /// assert_eq!(ld.get(&locale!("ar")), Some(Direction::RightToLeft));
163 ///
164 /// assert_eq!(ld.get(&locale!("en-Arab")), Some(Direction::RightToLeft));
165 ///
166 /// assert_eq!(ld.get(&locale!("foo")), None);
167 /// ```
168 ///
169 /// Using a script directly:
170 ///
171 /// ```
172 /// use icu::locid::subtags::script;
173 /// use icu::locid::Locale;
174 /// use icu::locid_transform::{Direction, LocaleDirectionality};
175 ///
176 /// let ld = LocaleDirectionality::new();
177 ///
178 /// assert_eq!(
179 /// ld.get(&Locale::from(Some(script!("Latn")))),
180 /// Some(Direction::LeftToRight)
181 /// );
182 /// ```
183 pub fn get(&self, locale: impl AsRef<LanguageIdentifier>) -> Option<Direction> {
184 let script = self.expander.get_likely_script(locale.as_ref())?;
185
186 if self.script_in_ltr(script) {
187 Some(Direction::LeftToRight)
188 } else if self.script_in_rtl(script) {
189 Some(Direction::RightToLeft)
190 } else {
191 None
192 }
193 }
194
195 /// Returns whether the given locale is right-to-left.
196 ///
197 /// Note that if this method returns `false`, the locale is either left-to-right or
198 /// the [`LocaleDirectionality`] does not include data for the locale.
199 /// You should use [`LocaleDirectionality::get`] if you need to differentiate between these cases.
200 ///
201 /// See [`LocaleDirectionality::get`] for more information.
202 pub fn is_right_to_left(&self, locale: impl AsRef<LanguageIdentifier>) -> bool {
203 self.expander
204 .get_likely_script(locale.as_ref())
205 .map(|s| self.script_in_rtl(s))
206 .unwrap_or(false)
207 }
208
209 /// Returns whether the given locale is left-to-right.
210 ///
211 /// Note that if this method returns `false`, the locale is either right-to-left or
212 /// the [`LocaleDirectionality`] does not include data for the locale.
213 /// You should use [`LocaleDirectionality::get`] if you need to differentiate between these cases.
214 ///
215 /// See [`LocaleDirectionality::get`] for more information.
216 pub fn is_left_to_right(&self, locale: impl AsRef<LanguageIdentifier>) -> bool {
217 self.expander
218 .get_likely_script(locale.as_ref())
219 .map(|s| self.script_in_ltr(s))
220 .unwrap_or(false)
221 }
222
223 fn script_in_rtl(&self, script: Script) -> bool {
224 self.script_direction
225 .get()
226 .rtl
227 .binary_search(&script.into_tinystr().to_unvalidated())
228 .is_ok()
229 }
230
231 fn script_in_ltr(&self, script: Script) -> bool {
232 self.script_direction
233 .get()
234 .ltr
235 .binary_search(&script.into_tinystr().to_unvalidated())
236 .is_ok()
237 }
238}