Home | History | Annotate | Download | only in web
      1 /*
      2  * Copyright (C) 2009 Google Inc. All rights reserved.
      3  *
      4  * Redistribution and use in source and binary forms, with or without
      5  * modification, are permitted provided that the following conditions are
      6  * met:
      7  *
      8  *     * Redistributions of source code must retain the above copyright
      9  * notice, this list of conditions and the following disclaimer.
     10  *     * Redistributions in binary form must reproduce the above
     11  * copyright notice, this list of conditions and the following disclaimer
     12  * in the documentation and/or other materials provided with the
     13  * distribution.
     14  *     * Neither the name of Google Inc. nor the names of its
     15  * contributors may be used to endorse or promote products derived from
     16  * this software without specific prior written permission.
     17  *
     18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
     19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
     20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
     21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
     22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
     23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
     24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     29  */
     30 
     31 #include "config.h"
     32 #include "WebBindings.h"
     33 
     34 #include "V8Element.h"
     35 #include "V8Range.h"
     36 #include "WebArrayBufferView.h"
     37 #include "WebElement.h"
     38 #include "WebRange.h"
     39 #include "bindings/v8/NPV8Object.h"  // for PrivateIdentifier
     40 #include "bindings/v8/ScriptController.h"
     41 #include "bindings/v8/V8DOMWrapper.h"
     42 #include "bindings/v8/V8NPUtils.h"
     43 #include "bindings/v8/custom/V8ArrayBufferCustom.h"
     44 #include "bindings/v8/custom/V8ArrayBufferViewCustom.h"
     45 #include "bindings/v8/npruntime_impl.h"
     46 #include "bindings/v8/npruntime_priv.h"
     47 #include "core/dom/Range.h"
     48 #include "core/frame/DOMWindow.h"
     49 #include "core/frame/Frame.h"
     50 #include "public/platform/WebArrayBuffer.h"
     51 #include "wtf/ArrayBufferView.h"
     52 
     53 using namespace WebCore;
     54 
     55 namespace blink {
     56 
     57 bool WebBindings::construct(NPP npp, NPObject* object, const NPVariant* args, uint32_t argCount, NPVariant* result)
     58 {
     59     return _NPN_Construct(npp, object, args, argCount, result);
     60 }
     61 
     62 NPObject* WebBindings::createObject(NPP npp, NPClass* npClass)
     63 {
     64     return _NPN_CreateObject(npp, npClass);
     65 }
     66 
     67 bool WebBindings::enumerate(NPP npp, NPObject* object, NPIdentifier** identifier, uint32_t* identifierCount)
     68 {
     69     return _NPN_Enumerate(npp, object, identifier, identifierCount);
     70 }
     71 
     72 bool WebBindings::evaluate(NPP npp, NPObject* object, NPString* script, NPVariant* result)
     73 {
     74     return _NPN_Evaluate(npp, object, script, result);
     75 }
     76 
     77 bool WebBindings::evaluateHelper(NPP npp, bool popupsAllowed, NPObject* object, NPString* script, NPVariant* result)
     78 {
     79     return _NPN_EvaluateHelper(npp, popupsAllowed, object, script, result);
     80 }
     81 
     82 NPIdentifier WebBindings::getIntIdentifier(int32_t number)
     83 {
     84     return _NPN_GetIntIdentifier(number);
     85 }
     86 
     87 bool WebBindings::getProperty(NPP npp, NPObject* object, NPIdentifier property, NPVariant* result)
     88 {
     89     return _NPN_GetProperty(npp, object, property, result);
     90 }
     91 
     92 NPIdentifier WebBindings::getStringIdentifier(const NPUTF8* string)
     93 {
     94     return _NPN_GetStringIdentifier(string);
     95 }
     96 
     97 void WebBindings::getStringIdentifiers(const NPUTF8** names, int32_t nameCount, NPIdentifier* identifiers)
     98 {
     99     _NPN_GetStringIdentifiers(names, nameCount, identifiers);
    100 }
    101 
    102 bool WebBindings::hasMethod(NPP npp, NPObject* object, NPIdentifier method)
    103 {
    104     return _NPN_HasMethod(npp, object, method);
    105 }
    106 
    107 bool WebBindings::hasProperty(NPP npp, NPObject* object, NPIdentifier property)
    108 {
    109     return _NPN_HasProperty(npp, object, property);
    110 }
    111 
    112 bool WebBindings::identifierIsString(NPIdentifier identifier)
    113 {
    114     return _NPN_IdentifierIsString(identifier);
    115 }
    116 
    117 int32_t WebBindings::intFromIdentifier(NPIdentifier identifier)
    118 {
    119     return _NPN_IntFromIdentifier(identifier);
    120 }
    121 
    122 void WebBindings::initializeVariantWithStringCopy(NPVariant* variant, const NPString* value)
    123 {
    124     _NPN_InitializeVariantWithStringCopy(variant, value);
    125 }
    126 
    127 bool WebBindings::invoke(NPP npp, NPObject* object, NPIdentifier method, const NPVariant* args, uint32_t argCount, NPVariant* result)
    128 {
    129     return _NPN_Invoke(npp, object, method, args, argCount, result);
    130 }
    131 
    132 bool WebBindings::invokeDefault(NPP npp, NPObject* object, const NPVariant* args, uint32_t argCount, NPVariant* result)
    133 {
    134     return _NPN_InvokeDefault(npp, object, args, argCount, result);
    135 }
    136 
    137 void WebBindings::releaseObject(NPObject* object)
    138 {
    139     return _NPN_ReleaseObject(object);
    140 }
    141 
    142 void WebBindings::releaseVariantValue(NPVariant* variant)
    143 {
    144     _NPN_ReleaseVariantValue(variant);
    145 }
    146 
    147 bool WebBindings::removeProperty(NPP npp, NPObject* object, NPIdentifier identifier)
    148 {
    149     return _NPN_RemoveProperty(npp, object, identifier);
    150 }
    151 
    152 NPObject* WebBindings::retainObject(NPObject* object)
    153 {
    154     return _NPN_RetainObject(object);
    155 }
    156 
    157 void WebBindings::setException(NPObject* object, const NPUTF8* message)
    158 {
    159     _NPN_SetException(object, message);
    160 }
    161 
    162 bool WebBindings::setProperty(NPP npp, NPObject* object, NPIdentifier identifier, const NPVariant* value)
    163 {
    164     return _NPN_SetProperty(npp, object, identifier, value);
    165 }
    166 
    167 void WebBindings::registerObjectOwner(NPP)
    168 {
    169 }
    170 
    171 void WebBindings::unregisterObjectOwner(NPP)
    172 {
    173 }
    174 
    175 NPP WebBindings::getObjectOwner(NPObject*)
    176 {
    177     return 0;
    178 }
    179 
    180 void WebBindings::unregisterObject(NPObject* object)
    181 {
    182     _NPN_UnregisterObject(object);
    183 }
    184 
    185 NPUTF8* WebBindings::utf8FromIdentifier(NPIdentifier identifier)
    186 {
    187     return _NPN_UTF8FromIdentifier(identifier);
    188 }
    189 
    190 void WebBindings::extractIdentifierData(const NPIdentifier& identifier, const NPUTF8*& string, int32_t& number, bool& isString)
    191 {
    192     PrivateIdentifier* data = static_cast<PrivateIdentifier*>(identifier);
    193     if (!data) {
    194         isString = false;
    195         number = 0;
    196         return;
    197     }
    198 
    199     isString = data->isString;
    200     if (isString)
    201         string = data->value.string;
    202     else
    203         number = data->value.number;
    204 }
    205 
    206 static bool getRangeImpl(NPObject* object, WebRange* webRange, v8::Isolate* isolate)
    207 {
    208     if (!object)
    209         return false;
    210 
    211     V8NPObject* v8NPObject = npObjectToV8NPObject(object);
    212     if (!v8NPObject)
    213         return false;
    214 
    215     v8::HandleScope handleScope(isolate);
    216     v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
    217     if (v8Object.IsEmpty())
    218         return false;
    219     if (!V8Range::wrapperTypeInfo.equals(toWrapperTypeInfo(v8Object)))
    220         return false;
    221 
    222     Range* native = V8Range::hasInstanceInAnyWorld(v8Object, isolate) ? V8Range::toNative(v8Object) : 0;
    223     if (!native)
    224         return false;
    225 
    226     *webRange = WebRange(native);
    227     return true;
    228 }
    229 
    230 static bool getNodeImpl(NPObject* object, WebNode* webNode, v8::Isolate* isolate)
    231 {
    232     if (!object)
    233         return false;
    234 
    235     V8NPObject* v8NPObject = npObjectToV8NPObject(object);
    236     if (!v8NPObject)
    237         return false;
    238 
    239     v8::HandleScope handleScope(isolate);
    240     v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
    241     if (v8Object.IsEmpty())
    242         return false;
    243     Node* native = V8Node::hasInstanceInAnyWorld(v8Object, isolate) ? V8Node::toNative(v8Object) : 0;
    244     if (!native)
    245         return false;
    246 
    247     *webNode = WebNode(native);
    248     return true;
    249 }
    250 
    251 static bool getElementImpl(NPObject* object, WebElement* webElement, v8::Isolate* isolate)
    252 {
    253     if (!object)
    254         return false;
    255 
    256     V8NPObject* v8NPObject = npObjectToV8NPObject(object);
    257     if (!v8NPObject)
    258         return false;
    259 
    260     v8::HandleScope handleScope(isolate);
    261     v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
    262     if (v8Object.IsEmpty())
    263         return false;
    264     Element* native = V8Element::hasInstanceInAnyWorld(v8Object, isolate) ? V8Element::toNative(v8Object) : 0;
    265     if (!native)
    266         return false;
    267 
    268     *webElement = WebElement(native);
    269     return true;
    270 }
    271 
    272 static bool getArrayBufferImpl(NPObject* object, WebArrayBuffer* arrayBuffer, v8::Isolate* isolate)
    273 {
    274     if (!object)
    275         return false;
    276 
    277     V8NPObject* v8NPObject = npObjectToV8NPObject(object);
    278     if (!v8NPObject)
    279         return false;
    280 
    281     v8::HandleScope handleScope(isolate);
    282     v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
    283     if (v8Object.IsEmpty())
    284         return false;
    285     ArrayBuffer* native = V8ArrayBuffer::hasInstanceInAnyWorld(v8Object, isolate) ? V8ArrayBuffer::toNative(v8Object) : 0;
    286     if (!native)
    287         return false;
    288 
    289     *arrayBuffer = WebArrayBuffer(native);
    290     return true;
    291 }
    292 
    293 static bool getArrayBufferViewImpl(NPObject* object, WebArrayBufferView* arrayBufferView, v8::Isolate* isolate)
    294 {
    295     if (!object)
    296         return false;
    297 
    298     V8NPObject* v8NPObject = npObjectToV8NPObject(object);
    299     if (!v8NPObject)
    300         return false;
    301 
    302     v8::HandleScope handleScope(isolate);
    303     v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
    304     if (v8Object.IsEmpty())
    305         return false;
    306     ArrayBufferView* native = V8ArrayBufferView::hasInstanceInAnyWorld(v8Object, isolate) ? V8ArrayBufferView::toNative(v8Object) : 0;
    307     if (!native)
    308         return false;
    309 
    310     *arrayBufferView = WebArrayBufferView(native);
    311     return true;
    312 }
    313 
    314 static NPObject* makeIntArrayImpl(const WebVector<int>& data, v8::Isolate* isolate)
    315 {
    316     v8::HandleScope handleScope(isolate);
    317     v8::Handle<v8::Array> result = v8::Array::New(isolate, data.size());
    318     for (size_t i = 0; i < data.size(); ++i)
    319         result->Set(i, v8::Number::New(isolate, data[i]));
    320 
    321     DOMWindow* window = toDOMWindow(isolate->GetCurrentContext());
    322     return npCreateV8ScriptObject(0, result, window, isolate);
    323 }
    324 
    325 static NPObject* makeStringArrayImpl(const WebVector<WebString>& data, v8::Isolate* isolate)
    326 {
    327     v8::HandleScope handleScope(isolate);
    328     v8::Handle<v8::Array> result = v8::Array::New(isolate, data.size());
    329     for (size_t i = 0; i < data.size(); ++i)
    330         result->Set(i, v8String(isolate, data[i]));
    331 
    332     DOMWindow* window = toDOMWindow(isolate->GetCurrentContext());
    333     return npCreateV8ScriptObject(0, result, window, isolate);
    334 }
    335 
    336 bool WebBindings::getRange(NPObject* range, WebRange* webRange)
    337 {
    338     return getRangeImpl(range, webRange, v8::Isolate::GetCurrent());
    339 }
    340 
    341 bool WebBindings::getArrayBuffer(NPObject* arrayBuffer, WebArrayBuffer* webArrayBuffer)
    342 {
    343     return getArrayBufferImpl(arrayBuffer, webArrayBuffer, v8::Isolate::GetCurrent());
    344 }
    345 
    346 bool WebBindings::getArrayBufferView(NPObject* arrayBufferView, WebArrayBufferView* webArrayBufferView)
    347 {
    348     return getArrayBufferViewImpl(arrayBufferView, webArrayBufferView, v8::Isolate::GetCurrent());
    349 }
    350 
    351 bool WebBindings::getNode(NPObject* node, WebNode* webNode)
    352 {
    353     return getNodeImpl(node, webNode, v8::Isolate::GetCurrent());
    354 }
    355 
    356 bool WebBindings::getElement(NPObject* element, WebElement* webElement)
    357 {
    358     return getElementImpl(element, webElement, v8::Isolate::GetCurrent());
    359 }
    360 
    361 NPObject* WebBindings::makeIntArray(const WebVector<int>& data)
    362 {
    363     return makeIntArrayImpl(data, v8::Isolate::GetCurrent());
    364 }
    365 
    366 NPObject* WebBindings::makeStringArray(const WebVector<WebString>& data)
    367 {
    368     return makeStringArrayImpl(data, v8::Isolate::GetCurrent());
    369 }
    370 
    371 void WebBindings::pushExceptionHandler(ExceptionHandler handler, void* data)
    372 {
    373     WebCore::pushExceptionHandler(handler, data);
    374 }
    375 
    376 void WebBindings::popExceptionHandler()
    377 {
    378     WebCore::popExceptionHandler();
    379 }
    380 
    381 void WebBindings::toNPVariant(v8::Local<v8::Value> object, NPObject* root, NPVariant* result)
    382 {
    383     WebCore::convertV8ObjectToNPVariant(object, root, result, v8::Isolate::GetCurrent());
    384 }
    385 
    386 v8::Handle<v8::Value> WebBindings::toV8Value(const NPVariant* variant)
    387 {
    388     v8::Isolate* isolate = v8::Isolate::GetCurrent();
    389     if (variant->type == NPVariantType_Object) {
    390         NPObject* object = NPVARIANT_TO_OBJECT(*variant);
    391         V8NPObject* v8Object = npObjectToV8NPObject(object);
    392         if (!v8Object)
    393             return v8::Undefined(isolate);
    394         return convertNPVariantToV8Object(variant, v8Object->rootObject->frame()->script().windowScriptNPObject(), isolate);
    395     }
    396     // Safe to pass 0 since we have checked the script object class to make sure the
    397     // argument is a primitive v8 type.
    398     return convertNPVariantToV8Object(variant, 0, isolate);
    399 }
    400 
    401 } // namespace blink
    402