Element's renderer factory should return RenderPtrs.
[WebKit-https.git] / Source / WebCore / html / HTMLFieldSetElement.cpp
1 /*
2  * Copyright (C) 1999 Lars Knoll (knoll@kde.org)
3  *           (C) 1999 Antti Koivisto (koivisto@kde.org)
4  *           (C) 2001 Dirk Mueller (mueller@kde.org)
5  * Copyright (C) 2004, 2005, 2006, 2010 Apple Inc. All rights reserved.
6  *           (C) 2006 Alexey Proskuryakov (ap@nypop.com)
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Library General Public
10  * License as published by the Free Software Foundation; either
11  * version 2 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Library General Public License for more details.
17  *
18  * You should have received a copy of the GNU Library General Public License
19  * along with this library; see the file COPYING.LIB.  If not, write to
20  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
21  * Boston, MA 02110-1301, USA.
22  *
23  */
24
25 #include "config.h"
26 #include "HTMLFieldSetElement.h"
27
28 #include "ElementIterator.h"
29 #include "HTMLCollection.h"
30 #include "HTMLLegendElement.h"
31 #include "HTMLNames.h"
32 #include "HTMLObjectElement.h"
33 #include "RenderFieldset.h"
34 #include <wtf/StdLibExtras.h>
35
36 namespace WebCore {
37
38 using namespace HTMLNames;
39
40 inline HTMLFieldSetElement::HTMLFieldSetElement(const QualifiedName& tagName, Document& document, HTMLFormElement* form)
41     : HTMLFormControlElement(tagName, document, form)
42     , m_documentVersion(0)
43 {
44     ASSERT(hasTagName(fieldsetTag));
45 }
46
47 PassRefPtr<HTMLFieldSetElement> HTMLFieldSetElement::create(const QualifiedName& tagName, Document& document, HTMLFormElement* form)
48 {
49     return adoptRef(new HTMLFieldSetElement(tagName, document, form));
50 }
51
52 void HTMLFieldSetElement::invalidateDisabledStateUnder(Element* base)
53 {
54     for (auto& control : descendantsOfType<HTMLFormControlElement>(*base))
55         control.ancestorDisabledStateWasChanged();
56 }
57
58 void HTMLFieldSetElement::disabledAttributeChanged()
59 {
60     // This element must be updated before the style of nodes in its subtree gets recalculated.
61     HTMLFormControlElement::disabledAttributeChanged();
62     invalidateDisabledStateUnder(this);
63 }
64
65 void HTMLFieldSetElement::childrenChanged(const ChildChange& change)
66 {
67     HTMLFormControlElement::childrenChanged(change);
68
69     for (auto& legend : childrenOfType<HTMLLegendElement>(*this))
70         invalidateDisabledStateUnder(&legend);
71 }
72
73 bool HTMLFieldSetElement::supportsFocus() const
74 {
75     return HTMLElement::supportsFocus();
76 }
77
78 const AtomicString& HTMLFieldSetElement::formControlType() const
79 {
80     DEFINE_STATIC_LOCAL(const AtomicString, fieldset, ("fieldset", AtomicString::ConstructFromLiteral));
81     return fieldset;
82 }
83
84 RenderPtr<RenderElement> HTMLFieldSetElement::createElementRenderer(PassRef<RenderStyle> style)
85 {
86     return createRenderer<RenderFieldset>(*this, std::move(style));
87 }
88
89 HTMLLegendElement* HTMLFieldSetElement::legend() const
90 {
91     return const_cast<HTMLLegendElement*>(descendantsOfType<HTMLLegendElement>(*this).first());
92 }
93
94 PassRefPtr<HTMLCollection> HTMLFieldSetElement::elements()
95 {
96     return ensureCachedHTMLCollection(FormControls);
97 }
98
99 void HTMLFieldSetElement::refreshElementsIfNeeded() const
100 {
101     uint64_t documentVersion = document().domTreeVersion();
102     if (m_documentVersion == documentVersion)
103         return;
104
105     m_documentVersion = documentVersion;
106
107     m_associatedElements.clear();
108
109     for (auto& element : elementDescendants(const_cast<HTMLFieldSetElement&>(*this))) {
110         if (element.hasTagName(objectTag))
111             m_associatedElements.append(&toHTMLObjectElement(element));
112         else if (element.isFormControlElement())
113             m_associatedElements.append(&toHTMLFormControlElement(element));
114     }
115 }
116
117 const Vector<FormAssociatedElement*>& HTMLFieldSetElement::associatedElements() const
118 {
119     refreshElementsIfNeeded();
120     return m_associatedElements;
121 }
122
123 unsigned HTMLFieldSetElement::length() const
124 {
125     refreshElementsIfNeeded();
126     unsigned length = 0;
127     for (unsigned i = 0; i < m_associatedElements.size(); ++i) {
128         if (m_associatedElements[i]->isEnumeratable())
129             ++length;
130     }
131     return length;
132 }
133
134 } // namespace