use crate::{calendar, options::DateTimeFormatterOptions, raw};
use alloc::string::String;
use icu_provider::prelude::*;
use crate::input::{DateTimeInput, ExtractedDateTimeInput, TimeZoneInput};
use crate::provider::{self, calendar::*, date_time::PatternSelector};
use crate::time_zone::TimeZoneFormatterOptions;
use crate::{DateTimeError, FormattedZonedDateTime};
use icu_calendar::any_calendar::{AnyCalendar, AnyCalendarKind};
use icu_calendar::provider::{
JapaneseErasV1Marker, JapaneseExtendedErasV1Marker, WeekDataV1Marker,
};
use icu_calendar::{types::Time, DateTime};
use icu_decimal::provider::DecimalSymbolsV1Marker;
use icu_plurals::provider::OrdinalV1Marker;
use writeable::Writeable;
pub struct ZonedDateTimeFormatter(raw::ZonedDateTimeFormatter, AnyCalendar);
impl ZonedDateTimeFormatter {
#[cfg(feature = "experimental")]
#[inline]
#[allow(clippy::too_many_arguments)]
pub fn try_new_experimental_unstable<P>(
provider: &P,
locale: &DataLocale,
date_time_format_options: DateTimeFormatterOptions,
time_zone_format_options: TimeZoneFormatterOptions,
) -> Result<Self, DateTimeError>
where
P: DataProvider<TimeSymbolsV1Marker>
+ DataProvider<TimeLengthsV1Marker>
+ DataProvider<crate::provider::calendar::DateSkeletonPatternsV1Marker>
+ DataProvider<WeekDataV1Marker>
+ DataProvider<provider::time_zones::TimeZoneFormatsV1Marker>
+ DataProvider<provider::time_zones::ExemplarCitiesV1Marker>
+ DataProvider<provider::time_zones::MetazoneGenericNamesLongV1Marker>
+ DataProvider<provider::time_zones::MetazoneGenericNamesShortV1Marker>
+ DataProvider<provider::time_zones::MetazoneSpecificNamesLongV1Marker>
+ DataProvider<provider::time_zones::MetazoneSpecificNamesShortV1Marker>
+ DataProvider<OrdinalV1Marker>
+ 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(provider, locale)?;
let kind = calendar.kind();
let patterns = PatternSelector::for_options_experimental(
provider,
calendar::load_lengths_for_any_calendar_kind(provider, locale, kind)?,
locale,
&kind.as_bcp47_value(),
&date_time_format_options,
)?;
Ok(Self(
raw::ZonedDateTimeFormatter::try_new(
provider,
patterns,
|| calendar::load_symbols_for_any_calendar_kind(provider, locale, kind),
locale,
time_zone_format_options,
)?,
calendar,
))
}
#[inline]
#[allow(clippy::too_many_arguments)]
pub fn try_new_unstable<P>(
provider: &P,
locale: &DataLocale,
date_time_format_options: DateTimeFormatterOptions,
time_zone_format_options: TimeZoneFormatterOptions,
) -> Result<Self, DateTimeError>
where
P: DataProvider<TimeSymbolsV1Marker>
+ DataProvider<TimeLengthsV1Marker>
+ DataProvider<WeekDataV1Marker>
+ DataProvider<provider::time_zones::TimeZoneFormatsV1Marker>
+ DataProvider<provider::time_zones::ExemplarCitiesV1Marker>
+ DataProvider<provider::time_zones::MetazoneGenericNamesLongV1Marker>
+ DataProvider<provider::time_zones::MetazoneGenericNamesShortV1Marker>
+ DataProvider<provider::time_zones::MetazoneSpecificNamesLongV1Marker>
+ DataProvider<provider::time_zones::MetazoneSpecificNamesShortV1Marker>
+ DataProvider<OrdinalV1Marker>
+ 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(provider, locale)?;
let kind = calendar.kind();
let patterns = PatternSelector::for_options(
provider,
calendar::load_lengths_for_any_calendar_kind(provider, locale, kind)?,
locale,
&date_time_format_options,
)?;
Ok(Self(
raw::ZonedDateTimeFormatter::try_new(
provider,
patterns,
|| calendar::load_symbols_for_any_calendar_kind(provider, locale, kind),
locale,
time_zone_format_options,
)?,
calendar,
))
}
#[inline]
pub fn try_new_with_any_provider<P>(
data_provider: &P,
locale: &DataLocale,
options: DateTimeFormatterOptions,
time_zone_format_options: TimeZoneFormatterOptions,
) -> Result<Self, DateTimeError>
where
P: AnyProvider,
{
let downcasting = data_provider.as_downcasting();
Self::try_new_unstable(&downcasting, locale, options, time_zone_format_options)
}
#[inline]
#[cfg(feature = "serde")]
pub fn try_new_with_buffer_provider<P>(
data_provider: &P,
locale: &DataLocale,
options: DateTimeFormatterOptions,
time_zone_format_options: TimeZoneFormatterOptions,
) -> Result<Self, DateTimeError>
where
P: BufferProvider,
{
let deserializing = data_provider.as_deserializing();
Self::try_new_unstable(&deserializing, locale, options, time_zone_format_options)
}
#[inline]
pub fn format<'l>(
&'l self,
date: &impl DateTimeInput<Calendar = AnyCalendar>,
time_zone: &impl TimeZoneInput,
) -> Result<FormattedZonedDateTime<'l>, DateTimeError> {
if let Some(converted) = self.convert_if_necessary(date)? {
Ok(self.0.format(&converted, time_zone))
} else {
Ok(self.0.format(date, time_zone))
}
}
#[inline]
pub fn format_to_string(
&self,
date: &impl DateTimeInput<Calendar = AnyCalendar>,
time_zone: &impl TimeZoneInput,
) -> Result<String, DateTimeError> {
Ok(self.format(date, time_zone)?.write_to_string().into_owned())
}
fn convert_if_necessary(
&self,
value: &impl DateTimeInput<Calendar = AnyCalendar>,
) -> Result<Option<ExtractedDateTimeInput>, 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();
let time = Time::new(
value.hour().unwrap_or_default(),
value.minute().unwrap_or_default(),
value.second().unwrap_or_default(),
value.nanosecond().unwrap_or_default(),
);
let datetime = DateTime::new(date, time).to_any();
let converted = self.1.convert_any_datetime(&datetime);
let converted = ExtractedDateTimeInput::extract_from(&converted);
Ok(Some(converted))
} else {
Ok(None)
}
}
}