2019-12-19 19:15:52 -05:00
|
|
|
use crate::isolate::Isolate;
|
2020-01-02 10:41:40 -05:00
|
|
|
use crate::support::int;
|
2020-01-04 09:23:36 -05:00
|
|
|
use crate::support::MapFnTo;
|
2019-12-24 16:10:40 -05:00
|
|
|
use crate::support::MaybeBool;
|
2020-01-04 09:23:36 -05:00
|
|
|
use crate::AccessorNameGetterCallback;
|
2020-01-02 10:41:40 -05:00
|
|
|
use crate::Array;
|
2019-12-24 16:10:40 -05:00
|
|
|
use crate::Context;
|
2019-12-09 19:14:07 -05:00
|
|
|
use crate::Local;
|
2020-04-02 13:37:13 -04:00
|
|
|
use crate::Map;
|
2019-12-09 19:14:07 -05:00
|
|
|
use crate::Name;
|
2019-12-30 09:28:39 -05:00
|
|
|
use crate::Object;
|
2020-01-20 11:16:24 -05:00
|
|
|
use crate::PropertyAttribute;
|
2019-12-24 18:31:36 -05:00
|
|
|
use crate::ToLocal;
|
2019-12-09 19:14:07 -05:00
|
|
|
use crate::Value;
|
|
|
|
|
|
|
|
extern "C" {
|
2020-04-13 08:43:56 -04:00
|
|
|
fn v8__Object__New(isolate: *mut Isolate) -> *const Object;
|
2020-01-22 16:58:31 -05:00
|
|
|
fn v8__Object__New__with_prototype_and_properties(
|
2019-12-20 10:01:45 -05:00
|
|
|
isolate: *mut Isolate,
|
2020-04-13 08:43:56 -04:00
|
|
|
prototype_or_null: *const Value,
|
|
|
|
names: *const *const Name,
|
|
|
|
values: *const *const Value,
|
2019-12-09 19:14:07 -05:00
|
|
|
length: usize,
|
2020-04-13 08:43:56 -04:00
|
|
|
) -> *const Object;
|
2020-01-02 12:01:36 -05:00
|
|
|
fn v8__Object__SetAccessor(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
|
|
|
key: *const Name,
|
2020-01-02 12:01:36 -05:00
|
|
|
getter: AccessorNameGetterCallback,
|
|
|
|
) -> MaybeBool;
|
2019-12-24 16:10:40 -05:00
|
|
|
fn v8__Object__Get(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
|
|
|
key: *const Value,
|
|
|
|
) -> *const Value;
|
2020-02-14 09:42:54 -05:00
|
|
|
fn v8__Object__GetIndex(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
2020-02-14 09:42:54 -05:00
|
|
|
index: u32,
|
2020-04-13 08:43:56 -04:00
|
|
|
) -> *const Value;
|
|
|
|
fn v8__Object__GetPrototype(this: *const Object) -> *const Value;
|
2019-12-26 14:38:16 -05:00
|
|
|
fn v8__Object__Set(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
|
|
|
key: *const Value,
|
|
|
|
value: *const Value,
|
2019-12-26 14:38:16 -05:00
|
|
|
) -> MaybeBool;
|
2020-02-14 09:42:54 -05:00
|
|
|
fn v8__Object__SetIndex(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
2020-02-14 09:42:54 -05:00
|
|
|
index: u32,
|
2020-04-13 08:43:56 -04:00
|
|
|
value: *const Value,
|
2020-02-14 09:42:54 -05:00
|
|
|
) -> MaybeBool;
|
2020-03-23 11:39:43 -04:00
|
|
|
fn v8__Object__SetPrototype(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
|
|
|
prototype: *const Value,
|
2020-03-23 11:39:43 -04:00
|
|
|
) -> MaybeBool;
|
2019-12-24 16:10:40 -05:00
|
|
|
fn v8__Object__CreateDataProperty(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
|
|
|
key: *const Name,
|
|
|
|
value: *const Value,
|
2019-12-24 16:10:40 -05:00
|
|
|
) -> MaybeBool;
|
2020-01-20 11:16:24 -05:00
|
|
|
fn v8__Object__DefineOwnProperty(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Object,
|
|
|
|
context: *const Context,
|
|
|
|
key: *const Name,
|
|
|
|
value: *const Value,
|
2020-01-20 11:16:24 -05:00
|
|
|
attr: PropertyAttribute,
|
|
|
|
) -> MaybeBool;
|
2020-04-13 08:43:56 -04:00
|
|
|
fn v8__Object__GetIdentityHash(this: *const Object) -> int;
|
|
|
|
fn v8__Object__CreationContext(this: *const Object) -> *const Context;
|
2020-01-02 10:41:40 -05:00
|
|
|
|
2020-04-13 08:43:56 -04:00
|
|
|
fn v8__Array__New(isolate: *mut Isolate, length: int) -> *const Array;
|
2020-02-14 09:42:54 -05:00
|
|
|
fn v8__Array__New_with_elements(
|
|
|
|
isolate: *mut Isolate,
|
2020-04-13 08:43:56 -04:00
|
|
|
elements: *const *const Value,
|
2020-02-14 09:42:54 -05:00
|
|
|
length: usize,
|
2020-04-13 08:43:56 -04:00
|
|
|
) -> *const Array;
|
|
|
|
fn v8__Array__Length(array: *const Array) -> u32;
|
|
|
|
fn v8__Map__Size(map: *const Map) -> usize;
|
|
|
|
fn v8__Map__As__Array(this: *const Map) -> *const Array;
|
2019-12-09 19:14:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Object {
|
2019-12-30 12:14:06 -05:00
|
|
|
/// Creates an empty object.
|
|
|
|
pub fn new<'sc>(scope: &mut impl ToLocal<'sc>) -> Local<'sc, Object> {
|
|
|
|
let ptr = unsafe { v8__Object__New(scope.isolate()) };
|
|
|
|
unsafe { scope.to_local(ptr) }.unwrap()
|
|
|
|
}
|
|
|
|
|
2019-12-09 19:14:07 -05:00
|
|
|
/// 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.
|
2020-01-22 16:58:31 -05:00
|
|
|
pub fn with_prototype_and_properties<'sc>(
|
2019-12-24 18:31:36 -05:00
|
|
|
scope: &mut impl ToLocal<'sc>,
|
2020-01-22 16:58:31 -05:00
|
|
|
prototype_or_null: Local<'sc, Value>,
|
|
|
|
names: &[Local<Name>],
|
|
|
|
values: &[Local<Value>],
|
2019-12-20 10:01:45 -05:00
|
|
|
) -> Local<'sc, Object> {
|
2020-01-22 16:58:31 -05:00
|
|
|
assert_eq!(names.len(), values.len());
|
2020-04-13 08:43:56 -04:00
|
|
|
let names = Local::slice_into_raw(names);
|
|
|
|
let values = Local::slice_into_raw(values);
|
2020-01-22 16:58:31 -05:00
|
|
|
unsafe {
|
|
|
|
let object = v8__Object__New__with_prototype_and_properties(
|
2019-12-24 18:31:36 -05:00
|
|
|
scope.isolate(),
|
2020-04-13 08:43:56 -04:00
|
|
|
&*prototype_or_null,
|
|
|
|
names.as_ptr(),
|
|
|
|
values.as_ptr(),
|
2020-01-22 16:58:31 -05:00
|
|
|
names.len(),
|
|
|
|
);
|
|
|
|
scope.to_local(object).unwrap()
|
|
|
|
}
|
2019-12-09 19:14:07 -05:00
|
|
|
}
|
2019-12-19 08:13:33 -05:00
|
|
|
|
2019-12-26 14:38:16 -05:00
|
|
|
/// Set only return Just(true) or Empty(), so if it should never fail, use
|
|
|
|
/// result.Check().
|
|
|
|
pub fn set(
|
|
|
|
&self,
|
|
|
|
context: Local<Context>,
|
|
|
|
key: Local<Value>,
|
|
|
|
value: Local<Value>,
|
2020-01-22 16:58:31 -05:00
|
|
|
) -> Option<bool> {
|
2020-04-13 08:43:56 -04:00
|
|
|
unsafe { v8__Object__Set(self, &*context, &*key, &*value) }.into()
|
2019-12-26 14:38:16 -05:00
|
|
|
}
|
|
|
|
|
2020-02-14 09:42:54 -05:00
|
|
|
/// Set only return Just(true) or Empty(), so if it should never fail, use
|
|
|
|
/// result.Check().
|
|
|
|
pub fn set_index(
|
|
|
|
&self,
|
|
|
|
context: Local<Context>,
|
|
|
|
index: u32,
|
|
|
|
value: Local<Value>,
|
|
|
|
) -> Option<bool> {
|
2020-04-13 08:43:56 -04:00
|
|
|
unsafe { v8__Object__SetIndex(self, &*context, index, &*value) }.into()
|
2020-02-14 09:42:54 -05:00
|
|
|
}
|
|
|
|
|
2020-03-23 11:39:43 -04:00
|
|
|
/// Set the prototype object. This does not skip objects marked to be
|
|
|
|
/// skipped by proto and it does not consult the security handler.
|
|
|
|
pub fn set_prototype(
|
|
|
|
&self,
|
|
|
|
context: Local<Context>,
|
|
|
|
prototype: Local<Value>,
|
|
|
|
) -> Option<bool> {
|
2020-04-13 08:43:56 -04:00
|
|
|
unsafe { v8__Object__SetPrototype(self, &*context, &*prototype) }.into()
|
2020-03-23 11:39:43 -04: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>,
|
2020-01-22 16:58:31 -05:00
|
|
|
) -> Option<bool> {
|
2020-04-13 08:43:56 -04:00
|
|
|
unsafe { v8__Object__CreateDataProperty(self, &*context, &*key, &*value) }
|
|
|
|
.into()
|
2019-12-24 16:10:40 -05:00
|
|
|
}
|
|
|
|
|
2020-01-20 11:16:24 -05:00
|
|
|
/// Implements DefineOwnProperty.
|
|
|
|
///
|
|
|
|
/// In general, CreateDataProperty will be faster, however, does not allow
|
|
|
|
/// for specifying attributes.
|
|
|
|
///
|
|
|
|
/// Returns true on success.
|
|
|
|
pub fn define_own_property(
|
|
|
|
&self,
|
|
|
|
context: Local<Context>,
|
|
|
|
key: Local<Name>,
|
|
|
|
value: Local<Value>,
|
|
|
|
attr: PropertyAttribute,
|
2020-01-22 16:58:31 -05:00
|
|
|
) -> Option<bool> {
|
2020-04-13 08:43:56 -04:00
|
|
|
unsafe {
|
|
|
|
v8__Object__DefineOwnProperty(self, &*context, &*key, &*value, attr)
|
|
|
|
}
|
|
|
|
.into()
|
2020-01-20 11:16:24 -05:00
|
|
|
}
|
|
|
|
|
2019-12-24 16:10:40 -05:00
|
|
|
pub fn get<'a>(
|
|
|
|
&self,
|
2019-12-24 18:31:36 -05:00
|
|
|
scope: &mut impl ToLocal<'a>,
|
2019-12-24 16:10:40 -05:00
|
|
|
context: Local<Context>,
|
|
|
|
key: Local<Value>,
|
|
|
|
) -> Option<Local<'a, Value>> {
|
|
|
|
unsafe {
|
2020-04-13 08:43:56 -04:00
|
|
|
let ptr = v8__Object__Get(self, &*context, &*key);
|
2019-12-24 18:31:36 -05:00
|
|
|
scope.to_local(ptr)
|
2019-12-24 16:10:40 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-14 09:42:54 -05:00
|
|
|
pub fn get_index<'a>(
|
|
|
|
&self,
|
|
|
|
scope: &mut impl ToLocal<'a>,
|
|
|
|
context: Local<Context>,
|
|
|
|
index: u32,
|
|
|
|
) -> Option<Local<'a, Value>> {
|
|
|
|
unsafe {
|
2020-04-13 08:43:56 -04:00
|
|
|
let ptr = v8__Object__GetIndex(self, &*context, index);
|
2020-02-14 09:42:54 -05:00
|
|
|
scope.to_local(ptr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-23 11:39:43 -04:00
|
|
|
/// Get the prototype object. This does not skip objects marked to be
|
|
|
|
/// skipped by proto and it does not consult the security handler.
|
|
|
|
pub fn get_prototype<'a>(
|
|
|
|
&self,
|
|
|
|
scope: &mut impl ToLocal<'a>,
|
|
|
|
) -> Option<Local<'a, Value>> {
|
|
|
|
unsafe {
|
|
|
|
let ptr = v8__Object__GetPrototype(self);
|
|
|
|
scope.to_local(ptr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-02 12:01:36 -05:00
|
|
|
/// Note: SideEffectType affects the getter only, not the setter.
|
|
|
|
pub fn set_accessor(
|
|
|
|
&mut self,
|
|
|
|
context: Local<Context>,
|
|
|
|
name: Local<Name>,
|
2020-01-17 01:39:05 -05:00
|
|
|
getter: impl for<'s> MapFnTo<AccessorNameGetterCallback<'s>>,
|
2020-01-22 16:58:31 -05:00
|
|
|
) -> Option<bool> {
|
2020-04-13 08:43:56 -04:00
|
|
|
unsafe {
|
|
|
|
v8__Object__SetAccessor(self, &*context, &*name, getter.map_fn_to())
|
|
|
|
}
|
|
|
|
.into()
|
2019-12-19 08:13:33 -05:00
|
|
|
}
|
2020-01-02 13:56:28 -05:00
|
|
|
|
|
|
|
/// Returns the identity hash for this object. The current implementation
|
|
|
|
/// uses a hidden property on the object to store the identity hash.
|
|
|
|
///
|
|
|
|
/// The return value will never be 0. Also, it is not guaranteed to be
|
|
|
|
/// unique.
|
|
|
|
pub fn get_identity_hash(&self) -> int {
|
|
|
|
unsafe { v8__Object__GetIdentityHash(self) }
|
|
|
|
}
|
2020-01-26 11:42:28 -05:00
|
|
|
|
|
|
|
/// Returns the context in which the object was created.
|
|
|
|
pub fn creation_context<'a>(
|
|
|
|
&self,
|
|
|
|
scope: &mut impl ToLocal<'a>,
|
|
|
|
) -> Local<'a, Context> {
|
|
|
|
unsafe {
|
|
|
|
let ptr = v8__Object__CreationContext(self);
|
|
|
|
scope.to_local(ptr).unwrap()
|
|
|
|
}
|
|
|
|
}
|
2019-12-09 19:14:07 -05:00
|
|
|
}
|
2020-01-02 10:41:40 -05:00
|
|
|
|
|
|
|
impl Array {
|
|
|
|
/// Creates a JavaScript array with the given length. If the length
|
|
|
|
/// is negative the returned array will have length 0.
|
|
|
|
pub fn new<'sc>(
|
|
|
|
scope: &mut impl ToLocal<'sc>,
|
|
|
|
length: i32,
|
|
|
|
) -> Local<'sc, Array> {
|
|
|
|
let ptr = unsafe { v8__Array__New(scope.isolate(), length) };
|
|
|
|
unsafe { scope.to_local(ptr) }.unwrap()
|
|
|
|
}
|
2020-02-14 09:42:54 -05:00
|
|
|
|
|
|
|
/// Creates a JavaScript array out of a Local<Value> array with a known length.
|
|
|
|
pub fn new_with_elements<'sc>(
|
|
|
|
scope: &mut impl ToLocal<'sc>,
|
|
|
|
elements: &[Local<Value>],
|
|
|
|
) -> Local<'sc, Array> {
|
|
|
|
if elements.is_empty() {
|
|
|
|
return Self::new(scope, 0);
|
|
|
|
}
|
2020-04-13 08:43:56 -04:00
|
|
|
let elements = Local::slice_into_raw(elements);
|
2020-02-14 09:42:54 -05:00
|
|
|
let ptr = unsafe {
|
|
|
|
v8__Array__New_with_elements(
|
|
|
|
scope.isolate(),
|
2020-04-13 08:43:56 -04:00
|
|
|
elements.as_ptr(),
|
2020-02-14 09:42:54 -05:00
|
|
|
elements.len(),
|
|
|
|
)
|
|
|
|
};
|
|
|
|
unsafe { scope.to_local(ptr) }.unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn length(&self) -> u32 {
|
|
|
|
unsafe { v8__Array__Length(self) }
|
|
|
|
}
|
2020-01-02 10:41:40 -05:00
|
|
|
}
|
2020-04-02 13:37:13 -04:00
|
|
|
|
|
|
|
impl Map {
|
|
|
|
pub fn size(&self) -> usize {
|
|
|
|
unsafe { v8__Map__Size(self) }
|
|
|
|
}
|
|
|
|
/// Returns an array of length size() * 2, where index N is the Nth key and
|
|
|
|
/// index N + 1 is the Nth value.
|
|
|
|
pub fn as_array<'sc>(
|
|
|
|
&self,
|
|
|
|
scope: &mut impl ToLocal<'sc>,
|
|
|
|
) -> Local<'sc, Array> {
|
|
|
|
let ptr = unsafe { v8__Map__As__Array(self) };
|
|
|
|
unsafe { scope.to_local(ptr) }.unwrap()
|
|
|
|
}
|
|
|
|
}
|