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