Unreviewed build fix
[WebKit-https.git] / Source / WebCore / html / HTMLDocument.cpp
1 /*
2  * Copyright (C) 1999 Lars Knoll (knoll@kde.org)
3  *           (C) 1999 Antti Koivisto (koivisto@kde.org)
4  * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public License
17  * along with this library; see the file COPYING.LIB.  If not, write to
18  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  *
21  * Portions are Copyright (C) 2002 Netscape Communications Corporation.
22  * Other contributors: David Baron <dbaron@fas.harvard.edu>
23  *
24  * This library is free software; you can redistribute it and/or
25  * modify it under the terms of the GNU Lesser General Public
26  * License as published by the Free Software Foundation; either
27  * version 2.1 of the License, or (at your option) any later version.
28  *
29  * This library is distributed in the hope that it will be useful,
30  * but WITHOUT ANY WARRANTY; without even the implied warranty of
31  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
32  * Lesser General Public License for more details.
33  *
34  * You should have received a copy of the GNU Lesser General Public
35  * License along with this library; if not, write to the Free Software
36  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
37  *
38  * Alternatively, the document type parsing portions of this file may be used
39  * under the terms of either the Mozilla Public License Version 1.1, found at
40  * http://www.mozilla.org/MPL/ (the "MPL") or the GNU General Public
41  * License Version 2.0, found at http://www.fsf.org/copyleft/gpl.html
42  * (the "GPL"), in which case the provisions of the MPL or the GPL are
43  * applicable instead of those above.  If you wish to allow use of your
44  * version of this file only under the terms of one of those two
45  * licenses (the MPL or the GPL) and not to allow others to use your
46  * version of this file under the LGPL, indicate your decision by
47  * deleting the provisions above and replace them with the notice and
48  * other provisions required by the MPL or the GPL, as the case may be.
49  * If you do not delete the provisions above, a recipient may use your
50  * version of this file under any of the LGPL, the MPL or the GPL.
51  */
52
53 #include "config.h"
54 #include "HTMLDocument.h"
55
56 #include "CSSPropertyNames.h"
57 #include "CommonVM.h"
58 #include "CookieJar.h"
59 #include "CustomHeaderFields.h"
60 #include "DOMWindow.h"
61 #include "DocumentLoader.h"
62 #include "DocumentType.h"
63 #include "ElementChildIterator.h"
64 #include "FocusController.h"
65 #include "Frame.h"
66 #include "FrameLoader.h"
67 #include "FrameTree.h"
68 #include "FrameView.h"
69 #include "HTMLBodyElement.h"
70 #include "HTMLCollection.h"
71 #include "HTMLDocumentParser.h"
72 #include "HTMLElementFactory.h"
73 #include "HTMLFrameOwnerElement.h"
74 #include "HTMLFrameSetElement.h"
75 #include "HTMLHtmlElement.h"
76 #include "HTMLIFrameElement.h"
77 #include "HTMLNames.h"
78 #include "HashTools.h"
79 #include "ScriptController.h"
80 #include "StyleResolver.h"
81 #include <wtf/IsoMallocInlines.h>
82 #include <wtf/text/CString.h>
83
84 namespace WebCore {
85
86 WTF_MAKE_ISO_ALLOCATED_IMPL(HTMLDocument);
87
88 using namespace HTMLNames;
89
90 HTMLDocument::HTMLDocument(Frame* frame, const URL& url, DocumentClassFlags documentClasses, unsigned constructionFlags)
91     : Document(frame, url, documentClasses | HTMLDocumentClass, constructionFlags)
92 {
93     clearXMLVersion();
94 }
95
96 HTMLDocument::~HTMLDocument() = default;
97
98 int HTMLDocument::width()
99 {
100     updateLayoutIgnorePendingStylesheets();
101     RefPtr<FrameView> frameView = view();
102     return frameView ? frameView->contentsWidth() : 0;
103 }
104
105 int HTMLDocument::height()
106 {
107     updateLayoutIgnorePendingStylesheets();
108     RefPtr<FrameView> frameView = view();
109     return frameView ? frameView->contentsHeight() : 0;
110 }
111
112 Ref<DocumentParser> HTMLDocument::createParser()
113 {
114     return HTMLDocumentParser::create(*this);
115 }
116
117 // https://html.spec.whatwg.org/multipage/dom.html#dom-document-nameditem
118 Optional<Variant<RefPtr<WindowProxy>, RefPtr<Element>, RefPtr<HTMLCollection>>> HTMLDocument::namedItem(const AtomicString& name)
119 {
120     if (name.isNull() || !hasDocumentNamedItem(*name.impl()))
121         return WTF::nullopt;
122
123     if (UNLIKELY(documentNamedItemContainsMultipleElements(*name.impl()))) {
124         auto collection = documentNamedItems(name);
125         ASSERT(collection->length() > 1);
126         return Variant<RefPtr<WindowProxy>, RefPtr<Element>, RefPtr<HTMLCollection>> { RefPtr<HTMLCollection> { WTFMove(collection) } };
127     }
128
129     auto& element = *documentNamedItem(*name.impl());
130     if (UNLIKELY(is<HTMLIFrameElement>(element))) {
131         if (auto domWindow = makeRefPtr(downcast<HTMLIFrameElement>(element).contentWindow()))
132             return Variant<RefPtr<WindowProxy>, RefPtr<Element>, RefPtr<HTMLCollection>> { WTFMove(domWindow) };
133     }
134
135     return Variant<RefPtr<WindowProxy>, RefPtr<Element>, RefPtr<HTMLCollection>> { RefPtr<Element> { &element } };
136 }
137
138 Vector<AtomicString> HTMLDocument::supportedPropertyNames() const
139 {
140     // https://html.spec.whatwg.org/multipage/dom.html#dom-document-namedItem-which
141     //
142     // ... The supported property names of a Document object document at any moment consist of the following, in
143     // tree order according to the element that contributed them, ignoring later duplicates, and with values from
144     // id attributes coming before values from name attributes when the same element contributes both:
145     //
146     // - the value of the name content attribute for all applet, exposed embed, form, iframe, img, and exposed
147     //   object elements that have a non-empty name content attribute and are in a document tree with document
148     //   as their root;
149     // - the value of the id content attribute for all applet and exposed object elements that have a non-empty
150     //   id content attribute and are in a document tree with document as their root; and
151     // - the value of the id content attribute for all img elements that have both a non-empty id content attribute
152     //   and a non-empty name content attribute, and are in a document tree with document as their root.
153
154     // FIXME: Implement.
155     return { };
156 }
157
158 void HTMLDocument::addDocumentNamedItem(const AtomicStringImpl& name, Element& item)
159 {
160     m_documentNamedItem.add(name, item, *this);
161     addImpureProperty(AtomicString(const_cast<AtomicStringImpl*>(&name)));
162 }
163
164 void HTMLDocument::removeDocumentNamedItem(const AtomicStringImpl& name, Element& item)
165 {
166     m_documentNamedItem.remove(name, item);
167 }
168
169 void HTMLDocument::addWindowNamedItem(const AtomicStringImpl& name, Element& item)
170 {
171     m_windowNamedItem.add(name, item, *this);
172 }
173
174 void HTMLDocument::removeWindowNamedItem(const AtomicStringImpl& name, Element& item)
175 {
176     m_windowNamedItem.remove(name, item);
177 }
178
179 bool HTMLDocument::isCaseSensitiveAttribute(const QualifiedName& attributeName)
180 {
181     static const auto caseInsensitiveAttributeSet = makeNeverDestroyed([] {
182         // This is the list of attributes in HTML 4.01 with values marked as "[CI]" or case-insensitive
183         // Mozilla treats all other values as case-sensitive, thus so do we.
184         static const QualifiedName* const names[] = {
185             &accept_charsetAttr.get(),
186             &acceptAttr.get(),
187             &alignAttr.get(),
188             &alinkAttr.get(),
189             &axisAttr.get(),
190             &bgcolorAttr.get(),
191             &charsetAttr.get(),
192             &checkedAttr.get(),
193             &clearAttr.get(),
194             &codetypeAttr.get(),
195             &colorAttr.get(),
196             &compactAttr.get(),
197             &declareAttr.get(),
198             &deferAttr.get(),
199             &dirAttr.get(),
200             &disabledAttr.get(),
201             &enctypeAttr.get(),
202             &faceAttr.get(),
203             &frameAttr.get(),
204             &hreflangAttr.get(),
205             &http_equivAttr.get(),
206             &langAttr.get(),
207             &languageAttr.get(),
208             &linkAttr.get(),
209             &mediaAttr.get(),
210             &methodAttr.get(),
211             &multipleAttr.get(),
212             &nohrefAttr.get(),
213             &noresizeAttr.get(),
214             &noshadeAttr.get(),
215             &nowrapAttr.get(),
216             &readonlyAttr.get(),
217             &relAttr.get(),
218             &revAttr.get(),
219             &rulesAttr.get(),
220             &scopeAttr.get(),
221             &scrollingAttr.get(),
222             &selectedAttr.get(),
223             &shapeAttr.get(),
224             &targetAttr.get(),
225             &textAttr.get(),
226             &typeAttr.get(),
227             &valignAttr.get(),
228             &valuetypeAttr.get(),
229             &vlinkAttr.get(),
230         };
231         HashSet<AtomicString> set;
232         for (auto* name : names)
233             set.add(name->localName());
234         return set;
235     }());
236
237     bool isPossibleHTMLAttr = !attributeName.hasPrefix() && attributeName.namespaceURI().isNull();
238     return !isPossibleHTMLAttr || !caseInsensitiveAttributeSet.get().contains(attributeName.localName());
239 }
240
241 bool HTMLDocument::isFrameSet() const
242 {
243     if (!documentElement())
244         return false;
245     return !!childrenOfType<HTMLFrameSetElement>(*documentElement()).first();
246 }
247
248 Ref<Document> HTMLDocument::cloneDocumentWithoutChildren() const
249 {
250     return create(nullptr, url());
251 }
252
253 }