Source code

Revision control

Copy as Markdown

Other Tools

// This file is part of ICU4X. For terms of use, please see the file
// called LICENSE at the top level of the ICU4X source tree
#[diplomat::bridge]
pub mod ffi {
use alloc::boxed::Box;
use icu_collator::{Collator, CollatorOptions};
use crate::{
common::ffi::ICU4XOrdering, errors::ffi::ICU4XError, locale::ffi::ICU4XLocale,
provider::ffi::ICU4XDataProvider,
};
#[diplomat::opaque]
#[diplomat::rust_link(icu::collator::Collator, Struct)]
pub struct ICU4XCollator(pub Collator);
#[diplomat::rust_link(icu::collator::CollatorOptions, Struct)]
#[diplomat::rust_link(icu::collator::CollatorOptions::new, FnInStruct, hidden)]
#[diplomat::attr(dart, rename = "CollatorOptions")]
pub struct ICU4XCollatorOptionsV1 {
pub strength: ICU4XCollatorStrength,
pub alternate_handling: ICU4XCollatorAlternateHandling,
pub case_first: ICU4XCollatorCaseFirst,
pub max_variable: ICU4XCollatorMaxVariable,
pub case_level: ICU4XCollatorCaseLevel,
pub numeric: ICU4XCollatorNumeric,
pub backward_second_level: ICU4XCollatorBackwardSecondLevel,
}
// Note the flipped order of the words `Collator` and `Resolved`, because
// in FFI `Collator` is part of the `ICU4XCollator` prefix, but in Rust,
// `ResolvedCollatorOptions` makes more sense as English.
#[diplomat::rust_link(icu::collator::ResolvedCollatorOptions, Struct)]
#[diplomat::out]
#[diplomat::attr(dart, rename = "ResolvedCollatorOptions")]
pub struct ICU4XCollatorResolvedOptionsV1 {
pub strength: ICU4XCollatorStrength,
pub alternate_handling: ICU4XCollatorAlternateHandling,
pub case_first: ICU4XCollatorCaseFirst,
pub max_variable: ICU4XCollatorMaxVariable,
pub case_level: ICU4XCollatorCaseLevel,
pub numeric: ICU4XCollatorNumeric,
pub backward_second_level: ICU4XCollatorBackwardSecondLevel,
}
#[diplomat::rust_link(icu::collator::Strength, Enum)]
#[derive(Eq, PartialEq, Debug, PartialOrd, Ord)]
pub enum ICU4XCollatorStrength {
Auto = 0,
Primary = 1,
Secondary = 2,
Tertiary = 3,
Quaternary = 4,
Identical = 5,
}
#[diplomat::rust_link(icu::collator::AlternateHandling, Enum)]
#[derive(Eq, PartialEq, Debug, PartialOrd, Ord)]
pub enum ICU4XCollatorAlternateHandling {
Auto = 0,
NonIgnorable = 1,
Shifted = 2,
}
#[diplomat::rust_link(icu::collator::CaseFirst, Enum)]
#[derive(Eq, PartialEq, Debug, PartialOrd, Ord)]
pub enum ICU4XCollatorCaseFirst {
Auto = 0,
Off = 1,
LowerFirst = 2,
UpperFirst = 3,
}
#[diplomat::rust_link(icu::collator::MaxVariable, Enum)]
#[derive(Eq, PartialEq, Debug, PartialOrd, Ord)]
pub enum ICU4XCollatorMaxVariable {
Auto = 0,
Space = 1,
Punctuation = 2,
Symbol = 3,
Currency = 4,
}
#[diplomat::rust_link(icu::collator::CaseLevel, Enum)]
#[derive(Eq, PartialEq, Debug, PartialOrd, Ord)]
pub enum ICU4XCollatorCaseLevel {
Auto = 0,
Off = 1,
On = 2,
}
#[diplomat::rust_link(icu::collator::Numeric, Enum)]
#[derive(Eq, PartialEq, Debug, PartialOrd, Ord)]
pub enum ICU4XCollatorNumeric {
Auto = 0,
Off = 1,
On = 2,
}
#[diplomat::rust_link(icu::collator::BackwardSecondLevel, Enum)]
#[derive(Eq, PartialEq, Debug, PartialOrd, Ord)]
pub enum ICU4XCollatorBackwardSecondLevel {
Auto = 0,
Off = 1,
On = 2,
}
impl ICU4XCollator {
/// Construct a new Collator instance.
#[diplomat::rust_link(icu::collator::Collator::try_new, FnInStruct)]
#[diplomat::attr(all(supports = constructors, supports = fallible_constructors), constructor)]
pub fn create_v1(
provider: &ICU4XDataProvider,
locale: &ICU4XLocale,
options: ICU4XCollatorOptionsV1,
) -> Result<Box<ICU4XCollator>, ICU4XError> {
let locale = locale.to_datalocale();
let options = CollatorOptions::from(options);
Ok(Box::new(ICU4XCollator(call_constructor!(
Collator::try_new,
Collator::try_new_with_any_provider,
Collator::try_new_with_buffer_provider,
provider,
&locale,
options,
)?)))
}
/// Compare two strings.
///
/// Ill-formed input is treated as if errors had been replaced with REPLACEMENT CHARACTERs according
/// to the WHATWG Encoding Standard.
#[diplomat::rust_link(icu::collator::Collator::compare_utf8, FnInStruct)]
#[diplomat::attr(dart, disable)]
pub fn compare(&self, left: &DiplomatStr, right: &DiplomatStr) -> ICU4XOrdering {
self.0.compare_utf8(left, right).into()
}
/// Compare two strings.
#[diplomat::rust_link(icu::collator::Collator::compare, FnInStruct)]
#[diplomat::attr(*, disable)]
pub fn compare_valid_utf8(&self, left: &str, right: &str) -> ICU4XOrdering {
self.0.compare(left, right).into()
}
/// Compare two strings.
///
/// Ill-formed input is treated as if errors had been replaced with REPLACEMENT CHARACTERs according
/// to the WHATWG Encoding Standard.
#[diplomat::rust_link(icu::collator::Collator::compare_utf16, FnInStruct)]
#[diplomat::attr(dart, disable)]
pub fn compare_utf16(&self, left: &DiplomatStr16, right: &DiplomatStr16) -> ICU4XOrdering {
self.0.compare_utf16(left, right).into()
}
/// Compare two strings.
///
/// Ill-formed input is treated as if errors had been replaced with REPLACEMENT CHARACTERs according
/// to the WHATWG Encoding Standard.
#[diplomat::rust_link(icu::collator::Collator::compare_utf16, FnInStruct)]
#[diplomat::skip_if_ast]
#[diplomat::attr(dart, rename = "compare")]
pub fn compare_utf16_(
&self,
left: &DiplomatStr16,
right: &DiplomatStr16,
) -> core::cmp::Ordering {
self.0.compare_utf16(left, right)
}
/// The resolved options showing how the default options, the requested options,
/// and the options from locale data were combined. None of the struct fields
/// will have `Auto` as the value.
#[diplomat::rust_link(icu::collator::Collator::resolved_options, FnInStruct)]
#[diplomat::attr(supports = accessors, getter)]
pub fn resolved_options(&self) -> ICU4XCollatorResolvedOptionsV1 {
self.0.resolved_options().into()
}
}
}
use icu_collator::{
AlternateHandling, BackwardSecondLevel, CaseFirst, CaseLevel, CollatorOptions, MaxVariable,
Numeric, ResolvedCollatorOptions, Strength,
};
impl From<ffi::ICU4XCollatorStrength> for Option<Strength> {
fn from(strength: ffi::ICU4XCollatorStrength) -> Option<Strength> {
match strength {
ffi::ICU4XCollatorStrength::Auto => None,
ffi::ICU4XCollatorStrength::Primary => Some(Strength::Primary),
ffi::ICU4XCollatorStrength::Secondary => Some(Strength::Secondary),
ffi::ICU4XCollatorStrength::Tertiary => Some(Strength::Tertiary),
ffi::ICU4XCollatorStrength::Quaternary => Some(Strength::Quaternary),
ffi::ICU4XCollatorStrength::Identical => Some(Strength::Identical),
}
}
}
impl From<ffi::ICU4XCollatorAlternateHandling> for Option<AlternateHandling> {
fn from(alternate_handling: ffi::ICU4XCollatorAlternateHandling) -> Option<AlternateHandling> {
match alternate_handling {
ffi::ICU4XCollatorAlternateHandling::Auto => None,
ffi::ICU4XCollatorAlternateHandling::NonIgnorable => {
Some(AlternateHandling::NonIgnorable)
}
ffi::ICU4XCollatorAlternateHandling::Shifted => Some(AlternateHandling::Shifted),
}
}
}
impl From<ffi::ICU4XCollatorCaseFirst> for Option<CaseFirst> {
fn from(case_first: ffi::ICU4XCollatorCaseFirst) -> Option<CaseFirst> {
match case_first {
ffi::ICU4XCollatorCaseFirst::Auto => None,
ffi::ICU4XCollatorCaseFirst::Off => Some(CaseFirst::Off),
ffi::ICU4XCollatorCaseFirst::LowerFirst => Some(CaseFirst::LowerFirst),
ffi::ICU4XCollatorCaseFirst::UpperFirst => Some(CaseFirst::UpperFirst),
}
}
}
impl From<ffi::ICU4XCollatorMaxVariable> for Option<MaxVariable> {
fn from(max_variable: ffi::ICU4XCollatorMaxVariable) -> Option<MaxVariable> {
match max_variable {
ffi::ICU4XCollatorMaxVariable::Auto => None,
ffi::ICU4XCollatorMaxVariable::Space => Some(MaxVariable::Space),
ffi::ICU4XCollatorMaxVariable::Punctuation => Some(MaxVariable::Punctuation),
ffi::ICU4XCollatorMaxVariable::Symbol => Some(MaxVariable::Symbol),
ffi::ICU4XCollatorMaxVariable::Currency => Some(MaxVariable::Currency),
}
}
}
impl From<ffi::ICU4XCollatorCaseLevel> for Option<CaseLevel> {
fn from(case_level: ffi::ICU4XCollatorCaseLevel) -> Option<CaseLevel> {
match case_level {
ffi::ICU4XCollatorCaseLevel::Auto => None,
ffi::ICU4XCollatorCaseLevel::Off => Some(CaseLevel::Off),
ffi::ICU4XCollatorCaseLevel::On => Some(CaseLevel::On),
}
}
}
impl From<ffi::ICU4XCollatorNumeric> for Option<Numeric> {
fn from(numeric: ffi::ICU4XCollatorNumeric) -> Option<Numeric> {
match numeric {
ffi::ICU4XCollatorNumeric::Auto => None,
ffi::ICU4XCollatorNumeric::Off => Some(Numeric::Off),
ffi::ICU4XCollatorNumeric::On => Some(Numeric::On),
}
}
}
impl From<ffi::ICU4XCollatorBackwardSecondLevel> for Option<BackwardSecondLevel> {
fn from(
backward_second_level: ffi::ICU4XCollatorBackwardSecondLevel,
) -> Option<BackwardSecondLevel> {
match backward_second_level {
ffi::ICU4XCollatorBackwardSecondLevel::Auto => None,
ffi::ICU4XCollatorBackwardSecondLevel::Off => Some(BackwardSecondLevel::Off),
ffi::ICU4XCollatorBackwardSecondLevel::On => Some(BackwardSecondLevel::On),
}
}
}
impl From<Strength> for ffi::ICU4XCollatorStrength {
fn from(strength: Strength) -> ffi::ICU4XCollatorStrength {
match strength {
Strength::Primary => ffi::ICU4XCollatorStrength::Primary,
Strength::Secondary => ffi::ICU4XCollatorStrength::Secondary,
Strength::Tertiary => ffi::ICU4XCollatorStrength::Tertiary,
Strength::Quaternary => ffi::ICU4XCollatorStrength::Quaternary,
Strength::Identical => ffi::ICU4XCollatorStrength::Identical,
_ => {
debug_assert!(false, "FFI out of sync");
ffi::ICU4XCollatorStrength::Identical // Highest we know of
}
}
}
}
impl From<AlternateHandling> for ffi::ICU4XCollatorAlternateHandling {
fn from(alternate_handling: AlternateHandling) -> ffi::ICU4XCollatorAlternateHandling {
match alternate_handling {
AlternateHandling::NonIgnorable => ffi::ICU4XCollatorAlternateHandling::NonIgnorable,
AlternateHandling::Shifted => ffi::ICU4XCollatorAlternateHandling::Shifted,
_ => {
debug_assert!(false, "FFI out of sync");
// Possible future values: ShiftTrimmed, Blanked
ffi::ICU4XCollatorAlternateHandling::Shifted // Highest we know of
}
}
}
}
impl From<CaseFirst> for ffi::ICU4XCollatorCaseFirst {
fn from(case_first: CaseFirst) -> ffi::ICU4XCollatorCaseFirst {
match case_first {
CaseFirst::Off => ffi::ICU4XCollatorCaseFirst::Off,
CaseFirst::LowerFirst => ffi::ICU4XCollatorCaseFirst::LowerFirst,
CaseFirst::UpperFirst => ffi::ICU4XCollatorCaseFirst::UpperFirst,
_ => {
debug_assert!(false, "FFI out of sync");
// Does it even make sense that `CaseFirst` is non-exhaustive?
ffi::ICU4XCollatorCaseFirst::Off // The most neutral value
}
}
}
}
impl From<MaxVariable> for ffi::ICU4XCollatorMaxVariable {
fn from(max_variable: MaxVariable) -> ffi::ICU4XCollatorMaxVariable {
match max_variable {
MaxVariable::Space => ffi::ICU4XCollatorMaxVariable::Space,
MaxVariable::Punctuation => ffi::ICU4XCollatorMaxVariable::Punctuation,
MaxVariable::Symbol => ffi::ICU4XCollatorMaxVariable::Symbol,
MaxVariable::Currency => ffi::ICU4XCollatorMaxVariable::Currency,
_ => {
debug_assert!(false, "FFI out of sync");
ffi::ICU4XCollatorMaxVariable::Currency // Highest we know of
}
}
}
}
impl From<CaseLevel> for ffi::ICU4XCollatorCaseLevel {
fn from(case_level: CaseLevel) -> ffi::ICU4XCollatorCaseLevel {
match case_level {
CaseLevel::Off => ffi::ICU4XCollatorCaseLevel::Off,
CaseLevel::On => ffi::ICU4XCollatorCaseLevel::On,
_ => {
debug_assert!(false, "FFI out of sync");
ffi::ICU4XCollatorCaseLevel::On // The most enabled that we know of
}
}
}
}
impl From<Numeric> for ffi::ICU4XCollatorNumeric {
fn from(numeric: Numeric) -> ffi::ICU4XCollatorNumeric {
match numeric {
Numeric::Off => ffi::ICU4XCollatorNumeric::Off,
Numeric::On => ffi::ICU4XCollatorNumeric::On,
_ => {
debug_assert!(false, "FFI out of sync");
ffi::ICU4XCollatorNumeric::On // The most enabled that we know of
}
}
}
}
impl From<BackwardSecondLevel> for ffi::ICU4XCollatorBackwardSecondLevel {
fn from(backward_second_level: BackwardSecondLevel) -> ffi::ICU4XCollatorBackwardSecondLevel {
match backward_second_level {
BackwardSecondLevel::Off => ffi::ICU4XCollatorBackwardSecondLevel::Off,
BackwardSecondLevel::On => ffi::ICU4XCollatorBackwardSecondLevel::On,
_ => {
debug_assert!(false, "FFI out of sync");
ffi::ICU4XCollatorBackwardSecondLevel::On // The most enabled that we know of
}
}
}
}
impl From<ffi::ICU4XCollatorOptionsV1> for CollatorOptions {
fn from(options: ffi::ICU4XCollatorOptionsV1) -> CollatorOptions {
let mut result = CollatorOptions::new();
result.strength = options.strength.into();
result.alternate_handling = options.alternate_handling.into();
result.case_first = options.case_first.into();
result.max_variable = options.max_variable.into();
result.case_level = options.case_level.into();
result.numeric = options.numeric.into();
result.backward_second_level = options.backward_second_level.into();
result
}
}
impl From<ResolvedCollatorOptions> for ffi::ICU4XCollatorResolvedOptionsV1 {
fn from(options: ResolvedCollatorOptions) -> ffi::ICU4XCollatorResolvedOptionsV1 {
Self {
strength: options.strength.into(),
alternate_handling: options.alternate_handling.into(),
case_first: options.case_first.into(),
max_variable: options.max_variable.into(),
case_level: options.case_level.into(),
numeric: options.numeric.into(),
backward_second_level: options.backward_second_level.into(),
}
}
}