From 3d36a6a7938af3bc10e820a73b433605b80ab8aa Mon Sep 17 00:00:00 2001 From: Cheng Zhao Date: Mon, 7 Jul 2014 22:39:39 +0800 Subject: [PATCH] views: Make resizable frameless window work. --- atom/browser/native_window_views.cc | 15 +++++++++++++-- atom/browser/ui/views/linux_frame_view.cc | 15 ++++++++++++++- atom/browser/ui/views/linux_frame_view.h | 3 +++ 3 files changed, 30 insertions(+), 3 deletions(-) diff --git a/atom/browser/native_window_views.cc b/atom/browser/native_window_views.cc index 37148e12d6f1..6cbd506184be 100644 --- a/atom/browser/native_window_views.cc +++ b/atom/browser/native_window_views.cc @@ -14,6 +14,7 @@ #include "content/public/browser/native_web_keyboard_event.h" #include "content/public/browser/web_contents_view.h" #include "native_mate/dictionary.h" +#include "ui/base/hit_test.h" #include "ui/gfx/image/image.h" #include "ui/gfx/image/image_skia.h" #include "ui/views/background.h" @@ -335,8 +336,18 @@ bool NativeWindowViews::ShouldDescendIntoChildForEventHandling( gfx::NativeView child, const gfx::Point& location) { // App window should claim mouse events that fall within the draggable region. - return !draggable_region_.get() || - !draggable_region_->contains(location.x(), location.y()); + if (draggable_region_.get() && + draggable_region_->contains(location.x(), location.y())) + return false; + + // And the events on border for dragging resizable frameless window. + if (!has_frame_ && CanResize()) { + LinuxFrameView* frame = static_cast( + window_->non_client_view()->frame_view()); + return frame->ResizingBorderHitTest(location) == HTNOWHERE; + } + + return true; } views::ClientView* NativeWindowViews::CreateClientView(views::Widget* widget) { diff --git a/atom/browser/ui/views/linux_frame_view.cc b/atom/browser/ui/views/linux_frame_view.cc index 3d31f5feaa43..7832913657ee 100644 --- a/atom/browser/ui/views/linux_frame_view.cc +++ b/atom/browser/ui/views/linux_frame_view.cc @@ -84,7 +84,7 @@ LinuxFrameView::LinuxFrameView() restore_button_(NULL), close_button_(NULL), should_show_maximize_button_(false), - frame_background_(new views::FrameBackground()) { + frame_background_(new views::FrameBackground) { } LinuxFrameView::~LinuxFrameView() { @@ -118,6 +118,12 @@ void LinuxFrameView::Init(NativeWindowViews* window, views::Widget* frame) { } } +int LinuxFrameView::ResizingBorderHitTest(const gfx::Point& point) { + return GetHTComponentForFrame(point, FrameBorderThickness(), + FrameBorderThickness() + kClientEdgeThickness, kResizeAreaCornerSize, + kResizeAreaCornerSize, true); +} + /////////////////////////////////////////////////////////////////////////////// // LinuxFrameView, NonClientFrameView implementation: @@ -146,6 +152,13 @@ int LinuxFrameView::NonClientHitTest(const gfx::Point& point) { if (draggable_region && draggable_region->contains(point.x(), point.y())) return HTCAPTION; + // Support resizing frameless window by dragging the border. + if (!window_->has_frame() && frame_->widget_delegate()->CanResize()) { + int window_component = ResizingBorderHitTest(point); + if (window_component != HTNOWHERE) + return window_component; + } + int frame_component = frame_->client_view()->NonClientHitTest(point); // See if we're in the sysmenu region. (We check the ClientView first to be diff --git a/atom/browser/ui/views/linux_frame_view.h b/atom/browser/ui/views/linux_frame_view.h index bb059aab7ab6..b1d8d7069f20 100644 --- a/atom/browser/ui/views/linux_frame_view.h +++ b/atom/browser/ui/views/linux_frame_view.h @@ -56,6 +56,9 @@ class LinuxFrameView : public views::NonClientFrameView, virtual void ButtonPressed(views::Button* sender, const ui::Event& event) OVERRIDE; + // Returns whether the |point| is on frameless window's resizing border. + int ResizingBorderHitTest(const gfx::Point& point); + private: // Returns the thickness of the border that makes up the window frame edges. // This does not include any client edge.