| 
									
										
										
										
											2014-10-31 11:17:05 -07:00
										 |  |  | // Copyright (c) 2014 GitHub, Inc.
 | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/ui/views/frameless_view.h"
 | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/native_window_views.h"
 | 
					
						
							| 
									
										
											  
											
												refactor: remove InspectableWebContentsViewMac in favor of the Views version (#44628)
* refactor: remove InspectableWebContentsViewMac in favor of the Views version
* cherry-pick: refactor: remove InspectableWebContentsViewMac in favor of the Views version (#41326)
commit e67ab9a93dadccecff30de50ab4555191c30b6c4
Confilcts not resolved, except removal of the files removed
by the original commit.
* resolved conflicts and build issues after cherry-pick
* cherry-picked: fix: add method allowing to disable headless mode in native widget
https://github.com/electron/electron/pull/42996
fixing
https://github.com/electron/electron/issues/42995
* fix: displaying select popup in window created as fullscreen window
`constrainFrameRect:toScreen:` is not being call for windows created
with `fullscreen: true` therefore `headless` mode was not being removed
and `RenderWidgetHostNSViewBridge::DisplayPopupMenu` ignored displaying
popup.
Issue could be fixed by placing additional removal of `headless` mode
in the `toggleFullScreen:`, but `orderWindow:relativeTo:` is called
both for a regular and a fullscreen window, therefore there will be
a single place fixing both cases.
Because `electron::NativeWindowMac` lifetime may be shorter than
`ElectronNSWindow` on which macOS may execute `orderWindow:relativeTo:`
we need to clear `shell_` when `NativeWindow` is being closed.
Fixes #43010.
* fix: Content visibility when using `vibrancy`
We need to put `NSVisualEffectView` before `ViewsCompositorSuperview`
otherwise when using `vibrancy` in `BrowserWindow` `NSVisualEffectView`
will hide content displayed by the compositor.
Fixes #43003
Fixes #42336
In fact main issues reported in these tickets were not present after
cherry-picking original refactor switching to `views::WebView`, so
text could be selected and click event was properly generated. However
both issues testcases were using `vibrancy` and actual content was
invisible, because it was covered by the `NSVisualEffectView`.
* fix: EXCEPTION_ACCESS_VIOLATION crash on BrowserWindow.destroy()
Restored postponed deletion of the `NativeWindow`.
Restoration caused `DCHECK(new_parent_ui_layer->GetCompositor());` failure
in `BrowserCompositorMac::SetParentUiLayer` after the spec test:
`chrome extensions chrome.webRequest does not take precedence over Electron webRequest - http`
with stack:
```
7   Electron Framework 0x000000011fe07830 content::BrowserCompositorMac::SetParentUiLayer(ui::Layer*) + 628
8   Electron Framework 0x000000011fe0c154 content::RenderWidgetHostViewMac::SetParentUiLayer(ui::Layer*) + 220
9   Electron Framework 0x000000011fe226a8 content::WebContentsViewMac::CreateViewForWidget(content::RenderWidgetHost*) + 600
10  Electron Framework 0x000000011fd37e4c content::WebContentsImpl::CreateRenderWidgetHostViewForRenderManager(content::RenderViewHost*) + 164
11  Electron Framework 0x000000011fb32278 content::RenderFrameHostManager::CreateSpeculativeRenderFrame(content::SiteInstanceImpl*, bool, scoped_refptr<content::BrowsingContextState> const&) + 816
12  Electron Framework 0x000000011fb2ab8c content::RenderFrameHostManager::CreateSpeculativeRenderFrameHost(content::SiteInstanceImpl*, content::SiteInstanceImpl*, bool) + 1308
13  Electron Framework 0x000000011fb28598 content::RenderFrameHostManager::GetFrameHostForNavigation(content::NavigationRequest*, content::BrowsingContextGroupSwap*, std::__Cr::basic_string<char, std::__Cr::char_traits<char>, std::__Cr::allocator<char>>*) + 1796
14  Electron Framework 0x000000011fa78660 content::NavigationRequest::SelectFrameHostForOnRequestFailedInternal(bool, bool, std::__Cr::optional<std::__Cr::basic_string<char, std::__Cr::char_traits<char>, std::__Cr::allocator<char>>> const&) + 280
15  Electron Framework 0x000000011fa6a994 content::NavigationRequest::OnRequestFailedInternal(network::URLLoaderCompletionStatus const&, bool, std::__Cr::optional<std::__Cr::basic_string<char, std::__Cr::char_traits<char>, std::__Cr::allocator<char>>> const&, bo
+ 1008
16  Electron Framework 0x000000011fa7772c content::NavigationRequest::OnRequestFailed(network::URLLoaderCompletionStatus const&) + 72
17  Electron Framework 0x000000011f8554ac content::NavigationURLLoaderImpl::NotifyRequestFailed(network::URLLoaderCompletionStatus const&) + 248
```
This was probably the reason of removing `NativeWindow` immediately
in order to cleanup `views_host_` in `WebContentsViewMac` to prevent
using layer without compositor in `WebContentsViewMac::CreateViewForWidget`.
`[ElectronNSWindowDelegate windowWillClose:]` is deleting window host
and the compositor used by the `NativeWindow` therefore detach `NativeWindow`
contents from parent. This will clear `views_host_` and prevent failing
mentioned `DCHECK`.
Fixes #42975
* chore: Applied review suggestions
* refactor: directly cleanup shell
---------
Co-authored-by: Samuel Maddock <smaddock@slack-corp.com>
											
										 
											2025-01-17 16:21:10 +01:00
										 |  |  | #include "shell/browser/ui/inspectable_web_contents_view.h"
 | 
					
						
							|  |  |  | #include "ui/aura/window.h"
 | 
					
						
							| 
									
										
										
										
											2014-07-16 16:00:08 +08:00
										 |  |  | #include "ui/base/hit_test.h"
 | 
					
						
							| 
									
										
										
										
											2024-02-21 15:27:05 -05:00
										 |  |  | #include "ui/base/metadata/metadata_impl_macros.h"
 | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08: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 15:40:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-16 18:12:00 -04:00
										 |  |  | FramelessView::FramelessView() = default; | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-16 18:12:00 -04:00
										 |  |  | FramelessView::~FramelessView() = default; | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | void FramelessView::Init(NativeWindowViews* window, views::Widget* frame) { | 
					
						
							|  |  |  |   window_ = window; | 
					
						
							|  |  |  |   frame_ = frame; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int FramelessView::ResizingBorderHitTest(const gfx::Point& point) { | 
					
						
							| 
									
										
										
										
											2022-01-26 15:59:09 -06:00
										 |  |  |   return ResizingBorderHitTestImpl(point, gfx::Insets(kResizeInsideBoundsSize)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int FramelessView::ResizingBorderHitTestImpl(const gfx::Point& point, | 
					
						
							|  |  |  |                                              const gfx::Insets& resize_border) { | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  |   // to be used for resize handles.
 | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |   bool can_ever_resize = frame_->widget_delegate() | 
					
						
							|  |  |  |                              ? frame_->widget_delegate()->CanResize() | 
					
						
							|  |  |  |                              : false; | 
					
						
							| 
									
										
										
										
											2021-06-21 09:10:40 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // https://github.com/electron/electron/issues/611
 | 
					
						
							| 
									
										
										
										
											2021-10-05 12:48:22 -07:00
										 |  |  |   // If window isn't resizable, we should always return HTNOWHERE, otherwise the
 | 
					
						
							| 
									
										
										
										
											2021-06-21 09:10:40 +08:00
										 |  |  |   // hover state of DOM will not be cleared probably.
 | 
					
						
							|  |  |  |   if (!can_ever_resize) | 
					
						
							| 
									
										
										
										
											2021-10-05 12:48:22 -07:00
										 |  |  |     return HTNOWHERE; | 
					
						
							| 
									
										
										
										
											2021-06-21 09:10:40 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08: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 15:59:09 -06: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 15:33:40 +08: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 10:15:57 -08:00
										 |  |  | int FramelessView::NonClientHitTest(const gfx::Point& point) { | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  |   if (frame_->IsFullscreen()) | 
					
						
							|  |  |  |     return HTCLIENT; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-07 10:15:57 -08:00
										 |  |  |   int contents_hit_test = window_->NonClientHitTest(point); | 
					
						
							|  |  |  |   if (contents_hit_test != HTNOWHERE) | 
					
						
							|  |  |  |     return contents_hit_test; | 
					
						
							| 
									
										
										
										
											2020-12-01 15:03:00 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-11 11:07:36 -07:00
										 |  |  |   // Support resizing frameless window by dragging the border.
 | 
					
						
							| 
									
										
										
										
											2022-11-07 10:15:57 -08:00
										 |  |  |   int frame_component = ResizingBorderHitTest(point); | 
					
						
							| 
									
										
										
										
											2021-08-11 11:07:36 -07:00
										 |  |  |   if (frame_component != HTNOWHERE) | 
					
						
							|  |  |  |     return frame_component; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  |   return HTCLIENT; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-29 02:34:12 +02: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 17:18:35 -04:00
										 |  |  | gfx::Size FramelessView::CalculatePreferredSize( | 
					
						
							|  |  |  |     const views::SizeBounds& available_size) const { | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |   return frame_->non_client_view() | 
					
						
							| 
									
										
										
										
											2024-06-07 17:18:35 -04:00
										 |  |  |       ->GetWindowBoundsForClientBounds(gfx::Rect( | 
					
						
							|  |  |  |           frame_->client_view()->CalculatePreferredSize(available_size))) | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |       .size(); | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-01 20:10:14 +08:00
										 |  |  | gfx::Size FramelessView::GetMinimumSize() const { | 
					
						
							| 
									
										
										
										
											2018-05-12 17:37:31 +02:00
										 |  |  |   return window_->GetContentMinimumSize(); | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-01 20:10:14 +08:00
										 |  |  | gfx::Size FramelessView::GetMaximumSize() const { | 
					
						
							| 
									
										
										
										
											2021-01-05 17:17:33 +09: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 15:33:40 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-21 15:27:05 -05:00
										 |  |  | BEGIN_METADATA(FramelessView) | 
					
						
							|  |  |  | END_METADATA | 
					
						
							| 
									
										
										
										
											2014-07-16 15:33:40 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |