Adding a ShadowRoot to image-backed element causes a crash
[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
31 namespace WebCore {
32
33 using namespace HTMLNames;
34
35 RenderButton::RenderButton(Node* node)
36     : RenderDeprecatedFlexibleBox(node)
37     , m_buttonText(0)
38     , m_inner(0)
39     , m_default(false)
40 {
41 }
42
43 RenderButton::~RenderButton()
44 {
45 }
46
47 void RenderButton::addChild(RenderObject* newChild, RenderObject* beforeChild)
48 {
49     if (!m_inner) {
50         // Create an anonymous block.
51         ASSERT(!firstChild());
52         bool isFlexibleBox = style()->display() == BOX || style()->display() == INLINE_BOX;
53         m_inner = createAnonymousBlock(isFlexibleBox);
54         setupInnerStyle(m_inner->style());
55         RenderDeprecatedFlexibleBox::addChild(m_inner);
56     }
57     
58     m_inner->addChild(newChild, beforeChild);
59 }
60
61 void RenderButton::removeChild(RenderObject* oldChild)
62 {
63     if (oldChild == m_inner || !m_inner) {
64         RenderDeprecatedFlexibleBox::removeChild(oldChild);
65         m_inner = 0;
66     } else
67         m_inner->removeChild(oldChild);
68 }
69
70 void RenderButton::styleWillChange(StyleDifference diff, const RenderStyle* newStyle)
71 {
72     if (m_inner) {
73         // RenderBlock::setStyle is going to apply a new style to the inner block, which
74         // will have the initial box flex value, 0. The current value is 1, because we set
75         // it right below. Here we change it back to 0 to avoid getting a spurious layout hint
76         // because of the difference.
77         m_inner->style()->setBoxFlex(0);
78     }
79     RenderBlock::styleWillChange(diff, newStyle);
80 }
81
82 void RenderButton::styleDidChange(StyleDifference diff, const RenderStyle* oldStyle)
83 {
84     RenderBlock::styleDidChange(diff, oldStyle);
85
86     if (m_buttonText)
87         m_buttonText->setStyle(style());
88     if (m_inner) // RenderBlock handled updating the anonymous block's style.
89         setupInnerStyle(m_inner->style());
90
91     if (!m_default && theme()->isDefault(this)) {
92         if (!m_timer)
93             m_timer = adoptPtr(new Timer<RenderButton>(this, &RenderButton::timerFired));
94         m_timer->startRepeating(0.03);
95         m_default = true;
96     } else if (m_default && !theme()->isDefault(this)) {
97         m_default = false;
98         m_timer.clear();
99     }
100 }
101
102 void RenderButton::setupInnerStyle(RenderStyle* innerStyle) 
103 {
104     ASSERT(innerStyle->refCount() == 1);
105     // RenderBlock::createAnonymousBlock creates a new RenderStyle, so this is
106     // safe to modify.
107     innerStyle->setBoxFlex(1.0f);
108     innerStyle->setBoxOrient(style()->boxOrient());
109 }
110
111 void RenderButton::updateFromElement()
112 {
113     // If we're an input element, we may need to change our button text.
114     if (node()->hasTagName(inputTag)) {
115         HTMLInputElement* input = static_cast<HTMLInputElement*>(node());
116         String value = input->valueWithDefault();
117         setText(value);
118     }
119 }
120
121 void RenderButton::setText(const String& str)
122 {
123     if (str.isEmpty()) {
124         if (m_buttonText) {
125             m_buttonText->destroy();
126             m_buttonText = 0;
127         }
128     } else {
129         if (m_buttonText)
130             m_buttonText->setText(str.impl());
131         else {
132             m_buttonText = new (renderArena()) RenderTextFragment(document(), str.impl());
133             m_buttonText->setStyle(style());
134             addChild(m_buttonText);
135         }
136     }
137 }
138
139 String RenderButton::text() const
140 {
141     return m_buttonText ? m_buttonText->text() : 0;
142 }
143
144 void RenderButton::updateBeforeAfterContent(PseudoId type)
145 {
146     if (m_inner)
147         m_inner->children()->updateBeforeAfterContent(m_inner, type, this);
148     else
149         children()->updateBeforeAfterContent(this, type);
150 }
151
152 RenderText* RenderButton::buttonText() const
153 {
154     return m_buttonText;
155 }
156
157 LayoutRect RenderButton::controlClipRect(const LayoutPoint& additionalOffset) const
158 {
159     // Clip to the padding box to at least give content the extra padding space.
160     return LayoutRect(additionalOffset.x() + borderLeft(), additionalOffset.y() + borderTop(), width() - borderLeft() - borderRight(), height() - borderTop() - borderBottom());
161 }
162
163 void RenderButton::timerFired(Timer<RenderButton>*)
164 {
165     // FIXME Bug 25110: Ideally we would stop our timer when our Document
166     // enters the page cache. But we currently have no way of being notified
167     // when that happens, so we'll just ignore the timer firing as long as
168     // we're in the cache.
169     if (document()->inPageCache())
170         return;
171
172     repaint();
173 }
174
175 } // namespace WebCore