| 
									
										
										
										
											2021-08-08 18:27:28 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2021, Linus Groh <linusg@serenityos.org> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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>
 | 
					
						
							| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2022-01-30 17:36:23 -05:00
										 |  |  | #include <LibJS/Runtime/Temporal/TimeZone.h>
 | 
					
						
							|  |  |  | #include <LibUnicode/CurrencyCode.h>
 | 
					
						
							|  |  |  | #include <LibUnicode/DateTimeFormat.h>
 | 
					
						
							|  |  |  | #include <LibUnicode/Locale.h>
 | 
					
						
							|  |  |  | #include <LibUnicode/NumberFormat.h>
 | 
					
						
							| 
									
										
										
										
											2021-08-08 18:27:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS::Intl { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // 8 The Intl Object, https://tc39.es/ecma402/#intl-object
 | 
					
						
							|  |  |  | Intl::Intl(GlobalObject& global_object) | 
					
						
							|  |  |  |     : Object(*global_object.object_prototype()) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Intl::initialize(GlobalObject& global_object) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Object::initialize(global_object); | 
					
						
							| 
									
										
										
										
											2021-08-08 18:39:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto& vm = this->vm(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 8.1.1 Intl[ @@toStringTag ], https://tc39.es/ecma402/#sec-Intl-toStringTag
 | 
					
						
							|  |  |  |     define_direct_property(*vm.well_known_symbol_to_string_tag(), js_string(vm, "Intl"), Attribute::Configurable); | 
					
						
							| 
									
										
										
										
											2021-08-24 22:27:05 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     u8 attr = Attribute::Writable | Attribute::Configurable; | 
					
						
							| 
									
										
										
										
											2021-09-06 22:18:48 -04:00
										 |  |  |     define_direct_property(vm.names.Collator, global_object.intl_collator_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2021-11-18 10:30:31 -05:00
										 |  |  |     define_direct_property(vm.names.DateTimeFormat, global_object.intl_date_time_format_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2021-08-24 22:27:05 -04:00
										 |  |  |     define_direct_property(vm.names.DisplayNames, global_object.intl_display_names_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2021-09-05 23:05:16 -04:00
										 |  |  |     define_direct_property(vm.names.ListFormat, global_object.intl_list_format_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2021-09-02 08:32:43 -04:00
										 |  |  |     define_direct_property(vm.names.Locale, global_object.intl_locale_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2021-09-08 21:34:27 -04:00
										 |  |  |     define_direct_property(vm.names.NumberFormat, global_object.intl_number_format_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2022-01-28 12:56:04 -05:00
										 |  |  |     define_direct_property(vm.names.PluralRules, global_object.intl_plural_rules_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2022-01-25 10:41:57 -05:00
										 |  |  |     define_direct_property(vm.names.RelativeTimeFormat, global_object.intl_relative_time_format_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2022-01-29 23:47:29 +02:00
										 |  |  |     define_direct_property(vm.names.Segmenter, global_object.intl_segmenter_constructor(), attr); | 
					
						
							| 
									
										
										
										
											2021-08-25 09:41:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-22 22:45:54 +01:00
										 |  |  |     define_native_function(vm.names.getCanonicalLocales, get_canonical_locales, 1, attr); | 
					
						
							| 
									
										
										
										
											2022-01-30 17:36:23 -05:00
										 |  |  |     define_native_function(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
										 |  |  | { | 
					
						
							|  |  |  |     auto locales = vm.argument(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 1. Let ll be ? CanonicalizeLocaleList(locales).
 | 
					
						
							| 
									
										
										
										
											2021-10-22 22:45:54 +01:00
										 |  |  |     auto locale_list = TRY(canonicalize_locale_list(global_object, locales)); | 
					
						
							| 
									
										
										
										
											2021-08-25 09:41:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 10:06:40 +00:00
										 |  |  |     MarkedVector<Value> marked_locale_list { vm.heap() }; | 
					
						
							| 
									
										
										
										
											2021-08-25 09:41:25 -04:00
										 |  |  |     marked_locale_list.ensure_capacity(locale_list.size()); | 
					
						
							|  |  |  |     for (auto& locale : locale_list) | 
					
						
							|  |  |  |         marked_locale_list.append(js_string(vm, move(locale))); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-28 08:30:48 -04:00
										 |  |  |     // 2. Return ! CreateArrayFromList(ll).
 | 
					
						
							| 
									
										
										
										
											2021-08-25 09:41:25 -04:00
										 |  |  |     return Array::create_from(global_object, marked_locale_list); | 
					
						
							| 
									
										
										
										
											2021-08-08 18:27:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-30 17:36:23 -05:00
										 |  |  | // 1.4.4 AvailableTimeZones (), https://tc39.es/proposal-intl-enumeration/#sec-availablecurrencies
 | 
					
						
							|  |  |  | static Vector<StringView> available_time_zones() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // 1. Let names be a List of all supported Zone and Link names in the IANA Time Zone Database.
 | 
					
						
							|  |  |  |     auto names = TimeZone::all_time_zones(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. Let result be a new empty List.
 | 
					
						
							|  |  |  |     Vector<StringView> result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 3. For each element name of names, do
 | 
					
						
							|  |  |  |     for (auto name : names) { | 
					
						
							|  |  |  |         // a. Assert: ! IsValidTimeZoneName( name ) is true.
 | 
					
						
							|  |  |  |         // b. Let canonical be ! CanonicalizeTimeZoneName( name ).
 | 
					
						
							|  |  |  |         auto canonical = TimeZone::canonicalize_time_zone(name).value(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // c. If result does not contain an element equal to canonical, then
 | 
					
						
							|  |  |  |         if (!result.contains_slow(canonical)) { | 
					
						
							|  |  |  |             // i. Append canonical to the end of result.
 | 
					
						
							|  |  |  |             result.append(canonical); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 4. Sort result in order as if an Array of the same values had been sorted using %Array.prototype.sort% using undefined as comparefn.
 | 
					
						
							|  |  |  |     quick_sort(result); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 5. Return result.
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // 2.2.2 Intl.supportedValuesOf ( key ), https://tc39.es/proposal-intl-enumeration/#sec-intl.supportedvaluesof
 | 
					
						
							|  |  |  | JS_DEFINE_NATIVE_FUNCTION(Intl::supported_values_of) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // 1. Let key be ? ToString(key).
 | 
					
						
							|  |  |  |     auto key = TRY(vm.argument(0).to_string(global_object)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Span<StringView const> list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 2. If key is "calendar", then
 | 
					
						
							|  |  |  |     if (key == "calendar"sv) { | 
					
						
							|  |  |  |         // a. Let list be ! AvailableCalendars( ).
 | 
					
						
							|  |  |  |         list = Unicode::get_available_calendars(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // 3. Else if key is "collation", then
 | 
					
						
							|  |  |  |     else if (key == "collation"sv) { | 
					
						
							|  |  |  |         // a. Let list be ! AvailableCollations( ).
 | 
					
						
							|  |  |  |         // NOTE: We don't yet parse any collation data, but "default" is allowed.
 | 
					
						
							|  |  |  |         static constexpr auto collations = AK::Array { "default"sv }; | 
					
						
							|  |  |  |         list = collations.span(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // 4. Else if key is "currency", then
 | 
					
						
							|  |  |  |     else if (key == "currency"sv) { | 
					
						
							|  |  |  |         // a. Let list be ! AvailableCurrencies( ).
 | 
					
						
							|  |  |  |         list = Unicode::get_available_currencies(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // 5. Else if key is "numberingSystem", then
 | 
					
						
							|  |  |  |     else if (key == "numberingSystem"sv) { | 
					
						
							|  |  |  |         // a. Let list be ! AvailableNumberingSystems( ).
 | 
					
						
							|  |  |  |         list = Unicode::get_available_number_systems(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // 6. Else if key is "timeZone", then
 | 
					
						
							|  |  |  |     else if (key == "timeZone"sv) { | 
					
						
							|  |  |  |         // a. Let list be ! AvailableTimeZones( ).
 | 
					
						
							|  |  |  |         static auto time_zones = available_time_zones(); | 
					
						
							|  |  |  |         list = time_zones.span(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // 7. Else if key is "unit", then
 | 
					
						
							|  |  |  |     else if (key == "unit"sv) { | 
					
						
							|  |  |  |         // a. Let list be ! AvailableUnits( ).
 | 
					
						
							|  |  |  |         static auto units = sanctioned_simple_unit_identifiers(); | 
					
						
							|  |  |  |         list = units.span(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // 8. Else,
 | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         // a. Throw a RangeError exception.
 | 
					
						
							|  |  |  |         return vm.throw_completion<RangeError>(global_object, ErrorType::IntlInvalidKey, key); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // 9. Return ! CreateArrayFromList( list ).
 | 
					
						
							|  |  |  |     return Array::create_from<StringView>(global_object, list, [&](auto value) { return js_string(vm, value); }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-08 18:27:28 +01:00
										 |  |  | } |