0ce2b06ffe85861f36ed5105202dcedae74d73e7
[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 "CookieJar.h"
58 #include "DocumentLoader.h"
59 #include "DocumentType.h"
60 #include "ExceptionCode.h"
61 #include "FocusController.h"
62 #include "Frame.h"
63 #include "FrameLoader.h"
64 #include "FrameTree.h"
65 #include "FrameView.h"
66 #include "HashTools.h"
67 #include "HTMLDocumentParser.h"
68 #include "HTMLBodyElement.h"
69 #include "HTMLElementFactory.h"
70 #include "HTMLFrameOwnerElement.h"
71 #include "HTMLFrameSetElement.h"
72 #include "HTMLNames.h"
73 #include "JSDOMBinding.h"
74 #include "Page.h"
75 #include "ScriptController.h"
76 #include "Settings.h"
77 #include "StyleResolver.h"
78 #include <wtf/text/CString.h>
79
80 namespace WebCore {
81
82 using namespace HTMLNames;
83
84 HTMLDocument::HTMLDocument(Frame* frame, const URL& url, DocumentClassFlags documentClasses, unsigned constructionFlags)
85     : Document(frame, url, documentClasses | HTMLDocumentClass, constructionFlags)
86 {
87     clearXMLVersion();
88 }
89
90 HTMLDocument::~HTMLDocument()
91 {
92 }
93
94 int HTMLDocument::width()
95 {
96     updateLayoutIgnorePendingStylesheets();
97     FrameView* frameView = view();
98     return frameView ? frameView->contentsWidth() : 0;
99 }
100
101 int HTMLDocument::height()
102 {
103     updateLayoutIgnorePendingStylesheets();
104     FrameView* frameView = view();
105     return frameView ? frameView->contentsHeight() : 0;
106 }
107
108 String HTMLDocument::dir()
109 {
110     if (auto* body = this->body())
111         return body->fastGetAttribute(dirAttr);
112     return String();
113 }
114
115 void HTMLDocument::setDir(const String& value)
116 {
117     HTMLElement* b = body();
118     if (b)
119         b->setAttribute(dirAttr, value);
120 }
121
122 String HTMLDocument::designMode() const
123 {
124     return inDesignMode() ? "on" : "off";
125 }
126
127 void HTMLDocument::setDesignMode(const String& value)
128 {
129     InheritedBool mode;
130     if (equalIgnoringCase(value, "on"))
131         mode = on;
132     else if (equalIgnoringCase(value, "off"))
133         mode = off;
134     else
135         mode = inherit;
136     Document::setDesignMode(mode);
137 }
138
139 const AtomicString& HTMLDocument::bgColor() const
140 {
141     HTMLElement* bodyElement = body();
142     if (!bodyElement || !isHTMLBodyElement(bodyElement))
143         return emptyAtom;
144     return bodyElement->fastGetAttribute(bgcolorAttr);
145 }
146
147 void HTMLDocument::setBgColor(const String& value)
148 {
149     HTMLElement* bodyElement = body();
150     if (!bodyElement || !isHTMLBodyElement(bodyElement))
151         return;
152     bodyElement->setAttribute(bgcolorAttr, value);
153 }
154
155 const AtomicString& HTMLDocument::fgColor() const
156 {
157     HTMLElement* bodyElement = body();
158     if (!bodyElement || !isHTMLBodyElement(bodyElement))
159         return emptyAtom;
160     return bodyElement->fastGetAttribute(textAttr);
161 }
162
163 void HTMLDocument::setFgColor(const String& value)
164 {
165     HTMLElement* bodyElement = body();
166     if (!bodyElement || !isHTMLBodyElement(bodyElement))
167         return;
168     bodyElement->setAttribute(textAttr, value);
169 }
170
171 const AtomicString& HTMLDocument::alinkColor() const
172 {
173     HTMLElement* bodyElement = body();
174     if (!bodyElement || !isHTMLBodyElement(bodyElement))
175         return emptyAtom;
176     return bodyElement->fastGetAttribute(alinkAttr);
177 }
178
179 void HTMLDocument::setAlinkColor(const String& value)
180 {
181     HTMLElement* bodyElement = body();
182     if (!bodyElement || !isHTMLBodyElement(bodyElement))
183         return;
184     bodyElement->setAttribute(alinkAttr, value);
185 }
186
187 const AtomicString& HTMLDocument::linkColor() const
188 {
189     HTMLElement* bodyElement = body();
190     if (!bodyElement || !isHTMLBodyElement(bodyElement))
191         return emptyAtom;
192     return bodyElement->fastGetAttribute(linkAttr);
193 }
194
195 void HTMLDocument::setLinkColor(const String& value)
196 {
197     HTMLElement* bodyElement = body();
198     if (!bodyElement || !isHTMLBodyElement(bodyElement))
199         return;
200     return bodyElement->setAttribute(linkAttr, value);
201 }
202
203 const AtomicString& HTMLDocument::vlinkColor() const
204 {
205     HTMLElement* bodyElement = body();
206     if (!bodyElement || !isHTMLBodyElement(bodyElement))
207         return emptyAtom;
208     return bodyElement->fastGetAttribute(vlinkAttr);
209 }
210
211 void HTMLDocument::setVlinkColor(const String& value)
212 {
213     HTMLElement* bodyElement = body();
214     if (!bodyElement || !isHTMLBodyElement(bodyElement))
215         return;
216     return bodyElement->setAttribute(vlinkAttr, value);
217 }
218
219 void HTMLDocument::captureEvents()
220 {
221 }
222
223 void HTMLDocument::releaseEvents()
224 {
225 }
226
227 PassRefPtr<DocumentParser> HTMLDocument::createParser()
228 {
229     return HTMLDocumentParser::create(*this);
230 }
231
232 // --------------------------------------------------------------------------
233 // not part of the DOM
234 // --------------------------------------------------------------------------
235
236 PassRefPtr<Element> HTMLDocument::createElement(const AtomicString& name, ExceptionCode& ec)
237 {
238     if (!isValidName(name)) {
239         ec = INVALID_CHARACTER_ERR;
240         return 0;
241     }
242     return HTMLElementFactory::createElement(QualifiedName(nullAtom, name.lower(), xhtmlNamespaceURI), *this);
243 }
244
245 static void addLocalNameToSet(HashSet<AtomicStringImpl*>* set, const QualifiedName& qName)
246 {
247     set->add(qName.localName().impl());
248 }
249
250 static HashSet<AtomicStringImpl*>* createHtmlCaseInsensitiveAttributesSet()
251 {
252     // This is the list of attributes in HTML 4.01 with values marked as "[CI]" or case-insensitive
253     // Mozilla treats all other values as case-sensitive, thus so do we.
254     HashSet<AtomicStringImpl*>* attrSet = new HashSet<AtomicStringImpl*>;
255
256     addLocalNameToSet(attrSet, accept_charsetAttr);
257     addLocalNameToSet(attrSet, acceptAttr);
258     addLocalNameToSet(attrSet, alignAttr);
259     addLocalNameToSet(attrSet, alinkAttr);
260     addLocalNameToSet(attrSet, axisAttr);
261     addLocalNameToSet(attrSet, bgcolorAttr);
262     addLocalNameToSet(attrSet, charsetAttr);
263     addLocalNameToSet(attrSet, checkedAttr);
264     addLocalNameToSet(attrSet, clearAttr);
265     addLocalNameToSet(attrSet, codetypeAttr);
266     addLocalNameToSet(attrSet, colorAttr);
267     addLocalNameToSet(attrSet, compactAttr);
268     addLocalNameToSet(attrSet, declareAttr);
269     addLocalNameToSet(attrSet, deferAttr);
270     addLocalNameToSet(attrSet, dirAttr);
271     addLocalNameToSet(attrSet, disabledAttr);
272     addLocalNameToSet(attrSet, enctypeAttr);
273     addLocalNameToSet(attrSet, faceAttr);
274     addLocalNameToSet(attrSet, frameAttr);
275     addLocalNameToSet(attrSet, hreflangAttr);
276     addLocalNameToSet(attrSet, http_equivAttr);
277     addLocalNameToSet(attrSet, langAttr);
278     addLocalNameToSet(attrSet, languageAttr);
279     addLocalNameToSet(attrSet, linkAttr);
280     addLocalNameToSet(attrSet, mediaAttr);
281     addLocalNameToSet(attrSet, methodAttr);
282     addLocalNameToSet(attrSet, multipleAttr);
283     addLocalNameToSet(attrSet, nohrefAttr);
284     addLocalNameToSet(attrSet, noresizeAttr);
285     addLocalNameToSet(attrSet, noshadeAttr);
286     addLocalNameToSet(attrSet, nowrapAttr);
287     addLocalNameToSet(attrSet, readonlyAttr);
288     addLocalNameToSet(attrSet, relAttr);
289     addLocalNameToSet(attrSet, revAttr);
290     addLocalNameToSet(attrSet, rulesAttr);
291     addLocalNameToSet(attrSet, scopeAttr);
292     addLocalNameToSet(attrSet, scrollingAttr);
293     addLocalNameToSet(attrSet, selectedAttr);
294     addLocalNameToSet(attrSet, shapeAttr);
295     addLocalNameToSet(attrSet, targetAttr);
296     addLocalNameToSet(attrSet, textAttr);
297     addLocalNameToSet(attrSet, typeAttr);
298     addLocalNameToSet(attrSet, valignAttr);
299     addLocalNameToSet(attrSet, valuetypeAttr);
300     addLocalNameToSet(attrSet, vlinkAttr);
301
302     return attrSet;
303 }
304
305 void HTMLDocument::addDocumentNamedItem(const AtomicStringImpl& name, Element& item)
306 {
307     m_documentNamedItem.add(name, item, *this);
308     addImpureProperty(AtomicString(const_cast<AtomicStringImpl*>(&name)));
309 }
310
311 void HTMLDocument::removeDocumentNamedItem(const AtomicStringImpl& name, Element& item)
312 {
313     m_documentNamedItem.remove(name, item);
314 }
315
316 void HTMLDocument::addWindowNamedItem(const AtomicStringImpl& name, Element& item)
317 {
318     m_windowNamedItem.add(name, item, *this);
319 }
320
321 void HTMLDocument::removeWindowNamedItem(const AtomicStringImpl& name, Element& item)
322 {
323     m_windowNamedItem.remove(name, item);
324 }
325
326 bool HTMLDocument::isCaseSensitiveAttribute(const QualifiedName& attributeName)
327 {
328     static HashSet<AtomicStringImpl*>* htmlCaseInsensitiveAttributesSet = createHtmlCaseInsensitiveAttributesSet();
329     bool isPossibleHTMLAttr = !attributeName.hasPrefix() && (attributeName.namespaceURI() == nullAtom);
330     return !isPossibleHTMLAttr || !htmlCaseInsensitiveAttributesSet->contains(attributeName.localName().impl());
331 }
332
333 void HTMLDocument::clear()
334 {
335     // FIXME: This does nothing, and that seems unlikely to be correct.
336     // We've long had a comment saying that IE doesn't support this.
337     // But I do see it in the documentation for Mozilla.
338 }
339
340 bool HTMLDocument::isFrameSet() const
341 {
342     HTMLElement* bodyElement = body();
343     return bodyElement && isHTMLFrameSetElement(bodyElement);
344 }
345
346 PassRefPtr<Document> HTMLDocument::cloneDocumentWithoutChildren() const
347 {
348     return create(nullptr, url());
349 }
350
351 }