0
0
Fork 0
mirror of https://github.com/denoland/rusty_v8.git synced 2024-12-31 19:44:16 -05:00
denoland-rusty-v8/src/object.rs

179 lines
5 KiB
Rust
Raw Normal View History

use crate::isolate::Isolate;
2020-01-02 10:41:40 -05:00
use crate::support::int;
use crate::support::MapFnTo;
use crate::support::MaybeBool;
use crate::AccessorNameGetterCallback;
2020-01-02 10:41:40 -05:00
use crate::Array;
use crate::Context;
2019-12-09 19:14:07 -05:00
use crate::Local;
use crate::Name;
2019-12-30 09:28:39 -05:00
use crate::Object;
use crate::PropertyAttribute;
use crate::ToLocal;
2019-12-09 19:14:07 -05:00
use crate::Value;
extern "C" {
2019-12-30 12:14:06 -05:00
fn v8__Object__New(isolate: *mut Isolate) -> *mut Object;
fn v8__Object__New__with_prototype_and_properties(
2019-12-20 10:01:45 -05:00
isolate: *mut Isolate,
prototype_or_null: Local<Value>,
names: *mut Local<Name>,
values: *mut Local<Value>,
2019-12-09 19:14:07 -05:00
length: usize,
) -> *mut Object;
2020-01-02 12:01:36 -05:00
fn v8__Object__SetAccessor(
self_: &Object,
context: Local<Context>,
name: Local<Name>,
2020-01-02 12:01:36 -05:00
getter: AccessorNameGetterCallback,
) -> MaybeBool;
fn v8__Object__Get(
object: &Object,
context: Local<Context>,
key: Local<Value>,
) -> *mut Value;
2019-12-26 14:38:16 -05:00
fn v8__Object__Set(
object: &Object,
context: Local<Context>,
key: Local<Value>,
value: Local<Value>,
2019-12-26 14:38:16 -05:00
) -> MaybeBool;
fn v8__Object__CreateDataProperty(
object: &Object,
context: Local<Context>,
key: Local<Name>,
value: Local<Value>,
) -> MaybeBool;
fn v8__Object__DefineOwnProperty(
object: &Object,
context: Local<Context>,
key: Local<Name>,
value: Local<Value>,
attr: PropertyAttribute,
) -> MaybeBool;
2020-01-02 13:56:28 -05:00
fn v8__Object__GetIdentityHash(object: &Object) -> int;
2020-01-02 10:41:40 -05:00
fn v8__Array__New(isolate: *mut Isolate, length: int) -> *mut 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.
pub fn with_prototype_and_properties<'sc>(
scope: &mut impl ToLocal<'sc>,
prototype_or_null: Local<'sc, Value>,
names: &[Local<Name>],
values: &[Local<Value>],
2019-12-20 10:01:45 -05:00
) -> Local<'sc, Object> {
assert_eq!(names.len(), values.len());
unsafe {
let object = v8__Object__New__with_prototype_and_properties(
scope.isolate(),
prototype_or_null,
names.as_ptr() as *mut Local<Name>,
values.as_ptr() as *mut Local<Value>,
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>,
) -> Option<bool> {
unsafe { v8__Object__Set(self, context, key, value) }.into()
2019-12-26 14:38:16 -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>,
) -> Option<bool> {
unsafe { v8__Object__CreateDataProperty(self, context, key, value) }.into()
}
/// 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,
) -> Option<bool> {
unsafe { v8__Object__DefineOwnProperty(self, context, key, value, attr) }
.into()
}
pub fn get<'a>(
&self,
scope: &mut impl ToLocal<'a>,
context: Local<Context>,
key: Local<Value>,
) -> Option<Local<'a, Value>> {
unsafe {
let ptr = v8__Object__Get(self, context, key);
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>,
getter: impl for<'s> MapFnTo<AccessorNameGetterCallback<'s>>,
) -> Option<bool> {
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) }
}
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()
}
}