1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
//! Templates for constructing functions and objects efficiently.
use v8_sys as v8;
use isolate;
use util;
use value;
use value::Data;
use context;
use std::os;
use std::ptr;
use std::mem;
use std::ops;
use std::ffi;

/// The superclass of object and function templates.
#[derive(Debug)]
pub struct Template(isolate::Isolate, v8::TemplateRef);

/// A FunctionTemplate is used to create functions at runtime.
///
/// There can only be one function created from a FunctionTemplate in a context.  Any modification
/// of a FunctionTemplate after first instantiation will trigger a crash.  A FunctionTemplate can
/// have properties, these properties are added to the function object when it is created.
#[derive(Debug)]
pub struct FunctionTemplate(isolate::Isolate, v8::FunctionTemplateRef);

/// An ObjectTemplate is used to create objects at runtime.
///
/// Properties added to an ObjectTemplate are added to each object created from the ObjectTemplate.
#[derive(Debug)]
pub struct ObjectTemplate(isolate::Isolate, v8::ObjectTemplateRef);

/// A Signature specifies which receiver is valid for a function.
#[derive(Debug)]
pub struct Signature(isolate::Isolate, v8::SignatureRef);

impl Signature {
    /// Creates a new signature.
    pub fn new(isolate: &isolate::Isolate) -> Signature {
        let raw = unsafe {
            util::invoke(isolate,
                         |c| v8::v8_Signature_New(c, isolate.as_raw(), ptr::null_mut()))
                .unwrap()
        };
        Signature(isolate.clone(), raw)
    }

    /// Creates a new signature with the specified receiver.
    pub fn new_with_receiver(isolate: &isolate::Isolate, receiver: &FunctionTemplate) -> Signature {
        let raw = unsafe {
            util::invoke(isolate,
                         |c| v8::v8_Signature_New(c, isolate.as_raw(), receiver.1))
                .unwrap()
        };
        Signature(isolate.clone(), raw)
    }
}

impl FunctionTemplate {
    /// Creates a function template.
    pub fn new(isolate: &isolate::Isolate,
               context: &context::Context,
               callback: Box<value::FunctionCallback>)
               -> FunctionTemplate {
        let raw = unsafe {
            let callback_ptr = Box::into_raw(Box::new(callback));
            let callback_ext = value::External::new::<Box<value::FunctionCallback>>(&isolate, callback_ptr);

            let template = ObjectTemplate::new(isolate);
            template.set_internal_field_count(1);

            let closure = template.new_instance(context);
            closure.set_internal_field(0, &callback_ext);

            util::invoke_ctx(isolate, context, |c| {
                    v8::v8_FunctionTemplate_New(c,
                                             context.as_raw(),
                                             Some(util::callback),
                                             (&closure as &value::Value).as_raw(),
                                             ptr::null_mut(),
                                             0,
                                             v8::ConstructorBehavior::ConstructorBehavior_kAllow)
                })
                .unwrap()
        };
        FunctionTemplate(isolate.clone(), raw)
    }

    /// Returns the unique function instance in the current execution context.
    pub fn get_function(self, context: &context::Context) -> value::Function {
        unsafe {
            let raw = util::invoke_ctx(&self.0, context, |c| {
                    v8::v8_FunctionTemplate_GetFunction(c, self.1, context.as_raw())
                })
                .unwrap();
            value::Function::from_raw(&self.0, raw)
        }
    }
}

impl ObjectTemplate {
    /// Creates an ObjectTemplate.
    pub fn new(isolate: &isolate::Isolate) -> ObjectTemplate {
        let raw = unsafe {
            util::invoke(isolate,
                         |c| v8::v8_ObjectTemplate_New(c, isolate.as_raw(), ptr::null_mut()))
                .unwrap()
        };
        ObjectTemplate(isolate.clone(), raw)
    }

    /// Sets the number of internal fields for objects generated from this template.
    pub fn set_internal_field_count(&self, value: usize) {
        unsafe {
            util::invoke(&self.0, |c| {
                    v8::v8_ObjectTemplate_SetInternalFieldCount(c, self.1, value as os::raw::c_int)
                })
                .unwrap()
        };
    }

    pub fn set(&self, name: &str, value: &value::Data) {
        let cname = ffi::CString::new(name).unwrap();
        let template: &Template = self;
        unsafe {
            util::invoke(&self.0, |c| {
                    v8::v8_Template_Set_Raw(c,
                                         template.1,
                                         self.0.as_raw(),
                                         mem::transmute(cname.as_ptr()),
                                         value.as_raw())
                })
                .unwrap()
        };
    }

    /// Creates a new object instance based off of this template.
    pub fn new_instance(&self, context: &context::Context) -> value::Object {
        unsafe {
            let raw = util::invoke_ctx(&self.0, context, |c| {
                    v8::v8_ObjectTemplate_NewInstance(c, self.1, context.as_raw())
                })
                .unwrap();
            value::Object::from_raw(&self.0, raw)
        }
    }

    /// Creates an object template from a set of raw pointers.
    pub unsafe fn from_raw(isolate: &isolate::Isolate,
                           raw: v8::ObjectTemplateRef)
                           -> ObjectTemplate {
        ObjectTemplate(isolate.clone(), raw)
    }

    /// Returns the underlying raw pointer behind this object template.
    pub fn as_raw(&self) -> v8::ObjectTemplateRef {
        self.1
    }
}

inherit!(Template, Data);
inherit!(ObjectTemplate, Template);
inherit!(FunctionTemplate, Template);
inherit!(Signature, Data);

reference!(Template, v8::v8_Template_CloneRef, v8::v8_Template_DestroyRef);
reference!(FunctionTemplate,
           v8::v8_FunctionTemplate_CloneRef,
           v8::v8_FunctionTemplate_DestroyRef);
reference!(ObjectTemplate,
           v8::v8_ObjectTemplate_CloneRef,
           v8::v8_ObjectTemplate_DestroyRef);
reference!(Signature, v8::v8_Signature_CloneRef, v8::v8_Signature_DestroyRef);