use boa_macros::js_str;
use crate::{
builtins::function::{set_function_name, OrdinaryFunction},
object::internal_methods::InternalMethodContext,
property::PropertyDescriptor,
vm::{opcode::Operation, CompletionType},
Context, JsResult,
};
#[derive(Debug, Clone, Copy)]
pub(crate) struct DefineClassStaticGetterByName;
impl DefineClassStaticGetterByName {
fn operation(context: &mut Context, index: usize) -> JsResult<CompletionType> {
let function = context.vm.pop();
let class = context.vm.pop();
let class = class.as_object().expect("class must be object");
let key = context
.vm
.frame()
.code_block()
.constant_string(index)
.into();
{
let function_object = function
.as_object()
.expect("method must be function object");
set_function_name(function_object, &key, Some(js_str!("get")), context);
function_object
.downcast_mut::<OrdinaryFunction>()
.expect("method must be function object")
.set_home_object(class.clone());
}
let set = class
.__get_own_property__(&key, &mut InternalMethodContext::new(context))?
.as_ref()
.and_then(PropertyDescriptor::set)
.cloned();
class.__define_own_property__(
&key,
PropertyDescriptor::builder()
.maybe_get(Some(function))
.maybe_set(set)
.enumerable(false)
.configurable(true)
.build(),
&mut InternalMethodContext::new(context),
)?;
Ok(CompletionType::Normal)
}
}
impl Operation for DefineClassStaticGetterByName {
const NAME: &'static str = "DefineClassStaticGetterByName";
const INSTRUCTION: &'static str = "INST - DefineClassStaticGetterByName";
const COST: u8 = 6;
fn execute(context: &mut Context) -> JsResult<CompletionType> {
let index = context.vm.read::<u8>() as usize;
Self::operation(context, index)
}
fn execute_with_u16_operands(context: &mut Context) -> JsResult<CompletionType> {
let index = context.vm.read::<u16>() as usize;
Self::operation(context, index)
}
fn execute_with_u32_operands(context: &mut Context) -> JsResult<CompletionType> {
let index = context.vm.read::<u32>() as usize;
Self::operation(context, index)
}
}
#[derive(Debug, Clone, Copy)]
pub(crate) struct DefineClassGetterByName;
impl DefineClassGetterByName {
fn operation(context: &mut Context, index: usize) -> JsResult<CompletionType> {
let function = context.vm.pop();
let class_proto = context.vm.pop();
let class_proto = class_proto.as_object().expect("class must be object");
let key = context
.vm
.frame()
.code_block()
.constant_string(index)
.into();
{
let function_object = function
.as_object()
.expect("method must be function object");
set_function_name(function_object, &key, Some(js_str!("get")), context);
function_object
.downcast_mut::<OrdinaryFunction>()
.expect("method must be function object")
.set_home_object(class_proto.clone());
}
let set = class_proto
.__get_own_property__(&key, &mut InternalMethodContext::new(context))?
.as_ref()
.and_then(PropertyDescriptor::set)
.cloned();
class_proto.__define_own_property__(
&key,
PropertyDescriptor::builder()
.maybe_get(Some(function))
.maybe_set(set)
.enumerable(false)
.configurable(true)
.build(),
&mut InternalMethodContext::new(context),
)?;
Ok(CompletionType::Normal)
}
}
impl Operation for DefineClassGetterByName {
const NAME: &'static str = "DefineClassGetterByName";
const INSTRUCTION: &'static str = "INST - DefineClassGetterByName";
const COST: u8 = 6;
fn execute(context: &mut Context) -> JsResult<CompletionType> {
let index = context.vm.read::<u8>() as usize;
Self::operation(context, index)
}
fn execute_with_u16_operands(context: &mut Context) -> JsResult<CompletionType> {
let index = context.vm.read::<u16>() as usize;
Self::operation(context, index)
}
fn execute_with_u32_operands(context: &mut Context) -> JsResult<CompletionType> {
let index = context.vm.read::<u32>() as usize;
Self::operation(context, index)
}
}
#[derive(Debug, Clone, Copy)]
pub(crate) struct DefineClassStaticGetterByValue;
impl Operation for DefineClassStaticGetterByValue {
const NAME: &'static str = "DefineClassStaticGetterByValue";
const INSTRUCTION: &'static str = "INST - DefineClassStaticGetterByValue";
const COST: u8 = 6;
fn execute(context: &mut Context) -> JsResult<CompletionType> {
let function = context.vm.pop();
let key = context.vm.pop();
let class = context.vm.pop();
let class = class.as_object().expect("class must be object");
let key = key
.to_property_key(context)
.expect("property key must already be valid");
{
let function_object = function
.as_object()
.expect("method must be function object");
set_function_name(function_object, &key, Some(js_str!("get")), context);
function_object
.downcast_mut::<OrdinaryFunction>()
.expect("method must be function object")
.set_home_object(class.clone());
}
let set = class
.__get_own_property__(&key, &mut InternalMethodContext::new(context))?
.as_ref()
.and_then(PropertyDescriptor::set)
.cloned();
class.define_property_or_throw(
key,
PropertyDescriptor::builder()
.maybe_get(Some(function))
.maybe_set(set)
.enumerable(false)
.configurable(true)
.build(),
context,
)?;
Ok(CompletionType::Normal)
}
}
#[derive(Debug, Clone, Copy)]
pub(crate) struct DefineClassGetterByValue;
impl Operation for DefineClassGetterByValue {
const NAME: &'static str = "DefineClassGetterByValue";
const INSTRUCTION: &'static str = "INST - DefineClassGetterByValue";
const COST: u8 = 6;
fn execute(context: &mut Context) -> JsResult<CompletionType> {
let function = context.vm.pop();
let key = context.vm.pop();
let class_proto = context.vm.pop();
let class_proto = class_proto.as_object().expect("class must be object");
let key = key
.to_property_key(context)
.expect("property key must already be valid");
{
let function_object = function
.as_object()
.expect("method must be function object");
set_function_name(function_object, &key, Some(js_str!("get")), context);
function_object
.downcast_mut::<OrdinaryFunction>()
.expect("method must be function object")
.set_home_object(class_proto.clone());
}
let set = class_proto
.__get_own_property__(&key, &mut InternalMethodContext::new(context))?
.as_ref()
.and_then(PropertyDescriptor::set)
.cloned();
class_proto.__define_own_property__(
&key,
PropertyDescriptor::builder()
.maybe_get(Some(function))
.maybe_set(set)
.enumerable(false)
.configurable(true)
.build(),
&mut InternalMethodContext::new(context),
)?;
Ok(CompletionType::Normal)
}
}