Home | History | Annotate | Download | only in gin
      1 // Copyright 2013 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "gin/object_template_builder.h"
      6 
      7 #include "gin/interceptor.h"
      8 #include "gin/per_isolate_data.h"
      9 #include "gin/public/wrapper_info.h"
     10 
     11 namespace gin {
     12 
     13 namespace {
     14 
     15 WrappableBase* WrappableFromV8(v8::Isolate* isolate,
     16                                v8::Handle<v8::Value> val) {
     17   if (!val->IsObject())
     18     return NULL;
     19   v8::Handle<v8::Object> obj = v8::Handle<v8::Object>::Cast(val);
     20   WrapperInfo* info = WrapperInfo::From(obj);
     21 
     22   // If this fails, the object is not managed by Gin.
     23   if (!info)
     24     return NULL;
     25 
     26   // We don't further validate the type of the object, but assume it's derived
     27   // from WrappableBase. We look up the pointer in a global registry, to make
     28   // sure it's actually pointed to a valid life object.
     29   return static_cast<WrappableBase*>(
     30       obj->GetAlignedPointerFromInternalField(kEncodedValueIndex));
     31 }
     32 
     33 NamedPropertyInterceptor* NamedInterceptorFromV8(v8::Isolate* isolate,
     34                                                  v8::Handle<v8::Value> val) {
     35   WrappableBase* base = WrappableFromV8(isolate, val);
     36   if (!base)
     37     return NULL;
     38   return PerIsolateData::From(isolate)->GetNamedPropertyInterceptor(base);
     39 }
     40 
     41 IndexedPropertyInterceptor* IndexedInterceptorFromV8(
     42     v8::Isolate* isolate,
     43     v8::Handle<v8::Value> val) {
     44   WrappableBase* base = WrappableFromV8(isolate, val);
     45   if (!base)
     46     return NULL;
     47   return PerIsolateData::From(isolate)->GetIndexedPropertyInterceptor(base);
     48 }
     49 
     50 void NamedPropertyGetter(v8::Local<v8::String> property,
     51                          const v8::PropertyCallbackInfo<v8::Value>& info) {
     52   v8::Isolate* isolate = info.GetIsolate();
     53   NamedPropertyInterceptor* interceptor =
     54       NamedInterceptorFromV8(isolate, info.Holder());
     55   if (!interceptor)
     56     return;
     57   std::string name;
     58   ConvertFromV8(isolate, property, &name);
     59   info.GetReturnValue().Set(interceptor->GetNamedProperty(isolate, name));
     60 }
     61 
     62 void NamedPropertySetter(v8::Local<v8::String> property,
     63                          v8::Local<v8::Value> value,
     64                          const v8::PropertyCallbackInfo<v8::Value>& info) {
     65   v8::Isolate* isolate = info.GetIsolate();
     66   NamedPropertyInterceptor* interceptor =
     67       NamedInterceptorFromV8(isolate, info.Holder());
     68   if (!interceptor)
     69     return;
     70   std::string name;
     71   ConvertFromV8(isolate, property, &name);
     72   if (interceptor->SetNamedProperty(isolate, name, value))
     73     info.GetReturnValue().Set(value);
     74 }
     75 
     76 void NamedPropertyQuery(v8::Local<v8::String> property,
     77                         const v8::PropertyCallbackInfo<v8::Integer>& info) {
     78   v8::Isolate* isolate = info.GetIsolate();
     79   NamedPropertyInterceptor* interceptor =
     80       NamedInterceptorFromV8(isolate, info.Holder());
     81   if (!interceptor)
     82     return;
     83   std::string name;
     84   ConvertFromV8(isolate, property, &name);
     85   if (interceptor->GetNamedProperty(isolate, name).IsEmpty())
     86     return;
     87   info.GetReturnValue().Set(0);
     88 }
     89 
     90 void NamedPropertyEnumerator(const v8::PropertyCallbackInfo<v8::Array>& info) {
     91   v8::Isolate* isolate = info.GetIsolate();
     92   NamedPropertyInterceptor* interceptor =
     93       NamedInterceptorFromV8(isolate, info.Holder());
     94   if (!interceptor)
     95     return;
     96   info.GetReturnValue().Set(v8::Handle<v8::Array>::Cast(
     97       ConvertToV8(isolate, interceptor->EnumerateNamedProperties(isolate))));
     98 }
     99 
    100 void IndexedPropertyGetter(uint32_t index,
    101                            const v8::PropertyCallbackInfo<v8::Value>& info) {
    102   v8::Isolate* isolate = info.GetIsolate();
    103   IndexedPropertyInterceptor* interceptor =
    104       IndexedInterceptorFromV8(isolate, info.Holder());
    105   if (!interceptor)
    106     return;
    107   info.GetReturnValue().Set(interceptor->GetIndexedProperty(isolate, index));
    108 }
    109 
    110 void IndexedPropertySetter(uint32_t index,
    111                            v8::Local<v8::Value> value,
    112                            const v8::PropertyCallbackInfo<v8::Value>& info) {
    113   v8::Isolate* isolate = info.GetIsolate();
    114   IndexedPropertyInterceptor* interceptor =
    115       IndexedInterceptorFromV8(isolate, info.Holder());
    116   if (!interceptor)
    117     return;
    118   if (interceptor->SetIndexedProperty(isolate, index, value))
    119     info.GetReturnValue().Set(value);
    120 }
    121 
    122 void IndexedPropertyEnumerator(
    123     const v8::PropertyCallbackInfo<v8::Array>& info) {
    124   v8::Isolate* isolate = info.GetIsolate();
    125   IndexedPropertyInterceptor* interceptor =
    126       IndexedInterceptorFromV8(isolate, info.Holder());
    127   if (!interceptor)
    128     return;
    129   info.GetReturnValue().Set(v8::Handle<v8::Array>::Cast(
    130       ConvertToV8(isolate, interceptor->EnumerateIndexedProperties(isolate))));
    131 }
    132 
    133 }  // namespace
    134 
    135 ObjectTemplateBuilder::ObjectTemplateBuilder(v8::Isolate* isolate)
    136     : isolate_(isolate), template_(v8::ObjectTemplate::New(isolate)) {
    137   template_->SetInternalFieldCount(kNumberOfInternalFields);
    138 }
    139 
    140 ObjectTemplateBuilder::~ObjectTemplateBuilder() {
    141 }
    142 
    143 ObjectTemplateBuilder& ObjectTemplateBuilder::AddNamedPropertyInterceptor() {
    144   template_->SetNamedPropertyHandler(&NamedPropertyGetter,
    145                                      &NamedPropertySetter,
    146                                      &NamedPropertyQuery,
    147                                      NULL,
    148                                      &NamedPropertyEnumerator);
    149   return *this;
    150 }
    151 
    152 ObjectTemplateBuilder& ObjectTemplateBuilder::AddIndexedPropertyInterceptor() {
    153   template_->SetIndexedPropertyHandler(&IndexedPropertyGetter,
    154                                        &IndexedPropertySetter,
    155                                        NULL,
    156                                        NULL,
    157                                        &IndexedPropertyEnumerator);
    158   return *this;
    159 }
    160 
    161 ObjectTemplateBuilder& ObjectTemplateBuilder::SetImpl(
    162     const base::StringPiece& name, v8::Handle<v8::Data> val) {
    163   template_->Set(StringToSymbol(isolate_, name), val);
    164   return *this;
    165 }
    166 
    167 ObjectTemplateBuilder& ObjectTemplateBuilder::SetPropertyImpl(
    168     const base::StringPiece& name, v8::Handle<v8::FunctionTemplate> getter,
    169     v8::Handle<v8::FunctionTemplate> setter) {
    170   template_->SetAccessorProperty(StringToSymbol(isolate_, name), getter,
    171                                  setter);
    172   return *this;
    173 }
    174 
    175 v8::Local<v8::ObjectTemplate> ObjectTemplateBuilder::Build() {
    176   v8::Local<v8::ObjectTemplate> result = template_;
    177   template_.Clear();
    178   return result;
    179 }
    180 
    181 }  // namespace gin
    182