darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 1 | /** |
| 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" |
darin | 98fa8b8 | 2006-03-20 08:03:57 +0000 | [diff] [blame] | 28 | #include "HTMLNames.h" |
darin | b53ebdc | 2006-07-09 15:10:21 +0000 | [diff] [blame] | 29 | #include "RenderStyle.h" |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 30 | |
| 31 | // The methods in this file are shared by all themes on every platform. |
| 32 | |
| 33 | namespace WebCore { |
| 34 | |
| 35 | using namespace HTMLNames; |
| 36 | |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 37 | void RenderTheme::adjustStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e, |
| 38 | bool UAHasAppearance, const BorderData& border, const BackgroundLayer& background, const Color& backgroundColor) |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 39 | { |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 40 | |
adele | 8816125 | 2006-03-23 21:58:32 +0000 | [diff] [blame] | 41 | // 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) |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 46 | style->setDisplay(INLINE_BLOCK); |
adele | 8816125 | 2006-03-23 21:58:32 +0000 | [diff] [blame] | 47 | else if (style->display() == COMPACT || style->display() == RUN_IN || style->display() == LIST_ITEM || style->display() == TABLE) |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 48 | style->setDisplay(BLOCK); |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 49 | |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 50 | 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 | |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 57 | // 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); |
adele | 275a6ad | 2006-05-17 23:32:38 +0000 | [diff] [blame] | 69 | case TextAreaAppearance: |
| 70 | return adjustTextAreaStyle(selector, style, e); |
adele | e0a7503 | 2006-07-06 05:47:30 +0000 | [diff] [blame] | 71 | case MenulistAppearance: |
| 72 | return adjustMenuListStyle(selector, style, e); |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 73 | case MenulistButtonAppearance: |
| 74 | return adjustMenuListButtonStyle(selector, style, e); |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 75 | default: |
| 76 | break; |
| 77 | } |
| 78 | } |
| 79 | |
| 80 | bool 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); |
adele | e0a7503 | 2006-07-06 05:47:30 +0000 | [diff] [blame] | 103 | case MenulistAppearance: |
| 104 | return paintMenuList(o, i, r); |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 105 | case MenulistButtonAppearance: |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 106 | case TextFieldAppearance: |
adele | 275a6ad | 2006-05-17 23:32:38 +0000 | [diff] [blame] | 107 | case TextAreaAppearance: |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 108 | case ListboxAppearance: |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 109 | 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 | |
hyatt | 6180216 | 2006-03-31 19:03:19 +0000 | [diff] [blame] | 117 | bool RenderTheme::paintBorderOnly(RenderObject* o, const RenderObject::PaintInfo& i, const IntRect& r) |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 118 | { |
| 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); |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 126 | case ListboxAppearance: |
adele | 275a6ad | 2006-05-17 23:32:38 +0000 | [diff] [blame] | 127 | case TextAreaAppearance: |
| 128 | return paintTextArea(o, i, r); |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 129 | 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 | |
| 144 | bool 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: |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 155 | case ListboxAppearance: |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 156 | case CheckboxAppearance: |
| 157 | case RadioAppearance: |
| 158 | case PushButtonAppearance: |
| 159 | case SquareButtonAppearance: |
| 160 | case ButtonAppearance: |
adele | e0a7503 | 2006-07-06 05:47:30 +0000 | [diff] [blame] | 161 | case MenulistAppearance: |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 162 | default: |
| 163 | break; |
| 164 | } |
| 165 | |
hyatt | bb9bd59 | 2006-03-31 11:45:15 +0000 | [diff] [blame] | 166 | return false; |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 167 | } |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 168 | |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 169 | Color RenderTheme::activeSelectionBackgroundColor() const |
hyatt | 76a0c02 | 2006-06-09 20:20:45 +0000 | [diff] [blame] | 170 | { |
| 171 | static Color activeSelectionColor; |
| 172 | if (!activeSelectionColor.isValid()) |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 173 | activeSelectionColor = platformActiveSelectionBackgroundColor().blendWithWhite(); |
hyatt | 76a0c02 | 2006-06-09 20:20:45 +0000 | [diff] [blame] | 174 | return activeSelectionColor; |
| 175 | } |
| 176 | |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 177 | Color RenderTheme::inactiveSelectionBackgroundColor() const |
hyatt | 76a0c02 | 2006-06-09 20:20:45 +0000 | [diff] [blame] | 178 | { |
| 179 | static Color inactiveSelectionColor; |
| 180 | if (!inactiveSelectionColor.isValid()) |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 181 | inactiveSelectionColor = platformInactiveSelectionBackgroundColor().blendWithWhite(); |
hyatt | 76a0c02 | 2006-06-09 20:20:45 +0000 | [diff] [blame] | 182 | return inactiveSelectionColor; |
| 183 | } |
| 184 | |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 185 | Color RenderTheme::platformActiveSelectionBackgroundColor() const |
hyatt | 76a0c02 | 2006-06-09 20:20:45 +0000 | [diff] [blame] | 186 | { |
| 187 | // Use a blue color by default if the platform theme doesn't define anything. |
| 188 | return Color(0, 0, 255); |
| 189 | } |
| 190 | |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 191 | Color RenderTheme::platformInactiveSelectionBackgroundColor() const |
hyatt | 76a0c02 | 2006-06-09 20:20:45 +0000 | [diff] [blame] | 192 | { |
| 193 | // Use a grey color by default if the platform theme doesn't define anything. |
| 194 | return Color(128, 128, 128); |
| 195 | } |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 196 | |
| 197 | Color RenderTheme::platformActiveSelectionForegroundColor() const |
| 198 | { |
| 199 | return Color(); |
| 200 | } |
hyatt | 76a0c02 | 2006-06-09 20:20:45 +0000 | [diff] [blame] | 201 | |
hyatt | cb7f065 | 2006-06-12 23:03:56 +0000 | [diff] [blame] | 202 | Color RenderTheme::platformInactiveSelectionForegroundColor() const |
| 203 | { |
| 204 | return Color(); |
| 205 | } |
| 206 | |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 207 | Color RenderTheme::activeListBoxSelectionBackgroundColor() const |
| 208 | { |
adele | d08517f | 2006-10-13 14:36:26 +0000 | [diff] [blame] | 209 | return activeSelectionBackgroundColor(); |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 210 | } |
| 211 | |
| 212 | Color RenderTheme::activeListBoxSelectionForegroundColor() const |
| 213 | { |
adele | d08517f | 2006-10-13 14:36:26 +0000 | [diff] [blame] | 214 | // Use a white color by default if the platform theme doesn't define anything. |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 215 | return Color(255, 255, 255); |
| 216 | } |
| 217 | |
| 218 | Color RenderTheme::inactiveListBoxSelectionBackgroundColor() const |
| 219 | { |
adele | d08517f | 2006-10-13 14:36:26 +0000 | [diff] [blame] | 220 | return inactiveSelectionBackgroundColor(); |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 221 | } |
| 222 | |
| 223 | Color RenderTheme::inactiveListBoxSelectionForegroundColor() const |
| 224 | { |
adele | d08517f | 2006-10-13 14:36:26 +0000 | [diff] [blame] | 225 | // Use a black color by default if the platform theme doesn't define anything. |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 226 | return Color(0, 0, 0); |
| 227 | } |
| 228 | |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 229 | short RenderTheme::baselinePosition(const RenderObject* o) const |
| 230 | { |
| 231 | return o->height() + o->marginTop(); |
| 232 | } |
| 233 | |
| 234 | bool 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. |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 238 | return appearance != CheckboxAppearance && appearance != RadioAppearance; |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 239 | } |
| 240 | |
| 241 | bool 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: |
hyatt | 5f2591f | 2006-04-03 21:58:47 +0000 | [diff] [blame] | 247 | case ButtonAppearance: |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 248 | case ListboxAppearance: |
adele | e0a7503 | 2006-07-06 05:47:30 +0000 | [diff] [blame] | 249 | case MenulistAppearance: |
adele | 275a6ad | 2006-05-17 23:32:38 +0000 | [diff] [blame] | 250 | case TextFieldAppearance: |
| 251 | case TextAreaAppearance: { |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 252 | // 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 | |
| 264 | bool RenderTheme::supportsFocusRing(const RenderStyle* style) const |
| 265 | { |
adele | f989049 | 2006-09-29 23:13:49 +0000 | [diff] [blame] | 266 | return (style->hasAppearance() && style->appearance() != TextFieldAppearance && style->appearance() != TextAreaAppearance && style->appearance() != MenulistButtonAppearance && style->appearance() != ListboxAppearance); |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 267 | } |
| 268 | |
| 269 | bool RenderTheme::stateChanged(RenderObject* o, ControlState state) const |
| 270 | { |
| 271 | // Default implementation assumes the controls dont respond to changes in :hover state |
hyatt | 5f2591f | 2006-04-03 21:58:47 +0000 | [diff] [blame] | 272 | if (state == HoverState && !supportsHover(o->style())) |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 273 | 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 | |
| 284 | bool RenderTheme::isChecked(const RenderObject* o) const |
| 285 | { |
| 286 | if (!o->element()) |
| 287 | return false; |
| 288 | return o->element()->isChecked(); |
| 289 | } |
| 290 | |
| 291 | bool RenderTheme::isIndeterminate(const RenderObject* o) const |
| 292 | { |
| 293 | if (!o->element()) |
| 294 | return false; |
| 295 | return o->element()->isIndeterminate(); |
| 296 | } |
| 297 | |
| 298 | bool RenderTheme::isEnabled(const RenderObject* o) const |
| 299 | { |
| 300 | if (!o->element()) |
| 301 | return true; |
| 302 | return o->element()->isEnabled(); |
| 303 | } |
| 304 | |
| 305 | bool RenderTheme::isFocused(const RenderObject* o) const |
| 306 | { |
| 307 | if (!o->element()) |
| 308 | return false; |
ggaren | aa01dc0 | 2006-03-29 00:21:31 +0000 | [diff] [blame] | 309 | return o->element() == o->element()->document()->focusNode(); |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 310 | } |
| 311 | |
| 312 | bool RenderTheme::isPressed(const RenderObject* o) const |
| 313 | { |
| 314 | if (!o->element()) |
| 315 | return false; |
| 316 | return o->element()->active(); |
| 317 | } |
| 318 | |
adele | 6c464a1 | 2006-04-19 22:21:22 +0000 | [diff] [blame] | 319 | bool RenderTheme::isReadOnlyControl(const RenderObject* o) const |
hyatt | 5f2591f | 2006-04-03 21:58:47 +0000 | [diff] [blame] | 320 | { |
| 321 | if (!o->element()) |
| 322 | return false; |
adele | 6c464a1 | 2006-04-19 22:21:22 +0000 | [diff] [blame] | 323 | return o->element()->isReadOnlyControl(); |
hyatt | 5f2591f | 2006-04-03 21:58:47 +0000 | [diff] [blame] | 324 | } |
| 325 | |
| 326 | bool RenderTheme::isHovered(const RenderObject* o) const |
| 327 | { |
| 328 | if (!o->element()) |
| 329 | return false; |
| 330 | return o->element()->hovered(); |
| 331 | } |
| 332 | |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 333 | void 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 | |
| 348 | void 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 | |
| 363 | void 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 | |
| 370 | void RenderTheme::adjustTextFieldStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const |
| 371 | { |
adele | 275a6ad | 2006-05-17 23:32:38 +0000 | [diff] [blame] | 372 | } |
| 373 | |
| 374 | void RenderTheme::adjustTextAreaStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const |
| 375 | { |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 376 | } |
| 377 | |
adele | e0a7503 | 2006-07-06 05:47:30 +0000 | [diff] [blame] | 378 | void RenderTheme::adjustMenuListStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const |
| 379 | { |
| 380 | } |
| 381 | |
adele | 05abee3 | 2006-08-25 23:44:05 +0000 | [diff] [blame] | 382 | void RenderTheme::adjustMenuListButtonStyle(CSSStyleSelector* selector, RenderStyle* style, Element* e) const |
| 383 | { |
| 384 | } |
| 385 | |
darin | b9481ed | 2006-03-20 02:57:59 +0000 | [diff] [blame] | 386 | } |