2009-11-11 Yaar Schnitman <yaar@chromium.org>
[WebKit-https.git] / WebKit / chromium / src / WebBindings.cpp
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 "npruntime_impl.h"
35 #include "npruntime_priv.h"
36
37 #include "WebDragData.h"
38 #include "WebRange.h"
39
40 #if USE(V8)
41 #include "ChromiumDataObject.h"
42 #include "ClipboardChromium.h"
43 #include "EventNames.h"
44 #include "MouseEvent.h"
45 #include "NPV8Object.h"  // for PrivateIdentifier
46 #include "Range.h"
47 #include "V8DOMWrapper.h"
48 #include "V8Helpers.h"
49 #include "V8Proxy.h"
50 #elif USE(JSC)
51 #include "bridge/c/c_utility.h"
52 #endif
53
54 #if USE(JAVASCRIPTCORE_BINDINGS)
55 using JSC::Bindings::PrivateIdentifier;
56 #endif
57
58 using namespace WebCore;
59
60 namespace WebKit {
61
62 bool WebBindings::construct(NPP npp, NPObject *npobj, const NPVariant *args, uint32_t argCount, NPVariant* result)
63 {
64     return _NPN_Construct(npp, npobj, args, argCount, result);
65 }
66
67 NPObject* WebBindings::createObject(NPP npp, NPClass* npClass)
68 {
69     return _NPN_CreateObject(npp, npClass);
70 }
71
72 bool WebBindings::enumerate(NPP id, NPObject* obj, NPIdentifier** identifier, uint32_t* val)
73 {
74     return _NPN_Enumerate(id, obj, identifier, val);
75 }
76
77 bool WebBindings::evaluate(NPP npp, NPObject* npObject, NPString* npScript, NPVariant* result)
78 {
79     return _NPN_Evaluate(npp, npObject, npScript, result);
80 }
81
82 bool WebBindings::evaluateHelper(NPP npp, bool popups_allowed, NPObject* npobj, NPString* npscript, NPVariant* result)
83 {
84     return _NPN_EvaluateHelper(npp, popups_allowed, npobj, npscript, result);
85 }
86
87 NPIdentifier WebBindings::getIntIdentifier(int32_t number)
88 {
89     return _NPN_GetIntIdentifier(number);
90 }
91
92 bool WebBindings::getProperty(NPP npp, NPObject* obj, NPIdentifier propertyName, NPVariant *result)
93 {
94     return _NPN_GetProperty(npp, obj, propertyName, result);
95 }
96
97 NPIdentifier WebBindings::getStringIdentifier(const NPUTF8* string)
98 {
99     return _NPN_GetStringIdentifier(string);
100 }
101
102 void WebBindings::getStringIdentifiers(const NPUTF8** names, int32_t nameCount, NPIdentifier* identifiers)
103 {
104     _NPN_GetStringIdentifiers(names, nameCount, identifiers);
105 }
106
107 bool WebBindings::hasMethod(NPP npp, NPObject* npObject, NPIdentifier methodName)
108 {
109     return _NPN_HasMethod(npp, npObject, methodName);
110 }
111
112 bool WebBindings::hasProperty(NPP npp, NPObject* npObject, NPIdentifier propertyName)
113 {
114     return _NPN_HasProperty(npp, npObject, propertyName);
115 }
116
117 bool WebBindings::identifierIsString(NPIdentifier identifier)
118 {
119     return _NPN_IdentifierIsString(identifier);
120 }
121
122 int32_t WebBindings::intFromIdentifier(NPIdentifier identifier)
123 {
124     return _NPN_IntFromIdentifier(identifier);
125 }
126
127 void WebBindings::initializeVariantWithStringCopy(NPVariant* variant, const NPString* value)
128 {
129 #if USE(V8)
130     _NPN_InitializeVariantWithStringCopy(variant, value);
131 #else
132     NPN_InitializeVariantWithStringCopy(variant, value);
133 #endif
134 }
135
136 bool WebBindings::invoke(NPP npp, NPObject* npObject, NPIdentifier methodName, const NPVariant* arguments, uint32_t argumentCount, NPVariant* result)
137 {
138     return _NPN_Invoke(npp, npObject, methodName, arguments, argumentCount, result);
139 }
140
141 bool WebBindings::invokeDefault(NPP id, NPObject* obj, const NPVariant* args, uint32_t count, NPVariant* result)
142 {
143     return _NPN_InvokeDefault(id, obj, args, count, result);
144 }
145
146 void WebBindings::releaseObject(NPObject* npObject)
147 {
148     return _NPN_ReleaseObject(npObject);
149 }
150
151 void WebBindings::releaseVariantValue(NPVariant* variant)
152 {
153     _NPN_ReleaseVariantValue(variant);
154 }
155
156 bool WebBindings::removeProperty(NPP id, NPObject* object, NPIdentifier identifier)
157 {
158     return  _NPN_RemoveProperty(id, object, identifier);
159 }
160
161 NPObject* WebBindings::retainObject(NPObject* npObject)
162 {
163     return _NPN_RetainObject(npObject);
164 }
165
166 void WebBindings::setException(NPObject* obj, const NPUTF8* message)
167 {
168     _NPN_SetException(obj, message);
169 }
170
171 bool WebBindings::setProperty(NPP id, NPObject* obj, NPIdentifier identifier, const NPVariant* variant)
172 {
173     return _NPN_SetProperty(id, obj, identifier, variant);
174 }
175
176 void WebBindings::unregisterObject(NPObject* npObject)
177 {
178 #if USE(V8)
179     _NPN_UnregisterObject(npObject);
180 #endif
181 }
182
183 NPUTF8* WebBindings::utf8FromIdentifier(NPIdentifier identifier)
184 {
185     return _NPN_UTF8FromIdentifier(identifier);
186 }
187
188 void WebBindings::extractIdentifierData(const NPIdentifier& identifier, const NPUTF8*& string, int32_t& number, bool& isString)
189 {
190     PrivateIdentifier* priv = static_cast<PrivateIdentifier*>(identifier);
191     if (!priv) {
192         isString = false;
193         number = 0;
194         return;
195     }
196
197     isString = priv->isString;
198     if (isString)
199         string = priv->value.string;
200     else
201         number = priv->value.number;
202 }
203
204 #if USE(V8)
205
206 static v8::Local<v8::Value> getEvent(const v8::Handle<v8::Context>& context)
207 {
208     static v8::Persistent<v8::String> eventSymbol(v8::Persistent<v8::String>::New(v8::String::NewSymbol("event")));
209     return context->Global()->GetHiddenValue(eventSymbol);
210 }
211
212 static bool getDragDataImpl(NPObject* npobj, int* eventId, WebDragData* data)
213 {
214     if (!npobj)
215         return false;
216     if (npobj->_class != npScriptObjectClass)
217         return false;
218
219     v8::HandleScope handleScope;
220     v8::Handle<v8::Context> context = v8::Context::GetEntered();
221     if (context.IsEmpty())
222         return false;
223
224     // Get the current WebCore event.
225     v8::Handle<v8::Value> currentEvent(getEvent(context));
226     Event* event = V8DOMWrapper::convertToNativeEvent(currentEvent);
227     if (!event)
228         return false;
229
230     // Check that the given npobj is that event.
231     V8NPObject* object = reinterpret_cast<V8NPObject*>(npobj);
232     Event* given = V8DOMWrapper::convertToNativeEvent(object->v8Object);
233     if (given != event)
234         return false;
235
236     // Check the execution frames are same origin.
237     V8Proxy* current = V8Proxy::retrieve(V8Proxy::retrieveFrameForCurrentContext());
238     Frame* frame = V8Proxy::retrieveFrame(context);
239     if (!current || !current->canAccessFrame(frame, false))
240         return false;
241
242     const EventNames& names(eventNames());
243     const AtomicString& eventType(event->type());
244
245     enum DragTargetMouseEventId {
246         DragEnterId = 1, DragOverId = 2, DragLeaveId = 3, DropId = 4
247     };
248
249     // The event type should be a drag event.
250     if (eventType == names.dragenterEvent)
251         *eventId = DragEnterId;
252     else if (eventType == names.dragoverEvent)
253         *eventId = DragOverId;
254     else if (eventType == names.dragleaveEvent)
255         *eventId = DragLeaveId;
256     else if (eventType == names.dropEvent)
257         *eventId = DropId;
258     else
259         return false;
260
261     // Drag events are mouse events and should have a clipboard.
262     MouseEvent* me = static_cast<MouseEvent*>(event);
263     Clipboard* clipboard = me->clipboard();
264     if (!clipboard)
265         return false;
266
267     // And that clipboard should be accessible by WebKit policy.
268     ClipboardChromium* chrome = static_cast<ClipboardChromium*>(clipboard);
269     HashSet<String> accessible(chrome->types());
270     if (accessible.isEmpty())
271         return false;
272
273     RefPtr<ChromiumDataObject> dataObject(chrome->dataObject());
274     if (dataObject && data)
275         *data = WebDragData(dataObject);
276
277     return dataObject;
278 }
279
280 static bool getRangeImpl(NPObject* npobj, WebRange* range)
281 {
282     V8NPObject* v8npobject = reinterpret_cast<V8NPObject*>(npobj);
283     v8::Handle<v8::Object> v8object(v8npobject->v8Object);
284     if (V8ClassIndex::RANGE != V8DOMWrapper::domWrapperType(v8object))
285         return false;
286
287     Range* native = V8DOMWrapper::convertToNativeObject<WebCore::Range>(V8ClassIndex::RANGE, v8object);
288     if (!native)
289         return false;
290
291     *range = WebRange(native);
292     return true;
293 }
294
295 #endif
296
297 bool WebBindings::getDragData(NPObject* event, int* eventId, WebDragData* data)
298 {
299 #if USE(V8)
300     return getDragDataImpl(event, eventId, data);
301 #else
302     // Not supported on other ports (JSC, etc).
303     return false;
304 #endif
305 }
306
307 bool WebBindings::isDragEvent(NPObject* event)
308 {
309     int eventId;
310     return getDragData(event, &eventId, 0);
311 }
312
313 bool WebBindings::getRange(NPObject* range, WebRange* webrange)
314 {
315 #if USE(V8)
316     return getRangeImpl(range, webrange);
317 #else
318     // Not supported on other ports (JSC, etc).
319     return false;
320 #endif
321 }
322
323 } // namespace WebKit