use crate::cldr_serde;
use crate::SourceDataProvider;
use icu::experimental::dimension::provider::units::display_names::{
UnitsDisplayNames, UnitsDisplayNamesV1,
};
use icu_provider::prelude::*;
use icu_provider::DataMarkerAttributes;
use std::collections::HashSet;
impl DataProvider<UnitsDisplayNamesV1> for SourceDataProvider {
fn load(&self, req: DataRequest) -> Result<DataResponse<UnitsDisplayNamesV1>, DataError> {
self.check_req::<UnitsDisplayNamesV1>(req)?;
let (length, unit) = req.id.marker_attributes.split_once('-').ok_or_else(|| {
DataErrorKind::InvalidRequest
.into_error()
.with_req(UnitsDisplayNamesV1::INFO, req)
})?;
let units_format_data: &cldr_serde::units::data::Resource = self
.cldr()?
.units()
.read_and_parse(req.id.locale, "units.json")?;
let units_format_data = &units_format_data.main.value.units;
let unit_patterns = match length {
"long" => &units_format_data.long,
"short" => &units_format_data.short,
"narrow" => &units_format_data.narrow,
_ => {
return Err(DataErrorKind::InvalidRequest
.into_error()
.with_debug_context(length))
}
}
.categories
.iter()
.find_map(|(_, units_map)| units_map.get(unit))
.ok_or_else(|| {
DataErrorKind::IdentifierNotFound
.into_error()
.with_debug_context(length)
})?;
Ok(DataResponse {
metadata: Default::default(),
payload: DataPayload::from_owned(UnitsDisplayNames {
patterns: unit_patterns.try_into_plural_elements_packed_cow()?,
}),
})
}
}
impl crate::IterableDataProviderCached<UnitsDisplayNamesV1> for SourceDataProvider {
fn iter_ids_cached(&self) -> Result<HashSet<DataIdentifierCow<'static>>, DataError> {
let mut data_locales = HashSet::new();
let numbers = self.cldr()?.numbers();
let locales = numbers.list_locales()?;
for locale in locales {
let units_format_data: &cldr_serde::units::data::Resource =
self.cldr()?.units().read_and_parse(&locale, "units.json")?;
let units_format_data = &units_format_data.main.value.units;
for length in &["long", "short", "narrow"] {
let length_patterns = match *length {
"long" => &units_format_data.long,
"short" => &units_format_data.short,
"narrow" => &units_format_data.narrow,
_ => {
return Err(DataErrorKind::IdentifierNotFound
.into_error()
.with_debug_context(length))
}
};
for units_map in length_patterns.categories.values() {
for (unit, patterns) in units_map {
if patterns.other.is_none() {
continue;
}
data_locales.insert(DataIdentifierCow::from_owned(
DataMarkerAttributes::try_from_string(format!("{length}-{unit}"))
.map_err(|_| {
DataError::custom("Failed to parse the attribute")
.with_debug_context(&unit)
})?,
locale,
));
}
}
}
}
Ok(data_locales)
}
}
#[test]
fn test_basic() {
use icu::locale::langid;
use icu::plurals::PluralRules;
use icu_provider::prelude::*;
use writeable::assert_writeable_eq;
let provider = SourceDataProvider::new_testing();
let us_locale_long_meter: DataPayload<UnitsDisplayNamesV1> = provider
.load(DataRequest {
id: DataIdentifierBorrowed::for_marker_attributes_and_locale(
DataMarkerAttributes::from_str_or_panic("long-meter"),
&langid!("en").into(),
),
..Default::default()
})
.unwrap()
.payload;
let units_us = us_locale_long_meter.get().to_owned();
let en_rules = PluralRules::try_new_cardinal_unstable(&provider, langid!("en").into()).unwrap();
let long = units_us.patterns.get(1.into(), &en_rules).interpolate([1]);
assert_writeable_eq!(long, "1 meter");
let us_locale_short_meter: DataPayload<UnitsDisplayNamesV1> = provider
.load(DataRequest {
id: DataIdentifierBorrowed::for_marker_attributes_and_locale(
DataMarkerAttributes::from_str_or_panic("short-meter"),
&langid!("en").into(),
),
..Default::default()
})
.unwrap()
.payload;
let units_us_short = us_locale_short_meter.get().to_owned();
let short = units_us_short
.patterns
.get(5.into(), &en_rules)
.interpolate([5]);
assert_writeable_eq!(short, "5 m");
let ar_eg_locale: DataPayload<UnitsDisplayNamesV1> = provider
.load(DataRequest {
id: DataIdentifierBorrowed::for_marker_attributes_and_locale(
DataMarkerAttributes::from_str_or_panic("long-meter"),
&langid!("ar-EG").into(),
),
..Default::default()
})
.unwrap()
.payload;
let ar_eg_units = ar_eg_locale.get().to_owned();
let ar_rules = PluralRules::try_new_cardinal_unstable(&provider, langid!("ar").into()).unwrap();
let long = ar_eg_units
.patterns
.get(1.into(), &ar_rules)
.interpolate([1]);
assert_writeable_eq!(long, "متر");
let fr_locale: DataPayload<UnitsDisplayNamesV1> = provider
.load(DataRequest {
id: DataIdentifierBorrowed::for_marker_attributes_and_locale(
DataMarkerAttributes::from_str_or_panic("short-meter"),
&langid!("fr").into(),
),
..Default::default()
})
.unwrap()
.payload;
let fr_units = fr_locale.get().to_owned();
let fr_rules = PluralRules::try_new_cardinal_unstable(&provider, langid!("fr").into()).unwrap();
let short = fr_units.patterns.get(5.into(), &fr_rules).interpolate([5]);
assert_writeable_eq!(short, "5 m");
}