1
0
Fork 0
mirror of https://github.com/LadybirdBrowser/ladybird.git synced 2025-06-11 02:13:56 +09:00
ladybird/Userland/Libraries/LibUnicode/DisplayNames.cpp
Timothy Flynn ebdb92eef6 LibUnicode+Everywhere: Merge LibLocale back into LibUnicode
LibLocale was split off from LibUnicode a couple years ago to reduce the
number of applications on SerenityOS that depend on CLDR data. Now that
we use ICU, both LibUnicode and LibLocale are actually linking in this
data. And since vcpkg gives us static libraries, both libraries are over
30MB in size.

This patch reverts the separation and merges LibLocale into LibUnicode
again. We now have just one library that includes the ICU data.

Further, this will let LibUnicode share the locale cache that previously
would only exist in LibLocale.
2024-06-23 19:52:45 +02:00

259 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, TimeZone::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 == TimeZone::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);
}
}