| Index: mash/wm/frame/non_client_frame_view_mash.cc
|
| diff --git a/mash/wm/frame/non_client_frame_view_mash.cc b/mash/wm/frame/non_client_frame_view_mash.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..f3bee2debda40a26f0f43d5944442108b11f09a6
|
| --- /dev/null
|
| +++ b/mash/wm/frame/non_client_frame_view_mash.cc
|
| @@ -0,0 +1,351 @@
|
| +// Copyright 2015 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 "mash/wm/frame/non_client_frame_view_mash.h"
|
| +
|
| +#include <algorithm>
|
| +#include <vector>
|
| +
|
| +#include "components/mus/public/cpp/window.h"
|
| +#include "grit/mash_wm_resources.h"
|
| +#include "mash/wm/frame/caption_buttons/frame_caption_button_container_view.h"
|
| +#include "mash/wm/frame/default_header_painter.h"
|
| +#include "mash/wm/frame/frame_border_hit_test_controller.h"
|
| +#include "mash/wm/frame/header_painter.h"
|
| +#include "mash/wm/frame/move_loop.h"
|
| +#include "mojo/converters/input_events/input_events_type_converters.h"
|
| +#include "ui/base/resource/resource_bundle.h"
|
| +#include "ui/compositor/paint_recorder.h"
|
| +#include "ui/gfx/canvas.h"
|
| +#include "ui/gfx/geometry/rect.h"
|
| +#include "ui/gfx/geometry/rect_conversions.h"
|
| +#include "ui/gfx/geometry/size.h"
|
| +#include "ui/gfx/image/image.h"
|
| +#include "ui/views/view.h"
|
| +#include "ui/views/widget/widget.h"
|
| +#include "ui/views/widget/widget_delegate.h"
|
| +
|
| +namespace mash {
|
| +namespace wm {
|
| +
|
| +///////////////////////////////////////////////////////////////////////////////
|
| +// NonClientFrameViewMash::HeaderView
|
| +
|
| +// View which paints the header.
|
| +class NonClientFrameViewMash::HeaderView : public views::View {
|
| + public:
|
| + // |frame| is the widget that the caption buttons act on.
|
| + explicit HeaderView(views::Widget* frame);
|
| + ~HeaderView() override;
|
| +
|
| + // Schedules a repaint for the entire title.
|
| + void SchedulePaintForTitle();
|
| +
|
| + // Tells the window controls to reset themselves to the normal state.
|
| + void ResetWindowControls();
|
| +
|
| + // Returns the view's preferred height.
|
| + int GetPreferredHeight() const;
|
| +
|
| + // Returns the view's minimum width.
|
| + int GetMinimumWidth() const;
|
| +
|
| + void SizeConstraintsChanged();
|
| +
|
| + void SetFrameColors(SkColor active_frame_color, SkColor inactive_frame_color);
|
| +
|
| + // views::View:
|
| + void Layout() override;
|
| + void OnPaint(gfx::Canvas* canvas) override;
|
| + void ChildPreferredSizeChanged(views::View* child) override;
|
| +
|
| + FrameCaptionButtonContainerView* caption_button_container() {
|
| + return caption_button_container_;
|
| + }
|
| +
|
| + private:
|
| + // The widget that the caption buttons act on.
|
| + views::Widget* frame_;
|
| +
|
| + // Helper for painting the header.
|
| + scoped_ptr<DefaultHeaderPainter> header_painter_;
|
| +
|
| + // View which contains the window caption buttons.
|
| + FrameCaptionButtonContainerView* caption_button_container_;
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(HeaderView);
|
| +};
|
| +
|
| +NonClientFrameViewMash::HeaderView::HeaderView(views::Widget* frame)
|
| + : frame_(frame),
|
| + header_painter_(new DefaultHeaderPainter),
|
| + caption_button_container_(nullptr) {
|
| + caption_button_container_ = new FrameCaptionButtonContainerView(frame_);
|
| + caption_button_container_->UpdateSizeButtonVisibility();
|
| + AddChildView(caption_button_container_);
|
| +
|
| + header_painter_->Init(frame_, this, caption_button_container_);
|
| +}
|
| +
|
| +NonClientFrameViewMash::HeaderView::~HeaderView() {}
|
| +
|
| +void NonClientFrameViewMash::HeaderView::SchedulePaintForTitle() {
|
| + header_painter_->SchedulePaintForTitle();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::HeaderView::ResetWindowControls() {
|
| + caption_button_container_->ResetWindowControls();
|
| +}
|
| +
|
| +int NonClientFrameViewMash::HeaderView::GetPreferredHeight() const {
|
| + return header_painter_->GetHeaderHeightForPainting();
|
| +}
|
| +
|
| +int NonClientFrameViewMash::HeaderView::GetMinimumWidth() const {
|
| + return header_painter_->GetMinimumHeaderWidth();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::HeaderView::SizeConstraintsChanged() {
|
| + caption_button_container_->ResetWindowControls();
|
| + caption_button_container_->UpdateSizeButtonVisibility();
|
| + Layout();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::HeaderView::SetFrameColors(
|
| + SkColor active_frame_color,
|
| + SkColor inactive_frame_color) {
|
| + header_painter_->SetFrameColors(active_frame_color, inactive_frame_color);
|
| +}
|
| +
|
| +///////////////////////////////////////////////////////////////////////////////
|
| +// NonClientFrameViewMash::HeaderView, views::View overrides:
|
| +
|
| +void NonClientFrameViewMash::HeaderView::Layout() {
|
| + header_painter_->LayoutHeader();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::HeaderView::OnPaint(gfx::Canvas* canvas) {
|
| + bool paint_as_active =
|
| + frame_->non_client_view()->frame_view()->ShouldPaintAsActive();
|
| + caption_button_container_->SetPaintAsActive(paint_as_active);
|
| +
|
| + HeaderPainter::Mode header_mode = paint_as_active
|
| + ? HeaderPainter::MODE_ACTIVE
|
| + : HeaderPainter::MODE_INACTIVE;
|
| + header_painter_->PaintHeader(canvas, header_mode);
|
| +}
|
| +
|
| +void NonClientFrameViewMash::HeaderView::ChildPreferredSizeChanged(
|
| + views::View* child) {
|
| + // FrameCaptionButtonContainerView animates the visibility changes in
|
| + // UpdateSizeButtonVisibility(false). Due to this a new size is not available
|
| + // until the completion of the animation. Layout in response to the preferred
|
| + // size changes.
|
| + if (child != caption_button_container_)
|
| + return;
|
| + parent()->Layout();
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NonClientFrameViewMash, public:
|
| +
|
| +// static
|
| +const char NonClientFrameViewMash::kViewClassName[] = "NonClientFrameViewMash";
|
| +
|
| +NonClientFrameViewMash::NonClientFrameViewMash(views::Widget* frame,
|
| + mus::Window* window)
|
| + : frame_(frame), window_(window), header_view_(new HeaderView(frame)) {
|
| + // |header_view_| is set as the non client view's overlay view so that it can
|
| + // overlay the web contents in immersive fullscreen.
|
| + AddChildView(header_view_);
|
| + window_->AddObserver(this);
|
| +}
|
| +
|
| +NonClientFrameViewMash::~NonClientFrameViewMash() {
|
| + if (window_)
|
| + window_->RemoveObserver(this);
|
| +}
|
| +
|
| +// static
|
| +gfx::Insets NonClientFrameViewMash::GetPreferredClientAreaInsets() {
|
| + ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
|
| + const int header_height =
|
| + rb.GetImageSkiaNamed(IDR_MASH_WM_WINDOW_CONTROL_BACKGROUND_P)
|
| + ->size()
|
| + .height();
|
| + return gfx::Insets(header_height, 0, 0, 0);
|
| +}
|
| +
|
| +void NonClientFrameViewMash::SetFrameColors(SkColor active_frame_color,
|
| + SkColor inactive_frame_color) {
|
| + header_view_->SetFrameColors(active_frame_color, inactive_frame_color);
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NonClientFrameViewMash, views::NonClientFrameView overrides:
|
| +
|
| +gfx::Rect NonClientFrameViewMash::GetBoundsForClientView() const {
|
| + gfx::Rect result(GetLocalBounds());
|
| + result.Inset(window_->client_area());
|
| + return result;
|
| +}
|
| +
|
| +gfx::Rect NonClientFrameViewMash::GetWindowBoundsForClientBounds(
|
| + const gfx::Rect& client_bounds) const {
|
| + gfx::Rect window_bounds = client_bounds;
|
| + window_bounds.Inset(
|
| + window_->client_area().left(), window_->client_area().top(),
|
| + window_->client_area().right(), window_->client_area().bottom());
|
| + return window_bounds;
|
| +}
|
| +
|
| +int NonClientFrameViewMash::NonClientHitTest(const gfx::Point& point) {
|
| + return FrameBorderHitTestController::NonClientHitTest(
|
| + this, header_view_->caption_button_container(), point);
|
| +}
|
| +
|
| +void NonClientFrameViewMash::GetWindowMask(const gfx::Size& size,
|
| + gfx::Path* window_mask) {}
|
| +
|
| +void NonClientFrameViewMash::ResetWindowControls() {
|
| + header_view_->ResetWindowControls();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::UpdateWindowIcon() {}
|
| +
|
| +void NonClientFrameViewMash::UpdateWindowTitle() {
|
| + header_view_->SchedulePaintForTitle();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::SizeConstraintsChanged() {
|
| + header_view_->SizeConstraintsChanged();
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NonClientFrameViewMash, views::View overrides:
|
| +
|
| +void NonClientFrameViewMash::Layout() {
|
| + header_view_->SetBounds(0, 0, width(), header_view_->GetPreferredHeight());
|
| + header_view_->Layout();
|
| +}
|
| +
|
| +gfx::Size NonClientFrameViewMash::GetPreferredSize() const {
|
| + gfx::Size pref = frame_->client_view()->GetPreferredSize();
|
| + return frame_->non_client_view()
|
| + ->GetWindowBoundsForClientBounds(gfx::Rect(pref))
|
| + .size();
|
| +}
|
| +
|
| +const char* NonClientFrameViewMash::GetClassName() const {
|
| + return kViewClassName;
|
| +}
|
| +
|
| +gfx::Size NonClientFrameViewMash::GetMinimumSize() const {
|
| + gfx::Size min_client_view_size(frame_->client_view()->GetMinimumSize());
|
| + return gfx::Size(
|
| + std::max(header_view_->GetMinimumWidth(), min_client_view_size.width()),
|
| + NonClientTopBorderHeight() + min_client_view_size.height());
|
| +}
|
| +
|
| +gfx::Size NonClientFrameViewMash::GetMaximumSize() const {
|
| + gfx::Size max_client_size(frame_->client_view()->GetMaximumSize());
|
| + int width = 0;
|
| + int height = 0;
|
| +
|
| + if (max_client_size.width() > 0)
|
| + width = std::max(header_view_->GetMinimumWidth(), max_client_size.width());
|
| + if (max_client_size.height() > 0)
|
| + height = NonClientTopBorderHeight() + max_client_size.height();
|
| +
|
| + return gfx::Size(width, height);
|
| +}
|
| +
|
| +void NonClientFrameViewMash::OnPaint(gfx::Canvas* canvas) {
|
| + canvas->Save();
|
| + NonClientFrameView::OnPaint(canvas);
|
| + canvas->Restore();
|
| +
|
| + // The client app draws the client area. Make ours totally transparent so
|
| + // we only see the client apps client area.
|
| + canvas->FillRect(GetBoundsForClientView(), SK_ColorBLACK,
|
| + SkXfermode::kSrc_Mode);
|
| +}
|
| +
|
| +void NonClientFrameViewMash::PaintChildren(const ui::PaintContext& context) {
|
| + NonClientFrameView::PaintChildren(context);
|
| +
|
| + // The client app draws the client area. Make ours totally transparent so
|
| + // we only see the client apps client area.
|
| + ui::PaintRecorder recorder(context, size(), &paint_cache_);
|
| + recorder.canvas()->FillRect(GetBoundsForClientView(), SK_ColorBLACK,
|
| + SkXfermode::kSrc_Mode);
|
| +}
|
| +
|
| +bool NonClientFrameViewMash::OnMousePressed(const ui::MouseEvent& event) {
|
| + return StartMoveLoopIfNecessary(event);
|
| +}
|
| +
|
| +bool NonClientFrameViewMash::OnMouseDragged(const ui::MouseEvent& event) {
|
| + ContinueMove(event);
|
| + return move_loop_.get() != nullptr;
|
| +}
|
| +
|
| +void NonClientFrameViewMash::OnMouseReleased(const ui::MouseEvent& event) {
|
| + ContinueMove(event);
|
| +}
|
| +
|
| +void NonClientFrameViewMash::OnMouseCaptureLost() {
|
| + StopMove();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::OnWindowClientAreaChanged(
|
| + mus::Window* window,
|
| + const gfx::Insets& old_client_area) {
|
| + Layout();
|
| + // NonClientView (our parent) positions the client view based on bounds from
|
| + // us. We need to layout from parent to trigger a layout of the client view.
|
| + if (parent())
|
| + parent()->Layout();
|
| + SchedulePaint();
|
| +}
|
| +
|
| +void NonClientFrameViewMash::OnWindowDestroyed(mus::Window* window) {
|
| + window_->RemoveObserver(this);
|
| + window_ = nullptr;
|
| +}
|
| +
|
| +bool NonClientFrameViewMash::StartMoveLoopIfNecessary(const ui::Event& event) {
|
| + if (move_loop_)
|
| + return false;
|
| + // TODO(sky): convert MoveLoop to take ui::Event.
|
| + // TODO(sky): pass in hit test result.
|
| + move_loop_ = MoveLoop::Create(window_, *mus::mojom::Event::From(event));
|
| + return true;
|
| +}
|
| +
|
| +void NonClientFrameViewMash::ContinueMove(const ui::Event& event) {
|
| + // TODO(sky): convert MoveLoop to take ui::Event.
|
| + if (move_loop_ &&
|
| + move_loop_->Move(*mus::mojom::Event::From(event)) == MoveLoop::DONE) {
|
| + move_loop_.reset();
|
| + }
|
| +}
|
| +
|
| +void NonClientFrameViewMash::StopMove() {
|
| + move_loop_.reset();
|
| +}
|
| +
|
| +views::View* NonClientFrameViewMash::GetHeaderView() {
|
| + return header_view_;
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NonClientFrameViewMash, private:
|
| +
|
| +int NonClientFrameViewMash::NonClientTopBorderHeight() const {
|
| + return header_view_->GetPreferredHeight();
|
| +}
|
| +
|
| +} // namespace wm
|
| +} // namespace mash
|
|
|