CTTE: Element::createRenderer() should take references
[WebKit-https.git] / Source / WebCore / rendering / RenderButton.cpp
1 /**
2  * Copyright (C) 2005 Apple Computer, Inc.
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Library General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Library General Public License for more details.
13  *
14  * You should have received a copy of the GNU Library General Public License
15  * along with this library; see the file COPYING.LIB.  If not, write to
16  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
17  * Boston, MA 02110-1301, USA.
18  *
19  */
20
21 #include "config.h"
22 #include "RenderButton.h"
23
24 #include "Document.h"
25 #include "GraphicsContext.h"
26 #include "HTMLInputElement.h"
27 #include "HTMLNames.h"
28 #include "RenderTextFragment.h"
29 #include "RenderTheme.h"
30 #include "StyleInheritedData.h"
31
32 namespace WebCore {
33
34 using namespace HTMLNames;
35
36 RenderButton::RenderButton(HTMLFormControlElement& element)
37     : RenderFlexibleBox(&element)
38     , m_buttonText(0)
39     , m_inner(0)
40     , m_default(false)
41 {
42 }
43
44 RenderButton::~RenderButton()
45 {
46 }
47
48 HTMLFormControlElement& RenderButton::formControlElement() const
49 {
50     ASSERT(RenderObject::node());
51     return *toHTMLFormControlElement(RenderObject::node());
52 }
53
54 bool RenderButton::canBeSelectionLeaf() const
55 {
56     return formControlElement().rendererIsEditable();
57 }
58
59 bool RenderButton::hasLineIfEmpty() const
60 {
61     return formControlElement().toInputElement();
62 }
63
64 void RenderButton::addChild(RenderObject* newChild, RenderObject* beforeChild)
65 {
66     if (!m_inner) {
67         // Create an anonymous block.
68         ASSERT(!firstChild());
69         m_inner = createAnonymousBlock(style()->display());
70         setupInnerStyle(m_inner->style());
71         RenderFlexibleBox::addChild(m_inner);
72     }
73     
74     m_inner->addChild(newChild, beforeChild);
75 }
76
77 void RenderButton::removeChild(RenderObject* oldChild)
78 {
79     // m_inner should be the only child, but checking for direct children who
80     // are not m_inner prevents security problems when that assumption is
81     // violated.
82     if (oldChild == m_inner || !m_inner || oldChild->parent() == this) {
83         ASSERT(oldChild == m_inner || !m_inner);
84         RenderFlexibleBox::removeChild(oldChild);
85         m_inner = 0;
86     } else
87         m_inner->removeChild(oldChild);
88 }
89
90 void RenderButton::styleWillChange(StyleDifference diff, const RenderStyle* newStyle)
91 {
92     if (m_inner) {
93         // RenderBlock::setStyle is going to apply a new style to the inner block, which
94         // will have the initial flex value, 0. The current value is 1, because we set
95         // it right below. Here we change it back to 0 to avoid getting a spurious layout hint
96         // because of the difference. Same goes for the other properties.
97         // FIXME: Make this hack unnecessary.
98         m_inner->style()->setFlexGrow(newStyle->initialFlexGrow());
99         m_inner->style()->setMarginTop(newStyle->initialMargin());
100         m_inner->style()->setMarginBottom(newStyle->initialMargin());
101     }
102     RenderBlock::styleWillChange(diff, newStyle);
103 }
104
105 void RenderButton::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
106 {
107     RenderBlock::styleDidChange(diff, oldStyle);
108
109     if (m_buttonText)
110         m_buttonText->setStyle(style());
111     if (m_inner) // RenderBlock handled updating the anonymous block's style.
112         setupInnerStyle(m_inner->style());
113
114     if (!m_default && theme()->isDefault(this)) {
115         if (!m_timer)
116             m_timer = adoptPtr(new Timer<RenderButton>(this, &RenderButton::timerFired));
117         m_timer->startRepeating(0.03);
118         m_default = true;
119     } else if (m_default && !theme()->isDefault(this)) {
120         m_default = false;
121         m_timer.clear();
122     }
123 }
124
125 void RenderButton::setupInnerStyle(RenderStyle* innerStyle) 
126 {
127     ASSERT(innerStyle->refCount() == 1);
128     // RenderBlock::createAnonymousBlock creates a new RenderStyle, so this is
129     // safe to modify.
130     innerStyle->setFlexGrow(1.0f);
131     // Use margin:auto instead of align-items:center to get safe centering, i.e.
132     // when the content overflows, treat it the same as align-items: flex-start.
133     innerStyle->setMarginTop(Length());
134     innerStyle->setMarginBottom(Length());
135     innerStyle->setFlexDirection(style()->flexDirection());
136 }
137
138 void RenderButton::updateFromElement()
139 {
140     // If we're an input element, we may need to change our button text.
141     if (isHTMLInputElement(formControlElement())) {
142         HTMLInputElement& input = toHTMLInputElement(formControlElement());
143         String value = input.valueWithDefault();
144         setText(value);
145     }
146 }
147
148 void RenderButton::setText(const String& str)
149 {
150     if (str.isEmpty()) {
151         if (m_buttonText) {
152             m_buttonText->destroy();
153             m_buttonText = 0;
154         }
155     } else {
156         if (m_buttonText)
157             m_buttonText->setText(str.impl());
158         else {
159             m_buttonText = new (*renderArena()) RenderTextFragment(&document(), str.impl());
160             m_buttonText->setStyle(style());
161             addChild(m_buttonText);
162         }
163     }
164 }
165
166 String RenderButton::text() const
167 {
168     return m_buttonText ? m_buttonText->text() : 0;
169 }
170
171 bool RenderButton::canHaveGeneratedChildren() const
172 {
173     // Input elements can't have generated children, but button elements can. We'll
174     // write the code assuming any other button types that might emerge in the future
175     // can also have children.
176     return !isHTMLInputElement(formControlElement());
177 }
178
179 LayoutRect RenderButton::controlClipRect(const LayoutPoint& additionalOffset) const
180 {
181     // Clip to the padding box to at least give content the extra padding space.
182     return LayoutRect(additionalOffset.x() + borderLeft(), additionalOffset.y() + borderTop(), width() - borderLeft() - borderRight(), height() - borderTop() - borderBottom());
183 }
184
185 void RenderButton::timerFired(Timer<RenderButton>*)
186 {
187     // FIXME Bug 25110: Ideally we would stop our timer when our Document
188     // enters the page cache. But we currently have no way of being notified
189     // when that happens, so we'll just ignore the timer firing as long as
190     // we're in the cache.
191     if (document().inPageCache())
192         return;
193
194     repaint();
195 }
196
197 } // namespace WebCore