mirror of
https://github.com/RGBCube/serenity
synced 2025-07-25 13:37:45 +00:00
LibJS: Convert ordinary_to_primitive() to ThrowCompletionOr
Also add spec step comments to it while we're here.
This commit is contained in:
parent
867b19affb
commit
fa2ac5b759
4 changed files with 26 additions and 13 deletions
|
@ -887,7 +887,7 @@ JS_DEFINE_NATIVE_FUNCTION(DatePrototype::symbol_to_primitive)
|
||||||
vm.throw_exception<TypeError>(global_object, ErrorType::InvalidHint, hint);
|
vm.throw_exception<TypeError>(global_object, ErrorType::InvalidHint, hint);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
return this_value.as_object().ordinary_to_primitive(try_first);
|
return TRY_OR_DISCARD(this_value.as_object().ordinary_to_primitive(try_first));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1146,30 +1146,43 @@ void Object::visit_edges(Cell::Visitor& visitor)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7.1.1.1 OrdinaryToPrimitive ( O, hint ), https://tc39.es/ecma262/#sec-ordinarytoprimitive
|
// 7.1.1.1 OrdinaryToPrimitive ( O, hint ), https://tc39.es/ecma262/#sec-ordinarytoprimitive
|
||||||
Value Object::ordinary_to_primitive(Value::PreferredType preferred_type) const
|
ThrowCompletionOr<Value> Object::ordinary_to_primitive(Value::PreferredType preferred_type) const
|
||||||
{
|
{
|
||||||
VERIFY(preferred_type == Value::PreferredType::String || preferred_type == Value::PreferredType::Number);
|
VERIFY(preferred_type == Value::PreferredType::String || preferred_type == Value::PreferredType::Number);
|
||||||
|
|
||||||
auto& vm = this->vm();
|
auto& vm = this->vm();
|
||||||
|
|
||||||
AK::Array<PropertyName, 2> method_names;
|
AK::Array<PropertyName, 2> method_names;
|
||||||
if (preferred_type == Value::PreferredType::String)
|
|
||||||
method_names = { vm.names.toString, vm.names.valueOf };
|
|
||||||
else
|
|
||||||
method_names = { vm.names.valueOf, vm.names.toString };
|
|
||||||
|
|
||||||
|
// 1. If hint is string, then
|
||||||
|
if (preferred_type == Value::PreferredType::String) {
|
||||||
|
// a. Let methodNames be « "toString", "valueOf" ».
|
||||||
|
method_names = { vm.names.toString, vm.names.valueOf };
|
||||||
|
} else {
|
||||||
|
// a. Let methodNames be « "valueOf", "toString" ».
|
||||||
|
method_names = { vm.names.valueOf, vm.names.toString };
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. For each element name of methodNames, do
|
||||||
for (auto& method_name : method_names) {
|
for (auto& method_name : method_names) {
|
||||||
|
// a. Let method be ? Get(O, name).
|
||||||
auto method = get(method_name);
|
auto method = get(method_name);
|
||||||
if (vm.exception())
|
if (auto* exception = vm.exception())
|
||||||
return {};
|
return throw_completion(exception->value());
|
||||||
|
|
||||||
|
// b. If IsCallable(method) is true, then
|
||||||
if (method.is_function()) {
|
if (method.is_function()) {
|
||||||
auto result = TRY_OR_DISCARD(vm.call(method.as_function(), const_cast<Object*>(this)));
|
// i. Let result be ? Call(method, O).
|
||||||
|
auto result = TRY(vm.call(method.as_function(), const_cast<Object*>(this)));
|
||||||
|
|
||||||
|
// ii. If Type(result) is not Object, return result.
|
||||||
if (!result.is_object())
|
if (!result.is_object())
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
vm.throw_exception<TypeError>(global_object(), ErrorType::Convert, "object", preferred_type == Value::PreferredType::String ? "string" : "number");
|
|
||||||
return {};
|
// 4. Throw a TypeError exception.
|
||||||
|
return vm.throw_completion<TypeError>(global_object(), ErrorType::Convert, "object", preferred_type == Value::PreferredType::String ? "string" : "number");
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -67,7 +67,7 @@ public:
|
||||||
|
|
||||||
// 7.1 Type Conversion, https://tc39.es/ecma262/#sec-type-conversion
|
// 7.1 Type Conversion, https://tc39.es/ecma262/#sec-type-conversion
|
||||||
|
|
||||||
Value ordinary_to_primitive(Value::PreferredType preferred_type) const;
|
ThrowCompletionOr<Value> ordinary_to_primitive(Value::PreferredType preferred_type) const;
|
||||||
|
|
||||||
// 7.2 Testing and Comparison Operations, https://tc39.es/ecma262/#sec-testing-and-comparison-operations
|
// 7.2 Testing and Comparison Operations, https://tc39.es/ecma262/#sec-testing-and-comparison-operations
|
||||||
|
|
||||||
|
|
|
@ -437,7 +437,7 @@ Value Value::to_primitive(GlobalObject& global_object, PreferredType preferred_t
|
||||||
}
|
}
|
||||||
if (preferred_type == PreferredType::Default)
|
if (preferred_type == PreferredType::Default)
|
||||||
preferred_type = PreferredType::Number;
|
preferred_type = PreferredType::Number;
|
||||||
return as_object().ordinary_to_primitive(preferred_type);
|
return TRY_OR_DISCARD(as_object().ordinary_to_primitive(preferred_type));
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue