/* * Copyright (c) 2021-2022, Tim Flynn * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include namespace Unicode { HourCycle hour_cycle_from_string(StringView hour_cycle) { if (hour_cycle == "h11"sv) return Unicode::HourCycle::H11; else if (hour_cycle == "h12"sv) return Unicode::HourCycle::H12; else if (hour_cycle == "h23"sv) return Unicode::HourCycle::H23; else if (hour_cycle == "h24"sv) return Unicode::HourCycle::H24; VERIFY_NOT_REACHED(); } StringView hour_cycle_to_string(HourCycle hour_cycle) { switch (hour_cycle) { case HourCycle::H11: return "h11"sv; case HourCycle::H12: return "h12"sv; case HourCycle::H23: return "h23"sv; case HourCycle::H24: return "h24"sv; default: VERIFY_NOT_REACHED(); } } CalendarPatternStyle calendar_pattern_style_from_string(StringView style) { if (style == "narrow"sv) return CalendarPatternStyle::Narrow; if (style == "short"sv) return CalendarPatternStyle::Short; if (style == "long"sv) return CalendarPatternStyle::Long; if (style == "numeric"sv) return CalendarPatternStyle::Numeric; if (style == "2-digit"sv) return CalendarPatternStyle::TwoDigit; if (style == "shortOffset"sv) return CalendarPatternStyle::ShortOffset; if (style == "longOffset"sv) return CalendarPatternStyle::LongOffset; if (style == "shortGeneric"sv) return CalendarPatternStyle::ShortGeneric; if (style == "longGeneric"sv) return CalendarPatternStyle::LongGeneric; VERIFY_NOT_REACHED(); } StringView calendar_pattern_style_to_string(CalendarPatternStyle style) { switch (style) { case CalendarPatternStyle::Narrow: return "narrow"sv; case CalendarPatternStyle::Short: return "short"sv; case CalendarPatternStyle::Long: return "long"sv; case CalendarPatternStyle::Numeric: return "numeric"sv; case CalendarPatternStyle::TwoDigit: return "2-digit"sv; case CalendarPatternStyle::ShortOffset: return "shortOffset"sv; case CalendarPatternStyle::LongOffset: return "longOffset"sv; case CalendarPatternStyle::ShortGeneric: return "shortGeneric"sv; case CalendarPatternStyle::LongGeneric: return "longGeneric"sv; default: VERIFY_NOT_REACHED(); } } Optional __attribute__((weak)) calendar_from_string(StringView) { return {}; } Optional __attribute__((weak)) hour_cycle_region_from_string(StringView) { return {}; } Span __attribute__((weak)) get_available_calendars() { return {}; } Vector __attribute__((weak)) get_regional_hour_cycles(StringView) { return {}; } // https://unicode.org/reports/tr35/tr35-dates.html#Date_Field_Symbol_Table Vector get_locale_hour_cycles(StringView locale) { if (auto hour_cycles = get_regional_hour_cycles(locale); !hour_cycles.is_empty()) return hour_cycles; auto return_default_hour_cycles = [&]() { return get_regional_hour_cycles("001"sv); }; auto language = parse_unicode_language_id(locale); if (!language.has_value()) return return_default_hour_cycles(); if (!language->region.has_value()) language = add_likely_subtags(*language); if (!language.has_value() || !language->region.has_value()) return return_default_hour_cycles(); if (auto hour_cycles = get_regional_hour_cycles(*language->region); !hour_cycles.is_empty()) return hour_cycles; return return_default_hour_cycles(); } Optional get_default_regional_hour_cycle(StringView locale) { if (auto hour_cycles = get_locale_hour_cycles(locale); !hour_cycles.is_empty()) return hour_cycles.first(); return {}; } String combine_skeletons(StringView first, StringView second) { // https://unicode.org/reports/tr35/tr35-dates.html#availableFormats_appendItems constexpr auto field_order = Array { "G"sv, // Era "yYuUr"sv, // Year "ML"sv, // Month "dDFg"sv, // Day "Eec"sv, // Weekday "abB"sv, // Period "hHKk"sv, // Hour "m"sv, // Minute "sSA"sv, // Second "zZOvVXx"sv, // Zone }; StringBuilder builder; auto append_from_skeleton = [&](auto skeleton, auto ch) { auto first_index = skeleton.find(ch); if (!first_index.has_value()) return false; auto last_index = skeleton.find_last(ch); builder.append(skeleton.substring_view(*first_index, *last_index - *first_index + 1)); return true; }; for (auto fields : field_order) { for (auto ch : fields) { if (append_from_skeleton(first, ch)) break; if (append_from_skeleton(second, ch)) break; } } return builder.build(); } Optional __attribute__((weak)) get_calendar_date_format(StringView, StringView) { return {}; } Optional __attribute__((weak)) get_calendar_time_format(StringView, StringView) { return {}; } Optional __attribute__((weak)) get_calendar_date_time_format(StringView, StringView) { return {}; } Optional get_calendar_format(StringView locale, StringView calendar, CalendarFormatType type) { switch (type) { case CalendarFormatType::Date: return get_calendar_date_format(locale, calendar); case CalendarFormatType::Time: return get_calendar_time_format(locale, calendar); case CalendarFormatType::DateTime: return get_calendar_date_time_format(locale, calendar); default: VERIFY_NOT_REACHED(); } } Vector __attribute__((weak)) get_calendar_available_formats(StringView, StringView) { return {}; } Optional __attribute__((weak)) get_calendar_default_range_format(StringView, StringView) { return {}; } Vector __attribute__((weak)) get_calendar_range_formats(StringView, StringView, StringView) { return {}; } Vector __attribute__((weak)) get_calendar_range12_formats(StringView, StringView, StringView) { return {}; } Optional __attribute__((weak)) get_calendar_era_symbol(StringView, StringView, CalendarPatternStyle, Era) { return {}; } Optional __attribute__((weak)) get_calendar_month_symbol(StringView, StringView, CalendarPatternStyle, Month) { return {}; } Optional __attribute__((weak)) get_calendar_weekday_symbol(StringView, StringView, CalendarPatternStyle, Weekday) { return {}; } Optional __attribute__((weak)) get_calendar_day_period_symbol(StringView, StringView, CalendarPatternStyle, DayPeriod) { return {}; } Optional __attribute__((weak)) get_calendar_day_period_symbol_for_hour(StringView, StringView, CalendarPatternStyle, u8) { return {}; } Optional __attribute__((weak)) get_time_zone_name(StringView, StringView, CalendarPatternStyle, TimeZone::InDST) { return {}; } Optional __attribute__((weak)) get_time_zone_format(StringView) { return {}; } static Optional format_time_zone_offset(StringView locale, CalendarPatternStyle style, i64 offset_seconds) { auto formats = get_time_zone_format(locale); if (!formats.has_value()) return {}; auto number_system = get_default_number_system(locale); if (!number_system.has_value()) return {}; if (offset_seconds == 0) return formats->gmt_zero_format; auto sign = offset_seconds > 0 ? formats->symbol_ahead_sign : formats->symbol_behind_sign; auto separator = offset_seconds > 0 ? formats->symbol_ahead_separator : formats->symbol_behind_separator; offset_seconds = llabs(offset_seconds); auto offset_hours = offset_seconds / 3'600; offset_seconds %= 3'600; auto offset_minutes = offset_seconds / 60; offset_seconds %= 60; StringBuilder builder; builder.append(sign); switch (style) { // The long format always uses 2-digit hours field and minutes field, with optional 2-digit seconds field. case CalendarPatternStyle::LongOffset: builder.appendff("{:02}{}{:02}", offset_hours, separator, offset_minutes); if (offset_seconds > 0) builder.appendff("{}{:02}", separator, offset_seconds); break; // The short format is intended for the shortest representation and uses hour fields without leading zero, with optional 2-digit minutes and seconds fields. case CalendarPatternStyle::ShortOffset: builder.appendff("{}", offset_hours); if (offset_minutes > 0) { builder.appendff("{}{:02}", separator, offset_minutes); if (offset_seconds > 0) builder.appendff("{}{:02}", separator, offset_seconds); } break; default: VERIFY_NOT_REACHED(); } // The digits used for hours, minutes and seconds fields in this format are the locale's default decimal digits. auto result = replace_digits_for_number_system(*number_system, builder.build()); return formats->gmt_format.replace("{0}"sv, result); } // https://unicode.org/reports/tr35/tr35-dates.html#Time_Zone_Format_Terminology String format_time_zone(StringView locale, StringView time_zone, CalendarPatternStyle style, AK::Time time) { auto offset = TimeZone::get_time_zone_offset(time_zone, time); if (!offset.has_value()) return time_zone; switch (style) { case CalendarPatternStyle::Short: case CalendarPatternStyle::Long: case CalendarPatternStyle::ShortGeneric: case CalendarPatternStyle::LongGeneric: if (auto name = get_time_zone_name(locale, time_zone, style, offset->in_dst); name.has_value()) return *name; break; case CalendarPatternStyle::ShortOffset: case CalendarPatternStyle::LongOffset: return format_time_zone_offset(locale, style, offset->seconds).value_or(time_zone); default: VERIFY_NOT_REACHED(); } // If more styles are added, consult the following table to ensure always falling back to GMT offset is still correct: // https://unicode.org/reports/tr35/tr35-dates.html#dfst-zone switch (style) { case CalendarPatternStyle::Short: case CalendarPatternStyle::ShortGeneric: return format_time_zone(locale, time_zone, CalendarPatternStyle::ShortOffset, time); case CalendarPatternStyle::Long: case CalendarPatternStyle::LongGeneric: return format_time_zone(locale, time_zone, CalendarPatternStyle::LongOffset, time); default: VERIFY_NOT_REACHED(); } } }