Add document.defineCustomElement
[WebKit.git] / Source / WebCore / bindings / js / JSHTMLElementCustom.cpp
1 /*
2  * Copyright (C) 2007 Apple 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
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "JSHTMLElement.h"
28
29 #include "CustomElementDefinitions.h"
30 #include "Document.h"
31 #include "HTMLFormElement.h"
32 #include <runtime/InternalFunction.h>
33 #include <runtime/JSWithScope.h>
34
35 namespace WebCore {
36
37 using namespace JSC;
38
39 #if ENABLE(CUSTOM_ELEMENTS)
40 EncodedJSValue JSC_HOST_CALL constructJSHTMLElement(ExecState* state)
41 {
42     auto* jsConstructor = jsCast<DOMConstructorObject*>(state->callee());
43
44     auto* context = jsConstructor->scriptExecutionContext();
45     if (!is<Document>(context))
46         return throwConstructorDocumentUnavailableError(*state, "HTMLElement");
47     auto& document = downcast<Document>(*context);
48
49     auto* definitions = document.customElementDefinitions();
50     if (!definitions)
51         return throwVMTypeError(state, "new.target is not a valid custom element constructor");
52
53     VM& vm = state->vm();
54     JSValue newTargetValue = state->thisValue();
55     JSObject* newTarget = newTargetValue.getObject();
56     QualifiedName fullName = definitions->findName(newTarget);
57     if (fullName == nullQName()) {
58         if (UNLIKELY(state->argumentCount() < 1))
59             return throwVMError(state, createNotEnoughArgumentsError(state));
60     }
61
62     if (state->argumentCount()) {
63         String name;
64         if (!state->argument(0).getString(state, name))
65             return throwVMTypeError(state, "The first argument is not a valid custom element name");
66         
67         auto* interface = definitions->findInterface(name);
68         if (!interface)
69             return throwVMTypeError(state, "The first argument is not a valid custom element name");
70         
71         if (newTarget != interface->constructor())
72             return throwVMTypeError(state, "Attempt to construct a custom element with a wrong interface");
73         
74         fullName = QualifiedName(nullAtom, name, HTMLNames::xhtmlNamespaceURI);
75     }
76
77     auto* globalObject = jsConstructor->globalObject();
78     Structure* baseStructure = getDOMStructure<JSHTMLElement>(vm, *globalObject);
79     auto* newElementStructure = InternalFunction::createSubclassStructure(state, newTargetValue, baseStructure);
80     if (UNLIKELY(state->hadException()))
81         return JSValue::encode(jsUndefined());
82
83     Ref<HTMLElement> element = HTMLElement::create(fullName, document);
84     auto* jsElement = JSHTMLElement::create(newElementStructure, globalObject, element.get());
85     cacheWrapper(globalObject->world(), element.ptr(), jsElement);
86     return JSValue::encode(jsElement);
87 }
88 #endif
89
90 JSScope* JSHTMLElement::pushEventHandlerScope(ExecState* exec, JSScope* scope) const
91 {
92     HTMLElement& element = wrapped();
93
94     // The document is put on first, fall back to searching it only after the element and form.
95     scope = JSWithScope::create(exec, asObject(toJS(exec, globalObject(), &element.document())), scope);
96
97     // The form is next, searched before the document, but after the element itself.
98     if (HTMLFormElement* form = element.form())
99         scope = JSWithScope::create(exec, asObject(toJS(exec, globalObject(), form)), scope);
100
101     // The element is on top, searched first.
102     return JSWithScope::create(exec, asObject(toJS(exec, globalObject(), &element)), scope);
103 }
104
105 } // namespace WebCore