arkcompiler_ets_runtime/ecmascript/builtins/builtins_displaynames.cpp
2023-07-14 16:16:09 +08:00

156 lines
7.1 KiB
C++

/*
* Copyright (c) 2021-2022 Huawei Device Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "builtins_displaynames.h"
#include "ecmascript/intl/locale_helper.h"
#include "ecmascript/ecma_vm.h"
#include "ecmascript/global_env.h"
#include "ecmascript/js_displaynames.h"
#include "ecmascript/js_intl.h"
#include "ecmascript/js_locale.h"
#include "ecmascript/js_object.h"
#include "ecmascript/object_factory.h"
namespace panda::ecmascript::builtins {
// 12.2.1 Intl.DisplayNames ( [ locales [ , options ] ] )
JSTaggedValue BuiltinsDisplayNames::DisplayNamesConstructor(EcmaRuntimeCallInfo *argv)
{
JSThread *thread = argv->GetThread();
BUILTINS_API_TRACE(thread, DisplayNames, Constructor);
[[maybe_unused]] EcmaHandleScope scope(thread);
EcmaVM *ecmaVm = thread->GetEcmaVM();
ObjectFactory *factory = ecmaVm->GetFactory();
// 1. If NewTarget is undefined, throw a TypeError exception.
JSHandle<JSTaggedValue> newTarget = GetNewTarget(argv);
if (newTarget->IsUndefined()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "newTarget is undefined", JSTaggedValue::Exception());
}
// 2. Let displayNames be ? OrdinaryCreateFromConstructor(NewTarget, "%DisplayNames.prototype%",
// « [[InitializedDisplayNames]], [[Locale]], [[Style]], [[Type]], [[Fallback]], [[Fields]] »).
JSHandle<JSTaggedValue> constructor = GetConstructor(argv);
JSHandle<JSDisplayNames> displayNames =
JSHandle<JSDisplayNames>::Cast(factory->NewJSObjectByConstructor(JSHandle<JSFunction>(constructor), newTarget));
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
// 3. Perform ? InitializeDisplayNames(displayNames, locales, options).
JSHandle<JSTaggedValue> locales = GetCallArg(argv, 0);
JSHandle<JSTaggedValue> options = GetCallArg(argv, 1);
JSDisplayNames::InitializeDisplayNames(thread, displayNames, locales, options);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
return displayNames.GetTaggedValue();
}
// 12.3.2 Intl.DisplayNames.supportedLocalesOf ( locales [ , options ] )
JSTaggedValue BuiltinsDisplayNames::SupportedLocalesOf(EcmaRuntimeCallInfo *argv)
{
JSThread *thread = argv->GetThread();
BUILTINS_API_TRACE(thread, DisplayNames, SupportedLocalesOf);
[[maybe_unused]] EcmaHandleScope scope(thread);
// 1. Let availableLocales be %DisplayNames%.[[AvailableLocales]].
JSHandle<TaggedArray> availableLocales = JSDisplayNames::GetAvailableLocales(thread);
// 2. Let requestedLocales be ? CanonicaliezLocaleList(locales).
JSHandle<JSTaggedValue> locales = GetCallArg(argv, 0);
JSHandle<TaggedArray> requestedLocales = intl::LocaleHelper::CanonicalizeLocaleList(thread, locales);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
// 3. Return ? SupportedLocales(availableLocales, requestedLocales, options).
JSHandle<JSTaggedValue> options = GetCallArg(argv, 1);
JSHandle<JSArray> result = JSLocale::SupportedLocales(thread, availableLocales, requestedLocales, options);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
return result.GetTaggedValue();
}
// 12.4.3 get Intl.DisplayNames.prototype.of
JSTaggedValue BuiltinsDisplayNames::Of(EcmaRuntimeCallInfo *argv)
{
JSThread *thread = argv->GetThread();
BUILTINS_API_TRACE(thread, DisplayNames, Of);
[[maybe_unused]] EcmaHandleScope scope(thread);
// 1. Let displayNames be this value.
JSHandle<JSTaggedValue> thisValue = GetThis(argv);
// 2. Perform ? RequireInternalSlot(displayNames, [[InitializedDisplayNames]]).
if (!thisValue->IsJSDisplayNames()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "this is not dn object", JSTaggedValue::Exception());
}
// 3. Let code be ? ToString(code).
JSHandle<JSTaggedValue> codeValue = GetCallArg(argv, 0);
JSHandle<EcmaString> codeTemp = JSTaggedValue::ToString(thread, codeValue);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
// 4. Let code be ? CanonicalCodeForDisplayNames(displayNames.[[Type]], code).
// 5. Let fields be displayNames.[[Fields]].
// 6. If fields has a field [[<code>]], return fields.[[<code>]].
JSHandle<JSDisplayNames> displayNames = JSHandle<JSDisplayNames>::Cast(thisValue);
TypednsOption typeOpt = displayNames->GetType();
JSHandle<EcmaString> code = JSDisplayNames::CanonicalCodeForDisplayNames(thread, displayNames, typeOpt, codeTemp);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
std::string codeString = intl::LocaleHelper::ConvertToStdString(code);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
if (codeString.size()) {
JSHandle<JSTaggedValue> codeStr = JSHandle<JSTaggedValue>::Cast(code);
return codeStr.GetTaggedValue();
}
// 7. If displayNames.[[Fallback]] is "code", return code.
FallbackOption fallback = displayNames->GetFallback();
if (fallback == FallbackOption::CODE) {
return codeValue.GetTaggedValue();
}
// 8. Return undefined.
return JSTaggedValue::Undefined();
}
// 12.4.4 Intl.DisplayNames.prototype.resolvedOptions ()
JSTaggedValue BuiltinsDisplayNames::ResolvedOptions(EcmaRuntimeCallInfo *argv)
{
JSThread *thread = argv->GetThread();
BUILTINS_API_TRACE(thread, DisplayNames, ResolvedOptions);
[[maybe_unused]] EcmaHandleScope scope(thread);
// 1. Let DisplayNames be the this value.
JSHandle<JSTaggedValue> thisValue = GetThis(argv);
// 2. Perform ? RequireInternalSlot(DisplayNames, [[InitializedDisplayNames]]).
if (!thisValue->IsJSDisplayNames()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "this is not dn object", JSTaggedValue::Exception());
}
// 3. Let options be ! OrdinaryObjectCreate(%ObjectPrototype%).
auto ecmaVm = thread->GetEcmaVM();
JSHandle<GlobalEnv> env = ecmaVm->GetGlobalEnv();
ObjectFactory *factory = ecmaVm->GetFactory();
JSHandle<JSFunction> ctor(env->GetObjectFunction());
JSHandle<JSObject> options(factory->NewJSObjectByConstructor(ctor));
// 4. For each row of Table 8, except the header row, in table order, do
// Let p be the Property value of the current row.
// Let v be the value of displayNames's internal slot whose name is the Internal Slot value of the current row.
// Assert: v is not undefined.
// Perform ! CreateDataPropertyOrThrow(options, p, v).
JSHandle<JSDisplayNames> displayNames = JSHandle<JSDisplayNames>::Cast(thisValue);
JSDisplayNames::ResolvedOptions(thread, displayNames, options);
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
// 5. Return options.
return options.GetTaggedValue();
}
} // namespace panda::ecmascript::builtins