use crate::{
exec::Executable,
gc::{Finalize, Trace},
property::{AccessorDescriptor, Attribute, DataDescriptor, PropertyDescriptor},
syntax::ast::node::{MethodDefinitionKind, Node, PropertyDefinition},
Context, Result, Value,
};
use std::fmt;
#[cfg(feature = "deser")]
use serde::{Deserialize, Serialize};
#[cfg_attr(feature = "deser", derive(Serialize, Deserialize))]
#[cfg_attr(feature = "deser", serde(transparent))]
#[derive(Clone, Debug, Trace, Finalize, PartialEq)]
pub struct Object {
properties: Box<[PropertyDefinition]>,
}
impl Object {
pub fn properties(&self) -> &[PropertyDefinition] {
&self.properties
}
pub(in crate::syntax::ast::node) fn display(
&self,
f: &mut fmt::Formatter<'_>,
indent: usize,
) -> fmt::Result {
f.write_str("{\n")?;
for property in self.properties().iter() {
match property {
PropertyDefinition::IdentifierReference(key) => {
write!(f, "{} {},", indent, key)?;
}
PropertyDefinition::Property(key, value) => {
write!(f, "{} {}: {},", indent, key, value)?;
}
PropertyDefinition::SpreadObject(key) => {
write!(f, "{} ...{},", indent, key)?;
}
PropertyDefinition::MethodDefinition(_kind, _key, _node) => {
unimplemented!("Display for PropertyDefinition::MethodDefinition");
}
}
}
f.write_str("}")
}
}
impl Executable for Object {
fn run(&self, context: &mut Context) -> Result<Value> {
let obj = Value::new_object(context);
for property in self.properties().iter() {
match property {
PropertyDefinition::Property(key, value) => {
obj.set_property(
key.clone(),
PropertyDescriptor::Data(DataDescriptor::new(
value.run(context)?,
Attribute::all(),
)),
);
}
PropertyDefinition::MethodDefinition(kind, name, func) => match kind {
MethodDefinitionKind::Ordinary => {
obj.set_property(
name.clone(),
PropertyDescriptor::Data(DataDescriptor::new(
func.run(context)?,
Attribute::all(),
)),
);
}
MethodDefinitionKind::Get => {
let set = obj
.get_property(name.clone())
.as_ref()
.and_then(|p| p.as_accessor_descriptor())
.and_then(|a| a.setter().cloned());
obj.set_property(
name.clone(),
PropertyDescriptor::Accessor(AccessorDescriptor {
get: func.run(context)?.as_object(),
set,
attributes: Attribute::WRITABLE
| Attribute::ENUMERABLE
| Attribute::CONFIGURABLE,
}),
)
}
MethodDefinitionKind::Set => {
let get = obj
.get_property(name.clone())
.as_ref()
.and_then(|p| p.as_accessor_descriptor())
.and_then(|a| a.getter().cloned());
obj.set_property(
name.clone(),
PropertyDescriptor::Accessor(AccessorDescriptor {
get,
set: func.run(context)?.as_object(),
attributes: Attribute::WRITABLE
| Attribute::ENUMERABLE
| Attribute::CONFIGURABLE,
}),
)
}
},
_ => {} }
}
Ok(obj)
}
}
impl fmt::Display for Object {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.display(f, 0)
}
}
impl<T> From<T> for Object
where
T: Into<Box<[PropertyDefinition]>>,
{
fn from(props: T) -> Self {
Self {
properties: props.into(),
}
}
}
impl From<Object> for Node {
fn from(obj: Object) -> Self {
Self::Object(obj)
}
}