| // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "ui/base/native_theme/native_theme_android.h" |
| |
| #include <limits> |
| |
| #include "base/basictypes.h" |
| #include "base/logging.h" |
| #include "base/memory/scoped_ptr.h" |
| #include "grit/ui_resources.h" |
| #include "third_party/skia/include/effects/SkGradientShader.h" |
| #include "ui/base/layout.h" |
| #include "ui/base/resource/resource_bundle.h" |
| #include "ui/gfx/canvas.h" |
| #include "ui/gfx/color_utils.h" |
| #include "ui/gfx/rect.h" |
| #include "ui/gfx/size.h" |
| |
| namespace ui { |
| |
| const unsigned int kButtonLength = 14; |
| const unsigned int kScrollbarWidth = 15; |
| const unsigned int kThumbInactiveColor = 0xeaeaea; |
| const unsigned int kTrackColor= 0xd3d3d3; |
| |
| // These are the default dimensions of radio buttons and checkboxes. |
| const int kCheckboxAndRadioWidth = 13; |
| const int kCheckboxAndRadioHeight = 13; |
| |
| // These sizes match the sizes in Chromium Win. |
| const int kSliderThumbWidth = 11; |
| const int kSliderThumbHeight = 21; |
| |
| const SkColor kSliderTrackBackgroundColor = SkColorSetRGB(0xe3, 0xdd, 0xd8); |
| const SkColor kSliderThumbLightGrey = SkColorSetRGB(0xf4, 0xf2, 0xef); |
| const SkColor kSliderThumbDarkGrey = SkColorSetRGB(0xea, 0xe5, 0xe0); |
| const SkColor kSliderThumbBorderDarkGrey = SkColorSetRGB(0x9d, 0x96, 0x8e); |
| |
| // Get lightness adjusted color. |
| SkColor BrightenColor(const color_utils::HSL& hsl, |
| SkAlpha alpha, |
| double lightness_amount) { |
| color_utils::HSL adjusted = hsl; |
| adjusted.l += lightness_amount; |
| if (adjusted.l > 1.0) |
| adjusted.l = 1.0; |
| if (adjusted.l < 0.0) |
| adjusted.l = 0.0; |
| |
| return color_utils::HSLToSkColor(adjusted, alpha); |
| } |
| |
| // static |
| const NativeTheme* NativeTheme::instance() { |
| return NativeThemeAndroid::instance(); |
| } |
| |
| // static |
| const NativeThemeAndroid* NativeThemeAndroid::instance() { |
| CR_DEFINE_STATIC_LOCAL(NativeThemeAndroid, s_native_theme, ()); |
| return &s_native_theme; |
| } |
| |
| gfx::Size NativeThemeAndroid::GetPartSize(Part part, |
| State state, |
| const ExtraParams& extra) const { |
| switch (part) { |
| case kScrollbarDownArrow: |
| case kScrollbarUpArrow: |
| return gfx::Size(kScrollbarWidth, kButtonLength); |
| case kScrollbarLeftArrow: |
| case kScrollbarRightArrow: |
| return gfx::Size(kButtonLength, kScrollbarWidth); |
| case kCheckbox: |
| case kRadio: |
| return gfx::Size(kCheckboxAndRadioWidth, kCheckboxAndRadioHeight); |
| case kSliderThumb: |
| // These sizes match the sizes in Chromium Win. |
| return gfx::Size(kSliderThumbWidth, kSliderThumbHeight); |
| case kInnerSpinButton: |
| return gfx::Size(kScrollbarWidth, 0); |
| case kPushButton: |
| case kTextField: |
| case kMenuList: |
| case kSliderTrack: |
| case kProgressBar: |
| return gfx::Size(); // No default size. |
| default: |
| NOTREACHED(); |
| } |
| return gfx::Size(); |
| } |
| |
| void NativeThemeAndroid::Paint(SkCanvas* canvas, |
| Part part, |
| State state, |
| const gfx::Rect& rect, |
| const ExtraParams& extra) const { |
| switch (part) { |
| case kScrollbarDownArrow: |
| case kScrollbarUpArrow: |
| case kScrollbarLeftArrow: |
| case kScrollbarRightArrow: |
| PaintArrowButton(canvas, rect, part, state); |
| break; |
| case kCheckbox: |
| PaintCheckbox(canvas, state, rect, extra.button); |
| break; |
| case kRadio: |
| PaintRadio(canvas, state, rect, extra.button); |
| break; |
| case kPushButton: |
| PaintButton(canvas, state, rect, extra.button); |
| break; |
| case kTextField: |
| PaintTextField(canvas, state, rect, extra.text_field); |
| break; |
| case kMenuList: |
| PaintMenuList(canvas, state, rect, extra.menu_list); |
| break; |
| case kSliderTrack: |
| PaintSliderTrack(canvas, state, rect, extra.slider); |
| break; |
| case kSliderThumb: |
| PaintSliderThumb(canvas, state, rect, extra.slider); |
| break; |
| case kInnerSpinButton: |
| PaintInnerSpinButton(canvas, state, rect, extra.inner_spin); |
| break; |
| case kProgressBar: |
| PaintProgressBar(canvas, state, rect, extra.progress_bar); |
| break; |
| default: |
| NOTREACHED(); |
| } |
| } |
| |
| SkColor NativeThemeAndroid::GetSystemColor(ColorId color_id) const { |
| NOTIMPLEMENTED(); |
| return SK_ColorBLACK; |
| } |
| |
| NativeThemeAndroid::NativeThemeAndroid() { |
| } |
| |
| NativeThemeAndroid::~NativeThemeAndroid() { |
| } |
| |
| void NativeThemeAndroid::PaintArrowButton(SkCanvas* canvas, |
| const gfx::Rect& rect, |
| Part direction, |
| State state) const { |
| int widthMiddle; |
| int lengthMiddle; |
| SkPaint paint; |
| if (direction == kScrollbarUpArrow || direction == kScrollbarDownArrow) { |
| widthMiddle = rect.width() / 2 + 1; |
| lengthMiddle = rect.height() / 2 + 1; |
| } else { |
| lengthMiddle = rect.width() / 2 + 1; |
| widthMiddle = rect.height() / 2 + 1; |
| } |
| |
| // Calculate button color. |
| SkScalar trackHSV[3]; |
| SkColorToHSV(kTrackColor, trackHSV); |
| SkColor buttonColor = SaturateAndBrighten(trackHSV, 0, 0.2); |
| SkColor backgroundColor = buttonColor; |
| if (state == kPressed) { |
| SkScalar buttonHSV[3]; |
| SkColorToHSV(buttonColor, buttonHSV); |
| buttonColor = SaturateAndBrighten(buttonHSV, 0, -0.1); |
| } else if (state == kHovered) { |
| SkScalar buttonHSV[3]; |
| SkColorToHSV(buttonColor, buttonHSV); |
| buttonColor = SaturateAndBrighten(buttonHSV, 0, 0.05); |
| } |
| |
| SkIRect skrect; |
| skrect.set(rect.x(), rect.y(), rect.x() + rect.width(), rect.y() |
| + rect.height()); |
| // Paint the background (the area visible behind the rounded corners). |
| paint.setColor(backgroundColor); |
| canvas->drawIRect(skrect, paint); |
| |
| // Paint the button's outline and fill the middle |
| SkPath outline; |
| switch (direction) { |
| case kScrollbarUpArrow: |
| outline.moveTo(rect.x() + 0.5, rect.y() + rect.height() + 0.5); |
| outline.rLineTo(0, -(rect.height() - 2)); |
| outline.rLineTo(2, -2); |
| outline.rLineTo(rect.width() - 5, 0); |
| outline.rLineTo(2, 2); |
| outline.rLineTo(0, rect.height() - 2); |
| break; |
| case kScrollbarDownArrow: |
| outline.moveTo(rect.x() + 0.5, rect.y() - 0.5); |
| outline.rLineTo(0, rect.height() - 2); |
| outline.rLineTo(2, 2); |
| outline.rLineTo(rect.width() - 5, 0); |
| outline.rLineTo(2, -2); |
| outline.rLineTo(0, -(rect.height() - 2)); |
| break; |
| case kScrollbarRightArrow: |
| outline.moveTo(rect.x() - 0.5, rect.y() + 0.5); |
| outline.rLineTo(rect.width() - 2, 0); |
| outline.rLineTo(2, 2); |
| outline.rLineTo(0, rect.height() - 5); |
| outline.rLineTo(-2, 2); |
| outline.rLineTo(-(rect.width() - 2), 0); |
| break; |
| case kScrollbarLeftArrow: |
| outline.moveTo(rect.x() + rect.width() + 0.5, rect.y() + 0.5); |
| outline.rLineTo(-(rect.width() - 2), 0); |
| outline.rLineTo(-2, 2); |
| outline.rLineTo(0, rect.height() - 5); |
| outline.rLineTo(2, 2); |
| outline.rLineTo(rect.width() - 2, 0); |
| break; |
| default: |
| break; |
| } |
| outline.close(); |
| |
| paint.setStyle(SkPaint::kFill_Style); |
| paint.setColor(buttonColor); |
| canvas->drawPath(outline, paint); |
| |
| paint.setAntiAlias(true); |
| paint.setStyle(SkPaint::kStroke_Style); |
| SkScalar thumbHSV[3]; |
| SkColorToHSV(kThumbInactiveColor, thumbHSV); |
| paint.setColor(OutlineColor(trackHSV, thumbHSV)); |
| canvas->drawPath(outline, paint); |
| |
| // If the button is disabled or read-only, the arrow is drawn with the |
| // outline color. |
| if (state != kDisabled) |
| paint.setColor(SK_ColorBLACK); |
| |
| paint.setAntiAlias(false); |
| paint.setStyle(SkPaint::kFill_Style); |
| |
| SkPath path; |
| // The constants in this block of code are hand-tailored to produce good |
| // looking arrows without anti-aliasing. |
| switch (direction) { |
| case kScrollbarUpArrow: |
| path.moveTo(rect.x() + widthMiddle - 4, rect.y() + lengthMiddle + 2); |
| path.rLineTo(7, 0); |
| path.rLineTo(-4, -4); |
| break; |
| case kScrollbarDownArrow: |
| path.moveTo(rect.x() + widthMiddle - 4, rect.y() + lengthMiddle - 3); |
| path.rLineTo(7, 0); |
| path.rLineTo(-4, 4); |
| break; |
| case kScrollbarRightArrow: |
| path.moveTo(rect.x() + lengthMiddle - 3, rect.y() + widthMiddle - 4); |
| path.rLineTo(0, 7); |
| path.rLineTo(4, -4); |
| break; |
| case kScrollbarLeftArrow: |
| path.moveTo(rect.x() + lengthMiddle + 1, rect.y() + widthMiddle - 5); |
| path.rLineTo(0, 9); |
| path.rLineTo(-4, -4); |
| break; |
| default: |
| break; |
| } |
| path.close(); |
| |
| canvas->drawPath(path, paint); |
| } |
| |
| void NativeThemeAndroid::PaintCheckbox(SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const ButtonExtraParams& button) const { |
| ResourceBundle& rb = ResourceBundle::GetSharedInstance(); |
| gfx::ImageSkia* image = NULL; |
| if (button.indeterminate) { |
| image = state == kDisabled ? |
| rb.GetImageSkiaNamed(IDR_CHECKBOX_DISABLED_INDETERMINATE) : |
| rb.GetImageSkiaNamed(IDR_CHECKBOX_INDETERMINATE); |
| } else if (button.checked) { |
| image = state == kDisabled ? |
| rb.GetImageSkiaNamed(IDR_CHECKBOX_DISABLED_ON) : |
| rb.GetImageSkiaNamed(IDR_CHECKBOX_ON); |
| } else { |
| image = state == kDisabled ? |
| rb.GetImageSkiaNamed(IDR_CHECKBOX_DISABLED_OFF) : |
| rb.GetImageSkiaNamed(IDR_CHECKBOX_OFF); |
| } |
| |
| gfx::Rect bounds = rect.Center(gfx::Size(image->width(), image->height())); |
| DrawImageInt(canvas, *image, 0, 0, image->width(), image->height(), |
| bounds.x(), bounds.y(), bounds.width(), bounds.height()); |
| } |
| |
| void NativeThemeAndroid::PaintRadio(SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const ButtonExtraParams& button) const { |
| ResourceBundle& rb = ResourceBundle::GetSharedInstance(); |
| gfx::ImageSkia* image = NULL; |
| if (state == kDisabled) { |
| image = button.checked ? |
| rb.GetImageSkiaNamed(IDR_RADIO_DISABLED_ON) : |
| rb.GetImageSkiaNamed(IDR_RADIO_DISABLED_OFF); |
| } else { |
| image = button.checked ? |
| rb.GetImageSkiaNamed(IDR_RADIO_ON) : |
| rb.GetImageSkiaNamed(IDR_RADIO_OFF); |
| } |
| |
| gfx::Rect bounds = rect.Center(gfx::Size(image->width(), image->height())); |
| DrawImageInt(canvas, *image, 0, 0, image->width(), image->height(), |
| bounds.x(), bounds.y(), bounds.width(), bounds.height()); |
| } |
| |
| void NativeThemeAndroid::PaintButton(SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const ButtonExtraParams& button) const { |
| SkPaint paint; |
| SkRect skrect; |
| int kRight = rect.right(); |
| int kBottom = rect.bottom(); |
| SkColor base_color = button.background_color; |
| |
| color_utils::HSL base_hsl; |
| color_utils::SkColorToHSL(base_color, &base_hsl); |
| |
| // Our standard gradient is from 0xdd to 0xf8. This is the amount of |
| // increased luminance between those values. |
| SkColor light_color(BrightenColor(base_hsl, SkColorGetA(base_color), 0.105)); |
| |
| // If the button is too small, fallback to drawing a single, solid color |
| if (rect.width() < 5 || rect.height() < 5) { |
| paint.setColor(base_color); |
| skrect.set(rect.x(), rect.y(), kRight, kBottom); |
| canvas->drawRect(skrect, paint); |
| return; |
| } |
| |
| if (button.has_border) { |
| int kBorderAlpha = state == kHovered ? 0x80 : 0x55; |
| paint.setARGB(kBorderAlpha, 0, 0, 0); |
| canvas->drawLine(rect.x() + 1, rect.y(), kRight - 1, rect.y(), paint); |
| canvas->drawLine(kRight - 1, rect.y() + 1, kRight - 1, kBottom - 1, paint); |
| canvas->drawLine(rect.x() + 1, kBottom - 1, kRight - 1, kBottom - 1, paint); |
| canvas->drawLine(rect.x(), rect.y() + 1, rect.x(), kBottom - 1, paint); |
| } |
| |
| paint.setColor(SK_ColorBLACK); |
| int kLightEnd = state == kPressed ? 1 : 0; |
| int kDarkEnd = !kLightEnd; |
| SkPoint gradient_bounds[2]; |
| gradient_bounds[kLightEnd].iset(rect.x(), rect.y()); |
| gradient_bounds[kDarkEnd].iset(rect.x(), kBottom - 1); |
| SkColor colors[2]; |
| colors[0] = light_color; |
| colors[1] = base_color; |
| |
| SkShader* shader = SkGradientShader::CreateLinear( |
| gradient_bounds, colors, NULL, 2, SkShader::kClamp_TileMode, NULL); |
| paint.setStyle(SkPaint::kFill_Style); |
| paint.setShader(shader); |
| shader->unref(); |
| |
| if (button.has_border) { |
| skrect.set(rect.x() + 1, rect.y() + 1, kRight - 1, kBottom - 1); |
| } else { |
| skrect.set(rect.x(), rect.y(), kRight, kBottom); |
| } |
| canvas->drawRect(skrect, paint); |
| paint.setShader(NULL); |
| |
| if (button.has_border) { |
| paint.setColor(BrightenColor(base_hsl, SkColorGetA(base_color), -0.0588)); |
| canvas->drawPoint(rect.x() + 1, rect.y() + 1, paint); |
| canvas->drawPoint(kRight - 2, rect.y() + 1, paint); |
| canvas->drawPoint(rect.x() + 1, kBottom - 2, paint); |
| canvas->drawPoint(kRight - 2, kBottom - 2, paint); |
| } |
| } |
| |
| void NativeThemeAndroid::PaintTextField( |
| SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const TextFieldExtraParams& text) const { |
| // The following drawing code simulates the user-agent css border for |
| // text area and text input so that we do not break layout tests. Once we |
| // have decided the desired looks, we should update the code here and |
| // the layout test expectations. |
| SkRect bounds; |
| bounds.set(rect.x(), rect.y(), rect.right() - 1, rect.bottom() - 1); |
| |
| SkPaint fill_paint; |
| fill_paint.setStyle(SkPaint::kFill_Style); |
| fill_paint.setColor(text.background_color); |
| canvas->drawRect(bounds, fill_paint); |
| |
| if (text.is_text_area) { |
| // Draw text area border: 1px solid black |
| SkPaint stroke_paint; |
| fill_paint.setStyle(SkPaint::kStroke_Style); |
| fill_paint.setColor(SK_ColorBLACK); |
| canvas->drawRect(bounds, fill_paint); |
| } else { |
| // Draw text input and listbox inset border |
| // Text Input: 2px inset #eee |
| // Listbox: 1px inset #808080 |
| SkColor kLightColor = text.is_listbox ? |
| SkColorSetRGB(0x80, 0x80, 0x80) : SkColorSetRGB(0xee, 0xee, 0xee); |
| SkColor kDarkColor = text.is_listbox ? |
| SkColorSetRGB(0x2c, 0x2c, 0x2c) : SkColorSetRGB(0x9a, 0x9a, 0x9a); |
| int kBorderWidth = text.is_listbox ? 1 : 2; |
| |
| SkPaint dark_paint; |
| dark_paint.setAntiAlias(true); |
| dark_paint.setStyle(SkPaint::kFill_Style); |
| dark_paint.setColor(kDarkColor); |
| |
| SkPaint light_paint; |
| light_paint.setAntiAlias(true); |
| light_paint.setStyle(SkPaint::kFill_Style); |
| light_paint.setColor(kLightColor); |
| |
| int left = rect.x(); |
| int top = rect.y(); |
| int right = rect.right(); |
| int bottom = rect.bottom(); |
| |
| SkPath path; |
| path.incReserve(4); |
| |
| // Top |
| path.moveTo(SkIntToScalar(left), SkIntToScalar(top)); |
| path.lineTo(SkIntToScalar(left + kBorderWidth), |
| SkIntToScalar(top + kBorderWidth)); |
| path.lineTo(SkIntToScalar(right - kBorderWidth), |
| SkIntToScalar(top + kBorderWidth)); |
| path.lineTo(SkIntToScalar(right), SkIntToScalar(top)); |
| canvas->drawPath(path, dark_paint); |
| |
| // Bottom |
| path.reset(); |
| path.moveTo(SkIntToScalar(left + kBorderWidth), |
| SkIntToScalar(bottom - kBorderWidth)); |
| path.lineTo(SkIntToScalar(left), SkIntToScalar(bottom)); |
| path.lineTo(SkIntToScalar(right), SkIntToScalar(bottom)); |
| path.lineTo(SkIntToScalar(right - kBorderWidth), |
| SkIntToScalar(bottom - kBorderWidth)); |
| canvas->drawPath(path, light_paint); |
| |
| // Left |
| path.reset(); |
| path.moveTo(SkIntToScalar(left), SkIntToScalar(top)); |
| path.lineTo(SkIntToScalar(left), SkIntToScalar(bottom)); |
| path.lineTo(SkIntToScalar(left + kBorderWidth), |
| SkIntToScalar(bottom - kBorderWidth)); |
| path.lineTo(SkIntToScalar(left + kBorderWidth), |
| SkIntToScalar(top + kBorderWidth)); |
| canvas->drawPath(path, dark_paint); |
| |
| // Right |
| path.reset(); |
| path.moveTo(SkIntToScalar(right - kBorderWidth), |
| SkIntToScalar(top + kBorderWidth)); |
| path.lineTo(SkIntToScalar(right - kBorderWidth), SkIntToScalar(bottom)); |
| path.lineTo(SkIntToScalar(right), SkIntToScalar(bottom)); |
| path.lineTo(SkIntToScalar(right), SkIntToScalar(top)); |
| canvas->drawPath(path, light_paint); |
| } |
| } |
| |
| void NativeThemeAndroid::PaintMenuList( |
| SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const MenuListExtraParams& menu_list) const { |
| // If a border radius is specified, we let the WebCore paint the background |
| // and the border of the control. |
| if (!menu_list.has_border_radius) { |
| ButtonExtraParams button = { 0 }; |
| button.background_color = menu_list.background_color; |
| button.has_border = menu_list.has_border; |
| PaintButton(canvas, state, rect, button); |
| } |
| |
| SkPaint paint; |
| paint.setColor(SK_ColorBLACK); |
| paint.setAntiAlias(true); |
| paint.setStyle(SkPaint::kFill_Style); |
| |
| SkPath path; |
| path.moveTo(menu_list.arrow_x, menu_list.arrow_y - 3); |
| path.rLineTo(6, 0); |
| path.rLineTo(-3, 6); |
| path.close(); |
| canvas->drawPath(path, paint); |
| } |
| |
| void NativeThemeAndroid::PaintSliderTrack( |
| SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const SliderExtraParams& slider) const { |
| int kMidX = rect.x() + rect.width() / 2; |
| int kMidY = rect.y() + rect.height() / 2; |
| |
| SkPaint paint; |
| paint.setColor(kSliderTrackBackgroundColor); |
| |
| SkRect skrect; |
| if (slider.vertical) { |
| skrect.set(std::max(rect.x(), kMidX - 2), |
| rect.y(), |
| std::min(rect.right(), kMidX + 2), |
| rect.bottom()); |
| } else { |
| skrect.set(rect.x(), |
| std::max(rect.y(), kMidY - 2), |
| rect.right(), |
| std::min(rect.bottom(), kMidY + 2)); |
| } |
| canvas->drawRect(skrect, paint); |
| } |
| |
| void NativeThemeAndroid::PaintSliderThumb( |
| SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const SliderExtraParams& slider) const { |
| bool hovered = (state == kHovered) || slider.in_drag; |
| int kMidX = rect.x() + rect.width() / 2; |
| int kMidY = rect.y() + rect.height() / 2; |
| |
| SkPaint paint; |
| paint.setColor(hovered ? SK_ColorWHITE : kSliderThumbLightGrey); |
| |
| SkIRect skrect; |
| if (slider.vertical) |
| skrect.set(rect.x(), rect.y(), kMidX + 1, rect.bottom()); |
| else |
| skrect.set(rect.x(), rect.y(), rect.right(), kMidY + 1); |
| |
| canvas->drawIRect(skrect, paint); |
| |
| paint.setColor(hovered ? kSliderThumbLightGrey : kSliderThumbDarkGrey); |
| |
| if (slider.vertical) |
| skrect.set(kMidX + 1, rect.y(), rect.right(), rect.bottom()); |
| else |
| skrect.set(rect.x(), kMidY + 1, rect.right(), rect.bottom()); |
| |
| canvas->drawIRect(skrect, paint); |
| |
| paint.setColor(kSliderThumbBorderDarkGrey); |
| DrawBox(canvas, rect, paint); |
| |
| if (rect.height() > 10 && rect.width() > 10) { |
| DrawHorizLine(canvas, kMidX - 2, kMidX + 2, kMidY, paint); |
| DrawHorizLine(canvas, kMidX - 2, kMidX + 2, kMidY - 3, paint); |
| DrawHorizLine(canvas, kMidX - 2, kMidX + 2, kMidY + 3, paint); |
| } |
| } |
| |
| void NativeThemeAndroid::PaintInnerSpinButton( |
| SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const InnerSpinButtonExtraParams& spin_button) const { |
| if (spin_button.read_only) |
| state = kDisabled; |
| |
| State north_state = state; |
| State south_state = state; |
| if (spin_button.spin_up) |
| south_state = south_state != kDisabled ? kNormal : kDisabled; |
| else |
| north_state = north_state != kDisabled ? kNormal : kDisabled; |
| |
| gfx::Rect half = rect; |
| half.set_height(rect.height() / 2); |
| PaintArrowButton(canvas, half, kScrollbarUpArrow, north_state); |
| |
| half.set_y(rect.y() + rect.height() / 2); |
| PaintArrowButton(canvas, half, kScrollbarDownArrow, south_state); |
| } |
| |
| void NativeThemeAndroid::PaintProgressBar( |
| SkCanvas* canvas, |
| State state, |
| const gfx::Rect& rect, |
| const ProgressBarExtraParams& progress_bar) const { |
| ResourceBundle& rb = ResourceBundle::GetSharedInstance(); |
| gfx::ImageSkia* bar_image = rb.GetImageSkiaNamed(IDR_PROGRESS_BAR); |
| gfx::ImageSkia* left_border_image = rb.GetImageSkiaNamed( |
| IDR_PROGRESS_BORDER_LEFT); |
| gfx::ImageSkia* right_border_image = rb.GetImageSkiaNamed( |
| IDR_PROGRESS_BORDER_RIGHT); |
| |
| float tile_scale = static_cast<float>(rect.height()) / |
| bar_image->height(); |
| |
| int new_tile_width = static_cast<int>(bar_image->width() * tile_scale); |
| float tile_scale_x = static_cast<float>(new_tile_width) / |
| bar_image->width(); |
| |
| DrawTiledImage(canvas, *bar_image, 0, 0, tile_scale_x, tile_scale, |
| rect.x(), rect.y(), rect.width(), rect.height()); |
| |
| if (progress_bar.value_rect_width) { |
| gfx::ImageSkia* value_image = rb.GetImageSkiaNamed(IDR_PROGRESS_VALUE); |
| |
| new_tile_width = static_cast<int>(value_image->width() * tile_scale); |
| tile_scale_x = static_cast<float>(new_tile_width) / |
| value_image->width(); |
| |
| DrawTiledImage(canvas, *value_image, 0, 0, tile_scale_x, tile_scale, |
| progress_bar.value_rect_x, |
| progress_bar.value_rect_y, |
| progress_bar.value_rect_width, |
| progress_bar.value_rect_height); |
| } |
| |
| int dest_left_border_width = static_cast<int>(left_border_image->width() * |
| tile_scale); |
| DrawImageInt(canvas, *left_border_image, 0, 0, left_border_image->width(), |
| left_border_image->height(), rect.x(), rect.y(), dest_left_border_width, |
| rect.height()); |
| |
| int dest_right_border_width = static_cast<int>(right_border_image->width() * |
| tile_scale); |
| int dest_x = rect.right() - dest_right_border_width; |
| DrawImageInt(canvas, *right_border_image, 0, 0, right_border_image->width(), |
| right_border_image->height(), dest_x, rect.y(), dest_right_border_width, |
| rect.height()); |
| } |
| |
| bool NativeThemeAndroid::IntersectsClipRectInt(SkCanvas* canvas, |
| int x, |
| int y, |
| int w, |
| int h) const { |
| SkRect clip; |
| return canvas->getClipBounds(&clip) && |
| clip.intersect(SkIntToScalar(x), SkIntToScalar(y), SkIntToScalar(x + w), |
| SkIntToScalar(y + h)); |
| } |
| |
| void NativeThemeAndroid::DrawImageInt(SkCanvas* sk_canvas, |
| const gfx::ImageSkia& image, |
| int src_x, |
| int src_y, |
| int src_w, |
| int src_h, |
| int dest_x, |
| int dest_y, |
| int dest_w, |
| int dest_h) const { |
| // TODO(pkotwicz): Do something better and don't infer device |
| // scale factor from canvas scale. |
| SkMatrix m = sk_canvas->getTotalMatrix(); |
| ui::ScaleFactor device_scale_factor = ui::GetScaleFactorFromScale( |
| SkScalarAbs(m.getScaleX())); |
| scoped_ptr<gfx::Canvas> canvas(gfx::Canvas::CreateCanvasWithoutScaling( |
| sk_canvas, device_scale_factor)); |
| canvas->DrawImageInt(image, src_x, src_y, src_w, src_h, |
| dest_x, dest_y, dest_w, dest_h, true); |
| } |
| |
| void NativeThemeAndroid::DrawTiledImage(SkCanvas* sk_canvas, |
| const gfx::ImageSkia& image, |
| int src_x, |
| int src_y, |
| float tile_scale_x, |
| float tile_scale_y, |
| int dest_x, |
| int dest_y, |
| int w, |
| int h) const { |
| // TODO(pkotwicz): Do something better and don't infer device |
| // scale factor from canvas scale. |
| SkMatrix m = sk_canvas->getTotalMatrix(); |
| ui::ScaleFactor device_scale_factor = ui::GetScaleFactorFromScale( |
| SkScalarAbs(m.getScaleX())); |
| scoped_ptr<gfx::Canvas> canvas(gfx::Canvas::CreateCanvasWithoutScaling( |
| sk_canvas, device_scale_factor)); |
| canvas->TileImageInt(image, src_x, src_y, tile_scale_x, |
| tile_scale_y, dest_x, dest_y, w, h); |
| } |
| |
| SkColor NativeThemeAndroid::SaturateAndBrighten( |
| SkScalar* hsv, |
| SkScalar saturate_amount, |
| SkScalar brighten_amount) const { |
| SkScalar color[3]; |
| color[0] = hsv[0]; |
| color[1] = Clamp(hsv[1] + saturate_amount, 0.0, 1.0); |
| color[2] = Clamp(hsv[2] + brighten_amount, 0.0, 1.0); |
| return SkHSVToColor(color); |
| } |
| |
| SkScalar NativeThemeAndroid::Clamp(SkScalar value, |
| SkScalar min, |
| SkScalar max) const { |
| return std::min(std::max(value, min), max); |
| } |
| |
| void NativeThemeAndroid::DrawVertLine(SkCanvas* canvas, |
| int x, |
| int y1, |
| int y2, |
| const SkPaint& paint) const { |
| SkIRect skrect; |
| skrect.set(x, y1, x + 1, y2 + 1); |
| canvas->drawIRect(skrect, paint); |
| } |
| |
| void NativeThemeAndroid::DrawHorizLine(SkCanvas* canvas, |
| int x1, |
| int x2, |
| int y, |
| const SkPaint& paint) const { |
| SkIRect skrect; |
| skrect.set(x1, y, x2 + 1, y + 1); |
| canvas->drawIRect(skrect, paint); |
| } |
| |
| void NativeThemeAndroid::DrawBox(SkCanvas* canvas, |
| const gfx::Rect& rect, |
| const SkPaint& paint) const { |
| int right = rect.x() + rect.width() - 1; |
| int bottom = rect.y() + rect.height() - 1; |
| DrawHorizLine(canvas, rect.x(), right, rect.y(), paint); |
| DrawVertLine(canvas, right, rect.y(), bottom, paint); |
| DrawHorizLine(canvas, rect.x(), right, bottom, paint); |
| DrawVertLine(canvas, rect.x(), rect.y(), bottom, paint); |
| } |
| |
| SkColor NativeThemeAndroid::OutlineColor(SkScalar* hsv1, SkScalar* hsv2) const { |
| SkScalar min_diff = Clamp((hsv1[1] + hsv2[1]) * 1.2, 0.28, 0.5); |
| SkScalar diff = Clamp(fabs(hsv1[2] - hsv2[2]) / 2, min_diff, 0.5); |
| |
| if (hsv1[2] + hsv2[2] > 1.0) |
| diff = -diff; |
| |
| return SaturateAndBrighten(hsv2, -0.2, diff); |
| } |
| |
| } // namespace ui |