diff options
Diffstat (limited to 'ui/gfx/canvas_skia_win.cc')
-rw-r--r-- | ui/gfx/canvas_skia_win.cc | 277 |
1 files changed, 4 insertions, 273 deletions
diff --git a/ui/gfx/canvas_skia_win.cc b/ui/gfx/canvas_skia_win.cc index c498bb6..b343234 100644 --- a/ui/gfx/canvas_skia_win.cc +++ b/ui/gfx/canvas_skia_win.cc @@ -1,4 +1,4 @@ -// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Copyright (c) 2010 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. @@ -7,21 +7,12 @@ #include <limits> #include "base/i18n/rtl.h" -#include "base/logging.h" -#include "base/scoped_ptr.h" -#include "skia/ext/bitmap_platform_device.h" -#include "skia/ext/skia_utils_win.h" #include "third_party/skia/include/core/SkShader.h" #include "ui/gfx/font.h" #include "ui/gfx/rect.h" namespace { -static inline int Round(double x) { - // Why oh why is this not in a standard header? - return static_cast<int>(floor(x + 0.5)); -} - // We make sure that LTR text we draw in an RTL context is modified // appropriately to make sure it maintains it LTR orientation. void DoDrawText(HDC hdc, @@ -133,136 +124,6 @@ int ComputeFormatFlags(int flags, const string16& text) { return f; } -// Changes the alpha of the given bitmap. -// If |fade_to_right| is true then the rect fades from opaque to clear, -// otherwise the rect fades from clear to opaque. -void FadeBitmapRect(skia::BitmapPlatformDevice& bmp_device, - const gfx::Rect& rect, - bool fade_to_right) { - SkBitmap bmp = bmp_device.accessBitmap(true); - DCHECK_EQ(SkBitmap::kARGB_8888_Config, bmp.config()); - SkAutoLockPixels lock(bmp); - float total_width = static_cast<float>(rect.width()); - - for (int x = rect.x(); x < rect.right(); x++) { - float cur_width = static_cast<float>(fade_to_right ? - rect.right() - x : x - rect.x()); - // We want the fade effect to go from 0.2 to 1.0. - float alpha_percent = ((cur_width / total_width) * 0.8f) + 0.2f; - - for (int y = rect.y(); y < rect.bottom(); y++) { - SkColor color = bmp.getColor(x, y); - SkAlpha alpha = static_cast<SkAlpha>(SkColorGetA(color) * alpha_percent); - *bmp.getAddr32(x, y) = SkPreMultiplyColor(SkColorSetA(color, alpha)); - } - } -} - -// DrawText() doesn't support alpha channels. To create a transparent background -// this function draws black on white. It then uses the intensity of black -// to determine how much alpha to use. The text is drawn in |gfx_text_rect| and -// clipped to |gfx_draw_rect|. -void DrawTextAndClearBackground(skia::BitmapPlatformDevice& bmp_device, - HFONT font, - COLORREF text_color, - const string16& text, - int flags, - const gfx::Rect& gfx_text_rect, - const gfx::Rect& gfx_draw_rect) { - // Clear the background by filling with white. - HBRUSH fill_brush = static_cast<HBRUSH>(GetStockObject(WHITE_BRUSH)); - HANDLE old_brush = SelectObject(bmp_device.getBitmapDC(), fill_brush); - RECT draw_rect = gfx_draw_rect.ToRECT(); - FillRect(bmp_device.getBitmapDC(), &draw_rect, fill_brush); - SelectObject(bmp_device.getBitmapDC(), old_brush); - - // Set black text with trasparent background. - SetBkMode(bmp_device.getBitmapDC(), TRANSPARENT); - SetTextColor(bmp_device.getBitmapDC(), 0); - - // Draw the text. - int save_dc_id = SaveDC(bmp_device.getBitmapDC()); - // Clip the text to the draw destination. - IntersectClipRect(bmp_device.getBitmapDC(), draw_rect.left, draw_rect.top, - draw_rect.right, draw_rect.bottom); - SelectObject(bmp_device.getBitmapDC(), font); - RECT text_rect = gfx_text_rect.ToRECT(); - DoDrawText(bmp_device.getBitmapDC(), text, &text_rect, - ComputeFormatFlags(flags, text)); - RestoreDC(bmp_device.getBitmapDC(), save_dc_id); - - BYTE text_color_r = GetRValue(text_color); - BYTE text_color_g = GetGValue(text_color); - BYTE text_color_b = GetBValue(text_color); - - SkBitmap bmp = bmp_device.accessBitmap(true); - DCHECK_EQ(SkBitmap::kARGB_8888_Config, bmp.config()); - SkAutoLockPixels lock(bmp); - - // At this point the bitmap has black text on white. - // The intensity of black tells us the alpha value of the text. - for (int y = draw_rect.top; y < draw_rect.bottom; y++) { - for (int x = draw_rect.left; x < draw_rect.right; x++) { - // Gets the color directly. DrawText doesn't premultiply alpha so - // using SkBitmap::getColor() won't work here. - SkColor color = *bmp.getAddr32(x, y); - // Use any of the color channels as alpha. - BYTE alpha = 0xFF - SkColorGetB(color); - *bmp.getAddr32(x, y) = SkPreMultiplyColor( - SkColorSetARGB(alpha, text_color_r, text_color_b, text_color_g)); - } - } -} - -// Draws the given text with a fade out gradient. |bmp_device| is a bitmap -// that is used to temporary drawing. The text is drawn in |text_rect| and -// clipped to |draw_rect|. -void DrawTextGradientPart(HDC hdc, - skia::BitmapPlatformDevice& bmp_device, - const string16& text, - const SkColor& color, - HFONT font, - const gfx::Rect& text_rect, - const gfx::Rect& draw_rect, - bool fade_to_right, - int flags) { - DrawTextAndClearBackground(bmp_device, font, skia::SkColorToCOLORREF(color), - text, flags, text_rect, draw_rect); - FadeBitmapRect(bmp_device, draw_rect, fade_to_right); - BLENDFUNCTION blend = {AC_SRC_OVER, 0, 255, AC_SRC_ALPHA}; - AlphaBlend(hdc, draw_rect.x(), draw_rect.y(), draw_rect.width(), - draw_rect.height(), bmp_device.getBitmapDC(), draw_rect.x(), - draw_rect.y(), draw_rect.width(), draw_rect.height(), blend); -} - -enum PrimarySide { - PrimaryOnLeft, - PrimaryOnRight, -}; - -// Divides |rect| horizontally into a |primary| of width |primary_width| and a -// |secondary| taking up the remainder. -void DivideRect(const gfx::Rect& rect, - PrimarySide primary_side, - int primary_width, - gfx::Rect* primary, - gfx::Rect* secondary) { - *primary = rect; - *secondary = rect; - int remainder = rect.width() - primary_width; - - switch (primary_side) { - case PrimaryOnLeft: - primary->Inset(0, 0, remainder, 0); - secondary->Inset(primary_width, 0, 0, 0); - break; - case PrimaryOnRight: - primary->Inset(remainder, 0, 0, 0); - secondary->Inset(0, 0, primary_width, 0); - break; - } -} - } // anonymous namespace namespace gfx { @@ -321,13 +182,7 @@ void CanvasSkia::DrawStringInt(const string16& text, const SkColor& color, int x, int y, int w, int h, int flags) { - SkRect fclip; - if (!getClipBounds(&fclip)) - return; - RECT text_bounds = { x, y, x + w, y + h }; - SkIRect clip; - fclip.round(&clip); - if (!clip.intersect(skia::RECTToSkIRect(text_bounds))) + if (!IntersectsClipRectInt(x, y, w, h)) return; // Clamp the max amount of text we'll draw to 32K. There seem to be bugs in @@ -337,6 +192,7 @@ void CanvasSkia::DrawStringInt(const string16& text, const int kMaxStringLength = 32768 - 1; // So the trailing \0 fits in 32K. string16 clamped_string(text.substr(0, kMaxStringLength)); + RECT text_bounds = { x, y, x + w, y + h }; HDC dc = beginPlatformPaint(); SetBkMode(dc, TRANSPARENT); HFONT old_font = (HFONT)SelectObject(dc, font); @@ -355,8 +211,7 @@ void CanvasSkia::DrawStringInt(const string16& text, // Windows will have cleared the alpha channel of the text we drew. Assume // we're drawing to an opaque surface, or at least the text rect area is // opaque. - getTopPlatformDevice().makeOpaque(clip.fLeft, clip.fTop, - clip.width(), clip.height()); + getTopPlatformDevice().makeOpaque(x, y, w, h); } void CanvasSkia::DrawStringInt(const string16& text, @@ -448,128 +303,4 @@ ui::TextureID CanvasSkia::GetTextureID() { return 0; } -void CanvasSkia::DrawFadeTruncatingString( - const string16& text, - CanvasSkia::TruncateFadeMode truncate_mode, - size_t desired_characters_to_truncate_from_head, - const gfx::Font& font, - const SkColor& color, - const gfx::Rect& display_rect) { - int flags = NO_ELLIPSIS; - - // If the whole string fits in the destination then just draw it directly. - int total_string_width; - int total_string_height; - SizeStringInt(text, font, &total_string_width, &total_string_height, - flags | TEXT_VALIGN_TOP); - if (total_string_width <= display_rect.width()) { - DrawStringInt(text, font, color, display_rect.x(), display_rect.y(), - display_rect.width(), display_rect.height(), flags); - return; - } - - int average_character_width = font.GetAverageCharacterWidth(); - int clipped_string_width = total_string_width - display_rect.width(); - - // Clip the string by drawing it to the left by |offset_x|. - int offset_x = 0; - switch (truncate_mode) { - case TruncateFadeHead: - offset_x = clipped_string_width; - break; - case TruncateFadeHeadAndTail: - DCHECK_GT(desired_characters_to_truncate_from_head, 0u); - // Get the width of the beginning of the string we're clipping. - string16 clipped_head_string = - text.substr(0, desired_characters_to_truncate_from_head); - int clipped_width; - int clipped_height; - SizeStringInt(clipped_head_string, font, - &clipped_width, &clipped_height, flags); - - // This is the offset at which we start drawing. This causes the - // beginning of the string to get clipped. - offset_x = clipped_width; - - // Due to the fade effect the first character is hard to see. - // We want to make sure that the first character starting at - // |desired_characters_to_truncate_from_head| is readable so we reduce - // the offset by a little bit. - offset_x = std::max(0, Round(offset_x - average_character_width * 2)); - - // If the offset is so large that there's empty space at the tail - // then reduce the offset so we can use up the empty space. - offset_x = std::min(offset_x, clipped_string_width); - break; - } - bool is_truncating_head = offset_x > 0; - bool is_truncating_tail = clipped_string_width > offset_x; - - bool is_rtl = (ComputeFormatFlags(flags, text) & DT_RTLREADING) != 0; - // |is_rtl| tells us if the given text is right to left or not. |is_rtl| can - // be false even if the UI is set to a right to left language. - // Now, normally, we right align all text if the UI is set to a right to - // left language. In this case though we don't want that because we render - // the ends of the string ourselves. - if (!is_rtl) - flags |= TEXT_ALIGN_LEFT; - - // Fade in/out about 2.5 characters of the beginning/end of the string. - // The .5 here is helpful if one of the characters is a space. - // Use a quarter of the display width if the string is very short. - int gradient_width = Round(std::min(average_character_width * 2.5, - display_rect.width() / 4.0)); - - // Move the origin to |display_rect.origin()|. This simplifies all the - // drawing so that both the source and destination can be (0,0). - save(kMatrix_SaveFlag); - TranslateInt(display_rect.x(), display_rect.y()); - - gfx::Rect solid_part(gfx::Point(), display_rect.size()); - gfx::Rect head_part; - gfx::Rect tail_part; - if (is_truncating_head) - DivideRect(solid_part, is_rtl ? PrimaryOnRight : PrimaryOnLeft, - gradient_width, &head_part, &solid_part); - if (is_truncating_tail) - DivideRect(solid_part, is_rtl ? PrimaryOnLeft : PrimaryOnRight, - gradient_width, &tail_part, &solid_part); - - // Grow |display_rect| by |offset_x|. - gfx::Rect text_rect(gfx::Point(), display_rect.size()); - if (!is_rtl) - text_rect.set_x(text_rect.x() - offset_x); - text_rect.set_width(text_rect.width() + offset_x); - - // Disable clear type. This makes the text render in gray scale which - // makes it easier to compute the alpha value. - LOGFONT font_info; - GetObject(font.GetNativeFont(), sizeof(font_info), &font_info); - font_info.lfQuality = ANTIALIASED_QUALITY; - HFONT gray_scale_font = CreateFontIndirect(&font_info); - - HDC hdc = beginPlatformPaint(); - scoped_ptr<skia::BitmapPlatformDevice> gradient_bitmap( - skia::BitmapPlatformDevice::create(NULL, display_rect.width(), - display_rect.height(), false, NULL)); - if (is_truncating_head) - DrawTextGradientPart(hdc, *gradient_bitmap, text, color, gray_scale_font, - text_rect, head_part, is_rtl, flags); - if (is_truncating_tail) - DrawTextGradientPart(hdc, *gradient_bitmap, text, color, gray_scale_font, - text_rect, tail_part, !is_rtl, flags); - endPlatformPaint(); - - // Draw the solid part. - save(kClip_SaveFlag); - ClipRectInt(solid_part.x(), solid_part.y(), - solid_part.width(), solid_part.height()); - DrawStringInt(text, font, color, - text_rect.x(), text_rect.y(), - text_rect.width(), text_rect.height(), - flags); - restore(); - restore(); -} - } // namespace gfx |