f0dee3dd44b36ba33ac98f7fa6ad5eadeb102e2e
[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 "DocumentLoader.h"
60 #include "DocumentType.h"
61 #include "ElementChildIterator.h"
62 #include "FocusController.h"
63 #include "Frame.h"
64 #include "FrameLoader.h"
65 #include "FrameTree.h"
66 #include "FrameView.h"
67 #include "HTMLBodyElement.h"
68 #include "HTMLCollection.h"
69 #include "HTMLDocumentParser.h"
70 #include "HTMLElementFactory.h"
71 #include "HTMLFrameOwnerElement.h"
72 #include "HTMLFrameSetElement.h"
73 #include "HTMLHtmlElement.h"
74 #include "HTMLIFrameElement.h"
75 #include "HTMLNames.h"
76 #include "HashTools.h"
77 #include "ScriptController.h"
78 #include "StyleResolver.h"
79 #include <wtf/text/CString.h>
80
81 namespace WebCore {
82
83 using namespace HTMLNames;
84
85 HTMLDocument::HTMLDocument(Frame* frame, const URL& url, DocumentClassFlags documentClasses, unsigned constructionFlags)
86     : Document(frame, url, documentClasses | HTMLDocumentClass, constructionFlags)
87 {
88     clearXMLVersion();
89 }
90
91 HTMLDocument::~HTMLDocument()
92 {
93 }
94
95 int HTMLDocument::width()
96 {
97     updateLayoutIgnorePendingStylesheets();
98     FrameView* frameView = view();
99     return frameView ? frameView->contentsWidth() : 0;
100 }
101
102 int HTMLDocument::height()
103 {
104     updateLayoutIgnorePendingStylesheets();
105     FrameView* frameView = view();
106     return frameView ? frameView->contentsHeight() : 0;
107 }
108
109 Ref<DocumentParser> HTMLDocument::createParser()
110 {
111     return HTMLDocumentParser::create(*this);
112 }
113
114 static void addLocalNameToSet(HashSet<AtomicStringImpl*>* set, const QualifiedName& qName)
115 {
116     set->add(qName.localName().impl());
117 }
118
119 static HashSet<AtomicStringImpl*>* createHtmlCaseInsensitiveAttributesSet()
120 {
121     // This is the list of attributes in HTML 4.01 with values marked as "[CI]" or case-insensitive
122     // Mozilla treats all other values as case-sensitive, thus so do we.
123     HashSet<AtomicStringImpl*>* attrSet = new HashSet<AtomicStringImpl*>;
124
125     addLocalNameToSet(attrSet, accept_charsetAttr);
126     addLocalNameToSet(attrSet, acceptAttr);
127     addLocalNameToSet(attrSet, alignAttr);
128     addLocalNameToSet(attrSet, alinkAttr);
129     addLocalNameToSet(attrSet, axisAttr);
130     addLocalNameToSet(attrSet, bgcolorAttr);
131     addLocalNameToSet(attrSet, charsetAttr);
132     addLocalNameToSet(attrSet, checkedAttr);
133     addLocalNameToSet(attrSet, clearAttr);
134     addLocalNameToSet(attrSet, codetypeAttr);
135     addLocalNameToSet(attrSet, colorAttr);
136     addLocalNameToSet(attrSet, compactAttr);
137     addLocalNameToSet(attrSet, declareAttr);
138     addLocalNameToSet(attrSet, deferAttr);
139     addLocalNameToSet(attrSet, dirAttr);
140     addLocalNameToSet(attrSet, disabledAttr);
141     addLocalNameToSet(attrSet, enctypeAttr);
142     addLocalNameToSet(attrSet, faceAttr);
143     addLocalNameToSet(attrSet, frameAttr);
144     addLocalNameToSet(attrSet, hreflangAttr);
145     addLocalNameToSet(attrSet, http_equivAttr);
146     addLocalNameToSet(attrSet, langAttr);
147     addLocalNameToSet(attrSet, languageAttr);
148     addLocalNameToSet(attrSet, linkAttr);
149     addLocalNameToSet(attrSet, mediaAttr);
150     addLocalNameToSet(attrSet, methodAttr);
151     addLocalNameToSet(attrSet, multipleAttr);
152     addLocalNameToSet(attrSet, nohrefAttr);
153     addLocalNameToSet(attrSet, noresizeAttr);
154     addLocalNameToSet(attrSet, noshadeAttr);
155     addLocalNameToSet(attrSet, nowrapAttr);
156     addLocalNameToSet(attrSet, readonlyAttr);
157     addLocalNameToSet(attrSet, relAttr);
158     addLocalNameToSet(attrSet, revAttr);
159     addLocalNameToSet(attrSet, rulesAttr);
160     addLocalNameToSet(attrSet, scopeAttr);
161     addLocalNameToSet(attrSet, scrollingAttr);
162     addLocalNameToSet(attrSet, selectedAttr);
163     addLocalNameToSet(attrSet, shapeAttr);
164     addLocalNameToSet(attrSet, targetAttr);
165     addLocalNameToSet(attrSet, textAttr);
166     addLocalNameToSet(attrSet, typeAttr);
167     addLocalNameToSet(attrSet, valignAttr);
168     addLocalNameToSet(attrSet, valuetypeAttr);
169     addLocalNameToSet(attrSet, vlinkAttr);
170
171     return attrSet;
172 }
173
174 // https://html.spec.whatwg.org/multipage/dom.html#dom-document-nameditem
175 std::optional<Variant<RefPtr<DOMWindow>, RefPtr<Element>, RefPtr<HTMLCollection>>> HTMLDocument::namedItem(const AtomicString& name)
176 {
177     if (name.isNull() || !hasDocumentNamedItem(*name.impl()))
178         return std::nullopt;
179
180     if (UNLIKELY(documentNamedItemContainsMultipleElements(*name.impl()))) {
181         auto collection = documentNamedItems(name);
182         ASSERT(collection->length() > 1);
183         return Variant<RefPtr<DOMWindow>, RefPtr<Element>, RefPtr<HTMLCollection>> { RefPtr<HTMLCollection> { WTFMove(collection) } };
184     }
185
186     auto& element = *documentNamedItem(*name.impl());
187     if (UNLIKELY(is<HTMLIFrameElement>(element))) {
188         if (auto* domWindow = downcast<HTMLIFrameElement>(element).contentWindow())
189             return Variant<RefPtr<DOMWindow>, RefPtr<Element>, RefPtr<HTMLCollection>> { RefPtr<DOMWindow> { domWindow } };
190     }
191
192     return Variant<RefPtr<DOMWindow>, RefPtr<Element>, RefPtr<HTMLCollection>> { RefPtr<Element> { &element } };
193 }
194
195 Vector<AtomicString> HTMLDocument::supportedPropertyNames() const
196 {
197     // https://html.spec.whatwg.org/multipage/dom.html#dom-document-namedItem-which
198     //
199     // ... The supported property names of a Document object document at any moment consist of the following, in
200     // tree order according to the element that contributed them, ignoring later duplicates, and with values from
201     // id attributes coming before values from name attributes when the same element contributes both:
202     //
203     // - the value of the name content attribute for all applet, exposed embed, form, iframe, img, and exposed
204     //   object elements that have a non-empty name content attribute and are in a document tree with document
205     //   as their root;
206     // - the value of the id content attribute for all applet and exposed object elements that have a non-empty
207     //   id content attribute and are in a document tree with document as their root; and
208     // - the value of the id content attribute for all img elements that have both a non-empty id content attribute
209     //   and a non-empty name content attribute, and are in a document tree with document as their root.
210
211     // FIXME: Implement.
212     return { };
213 }
214
215 void HTMLDocument::addDocumentNamedItem(const AtomicStringImpl& name, Element& item)
216 {
217     m_documentNamedItem.add(name, item, *this);
218     addImpureProperty(AtomicString(const_cast<AtomicStringImpl*>(&name)));
219 }
220
221 void HTMLDocument::removeDocumentNamedItem(const AtomicStringImpl& name, Element& item)
222 {
223     m_documentNamedItem.remove(name, item);
224 }
225
226 void HTMLDocument::addWindowNamedItem(const AtomicStringImpl& name, Element& item)
227 {
228     m_windowNamedItem.add(name, item, *this);
229 }
230
231 void HTMLDocument::removeWindowNamedItem(const AtomicStringImpl& name, Element& item)
232 {
233     m_windowNamedItem.remove(name, item);
234 }
235
236 bool HTMLDocument::isCaseSensitiveAttribute(const QualifiedName& attributeName)
237 {
238     static HashSet<AtomicStringImpl*>* htmlCaseInsensitiveAttributesSet = createHtmlCaseInsensitiveAttributesSet();
239     bool isPossibleHTMLAttr = !attributeName.hasPrefix() && (attributeName.namespaceURI() == nullAtom);
240     return !isPossibleHTMLAttr || !htmlCaseInsensitiveAttributesSet->contains(attributeName.localName().impl());
241 }
242
243 bool HTMLDocument::isFrameSet() const
244 {
245     if (!documentElement())
246         return false;
247     return !!childrenOfType<HTMLFrameSetElement>(*documentElement()).first();
248 }
249
250 Ref<Document> HTMLDocument::cloneDocumentWithoutChildren() const
251 {
252     return create(nullptr, url());
253 }
254
255 }