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