Add WTF::move()
[WebKit-https.git] / Source / WebCore / rendering / RenderButton.cpp
1 /**
2  * Copyright (C) 2005 Apple 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 #if PLATFORM(IOS)
33 #include "RenderThemeIOS.h"
34 #endif
35
36 namespace WebCore {
37
38 using namespace HTMLNames;
39
40 RenderButton::RenderButton(HTMLFormControlElement& element, PassRef<RenderStyle> style)
41     : RenderFlexibleBox(element, WTF::move(style))
42     , m_buttonText(0)
43     , m_inner(0)
44     , m_default(false)
45 {
46 }
47
48 RenderButton::~RenderButton()
49 {
50 }
51
52 HTMLFormControlElement& RenderButton::formControlElement() const
53 {
54     return toHTMLFormControlElement(nodeForNonAnonymous());
55 }
56
57 bool RenderButton::canBeSelectionLeaf() const
58 {
59     return formControlElement().hasEditableStyle();
60 }
61
62 bool RenderButton::hasLineIfEmpty() const
63 {
64     return formControlElement().toInputElement();
65 }
66
67 void RenderButton::addChild(RenderObject* newChild, RenderObject* beforeChild)
68 {
69     if (!m_inner) {
70         // Create an anonymous block.
71         ASSERT(!firstChild());
72         m_inner = createAnonymousBlock(style().display());
73         setupInnerStyle(&m_inner->style());
74         RenderFlexibleBox::addChild(m_inner);
75     }
76     
77     m_inner->addChild(newChild, beforeChild);
78 }
79
80 RenderObject* RenderButton::removeChild(RenderObject& oldChild)
81 {
82     // m_inner should be the only child, but checking for direct children who
83     // are not m_inner prevents security problems when that assumption is
84     // violated.
85     if (&oldChild == m_inner || !m_inner || oldChild.parent() == this) {
86         ASSERT(&oldChild == m_inner || !m_inner);
87         RenderObject* next = RenderFlexibleBox::removeChild(oldChild);
88         m_inner = nullptr;
89         return next;
90     } else
91         return m_inner->removeChild(oldChild);
92 }
93
94 void RenderButton::styleWillChange(StyleDifference diff, const RenderStyle& newStyle)
95 {
96     if (m_inner) {
97         // RenderBlock::setStyle is going to apply a new style to the inner block, which
98         // will have the initial flex value, 0. The current value is 1, because we set
99         // it right below. Here we change it back to 0 to avoid getting a spurious layout hint
100         // because of the difference. Same goes for the other properties.
101         // FIXME: Make this hack unnecessary.
102         m_inner->style().setFlexGrow(newStyle.initialFlexGrow());
103         m_inner->style().setMarginTop(newStyle.initialMargin());
104         m_inner->style().setMarginBottom(newStyle.initialMargin());
105     }
106     RenderBlock::styleWillChange(diff, newStyle);
107 }
108
109 void RenderButton::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
110 {
111     RenderBlock::styleDidChange(diff, oldStyle);
112
113     if (m_inner) // RenderBlock handled updating the anonymous block's style.
114         setupInnerStyle(&m_inner->style());
115
116     if (!m_default && theme().isDefault(*this)) {
117         if (!m_timer)
118             m_timer = std::make_unique<Timer<RenderButton>>(this, &RenderButton::timerFired);
119         m_timer->startRepeating(0.03);
120         m_default = true;
121     } else if (m_default && !theme().isDefault(*this)) {
122         m_default = false;
123         m_timer = nullptr;
124     }
125 }
126
127 void RenderButton::setupInnerStyle(RenderStyle* innerStyle) 
128 {
129     ASSERT(innerStyle->refCount() == 1);
130     // RenderBlock::createAnonymousBlock creates a new RenderStyle, so this is
131     // safe to modify.
132     // FIXME: I don't see how the comment above is accurate when this is called
133     // from the RenderButton::styleDidChange function.
134     innerStyle->setFlexGrow(1.0f);
135     // Use margin:auto instead of align-items:center to get safe centering, i.e.
136     // when the content overflows, treat it the same as align-items: flex-start.
137     innerStyle->setMarginTop(Length());
138     innerStyle->setMarginBottom(Length());
139     innerStyle->setFlexDirection(style().flexDirection());
140 }
141
142 void RenderButton::updateFromElement()
143 {
144     // If we're an input element, we may need to change our button text.
145     if (isHTMLInputElement(formControlElement())) {
146         HTMLInputElement& input = toHTMLInputElement(formControlElement());
147         String value = input.valueWithDefault();
148         setText(value);
149     }
150 }
151
152 void RenderButton::setText(const String& str)
153 {
154     if (str.isEmpty()) {
155         if (m_buttonText) {
156             m_buttonText->destroy();
157             m_buttonText = 0;
158         }
159     } else {
160         if (m_buttonText)
161             m_buttonText->setText(str.impl());
162         else {
163             m_buttonText = new RenderTextFragment(document(), str);
164             addChild(m_buttonText);
165         }
166     }
167 }
168
169 String RenderButton::text() const
170 {
171     return m_buttonText ? m_buttonText->text() : 0;
172 }
173
174 bool RenderButton::canHaveGeneratedChildren() const
175 {
176     // Input elements can't have generated children, but button elements can. We'll
177     // write the code assuming any other button types that might emerge in the future
178     // can also have children.
179     return !isHTMLInputElement(formControlElement());
180 }
181
182 LayoutRect RenderButton::controlClipRect(const LayoutPoint& additionalOffset) const
183 {
184     // Clip to the padding box to at least give content the extra padding space.
185     return LayoutRect(additionalOffset.x() + borderLeft(), additionalOffset.y() + borderTop(), width() - borderLeft() - borderRight(), height() - borderTop() - borderBottom());
186 }
187
188 void RenderButton::timerFired(Timer<RenderButton>&)
189 {
190     // FIXME Bug 25110: Ideally we would stop our timer when our Document
191     // enters the page cache. But we currently have no way of being notified
192     // when that happens, so we'll just ignore the timer firing as long as
193     // we're in the cache.
194     if (document().inPageCache())
195         return;
196
197     repaint();
198 }
199
200 #if PLATFORM(IOS)
201 void RenderButton::layout()
202 {
203     RenderFlexibleBox::layout();
204
205     // FIXME: We should not be adjusting styles during layout. See <rdar://problem/7675493>.
206     RenderThemeIOS::adjustRoundBorderRadius(style(), *this);
207 }
208 #endif
209
210 } // namespace WebCore