Add setJSWrapperForActiveDOMNode and use it for Nodes that are also ActiveDOMObjects
[WebKit-https.git] / Source / WebCore / bindings / v8 / V8DOMWrapper.h
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 #ifndef V8DOMWrapper_h
32 #define V8DOMWrapper_h
33
34 #include "DOMDataStore.h"
35 #include "Event.h"
36 #include "IsolatedWorld.h"
37 #include "Node.h"
38 #include "NodeFilter.h"
39 #include "PlatformString.h"
40 #include "V8CustomXPathNSResolver.h"
41 #include "V8DOMMap.h"
42 #include "V8Event.h"
43 #include "V8IsolatedContext.h"
44 #include "V8Utilities.h"
45 #include "V8XPathNSResolver.h"
46 #include "WrapperTypeInfo.h"
47 #include "XPathNSResolver.h"
48 #include <v8.h>
49 #include <wtf/MainThread.h>
50 #include <wtf/PassRefPtr.h>
51
52 namespace WebCore {
53
54     class DOMWindow;
55     class EventTarget;
56     class Frame;
57     class Node;
58     class V8BindingPerContextData;
59     class V8Proxy;
60     class WorkerContext;
61
62     enum ListenerLookupType {
63         ListenerFindOnly,
64         ListenerFindOrCreate,
65     };
66
67     class V8DOMWrapper {
68     public:
69 #ifndef NDEBUG
70         // Checks if a v8 value can be a DOM wrapper
71         static bool maybeDOMWrapper(v8::Handle<v8::Value>);
72 #endif
73
74         // Sets contents of a DOM wrapper.
75         static void setDOMWrapper(v8::Handle<v8::Object> object, WrapperTypeInfo* type, void* cptr)
76         {
77             ASSERT(object->InternalFieldCount() >= 2);
78             object->SetPointerInInternalField(v8DOMWrapperObjectIndex, cptr);
79             object->SetPointerInInternalField(v8DOMWrapperTypeIndex, type);
80         }
81
82         static v8::Handle<v8::Object> lookupDOMWrapper(v8::Handle<v8::FunctionTemplate> functionTemplate, v8::Handle<v8::Object> object)
83         {
84             return object.IsEmpty() ? object : object->FindInstanceInPrototypeChain(functionTemplate);
85         }
86
87         static WrapperTypeInfo* domWrapperType(v8::Handle<v8::Object>);
88
89         static v8::Handle<v8::Value> convertEventTargetToV8Object(PassRefPtr<EventTarget> eventTarget)
90         {
91             return convertEventTargetToV8Object(eventTarget.get());
92         }
93
94         static v8::Handle<v8::Value> convertEventTargetToV8Object(EventTarget*);
95
96         static PassRefPtr<EventListener> getEventListener(v8::Local<v8::Value> value, bool isAttribute, ListenerLookupType lookup);
97
98         // XPath-related utilities
99         static RefPtr<XPathNSResolver> getXPathNSResolver(v8::Handle<v8::Value> value, V8Proxy* proxy = 0);
100
101         // Wrap JS node filter in C++.
102         static PassRefPtr<NodeFilter> wrapNativeNodeFilter(v8::Handle<v8::Value>);
103
104         static v8::Local<v8::Function> constructorForType(WrapperTypeInfo*, DOMWindow*);
105 #if ENABLE(WORKERS)
106         static v8::Local<v8::Function> constructorForType(WrapperTypeInfo*, WorkerContext*);
107 #endif
108
109         template<typename T> static void setJSWrapperForDOMObject(PassRefPtr<T>, v8::Persistent<v8::Object>);
110         template<typename T> static void setJSWrapperForActiveDOMObject(PassRefPtr<T>, v8::Persistent<v8::Object>);
111         static void setJSWrapperForDOMNode(PassRefPtr<Node>, v8::Persistent<v8::Object>);
112         static void setJSWrapperForActiveDOMNode(PassRefPtr<Node>, v8::Persistent<v8::Object>);
113
114         static bool isValidDOMObject(v8::Handle<v8::Value>);
115
116         // Check whether a V8 value is a wrapper of type |classType|.
117         static bool isWrapperOfType(v8::Handle<v8::Value>, WrapperTypeInfo*);
118
119         // Proper object lifetime support.
120         //
121         // Helper functions to make sure the child object stays alive
122         // while the parent is alive. Using the name more than once
123         // overwrites previous references making it possible to free
124         // old children.
125         static void setNamedHiddenReference(v8::Handle<v8::Object> parent, const char* name, v8::Handle<v8::Value> child);
126         static void setNamedHiddenWindowReference(Frame*, const char*, v8::Handle<v8::Value>);
127
128         static v8::Local<v8::Object> instantiateV8Object(V8Proxy* proxy, WrapperTypeInfo*, void* impl);
129
130         static v8::Handle<v8::Object> getCachedWrapper(Node* node)
131         {
132             ASSERT(isMainThread());
133             if (LIKELY(!IsolatedWorld::count())) {
134                 v8::Persistent<v8::Object>* wrapper = node->wrapper();
135                 if (LIKELY(!!wrapper))
136                     return *wrapper;
137             }
138
139             V8IsolatedContext* context = V8IsolatedContext::getEntered();
140             if (LIKELY(!context)) {
141                 v8::Persistent<v8::Object>* wrapper = node->wrapper();
142                 if (!wrapper)
143                     return v8::Handle<v8::Object>();
144                 return *wrapper;
145             }
146             DOMDataStore* store = context->world()->domDataStore();
147             DOMNodeMapping& domNodeMap = node->isActiveNode() ? store->activeDomNodeMap() : store->domNodeMap();
148             return domNodeMap.get(node);
149         }
150     private:
151         static V8BindingPerContextData* perContextData(V8Proxy*);
152 #if ENABLE(WORKERS)
153         static V8BindingPerContextData* perContextData(WorkerContext*);
154 #endif
155     };
156
157     template<typename T>
158     void V8DOMWrapper::setJSWrapperForDOMObject(PassRefPtr<T> object, v8::Persistent<v8::Object> wrapper)
159     {
160         ASSERT(maybeDOMWrapper(wrapper));
161         ASSERT(!domWrapperType(wrapper)->toActiveDOMObjectFunction);
162         getDOMObjectMap().set(object.leakRef(), wrapper);
163     }
164
165     template<typename T>
166     void V8DOMWrapper::setJSWrapperForActiveDOMObject(PassRefPtr<T> object, v8::Persistent<v8::Object> wrapper)
167     {
168         ASSERT(maybeDOMWrapper(wrapper));
169         ASSERT(domWrapperType(wrapper)->toActiveDOMObjectFunction);
170         getActiveDOMObjectMap().set(object.leakRef(), wrapper);
171     }
172 }
173
174 #endif // V8DOMWrapper_h