OLD | NEW |
| (Empty) |
1 /* | |
2 * Copyright (C) 2013 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 "bindings/v8/CustomElementConstructorBuilder.h" | |
33 | |
34 #include "bindings/core/v8/V8Document.h" | |
35 #include "bindings/v8/CustomElementBinding.h" | |
36 #include "bindings/v8/DOMWrapperWorld.h" | |
37 #include "bindings/v8/Dictionary.h" | |
38 #include "bindings/v8/ExceptionState.h" | |
39 #include "bindings/v8/V8Binding.h" | |
40 #include "bindings/v8/V8HiddenValue.h" | |
41 #include "bindings/v8/V8PerContextData.h" | |
42 #include "core/HTMLNames.h" | |
43 #include "core/SVGNames.h" | |
44 #include "core/V8HTMLElementWrapperFactory.h" // FIXME: should be bindings/core/
v8 | |
45 #include "core/V8SVGElementWrapperFactory.h" // FIXME: should be bindings/core/v
8 | |
46 #include "core/dom/Document.h" | |
47 #include "core/dom/custom/CustomElementCallbackDispatcher.h" | |
48 #include "core/dom/custom/CustomElementDefinition.h" | |
49 #include "core/dom/custom/CustomElementDescriptor.h" | |
50 #include "core/dom/custom/CustomElementException.h" | |
51 #include "wtf/Assertions.h" | |
52 | |
53 namespace WebCore { | |
54 | |
55 static void constructCustomElement(const v8::FunctionCallbackInfo<v8::Value>&); | |
56 | |
57 CustomElementConstructorBuilder::CustomElementConstructorBuilder(ScriptState* sc
riptState, const Dictionary* options) | |
58 : m_scriptState(scriptState) | |
59 , m_options(options) | |
60 , m_wrapperType(0) | |
61 { | |
62 ASSERT(m_scriptState->context() == m_scriptState->isolate()->GetCurrentConte
xt()); | |
63 } | |
64 | |
65 bool CustomElementConstructorBuilder::isFeatureAllowed() const | |
66 { | |
67 return m_scriptState->world().isMainWorld(); | |
68 } | |
69 | |
70 bool CustomElementConstructorBuilder::validateOptions(const AtomicString& type,
QualifiedName& tagName, ExceptionState& exceptionState) | |
71 { | |
72 ASSERT(m_prototype.IsEmpty()); | |
73 | |
74 v8::TryCatch tryCatch; | |
75 | |
76 ScriptValue prototypeScriptValue; | |
77 if (m_options->get("prototype", prototypeScriptValue) && !prototypeScriptVal
ue.isNull()) { | |
78 ASSERT(!tryCatch.HasCaught()); | |
79 if (!prototypeScriptValue.isObject()) { | |
80 CustomElementException::throwException(CustomElementException::Proto
typeNotAnObject, type, exceptionState); | |
81 tryCatch.ReThrow(); | |
82 return false; | |
83 } | |
84 m_prototype = prototypeScriptValue.v8Value().As<v8::Object>(); | |
85 } else if (!tryCatch.HasCaught()) { | |
86 m_prototype = v8::Object::New(m_scriptState->isolate()); | |
87 v8::Local<v8::Object> basePrototype = m_scriptState->perContextData()->p
rototypeForType(&V8HTMLElement::wrapperTypeInfo); | |
88 if (!basePrototype.IsEmpty()) | |
89 m_prototype->SetPrototype(basePrototype); | |
90 } | |
91 | |
92 if (tryCatch.HasCaught()) { | |
93 tryCatch.ReThrow(); | |
94 return false; | |
95 } | |
96 | |
97 AtomicString extends; | |
98 bool extendsProvidedAndNonNull = m_options->get("extends", extends); | |
99 | |
100 if (tryCatch.HasCaught()) { | |
101 tryCatch.ReThrow(); | |
102 return false; | |
103 } | |
104 | |
105 if (!m_scriptState->perContextData()) { | |
106 // FIXME: This should generate an InvalidContext exception at a later po
int. | |
107 CustomElementException::throwException(CustomElementException::ContextDe
stroyedCheckingPrototype, type, exceptionState); | |
108 tryCatch.ReThrow(); | |
109 return false; | |
110 } | |
111 | |
112 AtomicString namespaceURI = HTMLNames::xhtmlNamespaceURI; | |
113 if (hasValidPrototypeChainFor(&V8SVGElement::wrapperTypeInfo)) | |
114 namespaceURI = SVGNames::svgNamespaceURI; | |
115 | |
116 ASSERT(!tryCatch.HasCaught()); | |
117 | |
118 AtomicString localName; | |
119 | |
120 if (extendsProvidedAndNonNull) { | |
121 localName = extends.lower(); | |
122 | |
123 if (!Document::isValidName(localName)) { | |
124 CustomElementException::throwException(CustomElementException::Exten
dsIsInvalidName, type, exceptionState); | |
125 tryCatch.ReThrow(); | |
126 return false; | |
127 } | |
128 if (CustomElement::isValidName(localName)) { | |
129 CustomElementException::throwException(CustomElementException::Exten
dsIsCustomElementName, type, exceptionState); | |
130 tryCatch.ReThrow(); | |
131 return false; | |
132 } | |
133 } else { | |
134 if (namespaceURI == SVGNames::svgNamespaceURI) { | |
135 CustomElementException::throwException(CustomElementException::Exten
dsIsInvalidName, type, exceptionState); | |
136 tryCatch.ReThrow(); | |
137 return false; | |
138 } | |
139 localName = type; | |
140 } | |
141 | |
142 if (!extendsProvidedAndNonNull) | |
143 m_wrapperType = &V8HTMLElement::wrapperTypeInfo; | |
144 else if (namespaceURI == HTMLNames::xhtmlNamespaceURI) | |
145 m_wrapperType = findWrapperTypeForHTMLTagName(localName); | |
146 else | |
147 m_wrapperType = findWrapperTypeForSVGTagName(localName); | |
148 | |
149 ASSERT(!tryCatch.HasCaught()); | |
150 ASSERT(m_wrapperType); | |
151 tagName = QualifiedName(nullAtom, localName, namespaceURI); | |
152 return m_wrapperType; | |
153 } | |
154 | |
155 PassRefPtr<CustomElementLifecycleCallbacks> CustomElementConstructorBuilder::cre
ateCallbacks() | |
156 { | |
157 ASSERT(!m_prototype.IsEmpty()); | |
158 | |
159 v8::TryCatch exceptionCatcher; | |
160 exceptionCatcher.SetVerbose(true); | |
161 | |
162 v8::Isolate* isolate = m_scriptState->isolate(); | |
163 v8::Handle<v8::Function> created = retrieveCallback(isolate, "createdCallbac
k"); | |
164 v8::Handle<v8::Function> attached = retrieveCallback(isolate, "attachedCallb
ack"); | |
165 v8::Handle<v8::Function> detached = retrieveCallback(isolate, "detachedCallb
ack"); | |
166 v8::Handle<v8::Function> attributeChanged = retrieveCallback(isolate, "attri
buteChangedCallback"); | |
167 | |
168 m_callbacks = V8CustomElementLifecycleCallbacks::create(m_scriptState.get(),
m_prototype, created, attached, detached, attributeChanged); | |
169 return m_callbacks.get(); | |
170 } | |
171 | |
172 v8::Handle<v8::Function> CustomElementConstructorBuilder::retrieveCallback(v8::I
solate* isolate, const char* name) | |
173 { | |
174 v8::Handle<v8::Value> value = m_prototype->Get(v8String(isolate, name)); | |
175 if (value.IsEmpty() || !value->IsFunction()) | |
176 return v8::Handle<v8::Function>(); | |
177 return value.As<v8::Function>(); | |
178 } | |
179 | |
180 bool CustomElementConstructorBuilder::createConstructor(Document* document, Cust
omElementDefinition* definition, ExceptionState& exceptionState) | |
181 { | |
182 ASSERT(!m_prototype.IsEmpty()); | |
183 ASSERT(m_constructor.IsEmpty()); | |
184 ASSERT(document); | |
185 | |
186 v8::Isolate* isolate = m_scriptState->isolate(); | |
187 | |
188 if (!prototypeIsValid(definition->descriptor().type(), exceptionState)) | |
189 return false; | |
190 | |
191 v8::Local<v8::FunctionTemplate> constructorTemplate = v8::FunctionTemplate::
New(isolate); | |
192 constructorTemplate->SetCallHandler(constructCustomElement); | |
193 m_constructor = constructorTemplate->GetFunction(); | |
194 if (m_constructor.IsEmpty()) { | |
195 CustomElementException::throwException(CustomElementException::ContextDe
stroyedRegisteringDefinition, definition->descriptor().type(), exceptionState); | |
196 return false; | |
197 } | |
198 | |
199 const CustomElementDescriptor& descriptor = definition->descriptor(); | |
200 | |
201 v8::Handle<v8::String> v8TagName = v8String(isolate, descriptor.localName())
; | |
202 v8::Handle<v8::Value> v8Type; | |
203 if (descriptor.isTypeExtension()) | |
204 v8Type = v8String(isolate, descriptor.type()); | |
205 else | |
206 v8Type = v8::Null(isolate); | |
207 | |
208 m_constructor->SetName(v8Type->IsNull() ? v8TagName : v8Type.As<v8::String>(
)); | |
209 | |
210 V8HiddenValue::setHiddenValue(isolate, m_constructor, V8HiddenValue::customE
lementDocument(isolate), toV8(document, m_scriptState->context()->Global(), isol
ate)); | |
211 V8HiddenValue::setHiddenValue(isolate, m_constructor, V8HiddenValue::customE
lementNamespaceURI(isolate), v8String(isolate, descriptor.namespaceURI())); | |
212 V8HiddenValue::setHiddenValue(isolate, m_constructor, V8HiddenValue::customE
lementTagName(isolate), v8TagName); | |
213 V8HiddenValue::setHiddenValue(isolate, m_constructor, V8HiddenValue::customE
lementType(isolate), v8Type); | |
214 | |
215 v8::Handle<v8::String> prototypeKey = v8String(isolate, "prototype"); | |
216 ASSERT(m_constructor->HasOwnProperty(prototypeKey)); | |
217 // This sets the property *value*; calling Set is safe because | |
218 // "prototype" is a non-configurable data property so there can be | |
219 // no side effects. | |
220 m_constructor->Set(prototypeKey, m_prototype); | |
221 // This *configures* the property. ForceSet of a function's | |
222 // "prototype" does not affect the value, but can reconfigure the | |
223 // property. | |
224 m_constructor->ForceSet(prototypeKey, m_prototype, v8::PropertyAttribute(v8:
:ReadOnly | v8::DontEnum | v8::DontDelete)); | |
225 | |
226 V8HiddenValue::setHiddenValue(isolate, m_prototype, V8HiddenValue::customEle
mentIsInterfacePrototypeObject(isolate), v8::True(isolate)); | |
227 m_prototype->ForceSet(v8String(isolate, "constructor"), m_constructor, v8::D
ontEnum); | |
228 | |
229 return true; | |
230 } | |
231 | |
232 bool CustomElementConstructorBuilder::prototypeIsValid(const AtomicString& type,
ExceptionState& exceptionState) const | |
233 { | |
234 if (m_prototype->InternalFieldCount() || !V8HiddenValue::getHiddenValue(m_sc
riptState->isolate(), m_prototype, V8HiddenValue::customElementIsInterfaceProtot
ypeObject(m_scriptState->isolate())).IsEmpty()) { | |
235 CustomElementException::throwException(CustomElementException::Prototype
InUse, type, exceptionState); | |
236 return false; | |
237 } | |
238 | |
239 if (m_prototype->GetPropertyAttributes(v8String(m_scriptState->isolate(), "c
onstructor")) & v8::DontDelete) { | |
240 CustomElementException::throwException(CustomElementException::Construct
orPropertyNotConfigurable, type, exceptionState); | |
241 return false; | |
242 } | |
243 | |
244 return true; | |
245 } | |
246 | |
247 bool CustomElementConstructorBuilder::didRegisterDefinition(CustomElementDefinit
ion* definition) const | |
248 { | |
249 ASSERT(!m_constructor.IsEmpty()); | |
250 | |
251 return m_callbacks->setBinding(definition, CustomElementBinding::create(m_sc
riptState->isolate(), m_prototype, m_wrapperType)); | |
252 } | |
253 | |
254 ScriptValue CustomElementConstructorBuilder::bindingsReturnValue() const | |
255 { | |
256 return ScriptValue(m_scriptState.get(), m_constructor); | |
257 } | |
258 | |
259 bool CustomElementConstructorBuilder::hasValidPrototypeChainFor(const WrapperTyp
eInfo* type) const | |
260 { | |
261 v8::Handle<v8::Object> elementPrototype = m_scriptState->perContextData()->p
rototypeForType(type); | |
262 if (elementPrototype.IsEmpty()) | |
263 return false; | |
264 | |
265 v8::Handle<v8::Value> chain = m_prototype; | |
266 while (!chain.IsEmpty() && chain->IsObject()) { | |
267 if (chain == elementPrototype) | |
268 return true; | |
269 chain = chain.As<v8::Object>()->GetPrototype(); | |
270 } | |
271 | |
272 return false; | |
273 } | |
274 | |
275 static void constructCustomElement(const v8::FunctionCallbackInfo<v8::Value>& in
fo) | |
276 { | |
277 v8::Isolate* isolate = info.GetIsolate(); | |
278 | |
279 if (!info.IsConstructCall()) { | |
280 throwTypeError("DOM object constructor cannot be called as a function.",
isolate); | |
281 return; | |
282 } | |
283 | |
284 if (info.Length() > 0) { | |
285 throwTypeError("This constructor should be called without arguments.", i
solate); | |
286 return; | |
287 } | |
288 | |
289 Document* document = V8Document::toNative(V8HiddenValue::getHiddenValue(info
.GetIsolate(), info.Callee(), V8HiddenValue::customElementDocument(isolate)).As<
v8::Object>()); | |
290 TOSTRING_VOID(V8StringResource<>, namespaceURI, V8HiddenValue::getHiddenValu
e(isolate, info.Callee(), V8HiddenValue::customElementNamespaceURI(isolate))); | |
291 TOSTRING_VOID(V8StringResource<>, tagName, V8HiddenValue::getHiddenValue(iso
late, info.Callee(), V8HiddenValue::customElementTagName(isolate))); | |
292 v8::Handle<v8::Value> maybeType = V8HiddenValue::getHiddenValue(info.GetIsol
ate(), info.Callee(), V8HiddenValue::customElementType(isolate)); | |
293 TOSTRING_VOID(V8StringResource<>, type, maybeType); | |
294 | |
295 ExceptionState exceptionState(ExceptionState::ConstructionContext, "CustomEl
ement", info.Holder(), info.GetIsolate()); | |
296 CustomElementCallbackDispatcher::CallbackDeliveryScope deliveryScope; | |
297 RefPtrWillBeRawPtr<Element> element = document->createElementNS(namespaceURI
, tagName, maybeType->IsNull() ? nullAtom : type, exceptionState); | |
298 if (exceptionState.throwIfNeeded()) | |
299 return; | |
300 v8SetReturnValueFast(info, element.release(), document); | |
301 } | |
302 | |
303 } // namespace WebCore | |
OLD | NEW |