// Copyright 2018 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_INTL_SUPPORT #error Internationalization is expected to be enabled. #endif // V8_INTL_SUPPORT #include "src/objects/js-plural-rules.h" #include "src/execution/isolate-inl.h" #include "src/objects/intl-objects.h" #include "src/objects/js-number-format.h" #include "src/objects/js-plural-rules-inl.h" #include "unicode/locid.h" #include "unicode/numberformatter.h" #include "unicode/plurrule.h" #include "unicode/unumberformatter.h" namespace v8 { namespace internal { namespace { bool CreateICUPluralRules(Isolate* isolate, const icu::Locale& icu_locale, JSPluralRules::Type type, std::unique_ptr<icu::PluralRules>* pl) { // Make formatter from options. Numbering system is added // to the locale as Unicode extension (if it was specified at all). UErrorCode status = U_ZERO_ERROR; UPluralType icu_type = UPLURAL_TYPE_CARDINAL; if (type == JSPluralRules::Type::ORDINAL) { icu_type = UPLURAL_TYPE_ORDINAL; } else { DCHECK_EQ(JSPluralRules::Type::CARDINAL, type); } std::unique_ptr<icu::PluralRules> plural_rules( icu::PluralRules::forLocale(icu_locale, icu_type, status)); if (U_FAILURE(status)) { return false; } DCHECK_NOT_NULL(plural_rules.get()); *pl = std::move(plural_rules); return true; } } // namespace Handle<String> JSPluralRules::TypeAsString() const { switch (type()) { case Type::CARDINAL: return GetReadOnlyRoots().cardinal_string_handle(); case Type::ORDINAL: return GetReadOnlyRoots().ordinal_string_handle(); } UNREACHABLE(); } // static MaybeHandle<JSPluralRules> JSPluralRules::New(Isolate* isolate, Handle<Map> map, Handle<Object> locales, Handle<Object> options_obj) { // 1. Let requestedLocales be ? CanonicalizeLocaleList(locales). Maybe<std::vector<std::string>> maybe_requested_locales = Intl::CanonicalizeLocaleList(isolate, locales); MAYBE_RETURN(maybe_requested_locales, Handle<JSPluralRules>()); std::vector<std::string> requested_locales = maybe_requested_locales.FromJust(); // 2. If options is undefined, then if (options_obj->IsUndefined(isolate)) { // 2. a. Let options be ObjectCreate(null). options_obj = isolate->factory()->NewJSObjectWithNullProto(); } else { // 3. Else // 3. a. Let options be ? ToObject(options). ASSIGN_RETURN_ON_EXCEPTION( isolate, options_obj, Object::ToObject(isolate, options_obj, "Intl.PluralRules"), JSPluralRules); } // At this point, options_obj can either be a JSObject or a JSProxy only. Handle<JSReceiver> options = Handle<JSReceiver>::cast(options_obj); // 5. Let matcher be ? GetOption(options, "localeMatcher", "string", // « "lookup", "best fit" », "best fit"). // 6. Set opt.[[localeMatcher]] to matcher. Maybe<Intl::MatcherOption> maybe_locale_matcher = Intl::GetLocaleMatcher(isolate, options, "Intl.PluralRules"); MAYBE_RETURN(maybe_locale_matcher, MaybeHandle<JSPluralRules>()); Intl::MatcherOption matcher = maybe_locale_matcher.FromJust(); // 7. Let t be ? GetOption(options, "type", "string", « "cardinal", // "ordinal" », "cardinal"). Maybe<Type> maybe_type = Intl::GetStringOption<Type>( isolate, options, "type", "Intl.PluralRules", {"cardinal", "ordinal"}, {Type::CARDINAL, Type::ORDINAL}, Type::CARDINAL); MAYBE_RETURN(maybe_type, MaybeHandle<JSPluralRules>()); Type type = maybe_type.FromJust(); // Note: The spec says we should do ResolveLocale after performing // SetNumberFormatDigitOptions but we need the locale to create all // the ICU data structures. // // This isn't observable so we aren't violating the spec. // 11. Let r be ResolveLocale(%PluralRules%.[[AvailableLocales]], // requestedLocales, opt, %PluralRules%.[[RelevantExtensionKeys]], // localeData). Maybe<Intl::ResolvedLocale> maybe_resolve_locale = Intl::ResolveLocale(isolate, JSPluralRules::GetAvailableLocales(), requested_locales, matcher, {}); if (maybe_resolve_locale.IsNothing()) { THROW_NEW_ERROR(isolate, NewRangeError(MessageTemplate::kIcuError), JSPluralRules); } Intl::ResolvedLocale r = maybe_resolve_locale.FromJust(); Handle<String> locale_str = isolate->factory()->NewStringFromAsciiChecked(r.locale.c_str()); icu::number::LocalizedNumberFormatter icu_number_formatter = icu::number::NumberFormatter::withLocale(r.icu_locale) .roundingMode(UNUM_ROUND_HALFUP); std::unique_ptr<icu::PluralRules> icu_plural_rules; bool success = CreateICUPluralRules(isolate, r.icu_locale, type, &icu_plural_rules); if (!success || icu_plural_rules.get() == nullptr) { // Remove extensions and try again. icu::Locale no_extension_locale(r.icu_locale.getBaseName()); success = CreateICUPluralRules(isolate, no_extension_locale, type, &icu_plural_rules); icu_number_formatter = icu::number::NumberFormatter::withLocale(no_extension_locale) .roundingMode(UNUM_ROUND_HALFUP); if (!success || icu_plural_rules.get() == nullptr) { THROW_NEW_ERROR(isolate, NewRangeError(MessageTemplate::kIcuError), JSPluralRules); } } // 9. Perform ? SetNumberFormatDigitOptions(pluralRules, options, 0, 3). Maybe<Intl::NumberFormatDigitOptions> maybe_digit_options = Intl::SetNumberFormatDigitOptions(isolate, options, 0, 3, false); MAYBE_RETURN(maybe_digit_options, MaybeHandle<JSPluralRules>()); Intl::NumberFormatDigitOptions digit_options = maybe_digit_options.FromJust(); icu_number_formatter = JSNumberFormat::SetDigitOptionsToFormatter( icu_number_formatter, digit_options); Handle<Managed<icu::PluralRules>> managed_plural_rules = Managed<icu::PluralRules>::FromUniquePtr(isolate, 0, std::move(icu_plural_rules)); Handle<Managed<icu::number::LocalizedNumberFormatter>> managed_number_formatter = Managed<icu::number::LocalizedNumberFormatter>::FromRawPtr( isolate, 0, new icu::number::LocalizedNumberFormatter(icu_number_formatter)); // Now all properties are ready, so we can allocate the result object. Handle<JSPluralRules> plural_rules = Handle<JSPluralRules>::cast( isolate->factory()->NewFastOrSlowJSObjectFromMap(map)); DisallowGarbageCollection no_gc; plural_rules->set_flags(0); // 8. Set pluralRules.[[Type]] to t. plural_rules->set_type(type); // 12. Set pluralRules.[[Locale]] to the value of r.[[locale]]. plural_rules->set_locale(*locale_str); plural_rules->set_icu_plural_rules(*managed_plural_rules); plural_rules->set_icu_number_formatter(*managed_number_formatter); // 13. Return pluralRules. return plural_rules; } MaybeHandle<String> JSPluralRules::ResolvePlural( Isolate* isolate, Handle<JSPluralRules> plural_rules, double number) { icu::PluralRules* icu_plural_rules = plural_rules->icu_plural_rules().raw(); DCHECK_NOT_NULL(icu_plural_rules); icu::number::LocalizedNumberFormatter* fmt = plural_rules->icu_number_formatter().raw(); DCHECK_NOT_NULL(fmt); UErrorCode status = U_ZERO_ERROR; icu::number::FormattedNumber formatted_number = fmt->formatDouble(number, status); DCHECK(U_SUCCESS(status)); icu::UnicodeString result = icu_plural_rules->select(formatted_number, status); DCHECK(U_SUCCESS(status)); return Intl::ToString(isolate, result); } namespace { void CreateDataPropertyForOptions(Isolate* isolate, Handle<JSObject> options, Handle<Object> value, const char* key) { Handle<String> key_str = isolate->factory()->NewStringFromAsciiChecked(key); // This is a brand new JSObject that shouldn't already have the same // key so this shouldn't fail. Maybe<bool> maybe = JSReceiver::CreateDataProperty(isolate, options, key_str, value, Just(kDontThrow)); DCHECK(maybe.FromJust()); USE(maybe); } void CreateDataPropertyForOptions(Isolate* isolate, Handle<JSObject> options, int value, const char* key) { Handle<Smi> value_smi(Smi::FromInt(value), isolate); CreateDataPropertyForOptions(isolate, options, value_smi, key); } } // namespace Handle<JSObject> JSPluralRules::ResolvedOptions( Isolate* isolate, Handle<JSPluralRules> plural_rules) { Handle<JSObject> options = isolate->factory()->NewJSObject(isolate->object_function()); Handle<String> locale_value(plural_rules->locale(), isolate); CreateDataPropertyForOptions(isolate, options, locale_value, "locale"); CreateDataPropertyForOptions(isolate, options, plural_rules->TypeAsString(), "type"); UErrorCode status = U_ZERO_ERROR; icu::number::LocalizedNumberFormatter* icu_number_formatter = plural_rules->icu_number_formatter().raw(); icu::UnicodeString skeleton = icu_number_formatter->toSkeleton(status); DCHECK(U_SUCCESS(status)); CreateDataPropertyForOptions( isolate, options, JSNumberFormat::MinimumIntegerDigitsFromSkeleton(skeleton), "minimumIntegerDigits"); int32_t min = 0, max = 0; if (JSNumberFormat::SignificantDigitsFromSkeleton(skeleton, &min, &max)) { CreateDataPropertyForOptions(isolate, options, min, "minimumSignificantDigits"); CreateDataPropertyForOptions(isolate, options, max, "maximumSignificantDigits"); } else { JSNumberFormat::FractionDigitsFromSkeleton(skeleton, &min, &max); CreateDataPropertyForOptions(isolate, options, min, "minimumFractionDigits"); CreateDataPropertyForOptions(isolate, options, max, "maximumFractionDigits"); } // 6. Let pluralCategories be a List of Strings representing the // possible results of PluralRuleSelect for the selected locale pr. icu::PluralRules* icu_plural_rules = plural_rules->icu_plural_rules().raw(); DCHECK_NOT_NULL(icu_plural_rules); std::unique_ptr<icu::StringEnumeration> categories( icu_plural_rules->getKeywords(status)); DCHECK(U_SUCCESS(status)); int32_t count = categories->count(status); DCHECK(U_SUCCESS(status)); Handle<FixedArray> plural_categories = isolate->factory()->NewFixedArray(count); for (int32_t i = 0; i < count; i++) { const icu::UnicodeString* category = categories->snext(status); DCHECK(U_SUCCESS(status)); if (category == nullptr) break; std::string keyword; Handle<String> value = isolate->factory()->NewStringFromAsciiChecked( category->toUTF8String(keyword).data()); plural_categories->set(i, *value); } // 7. Perform ! CreateDataProperty(options, "pluralCategories", // CreateArrayFromList(pluralCategories)). Handle<JSArray> plural_categories_value = isolate->factory()->NewJSArrayWithElements(plural_categories); CreateDataPropertyForOptions(isolate, options, plural_categories_value, "pluralCategories"); return options; } namespace { class PluralRulesAvailableLocales { public: PluralRulesAvailableLocales() { UErrorCode status = U_ZERO_ERROR; std::unique_ptr<icu::StringEnumeration> locales( icu::PluralRules::getAvailableLocales(status)); DCHECK(U_SUCCESS(status)); int32_t len = 0; const char* locale = nullptr; while ((locale = locales->next(&len, status)) != nullptr && U_SUCCESS(status)) { std::string str(locale); if (len > 3) { std::replace(str.begin(), str.end(), '_', '-'); } set_.insert(std::move(str)); } } const std::set<std::string>& Get() const { return set_; } private: std::set<std::string> set_; }; } // namespace const std::set<std::string>& JSPluralRules::GetAvailableLocales() { static base::LazyInstance<PluralRulesAvailableLocales>::type available_locales = LAZY_INSTANCE_INITIALIZER; return available_locales.Pointer()->Get(); } } // namespace internal } // namespace v8