blob: f8aa7719701cd00a89b6e881340062ae813991d2 [file] [log] [blame]
// Copyright 2018 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/shelf/window_preview.h"
#include "ash/resources/vector_icons/vector_icons.h"
#include "ash/shelf/shelf_constants.h"
#include "ash/wm/window_preview_view.h"
#include "ash/wm/window_util.h"
#include "ui/aura/window.h"
#include "ui/gfx/color_palette.h"
#include "ui/gfx/paint_vector_icon.h"
#include "ui/views/background.h"
#include "ui/views/controls/button/image_button.h"
#include "ui/views/controls/label.h"
namespace ash {
// The margins around window titles.
constexpr int kTitleLineHeight = 20;
constexpr int kTitleMarginTop = 10;
constexpr int kTitleMarginBottom = 10;
constexpr int kTitleMarginRight = 16;
// The width and height of close buttons.
constexpr int kCloseButtonSize = 36;
constexpr int kCloseButtonImageSize = 24;
constexpr int kCloseButtonSideBleed = 8;
constexpr SkColor kCloseButtonColor = SK_ColorWHITE;
constexpr SkColor kPreviewContainerBgColor =
SkColorSetA(gfx::kGoogleGrey100, 0x24);
constexpr int kPreviewBorderRadius = 4;
WindowPreview::WindowPreview(aura::Window* window,
Delegate* delegate,
const ui::NativeTheme* theme)
: delegate_(delegate) {
preview_view_ =
new wm::WindowPreviewView(window, /*trilinear_filtering_on_init=*/false);
preview_container_view_ = new views::View();
preview_container_view_->SetBackground(views::CreateRoundedRectBackground(
kPreviewContainerBgColor, kPreviewBorderRadius));
title_ = new views::Label(window->GetTitle());
close_button_ = new views::ImageButton(this);
AddChildView(preview_container_view_);
AddChildView(preview_view_);
AddChildView(title_);
AddChildView(close_button_);
SetStyling(theme);
}
WindowPreview::~WindowPreview() = default;
gfx::Size WindowPreview::CalculatePreferredSize() const {
// The preview itself will always be strictly contained within its container,
// so only the container's size matters to calculate the preferred size.
const gfx::Size container_size = GetPreviewContainerSize();
const int title_height_with_padding =
kTitleLineHeight + kTitleMarginTop + kTitleMarginBottom;
return gfx::Size(container_size.width(),
container_size.height() + title_height_with_padding);
}
void WindowPreview::Layout() {
gfx::Rect content_rect = GetContentsBounds();
gfx::Size title_size = title_->CalculatePreferredSize();
int title_height_with_padding =
kTitleLineHeight + kTitleMarginTop + kTitleMarginBottom;
int title_width =
std::min(title_size.width(),
content_rect.width() - kCloseButtonSize - kTitleMarginRight);
title_->SetBoundsRect(gfx::Rect(content_rect.x(),
content_rect.y() + kTitleMarginTop,
title_width, kTitleLineHeight));
close_button_->SetBoundsRect(
gfx::Rect(content_rect.right() - kCloseButtonSize + kCloseButtonSideBleed,
content_rect.y(), kCloseButtonSize, kCloseButtonSize));
const gfx::Size container_size = GetPreviewContainerSize();
gfx::Size mirror_size = preview_view_->CalculatePreferredSize();
float preview_ratio = static_cast<float>(mirror_size.width()) /
static_cast<float>(mirror_size.height());
int preview_height = kShelfTooltipPreviewHeight;
int preview_width = preview_height * preview_ratio;
if (preview_ratio > kShelfTooltipPreviewMaxRatio) {
// Very wide window.
preview_width = kShelfTooltipPreviewMaxWidth;
preview_height = kShelfTooltipPreviewMaxWidth / preview_ratio;
}
// Center the actual preview over the container, horizontally and vertically.
gfx::Point preview_offset_from_container(
(container_size.width() - preview_width) / 2,
(container_size.height() - preview_height) / 2);
const int preview_container_top =
content_rect.y() + title_height_with_padding;
preview_container_view_->SetBoundsRect(
gfx::Rect(content_rect.x(), preview_container_top, container_size.width(),
container_size.height()));
preview_view_->SetBoundsRect(
gfx::Rect(content_rect.x() + preview_offset_from_container.x(),
preview_container_top + preview_offset_from_container.y(),
preview_width, preview_height));
}
bool WindowPreview::OnMousePressed(const ui::MouseEvent& event) {
if (!preview_view_->bounds().Contains(event.location()))
return false;
aura::Window* target = preview_view_->window();
if (target) {
// The window might have been closed in the mean time.
// TODO: Use WindowObserver to listen to when previewed windows are
// being closed and remove this condition.
wm::ActivateWindow(target);
// This will have the effect of deleting this view.
delegate_->OnPreviewActivated(this);
}
return true;
}
void WindowPreview::ButtonPressed(views::Button* sender,
const ui::Event& event) {
// The close button was pressed.
DCHECK_EQ(sender, close_button_);
aura::Window* target = preview_view_->window();
// The window might have been closed in the mean time.
// TODO: Use WindowObserver to listen to when previewed windows are
// being closed and remove this condition.
if (!target)
return;
wm::CloseWidgetForWindow(target);
// This will have the effect of deleting this view.
delegate_->OnPreviewDismissed(this);
}
void WindowPreview::SetStyling(const ui::NativeTheme* theme) {
SkColor background_color =
theme->GetSystemColor(ui::NativeTheme::kColorId_TooltipBackground);
title_->SetEnabledColor(
theme->GetSystemColor(ui::NativeTheme::kColorId_TooltipText));
title_->SetBackgroundColor(background_color);
// The background is not opaque, so we can't do subpixel rendering.
title_->SetSubpixelRenderingEnabled(false);
close_button_->SetImage(
views::Button::STATE_NORMAL,
gfx::CreateVectorIcon(kOverviewWindowCloseIcon, kCloseButtonColor));
close_button_->SetImageAlignment(views::ImageButton::ALIGN_CENTER,
views::ImageButton::ALIGN_MIDDLE);
close_button_->SetMinimumImageSize(
gfx::Size(kCloseButtonImageSize, kCloseButtonImageSize));
}
gfx::Size WindowPreview::GetPreviewContainerSize() const {
return gfx::Size(
std::min(delegate_->GetMaxPreviewRatio() * kShelfTooltipPreviewHeight,
static_cast<float>(kShelfTooltipPreviewMaxWidth)),
kShelfTooltipPreviewHeight);
}
} // namespace ash