| Index: ui/gfx/native_theme_android.cc
|
| diff --git a/ui/gfx/native_theme_android.cc b/ui/gfx/native_theme_android.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..23d6cbaf4739058dc6ceda63406cb0577a42ef8f
|
| --- /dev/null
|
| +++ b/ui/gfx/native_theme_android.cc
|
| @@ -0,0 +1,793 @@
|
| +// Copyright (c) 2011 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/gfx/native_theme_android.h"
|
| +
|
| +#include <limits>
|
| +
|
| +#include "base/logging.h"
|
| +#include "grit/gfx_resources.h"
|
| +#include "third_party/skia/include/effects/SkGradientShader.h"
|
| +#include "ui/base/resource/resource_bundle.h"
|
| +#include "ui/gfx/color_utils.h"
|
| +#include "ui/gfx/rect.h"
|
| +#include "ui/gfx/size.h"
|
| +
|
| +namespace gfx {
|
| +
|
| +static const unsigned int kButtonLength = 14;
|
| +static const unsigned int kScrollbarWidth = 15;
|
| +static const unsigned int kThumbInactiveColor = 0xeaeaea;
|
| +static const unsigned int kTrackColor= 0xd3d3d3;
|
| +
|
| +// These are the default dimensions of radio buttons and checkboxes.
|
| +static const int kCheckboxAndRadioWidth = 13;
|
| +static const int kCheckboxAndRadioHeight = 13;
|
| +
|
| +// These sizes match the sizes in Chromium Win.
|
| +static const int kSliderThumbWidth = 11;
|
| +static const int kSliderThumbHeight = 21;
|
| +
|
| +static const SkColor kSliderTrackBackgroundColor =
|
| + SkColorSetRGB(0xe3, 0xdd, 0xd8);
|
| +static const SkColor kSliderThumbLightGrey = SkColorSetRGB(0xf4, 0xf2, 0xef);
|
| +static const SkColor kSliderThumbDarkGrey = SkColorSetRGB(0xea, 0xe5, 0xe0);
|
| +static const SkColor kSliderThumbBorderDarkGrey =
|
| + SkColorSetRGB(0x9d, 0x96, 0x8e);
|
| +
|
| +// Get lightness adjusted color.
|
| +static 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
|
| +NativeThemeAndroid* NativeThemeAndroid::instance() {
|
| + // The global NativeThemeAndroid instance.
|
| + static NativeThemeAndroid s_native_theme;
|
| + return &s_native_theme;
|
| +}
|
| +
|
| +gfx::Size NativeThemeAndroid::GetPartSize(Part part) const {
|
| + switch (part) {
|
| + case SCROLLBAR_DOWN_ARROW:
|
| + case SCROLLBAR_UP_ARROW:
|
| + return gfx::Size(kScrollbarWidth, kButtonLength);
|
| + case SCROLLBAR_LEFT_ARROW:
|
| + case SCROLLBAR_RIGHT_ARROW:
|
| + return gfx::Size(kButtonLength, kScrollbarWidth);
|
| + case CHECKBOX:
|
| + case RADIO:
|
| + return gfx::Size(kCheckboxAndRadioWidth, kCheckboxAndRadioHeight);
|
| + case SLIDER_TNUMB:
|
| + // These sizes match the sizes in Chromium Win.
|
| + return gfx::Size(kSliderThumbWidth, kSliderThumbHeight);
|
| + case INNER_SPIN_BUTTON:
|
| + return gfx::Size(kScrollbarWidth, 0);
|
| + case PUSH_BUTTON:
|
| + case TEXTFIELD:
|
| + case MENU_LIST:
|
| + case SLIDER_TRACK:
|
| + case PROGRESS_BAR:
|
| + return gfx::Size(); // No default size.
|
| + }
|
| + return gfx::Size();
|
| +}
|
| +
|
| +void NativeThemeAndroid::Paint(SkCanvas* canvas,
|
| + Part part,
|
| + State state,
|
| + const gfx::Rect& rect,
|
| + const ExtraParams& extra) {
|
| + switch (part) {
|
| + case SCROLLBAR_DOWN_ARROW:
|
| + case SCROLLBAR_UP_ARROW:
|
| + case SCROLLBAR_LEFT_ARROW:
|
| + case SCROLLBAR_RIGHT_ARROW:
|
| + PaintArrowButton(canvas, rect, part, state);
|
| + break;
|
| + case CHECKBOX:
|
| + PaintCheckbox(canvas, state, rect, extra.button);
|
| + break;
|
| + case RADIO:
|
| + PaintRadio(canvas, state, rect, extra.button);
|
| + break;
|
| + case PUSH_BUTTON:
|
| + PaintButton(canvas, state, rect, extra.button);
|
| + break;
|
| + case TEXTFIELD:
|
| + PaintTextField(canvas, state, rect, extra.text_field);
|
| + break;
|
| + case MENU_LIST:
|
| + PaintMenuList(canvas, state, rect, extra.menu_list);
|
| + break;
|
| + case SLIDER_TRACK:
|
| + PaintSliderTrack(canvas, state, rect, extra.slider);
|
| + break;
|
| + case SLIDER_TNUMB:
|
| + PaintSliderThumb(canvas, state, rect, extra.slider);
|
| + break;
|
| + case INNER_SPIN_BUTTON:
|
| + PaintInnerSpinButton(canvas, state, rect, extra.inner_spin);
|
| + break;
|
| + case PROGRESS_BAR:
|
| + PaintProgressBar(canvas, state, rect, extra.progress_bar);
|
| + break;
|
| + default:
|
| + NOTREACHED();
|
| + }
|
| +}
|
| +
|
| +NativeThemeAndroid::NativeThemeAndroid() {
|
| +}
|
| +
|
| +NativeThemeAndroid::~NativeThemeAndroid() {
|
| +}
|
| +
|
| +void NativeThemeAndroid::PaintArrowButton(SkCanvas* canvas,
|
| + const gfx::Rect& rect,
|
| + Part direction,
|
| + State state) {
|
| + int widthMiddle;
|
| + int lengthMiddle;
|
| + SkPaint paint;
|
| + if (direction == SCROLLBAR_UP_ARROW || direction == SCROLLBAR_DOWN_ARROW) {
|
| + 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 == PRESSED) {
|
| + SkScalar buttonHSV[3];
|
| + SkColorToHSV(buttonColor, buttonHSV);
|
| + buttonColor = SaturateAndBrighten(buttonHSV, 0, -0.1);
|
| + } else if (state == HOVERED) {
|
| + 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 SCROLLBAR_UP_ARROW:
|
| + 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 SCROLLBAR_DOWN_ARROW:
|
| + 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 SCROLLBAR_RIGHT_ARROW:
|
| + 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 SCROLLBAR_LEFT_ARROW:
|
| + 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 != DISABLED)
|
| + 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 SCROLLBAR_UP_ARROW:
|
| + path.moveTo(rect.x() + widthMiddle - 4, rect.y() + lengthMiddle + 2);
|
| + path.rLineTo(7, 0);
|
| + path.rLineTo(-4, -4);
|
| + break;
|
| + case SCROLLBAR_DOWN_ARROW:
|
| + path.moveTo(rect.x() + widthMiddle - 4, rect.y() + lengthMiddle - 3);
|
| + path.rLineTo(7, 0);
|
| + path.rLineTo(-4, 4);
|
| + break;
|
| + case SCROLLBAR_RIGHT_ARROW:
|
| + path.moveTo(rect.x() + lengthMiddle - 3, rect.y() + widthMiddle - 4);
|
| + path.rLineTo(0, 7);
|
| + path.rLineTo(4, -4);
|
| + break;
|
| + case SCROLLBAR_LEFT_ARROW:
|
| + 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) {
|
| + ResourceBundle& rb = ResourceBundle::GetSharedInstance();
|
| + SkBitmap* image = NULL;
|
| + if (button.indeterminate) {
|
| + image = state == DISABLED ?
|
| + rb.GetBitmapNamed(IDR_CHECKBOX_DISABLED_INDETERMINATE) :
|
| + rb.GetBitmapNamed(IDR_CHECKBOX_INDETERMINATE);
|
| + } else if (button.checked) {
|
| + image = state == DISABLED ?
|
| + rb.GetBitmapNamed(IDR_CHECKBOX_DISABLED_ON) :
|
| + rb.GetBitmapNamed(IDR_CHECKBOX_ON);
|
| + } else {
|
| + image = state == DISABLED ?
|
| + rb.GetBitmapNamed(IDR_CHECKBOX_DISABLED_OFF) :
|
| + rb.GetBitmapNamed(IDR_CHECKBOX_OFF);
|
| + }
|
| +
|
| + gfx::Rect bounds = rect.Center(gfx::Size(image->width(), image->height()));
|
| + DrawBitmapInt(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) {
|
| + ResourceBundle& rb = ResourceBundle::GetSharedInstance();
|
| + SkBitmap* image = NULL;
|
| + if (state == DISABLED) {
|
| + image = button.checked ?
|
| + rb.GetBitmapNamed(IDR_RADIO_DISABLED_ON) :
|
| + rb.GetBitmapNamed(IDR_RADIO_DISABLED_OFF);
|
| + } else {
|
| + image = button.checked ?
|
| + rb.GetBitmapNamed(IDR_RADIO_ON) :
|
| + rb.GetBitmapNamed(IDR_RADIO_OFF);
|
| + }
|
| +
|
| + gfx::Rect bounds = rect.Center(gfx::Size(image->width(), image->height()));
|
| + DrawBitmapInt(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) {
|
| + 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 == HOVERED ? 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 == PRESSED ? 1 : 0;
|
| + int kDarkEnd = !kLightEnd;
|
| + SkPoint gradient_bounds[2];
|
| + gradient_bounds[kLightEnd].set(SkIntToScalar(rect.x()),
|
| + SkIntToScalar(rect.y()));
|
| + gradient_bounds[kDarkEnd].set(SkIntToScalar(rect.x()),
|
| + SkIntToScalar(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) {
|
| + // 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) {
|
| + // 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) {
|
| + 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) {
|
| + bool hovered = (state == HOVERED) || 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) {
|
| + if (spin_button.read_only)
|
| + state = DISABLED;
|
| +
|
| + State north_state = state;
|
| + State south_state = state;
|
| + if (spin_button.spin_up)
|
| + south_state = south_state != DISABLED ? NORMAL : DISABLED;
|
| + else
|
| + north_state = north_state != DISABLED ? NORMAL : DISABLED;
|
| +
|
| + gfx::Rect half = rect;
|
| + half.set_height(rect.height() / 2);
|
| + PaintArrowButton(canvas, half, SCROLLBAR_UP_ARROW, north_state);
|
| +
|
| + half.set_y(rect.y() + rect.height() / 2);
|
| + PaintArrowButton(canvas, half, SCROLLBAR_DOWN_ARROW, south_state);
|
| +}
|
| +
|
| +void NativeThemeAndroid::PaintProgressBar(
|
| + SkCanvas* canvas,
|
| + State state,
|
| + const gfx::Rect& rect,
|
| + const ProgressBarExtraParams& progress_bar) {
|
| + ResourceBundle& rb = ResourceBundle::GetSharedInstance();
|
| + SkBitmap* bar_image = rb.GetBitmapNamed(IDR_PROGRESS_BAR);
|
| + SkBitmap* left_border_image = rb.GetBitmapNamed(IDR_PROGRESS_BORDER_LEFT);
|
| + SkBitmap* right_border_image = rb.GetBitmapNamed(IDR_PROGRESS_BORDER_RIGHT);
|
| +
|
| + double tile_scale = static_cast<double>(rect.height()) /
|
| + bar_image->height();
|
| +
|
| + int new_tile_width = static_cast<int>(bar_image->width() * tile_scale);
|
| + double tile_scale_x = static_cast<double>(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) {
|
| + SkBitmap* value_image = rb.GetBitmapNamed(IDR_PROGRESS_VALUE);
|
| +
|
| + new_tile_width = static_cast<int>(value_image->width() * tile_scale);
|
| + tile_scale_x = static_cast<double>(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);
|
| + SkRect dest_rect = {
|
| + SkIntToScalar(rect.x()),
|
| + SkIntToScalar(rect.y()),
|
| + SkIntToScalar(rect.x() + dest_left_border_width),
|
| + SkIntToScalar(rect.bottom())
|
| + };
|
| + canvas->drawBitmapRect(*left_border_image, NULL, dest_rect);
|
| +
|
| + int dest_right_border_width = static_cast<int>(right_border_image->width() *
|
| + tile_scale);
|
| + dest_rect.set(SkIntToScalar(rect.right() - dest_right_border_width),
|
| + SkIntToScalar(rect.y()),
|
| + SkIntToScalar(rect.right()),
|
| + SkIntToScalar(rect.bottom()));
|
| + canvas->drawBitmapRect(*right_border_image, NULL, dest_rect);
|
| +}
|
| +
|
| +bool NativeThemeAndroid::IntersectsClipRectInt(SkCanvas* canvas,
|
| + int x,
|
| + int y,
|
| + int w,
|
| + int h) {
|
| + SkRect clip;
|
| + return canvas->getClipBounds(&clip) &&
|
| + clip.intersect(SkIntToScalar(x), SkIntToScalar(y), SkIntToScalar(x + w),
|
| + SkIntToScalar(y + h));
|
| +}
|
| +
|
| +void NativeThemeAndroid::DrawBitmapInt(SkCanvas* canvas,
|
| + const SkBitmap& bitmap,
|
| + int src_x,
|
| + int src_y,
|
| + int src_w,
|
| + int src_h,
|
| + int dest_x,
|
| + int dest_y,
|
| + int dest_w,
|
| + int dest_h) {
|
| + DLOG_ASSERT(src_x + src_w < std::numeric_limits<int16_t>::max() &&
|
| + src_y + src_h < std::numeric_limits<int16_t>::max());
|
| + if (src_w <= 0 || src_h <= 0 || dest_w <= 0 || dest_h <= 0) {
|
| + NOTREACHED() << "Attempting to draw bitmap to/from an empty rect!";
|
| + return;
|
| + }
|
| +
|
| + if (!IntersectsClipRectInt(canvas, dest_x, dest_y, dest_w, dest_h))
|
| + return;
|
| +
|
| + SkRect dest_rect = { SkIntToScalar(dest_x),
|
| + SkIntToScalar(dest_y),
|
| + SkIntToScalar(dest_x + dest_w),
|
| + SkIntToScalar(dest_y + dest_h) };
|
| +
|
| + if (src_w == dest_w && src_h == dest_h) {
|
| + // Workaround for apparent bug in Skia that causes image to occasionally
|
| + // shift.
|
| + SkIRect src_rect = { src_x, src_y, src_x + src_w, src_y + src_h };
|
| + canvas->drawBitmapRect(bitmap, &src_rect, dest_rect);
|
| + return;
|
| + }
|
| +
|
| + // Make a bitmap shader that contains the bitmap we want to draw. This is
|
| + // basically what SkCanvas.drawBitmap does internally, but it gives us
|
| + // more control over quality and will use the mipmap in the source image if
|
| + // it has one, whereas drawBitmap won't.
|
| + SkShader* shader = SkShader::CreateBitmapShader(bitmap,
|
| + SkShader::kRepeat_TileMode,
|
| + SkShader::kRepeat_TileMode);
|
| + SkMatrix shader_scale;
|
| + shader_scale.setScale(SkFloatToScalar(static_cast<float>(dest_w) / src_w),
|
| + SkFloatToScalar(static_cast<float>(dest_h) / src_h));
|
| + shader_scale.preTranslate(SkIntToScalar(-src_x), SkIntToScalar(-src_y));
|
| + shader_scale.postTranslate(SkIntToScalar(dest_x), SkIntToScalar(dest_y));
|
| + shader->setLocalMatrix(shader_scale);
|
| +
|
| + // The rect will be filled by the bitmap.
|
| + SkPaint p;
|
| + p.setFilterBitmap(true);
|
| + p.setShader(shader);
|
| + shader->unref();
|
| + canvas->drawRect(dest_rect, p);
|
| +}
|
| +
|
| +void NativeThemeAndroid::DrawTiledImage(SkCanvas* canvas,
|
| + const SkBitmap& bitmap,
|
| + int src_x,
|
| + int src_y,
|
| + double tile_scale_x,
|
| + double tile_scale_y,
|
| + int dest_x,
|
| + int dest_y,
|
| + int w,
|
| + int h) const {
|
| + SkShader* shader = SkShader::CreateBitmapShader(bitmap,
|
| + SkShader::kRepeat_TileMode,
|
| + SkShader::kRepeat_TileMode);
|
| + if (tile_scale_x != 1.0 || tile_scale_y != 1.0) {
|
| + SkMatrix shader_scale;
|
| + shader_scale.setScale(SkDoubleToScalar(tile_scale_x),
|
| + SkDoubleToScalar(tile_scale_y));
|
| + shader->setLocalMatrix(shader_scale);
|
| + }
|
| +
|
| + SkPaint paint;
|
| + paint.setShader(shader);
|
| + paint.setXfermodeMode(SkXfermode::kSrcOver_Mode);
|
| +
|
| + // CreateBitmapShader returns a Shader with a reference count of one, we
|
| + // need to unref after paint takes ownership of the shader.
|
| + shader->unref();
|
| + canvas->save();
|
| + canvas->translate(SkIntToScalar(dest_x - src_x),
|
| + SkIntToScalar(dest_y - src_y));
|
| + canvas->clipRect(SkRect::MakeXYWH(src_x, src_y, w, h));
|
| + canvas->drawPaint(paint);
|
| + canvas->restore();
|
| +}
|
| +
|
| +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 gfx
|
|
|