2014-10-31 18:17:05 +00:00
|
|
|
// Copyright (c) 2014 GitHub, Inc.
|
2014-07-16 07:33:40 +00:00
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2019-06-19 20:46:59 +00:00
|
|
|
#include "shell/browser/ui/views/frameless_view.h"
|
2014-07-16 07:33:40 +00:00
|
|
|
|
2019-06-19 20:46:59 +00:00
|
|
|
#include "shell/browser/native_window_views.h"
|
2014-07-16 08:00:08 +00:00
|
|
|
#include "ui/base/hit_test.h"
|
2024-02-21 20:27:05 +00:00
|
|
|
#include "ui/base/metadata/metadata_impl_macros.h"
|
2014-07-16 07:33:40 +00:00
|
|
|
#include "ui/views/widget/widget.h"
|
|
|
|
#include "ui/views/widget/widget_delegate.h"
|
|
|
|
|
|
|
|
namespace electron {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
const int kResizeInsideBoundsSize = 5;
|
|
|
|
const int kResizeAreaCornerSize = 16;
|
2014-07-16 07:40:22 +00:00
|
|
|
|
2014-07-16 07:33:40 +00:00
|
|
|
} // namespace
|
|
|
|
|
2019-09-16 22:12:00 +00:00
|
|
|
FramelessView::FramelessView() = default;
|
2014-07-16 07:33:40 +00:00
|
|
|
|
2019-09-16 22:12:00 +00:00
|
|
|
FramelessView::~FramelessView() = default;
|
2014-07-16 07:33:40 +00:00
|
|
|
|
|
|
|
void FramelessView::Init(NativeWindowViews* window, views::Widget* frame) {
|
|
|
|
window_ = window;
|
|
|
|
frame_ = frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
int FramelessView::ResizingBorderHitTest(const gfx::Point& point) {
|
2022-01-26 21:59:09 +00:00
|
|
|
return ResizingBorderHitTestImpl(point, gfx::Insets(kResizeInsideBoundsSize));
|
|
|
|
}
|
|
|
|
|
|
|
|
int FramelessView::ResizingBorderHitTestImpl(const gfx::Point& point,
|
|
|
|
const gfx::Insets& resize_border) {
|
2014-07-16 07:33:40 +00:00
|
|
|
// to be used for resize handles.
|
2018-04-18 01:55:30 +00:00
|
|
|
bool can_ever_resize = frame_->widget_delegate()
|
|
|
|
? frame_->widget_delegate()->CanResize()
|
|
|
|
: false;
|
2021-06-21 01:10:40 +00:00
|
|
|
|
|
|
|
// https://github.com/electron/electron/issues/611
|
2021-10-05 19:48:22 +00:00
|
|
|
// If window isn't resizable, we should always return HTNOWHERE, otherwise the
|
2021-06-21 01:10:40 +00:00
|
|
|
// hover state of DOM will not be cleared probably.
|
|
|
|
if (!can_ever_resize)
|
2021-10-05 19:48:22 +00:00
|
|
|
return HTNOWHERE;
|
2021-06-21 01:10:40 +00:00
|
|
|
|
2014-07-16 07:33:40 +00:00
|
|
|
// Don't allow overlapping resize handles when the window is maximized or
|
|
|
|
// fullscreen, as it can't be resized in those states.
|
2022-01-26 21:59:09 +00:00
|
|
|
bool allow_overlapping_handles =
|
|
|
|
!frame_->IsMaximized() && !frame_->IsFullscreen();
|
|
|
|
return GetHTComponentForFrame(
|
|
|
|
point, allow_overlapping_handles ? resize_border : gfx::Insets(),
|
|
|
|
kResizeAreaCornerSize, kResizeAreaCornerSize, can_ever_resize);
|
2014-07-16 07:33:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2022-11-07 18:15:57 +00:00
|
|
|
int FramelessView::NonClientHitTest(const gfx::Point& point) {
|
2014-07-16 07:33:40 +00:00
|
|
|
if (frame_->IsFullscreen())
|
|
|
|
return HTCLIENT;
|
|
|
|
|
2022-11-07 18:15:57 +00:00
|
|
|
int contents_hit_test = window_->NonClientHitTest(point);
|
|
|
|
if (contents_hit_test != HTNOWHERE)
|
|
|
|
return contents_hit_test;
|
2020-12-01 23:03:00 +00:00
|
|
|
|
2021-08-11 18:07:36 +00:00
|
|
|
// Support resizing frameless window by dragging the border.
|
2022-11-07 18:15:57 +00:00
|
|
|
int frame_component = ResizingBorderHitTest(point);
|
2021-08-11 18:07:36 +00:00
|
|
|
if (frame_component != HTNOWHERE)
|
|
|
|
return frame_component;
|
|
|
|
|
2014-07-16 07:33:40 +00:00
|
|
|
return HTCLIENT;
|
|
|
|
}
|
|
|
|
|
2022-04-29 00:34:12 +00:00
|
|
|
views::View* FramelessView::TargetForRect(views::View* root,
|
|
|
|
const gfx::Rect& rect) {
|
|
|
|
CHECK_EQ(root, this);
|
|
|
|
|
|
|
|
if (NonClientHitTest(rect.origin()) != HTCLIENT)
|
|
|
|
return this;
|
|
|
|
|
|
|
|
return NonClientFrameView::TargetForRect(root, rect);
|
|
|
|
}
|
|
|
|
|
2024-06-07 21:18:35 +00:00
|
|
|
gfx::Size FramelessView::CalculatePreferredSize(
|
|
|
|
const views::SizeBounds& available_size) const {
|
2018-04-18 01:55:30 +00:00
|
|
|
return frame_->non_client_view()
|
2024-06-07 21:18:35 +00:00
|
|
|
->GetWindowBoundsForClientBounds(gfx::Rect(
|
|
|
|
frame_->client_view()->CalculatePreferredSize(available_size)))
|
2018-04-18 01:55:30 +00:00
|
|
|
.size();
|
2014-07-16 07:33:40 +00:00
|
|
|
}
|
|
|
|
|
2014-09-01 12:10:14 +00:00
|
|
|
gfx::Size FramelessView::GetMinimumSize() const {
|
2018-05-12 15:37:31 +00:00
|
|
|
return window_->GetContentMinimumSize();
|
2014-07-16 07:33:40 +00:00
|
|
|
}
|
|
|
|
|
2014-09-01 12:10:14 +00:00
|
|
|
gfx::Size FramelessView::GetMaximumSize() const {
|
2021-01-05 08:17:33 +00:00
|
|
|
gfx::Size size = window_->GetContentMaximumSize();
|
|
|
|
// Electron public APIs returns (0, 0) when maximum size is not set, but it
|
|
|
|
// would break internal window APIs like HWNDMessageHandler::SetAspectRatio.
|
|
|
|
return size.IsEmpty() ? gfx::Size(INT_MAX, INT_MAX) : size;
|
2014-07-16 07:33:40 +00:00
|
|
|
}
|
|
|
|
|
2024-02-21 20:27:05 +00:00
|
|
|
BEGIN_METADATA(FramelessView)
|
|
|
|
END_METADATA
|
2014-07-16 07:33:40 +00:00
|
|
|
|
|
|
|
} // namespace electron
|