2021-08-08 18:27:28 +01:00
|
|
|
/*
|
2023-04-13 00:47:15 +02:00
|
|
|
* Copyright (c) 2021-2023, Linus Groh <linusg@serenityos.org>
|
2024-06-25 11:06:08 -04:00
|
|
|
* Copyright (c) 2024, Tim Flynn <trflynn89@serenityos.org>
|
2021-08-08 18:27:28 +01:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2022-01-30 17:36:23 -05:00
|
|
|
#include <AK/QuickSort.h>
|
2021-08-25 09:41:25 -04:00
|
|
|
#include <LibJS/Runtime/Array.h>
|
2024-06-25 11:06:08 -04:00
|
|
|
#include <LibJS/Runtime/Date.h>
|
2021-08-08 18:27:28 +01:00
|
|
|
#include <LibJS/Runtime/GlobalObject.h>
|
2021-08-25 09:41:25 -04:00
|
|
|
#include <LibJS/Runtime/Intl/AbstractOperations.h>
|
2021-09-06 22:18:48 -04:00
|
|
|
#include <LibJS/Runtime/Intl/CollatorConstructor.h>
|
2021-11-18 10:30:31 -05:00
|
|
|
#include <LibJS/Runtime/Intl/DateTimeFormatConstructor.h>
|
2021-08-24 22:27:05 -04:00
|
|
|
#include <LibJS/Runtime/Intl/DisplayNamesConstructor.h>
|
2022-06-30 00:58:40 +03:00
|
|
|
#include <LibJS/Runtime/Intl/DurationFormatConstructor.h>
|
2021-08-08 18:27:28 +01:00
|
|
|
#include <LibJS/Runtime/Intl/Intl.h>
|
2021-09-05 23:05:16 -04:00
|
|
|
#include <LibJS/Runtime/Intl/ListFormatConstructor.h>
|
2021-09-02 08:32:43 -04:00
|
|
|
#include <LibJS/Runtime/Intl/LocaleConstructor.h>
|
2021-09-08 21:34:27 -04:00
|
|
|
#include <LibJS/Runtime/Intl/NumberFormatConstructor.h>
|
2022-01-28 12:56:04 -05:00
|
|
|
#include <LibJS/Runtime/Intl/PluralRulesConstructor.h>
|
2022-01-25 10:41:57 -05:00
|
|
|
#include <LibJS/Runtime/Intl/RelativeTimeFormatConstructor.h>
|
2022-01-29 23:47:29 +02:00
|
|
|
#include <LibJS/Runtime/Intl/SegmenterConstructor.h>
|
2024-06-17 16:24:58 -04:00
|
|
|
#include <LibJS/Runtime/Intl/SingleUnitIdentifiers.h>
|
2024-06-23 09:14:27 -04:00
|
|
|
#include <LibUnicode/DateTimeFormat.h>
|
|
|
|
#include <LibUnicode/Locale.h>
|
|
|
|
#include <LibUnicode/NumberFormat.h>
|
|
|
|
#include <LibUnicode/UnicodeKeywords.h>
|
2021-08-08 18:27:28 +01:00
|
|
|
|
|
|
|
namespace JS::Intl {
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC_DEFINE_ALLOCATOR(Intl);
|
2023-11-19 09:45:05 +01:00
|
|
|
|
2021-08-08 18:27:28 +01:00
|
|
|
// 8 The Intl Object, https://tc39.es/ecma402/#intl-object
|
2022-08-16 00:20:49 +01:00
|
|
|
Intl::Intl(Realm& realm)
|
2023-04-13 00:47:15 +02:00
|
|
|
: Object(ConstructWithPrototypeTag::Tag, realm.intrinsics().object_prototype())
|
2021-08-08 18:27:28 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-08-07 08:41:28 +02:00
|
|
|
void Intl::initialize(Realm& realm)
|
2021-08-08 18:27:28 +01:00
|
|
|
{
|
2023-08-07 08:41:28 +02:00
|
|
|
Base::initialize(realm);
|
2021-08-08 18:39:20 +01:00
|
|
|
|
|
|
|
auto& vm = this->vm();
|
|
|
|
|
|
|
|
// 8.1.1 Intl[ @@toStringTag ], https://tc39.es/ecma402/#sec-Intl-toStringTag
|
2023-08-08 18:25:57 +02:00
|
|
|
define_direct_property(vm.well_known_symbol_to_string_tag(), PrimitiveString::create(vm, "Intl"_string), Attribute::Configurable);
|
2021-08-24 22:27:05 -04:00
|
|
|
|
|
|
|
u8 attr = Attribute::Writable | Attribute::Configurable;
|
2022-11-24 09:43:53 -05:00
|
|
|
define_intrinsic_accessor(vm.names.Collator, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_collator_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.DateTimeFormat, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_date_time_format_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.DisplayNames, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_display_names_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.DurationFormat, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_duration_format_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.ListFormat, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_list_format_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.Locale, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_locale_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.NumberFormat, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_number_format_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.PluralRules, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_plural_rules_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.RelativeTimeFormat, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_relative_time_format_constructor(); });
|
|
|
|
define_intrinsic_accessor(vm.names.Segmenter, attr, [](auto& realm) -> Value { return realm.intrinsics().intl_segmenter_constructor(); });
|
2021-08-25 09:41:25 -04:00
|
|
|
|
2022-08-22 21:47:35 +01:00
|
|
|
define_native_function(realm, vm.names.getCanonicalLocales, get_canonical_locales, 1, attr);
|
|
|
|
define_native_function(realm, vm.names.supportedValuesOf, supported_values_of, 1, attr);
|
2021-08-25 09:41:25 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// 8.3.1 Intl.getCanonicalLocales ( locales ), https://tc39.es/ecma402/#sec-intl.getcanonicallocales
|
2021-10-22 22:45:54 +01:00
|
|
|
JS_DEFINE_NATIVE_FUNCTION(Intl::get_canonical_locales)
|
2021-08-25 09:41:25 -04:00
|
|
|
{
|
2022-08-22 11:48:08 +01:00
|
|
|
auto& realm = *vm.current_realm();
|
2022-08-16 00:20:49 +01:00
|
|
|
|
2021-08-25 09:41:25 -04:00
|
|
|
auto locales = vm.argument(0);
|
|
|
|
|
|
|
|
// 1. Let ll be ? CanonicalizeLocaleList(locales).
|
2022-08-20 08:25:24 +01:00
|
|
|
auto locale_list = TRY(canonicalize_locale_list(vm, locales));
|
2021-08-25 09:41:25 -04:00
|
|
|
|
2024-12-26 14:32:52 +01:00
|
|
|
GC::RootVector<Value> marked_locale_list { vm.heap() };
|
2023-08-30 11:36:23 -04:00
|
|
|
marked_locale_list.ensure_capacity(locale_list.size());
|
2023-02-02 19:46:33 -05:00
|
|
|
|
2021-08-25 09:41:25 -04:00
|
|
|
for (auto& locale : locale_list)
|
2023-02-02 19:46:33 -05:00
|
|
|
marked_locale_list.unchecked_append(PrimitiveString::create(vm, move(locale)));
|
2021-08-25 09:41:25 -04:00
|
|
|
|
2022-05-02 20:54:39 +02:00
|
|
|
// 2. Return CreateArrayFromList(ll).
|
2022-08-16 00:20:49 +01:00
|
|
|
return Array::create_from(realm, marked_locale_list);
|
2021-08-08 18:27:28 +01:00
|
|
|
}
|
|
|
|
|
2024-06-25 11:06:08 -04:00
|
|
|
// 6.5.4 AvailablePrimaryTimeZoneIdentifiers ( ), https://tc39.es/ecma402/#sec-availableprimarytimezoneidentifiers
|
|
|
|
static Vector<String> available_primary_time_zone_identifiers()
|
2022-01-30 17:36:23 -05:00
|
|
|
{
|
2024-06-25 11:06:08 -04:00
|
|
|
// 1. Let records be AvailableNamedTimeZoneIdentifiers().
|
|
|
|
auto const& records = available_named_time_zone_identifiers();
|
2022-01-30 17:36:23 -05:00
|
|
|
|
|
|
|
// 2. Let result be a new empty List.
|
2024-06-25 11:06:08 -04:00
|
|
|
Vector<String> result;
|
|
|
|
|
|
|
|
// 3. For each element timeZoneIdentifierRecord of records, do
|
|
|
|
for (auto const& time_zone_identifier_record : records) {
|
|
|
|
// a. If timeZoneIdentifierRecord.[[Identifier]] is timeZoneIdentifierRecord.[[PrimaryIdentifier]], then
|
|
|
|
if (time_zone_identifier_record.identifier == time_zone_identifier_record.primary_identifier) {
|
|
|
|
// i. Append timeZoneIdentifierRecord.[[Identifier]] to result.
|
|
|
|
result.append(time_zone_identifier_record.identifier);
|
2022-01-30 17:36:23 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-25 11:06:08 -04:00
|
|
|
// 4. Return result.
|
2022-01-30 17:36:23 -05:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-04-05 11:32:30 -04:00
|
|
|
// 8.3.2 Intl.supportedValuesOf ( key ), https://tc39.es/ecma402/#sec-intl.supportedvaluesof
|
2022-01-30 17:36:23 -05:00
|
|
|
JS_DEFINE_NATIVE_FUNCTION(Intl::supported_values_of)
|
|
|
|
{
|
2022-08-22 11:48:08 +01:00
|
|
|
auto& realm = *vm.current_realm();
|
2022-08-16 00:20:49 +01:00
|
|
|
|
2022-01-30 17:36:23 -05:00
|
|
|
// 1. Let key be ? ToString(key).
|
2023-01-15 10:31:39 -05:00
|
|
|
auto key = TRY(vm.argument(0).to_string(vm));
|
2022-01-30 17:36:23 -05:00
|
|
|
|
2024-06-08 11:22:05 -04:00
|
|
|
Optional<Variant<ReadonlySpan<StringView>, ReadonlySpan<String>>> list;
|
2022-01-30 17:36:23 -05:00
|
|
|
|
|
|
|
// 2. If key is "calendar", then
|
|
|
|
if (key == "calendar"sv) {
|
2022-11-03 11:23:30 -04:00
|
|
|
// a. Let list be ! AvailableCanonicalCalendars( ).
|
2024-06-23 09:14:27 -04:00
|
|
|
list = Unicode::available_calendars().span();
|
2022-01-30 17:36:23 -05:00
|
|
|
}
|
|
|
|
// 3. Else if key is "collation", then
|
|
|
|
else if (key == "collation"sv) {
|
2022-11-03 11:23:30 -04:00
|
|
|
// a. Let list be ! AvailableCanonicalCollations( ).
|
2024-06-23 09:14:27 -04:00
|
|
|
list = Unicode::available_collations().span();
|
2022-01-30 17:36:23 -05:00
|
|
|
}
|
|
|
|
// 4. Else if key is "currency", then
|
|
|
|
else if (key == "currency"sv) {
|
2022-11-03 11:23:30 -04:00
|
|
|
// a. Let list be ! AvailableCanonicalCurrencies( ).
|
2024-06-23 09:14:27 -04:00
|
|
|
list = Unicode::available_currencies().span();
|
2022-01-30 17:36:23 -05:00
|
|
|
}
|
|
|
|
// 5. Else if key is "numberingSystem", then
|
|
|
|
else if (key == "numberingSystem"sv) {
|
2022-11-03 11:23:30 -04:00
|
|
|
// a. Let list be ! AvailableCanonicalNumberingSystems( ).
|
2024-06-23 09:14:27 -04:00
|
|
|
list = Unicode::available_number_systems().span();
|
2022-01-30 17:36:23 -05:00
|
|
|
}
|
|
|
|
// 6. Else if key is "timeZone", then
|
|
|
|
else if (key == "timeZone"sv) {
|
2024-06-25 11:06:08 -04:00
|
|
|
// a. Let list be ! AvailablePrimaryTimeZoneIdentifiers( ).
|
|
|
|
static auto const time_zones = available_primary_time_zone_identifiers();
|
2022-01-30 17:36:23 -05:00
|
|
|
list = time_zones.span();
|
|
|
|
}
|
|
|
|
// 7. Else if key is "unit", then
|
|
|
|
else if (key == "unit"sv) {
|
2022-11-03 11:23:30 -04:00
|
|
|
// a. Let list be ! AvailableCanonicalUnits( ).
|
2024-06-08 11:22:05 -04:00
|
|
|
static auto const units = sanctioned_single_unit_identifiers();
|
2022-01-30 17:36:23 -05:00
|
|
|
list = units.span();
|
|
|
|
}
|
|
|
|
// 8. Else,
|
|
|
|
else {
|
|
|
|
// a. Throw a RangeError exception.
|
2022-08-16 20:33:17 +01:00
|
|
|
return vm.throw_completion<RangeError>(ErrorType::IntlInvalidKey, key);
|
2022-01-30 17:36:23 -05:00
|
|
|
}
|
|
|
|
|
2022-05-02 20:54:39 +02:00
|
|
|
// 9. Return CreateArrayFromList( list ).
|
2024-06-08 11:22:05 -04:00
|
|
|
return list->visit([&]<typename T>(ReadonlySpan<T> list) {
|
|
|
|
return Array::create_from<T>(realm, list, [&](auto value) {
|
|
|
|
return PrimitiveString::create(vm, value);
|
|
|
|
});
|
2023-08-08 18:25:57 +02:00
|
|
|
});
|
2022-01-30 17:36:23 -05:00
|
|
|
}
|
|
|
|
|
2021-08-08 18:27:28 +01:00
|
|
|
}
|