blob: 2a3a2bd0227013764c481b2292c36b9948a8bc25 [file] [log] [blame]
darinb9481ed2006-03-20 02:57:59 +00001/**
2 * This file is part of the theme implementation for form controls in WebCore.
3 *
4 * Copyright (C) 2005 Apple Computer, Inc.
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Library General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Library General Public License for more details.
15 *
16 * You should have received a copy of the GNU Library General Public License
17 * along with this library; see the file COPYING.LIB. If not, write to
18 * the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 * Boston, MA 02111-1307, USA.
20 */
21
22#include "config.h"
23#include "RenderTheme.h"
24
25#include "Document.h"
26#include "GraphicsContext.h"
27#include "HTMLInputElement.h"
darin98fa8b82006-03-20 08:03:57 +000028#include "HTMLNames.h"
darinb53ebdc2006-07-09 15:10:21 +000029#include "RenderStyle.h"
darinb9481ed2006-03-20 02:57:59 +000030
31// The methods in this file are shared by all themes on every platform.
32
33namespace WebCore {
34
35using namespace HTMLNames;
36
adele05abee32006-08-25 23:44:05 +000037void RenderTheme::adjustStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e,
38 bool UAHasAppearance, const BorderData& border, const BackgroundLayer& background, const Color& backgroundColor)
darinb9481ed2006-03-20 02:57:59 +000039{
adele05abee32006-08-25 23:44:05 +000040
adele88161252006-03-23 21:58:32 +000041 // Force inline and table display styles to be inline-block (except for table- which is block)
42 if (style->display() == INLINE || style->display() == INLINE_TABLE || style->display() == TABLE_ROW_GROUP ||
43 style->display() == TABLE_HEADER_GROUP || style->display() == TABLE_FOOTER_GROUP ||
44 style->display() == TABLE_ROW || style->display() == TABLE_COLUMN_GROUP || style->display() == TABLE_COLUMN ||
45 style->display() == TABLE_CELL || style->display() == TABLE_CAPTION)
darinb9481ed2006-03-20 02:57:59 +000046 style->setDisplay(INLINE_BLOCK);
adele88161252006-03-23 21:58:32 +000047 else if (style->display() == COMPACT || style->display() == RUN_IN || style->display() == LIST_ITEM || style->display() == TABLE)
darinb9481ed2006-03-20 02:57:59 +000048 style->setDisplay(BLOCK);
darinb9481ed2006-03-20 02:57:59 +000049
adele05abee32006-08-25 23:44:05 +000050 if (UAHasAppearance && theme()->isControlStyled(style, border, background, backgroundColor)) {
51 if (style->appearance() == MenulistAppearance)
52 style->setAppearance(MenulistButtonAppearance);
53 else
54 style->setAppearance(NoAppearance);
55 }
56
darinb9481ed2006-03-20 02:57:59 +000057 // Call the appropriate style adjustment method based off the appearance value.
58 switch (style->appearance()) {
59 case CheckboxAppearance:
60 return adjustCheckboxStyle(selector, style, e);
61 case RadioAppearance:
62 return adjustRadioStyle(selector, style, e);
63 case PushButtonAppearance:
64 case SquareButtonAppearance:
65 case ButtonAppearance:
66 return adjustButtonStyle(selector, style, e);
67 case TextFieldAppearance:
68 return adjustTextFieldStyle(selector, style, e);
adele275a6ad2006-05-17 23:32:38 +000069 case TextAreaAppearance:
70 return adjustTextAreaStyle(selector, style, e);
adelee0a75032006-07-06 05:47:30 +000071 case MenulistAppearance:
72 return adjustMenuListStyle(selector, style, e);
adele05abee32006-08-25 23:44:05 +000073 case MenulistButtonAppearance:
74 return adjustMenuListButtonStyle(selector, style, e);
darinb9481ed2006-03-20 02:57:59 +000075 default:
76 break;
77 }
78}
79
80bool RenderTheme::paint(RenderObject* o, const RenderObject::PaintInfo& i, const IntRect& r)
81{
82 // If painting is disabled, but we aren't updating control tints, then just bail.
83 // If we are updating control tints, just schedule a repaint if the theme supports tinting
84 // for that control.
85 if (i.p->updatingControlTints()) {
86 if (controlSupportsTints(o))
87 o->repaint();
88 return false;
89 }
90 if (i.p->paintingDisabled())
91 return false;
92
93 // Call the appropriate paint method based off the appearance value.
94 switch (o->style()->appearance()) {
95 case CheckboxAppearance:
96 return paintCheckbox(o, i, r);
97 case RadioAppearance:
98 return paintRadio(o, i, r);
99 case PushButtonAppearance:
100 case SquareButtonAppearance:
101 case ButtonAppearance:
102 return paintButton(o, i, r);
adelee0a75032006-07-06 05:47:30 +0000103 case MenulistAppearance:
104 return paintMenuList(o, i, r);
adele05abee32006-08-25 23:44:05 +0000105 case MenulistButtonAppearance:
darinb9481ed2006-03-20 02:57:59 +0000106 case TextFieldAppearance:
adele275a6ad2006-05-17 23:32:38 +0000107 case TextAreaAppearance:
adelef9890492006-09-29 23:13:49 +0000108 case ListboxAppearance:
darinb9481ed2006-03-20 02:57:59 +0000109 return true;
110 default:
111 break;
112 }
113
114 return true; // We don't support the appearance, so let the normal background/border paint.
115}
116
hyatt61802162006-03-31 19:03:19 +0000117bool RenderTheme::paintBorderOnly(RenderObject* o, const RenderObject::PaintInfo& i, const IntRect& r)
darinb9481ed2006-03-20 02:57:59 +0000118{
119 if (i.p->paintingDisabled())
120 return false;
121
122 // Call the appropriate paint method based off the appearance value.
123 switch (o->style()->appearance()) {
124 case TextFieldAppearance:
125 return paintTextField(o, i, r);
adelef9890492006-09-29 23:13:49 +0000126 case ListboxAppearance:
adele275a6ad2006-05-17 23:32:38 +0000127 case TextAreaAppearance:
128 return paintTextArea(o, i, r);
adele05abee32006-08-25 23:44:05 +0000129 case MenulistButtonAppearance:
130 return true;
131 case CheckboxAppearance:
132 case RadioAppearance:
133 case PushButtonAppearance:
134 case SquareButtonAppearance:
135 case ButtonAppearance:
136 case MenulistAppearance:
137 default:
138 break;
139 }
140
141 return false;
142}
143
144bool RenderTheme::paintDecorations(RenderObject* o, const RenderObject::PaintInfo& i, const IntRect& r)
145{
146 if (i.p->paintingDisabled())
147 return false;
148
149 // Call the appropriate paint method based off the appearance value.
150 switch (o->style()->appearance()) {
151 case MenulistButtonAppearance:
152 return paintMenuListButton(o, i, r);
153 case TextFieldAppearance:
154 case TextAreaAppearance:
adelef9890492006-09-29 23:13:49 +0000155 case ListboxAppearance:
darinb9481ed2006-03-20 02:57:59 +0000156 case CheckboxAppearance:
157 case RadioAppearance:
158 case PushButtonAppearance:
159 case SquareButtonAppearance:
160 case ButtonAppearance:
adelee0a75032006-07-06 05:47:30 +0000161 case MenulistAppearance:
darinb9481ed2006-03-20 02:57:59 +0000162 default:
163 break;
164 }
165
hyattbb9bd592006-03-31 11:45:15 +0000166 return false;
darinb9481ed2006-03-20 02:57:59 +0000167}
darinb9481ed2006-03-20 02:57:59 +0000168
hyattcb7f0652006-06-12 23:03:56 +0000169Color RenderTheme::activeSelectionBackgroundColor() const
hyatt76a0c022006-06-09 20:20:45 +0000170{
171 static Color activeSelectionColor;
172 if (!activeSelectionColor.isValid())
hyattcb7f0652006-06-12 23:03:56 +0000173 activeSelectionColor = platformActiveSelectionBackgroundColor().blendWithWhite();
hyatt76a0c022006-06-09 20:20:45 +0000174 return activeSelectionColor;
175}
176
hyattcb7f0652006-06-12 23:03:56 +0000177Color RenderTheme::inactiveSelectionBackgroundColor() const
hyatt76a0c022006-06-09 20:20:45 +0000178{
179 static Color inactiveSelectionColor;
180 if (!inactiveSelectionColor.isValid())
hyattcb7f0652006-06-12 23:03:56 +0000181 inactiveSelectionColor = platformInactiveSelectionBackgroundColor().blendWithWhite();
hyatt76a0c022006-06-09 20:20:45 +0000182 return inactiveSelectionColor;
183}
184
hyattcb7f0652006-06-12 23:03:56 +0000185Color RenderTheme::platformActiveSelectionBackgroundColor() const
hyatt76a0c022006-06-09 20:20:45 +0000186{
187 // Use a blue color by default if the platform theme doesn't define anything.
188 return Color(0, 0, 255);
189}
190
hyattcb7f0652006-06-12 23:03:56 +0000191Color RenderTheme::platformInactiveSelectionBackgroundColor() const
hyatt76a0c022006-06-09 20:20:45 +0000192{
193 // Use a grey color by default if the platform theme doesn't define anything.
194 return Color(128, 128, 128);
195}
hyattcb7f0652006-06-12 23:03:56 +0000196
197Color RenderTheme::platformActiveSelectionForegroundColor() const
198{
199 return Color();
200}
hyatt76a0c022006-06-09 20:20:45 +0000201
hyattcb7f0652006-06-12 23:03:56 +0000202Color RenderTheme::platformInactiveSelectionForegroundColor() const
203{
204 return Color();
205}
206
adelef9890492006-09-29 23:13:49 +0000207Color RenderTheme::activeListBoxSelectionBackgroundColor() const
208{
adeled08517f2006-10-13 14:36:26 +0000209 return activeSelectionBackgroundColor();
adelef9890492006-09-29 23:13:49 +0000210}
211
212Color RenderTheme::activeListBoxSelectionForegroundColor() const
213{
adeled08517f2006-10-13 14:36:26 +0000214 // Use a white color by default if the platform theme doesn't define anything.
adelef9890492006-09-29 23:13:49 +0000215 return Color(255, 255, 255);
216}
217
218Color RenderTheme::inactiveListBoxSelectionBackgroundColor() const
219{
adeled08517f2006-10-13 14:36:26 +0000220 return inactiveSelectionBackgroundColor();
adelef9890492006-09-29 23:13:49 +0000221}
222
223Color RenderTheme::inactiveListBoxSelectionForegroundColor() const
224{
adeled08517f2006-10-13 14:36:26 +0000225 // Use a black color by default if the platform theme doesn't define anything.
adelef9890492006-09-29 23:13:49 +0000226 return Color(0, 0, 0);
227}
228
darinb9481ed2006-03-20 02:57:59 +0000229short RenderTheme::baselinePosition(const RenderObject* o) const
230{
231 return o->height() + o->marginTop();
232}
233
234bool RenderTheme::isControlContainer(EAppearance appearance) const
235{
236 // There are more leaves than this, but we'll patch this function as we add support for
237 // more controls.
adele05abee32006-08-25 23:44:05 +0000238 return appearance != CheckboxAppearance && appearance != RadioAppearance;
darinb9481ed2006-03-20 02:57:59 +0000239}
240
241bool RenderTheme::isControlStyled(const RenderStyle* style, const BorderData& border, const BackgroundLayer& background,
242 const Color& backgroundColor) const
243{
244 switch (style->appearance()) {
245 case PushButtonAppearance:
246 case SquareButtonAppearance:
hyatt5f2591f2006-04-03 21:58:47 +0000247 case ButtonAppearance:
adelef9890492006-09-29 23:13:49 +0000248 case ListboxAppearance:
adelee0a75032006-07-06 05:47:30 +0000249 case MenulistAppearance:
adele275a6ad2006-05-17 23:32:38 +0000250 case TextFieldAppearance:
251 case TextAreaAppearance: {
darinb9481ed2006-03-20 02:57:59 +0000252 // Test the style to see if the UA border and background match.
253 return (style->border() != border ||
254 *style->backgroundLayers() != background ||
255 style->backgroundColor() != backgroundColor);
256 }
257 default:
258 return false;
259 }
260
261 return false;
262}
263
264bool RenderTheme::supportsFocusRing(const RenderStyle* style) const
265{
adelef9890492006-09-29 23:13:49 +0000266 return (style->hasAppearance() && style->appearance() != TextFieldAppearance && style->appearance() != TextAreaAppearance && style->appearance() != MenulistButtonAppearance && style->appearance() != ListboxAppearance);
darinb9481ed2006-03-20 02:57:59 +0000267}
268
269bool RenderTheme::stateChanged(RenderObject* o, ControlState state) const
270{
271 // Default implementation assumes the controls dont respond to changes in :hover state
hyatt5f2591f2006-04-03 21:58:47 +0000272 if (state == HoverState && !supportsHover(o->style()))
darinb9481ed2006-03-20 02:57:59 +0000273 return false;
274
275 // Assume pressed state is only responded to if the control is enabled.
276 if (state == PressedState && !isEnabled(o))
277 return false;
278
279 // Repaint the control.
280 o->repaint();
281 return true;
282}
283
284bool RenderTheme::isChecked(const RenderObject* o) const
285{
286 if (!o->element())
287 return false;
288 return o->element()->isChecked();
289}
290
291bool RenderTheme::isIndeterminate(const RenderObject* o) const
292{
293 if (!o->element())
294 return false;
295 return o->element()->isIndeterminate();
296}
297
298bool RenderTheme::isEnabled(const RenderObject* o) const
299{
300 if (!o->element())
301 return true;
302 return o->element()->isEnabled();
303}
304
305bool RenderTheme::isFocused(const RenderObject* o) const
306{
307 if (!o->element())
308 return false;
ggarenaa01dc02006-03-29 00:21:31 +0000309 return o->element() == o->element()->document()->focusNode();
darinb9481ed2006-03-20 02:57:59 +0000310}
311
312bool RenderTheme::isPressed(const RenderObject* o) const
313{
314 if (!o->element())
315 return false;
316 return o->element()->active();
317}
318
adele6c464a12006-04-19 22:21:22 +0000319bool RenderTheme::isReadOnlyControl(const RenderObject* o) const
hyatt5f2591f2006-04-03 21:58:47 +0000320{
321 if (!o->element())
322 return false;
adele6c464a12006-04-19 22:21:22 +0000323 return o->element()->isReadOnlyControl();
hyatt5f2591f2006-04-03 21:58:47 +0000324}
325
326bool RenderTheme::isHovered(const RenderObject* o) const
327{
328 if (!o->element())
329 return false;
330 return o->element()->hovered();
331}
332
darinb9481ed2006-03-20 02:57:59 +0000333void RenderTheme::adjustCheckboxStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const
334{
335 // A summary of the rules for checkbox designed to match WinIE:
336 // width/height - honored (WinIE actually scales its control for small widths, but lets it overflow for small heights.)
337 // font-size - not honored (control has no text), but we use it to decide which control size to use.
338 setCheckboxSize(style);
339
340 // padding - not honored by WinIE, needs to be removed.
341 style->resetPadding();
342
343 // border - honored by WinIE, but looks terrible (just paints in the control box and turns off the Windows XP theme)
344 // for now, we will not honor it.
345 style->resetBorder();
346}
347
348void RenderTheme::adjustRadioStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const
349{
350 // A summary of the rules for checkbox designed to match WinIE:
351 // width/height - honored (WinIE actually scales its control for small widths, but lets it overflow for small heights.)
352 // font-size - not honored (control has no text), but we use it to decide which control size to use.
353 setRadioSize(style);
354
355 // padding - not honored by WinIE, needs to be removed.
356 style->resetPadding();
357
358 // border - honored by WinIE, but looks terrible (just paints in the control box and turns off the Windows XP theme)
359 // for now, we will not honor it.
360 style->resetBorder();
361}
362
363void RenderTheme::adjustButtonStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const
364{
365 // Most platforms will completely honor all CSS, and so we have no need to adjust the style
366 // at all by default. We will still allow the theme a crack at setting up a desired vertical size.
367 setButtonSize(style);
368}
369
370void RenderTheme::adjustTextFieldStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const
371{
adele275a6ad2006-05-17 23:32:38 +0000372}
373
374void RenderTheme::adjustTextAreaStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const
375{
darinb9481ed2006-03-20 02:57:59 +0000376}
377
adelee0a75032006-07-06 05:47:30 +0000378void RenderTheme::adjustMenuListStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const
379{
380}
381
adele05abee32006-08-25 23:44:05 +0000382void RenderTheme::adjustMenuListButtonStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const
383{
384}
385
darinb9481ed2006-03-20 02:57:59 +0000386}