ed8a1eb59114b73d4b32d86f5945e661a293c8c5
[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 "RenderTreeBuilder.h"
31 #include "StyleInheritedData.h"
32 #include <wtf/IsoMallocInlines.h>
33
34 #if PLATFORM(IOS)
35 #include "RenderThemeIOS.h"
36 #endif
37
38 namespace WebCore {
39
40 using namespace HTMLNames;
41
42 WTF_MAKE_ISO_ALLOCATED_IMPL(RenderButton);
43
44 RenderButton::RenderButton(HTMLFormControlElement& element, RenderStyle&& style)
45     : RenderFlexibleBox(element, WTFMove(style))
46 {
47 }
48
49 RenderButton::~RenderButton() = default;
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::setInnerRenderer(RenderBlock& innerRenderer)
67 {
68     ASSERT(!m_inner.get());
69     m_inner = makeWeakPtr(innerRenderer);
70     updateAnonymousChildStyle(m_inner->mutableStyle());
71 }
72
73 RenderPtr<RenderObject> RenderButton::takeChild(RenderTreeBuilder& builder, RenderObject& oldChild)
74 {
75     // m_inner should be the only child, but checking for direct children who
76     // are not m_inner prevents security problems when that assumption is
77     // violated.
78     if (&oldChild == m_inner || !m_inner || oldChild.parent() == this) {
79         ASSERT(&oldChild == m_inner || !m_inner);
80         return RenderFlexibleBox::takeChild(builder, oldChild);
81     }
82     return m_inner->takeChild(builder, oldChild);
83 }
84     
85 void RenderButton::updateAnonymousChildStyle(RenderStyle& childStyle) const
86 {
87     childStyle.setFlexGrow(1.0f);
88     // min-width: 0; is needed for correct shrinking.
89     childStyle.setMinWidth(Length(0, Fixed));
90     // Use margin:auto instead of align-items:center to get safe centering, i.e.
91     // when the content overflows, treat it the same as align-items: flex-start.
92     childStyle.setMarginTop(Length());
93     childStyle.setMarginBottom(Length());
94     childStyle.setFlexDirection(style().flexDirection());
95     childStyle.setJustifyContent(style().justifyContent());
96     childStyle.setFlexWrap(style().flexWrap());
97     childStyle.setAlignItems(style().alignItems());
98     childStyle.setAlignContent(style().alignContent());
99 }
100
101 void RenderButton::updateFromElement()
102 {
103     // If we're an input element, we may need to change our button text.
104     if (is<HTMLInputElement>(formControlElement())) {
105         HTMLInputElement& input = downcast<HTMLInputElement>(formControlElement());
106         String value = input.valueWithDefault();
107         setText(value);
108     }
109 }
110
111 void RenderButton::setText(const String& str)
112 {
113     if (!m_buttonText && str.isEmpty())
114         return;
115
116     if (!m_buttonText) {
117         auto newButtonText = createRenderer<RenderTextFragment>(document(), str);
118         m_buttonText = makeWeakPtr(*newButtonText);
119         // FIXME: This mutation should go through the normal RenderTreeBuilder path.
120         if (RenderTreeBuilder::current())
121             RenderTreeBuilder::current()->insertChild(*this, WTFMove(newButtonText));
122         else
123             RenderTreeBuilder(*document().renderView()).insertChild(*this, WTFMove(newButtonText));
124         return;
125     }
126
127     if (!str.isEmpty()) {
128         m_buttonText->setText(str.impl());
129         return;
130     }
131     if (RenderTreeBuilder::current())
132         m_buttonText->removeFromParentAndDestroy(*RenderTreeBuilder::current());
133     else {
134         RenderTreeBuilder builder(*document().renderView());
135         m_buttonText->removeFromParentAndDestroy(builder);
136     }
137 }
138
139 String RenderButton::text() const
140 {
141     if (m_buttonText)
142         return m_buttonText->text();
143     return { };
144 }
145
146 bool RenderButton::canHaveGeneratedChildren() const
147 {
148     // Input elements can't have generated children, but button elements can. We'll
149     // write the code assuming any other button types that might emerge in the future
150     // can also have children.
151     return !is<HTMLInputElement>(formControlElement());
152 }
153
154 LayoutRect RenderButton::controlClipRect(const LayoutPoint& additionalOffset) const
155 {
156     // Clip to the padding box to at least give content the extra padding space.
157     return LayoutRect(additionalOffset.x() + borderLeft(), additionalOffset.y() + borderTop(), width() - borderLeft() - borderRight(), height() - borderTop() - borderBottom());
158 }
159
160 #if PLATFORM(IOS)
161 void RenderButton::layout()
162 {
163     RenderFlexibleBox::layout();
164
165     // FIXME: We should not be adjusting styles during layout. See <rdar://problem/7675493>.
166     RenderThemeIOS::adjustRoundBorderRadius(mutableStyle(), *this);
167 }
168 #endif
169
170 } // namespace WebCore