| Index: ash/frame/default_header_painter.cc
|
| diff --git a/ash/frame/default_header_painter.cc b/ash/frame/default_header_painter.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..d76096190a20796f31f46cc5c23e1757bb7f3887
|
| --- /dev/null
|
| +++ b/ash/frame/default_header_painter.cc
|
| @@ -0,0 +1,319 @@
|
| +// 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);
|
| +// Size of header/content separator line.
|
| +const int kHeaderContentSeparatorSize = 1;
|
| +// Color of the active window header/content separator line.
|
| +const SkColor kHeaderContentSeparatorColor = SkColorSetRGB(180, 180, 182);
|
| +// Color of the inactive window header/content separator line.
|
| +const SkColor kHeaderContentSeparatorInactiveColor =
|
| + SkColorSetRGB(150, 150, 152);
|
| +// 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),
|
| + 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;
|
| +
|
| + ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
|
| + SkPaint paint;
|
| + if (inactive_alpha > 0) {
|
| + if (active_alpha > 0)
|
| + paint.setXfermodeMode(SkXfermode::kPlus_Mode);
|
| +
|
| + paint.setAlpha(inactive_alpha);
|
| + gfx::ImageSkia inactive_frame =
|
| + *rb.GetImageSkiaNamed(IDR_AURA_WINDOW_HEADER_BASE_INACTIVE);
|
| + TileRoundRect(canvas, inactive_frame, paint, GetLocalBounds(),
|
| + corner_radius);
|
| + }
|
| +
|
| + if (active_alpha > 0) {
|
| + paint.setAlpha(active_alpha);
|
| + gfx::ImageSkia active_frame =
|
| + *rb.GetImageSkiaNamed(IDR_AURA_WINDOW_HEADER_BASE_ACTIVE);
|
| + 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.
|
| + int icon_size = HeaderPainterUtil::GetIconSize();
|
| + int icon_offset_y = (caption_button_container_->height() - icon_size) / 2;
|
| + window_icon_->SetBounds(HeaderPainterUtil::GetIconXOffset(), icon_offset_y,
|
| + icon_size, icon_size);
|
| + }
|
| +
|
| + SetHeaderHeightForPainting(caption_button_container_->height() +
|
| + kHeaderContentSeparatorSize);
|
| +}
|
| +
|
| +int DefaultHeaderPainter::GetHeaderHeightForPainting() const {
|
| + return height_;
|
| +}
|
| +
|
| +void DefaultHeaderPainter::SetHeaderHeightForPainting(int height) {
|
| + height_ = height;
|
| +}
|
| +
|
| +void DefaultHeaderPainter::SchedulePaintForTitle() {
|
| + view_->SchedulePaintInRect(GetTitleBounds());
|
| +}
|
| +
|
| +///////////////////////////////////////////////////////////////////////////////
|
| +// 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;
|
| +
|
| + canvas->FillRect(gfx::Rect(0,
|
| + height_ - kHeaderContentSeparatorSize,
|
| + view_->width(),
|
| + kHeaderContentSeparatorSize),
|
| + color);
|
| +}
|
| +
|
| +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());
|
| +}
|
| +
|
| +} // namespace ash
|
|
|