// Copyright 2014 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 "ash/frame/default_header_painter.h" #include "ash/frame/caption_buttons/frame_caption_button_container_view.h" #include "ash/frame/header_painter_util.h" #include "base/debug/leak_annotations.h" #include "base/logging.h" // DCHECK #include "grit/ash_resources.h" #include "third_party/skia/include/core/SkColor.h" #include "third_party/skia/include/core/SkPaint.h" #include "third_party/skia/include/core/SkPath.h" #include "ui/base/resource/resource_bundle.h" #include "ui/gfx/animation/slide_animation.h" #include "ui/gfx/canvas.h" #include "ui/gfx/font_list.h" #include "ui/gfx/image/image.h" #include "ui/gfx/rect.h" #include "ui/gfx/skia_util.h" #include "ui/views/view.h" #include "ui/views/widget/native_widget_aura.h" #include "ui/views/widget/widget.h" #include "ui/views/widget/widget_delegate.h" using views::Widget; namespace { // Color for the window title text. const SkColor kTitleTextColor = SkColorSetRGB(40, 40, 40); // Color of the active window header/content separator line. const SkColor kHeaderContentSeparatorColor = SkColorSetRGB(150, 150, 152); // Color of the inactive window header/content separator line. const SkColor kHeaderContentSeparatorInactiveColor = SkColorSetRGB(180, 180, 182); // Duration of crossfade animation for activating and deactivating frame. const int kActivationCrossfadeDurationMs = 200; // Tiles an image into an area, rounding the top corners. void TileRoundRect(gfx::Canvas* canvas, const gfx::ImageSkia& image, const SkPaint& paint, const gfx::Rect& bounds, int corner_radius) { SkRect rect = gfx::RectToSkRect(bounds); const SkScalar corner_radius_scalar = SkIntToScalar(corner_radius); SkScalar radii[8] = { corner_radius_scalar, corner_radius_scalar, // top-left corner_radius_scalar, corner_radius_scalar, // top-right 0, 0, // bottom-right 0, 0}; // bottom-left SkPath path; path.addRoundRect(rect, radii, SkPath::kCW_Direction); canvas->DrawImageInPath(image, 0, 0, path, paint); } // Returns the FontList to use for the title. const gfx::FontList& GetTitleFontList() { static const gfx::FontList* title_font_list = new gfx::FontList(views::NativeWidgetAura::GetWindowTitleFontList()); ANNOTATE_LEAKING_OBJECT_PTR(title_font_list); return *title_font_list; } } // namespace namespace ash { /////////////////////////////////////////////////////////////////////////////// // DefaultHeaderPainter, public: DefaultHeaderPainter::DefaultHeaderPainter() : frame_(NULL), view_(NULL), window_icon_(NULL), window_icon_size_(HeaderPainterUtil::GetDefaultIconSize()), caption_button_container_(NULL), height_(0), mode_(MODE_INACTIVE), initial_paint_(true), activation_animation_(new gfx::SlideAnimation(this)) { } DefaultHeaderPainter::~DefaultHeaderPainter() { } void DefaultHeaderPainter::Init( views::Widget* frame, views::View* header_view, views::View* window_icon, FrameCaptionButtonContainerView* caption_button_container) { DCHECK(frame); DCHECK(header_view); // window_icon may be NULL. DCHECK(caption_button_container); frame_ = frame; view_ = header_view; window_icon_ = window_icon; caption_button_container_ = caption_button_container; caption_button_container_->SetButtonImages( CAPTION_BUTTON_ICON_MINIMIZE, IDR_AURA_WINDOW_CONTROL_ICON_MINIMIZE, IDR_AURA_WINDOW_CONTROL_ICON_MINIMIZE_I, IDR_AURA_WINDOW_CONTROL_BACKGROUND_H, IDR_AURA_WINDOW_CONTROL_BACKGROUND_P); caption_button_container_->SetButtonImages( CAPTION_BUTTON_ICON_MAXIMIZE_RESTORE, IDR_AURA_WINDOW_CONTROL_ICON_SIZE, IDR_AURA_WINDOW_CONTROL_ICON_SIZE_I, IDR_AURA_WINDOW_CONTROL_BACKGROUND_H, IDR_AURA_WINDOW_CONTROL_BACKGROUND_P); caption_button_container_->SetButtonImages( CAPTION_BUTTON_ICON_CLOSE, IDR_AURA_WINDOW_CONTROL_ICON_CLOSE, IDR_AURA_WINDOW_CONTROL_ICON_CLOSE_I, IDR_AURA_WINDOW_CONTROL_BACKGROUND_H, IDR_AURA_WINDOW_CONTROL_BACKGROUND_P); // There is no dedicated icon for the snap-left and snap-right buttons // when |frame_| is inactive because they should never be visible while // |frame_| is inactive. caption_button_container_->SetButtonImages( CAPTION_BUTTON_ICON_LEFT_SNAPPED, IDR_AURA_WINDOW_CONTROL_ICON_LEFT_SNAPPED, IDR_AURA_WINDOW_CONTROL_ICON_LEFT_SNAPPED, IDR_AURA_WINDOW_CONTROL_BACKGROUND_H, IDR_AURA_WINDOW_CONTROL_BACKGROUND_P); caption_button_container_->SetButtonImages( CAPTION_BUTTON_ICON_RIGHT_SNAPPED, IDR_AURA_WINDOW_CONTROL_ICON_RIGHT_SNAPPED, IDR_AURA_WINDOW_CONTROL_ICON_RIGHT_SNAPPED, IDR_AURA_WINDOW_CONTROL_BACKGROUND_H, IDR_AURA_WINDOW_CONTROL_BACKGROUND_P); } int DefaultHeaderPainter::GetMinimumHeaderWidth() const { // Ensure we have enough space for the window icon and buttons. We allow // the title string to collapse to zero width. return GetTitleBounds().x() + caption_button_container_->GetMinimumSize().width(); } void DefaultHeaderPainter::PaintHeader(gfx::Canvas* canvas, Mode mode) { Mode old_mode = mode_; mode_ = mode; if (mode_ != old_mode) { if (!initial_paint_ && HeaderPainterUtil::CanAnimateActivation(frame_)) { activation_animation_->SetSlideDuration(kActivationCrossfadeDurationMs); if (mode_ == MODE_ACTIVE) activation_animation_->Show(); else activation_animation_->Hide(); } else { if (mode_ == MODE_ACTIVE) activation_animation_->Reset(1); else activation_animation_->Reset(0); } initial_paint_ = false; } int corner_radius = (frame_->IsMaximized() || frame_->IsFullscreen()) ? 0 : HeaderPainterUtil::GetTopCornerRadiusWhenRestored(); int active_alpha = activation_animation_->CurrentValueBetween(0, 255); int inactive_alpha = 255 - active_alpha; SkPaint paint; if (inactive_alpha > 0) { if (active_alpha > 0) paint.setXfermodeMode(SkXfermode::kPlus_Mode); paint.setAlpha(inactive_alpha); gfx::ImageSkia inactive_frame = *GetInactiveFrameImage(); TileRoundRect(canvas, inactive_frame, paint, GetLocalBounds(), corner_radius); } if (active_alpha > 0) { paint.setAlpha(active_alpha); gfx::ImageSkia active_frame = *GetActiveFrameImage(); TileRoundRect(canvas, active_frame, paint, GetLocalBounds(), corner_radius); } if (!frame_->IsMaximized() && !frame_->IsFullscreen() && mode_ == MODE_INACTIVE) { PaintHighlightForInactiveRestoredWindow(canvas); } if (frame_->widget_delegate() && frame_->widget_delegate()->ShouldShowWindowTitle()) { PaintTitleBar(canvas); } PaintHeaderContentSeparator(canvas); } void DefaultHeaderPainter::LayoutHeader() { caption_button_container_->Layout(); gfx::Size caption_button_container_size = caption_button_container_->GetPreferredSize(); caption_button_container_->SetBounds( view_->width() - caption_button_container_size.width(), 0, caption_button_container_size.width(), caption_button_container_size.height()); if (window_icon_) { // Vertically center the window icon with respect to the caption button // container. // Floor when computing the center of |caption_button_container_|. int icon_offset_y = caption_button_container_->height() / 2 - window_icon_size_ / 2; window_icon_->SetBounds(HeaderPainterUtil::GetIconXOffset(), icon_offset_y, window_icon_size_, window_icon_size_); } // The header/content separator line overlays the caption buttons. SetHeaderHeightForPainting(caption_button_container_->height()); } int DefaultHeaderPainter::GetHeaderHeightForPainting() const { return height_; } void DefaultHeaderPainter::SetHeaderHeightForPainting(int height) { height_ = height; } void DefaultHeaderPainter::SchedulePaintForTitle() { view_->SchedulePaintInRect(GetTitleBounds()); } void DefaultHeaderPainter::UpdateWindowIcon(views::View* window_icon, int window_icon_size) { window_icon_ = window_icon; window_icon_size_ = window_icon_size; } /////////////////////////////////////////////////////////////////////////////// // gfx::AnimationDelegate overrides: void DefaultHeaderPainter::AnimationProgressed( const gfx::Animation* animation) { view_->SchedulePaintInRect(GetLocalBounds()); } /////////////////////////////////////////////////////////////////////////////// // DefaultHeaderPainter, private: void DefaultHeaderPainter::PaintHighlightForInactiveRestoredWindow( gfx::Canvas* canvas) { ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); gfx::ImageSkia top_edge = *rb.GetImageSkiaNamed( IDR_AURA_WINDOW_HEADER_SHADE_INACTIVE_TOP); gfx::ImageSkia left_edge = *rb.GetImageSkiaNamed( IDR_AURA_WINDOW_HEADER_SHADE_INACTIVE_LEFT); gfx::ImageSkia right_edge = *rb.GetImageSkiaNamed( IDR_AURA_WINDOW_HEADER_SHADE_INACTIVE_RIGHT); gfx::ImageSkia bottom_edge = *rb.GetImageSkiaNamed( IDR_AURA_WINDOW_HEADER_SHADE_INACTIVE_BOTTOM); int left_edge_width = left_edge.width(); int right_edge_width = right_edge.width(); canvas->DrawImageInt(left_edge, 0, 0); canvas->DrawImageInt(right_edge, view_->width() - right_edge_width, 0); canvas->TileImageInt( top_edge, left_edge_width, 0, view_->width() - left_edge_width - right_edge_width, top_edge.height()); DCHECK_EQ(left_edge.height(), right_edge.height()); int bottom = left_edge.height(); int bottom_height = bottom_edge.height(); canvas->TileImageInt( bottom_edge, left_edge_width, bottom - bottom_height, view_->width() - left_edge_width - right_edge_width, bottom_height); } void DefaultHeaderPainter::PaintTitleBar(gfx::Canvas* canvas) { // The window icon is painted by its own views::View. gfx::Rect title_bounds = GetTitleBounds(); title_bounds.set_x(view_->GetMirroredXForRect(title_bounds)); canvas->DrawStringRectWithFlags(frame_->widget_delegate()->GetWindowTitle(), GetTitleFontList(), kTitleTextColor, title_bounds, gfx::Canvas::NO_SUBPIXEL_RENDERING); } void DefaultHeaderPainter::PaintHeaderContentSeparator(gfx::Canvas* canvas) { SkColor color = (mode_ == MODE_ACTIVE) ? kHeaderContentSeparatorColor : kHeaderContentSeparatorInactiveColor; SkPaint paint; paint.setColor(color); // Draw the line as 1px thick regardless of scale factor. paint.setStrokeWidth(0); float thickness = 1 / canvas->image_scale(); SkScalar y = SkIntToScalar(height_) - SkFloatToScalar(thickness); canvas->sk_canvas()->drawLine(0, y, SkIntToScalar(view_->width()), y, paint); } gfx::Rect DefaultHeaderPainter::GetLocalBounds() const { return gfx::Rect(view_->width(), height_); } gfx::Rect DefaultHeaderPainter::GetTitleBounds() const { return HeaderPainterUtil::GetTitleBounds( window_icon_, caption_button_container_, GetTitleFontList()); } gfx::ImageSkia* DefaultHeaderPainter::GetActiveFrameImage() const { return ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( IDR_AURA_WINDOW_HEADER_BASE); } gfx::ImageSkia* DefaultHeaderPainter::GetInactiveFrameImage() const { int frame_image_id = (frame_->IsMaximized() || frame_->IsFullscreen()) ? IDR_AURA_WINDOW_HEADER_BASE : IDR_AURA_WINDOW_HEADER_BASE_RESTORED_INACTIVE; return ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( frame_image_id); } } // namespace ash