win: Implement frameless window.

This commit is contained in:
Cheng Zhao 2014-07-16 15:33:40 +08:00
parent db6c37bfbc
commit 6de595f036
8 changed files with 192 additions and 80 deletions

View file

@ -135,6 +135,8 @@
'atom/browser/ui/tray_icon_cocoa.mm', 'atom/browser/ui/tray_icon_cocoa.mm',
'atom/browser/ui/tray_icon_observer.h', 'atom/browser/ui/tray_icon_observer.h',
'atom/browser/ui/tray_icon_win.cc', 'atom/browser/ui/tray_icon_win.cc',
'atom/browser/ui/views/frameless_view.cc',
'atom/browser/ui/views/frameless_view.h',
'atom/browser/ui/views/global_menu_bar_x11.cc', 'atom/browser/ui/views/global_menu_bar_x11.cc',
'atom/browser/ui/views/global_menu_bar_x11.h', 'atom/browser/ui/views/global_menu_bar_x11.h',
'atom/browser/ui/views/linux_frame_view.cc', 'atom/browser/ui/views/linux_frame_view.cc',

View file

@ -81,6 +81,9 @@ NativeWindowViews::NativeWindowViews(content::WebContents* web_contents,
#if defined(USE_X11) #if defined(USE_X11)
// In X11 the window frame is drawn by the application. // In X11 the window frame is drawn by the application.
params.remove_standard_frame = true; params.remove_standard_frame = true;
#elif defined(OS_WIN)
if (!has_frame_)
params.remove_standard_frame = true;
#endif #endif
bool skip_taskbar = false; bool skip_taskbar = false;
@ -370,18 +373,16 @@ bool NativeWindowViews::ShouldDescendIntoChildForEventHandling(
gfx::NativeView child, gfx::NativeView child,
const gfx::Point& location) { const gfx::Point& location) {
// App window should claim mouse events that fall within the draggable region. // App window should claim mouse events that fall within the draggable region.
if (draggable_region_.get() && if (draggable_region_ &&
draggable_region_->contains(location.x(), location.y())) draggable_region_->contains(location.x(), location.y()))
return false; return false;
#if defined(USE_X11)
// And the events on border for dragging resizable frameless window. // And the events on border for dragging resizable frameless window.
if (!has_frame_ && CanResize()) { if (!has_frame_ && CanResize()) {
LinuxFrameView* frame = static_cast<LinuxFrameView*>( FramelessView* frame = static_cast<FramelessView*>(
window_->non_client_view()->frame_view()); window_->non_client_view()->frame_view());
return frame->ResizingBorderHitTest(location) == HTNOWHERE; return frame->ResizingBorderHitTest(location) == HTNOWHERE;
} }
#endif
return true; return true;
} }
@ -394,13 +395,11 @@ views::NonClientFrameView* NativeWindowViews::CreateNonClientFrameView(
views::Widget* widget) { views::Widget* widget) {
#if defined(USE_X11) #if defined(USE_X11)
LinuxFrameView* frame_view = new LinuxFrameView; LinuxFrameView* frame_view = new LinuxFrameView;
#else defined(OS_WIN)
WinFrameView* frame_view = new WinFrameView;
#endif
frame_view->Init(this, widget); frame_view->Init(this, widget);
return frame_view; return frame_view;
#elif defined(OS_WIN)
return new WinFrameView(widget);
#else
return NULL;
#endif
} }
void NativeWindowViews::HandleKeyboardEvent( void NativeWindowViews::HandleKeyboardEvent(

View file

@ -0,0 +1,113 @@
// Copyright (c) 2014 GitHub, Inc. All rights reserved.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#include "atom/browser/ui/views/frameless_view.h"
#include "atom/browser/native_window_views.h"
#include "ui/aura/window.h"
#include "ui/views/widget/widget.h"
#include "ui/views/widget/widget_delegate.h"
namespace atom {
namespace {
const int kResizeInsideBoundsSize = 5;
const int kResizeAreaCornerSize = 16;
const char kViewClassName[] = "FramelessView";
} // namespace
FramelessView::FramelessView() : window_(NULL), frame_(NULL) {
}
FramelessView::~FramelessView() {
}
void FramelessView::Init(NativeWindowViews* window, views::Widget* frame) {
window_ = window;
frame_ = frame;
}
int FramelessView::ResizingBorderHitTest(const gfx::Point& point) {
// Check the frame first, as we allow a small area overlapping the contents
// to be used for resize handles.
bool can_ever_resize = frame_->widget_delegate() ?
frame_->widget_delegate()->CanResize() :
false;
// Don't allow overlapping resize handles when the window is maximized or
// fullscreen, as it can't be resized in those states.
int resize_border =
frame_->IsMaximized() || frame_->IsFullscreen() ? 0 :
kResizeInsideBoundsSize;
return GetHTComponentForFrame(point, resize_border, resize_border,
kResizeAreaCornerSize, kResizeAreaCornerSize, can_ever_resize);
}
gfx::Rect FramelessView::GetBoundsForClientView() const {
return bounds();
}
gfx::Rect FramelessView::GetWindowBoundsForClientBounds(
const gfx::Rect& client_bounds) const {
gfx::Rect window_bounds = client_bounds;
// Enforce minimum size (1, 1) in case that client_bounds is passed with
// empty size. This could occur when the frameless window is being
// initialized.
if (window_bounds.IsEmpty()) {
window_bounds.set_width(1);
window_bounds.set_height(1);
}
return window_bounds;
}
int FramelessView::NonClientHitTest(const gfx::Point& cursor) {
if (frame_->IsFullscreen())
return HTCLIENT;
// Check for possible draggable region in the client area for the frameless
// window.
SkRegion* draggable_region = window_->draggable_region();
if (draggable_region && draggable_region->contains(cursor.x(), cursor.y()))
return HTCAPTION;
// Support resizing frameless window by dragging the border.
int frame_component = ResizingBorderHitTest(cursor);
if (frame_component != HTNOWHERE)
return frame_component;
return HTCLIENT;
}
void FramelessView::GetWindowMask(const gfx::Size& size,
gfx::Path* window_mask) {
}
void FramelessView::ResetWindowControls() {
}
void FramelessView::UpdateWindowIcon() {
}
void FramelessView::UpdateWindowTitle() {
}
gfx::Size FramelessView::GetPreferredSize() {
return frame_->non_client_view()->GetWindowBoundsForClientBounds(
gfx::Rect(frame_->client_view()->GetPreferredSize())).size();
}
gfx::Size FramelessView::GetMinimumSize() {
return window_->GetMinimumSize();
}
gfx::Size FramelessView::GetMaximumSize() {
return window_->GetMaximumSize();
}
const char* FramelessView::GetClassName() const {
return kViewClassName;
}
} // namespace atom

View file

@ -0,0 +1,56 @@
// Copyright (c) 2014 GitHub, Inc. All rights reserved.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#ifndef ATOM_BROWSER_UI_VIEWS_FRAMELESS_VIEW_H_
#define ATOM_BROWSER_UI_VIEWS_FRAMELESS_VIEW_H_
#include "ui/views/window/non_client_view.h"
namespace views {
class Widget;
}
namespace atom {
class NativeWindowViews;
class FramelessView : public views::NonClientFrameView {
public:
FramelessView();
virtual ~FramelessView();
virtual void Init(NativeWindowViews* window, views::Widget* frame);
// Returns whether the |point| is on frameless window's resizing border.
int ResizingBorderHitTest(const gfx::Point& point);
protected:
// views::NonClientFrameView:
virtual gfx::Rect GetBoundsForClientView() const OVERRIDE;
virtual gfx::Rect GetWindowBoundsForClientBounds(
const gfx::Rect& client_bounds) const OVERRIDE;
virtual int NonClientHitTest(const gfx::Point& point) OVERRIDE;
virtual void GetWindowMask(const gfx::Size& size,
gfx::Path* window_mask) OVERRIDE;
virtual void ResetWindowControls() OVERRIDE;
virtual void UpdateWindowIcon() OVERRIDE;
virtual void UpdateWindowTitle() OVERRIDE;
// Overridden from View:
virtual gfx::Size GetPreferredSize() OVERRIDE;
virtual gfx::Size GetMinimumSize() OVERRIDE;
virtual gfx::Size GetMaximumSize() OVERRIDE;
virtual const char* GetClassName() const OVERRIDE;
// Not owned.
NativeWindowViews* window_;
views::Widget* frame_;
private:
DISALLOW_COPY_AND_ASSIGN(FramelessView);
};
} // namespace atom
#endif // ATOM_BROWSER_UI_VIEWS_FRAMELESS_VIEW_H_

View file

@ -91,8 +91,7 @@ LinuxFrameView::~LinuxFrameView() {
} }
void LinuxFrameView::Init(NativeWindowViews* window, views::Widget* frame) { void LinuxFrameView::Init(NativeWindowViews* window, views::Widget* frame) {
window_ = window; FramelessView::Init(window, frame);
frame_ = frame;
close_button_ = new views::ImageButton(this); close_button_ = new views::ImageButton(this);
close_button_->SetAccessibleName( close_button_->SetAccessibleName(

View file

@ -5,12 +5,10 @@
#ifndef ATOM_BROWSER_UI_VIEWS_LINUX_FRAME_VIEW_H_ #ifndef ATOM_BROWSER_UI_VIEWS_LINUX_FRAME_VIEW_H_
#define ATOM_BROWSER_UI_VIEWS_LINUX_FRAME_VIEW_H_ #define ATOM_BROWSER_UI_VIEWS_LINUX_FRAME_VIEW_H_
#include "base/basictypes.h" #include "atom/browser/ui/views/frameless_view.h"
#include "base/compiler_specific.h"
#include "base/memory/scoped_ptr.h" #include "base/memory/scoped_ptr.h"
#include "ui/views/controls/button/button.h" #include "ui/views/controls/button/button.h"
#include "ui/views/window/frame_buttons.h" #include "ui/views/window/frame_buttons.h"
#include "ui/views/window/non_client_view.h"
namespace gfx { namespace gfx {
class ImageSkia; class ImageSkia;
@ -19,20 +17,17 @@ class ImageSkia;
namespace views { namespace views {
class FrameBackground; class FrameBackground;
class ImageButton; class ImageButton;
class Widget;
} }
namespace atom { namespace atom {
class NativeWindowViews; class LinuxFrameView : public FramelessView
class LinuxFrameView : public views::NonClientFrameView,
public views::ButtonListener { public views::ButtonListener {
public: public:
LinuxFrameView(); LinuxFrameView();
virtual ~LinuxFrameView(); virtual ~LinuxFrameView();
void Init(NativeWindowViews* window, views::Widget* frame); void Init(NativeWindowViews* window, views::Widget* frame) OVERRIDE;
// Overridden from NonClientFrameView: // Overridden from NonClientFrameView:
virtual gfx::Rect GetBoundsForClientView() const OVERRIDE; virtual gfx::Rect GetBoundsForClientView() const OVERRIDE;
@ -123,10 +118,6 @@ class LinuxFrameView : public views::NonClientFrameView,
// The layout rect of the title, if visible. // The layout rect of the title, if visible.
gfx::Rect title_bounds_; gfx::Rect title_bounds_;
// Not owned.
NativeWindowViews* window_;
views::Widget* frame_;
// The icon of this window. May be NULL. // The icon of this window. May be NULL.
views::ImageButton* window_icon_; views::ImageButton* window_icon_;

View file

@ -4,6 +4,7 @@
#include "atom/browser/ui/views/win_frame_view.h" #include "atom/browser/ui/views/win_frame_view.h"
#include "atom/browser/native_window_views.h"
#include "ui/views/widget/widget.h" #include "ui/views/widget/widget.h"
#include "ui/views/win/hwnd_util.h" #include "ui/views/win/hwnd_util.h"
@ -16,18 +17,13 @@ const char kViewClassName[] = "WinFrameView";
} // namespace } // namespace
WinFrameView::WinFrameView(views::Widget* frame) WinFrameView::WinFrameView() {
: frame_(frame) {
} }
WinFrameView::~WinFrameView() { WinFrameView::~WinFrameView() {
} }
gfx::Rect WinFrameView::GetBoundsForClientView() const {
return gfx::Rect(0, 0, width(), height());
}
gfx::Rect WinFrameView::GetWindowBoundsForClientBounds( gfx::Rect WinFrameView::GetWindowBoundsForClientBounds(
const gfx::Rect& client_bounds) const { const gfx::Rect& client_bounds) const {
gfx::Size size(client_bounds.size()); gfx::Size size(client_bounds.size());
@ -36,38 +32,10 @@ gfx::Rect WinFrameView::GetWindowBoundsForClientBounds(
} }
int WinFrameView::NonClientHitTest(const gfx::Point& point) { int WinFrameView::NonClientHitTest(const gfx::Point& point) {
if (window_->has_frame())
return frame_->client_view()->NonClientHitTest(point); return frame_->client_view()->NonClientHitTest(point);
} else
return FramelessView::NonClientHitTest(point);
void WinFrameView::GetWindowMask(const gfx::Size& size,
gfx::Path* window_mask) {
// Nothing to do, we use the default window mask.
}
void WinFrameView::ResetWindowControls() {
// Nothing to do.
}
void WinFrameView::UpdateWindowIcon() {
// Nothing to do.
}
void WinFrameView::UpdateWindowTitle() {
// Nothing to do.
}
gfx::Size WinFrameView::GetPreferredSize() {
gfx::Size client_preferred_size = frame_->client_view()->GetPreferredSize();
return frame_->non_client_view()->GetWindowBoundsForClientBounds(
gfx::Rect(client_preferred_size)).size();
}
gfx::Size WinFrameView::GetMinimumSize() {
return frame_->client_view()->GetMinimumSize();
}
gfx::Size WinFrameView::GetMaximumSize() {
return frame_->client_view()->GetMaximumSize();
} }
const char* WinFrameView::GetClassName() const { const char* WinFrameView::GetClassName() const {

View file

@ -5,42 +5,26 @@
#ifndef ATOM_BROWSER_UI_VIEWS_WIN_FRAME_VIEW_H_ #ifndef ATOM_BROWSER_UI_VIEWS_WIN_FRAME_VIEW_H_
#define ATOM_BROWSER_UI_VIEWS_WIN_FRAME_VIEW_H_ #define ATOM_BROWSER_UI_VIEWS_WIN_FRAME_VIEW_H_
#include "ui/views/window/non_client_view.h" #include "atom/browser/ui/views/frameless_view.h"
namespace views {
class Widget;
}
namespace atom { namespace atom {
class WinFrameView : public views::NonClientFrameView { class WinFrameView : public FramelessView {
public: public:
explicit WinFrameView(views::Widget* widget); WinFrameView();
virtual ~WinFrameView(); virtual ~WinFrameView();
// views::NonClientFrameView: // views::NonClientFrameView:
virtual gfx::Rect GetBoundsForClientView() const OVERRIDE;
virtual gfx::Rect GetWindowBoundsForClientBounds( virtual gfx::Rect GetWindowBoundsForClientBounds(
const gfx::Rect& client_bounds) const OVERRIDE; const gfx::Rect& client_bounds) const OVERRIDE;
virtual int NonClientHitTest(const gfx::Point& point) OVERRIDE; virtual int NonClientHitTest(const gfx::Point& point) OVERRIDE;
virtual void GetWindowMask(const gfx::Size& size,
gfx::Path* window_mask) OVERRIDE;
virtual void ResetWindowControls() OVERRIDE;
virtual void UpdateWindowIcon() OVERRIDE;
virtual void UpdateWindowTitle() OVERRIDE;
// views::View: // views::View:
virtual gfx::Size GetPreferredSize() OVERRIDE;
virtual gfx::Size GetMinimumSize() OVERRIDE;
virtual gfx::Size GetMaximumSize() OVERRIDE;
virtual const char* GetClassName() const OVERRIDE; virtual const char* GetClassName() const OVERRIDE;
private: private:
void ClientAreaSizeToWindowSize(gfx::Size* size) const; void ClientAreaSizeToWindowSize(gfx::Size* size) const;
// Our containing frame.
views::Widget* frame_;
DISALLOW_COPY_AND_ASSIGN(WinFrameView); DISALLOW_COPY_AND_ASSIGN(WinFrameView);
}; };