icu_properties/
bidi_data.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//! Data and APIs for supporting specific Bidi properties data in an efficient structure.
6//!
7//! Supported properties are:
8//! - `Bidi_Paired_Bracket`
9//! - `Bidi_Paired_Bracket_Type`
10//! - `Bidi_Mirrored`
11//! - `Bidi_Mirroring_Glyph`
12
13use crate::provider::bidi_data::{
14    BidiAuxiliaryPropertiesV1, BidiAuxiliaryPropertiesV1Marker, CheckedBidiPairedBracketType,
15};
16use crate::PropertiesError;
17
18use icu_provider::prelude::*;
19
20/// A wrapper around certain Bidi properties data. Can be obtained via [`bidi_auxiliary_properties()`] and
21/// related getters.
22///
23/// Most useful methods are on [`BidiAuxiliaryPropertiesBorrowed`] obtained by calling [`BidiAuxiliaryProperties::as_borrowed()`]
24#[derive(Debug)]
25pub struct BidiAuxiliaryProperties {
26    data: DataPayload<BidiAuxiliaryPropertiesV1Marker>,
27}
28
29impl BidiAuxiliaryProperties {
30    /// Construct a borrowed version of this type that can be queried.
31    ///
32    /// This avoids a potential small underlying cost per API call by consolidating it
33    /// up front.
34    #[inline]
35    pub fn as_borrowed(&self) -> BidiAuxiliaryPropertiesBorrowed<'_> {
36        BidiAuxiliaryPropertiesBorrowed {
37            data: self.data.get(),
38        }
39    }
40
41    /// Construct a new one from loaded data
42    ///
43    /// Typically it is preferable to use getters like [`bidi_auxiliary_properties()`] instead
44    pub fn from_data(data: DataPayload<BidiAuxiliaryPropertiesV1Marker>) -> Self {
45        Self { data }
46    }
47}
48
49/// This struct represents the properties Bidi_Mirrored and Bidi_Mirroring_Glyph.
50/// If Bidi_Mirroring_Glyph is not defined for a code point, then the value in the
51/// struct is `None`.
52#[derive(Debug, Eq, PartialEq)]
53#[non_exhaustive]
54pub struct BidiMirroringProperties {
55    /// Represents the Bidi_Mirroring_Glyph property value
56    pub mirroring_glyph: Option<char>,
57    /// Represents the Bidi_Mirrored property value
58    pub mirrored: bool,
59}
60
61/// The enum represents Bidi_Paired_Bracket_Type, the char represents Bidi_Paired_Bracket.
62/// Bidi_Paired_Bracket has a value of `None` when Bidi_Paired_Bracket_Type is `None`.
63#[derive(Debug, Eq, PartialEq)]
64#[non_exhaustive]
65pub enum BidiPairingProperties {
66    /// Represents Bidi_Paired_Bracket_Type=Open, and the Bidi_Paired_Bracket value for that code point.
67    Open(char),
68    /// Represents Bidi_Paired_Bracket_Type=Close, and the Bidi_Paired_Bracket value for that code point.
69    Close(char),
70    /// Represents Bidi_Paired_Bracket_Type=None, which cooccurs with Bidi_Paired_Bracket
71    /// being undefined for that code point.
72    None,
73}
74
75/// A borrowed wrapper around Bidi properties data, returned by
76/// [`BidiAuxiliaryProperties::as_borrowed()`]. More efficient to query.
77#[derive(Debug)]
78pub struct BidiAuxiliaryPropertiesBorrowed<'a> {
79    data: &'a BidiAuxiliaryPropertiesV1<'a>,
80}
81
82impl<'a> BidiAuxiliaryPropertiesBorrowed<'a> {
83    // The source data coming from icuexportdata will use 0 to represent the
84    // property value in cases for which the Bidi_Mirroring_Glyph property value
85    // of a code point is undefined. Since Rust types can be more expressive, we
86    // should represent these cases as None.
87    fn convert_mirroring_glyph_data(trie_data_char: char) -> Option<char> {
88        if trie_data_char as u32 == 0 {
89            None
90        } else {
91            Some(trie_data_char)
92        }
93    }
94
95    /// Return a struct for the given code point representing Bidi mirroring-related
96    /// property values. See [`BidiMirroringProperties`].
97    ///
98    /// # Examples
99    /// ```
100    /// use icu::properties::bidi_data;
101    ///
102    /// let bidi_data = bidi_data::bidi_auxiliary_properties();
103    ///
104    /// let open_paren = bidi_data.get32_mirroring_props('(' as u32);
105    /// assert_eq!(open_paren.mirroring_glyph, Some(')'));
106    /// assert_eq!(open_paren.mirrored, true);
107    /// let close_paren = bidi_data.get32_mirroring_props(')' as u32);
108    /// assert_eq!(close_paren.mirroring_glyph, Some('('));
109    /// assert_eq!(close_paren.mirrored, true);
110    /// let open_angle_bracket = bidi_data.get32_mirroring_props('<' as u32);
111    /// assert_eq!(open_angle_bracket.mirroring_glyph, Some('>'));
112    /// assert_eq!(open_angle_bracket.mirrored, true);
113    /// let close_angle_bracket = bidi_data.get32_mirroring_props('>' as u32);
114    /// assert_eq!(close_angle_bracket.mirroring_glyph, Some('<'));
115    /// assert_eq!(close_angle_bracket.mirrored, true);
116    /// let three = bidi_data.get32_mirroring_props('3' as u32);
117    /// assert_eq!(three.mirroring_glyph, None);
118    /// assert_eq!(three.mirrored, false);
119    /// ```
120    pub fn get32_mirroring_props(&self, code_point: u32) -> BidiMirroringProperties {
121        let bidi_aux_props = self.data.trie.get32(code_point);
122        let mirroring_glyph_opt =
123            Self::convert_mirroring_glyph_data(bidi_aux_props.mirroring_glyph);
124        BidiMirroringProperties {
125            mirroring_glyph: mirroring_glyph_opt,
126            mirrored: bidi_aux_props.mirrored,
127        }
128    }
129
130    /// Return a struct for the given code point representing Bidi bracket
131    /// pairing-related property values. See [`BidiPairingProperties`]
132    ///
133    /// # Examples
134    /// ```
135    /// use icu::properties::{bidi_data, bidi_data::BidiPairingProperties};
136    ///
137    /// let bidi_data = bidi_data::bidi_auxiliary_properties();
138    ///
139    /// let open_paren = bidi_data.get32_pairing_props('(' as u32);
140    /// assert_eq!(open_paren, BidiPairingProperties::Open(')'));
141    /// let close_paren = bidi_data.get32_pairing_props(')' as u32);
142    /// assert_eq!(close_paren, BidiPairingProperties::Close('('));
143    /// let open_angle_bracket = bidi_data.get32_pairing_props('<' as u32);
144    /// assert_eq!(open_angle_bracket, BidiPairingProperties::None);
145    /// let close_angle_bracket = bidi_data.get32_pairing_props('>' as u32);
146    /// assert_eq!(close_angle_bracket, BidiPairingProperties::None);
147    /// let three = bidi_data.get32_pairing_props('3' as u32);
148    /// assert_eq!(three, BidiPairingProperties::None);
149    /// ```
150    pub fn get32_pairing_props(&self, code_point: u32) -> BidiPairingProperties {
151        let bidi_aux_props = self.data.trie.get32(code_point);
152        let mirroring_glyph = bidi_aux_props.mirroring_glyph;
153        let paired_bracket_type = bidi_aux_props.paired_bracket_type;
154        match paired_bracket_type {
155            CheckedBidiPairedBracketType::Open => BidiPairingProperties::Open(mirroring_glyph),
156            CheckedBidiPairedBracketType::Close => BidiPairingProperties::Close(mirroring_glyph),
157            _ => BidiPairingProperties::None,
158        }
159    }
160}
161
162impl BidiAuxiliaryPropertiesBorrowed<'static> {
163    /// Cheaply converts a [`BidiAuxiliaryPropertiesBorrowed<'static>`] into a [`BidiAuxiliaryProperties`].
164    ///
165    /// Note: Due to branching and indirection, using [`BidiAuxiliaryProperties`] might inhibit some
166    /// compile-time optimizations that are possible with [`BidiAuxiliaryPropertiesBorrowed`].
167    pub const fn static_to_owned(self) -> BidiAuxiliaryProperties {
168        BidiAuxiliaryProperties {
169            data: DataPayload::from_static_ref(self.data),
170        }
171    }
172}
173
174/// Creates a [`BidiAuxiliaryPropertiesV1`] struct that represents the data for certain
175/// Bidi properties.
176///
177/// ✨ *Enabled with the `compiled_data` Cargo feature.*
178///
179/// [📚 Help choosing a constructor](icu_provider::constructors)
180///
181/// # Examples
182/// ```
183/// use icu::properties::bidi_data;
184///
185/// let bidi_data = bidi_data::bidi_auxiliary_properties();
186///
187/// let open_paren = bidi_data.get32_mirroring_props('(' as u32);
188/// assert_eq!(open_paren.mirroring_glyph, Some(')'));
189/// assert_eq!(open_paren.mirrored, true);
190/// ```
191#[cfg(feature = "compiled_data")]
192pub const fn bidi_auxiliary_properties() -> BidiAuxiliaryPropertiesBorrowed<'static> {
193    BidiAuxiliaryPropertiesBorrowed {
194        data: crate::provider::Baked::SINGLETON_PROPS_BIDIAUXILIARYPROPS_V1,
195    }
196}
197
198icu_provider::gen_any_buffer_data_constructors!(
199    locale: skip,
200    options: skip,
201    result: Result<BidiAuxiliaryProperties, PropertiesError>,
202    #[cfg(skip)]
203    functions: [
204        bidi_auxiliary_properties,
205        load_bidi_auxiliary_properties_with_any_provider,
206        load_bidi_auxiliary_properties_with_buffer_provider,
207        load_bidi_auxiliary_properties_unstable,
208    ]
209);
210
211#[doc = icu_provider::gen_any_buffer_unstable_docs!(UNSTABLE, bidi_auxiliary_properties)]
212pub fn load_bidi_auxiliary_properties_unstable(
213    provider: &(impl DataProvider<BidiAuxiliaryPropertiesV1Marker> + ?Sized),
214) -> Result<BidiAuxiliaryProperties, PropertiesError> {
215    Ok(provider
216        .load(Default::default())
217        .and_then(DataResponse::take_payload)
218        .map(BidiAuxiliaryProperties::from_data)?)
219}