use crate::provider::calendar::*;
use crate::{calendar, options::length, raw};
use crate::{input::DateInput, DateTimeError, FormattedDateTime};
use alloc::string::String;
use icu_calendar::any_calendar::{AnyCalendar, AnyCalendarKind};
use icu_calendar::provider::{
JapaneseErasV1Marker, JapaneseExtendedErasV1Marker, WeekDataV1Marker,
};
use icu_calendar::Date;
use icu_decimal::provider::DecimalSymbolsV1Marker;
use icu_plurals::provider::OrdinalV1Marker;
use icu_provider::prelude::*;
use icu_provider::DataLocale;
use writeable::Writeable;
pub struct DateFormatter(pub(crate) raw::DateFormatter, pub(crate) AnyCalendar);
impl DateFormatter {
#[inline]
pub fn try_new_with_length_with_any_provider<P>(
data_provider: &P,
locale: &DataLocale,
length: length::Date,
) -> Result<Self, DateTimeError>
where
P: AnyProvider,
{
let downcasting = data_provider.as_downcasting();
Self::try_new_with_length_unstable(&downcasting, locale, length)
}
#[inline]
#[cfg(feature = "serde")]
pub fn try_new_with_length_with_buffer_provider<P>(
data_provider: &P,
locale: &DataLocale,
length: length::Date,
) -> Result<Self, DateTimeError>
where
P: BufferProvider,
{
let deserializing = data_provider.as_deserializing();
Self::try_new_with_length_unstable(&deserializing, locale, length)
}
#[inline(never)]
pub fn try_new_with_length_unstable<P>(
data_provider: &P,
locale: &DataLocale,
length: length::Date,
) -> Result<Self, DateTimeError>
where
P: DataProvider<TimeSymbolsV1Marker>
+ DataProvider<TimeLengthsV1Marker>
+ DataProvider<OrdinalV1Marker>
+ DataProvider<WeekDataV1Marker>
+ DataProvider<DecimalSymbolsV1Marker>
+ DataProvider<GregorianDateLengthsV1Marker>
+ DataProvider<BuddhistDateLengthsV1Marker>
+ DataProvider<JapaneseDateLengthsV1Marker>
+ DataProvider<JapaneseExtendedDateLengthsV1Marker>
+ DataProvider<CopticDateLengthsV1Marker>
+ DataProvider<IndianDateLengthsV1Marker>
+ DataProvider<EthiopianDateLengthsV1Marker>
+ DataProvider<GregorianDateSymbolsV1Marker>
+ DataProvider<BuddhistDateSymbolsV1Marker>
+ DataProvider<JapaneseDateSymbolsV1Marker>
+ DataProvider<JapaneseExtendedDateSymbolsV1Marker>
+ DataProvider<CopticDateSymbolsV1Marker>
+ DataProvider<IndianDateSymbolsV1Marker>
+ DataProvider<EthiopianDateSymbolsV1Marker>
+ DataProvider<JapaneseErasV1Marker>
+ DataProvider<JapaneseExtendedErasV1Marker>
+ ?Sized,
{
let calendar = AnyCalendar::try_new_for_locale_unstable(data_provider, locale)?;
let kind = calendar.kind();
Ok(Self(
raw::DateFormatter::try_new(
data_provider,
calendar::load_lengths_for_any_calendar_kind(data_provider, locale, kind)?,
|| calendar::load_symbols_for_any_calendar_kind(data_provider, locale, kind),
locale,
length,
)?,
calendar,
))
}
#[inline]
pub fn format<'l, T>(&'l self, value: &T) -> Result<FormattedDateTime<'l>, DateTimeError>
where
T: DateInput<Calendar = AnyCalendar>,
{
if let Some(converted) = self.convert_if_necessary(value)? {
Ok(self.0.format(&converted))
} else {
Ok(self.0.format(value))
}
}
#[inline]
pub fn format_to_string(
&self,
value: &impl DateInput<Calendar = AnyCalendar>,
) -> Result<String, DateTimeError> {
Ok(self.format(value)?.write_to_string().into_owned())
}
fn convert_if_necessary<'a>(
&'a self,
value: &impl DateInput<Calendar = AnyCalendar>,
) -> Result<Option<Date<icu_calendar::Ref<'a, AnyCalendar>>>, DateTimeError> {
let this_calendar = self.1.kind();
let date_calendar = value.any_calendar_kind();
if Some(this_calendar) != date_calendar {
if date_calendar != Some(AnyCalendarKind::Iso) {
return Err(DateTimeError::MismatchedAnyCalendar(
this_calendar,
date_calendar,
));
}
let date = value.to_iso().to_any();
let converted = self.1.convert_any_date(&date);
Ok(Some(converted))
} else {
Ok(None)
}
}
}