2019-12-09 19:14:07 -05:00
|
|
|
use std::ops::Deref;
|
|
|
|
|
2019-12-19 19:15:52 -05:00
|
|
|
use crate::isolate::Isolate;
|
2019-12-24 16:10:40 -05:00
|
|
|
use crate::support::MaybeBool;
|
2019-12-09 19:14:07 -05:00
|
|
|
use crate::support::Opaque;
|
2019-12-24 16:10:40 -05:00
|
|
|
use crate::Context;
|
2019-12-20 10:01:45 -05:00
|
|
|
use crate::HandleScope;
|
2019-12-09 19:14:07 -05:00
|
|
|
use crate::Local;
|
|
|
|
use crate::Name;
|
|
|
|
use crate::Value;
|
|
|
|
|
|
|
|
/// A JavaScript object (ECMA-262, 4.3.3)
|
|
|
|
#[repr(C)]
|
|
|
|
pub struct Object(Opaque);
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
fn v8__Object__New(
|
2019-12-20 10:01:45 -05:00
|
|
|
isolate: *mut Isolate,
|
2019-12-09 19:14:07 -05:00
|
|
|
prototype_or_null: *mut Value,
|
|
|
|
names: *mut *mut Name,
|
|
|
|
values: *mut *mut Value,
|
|
|
|
length: usize,
|
|
|
|
) -> *mut Object;
|
2019-12-19 19:15:52 -05:00
|
|
|
fn v8__Object__GetIsolate(object: &Object) -> &mut Isolate;
|
2019-12-24 16:10:40 -05:00
|
|
|
|
|
|
|
fn v8__Object__Get(
|
|
|
|
object: &Object,
|
|
|
|
context: *const Context,
|
|
|
|
key: *const Value,
|
|
|
|
) -> *mut Value;
|
|
|
|
|
|
|
|
fn v8__Object__CreateDataProperty(
|
|
|
|
object: &Object,
|
|
|
|
context: *const Context,
|
|
|
|
key: *const Name,
|
|
|
|
value: *const Value,
|
|
|
|
) -> MaybeBool;
|
2019-12-09 19:14:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Object {
|
|
|
|
/// Creates a JavaScript object with the given properties, and
|
|
|
|
/// a the given prototype_or_null (which can be any JavaScript
|
|
|
|
/// value, and if it's null, the newly created object won't have
|
|
|
|
/// a prototype at all). This is similar to Object.create().
|
|
|
|
/// All properties will be created as enumerable, configurable
|
|
|
|
/// and writable properties.
|
2019-12-20 10:01:45 -05:00
|
|
|
pub fn new<'sc>(
|
|
|
|
scope: &mut HandleScope<'sc>,
|
|
|
|
mut prototype_or_null: Local<'sc, Value>,
|
|
|
|
names: Vec<Local<'sc, Name>>,
|
|
|
|
values: Vec<Local<'sc, Value>>,
|
2019-12-09 19:14:07 -05:00
|
|
|
length: usize,
|
2019-12-20 10:01:45 -05:00
|
|
|
) -> Local<'sc, Object> {
|
2019-12-09 19:14:07 -05:00
|
|
|
let mut names_: Vec<*mut Name> = vec![];
|
|
|
|
for mut name in names {
|
|
|
|
let n = &mut *name;
|
|
|
|
names_.push(n);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut values_: Vec<*mut Value> = vec![];
|
|
|
|
for mut value in values {
|
|
|
|
let n = &mut *value;
|
|
|
|
values_.push(n);
|
|
|
|
}
|
|
|
|
unsafe {
|
|
|
|
Local::from_raw(v8__Object__New(
|
2019-12-20 10:01:45 -05:00
|
|
|
scope.as_mut(),
|
2019-12-09 19:14:07 -05:00
|
|
|
&mut *prototype_or_null,
|
|
|
|
names_.as_mut_ptr(),
|
|
|
|
values_.as_mut_ptr(),
|
|
|
|
length,
|
|
|
|
))
|
|
|
|
.unwrap()
|
|
|
|
}
|
|
|
|
}
|
2019-12-19 08:13:33 -05:00
|
|
|
|
2019-12-24 16:10:40 -05:00
|
|
|
/// Implements CreateDataProperty (ECMA-262, 7.3.4).
|
|
|
|
///
|
|
|
|
/// Defines a configurable, writable, enumerable property with the given value
|
|
|
|
/// on the object unless the property already exists and is not configurable
|
|
|
|
/// or the object is not extensible.
|
|
|
|
///
|
|
|
|
/// Returns true on success.
|
|
|
|
pub fn create_data_property(
|
|
|
|
&self,
|
|
|
|
context: Local<Context>,
|
|
|
|
key: Local<Name>,
|
|
|
|
value: Local<Value>,
|
|
|
|
) -> MaybeBool {
|
|
|
|
unsafe { v8__Object__CreateDataProperty(self, &*context, &*key, &*value) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get<'a>(
|
|
|
|
&self,
|
|
|
|
context: Local<Context>,
|
|
|
|
key: Local<Value>,
|
|
|
|
) -> Option<Local<'a, Value>> {
|
|
|
|
unsafe {
|
|
|
|
let ptr = v8__Object__Get(self, &*context, &*key);
|
|
|
|
if ptr.is_null() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(Local::from_raw(ptr).unwrap())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-19 08:13:33 -05:00
|
|
|
/// Return the isolate to which the Object belongs to.
|
2019-12-19 16:58:39 -05:00
|
|
|
pub fn get_isolate(&self) -> &Isolate {
|
2019-12-19 08:13:33 -05:00
|
|
|
unsafe { v8__Object__GetIsolate(self) }
|
|
|
|
}
|
2019-12-09 19:14:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Deref for Object {
|
|
|
|
type Target = Value;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
unsafe { &*(self as *const _ as *const Value) }
|
|
|
|
}
|
|
|
|
}
|