ladybird/Userland/Libraries/LibUnicode/DisplayNames.cpp
Timothy Flynn 672a555f98 LibCore+LibJS+LibUnicode: Port retrieving time zone offsets to ICU
The changes to tests are due to LibTimeZone incorrectly interpreting
time stamps in the TZDB. The TZDB will list zone transitions in either
UTC or the zone's local time (which is then subject to DST offsets).
LibTimeZone did not handle the latter at all.

For example:

The following rule is in effect until November 18, 6PM UTC.

    America/Chicago -5:50:36 - LMT 1883 Nov 18 18:00u

The following rule is in effect until March 1, 2AM in Chicago time. But
at that time, a DST transition occurs, so the local time is actually
3AM.

    America/Chicago -6:00 Chicago C%sT 1936 Mar 1 2:00
2024-06-26 10:14:02 +02:00

260 lines
7.3 KiB
C++

/*
* Copyright (c) 2024, Tim Flynn <trflynn89@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#define AK_DONT_REPLACE_STD
#include <AK/Array.h>
#include <LibUnicode/DisplayNames.h>
#include <LibUnicode/ICU.h>
#include <unicode/dtptngen.h>
#include <unicode/localebuilder.h>
#include <unicode/locdspnm.h>
#include <unicode/tznames.h>
#include <unicode/udatpg.h>
namespace Unicode {
LanguageDisplay language_display_from_string(StringView language_display)
{
if (language_display == "standard"sv)
return LanguageDisplay::Standard;
if (language_display == "dialect"sv)
return LanguageDisplay::Dialect;
VERIFY_NOT_REACHED();
}
StringView language_display_to_string(LanguageDisplay language_display)
{
switch (language_display) {
case LanguageDisplay::Standard:
return "standard"sv;
case LanguageDisplay::Dialect:
return "dialect"sv;
default:
VERIFY_NOT_REACHED();
}
}
Optional<String> language_display_name(StringView locale, StringView language, LanguageDisplay display)
{
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
auto language_data = LocaleData::for_locale(language);
if (!language_data.has_value())
return {};
auto& display_names = display == LanguageDisplay::Standard
? locale_data->standard_display_names()
: locale_data->dialect_display_names();
icu::UnicodeString result;
display_names.localeDisplayName(language_data->locale().getName(), result);
return icu_string_to_string(result);
}
Optional<String> region_display_name(StringView locale, StringView region)
{
UErrorCode status = U_ZERO_ERROR;
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
auto icu_region = icu::LocaleBuilder().setRegion(icu_string_piece(region)).build(status);
if (icu_failure(status))
return {};
icu::UnicodeString result;
locale_data->standard_display_names().regionDisplayName(icu_region.getCountry(), result);
return icu_string_to_string(result);
}
Optional<String> script_display_name(StringView locale, StringView script)
{
UErrorCode status = U_ZERO_ERROR;
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
auto icu_script = icu::LocaleBuilder().setScript(icu_string_piece(script)).build(status);
if (icu_failure(status))
return {};
icu::UnicodeString result;
locale_data->standard_display_names().scriptDisplayName(icu_script.getScript(), result);
return icu_string_to_string(result);
}
Optional<String> calendar_display_name(StringView locale, StringView calendar)
{
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
if (calendar == "gregory"sv)
calendar = "gregorian"sv;
if (calendar == "islamicc"sv)
calendar = "islamic-civil"sv;
if (calendar == "ethioaa"sv)
calendar = "ethiopic-amete-alem"sv;
icu::UnicodeString result;
locale_data->standard_display_names().keyValueDisplayName("calendar", ByteString(calendar).characters(), result);
return icu_string_to_string(result);
}
static constexpr UDateTimePatternField icu_date_time_field(StringView field)
{
if (field == "day"sv)
return UDATPG_DAY_FIELD;
if (field == "dayPeriod"sv)
return UDATPG_DAYPERIOD_FIELD;
if (field == "era"sv)
return UDATPG_ERA_FIELD;
if (field == "hour"sv)
return UDATPG_HOUR_FIELD;
if (field == "minute"sv)
return UDATPG_MINUTE_FIELD;
if (field == "month"sv)
return UDATPG_MONTH_FIELD;
if (field == "quarter"sv)
return UDATPG_QUARTER_FIELD;
if (field == "second"sv)
return UDATPG_SECOND_FIELD;
if (field == "timeZoneName"sv)
return UDATPG_ZONE_FIELD;
if (field == "weekOfYear"sv)
return UDATPG_WEEK_OF_YEAR_FIELD;
if (field == "weekday"sv)
return UDATPG_WEEKDAY_FIELD;
if (field == "year"sv)
return UDATPG_YEAR_FIELD;
VERIFY_NOT_REACHED();
}
static constexpr UDateTimePGDisplayWidth icu_date_time_style(Style style)
{
switch (style) {
case Style::Long:
return UDATPG_WIDE;
case Style::Short:
return UDATPG_ABBREVIATED;
case Style::Narrow:
return UDATPG_NARROW;
}
VERIFY_NOT_REACHED();
}
Optional<String> date_time_field_display_name(StringView locale, StringView field, Style style)
{
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
auto icu_field = icu_date_time_field(field);
auto icu_style = icu_date_time_style(style);
icu::UnicodeString result;
result = locale_data->date_time_pattern_generator().getFieldDisplayName(icu_field, icu_style);
return icu_string_to_string(result);
}
Optional<String> time_zone_display_name(StringView locale, StringView time_zone_identifier, TimeZoneOffset::InDST in_dst, double time)
{
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
icu::UnicodeString time_zone_name;
auto type = in_dst == TimeZoneOffset::InDST::Yes ? UTZNM_LONG_DAYLIGHT : UTZNM_LONG_STANDARD;
locale_data->time_zone_names().getDisplayName(icu_string(time_zone_identifier), type, time, time_zone_name);
if (static_cast<bool>(time_zone_name.isBogus()))
return {};
return icu_string_to_string(time_zone_name);
}
static constexpr Array<UChar, 4> icu_currency_code(StringView currency)
{
VERIFY(currency.length() == 3);
return to_array({
static_cast<UChar>(currency[0]),
static_cast<UChar>(currency[1]),
static_cast<UChar>(currency[2]),
u'\0',
});
}
static constexpr UCurrNameStyle icu_currency_style(Style style)
{
switch (style) {
case Style::Long:
return UCURR_LONG_NAME;
case Style::Short:
return UCURR_SYMBOL_NAME;
case Style::Narrow:
return UCURR_NARROW_SYMBOL_NAME;
}
VERIFY_NOT_REACHED();
}
Optional<String> currency_display_name(StringView locale, StringView currency, Style style)
{
UErrorCode status = U_ZERO_ERROR;
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
auto icu_currency = icu_currency_code(currency);
i32 length = 0;
UChar const* result = ucurr_getName(icu_currency.data(), locale_data->locale().getName(), icu_currency_style(style), nullptr, &length, &status);
if (icu_failure(status))
return {};
if ((status == U_USING_DEFAULT_WARNING) && (result == icu_currency.data()))
return {};
return icu_string_to_string(result, length);
}
Optional<String> currency_numeric_display_name(StringView locale, StringView currency)
{
UErrorCode status = U_ZERO_ERROR;
auto locale_data = LocaleData::for_locale(locale);
if (!locale_data.has_value())
return {};
auto icu_currency = icu_currency_code(currency);
i32 length = 0;
UChar const* result = ucurr_getPluralName(icu_currency.data(), locale_data->locale().getName(), nullptr, "other", &length, &status);
if (icu_failure(status))
return {};
if ((status == U_USING_DEFAULT_WARNING) && (result == icu_currency.data()))
return {};
return icu_string_to_string(result, length);
}
}