• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "ui/native_theme/native_theme_win.h"
6 
7 #include <windows.h>
8 #include <uxtheme.h>
9 #include <vsstyle.h>
10 #include <vssym32.h>
11 
12 #include "base/basictypes.h"
13 #include "base/logging.h"
14 #include "base/memory/scoped_ptr.h"
15 #include "base/win/scoped_gdi_object.h"
16 #include "base/win/scoped_hdc.h"
17 #include "base/win/scoped_select_object.h"
18 #include "base/win/windows_version.h"
19 #include "skia/ext/bitmap_platform_device.h"
20 #include "skia/ext/platform_canvas.h"
21 #include "skia/ext/skia_utils_win.h"
22 #include "third_party/skia/include/core/SkCanvas.h"
23 #include "third_party/skia/include/core/SkColorPriv.h"
24 #include "third_party/skia/include/core/SkShader.h"
25 #include "ui/gfx/color_utils.h"
26 #include "ui/gfx/gdi_util.h"
27 #include "ui/gfx/rect.h"
28 #include "ui/gfx/rect_conversions.h"
29 #include "ui/gfx/win/dpi.h"
30 #include "ui/native_theme/common_theme.h"
31 
32 // This was removed from Winvers.h but is still used.
33 #if !defined(COLOR_MENUHIGHLIGHT)
34 #define COLOR_MENUHIGHLIGHT 29
35 #endif
36 
37 namespace {
38 
39 // TODO: Obtain the correct colors using GetSysColor.
40 // Theme colors returned by GetSystemColor().
41 const SkColor kInvalidColorIdColor = SkColorSetRGB(255, 0, 128);
42 // Dialogs:
43 const SkColor kDialogBackgroundColor = SkColorSetRGB(251, 251, 251);
44 // FocusableBorder:
45 const SkColor kFocusedBorderColor = SkColorSetRGB(0x4d, 0x90, 0xfe);
46 const SkColor kUnfocusedBorderColor = SkColorSetRGB(0xd9, 0xd9, 0xd9);
47 // Button:
48 const SkColor kButtonBackgroundColor = SkColorSetRGB(0xde, 0xde, 0xde);
49 const SkColor kButtonHighlightColor = SkColorSetARGB(200, 255, 255, 255);
50 const SkColor kButtonHoverColor = SkColorSetRGB(6, 45, 117);
51 const SkColor kButtonHoverBackgroundColor = SkColorSetRGB(0xEA, 0xEA, 0xEA);
52 // MenuItem:
53 const SkColor kEnabledMenuItemForegroundColor = SkColorSetRGB(6, 45, 117);
54 const SkColor kDisabledMenuItemForegroundColor = SkColorSetRGB(161, 161, 146);
55 const SkColor kFocusedMenuItemBackgroundColor = SkColorSetRGB(246, 249, 253);
56 const SkColor kMenuSeparatorColor = SkColorSetARGB(50, 0, 0, 0);
57 // Table:
58 const SkColor kTreeSelectionBackgroundUnfocused = SkColorSetRGB(240, 240, 240);
59 
60 // Windows system color IDs cached and updated by the native theme.
61 const int kSystemColors[] = {
62   COLOR_3DFACE,
63   COLOR_BTNTEXT,
64   COLOR_GRAYTEXT,
65   COLOR_HIGHLIGHT,
66   COLOR_HIGHLIGHTTEXT,
67   COLOR_SCROLLBAR,
68   COLOR_WINDOW,
69   COLOR_WINDOWTEXT,
70   COLOR_BTNFACE,
71   COLOR_MENUHIGHLIGHT,
72 };
73 
SetCheckerboardShader(SkPaint * paint,const RECT & align_rect)74 void SetCheckerboardShader(SkPaint* paint, const RECT& align_rect) {
75   // Create a 2x2 checkerboard pattern using the 3D face and highlight colors.
76   const SkColor face = color_utils::GetSysSkColor(COLOR_3DFACE);
77   const SkColor highlight = color_utils::GetSysSkColor(COLOR_3DHILIGHT);
78   SkColor buffer[] = { face, highlight, highlight, face };
79   // Confusing bit: we first create a temporary bitmap with our desired pattern,
80   // then copy it to another bitmap.  The temporary bitmap doesn't take
81   // ownership of the pixel data, and so will point to garbage when this
82   // function returns.  The copy will copy the pixel data into a place owned by
83   // the bitmap, which is in turn owned by the shader, etc., so it will live
84   // until we're done using it.
85   SkImageInfo info = SkImageInfo::MakeN32Premul(2, 2);
86   SkBitmap temp_bitmap;
87   temp_bitmap.installPixels(info, buffer, info.minRowBytes());
88   SkBitmap bitmap;
89   temp_bitmap.copyTo(&bitmap);
90 
91   // Align the pattern with the upper corner of |align_rect|.
92   SkMatrix local_matrix;
93   local_matrix.setTranslate(SkIntToScalar(align_rect.left),
94                             SkIntToScalar(align_rect.top));
95   skia::RefPtr<SkShader> shader =
96       skia::AdoptRef(SkShader::CreateBitmapShader(bitmap,
97                                                   SkShader::kRepeat_TileMode,
98                                                   SkShader::kRepeat_TileMode,
99                                                   &local_matrix));
100   paint->setShader(shader.get());
101 }
102 
103 //    <-a->
104 // [  *****             ]
105 //  ____ |              |
106 //  <-a-> <------b----->
107 // a: object_width
108 // b: frame_width
109 // *: animating object
110 //
111 // - the animation goes from "[" to "]" repeatedly.
112 // - the animation offset is at first "|"
113 //
ComputeAnimationProgress(int frame_width,int object_width,int pixels_per_second,double animated_seconds)114 int ComputeAnimationProgress(int frame_width,
115                              int object_width,
116                              int pixels_per_second,
117                              double animated_seconds) {
118   int animation_width = frame_width + object_width;
119   double interval = static_cast<double>(animation_width) / pixels_per_second;
120   double ratio = fmod(animated_seconds, interval) / interval;
121   return static_cast<int>(animation_width * ratio) - object_width;
122 }
123 
InsetRect(const RECT * rect,int size)124 RECT InsetRect(const RECT* rect, int size) {
125   gfx::Rect result(*rect);
126   result.Inset(size, size);
127   return result.ToRECT();
128 }
129 
130 }  // namespace
131 
132 namespace ui {
133 
IsThemingActive() const134 bool NativeThemeWin::IsThemingActive() const {
135   return is_theme_active_ && is_theme_active_();
136 }
137 
IsUsingHighContrastTheme() const138 bool NativeThemeWin::IsUsingHighContrastTheme() const {
139   if (is_using_high_contrast_valid_)
140     return is_using_high_contrast_;
141   HIGHCONTRAST result;
142   result.cbSize = sizeof(HIGHCONTRAST);
143   is_using_high_contrast_ =
144       SystemParametersInfo(SPI_GETHIGHCONTRAST, result.cbSize, &result, 0) &&
145       (result.dwFlags & HCF_HIGHCONTRASTON) == HCF_HIGHCONTRASTON;
146   is_using_high_contrast_valid_ = true;
147   return is_using_high_contrast_;
148 }
149 
GetThemeColor(ThemeName theme,int part_id,int state_id,int prop_id,SkColor * color) const150 HRESULT NativeThemeWin::GetThemeColor(ThemeName theme,
151                                       int part_id,
152                                       int state_id,
153                                       int prop_id,
154                                       SkColor* color) const {
155   HANDLE handle = GetThemeHandle(theme);
156   if (!handle || !get_theme_color_)
157     return E_NOTIMPL;
158   COLORREF color_ref;
159   if (get_theme_color_(handle, part_id, state_id, prop_id, &color_ref) != S_OK)
160     return E_NOTIMPL;
161   *color = skia::COLORREFToSkColor(color_ref);
162   return S_OK;
163 }
164 
GetThemeColorWithDefault(ThemeName theme,int part_id,int state_id,int prop_id,int default_sys_color) const165 SkColor NativeThemeWin::GetThemeColorWithDefault(ThemeName theme,
166                                                  int part_id,
167                                                  int state_id,
168                                                  int prop_id,
169                                                  int default_sys_color) const {
170   SkColor color;
171   return (GetThemeColor(theme, part_id, state_id, prop_id, &color) == S_OK) ?
172       color : color_utils::GetSysSkColor(default_sys_color);
173 }
174 
GetThemeBorderSize(ThemeName theme) const175 gfx::Size NativeThemeWin::GetThemeBorderSize(ThemeName theme) const {
176   // For simplicity use the wildcard state==0, part==0, since it works
177   // for the cases we currently depend on.
178   int border;
179   return (GetThemeInt(theme, 0, 0, TMT_BORDERSIZE, &border) == S_OK) ?
180       gfx::Size(border, border) :
181       gfx::Size(GetSystemMetrics(SM_CXEDGE), GetSystemMetrics(SM_CYEDGE));
182 }
183 
DisableTheming() const184 void NativeThemeWin::DisableTheming() const {
185   if (set_theme_properties_)
186     set_theme_properties_(0);
187 }
188 
CloseHandles() const189 void NativeThemeWin::CloseHandles() const {
190   if (!close_theme_)
191     return;
192 
193   for (int i = 0; i < LAST; ++i) {
194     if (theme_handles_[i]) {
195       close_theme_(theme_handles_[i]);
196       theme_handles_[i] = NULL;
197     }
198   }
199 }
200 
IsClassicTheme(ThemeName name) const201 bool NativeThemeWin::IsClassicTheme(ThemeName name) const {
202   return !theme_dll_ || !GetThemeHandle(name);
203 }
204 
205 // static
instance()206 NativeThemeWin* NativeThemeWin::instance() {
207   CR_DEFINE_STATIC_LOCAL(NativeThemeWin, s_native_theme, ());
208   return &s_native_theme;
209 }
210 
GetPartSize(Part part,State state,const ExtraParams & extra) const211 gfx::Size NativeThemeWin::GetPartSize(Part part,
212                                       State state,
213                                       const ExtraParams& extra) const {
214   gfx::Size part_size = CommonThemeGetPartSize(part, state, extra);
215   if (!part_size.IsEmpty())
216     return part_size;
217 
218   // The GetThemePartSize call below returns the default size without
219   // accounting for user customization (crbug/218291).
220   switch (part) {
221     case kScrollbarDownArrow:
222     case kScrollbarLeftArrow:
223     case kScrollbarRightArrow:
224     case kScrollbarUpArrow:
225     case kScrollbarHorizontalThumb:
226     case kScrollbarVerticalThumb:
227     case kScrollbarHorizontalTrack:
228     case kScrollbarVerticalTrack: {
229       int size = gfx::win::GetSystemMetricsInDIP(SM_CXVSCROLL);
230       if (size == 0)
231         size = 17;
232       return gfx::Size(size, size);
233     }
234     default:
235       break;
236   }
237 
238   int part_id = GetWindowsPart(part, state, extra);
239   int state_id = GetWindowsState(part, state, extra);
240 
241   base::win::ScopedGetDC screen_dc(NULL);
242   SIZE size;
243   if (SUCCEEDED(GetThemePartSize(GetThemeName(part), screen_dc, part_id,
244                                  state_id, NULL, TS_TRUE, &size)))
245     return gfx::Size(size.cx, size.cy);
246 
247   // TODO(rogerta): For now, we need to support radio buttons and checkboxes
248   // when theming is not enabled.  Support for other parts can be added
249   // if/when needed.
250   return (part == kCheckbox || part == kRadio) ?
251       gfx::Size(13, 13) : gfx::Size();
252 }
253 
Paint(SkCanvas * canvas,Part part,State state,const gfx::Rect & rect,const ExtraParams & extra) const254 void NativeThemeWin::Paint(SkCanvas* canvas,
255                            Part part,
256                            State state,
257                            const gfx::Rect& rect,
258                            const ExtraParams& extra) const {
259   if (rect.IsEmpty())
260     return;
261 
262   switch (part) {
263     case kComboboxArrow:
264       CommonThemePaintComboboxArrow(canvas, rect);
265       return;
266     case kMenuPopupGutter:
267       CommonThemePaintMenuGutter(canvas, rect);
268       return;
269     case kMenuPopupSeparator:
270       CommonThemePaintMenuSeparator(canvas, rect, extra.menu_separator);
271       return;
272     case kMenuPopupBackground:
273       CommonThemePaintMenuBackground(canvas, rect);
274       return;
275     case kMenuItemBackground:
276       CommonThemePaintMenuItemBackground(canvas, state, rect);
277       return;
278     default:
279       break;
280   }
281 
282   bool needs_paint_indirect = false;
283   if (!skia::SupportsPlatformPaint(canvas)) {
284     // This block will only get hit with --enable-accelerated-drawing flag.
285     needs_paint_indirect = true;
286   } else {
287     // Scrollbar components on Windows Classic theme (on all Windows versions)
288     // have particularly problematic alpha values, so always draw them
289     // indirectly. In addition, scrollbar thumbs and grippers for the Windows XP
290     // theme (available only on Windows XP) also need their alpha values
291     // fixed.
292     switch (part) {
293       case kScrollbarDownArrow:
294       case kScrollbarUpArrow:
295       case kScrollbarLeftArrow:
296       case kScrollbarRightArrow:
297         needs_paint_indirect = !GetThemeHandle(SCROLLBAR);
298         break;
299       case kScrollbarHorizontalThumb:
300       case kScrollbarVerticalThumb:
301       case kScrollbarHorizontalGripper:
302       case kScrollbarVerticalGripper:
303         needs_paint_indirect = !GetThemeHandle(SCROLLBAR) ||
304             base::win::GetVersion() == base::win::VERSION_XP;
305         break;
306       default:
307         break;
308     }
309   }
310 
311   if (needs_paint_indirect)
312     PaintIndirect(canvas, part, state, rect, extra);
313   else
314     PaintDirect(canvas, part, state, rect, extra);
315 }
316 
NativeThemeWin()317 NativeThemeWin::NativeThemeWin()
318     : theme_dll_(LoadLibrary(L"uxtheme.dll")),
319       draw_theme_(NULL),
320       draw_theme_ex_(NULL),
321       get_theme_color_(NULL),
322       get_theme_content_rect_(NULL),
323       get_theme_part_size_(NULL),
324       open_theme_(NULL),
325       close_theme_(NULL),
326       set_theme_properties_(NULL),
327       is_theme_active_(NULL),
328       get_theme_int_(NULL),
329       color_change_listener_(this),
330       is_using_high_contrast_(false),
331       is_using_high_contrast_valid_(false) {
332   if (theme_dll_) {
333     draw_theme_ = reinterpret_cast<DrawThemeBackgroundPtr>(
334         GetProcAddress(theme_dll_, "DrawThemeBackground"));
335     draw_theme_ex_ = reinterpret_cast<DrawThemeBackgroundExPtr>(
336         GetProcAddress(theme_dll_, "DrawThemeBackgroundEx"));
337     get_theme_color_ = reinterpret_cast<GetThemeColorPtr>(
338         GetProcAddress(theme_dll_, "GetThemeColor"));
339     get_theme_content_rect_ = reinterpret_cast<GetThemeContentRectPtr>(
340         GetProcAddress(theme_dll_, "GetThemeBackgroundContentRect"));
341     get_theme_part_size_ = reinterpret_cast<GetThemePartSizePtr>(
342         GetProcAddress(theme_dll_, "GetThemePartSize"));
343     open_theme_ = reinterpret_cast<OpenThemeDataPtr>(
344         GetProcAddress(theme_dll_, "OpenThemeData"));
345     close_theme_ = reinterpret_cast<CloseThemeDataPtr>(
346         GetProcAddress(theme_dll_, "CloseThemeData"));
347     set_theme_properties_ = reinterpret_cast<SetThemeAppPropertiesPtr>(
348         GetProcAddress(theme_dll_, "SetThemeAppProperties"));
349     is_theme_active_ = reinterpret_cast<IsThemeActivePtr>(
350         GetProcAddress(theme_dll_, "IsThemeActive"));
351     get_theme_int_ = reinterpret_cast<GetThemeIntPtr>(
352         GetProcAddress(theme_dll_, "GetThemeInt"));
353   }
354   memset(theme_handles_, 0, sizeof(theme_handles_));
355 
356   // Initialize the cached system colors.
357   UpdateSystemColors();
358 }
359 
~NativeThemeWin()360 NativeThemeWin::~NativeThemeWin() {
361   if (theme_dll_) {
362     // todo (cpu): fix this soon.  Making a call to CloseHandles() here breaks
363     // certain tests and the reliability bots.
364     // CloseHandles();
365     FreeLibrary(theme_dll_);
366   }
367 }
368 
OnSysColorChange()369 void NativeThemeWin::OnSysColorChange() {
370   UpdateSystemColors();
371   is_using_high_contrast_valid_ = false;
372   NotifyObservers();
373 }
374 
UpdateSystemColors()375 void NativeThemeWin::UpdateSystemColors() {
376   for (int i = 0; i < arraysize(kSystemColors); ++i) {
377     system_colors_[kSystemColors[i]] =
378         color_utils::GetSysSkColor(kSystemColors[i]);
379   }
380 }
381 
PaintDirect(SkCanvas * canvas,Part part,State state,const gfx::Rect & rect,const ExtraParams & extra) const382 void NativeThemeWin::PaintDirect(SkCanvas* canvas,
383                                  Part part,
384                                  State state,
385                                  const gfx::Rect& rect,
386                                  const ExtraParams& extra) const {
387   skia::ScopedPlatformPaint scoped_platform_paint(canvas);
388   HDC hdc = scoped_platform_paint.GetPlatformSurface();
389 
390   switch (part) {
391     case kCheckbox:
392       PaintCheckbox(hdc, part, state, rect, extra.button);
393       return;
394     case kInnerSpinButton:
395       PaintSpinButton(hdc, part, state, rect, extra.inner_spin);
396       return;
397     case kMenuList:
398       PaintMenuList(hdc, state, rect, extra.menu_list);
399       return;
400     case kMenuCheck:
401       PaintMenuCheck(hdc, state, rect, extra.menu_check);
402       return;
403     case kMenuCheckBackground:
404       PaintMenuCheckBackground(hdc, state, rect);
405       return;
406     case kMenuPopupArrow:
407       PaintMenuArrow(hdc, state, rect, extra.menu_arrow);
408       return;
409     case kMenuPopupBackground:
410       PaintMenuBackground(hdc, rect);
411       return;
412     case kMenuPopupGutter:
413       PaintMenuGutter(hdc, rect);
414       return;
415     case kMenuPopupSeparator:
416       PaintMenuSeparator(hdc, rect, extra.menu_separator);
417       return;
418     case kMenuItemBackground:
419       PaintMenuItemBackground(hdc, state, rect, extra.menu_item);
420       return;
421     case kProgressBar:
422       PaintProgressBar(hdc, rect, extra.progress_bar);
423       return;
424     case kPushButton:
425       PaintPushButton(hdc, part, state, rect, extra.button);
426       return;
427     case kRadio:
428       PaintRadioButton(hdc, part, state, rect, extra.button);
429       return;
430     case kScrollbarDownArrow:
431     case kScrollbarUpArrow:
432     case kScrollbarLeftArrow:
433     case kScrollbarRightArrow:
434       PaintScrollbarArrow(hdc, part, state, rect, extra.scrollbar_arrow);
435       return;
436     case kScrollbarHorizontalThumb:
437     case kScrollbarVerticalThumb:
438     case kScrollbarHorizontalGripper:
439     case kScrollbarVerticalGripper:
440       PaintScrollbarThumb(hdc, part, state, rect, extra.scrollbar_thumb);
441       return;
442     case kScrollbarHorizontalTrack:
443     case kScrollbarVerticalTrack:
444       PaintScrollbarTrack(canvas, hdc, part, state, rect,
445                           extra.scrollbar_track);
446       return;
447     case kScrollbarCorner:
448       canvas->drawColor(SK_ColorWHITE, SkXfermode::kSrc_Mode);
449       return;
450     case kTabPanelBackground:
451       PaintTabPanelBackground(hdc, rect);
452       return;
453     case kTextField:
454       PaintTextField(hdc, part, state, rect, extra.text_field);
455       return;
456     case kTrackbarThumb:
457     case kTrackbarTrack:
458       PaintTrackbar(canvas, hdc, part, state, rect, extra.trackbar);
459       return;
460     case kWindowResizeGripper:
461       PaintWindowResizeGripper(hdc, rect);
462       return;
463     case kComboboxArrow:
464     case kSliderTrack:
465     case kSliderThumb:
466     case kMaxPart:
467       NOTREACHED();
468   }
469 }
470 
GetSystemColor(ColorId color_id) const471 SkColor NativeThemeWin::GetSystemColor(ColorId color_id) const {
472   SkColor color;
473   if (CommonThemeGetSystemColor(color_id, &color))
474     return color;
475 
476   switch (color_id) {
477     // Windows
478     case kColorId_WindowBackground:
479       return system_colors_[COLOR_WINDOW];
480 
481     // Dialogs
482     case kColorId_DialogBackground:
483       return gfx::IsInvertedColorScheme() ?
484           color_utils::InvertColor(kDialogBackgroundColor) :
485           kDialogBackgroundColor;
486 
487     // FocusableBorder
488     case kColorId_FocusedBorderColor:
489       return kFocusedBorderColor;
490     case kColorId_UnfocusedBorderColor:
491       return kUnfocusedBorderColor;
492 
493     // Button
494     case kColorId_ButtonBackgroundColor:
495       return kButtonBackgroundColor;
496     case kColorId_ButtonEnabledColor:
497       return system_colors_[COLOR_BTNTEXT];
498     case kColorId_ButtonDisabledColor:
499       return system_colors_[COLOR_GRAYTEXT];
500     case kColorId_ButtonHighlightColor:
501       return kButtonHighlightColor;
502     case kColorId_ButtonHoverColor:
503       return kButtonHoverColor;
504     case kColorId_ButtonHoverBackgroundColor:
505       return kButtonHoverBackgroundColor;
506     case kColorId_BlueButtonEnabledColor:
507     case kColorId_BlueButtonDisabledColor:
508     case kColorId_BlueButtonPressedColor:
509     case kColorId_BlueButtonHoverColor:
510       NOTREACHED();
511       return kInvalidColorIdColor;
512 
513     // MenuItem
514     case kColorId_EnabledMenuItemForegroundColor:
515       return kEnabledMenuItemForegroundColor;
516     case kColorId_DisabledMenuItemForegroundColor:
517       return kDisabledMenuItemForegroundColor;
518     case kColorId_DisabledEmphasizedMenuItemForegroundColor:
519       return SK_ColorBLACK;
520     case kColorId_FocusedMenuItemBackgroundColor:
521       return kFocusedMenuItemBackgroundColor;
522     case kColorId_MenuSeparatorColor:
523       return kMenuSeparatorColor;
524     case kColorId_SelectedMenuItemForegroundColor:
525     case kColorId_HoverMenuItemBackgroundColor:
526     case kColorId_MenuBackgroundColor:
527     case kColorId_MenuBorderColor:
528       NOTREACHED();
529       return kInvalidColorIdColor;
530 
531     // MenuButton
532     case kColorId_EnabledMenuButtonBorderColor:
533     case kColorId_FocusedMenuButtonBorderColor:
534     case kColorId_HoverMenuButtonBorderColor:
535       NOTREACHED();
536       return kInvalidColorIdColor;
537 
538     // Label
539     case kColorId_LabelEnabledColor:
540       return system_colors_[COLOR_BTNTEXT];
541     case kColorId_LabelDisabledColor:
542       return system_colors_[COLOR_GRAYTEXT];
543     case kColorId_LabelBackgroundColor:
544       return system_colors_[COLOR_WINDOW];
545 
546     // Textfield
547     case kColorId_TextfieldDefaultColor:
548       return system_colors_[COLOR_WINDOWTEXT];
549     case kColorId_TextfieldDefaultBackground:
550       return system_colors_[COLOR_WINDOW];
551     case kColorId_TextfieldReadOnlyColor:
552       return system_colors_[COLOR_GRAYTEXT];
553     case kColorId_TextfieldReadOnlyBackground:
554       return system_colors_[COLOR_3DFACE];
555     case kColorId_TextfieldSelectionColor:
556       return system_colors_[COLOR_HIGHLIGHTTEXT];
557     case kColorId_TextfieldSelectionBackgroundFocused:
558       return system_colors_[COLOR_HIGHLIGHT];
559 
560     // Tooltip
561     case kColorId_TooltipBackground:
562     case kColorId_TooltipText:
563       NOTREACHED();
564       return kInvalidColorIdColor;
565 
566     // Tree
567     // NOTE: these aren't right for all themes, but as close as I could get.
568     case kColorId_TreeBackground:
569       return system_colors_[COLOR_WINDOW];
570     case kColorId_TreeText:
571       return system_colors_[COLOR_WINDOWTEXT];
572     case kColorId_TreeSelectedText:
573       return system_colors_[COLOR_HIGHLIGHTTEXT];
574     case kColorId_TreeSelectedTextUnfocused:
575       return system_colors_[COLOR_BTNTEXT];
576     case kColorId_TreeSelectionBackgroundFocused:
577       return system_colors_[COLOR_HIGHLIGHT];
578     case kColorId_TreeSelectionBackgroundUnfocused:
579       return system_colors_[IsUsingHighContrastTheme() ?
580                               COLOR_MENUHIGHLIGHT : COLOR_BTNFACE];
581     case kColorId_TreeArrow:
582       return system_colors_[COLOR_WINDOWTEXT];
583 
584     // Table
585     case kColorId_TableBackground:
586       return system_colors_[COLOR_WINDOW];
587     case kColorId_TableText:
588       return system_colors_[COLOR_WINDOWTEXT];
589     case kColorId_TableSelectedText:
590       return system_colors_[COLOR_HIGHLIGHTTEXT];
591     case kColorId_TableSelectedTextUnfocused:
592       return system_colors_[COLOR_BTNTEXT];
593     case kColorId_TableSelectionBackgroundFocused:
594       return system_colors_[COLOR_HIGHLIGHT];
595     case kColorId_TableSelectionBackgroundUnfocused:
596       return system_colors_[IsUsingHighContrastTheme() ?
597                               COLOR_MENUHIGHLIGHT : COLOR_BTNFACE];
598     case kColorId_TableGroupingIndicatorColor:
599       return system_colors_[COLOR_GRAYTEXT];
600 
601     // Results Tables
602     case kColorId_ResultsTableNormalBackground:
603       return system_colors_[COLOR_WINDOW];
604     case kColorId_ResultsTableHoveredBackground:
605       return color_utils::AlphaBlend(system_colors_[COLOR_HIGHLIGHT],
606                                      system_colors_[COLOR_WINDOW], 0x40);
607     case kColorId_ResultsTableSelectedBackground:
608       return system_colors_[COLOR_HIGHLIGHT];
609     case kColorId_ResultsTableNormalText:
610     case kColorId_ResultsTableHoveredText:
611       return system_colors_[COLOR_WINDOWTEXT];
612     case kColorId_ResultsTableSelectedText:
613       return system_colors_[COLOR_HIGHLIGHTTEXT];
614     case kColorId_ResultsTableNormalDimmedText:
615       return color_utils::AlphaBlend(system_colors_[COLOR_WINDOWTEXT],
616                                      system_colors_[COLOR_WINDOW], 0x80);
617     case kColorId_ResultsTableHoveredDimmedText:
618       return color_utils::AlphaBlend(
619           system_colors_[COLOR_WINDOWTEXT],
620           GetSystemColor(kColorId_ResultsTableHoveredBackground), 0x80);
621     case kColorId_ResultsTableSelectedDimmedText:
622       return color_utils::AlphaBlend(system_colors_[COLOR_HIGHLIGHTTEXT],
623                                      system_colors_[COLOR_HIGHLIGHT], 0x80);
624     case kColorId_ResultsTableNormalUrl:
625       return color_utils::GetReadableColor(SkColorSetRGB(0, 128, 0),
626                                            system_colors_[COLOR_WINDOW]);
627     case kColorId_ResultsTableHoveredUrl:
628       return color_utils::GetReadableColor(
629           SkColorSetRGB(0, 128, 0),
630           GetSystemColor(kColorId_ResultsTableHoveredBackground));
631     case kColorId_ResultsTableSelectedUrl:
632       return color_utils::GetReadableColor(SkColorSetRGB(0, 128, 0),
633                                            system_colors_[COLOR_HIGHLIGHT]);
634     case kColorId_ResultsTableNormalDivider:
635       return color_utils::AlphaBlend(system_colors_[COLOR_WINDOWTEXT],
636                                      system_colors_[COLOR_WINDOW], 0x34);
637     case kColorId_ResultsTableHoveredDivider:
638       return color_utils::AlphaBlend(
639           system_colors_[COLOR_WINDOWTEXT],
640           GetSystemColor(kColorId_ResultsTableHoveredBackground), 0x34);
641     case kColorId_ResultsTableSelectedDivider:
642       return color_utils::AlphaBlend(system_colors_[COLOR_HIGHLIGHTTEXT],
643                                      system_colors_[COLOR_HIGHLIGHT], 0x34);
644   }
645   NOTREACHED();
646   return kInvalidColorIdColor;
647 }
648 
PaintIndirect(SkCanvas * canvas,Part part,State state,const gfx::Rect & rect,const ExtraParams & extra) const649 void NativeThemeWin::PaintIndirect(SkCanvas* canvas,
650                                    Part part,
651                                    State state,
652                                    const gfx::Rect& rect,
653                                    const ExtraParams& extra) const {
654   // TODO(asvitkine): This path is pretty inefficient - for each paint operation
655   //                  it creates a new offscreen bitmap Skia canvas. This can
656   //                  be sped up by doing it only once per part/state and
657   //                  keeping a cache of the resulting bitmaps.
658 
659   // Create an offscreen canvas that is backed by an HDC.
660   skia::RefPtr<skia::BitmapPlatformDevice> device = skia::AdoptRef(
661       skia::BitmapPlatformDevice::Create(
662           rect.width(), rect.height(), false, NULL));
663   DCHECK(device);
664   SkCanvas offscreen_canvas(device.get());
665   DCHECK(skia::SupportsPlatformPaint(&offscreen_canvas));
666 
667   // Some of the Windows theme drawing operations do not write correct alpha
668   // values for fully-opaque pixels; instead the pixels get alpha 0. This is
669   // especially a problem on Windows XP or when using the Classic theme.
670   //
671   // To work-around this, mark all pixels with a placeholder value, to detect
672   // which pixels get touched by the paint operation. After paint, set any
673   // pixels that have alpha 0 to opaque and placeholders to fully-transparent.
674   const SkColor placeholder = SkColorSetARGB(1, 0, 0, 0);
675   offscreen_canvas.clear(placeholder);
676 
677   // Offset destination rects to have origin (0,0).
678   gfx::Rect adjusted_rect(rect.size());
679   ExtraParams adjusted_extra(extra);
680   switch (part) {
681     case kProgressBar:
682       adjusted_extra.progress_bar.value_rect_x = 0;
683       adjusted_extra.progress_bar.value_rect_y = 0;
684       break;
685     case kScrollbarHorizontalTrack:
686     case kScrollbarVerticalTrack:
687       adjusted_extra.scrollbar_track.track_x = 0;
688       adjusted_extra.scrollbar_track.track_y = 0;
689       break;
690     default:
691       break;
692   }
693   // Draw the theme controls using existing HDC-drawing code.
694   PaintDirect(&offscreen_canvas, part, state, adjusted_rect, adjusted_extra);
695 
696   // Copy the pixels to a bitmap that has ref-counted pixel storage, which is
697   // necessary to have when drawing to a SkPicture.
698   const SkBitmap& hdc_bitmap =
699       offscreen_canvas.getDevice()->accessBitmap(false);
700   SkBitmap bitmap;
701   hdc_bitmap.copyTo(&bitmap, kN32_SkColorType);
702 
703   // Post-process the pixels to fix up the alpha values (see big comment above).
704   const SkPMColor placeholder_value = SkPreMultiplyColor(placeholder);
705   const int pixel_count = rect.width() * rect.height();
706   SkPMColor* pixels = bitmap.getAddr32(0, 0);
707   for (int i = 0; i < pixel_count; i++) {
708     if (pixels[i] == placeholder_value) {
709       // Pixel wasn't touched - make it fully transparent.
710       pixels[i] = SkPackARGB32(0, 0, 0, 0);
711     } else if (SkGetPackedA32(pixels[i]) == 0) {
712       // Pixel was touched but has incorrect alpha of 0, make it fully opaque.
713       pixels[i] = SkPackARGB32(0xFF,
714                                SkGetPackedR32(pixels[i]),
715                                SkGetPackedG32(pixels[i]),
716                                SkGetPackedB32(pixels[i]));
717     }
718   }
719 
720   // Draw the offscreen bitmap to the destination canvas.
721   canvas->drawBitmap(bitmap, rect.x(), rect.y());
722 }
723 
GetThemePartSize(ThemeName theme_name,HDC hdc,int part_id,int state_id,RECT * rect,int ts,SIZE * size) const724 HRESULT NativeThemeWin::GetThemePartSize(ThemeName theme_name,
725                                          HDC hdc,
726                                          int part_id,
727                                          int state_id,
728                                          RECT* rect,
729                                          int ts,
730                                          SIZE* size) const {
731   HANDLE handle = GetThemeHandle(theme_name);
732   return (handle && get_theme_part_size_) ?
733       get_theme_part_size_(handle, hdc, part_id, state_id, rect, ts, size) :
734       E_NOTIMPL;
735 }
736 
PaintButton(HDC hdc,State state,const ButtonExtraParams & extra,int part_id,int state_id,RECT * rect) const737 HRESULT NativeThemeWin::PaintButton(HDC hdc,
738                                     State state,
739                                     const ButtonExtraParams& extra,
740                                     int part_id,
741                                     int state_id,
742                                     RECT* rect) const {
743   HANDLE handle = GetThemeHandle(BUTTON);
744   if (handle && draw_theme_)
745     return draw_theme_(handle, hdc, part_id, state_id, rect, NULL);
746 
747   // Adjust classic_state based on part, state, and extras.
748   int classic_state = extra.classic_state;
749   switch (part_id) {
750     case BP_CHECKBOX:
751       classic_state |= DFCS_BUTTONCHECK;
752       break;
753     case BP_RADIOBUTTON:
754       classic_state |= DFCS_BUTTONRADIO;
755       break;
756     case BP_PUSHBUTTON:
757       classic_state |= DFCS_BUTTONPUSH;
758       break;
759     default:
760       NOTREACHED();
761       break;
762   }
763 
764   switch (state) {
765     case kDisabled:
766       classic_state |= DFCS_INACTIVE;
767       break;
768     case kHovered:
769     case kNormal:
770       break;
771     case kPressed:
772       classic_state |= DFCS_PUSHED;
773       break;
774     case kNumStates:
775       NOTREACHED();
776       break;
777   }
778 
779   if (extra.checked)
780     classic_state |= DFCS_CHECKED;
781 
782   // Draw it manually.
783   // All pressed states have both low bits set, and no other states do.
784   const bool focused = ((state_id & ETS_FOCUSED) == ETS_FOCUSED);
785   const bool pressed = ((state_id & PBS_PRESSED) == PBS_PRESSED);
786   if ((BP_PUSHBUTTON == part_id) && (pressed || focused)) {
787     // BP_PUSHBUTTON has a focus rect drawn around the outer edge, and the
788     // button itself is shrunk by 1 pixel.
789     HBRUSH brush = GetSysColorBrush(COLOR_3DDKSHADOW);
790     if (brush) {
791       FrameRect(hdc, rect, brush);
792       InflateRect(rect, -1, -1);
793     }
794   }
795   DrawFrameControl(hdc, rect, DFC_BUTTON, classic_state);
796 
797   // Draw the focus rectangle (the dotted line box) only on buttons.  For radio
798   // and checkboxes, we let webkit draw the focus rectangle (orange glow).
799   if ((BP_PUSHBUTTON == part_id) && focused) {
800     // The focus rect is inside the button.  The exact number of pixels depends
801     // on whether we're in classic mode or using uxtheme.
802     if (handle && get_theme_content_rect_) {
803       get_theme_content_rect_(handle, hdc, part_id, state_id, rect, rect);
804     } else {
805       InflateRect(rect, -GetSystemMetrics(SM_CXEDGE),
806                   -GetSystemMetrics(SM_CYEDGE));
807     }
808     DrawFocusRect(hdc, rect);
809   }
810 
811   // Classic theme doesn't support indeterminate checkboxes.  We draw
812   // a recangle inside a checkbox like IE10 does.
813   if (part_id == BP_CHECKBOX && extra.indeterminate) {
814     RECT inner_rect = *rect;
815     // "4 / 13" is same as IE10 in classic theme.
816     int padding = (inner_rect.right - inner_rect.left) * 4 / 13;
817     InflateRect(&inner_rect, -padding, -padding);
818     int color_index = state == kDisabled ? COLOR_GRAYTEXT : COLOR_WINDOWTEXT;
819     FillRect(hdc, &inner_rect, GetSysColorBrush(color_index));
820   }
821 
822   return S_OK;
823 }
824 
PaintMenuSeparator(HDC hdc,const gfx::Rect & rect,const MenuSeparatorExtraParams & extra) const825 HRESULT NativeThemeWin::PaintMenuSeparator(
826     HDC hdc,
827     const gfx::Rect& rect,
828     const MenuSeparatorExtraParams& extra) const {
829   RECT rect_win = rect.ToRECT();
830 
831   HANDLE handle = GetThemeHandle(MENU);
832   if (handle && draw_theme_) {
833     // Delta is needed for non-classic to move separator up slightly.
834     --rect_win.top;
835     --rect_win.bottom;
836     return draw_theme_(handle, hdc, MENU_POPUPSEPARATOR, MPI_NORMAL, &rect_win,
837                        NULL);
838   }
839 
840   DrawEdge(hdc, &rect_win, EDGE_ETCHED, BF_TOP);
841   return S_OK;
842 }
843 
PaintMenuGutter(HDC hdc,const gfx::Rect & rect) const844 HRESULT NativeThemeWin::PaintMenuGutter(HDC hdc,
845                                         const gfx::Rect& rect) const {
846   RECT rect_win = rect.ToRECT();
847   HANDLE handle = GetThemeHandle(MENU);
848   return (handle && draw_theme_) ?
849       draw_theme_(handle, hdc, MENU_POPUPGUTTER, MPI_NORMAL, &rect_win, NULL) :
850       E_NOTIMPL;
851 }
852 
PaintMenuArrow(HDC hdc,State state,const gfx::Rect & rect,const MenuArrowExtraParams & extra) const853 HRESULT NativeThemeWin::PaintMenuArrow(
854     HDC hdc,
855     State state,
856     const gfx::Rect& rect,
857     const MenuArrowExtraParams& extra) const {
858   int state_id = MSM_NORMAL;
859   if (state == kDisabled)
860     state_id = MSM_DISABLED;
861 
862   HANDLE handle = GetThemeHandle(MENU);
863   RECT rect_win = rect.ToRECT();
864   if (handle && draw_theme_) {
865     if (extra.pointing_right) {
866       return draw_theme_(handle, hdc, MENU_POPUPSUBMENU, state_id, &rect_win,
867                          NULL);
868     }
869     // There is no way to tell the uxtheme API to draw a left pointing arrow; it
870     // doesn't have a flag equivalent to DFCS_MENUARROWRIGHT.  But they are
871     // needed for RTL locales on Vista.  So use a memory DC and mirror the
872     // region with GDI's StretchBlt.
873     gfx::Rect r(rect);
874     base::win::ScopedCreateDC mem_dc(CreateCompatibleDC(hdc));
875     base::win::ScopedBitmap mem_bitmap(CreateCompatibleBitmap(hdc, r.width(),
876                                                               r.height()));
877     base::win::ScopedSelectObject select_bitmap(mem_dc.Get(), mem_bitmap);
878     // Copy and horizontally mirror the background from hdc into mem_dc. Use
879     // a negative-width source rect, starting at the rightmost pixel.
880     StretchBlt(mem_dc.Get(), 0, 0, r.width(), r.height(),
881                hdc, r.right()-1, r.y(), -r.width(), r.height(), SRCCOPY);
882     // Draw the arrow.
883     RECT theme_rect = {0, 0, r.width(), r.height()};
884     HRESULT result = draw_theme_(handle, mem_dc.Get(), MENU_POPUPSUBMENU,
885                                   state_id, &theme_rect, NULL);
886     // Copy and mirror the result back into mem_dc.
887     StretchBlt(hdc, r.x(), r.y(), r.width(), r.height(),
888                mem_dc.Get(), r.width()-1, 0, -r.width(), r.height(), SRCCOPY);
889     return result;
890   }
891 
892   // For some reason, Windows uses the name DFCS_MENUARROWRIGHT to indicate a
893   // left pointing arrow. This makes the following statement counterintuitive.
894   UINT pfc_state = extra.pointing_right ? DFCS_MENUARROW : DFCS_MENUARROWRIGHT;
895   return PaintFrameControl(hdc, rect, DFC_MENU, pfc_state, extra.is_selected,
896                            state);
897 }
898 
PaintMenuBackground(HDC hdc,const gfx::Rect & rect) const899 HRESULT NativeThemeWin::PaintMenuBackground(HDC hdc,
900                                             const gfx::Rect& rect) const {
901   HANDLE handle = GetThemeHandle(MENU);
902   RECT rect_win = rect.ToRECT();
903   if (handle && draw_theme_) {
904     HRESULT result = draw_theme_(handle, hdc, MENU_POPUPBACKGROUND, 0,
905                                  &rect_win, NULL);
906     FrameRect(hdc, &rect_win, GetSysColorBrush(COLOR_3DSHADOW));
907     return result;
908   }
909 
910   FillRect(hdc, &rect_win, GetSysColorBrush(COLOR_MENU));
911   DrawEdge(hdc, &rect_win, EDGE_RAISED, BF_RECT);
912   return S_OK;
913 }
914 
PaintMenuCheck(HDC hdc,State state,const gfx::Rect & rect,const MenuCheckExtraParams & extra) const915 HRESULT NativeThemeWin::PaintMenuCheck(
916     HDC hdc,
917     State state,
918     const gfx::Rect& rect,
919     const MenuCheckExtraParams& extra) const {
920   HANDLE handle = GetThemeHandle(MENU);
921   if (handle && draw_theme_) {
922     const int state_id = extra.is_radio ?
923         ((state == kDisabled) ? MC_BULLETDISABLED : MC_BULLETNORMAL) :
924         ((state == kDisabled) ? MC_CHECKMARKDISABLED : MC_CHECKMARKNORMAL);
925     RECT rect_win = rect.ToRECT();
926     return draw_theme_(handle, hdc, MENU_POPUPCHECK, state_id, &rect_win, NULL);
927   }
928 
929   return PaintFrameControl(hdc, rect, DFC_MENU,
930                            extra.is_radio ? DFCS_MENUBULLET : DFCS_MENUCHECK,
931                            extra.is_selected, state);
932 }
933 
PaintMenuCheckBackground(HDC hdc,State state,const gfx::Rect & rect) const934 HRESULT NativeThemeWin::PaintMenuCheckBackground(HDC hdc,
935                                                  State state,
936                                                  const gfx::Rect& rect) const {
937   HANDLE handle = GetThemeHandle(MENU);
938   if (!handle || !draw_theme_)
939     return S_OK;  // Nothing to do for background.
940 
941   int state_id = state == kDisabled ? MCB_DISABLED : MCB_NORMAL;
942   RECT rect_win = rect.ToRECT();
943   return draw_theme_(handle, hdc, MENU_POPUPCHECKBACKGROUND, state_id,
944                      &rect_win, NULL);
945 }
946 
PaintMenuItemBackground(HDC hdc,State state,const gfx::Rect & rect,const MenuItemExtraParams & extra) const947 HRESULT NativeThemeWin::PaintMenuItemBackground(
948     HDC hdc,
949     State state,
950     const gfx::Rect& rect,
951     const MenuItemExtraParams& extra) const {
952   HANDLE handle = GetThemeHandle(MENU);
953   RECT rect_win = rect.ToRECT();
954   int state_id = MPI_NORMAL;
955   switch (state) {
956     case kDisabled:
957       state_id = extra.is_selected ? MPI_DISABLEDHOT : MPI_DISABLED;
958       break;
959     case kHovered:
960       state_id = MPI_HOT;
961       break;
962     case kNormal:
963       break;
964     case kPressed:
965     case kNumStates:
966       NOTREACHED();
967       break;
968   }
969 
970   if (handle && draw_theme_)
971     return draw_theme_(handle, hdc, MENU_POPUPITEM, state_id, &rect_win, NULL);
972 
973   if (extra.is_selected)
974     FillRect(hdc, &rect_win, GetSysColorBrush(COLOR_HIGHLIGHT));
975   return S_OK;
976 }
977 
PaintPushButton(HDC hdc,Part part,State state,const gfx::Rect & rect,const ButtonExtraParams & extra) const978 HRESULT NativeThemeWin::PaintPushButton(HDC hdc,
979                                         Part part,
980                                         State state,
981                                         const gfx::Rect& rect,
982                                         const ButtonExtraParams& extra) const {
983   int state_id = extra.is_default ? PBS_DEFAULTED : PBS_NORMAL;
984   switch (state) {
985     case kDisabled:
986       state_id = PBS_DISABLED;
987       break;
988     case kHovered:
989       state_id = PBS_HOT;
990       break;
991     case kNormal:
992       break;
993     case kPressed:
994       state_id = PBS_PRESSED;
995       break;
996     case kNumStates:
997       NOTREACHED();
998       break;
999   }
1000 
1001   RECT rect_win = rect.ToRECT();
1002   return PaintButton(hdc, state, extra, BP_PUSHBUTTON, state_id, &rect_win);
1003 }
1004 
PaintRadioButton(HDC hdc,Part part,State state,const gfx::Rect & rect,const ButtonExtraParams & extra) const1005 HRESULT NativeThemeWin::PaintRadioButton(HDC hdc,
1006                                          Part part,
1007                                          State state,
1008                                          const gfx::Rect& rect,
1009                                          const ButtonExtraParams& extra) const {
1010   int state_id = extra.checked ? RBS_CHECKEDNORMAL : RBS_UNCHECKEDNORMAL;
1011   switch (state) {
1012     case kDisabled:
1013       state_id = extra.checked ? RBS_CHECKEDDISABLED : RBS_UNCHECKEDDISABLED;
1014       break;
1015     case kHovered:
1016       state_id = extra.checked ? RBS_CHECKEDHOT : RBS_UNCHECKEDHOT;
1017       break;
1018     case kNormal:
1019       break;
1020     case kPressed:
1021       state_id = extra.checked ? RBS_CHECKEDPRESSED : RBS_UNCHECKEDPRESSED;
1022       break;
1023     case kNumStates:
1024       NOTREACHED();
1025       break;
1026   }
1027 
1028   RECT rect_win = rect.ToRECT();
1029   return PaintButton(hdc, state, extra, BP_RADIOBUTTON, state_id, &rect_win);
1030 }
1031 
PaintCheckbox(HDC hdc,Part part,State state,const gfx::Rect & rect,const ButtonExtraParams & extra) const1032 HRESULT NativeThemeWin::PaintCheckbox(HDC hdc,
1033                                       Part part,
1034                                       State state,
1035                                       const gfx::Rect& rect,
1036                                       const ButtonExtraParams& extra) const {
1037   int state_id = extra.checked ?
1038       CBS_CHECKEDNORMAL :
1039       (extra.indeterminate ? CBS_MIXEDNORMAL : CBS_UNCHECKEDNORMAL);
1040   switch (state) {
1041     case kDisabled:
1042       state_id = extra.checked ?
1043           CBS_CHECKEDDISABLED :
1044           (extra.indeterminate ? CBS_MIXEDDISABLED : CBS_UNCHECKEDDISABLED);
1045       break;
1046     case kHovered:
1047       state_id = extra.checked ?
1048           CBS_CHECKEDHOT :
1049           (extra.indeterminate ? CBS_MIXEDHOT : CBS_UNCHECKEDHOT);
1050       break;
1051     case kNormal:
1052       break;
1053     case kPressed:
1054       state_id = extra.checked ?
1055           CBS_CHECKEDPRESSED :
1056           (extra.indeterminate ? CBS_MIXEDPRESSED : CBS_UNCHECKEDPRESSED);
1057       break;
1058     case kNumStates:
1059       NOTREACHED();
1060       break;
1061   }
1062 
1063   RECT rect_win = rect.ToRECT();
1064   return PaintButton(hdc, state, extra, BP_CHECKBOX, state_id, &rect_win);
1065 }
1066 
PaintMenuList(HDC hdc,State state,const gfx::Rect & rect,const MenuListExtraParams & extra) const1067 HRESULT NativeThemeWin::PaintMenuList(HDC hdc,
1068                                       State state,
1069                                       const gfx::Rect& rect,
1070                                       const MenuListExtraParams& extra) const {
1071   HANDLE handle = GetThemeHandle(MENULIST);
1072   RECT rect_win = rect.ToRECT();
1073   int state_id = CBXS_NORMAL;
1074   switch (state) {
1075     case kDisabled:
1076       state_id = CBXS_DISABLED;
1077       break;
1078     case kHovered:
1079       state_id = CBXS_HOT;
1080       break;
1081     case kNormal:
1082       break;
1083     case kPressed:
1084       state_id = CBXS_PRESSED;
1085       break;
1086     case kNumStates:
1087       NOTREACHED();
1088       break;
1089   }
1090 
1091   if (handle && draw_theme_)
1092     return draw_theme_(handle, hdc, CP_DROPDOWNBUTTON, state_id, &rect_win,
1093                        NULL);
1094 
1095   // Draw it manually.
1096   DrawFrameControl(hdc, &rect_win, DFC_SCROLL,
1097                    DFCS_SCROLLCOMBOBOX | extra.classic_state);
1098   return S_OK;
1099 }
1100 
PaintScrollbarArrow(HDC hdc,Part part,State state,const gfx::Rect & rect,const ScrollbarArrowExtraParams & extra) const1101 HRESULT NativeThemeWin::PaintScrollbarArrow(
1102     HDC hdc,
1103     Part part,
1104     State state,
1105     const gfx::Rect& rect,
1106     const ScrollbarArrowExtraParams& extra) const {
1107   static const int state_id_matrix[4][kNumStates] = {
1108       ABS_DOWNDISABLED, ABS_DOWNHOT, ABS_DOWNNORMAL, ABS_DOWNPRESSED,
1109       ABS_LEFTDISABLED, ABS_LEFTHOT, ABS_LEFTNORMAL, ABS_LEFTPRESSED,
1110       ABS_RIGHTDISABLED, ABS_RIGHTHOT, ABS_RIGHTNORMAL, ABS_RIGHTPRESSED,
1111       ABS_UPDISABLED, ABS_UPHOT, ABS_UPNORMAL, ABS_UPPRESSED
1112   };
1113   HANDLE handle = GetThemeHandle(SCROLLBAR);
1114   RECT rect_win = rect.ToRECT();
1115   if (handle && draw_theme_) {
1116     int index = part - kScrollbarDownArrow;
1117     DCHECK_GE(index, 0);
1118     DCHECK_LT(static_cast<size_t>(index), arraysize(state_id_matrix));
1119     int state_id = state_id_matrix[index][state];
1120 
1121     // Hovering means that the cursor is over the scroolbar, but not over the
1122     // specific arrow itself.  We don't want to show it "hot" mode, but only
1123     // in "hover" mode.
1124     if (state == kHovered && extra.is_hovering) {
1125       switch (part) {
1126         case kScrollbarDownArrow:
1127           state_id = ABS_DOWNHOVER;
1128           break;
1129         case kScrollbarLeftArrow:
1130           state_id = ABS_LEFTHOVER;
1131           break;
1132         case kScrollbarRightArrow:
1133           state_id = ABS_RIGHTHOVER;
1134           break;
1135         case kScrollbarUpArrow:
1136           state_id = ABS_UPHOVER;
1137           break;
1138         default:
1139           NOTREACHED();
1140           break;
1141       }
1142     }
1143     return PaintScaledTheme(handle, hdc, SBP_ARROWBTN, state_id, rect);
1144   }
1145 
1146   int classic_state = DFCS_SCROLLDOWN;
1147   switch (part) {
1148     case kScrollbarDownArrow:
1149       break;
1150     case kScrollbarLeftArrow:
1151       classic_state = DFCS_SCROLLLEFT;
1152       break;
1153     case kScrollbarRightArrow:
1154       classic_state = DFCS_SCROLLRIGHT;
1155       break;
1156     case kScrollbarUpArrow:
1157       classic_state = DFCS_SCROLLUP;
1158       break;
1159     default:
1160       NOTREACHED();
1161       break;
1162   }
1163   switch (state) {
1164     case kDisabled:
1165       classic_state |= DFCS_INACTIVE;
1166       break;
1167     case kHovered:
1168       classic_state |= DFCS_HOT;
1169       break;
1170     case kNormal:
1171       break;
1172     case kPressed:
1173       classic_state |= DFCS_PUSHED;
1174       break;
1175     case kNumStates:
1176       NOTREACHED();
1177       break;
1178   }
1179   DrawFrameControl(hdc, &rect_win, DFC_SCROLL, classic_state);
1180   return S_OK;
1181 }
1182 
PaintScrollbarThumb(HDC hdc,Part part,State state,const gfx::Rect & rect,const ScrollbarThumbExtraParams & extra) const1183 HRESULT NativeThemeWin::PaintScrollbarThumb(
1184     HDC hdc,
1185     Part part,
1186     State state,
1187     const gfx::Rect& rect,
1188     const ScrollbarThumbExtraParams& extra) const {
1189   HANDLE handle = GetThemeHandle(SCROLLBAR);
1190   RECT rect_win = rect.ToRECT();
1191 
1192   int part_id = SBP_THUMBBTNVERT;
1193   switch (part) {
1194     case kScrollbarHorizontalThumb:
1195       part_id = SBP_THUMBBTNHORZ;
1196       break;
1197     case kScrollbarVerticalThumb:
1198       break;
1199     case kScrollbarHorizontalGripper:
1200       part_id = SBP_GRIPPERHORZ;
1201       break;
1202     case kScrollbarVerticalGripper:
1203       part_id = SBP_GRIPPERVERT;
1204       break;
1205     default:
1206       NOTREACHED();
1207       break;
1208   }
1209 
1210   int state_id = SCRBS_NORMAL;
1211   switch (state) {
1212     case kDisabled:
1213       state_id = SCRBS_DISABLED;
1214       break;
1215     case kHovered:
1216       state_id = extra.is_hovering ? SCRBS_HOVER : SCRBS_HOT;
1217       break;
1218     case kNormal:
1219       break;
1220     case kPressed:
1221       state_id = SCRBS_PRESSED;
1222       break;
1223     case kNumStates:
1224       NOTREACHED();
1225       break;
1226   }
1227 
1228   if (handle && draw_theme_)
1229     return PaintScaledTheme(handle, hdc, part_id, state_id, rect);
1230 
1231   // Draw it manually.
1232   if ((part_id == SBP_THUMBBTNHORZ) || (part_id == SBP_THUMBBTNVERT))
1233     DrawEdge(hdc, &rect_win, EDGE_RAISED, BF_RECT | BF_MIDDLE);
1234   // Classic mode doesn't have a gripper.
1235   return S_OK;
1236 }
1237 
PaintScrollbarTrack(SkCanvas * canvas,HDC hdc,Part part,State state,const gfx::Rect & rect,const ScrollbarTrackExtraParams & extra) const1238 HRESULT NativeThemeWin::PaintScrollbarTrack(
1239     SkCanvas* canvas,
1240     HDC hdc,
1241     Part part,
1242     State state,
1243     const gfx::Rect& rect,
1244     const ScrollbarTrackExtraParams& extra) const {
1245   HANDLE handle = GetThemeHandle(SCROLLBAR);
1246   RECT rect_win = rect.ToRECT();
1247 
1248   const int part_id = extra.is_upper ?
1249       ((part == kScrollbarHorizontalTrack) ?
1250           SBP_UPPERTRACKHORZ : SBP_UPPERTRACKVERT) :
1251       ((part == kScrollbarHorizontalTrack) ?
1252           SBP_LOWERTRACKHORZ : SBP_LOWERTRACKVERT);
1253 
1254   int state_id = SCRBS_NORMAL;
1255   switch (state) {
1256     case kDisabled:
1257       state_id = SCRBS_DISABLED;
1258       break;
1259     case kHovered:
1260       state_id = SCRBS_HOVER;
1261       break;
1262     case kNormal:
1263       break;
1264     case kPressed:
1265       state_id = SCRBS_PRESSED;
1266       break;
1267     case kNumStates:
1268       NOTREACHED();
1269       break;
1270   }
1271 
1272   if (handle && draw_theme_)
1273     return draw_theme_(handle, hdc, part_id, state_id, &rect_win, NULL);
1274 
1275   // Draw it manually.
1276   if ((system_colors_[COLOR_SCROLLBAR] != system_colors_[COLOR_3DFACE]) &&
1277       (system_colors_[COLOR_SCROLLBAR] != system_colors_[COLOR_WINDOW])) {
1278     FillRect(hdc, &rect_win, reinterpret_cast<HBRUSH>(COLOR_SCROLLBAR + 1));
1279   } else {
1280     SkPaint paint;
1281     RECT align_rect = gfx::Rect(extra.track_x, extra.track_y, extra.track_width,
1282                                 extra.track_height).ToRECT();
1283     SetCheckerboardShader(&paint, align_rect);
1284     canvas->drawIRect(skia::RECTToSkIRect(rect_win), paint);
1285   }
1286   if (extra.classic_state & DFCS_PUSHED)
1287     InvertRect(hdc, &rect_win);
1288   return S_OK;
1289 }
1290 
PaintSpinButton(HDC hdc,Part part,State state,const gfx::Rect & rect,const InnerSpinButtonExtraParams & extra) const1291 HRESULT NativeThemeWin::PaintSpinButton(
1292     HDC hdc,
1293     Part part,
1294     State state,
1295     const gfx::Rect& rect,
1296     const InnerSpinButtonExtraParams& extra) const {
1297   HANDLE handle = GetThemeHandle(SPIN);
1298   RECT rect_win = rect.ToRECT();
1299   int part_id = extra.spin_up ? SPNP_UP : SPNP_DOWN;
1300   int state_id = extra.spin_up ? UPS_NORMAL : DNS_NORMAL;
1301   switch (state) {
1302     case kDisabled:
1303       state_id = extra.spin_up ? UPS_DISABLED : DNS_DISABLED;
1304       break;
1305     case kHovered:
1306       state_id = extra.spin_up ? UPS_HOT : DNS_HOT;
1307       break;
1308     case kNormal:
1309       break;
1310     case kPressed:
1311       state_id = extra.spin_up ? UPS_PRESSED : DNS_PRESSED;
1312       break;
1313     case kNumStates:
1314       NOTREACHED();
1315       break;
1316   }
1317 
1318   if (handle && draw_theme_)
1319     return draw_theme_(handle, hdc, part_id, state_id, &rect_win, NULL);
1320   DrawFrameControl(hdc, &rect_win, DFC_SCROLL, extra.classic_state);
1321   return S_OK;
1322 }
1323 
PaintTrackbar(SkCanvas * canvas,HDC hdc,Part part,State state,const gfx::Rect & rect,const TrackbarExtraParams & extra) const1324 HRESULT NativeThemeWin::PaintTrackbar(
1325     SkCanvas* canvas,
1326     HDC hdc,
1327     Part part,
1328     State state,
1329     const gfx::Rect& rect,
1330     const TrackbarExtraParams& extra) const {
1331   const int part_id = extra.vertical ?
1332       ((part == kTrackbarTrack) ? TKP_TRACKVERT : TKP_THUMBVERT) :
1333       ((part == kTrackbarTrack) ? TKP_TRACK : TKP_THUMBBOTTOM);
1334 
1335   int state_id = TUS_NORMAL;
1336   switch (state) {
1337     case kDisabled:
1338       state_id = TUS_DISABLED;
1339       break;
1340     case kHovered:
1341       state_id = TUS_HOT;
1342       break;
1343     case kNormal:
1344       break;
1345     case kPressed:
1346       state_id = TUS_PRESSED;
1347       break;
1348     case kNumStates:
1349       NOTREACHED();
1350       break;
1351   }
1352 
1353   // Make the channel be 4 px thick in the center of the supplied rect.  (4 px
1354   // matches what XP does in various menus; GetThemePartSize() doesn't seem to
1355   // return good values here.)
1356   RECT rect_win = rect.ToRECT();
1357   RECT channel_rect = rect.ToRECT();
1358   const int channel_thickness = 4;
1359   if (part_id == TKP_TRACK) {
1360     channel_rect.top +=
1361         ((channel_rect.bottom - channel_rect.top - channel_thickness) / 2);
1362     channel_rect.bottom = channel_rect.top + channel_thickness;
1363   } else if (part_id == TKP_TRACKVERT) {
1364     channel_rect.left +=
1365         ((channel_rect.right - channel_rect.left - channel_thickness) / 2);
1366     channel_rect.right = channel_rect.left + channel_thickness;
1367   }  // else this isn't actually a channel, so |channel_rect| == |rect|.
1368 
1369   HANDLE handle = GetThemeHandle(TRACKBAR);
1370   if (handle && draw_theme_)
1371     return draw_theme_(handle, hdc, part_id, state_id, &channel_rect, NULL);
1372 
1373   // Classic mode, draw it manually.
1374   if ((part_id == TKP_TRACK) || (part_id == TKP_TRACKVERT)) {
1375     DrawEdge(hdc, &channel_rect, EDGE_SUNKEN, BF_RECT);
1376   } else if (part_id == TKP_THUMBVERT) {
1377     DrawEdge(hdc, &rect_win, EDGE_RAISED, BF_RECT | BF_SOFT | BF_MIDDLE);
1378   } else {
1379     // Split rect into top and bottom pieces.
1380     RECT top_section = rect.ToRECT();
1381     RECT bottom_section = rect.ToRECT();
1382     top_section.bottom -= ((bottom_section.right - bottom_section.left) / 2);
1383     bottom_section.top = top_section.bottom;
1384     DrawEdge(hdc, &top_section, EDGE_RAISED,
1385              BF_LEFT | BF_TOP | BF_RIGHT | BF_SOFT | BF_MIDDLE | BF_ADJUST);
1386 
1387     // Split triangular piece into two diagonals.
1388     RECT& left_half = bottom_section;
1389     RECT right_half = bottom_section;
1390     right_half.left += ((bottom_section.right - bottom_section.left) / 2);
1391     left_half.right = right_half.left;
1392     DrawEdge(hdc, &left_half, EDGE_RAISED,
1393              BF_DIAGONAL_ENDTOPLEFT | BF_SOFT | BF_MIDDLE | BF_ADJUST);
1394     DrawEdge(hdc, &right_half, EDGE_RAISED,
1395              BF_DIAGONAL_ENDBOTTOMLEFT | BF_SOFT | BF_MIDDLE | BF_ADJUST);
1396 
1397     // If the button is pressed, draw hatching.
1398     if (extra.classic_state & DFCS_PUSHED) {
1399       SkPaint paint;
1400       SetCheckerboardShader(&paint, rect_win);
1401 
1402       // Fill all three pieces with the pattern.
1403       canvas->drawIRect(skia::RECTToSkIRect(top_section), paint);
1404 
1405       SkScalar left_triangle_top = SkIntToScalar(left_half.top);
1406       SkScalar left_triangle_right = SkIntToScalar(left_half.right);
1407       SkPath left_triangle;
1408       left_triangle.moveTo(SkIntToScalar(left_half.left), left_triangle_top);
1409       left_triangle.lineTo(left_triangle_right, left_triangle_top);
1410       left_triangle.lineTo(left_triangle_right,
1411                            SkIntToScalar(left_half.bottom));
1412       left_triangle.close();
1413       canvas->drawPath(left_triangle, paint);
1414 
1415       SkScalar right_triangle_left = SkIntToScalar(right_half.left);
1416       SkScalar right_triangle_top = SkIntToScalar(right_half.top);
1417       SkPath right_triangle;
1418       right_triangle.moveTo(right_triangle_left, right_triangle_top);
1419       right_triangle.lineTo(SkIntToScalar(right_half.right),
1420                             right_triangle_top);
1421       right_triangle.lineTo(right_triangle_left,
1422                             SkIntToScalar(right_half.bottom));
1423       right_triangle.close();
1424       canvas->drawPath(right_triangle, paint);
1425     }
1426   }
1427   return S_OK;
1428 }
1429 
PaintProgressBar(HDC hdc,const gfx::Rect & rect,const ProgressBarExtraParams & extra) const1430 HRESULT NativeThemeWin::PaintProgressBar(
1431     HDC hdc,
1432     const gfx::Rect& rect,
1433     const ProgressBarExtraParams& extra) const {
1434   // There is no documentation about the animation speed, frame-rate, nor
1435   // size of moving overlay of the indeterminate progress bar.
1436   // So we just observed real-world programs and guessed following parameters.
1437   const int kDeterminateOverlayPixelsPerSecond = 300;
1438   const int kDeterminateOverlayWidth = 120;
1439   const int kIndeterminateOverlayPixelsPerSecond =  175;
1440   const int kVistaIndeterminateOverlayWidth = 120;
1441   const int kXPIndeterminateOverlayWidth = 55;
1442   // The thickness of the bar frame inside |value_rect|
1443   const int kXPBarPadding = 3;
1444 
1445   RECT bar_rect = rect.ToRECT();
1446   RECT value_rect = gfx::Rect(extra.value_rect_x,
1447                               extra.value_rect_y,
1448                               extra.value_rect_width,
1449                               extra.value_rect_height).ToRECT();
1450 
1451   HANDLE handle = GetThemeHandle(PROGRESS);
1452   if (!handle || !draw_theme_ || !draw_theme_ex_) {
1453     FillRect(hdc, &bar_rect, GetSysColorBrush(COLOR_BTNFACE));
1454     FillRect(hdc, &value_rect, GetSysColorBrush(COLOR_BTNSHADOW));
1455     DrawEdge(hdc, &bar_rect, EDGE_SUNKEN, BF_RECT | BF_ADJUST);
1456     return S_OK;
1457   }
1458 
1459   draw_theme_(handle, hdc, PP_BAR, 0, &bar_rect, NULL);
1460 
1461   bool pre_vista = base::win::GetVersion() < base::win::VERSION_VISTA;
1462   int bar_width = bar_rect.right - bar_rect.left;
1463   if (!extra.determinate) {
1464     // The glossy overlay for the indeterminate progress bar has a small pause
1465     // after each animation. We emulate this by adding an invisible margin the
1466     // animation has to traverse.
1467     int width_with_margin = bar_width + kIndeterminateOverlayPixelsPerSecond;
1468     int overlay_width = pre_vista ?
1469         kXPIndeterminateOverlayWidth : kVistaIndeterminateOverlayWidth;
1470     RECT overlay_rect = bar_rect;
1471     overlay_rect.left += ComputeAnimationProgress(
1472         width_with_margin, overlay_width, kIndeterminateOverlayPixelsPerSecond,
1473         extra.animated_seconds);
1474     overlay_rect.right = overlay_rect.left + overlay_width;
1475     if (pre_vista) {
1476       RECT shrunk_rect = InsetRect(&overlay_rect, kXPBarPadding);
1477       RECT shrunk_bar_rect = InsetRect(&bar_rect, kXPBarPadding);
1478       draw_theme_(handle, hdc, PP_CHUNK, 0, &shrunk_rect, &shrunk_bar_rect);
1479     } else {
1480       draw_theme_(handle, hdc, PP_MOVEOVERLAY, 0, &overlay_rect, &bar_rect);
1481     }
1482     return S_OK;
1483   }
1484 
1485   // We care about the direction here because PP_CHUNK painting is asymmetric.
1486   // TODO(morrita): This RTL guess can be wrong.  We should pass in the
1487   // direction from WebKit.
1488   const DTBGOPTS value_draw_options = {
1489     sizeof(DTBGOPTS),
1490     (bar_rect.right == value_rect.right && bar_rect.left != value_rect.left) ?
1491         DTBG_MIRRORDC : 0,
1492     bar_rect
1493   };
1494   if (pre_vista) {
1495     // On XP, the progress bar is chunk-style and has no glossy effect.  We need
1496     // to shrink the destination rect to fit the part inside the bar with an
1497     // appropriate margin.
1498     RECT shrunk_value_rect = InsetRect(&value_rect, kXPBarPadding);
1499     draw_theme_ex_(handle, hdc, PP_CHUNK, 0, &shrunk_value_rect,
1500                    &value_draw_options);
1501   } else  {
1502     // On Vista or later, the progress bar part has a single-block value part
1503     // and a glossy effect.  The value part has exactly same height as the bar
1504     // part, so we don't need to shrink the rect.
1505     draw_theme_ex_(handle, hdc, PP_FILL, 0, &value_rect, &value_draw_options);
1506 
1507     RECT overlay_rect = value_rect;
1508     overlay_rect.left += ComputeAnimationProgress(
1509         bar_width, kDeterminateOverlayWidth, kDeterminateOverlayPixelsPerSecond,
1510         extra.animated_seconds);
1511     overlay_rect.right = overlay_rect.left + kDeterminateOverlayWidth;
1512     draw_theme_(handle, hdc, PP_MOVEOVERLAY, 0, &overlay_rect, &value_rect);
1513   }
1514   return S_OK;
1515 }
1516 
PaintWindowResizeGripper(HDC hdc,const gfx::Rect & rect) const1517 HRESULT NativeThemeWin::PaintWindowResizeGripper(HDC hdc,
1518                                                  const gfx::Rect& rect) const {
1519   HANDLE handle = GetThemeHandle(STATUS);
1520   RECT rect_win = rect.ToRECT();
1521   if (handle && draw_theme_) {
1522     // Paint the status bar gripper.  There doesn't seem to be a standard
1523     // gripper in Windows for the space between scrollbars.  This is pretty
1524     // close, but it's supposed to be painted over a status bar.
1525     return draw_theme_(handle, hdc, SP_GRIPPER, 0, &rect_win, NULL);
1526   }
1527 
1528   // Draw a windows classic scrollbar gripper.
1529   DrawFrameControl(hdc, &rect_win, DFC_SCROLL, DFCS_SCROLLSIZEGRIP);
1530   return S_OK;
1531 }
1532 
PaintTabPanelBackground(HDC hdc,const gfx::Rect & rect) const1533 HRESULT NativeThemeWin::PaintTabPanelBackground(HDC hdc,
1534                                                 const gfx::Rect& rect) const {
1535   HANDLE handle = GetThemeHandle(TAB);
1536   RECT rect_win = rect.ToRECT();
1537   if (handle && draw_theme_)
1538     return draw_theme_(handle, hdc, TABP_BODY, 0, &rect_win, NULL);
1539 
1540   // Classic just renders a flat color background.
1541   FillRect(hdc, &rect_win, reinterpret_cast<HBRUSH>(COLOR_3DFACE + 1));
1542   return S_OK;
1543 }
1544 
PaintTextField(HDC hdc,Part part,State state,const gfx::Rect & rect,const TextFieldExtraParams & extra) const1545 HRESULT NativeThemeWin::PaintTextField(
1546     HDC hdc,
1547     Part part,
1548     State state,
1549     const gfx::Rect& rect,
1550     const TextFieldExtraParams& extra) const {
1551   int state_id = ETS_NORMAL;
1552   switch (state) {
1553     case kDisabled:
1554       state_id = ETS_DISABLED;
1555       break;
1556     case kHovered:
1557       state_id = ETS_HOT;
1558       break;
1559     case kNormal:
1560       if (extra.is_read_only)
1561         state_id = ETS_READONLY;
1562       else if (extra.is_focused)
1563         state_id = ETS_FOCUSED;
1564       break;
1565     case kPressed:
1566       state_id = ETS_SELECTED;
1567       break;
1568     case kNumStates:
1569       NOTREACHED();
1570       break;
1571   }
1572 
1573   RECT rect_win = rect.ToRECT();
1574   return PaintTextField(hdc, EP_EDITTEXT, state_id, extra.classic_state,
1575                         &rect_win,
1576                         skia::SkColorToCOLORREF(extra.background_color),
1577                         extra.fill_content_area, extra.draw_edges);
1578 }
1579 
PaintTextField(HDC hdc,int part_id,int state_id,int classic_state,RECT * rect,COLORREF color,bool fill_content_area,bool draw_edges) const1580 HRESULT NativeThemeWin::PaintTextField(HDC hdc,
1581                                        int part_id,
1582                                        int state_id,
1583                                        int classic_state,
1584                                        RECT* rect,
1585                                        COLORREF color,
1586                                        bool fill_content_area,
1587                                        bool draw_edges) const {
1588   // TODO(ojan): http://b/1210017 Figure out how to give the ability to
1589   // exclude individual edges from being drawn.
1590 
1591   HANDLE handle = GetThemeHandle(TEXTFIELD);
1592   // TODO(mpcomplete): can we detect if the color is specified by the user,
1593   // and if not, just use the system color?
1594   // CreateSolidBrush() accepts a RGB value but alpha must be 0.
1595   base::win::ScopedGDIObject<HBRUSH> bg_brush(CreateSolidBrush(color));
1596   // DrawThemeBackgroundEx was introduced in XP SP2, so that it's possible
1597   // draw_theme_ex_ is NULL and draw_theme_ is non-null.
1598   if (!handle || (!draw_theme_ex_ && (!draw_theme_ || !draw_edges))) {
1599     // Draw it manually.
1600     if (draw_edges)
1601       DrawEdge(hdc, rect, EDGE_SUNKEN, BF_RECT | BF_ADJUST);
1602 
1603     if (fill_content_area) {
1604       FillRect(hdc, rect, (classic_state & DFCS_INACTIVE) ?
1605                    reinterpret_cast<HBRUSH>(COLOR_BTNFACE + 1) : bg_brush);
1606     }
1607     return S_OK;
1608   }
1609 
1610   static const DTBGOPTS omit_border_options = {
1611     sizeof(DTBGOPTS),
1612     DTBG_OMITBORDER,
1613     { 0, 0, 0, 0 }
1614   };
1615   HRESULT hr = draw_theme_ex_ ?
1616     draw_theme_ex_(handle, hdc, part_id, state_id, rect,
1617                    draw_edges ? NULL : &omit_border_options) :
1618     draw_theme_(handle, hdc, part_id, state_id, rect, NULL);
1619 
1620   // TODO(maruel): Need to be fixed if get_theme_content_rect_ is NULL.
1621   if (fill_content_area && get_theme_content_rect_) {
1622     RECT content_rect;
1623     hr = get_theme_content_rect_(handle, hdc, part_id, state_id, rect,
1624                                   &content_rect);
1625     FillRect(hdc, &content_rect, bg_brush);
1626   }
1627   return hr;
1628 }
1629 
PaintScaledTheme(HANDLE theme,HDC hdc,int part_id,int state_id,const gfx::Rect & rect) const1630 HRESULT NativeThemeWin::PaintScaledTheme(HANDLE theme,
1631                                          HDC hdc,
1632                                          int part_id,
1633                                          int state_id,
1634                                          const gfx::Rect& rect) const {
1635   // Correct the scaling and positioning of sub-components such as scrollbar
1636   // arrows and thumb grippers in the event that the world transform applies
1637   // scaling (e.g. in high-DPI mode).
1638   XFORM save_transform;
1639   if (GetWorldTransform(hdc, &save_transform)) {
1640     float scale = save_transform.eM11;
1641     if (scale != 1 && save_transform.eM12 == 0) {
1642       ModifyWorldTransform(hdc, NULL, MWT_IDENTITY);
1643       gfx::Rect scaled_rect(gfx::ToEnclosedRect(gfx::ScaleRect(rect, scale)));
1644       scaled_rect.Offset(save_transform.eDx, save_transform.eDy);
1645       RECT bounds = scaled_rect.ToRECT();
1646       HRESULT result = draw_theme_(theme, hdc, part_id, state_id, &bounds,
1647                                    NULL);
1648       SetWorldTransform(hdc, &save_transform);
1649       return result;
1650     }
1651   }
1652   RECT bounds = rect.ToRECT();
1653   return draw_theme_(theme, hdc, part_id, state_id, &bounds, NULL);
1654 }
1655 
1656 // static
GetThemeName(Part part)1657 NativeThemeWin::ThemeName NativeThemeWin::GetThemeName(Part part) {
1658   switch (part) {
1659     case kCheckbox:
1660     case kPushButton:
1661     case kRadio:
1662       return BUTTON;
1663     case kInnerSpinButton:
1664       return SPIN;
1665     case kMenuList:
1666     case kMenuCheck:
1667     case kMenuPopupArrow:
1668     case kMenuPopupGutter:
1669     case kMenuPopupSeparator:
1670       return MENU;
1671     case kProgressBar:
1672       return PROGRESS;
1673     case kScrollbarDownArrow:
1674     case kScrollbarLeftArrow:
1675     case kScrollbarRightArrow:
1676     case kScrollbarUpArrow:
1677     case kScrollbarHorizontalThumb:
1678     case kScrollbarVerticalThumb:
1679     case kScrollbarHorizontalTrack:
1680     case kScrollbarVerticalTrack:
1681       return SCROLLBAR;
1682     case kSliderTrack:
1683     case kSliderThumb:
1684       return TRACKBAR;
1685     case kTextField:
1686       return TEXTFIELD;
1687     case kWindowResizeGripper:
1688       return STATUS;
1689     case kComboboxArrow:
1690     case kMenuCheckBackground:
1691     case kMenuPopupBackground:
1692     case kMenuItemBackground:
1693     case kScrollbarHorizontalGripper:
1694     case kScrollbarVerticalGripper:
1695     case kScrollbarCorner:
1696     case kTabPanelBackground:
1697     case kTrackbarThumb:
1698     case kTrackbarTrack:
1699     case kMaxPart:
1700       NOTREACHED();
1701   }
1702   return LAST;
1703 }
1704 
1705 // static
GetWindowsPart(Part part,State state,const ExtraParams & extra)1706 int NativeThemeWin::GetWindowsPart(Part part,
1707                                    State state,
1708                                    const ExtraParams& extra) {
1709   switch (part) {
1710     case kCheckbox:
1711       return BP_CHECKBOX;
1712     case kMenuCheck:
1713       return MENU_POPUPCHECK;
1714     case kMenuPopupArrow:
1715       return MENU_POPUPSUBMENU;
1716     case kMenuPopupGutter:
1717       return MENU_POPUPGUTTER;
1718     case kMenuPopupSeparator:
1719       return MENU_POPUPSEPARATOR;
1720     case kPushButton:
1721       return BP_PUSHBUTTON;
1722     case kRadio:
1723       return BP_RADIOBUTTON;
1724     case kScrollbarDownArrow:
1725     case kScrollbarLeftArrow:
1726     case kScrollbarRightArrow:
1727     case kScrollbarUpArrow:
1728       return SBP_ARROWBTN;
1729     case kScrollbarHorizontalThumb:
1730       return SBP_THUMBBTNHORZ;
1731     case kScrollbarVerticalThumb:
1732       return SBP_THUMBBTNVERT;
1733     case kWindowResizeGripper:
1734       return SP_GRIPPER;
1735     case kComboboxArrow:
1736     case kInnerSpinButton:
1737     case kMenuList:
1738     case kMenuCheckBackground:
1739     case kMenuPopupBackground:
1740     case kMenuItemBackground:
1741     case kProgressBar:
1742     case kScrollbarHorizontalTrack:
1743     case kScrollbarVerticalTrack:
1744     case kScrollbarHorizontalGripper:
1745     case kScrollbarVerticalGripper:
1746     case kScrollbarCorner:
1747     case kSliderTrack:
1748     case kSliderThumb:
1749     case kTabPanelBackground:
1750     case kTextField:
1751     case kTrackbarThumb:
1752     case kTrackbarTrack:
1753     case kMaxPart:
1754       NOTREACHED();
1755   }
1756   return 0;
1757 }
1758 
GetWindowsState(Part part,State state,const ExtraParams & extra)1759 int NativeThemeWin::GetWindowsState(Part part,
1760                                     State state,
1761                                     const ExtraParams& extra) {
1762   switch (part) {
1763     case kCheckbox:
1764       switch (state) {
1765         case kDisabled:
1766           return CBS_UNCHECKEDDISABLED;
1767         case kHovered:
1768           return CBS_UNCHECKEDHOT;
1769         case kNormal:
1770           return CBS_UNCHECKEDNORMAL;
1771         case kPressed:
1772           return CBS_UNCHECKEDPRESSED;
1773         case kNumStates:
1774           NOTREACHED();
1775           return 0;
1776       }
1777     case kMenuCheck:
1778       switch (state) {
1779         case kDisabled:
1780           return extra.menu_check.is_radio ?
1781               MC_BULLETDISABLED : MC_CHECKMARKDISABLED;
1782         case kHovered:
1783         case kNormal:
1784         case kPressed:
1785           return extra.menu_check.is_radio ?
1786               MC_BULLETNORMAL : MC_CHECKMARKNORMAL;
1787         case kNumStates:
1788           NOTREACHED();
1789           return 0;
1790       }
1791     case kMenuPopupArrow:
1792     case kMenuPopupGutter:
1793     case kMenuPopupSeparator:
1794       switch (state) {
1795         case kDisabled:
1796           return MBI_DISABLED;
1797         case kHovered:
1798           return MBI_HOT;
1799         case kNormal:
1800           return MBI_NORMAL;
1801         case kPressed:
1802           return MBI_PUSHED;
1803         case kNumStates:
1804           NOTREACHED();
1805           return 0;
1806       }
1807     case kPushButton:
1808       switch (state) {
1809         case kDisabled:
1810           return PBS_DISABLED;
1811         case kHovered:
1812           return PBS_HOT;
1813         case kNormal:
1814           return PBS_NORMAL;
1815         case kPressed:
1816           return PBS_PRESSED;
1817         case kNumStates:
1818           NOTREACHED();
1819           return 0;
1820       }
1821     case kRadio:
1822       switch (state) {
1823         case kDisabled:
1824           return RBS_UNCHECKEDDISABLED;
1825         case kHovered:
1826           return RBS_UNCHECKEDHOT;
1827         case kNormal:
1828           return RBS_UNCHECKEDNORMAL;
1829         case kPressed:
1830           return RBS_UNCHECKEDPRESSED;
1831         case kNumStates:
1832           NOTREACHED();
1833           return 0;
1834       }
1835     case kScrollbarDownArrow:
1836       switch (state) {
1837         case kDisabled:
1838           return ABS_DOWNDISABLED;
1839         case kHovered:
1840           // Mimic ScrollbarThemeChromiumWin.cpp in WebKit.
1841           return base::win::GetVersion() < base::win::VERSION_VISTA ?
1842               ABS_DOWNHOT : ABS_DOWNHOVER;
1843         case kNormal:
1844           return ABS_DOWNNORMAL;
1845         case kPressed:
1846           return ABS_DOWNPRESSED;
1847         case kNumStates:
1848           NOTREACHED();
1849           return 0;
1850       }
1851     case kScrollbarLeftArrow:
1852       switch (state) {
1853         case kDisabled:
1854           return ABS_LEFTDISABLED;
1855         case kHovered:
1856           // Mimic ScrollbarThemeChromiumWin.cpp in WebKit.
1857           return base::win::GetVersion() < base::win::VERSION_VISTA ?
1858               ABS_LEFTHOT : ABS_LEFTHOVER;
1859         case kNormal:
1860           return ABS_LEFTNORMAL;
1861         case kPressed:
1862           return ABS_LEFTPRESSED;
1863         case kNumStates:
1864           NOTREACHED();
1865           return 0;
1866       }
1867     case kScrollbarRightArrow:
1868       switch (state) {
1869         case kDisabled:
1870           return ABS_RIGHTDISABLED;
1871         case kHovered:
1872           // Mimic ScrollbarThemeChromiumWin.cpp in WebKit.
1873           return base::win::GetVersion() < base::win::VERSION_VISTA ?
1874               ABS_RIGHTHOT : ABS_RIGHTHOVER;
1875         case kNormal:
1876           return ABS_RIGHTNORMAL;
1877         case kPressed:
1878           return ABS_RIGHTPRESSED;
1879         case kNumStates:
1880           NOTREACHED();
1881           return 0;
1882       }
1883       break;
1884     case kScrollbarUpArrow:
1885       switch (state) {
1886         case kDisabled:
1887           return ABS_UPDISABLED;
1888         case kHovered:
1889           // Mimic ScrollbarThemeChromiumWin.cpp in WebKit.
1890           return base::win::GetVersion() < base::win::VERSION_VISTA ?
1891               ABS_UPHOT : ABS_UPHOVER;
1892         case kNormal:
1893           return ABS_UPNORMAL;
1894         case kPressed:
1895           return ABS_UPPRESSED;
1896         case kNumStates:
1897           NOTREACHED();
1898           return 0;
1899       }
1900       break;
1901     case kScrollbarHorizontalThumb:
1902     case kScrollbarVerticalThumb:
1903       switch (state) {
1904         case kDisabled:
1905           return SCRBS_DISABLED;
1906         case kHovered:
1907           // Mimic WebKit's behaviour in ScrollbarThemeChromiumWin.cpp.
1908           return base::win::GetVersion() < base::win::VERSION_VISTA ?
1909               SCRBS_HOT : SCRBS_HOVER;
1910         case kNormal:
1911           return SCRBS_NORMAL;
1912         case kPressed:
1913           return SCRBS_PRESSED;
1914         case kNumStates:
1915           NOTREACHED();
1916           return 0;
1917       }
1918     case kWindowResizeGripper:
1919       switch (state) {
1920         case kDisabled:
1921         case kHovered:
1922         case kNormal:
1923         case kPressed:
1924           return 1;  // gripper has no windows state
1925         case kNumStates:
1926           NOTREACHED();
1927           return 0;
1928       }
1929     case kComboboxArrow:
1930     case kInnerSpinButton:
1931     case kMenuList:
1932     case kMenuCheckBackground:
1933     case kMenuPopupBackground:
1934     case kMenuItemBackground:
1935     case kProgressBar:
1936     case kScrollbarHorizontalTrack:
1937     case kScrollbarVerticalTrack:
1938     case kScrollbarHorizontalGripper:
1939     case kScrollbarVerticalGripper:
1940     case kScrollbarCorner:
1941     case kSliderTrack:
1942     case kSliderThumb:
1943     case kTabPanelBackground:
1944     case kTextField:
1945     case kTrackbarThumb:
1946     case kTrackbarTrack:
1947     case kMaxPart:
1948       NOTREACHED();
1949   }
1950   return 0;
1951 }
1952 
GetThemeInt(ThemeName theme,int part_id,int state_id,int prop_id,int * value) const1953 HRESULT NativeThemeWin::GetThemeInt(ThemeName theme,
1954                                     int part_id,
1955                                     int state_id,
1956                                     int prop_id,
1957                                     int *value) const {
1958   HANDLE handle = GetThemeHandle(theme);
1959   return (handle && get_theme_int_) ?
1960       get_theme_int_(handle, part_id, state_id, prop_id, value) : E_NOTIMPL;
1961 }
1962 
PaintFrameControl(HDC hdc,const gfx::Rect & rect,UINT type,UINT state,bool is_selected,State control_state) const1963 HRESULT NativeThemeWin::PaintFrameControl(HDC hdc,
1964                                           const gfx::Rect& rect,
1965                                           UINT type,
1966                                           UINT state,
1967                                           bool is_selected,
1968                                           State control_state) const {
1969   const int width = rect.width();
1970   const int height = rect.height();
1971 
1972   // DrawFrameControl for menu arrow/check wants a monochrome bitmap.
1973   base::win::ScopedBitmap mask_bitmap(CreateBitmap(width, height, 1, 1, NULL));
1974 
1975   if (mask_bitmap == NULL)
1976     return E_OUTOFMEMORY;
1977 
1978   base::win::ScopedCreateDC bitmap_dc(CreateCompatibleDC(NULL));
1979   base::win::ScopedSelectObject select_bitmap(bitmap_dc.Get(), mask_bitmap);
1980   RECT local_rect = { 0, 0, width, height };
1981   DrawFrameControl(bitmap_dc.Get(), &local_rect, type, state);
1982 
1983   // We're going to use BitBlt with a b&w mask. This results in using the dest
1984   // dc's text color for the black bits in the mask, and the dest dc's
1985   // background color for the white bits in the mask. DrawFrameControl draws the
1986   // check in black, and the background in white.
1987   int bg_color_key = COLOR_MENU;
1988   int text_color_key = COLOR_MENUTEXT;
1989   switch (control_state) {
1990     case kDisabled:
1991       bg_color_key = is_selected ? COLOR_HIGHLIGHT : COLOR_MENU;
1992       text_color_key = COLOR_GRAYTEXT;
1993       break;
1994     case kHovered:
1995       bg_color_key = COLOR_HIGHLIGHT;
1996       text_color_key = COLOR_HIGHLIGHTTEXT;
1997       break;
1998     case kNormal:
1999       break;
2000     case kPressed:
2001     case kNumStates:
2002       NOTREACHED();
2003       break;
2004   }
2005   COLORREF old_bg_color = SetBkColor(hdc, GetSysColor(bg_color_key));
2006   COLORREF old_text_color = SetTextColor(hdc, GetSysColor(text_color_key));
2007   BitBlt(hdc, rect.x(), rect.y(), width, height, bitmap_dc.Get(), 0, 0,
2008          SRCCOPY);
2009   SetBkColor(hdc, old_bg_color);
2010   SetTextColor(hdc, old_text_color);
2011 
2012   return S_OK;
2013 }
2014 
GetThemeHandle(ThemeName theme_name) const2015 HANDLE NativeThemeWin::GetThemeHandle(ThemeName theme_name) const {
2016   if (!open_theme_ || theme_name < 0 || theme_name >= LAST)
2017     return 0;
2018 
2019   if (theme_handles_[theme_name])
2020     return theme_handles_[theme_name];
2021 
2022   // Not found, try to load it.
2023   HANDLE handle = 0;
2024   switch (theme_name) {
2025   case BUTTON:
2026     handle = open_theme_(NULL, L"Button");
2027     break;
2028   case LIST:
2029     handle = open_theme_(NULL, L"Listview");
2030     break;
2031   case MENU:
2032     handle = open_theme_(NULL, L"Menu");
2033     break;
2034   case MENULIST:
2035     handle = open_theme_(NULL, L"Combobox");
2036     break;
2037   case SCROLLBAR:
2038     handle = open_theme_(NULL, L"Scrollbar");
2039     break;
2040   case STATUS:
2041     handle = open_theme_(NULL, L"Status");
2042     break;
2043   case TAB:
2044     handle = open_theme_(NULL, L"Tab");
2045     break;
2046   case TEXTFIELD:
2047     handle = open_theme_(NULL, L"Edit");
2048     break;
2049   case TRACKBAR:
2050     handle = open_theme_(NULL, L"Trackbar");
2051     break;
2052   case WINDOW:
2053     handle = open_theme_(NULL, L"Window");
2054     break;
2055   case PROGRESS:
2056     handle = open_theme_(NULL, L"Progress");
2057     break;
2058   case SPIN:
2059     handle = open_theme_(NULL, L"Spin");
2060     break;
2061   case LAST:
2062     NOTREACHED();
2063     break;
2064   }
2065   theme_handles_[theme_name] = handle;
2066   return handle;
2067 }
2068 
2069 }  // namespace ui
2070