 cee51785e1
			
		
	
	
	
	
	cee51785e1* refactor: inline AutofillPopup::line_count() refactor: inline AutofillPopup::value_at() refactor: inline AutofillPopup::label_at() * refactor: inline NativeWindow::aspect_ratio() refactor: inline NativeWindow::aspect_ratio_extra_size() * refactor: inline BrowserProcessImpl::linux_storage_backend() * refactor: inline ElectronMenuModel::sharing_item() * refactor: inline Browser::badge_count() * refactor: inline WebContents::is_guest() refactor: inline InspectableWebContents::is_guest() * refactor: inline InspectableWebContents::dev_tool_bounds() * refactor: inline WebContents::type()
		
			
				
	
	
		
			407 lines
		
	
	
	
		
			14 KiB
			
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			407 lines
		
	
	
	
		
			14 KiB
			
		
	
	
	
		
			Text
		
	
	
	
	
	
| // Copyright (c) 2018 GitHub, Inc.
 | |
| // Use of this source code is governed by the MIT license that can be
 | |
| // found in the LICENSE file.
 | |
| 
 | |
| #include "shell/browser/ui/cocoa/electron_ns_window_delegate.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| 
 | |
| #include "base/mac/mac_util.h"
 | |
| #include "components/remote_cocoa/app_shim/native_widget_ns_window_bridge.h"
 | |
| #include "shell/browser/browser.h"
 | |
| #include "shell/browser/native_window_mac.h"
 | |
| #include "shell/browser/ui/cocoa/electron_preview_item.h"
 | |
| #include "shell/browser/ui/cocoa/electron_touch_bar.h"
 | |
| #include "ui/gfx/geometry/resize_utils.h"
 | |
| #include "ui/gfx/mac/coordinate_conversion.h"
 | |
| #include "ui/views/cocoa/native_widget_mac_ns_window_host.h"
 | |
| #include "ui/views/widget/native_widget_mac.h"
 | |
| 
 | |
| using TitleBarStyle = electron::NativeWindowMac::TitleBarStyle;
 | |
| using FullScreenTransitionState =
 | |
|     electron::NativeWindow::FullScreenTransitionState;
 | |
| 
 | |
| @implementation ElectronNSWindowDelegate
 | |
| 
 | |
| - (id)initWithShell:(electron::NativeWindowMac*)shell {
 | |
|   // The views library assumes the window delegate must be an instance of
 | |
|   // ViewsNSWindowDelegate, since we don't have a way to override the creation
 | |
|   // of NSWindowDelegate, we have to dynamically replace the window delegate
 | |
|   // on the fly.
 | |
|   // TODO(zcbenz): Add interface in NativeWidgetMac to allow overriding creating
 | |
|   // window delegate.
 | |
|   auto* bridge_host = views::NativeWidgetMacNSWindowHost::GetFromNativeWindow(
 | |
|       shell->GetNativeWindow());
 | |
|   auto* bridged_view = bridge_host->GetInProcessNSWindowBridge();
 | |
|   if ((self = [super initWithBridgedNativeWidget:bridged_view])) {
 | |
|     shell_ = shell;
 | |
|     is_zooming_ = false;
 | |
|     level_ = [shell_->GetNativeWindow().GetNativeNSWindow() level];
 | |
|   }
 | |
|   return self;
 | |
| }
 | |
| 
 | |
| #pragma mark - NSWindowDelegate
 | |
| 
 | |
| - (void)windowDidChangeOcclusionState:(NSNotification*)notification {
 | |
|   // notification.object is the window that changed its state.
 | |
|   // It's safe to use self.window instead if you don't assign one delegate to
 | |
|   // many windows
 | |
|   NSWindow* window = notification.object;
 | |
| 
 | |
|   // check occlusion binary flag
 | |
|   if (window.occlusionState & NSWindowOcclusionStateVisible) {
 | |
|     // There's a macOS bug where if a child window is minimized, and then both
 | |
|     // windows are restored via activation of the parent window, the child
 | |
|     // window is not properly deminiaturized. This causes traffic light bugs
 | |
|     // like the close and miniaturize buttons having no effect. We need to call
 | |
|     // deminiaturize on the child window to fix this. Unfortunately, this also
 | |
|     // hits ANOTHER bug where even after calling deminiaturize,
 | |
|     // windowDidDeminiaturize is not posted on the child window if it was
 | |
|     // incidentally restored by the parent, so we need to manually reset
 | |
|     // is_minimized_ here.
 | |
|     if (shell_->parent() && is_minimized_) {
 | |
|       shell_->Restore();
 | |
|       is_minimized_ = false;
 | |
|     }
 | |
| 
 | |
|     shell_->NotifyWindowShow();
 | |
|   } else {
 | |
|     shell_->NotifyWindowHide();
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Called when the user clicks the zoom button or selects it from the Window
 | |
| // menu to determine the "standard size" of the window.
 | |
| - (NSRect)windowWillUseStandardFrame:(NSWindow*)window
 | |
|                         defaultFrame:(NSRect)frame {
 | |
|   if (!shell_->zoom_to_page_width()) {
 | |
|     if (shell_->aspect_ratio() > 0.0)
 | |
|       shell_->set_default_frame_for_zoom(frame);
 | |
|     return frame;
 | |
|   }
 | |
| 
 | |
|   // If the shift key is down, maximize.
 | |
|   if ([[NSApp currentEvent] modifierFlags] & NSEventModifierFlagShift)
 | |
|     return frame;
 | |
| 
 | |
|   // Get preferred width from observers. Usually the page width.
 | |
|   int preferred_width = 0;
 | |
|   shell_->NotifyWindowRequestPreferredWidth(&preferred_width);
 | |
| 
 | |
|   // Never shrink from the current size on zoom.
 | |
|   NSRect window_frame = [window frame];
 | |
|   CGFloat zoomed_width =
 | |
|       std::max(static_cast<CGFloat>(preferred_width), NSWidth(window_frame));
 | |
| 
 | |
|   // |frame| determines our maximum extents. We need to set the origin of the
 | |
|   // frame -- and only move it left if necessary.
 | |
|   if (window_frame.origin.x + zoomed_width > NSMaxX(frame))
 | |
|     frame.origin.x = NSMaxX(frame) - zoomed_width;
 | |
|   else
 | |
|     frame.origin.x = window_frame.origin.x;
 | |
| 
 | |
|   // Set the width. Don't touch y or height.
 | |
|   frame.size.width = zoomed_width;
 | |
| 
 | |
|   if (shell_->aspect_ratio() > 0.0)
 | |
|     shell_->set_default_frame_for_zoom(frame);
 | |
| 
 | |
|   return frame;
 | |
| }
 | |
| 
 | |
| - (void)windowDidBecomeMain:(NSNotification*)notification {
 | |
|   shell_->NotifyWindowFocus();
 | |
|   shell_->RedrawTrafficLights();
 | |
| }
 | |
| 
 | |
| - (void)windowDidResignMain:(NSNotification*)notification {
 | |
|   shell_->NotifyWindowBlur();
 | |
|   shell_->RedrawTrafficLights();
 | |
| }
 | |
| 
 | |
| - (void)windowDidBecomeKey:(NSNotification*)notification {
 | |
|   shell_->NotifyWindowIsKeyChanged(true);
 | |
|   shell_->RedrawTrafficLights();
 | |
| }
 | |
| 
 | |
| - (void)windowDidResignKey:(NSNotification*)notification {
 | |
|   // If our app is still active and we're still the key window, ignore this
 | |
|   // message, since it just means that a menu extra (on the "system status bar")
 | |
|   // was activated; we'll get another |-windowDidResignKey| if we ever really
 | |
|   // lose key window status.
 | |
|   if ([NSApp isActive] && ([NSApp keyWindow] == [notification object]))
 | |
|     return;
 | |
| 
 | |
|   shell_->NotifyWindowIsKeyChanged(false);
 | |
|   shell_->RedrawTrafficLights();
 | |
| }
 | |
| 
 | |
| - (NSSize)windowWillResize:(NSWindow*)sender toSize:(NSSize)frameSize {
 | |
|   NSSize newSize = frameSize;
 | |
|   NSWindow* window = shell_->GetNativeWindow().GetNativeNSWindow();
 | |
| 
 | |
|   if (const double aspectRatio = shell_->aspect_ratio(); aspectRatio > 0.0) {
 | |
|     const gfx::Size windowSize = shell_->GetSize();
 | |
|     const gfx::Size contentSize = shell_->GetContentSize();
 | |
|     const gfx::Size extraSize = shell_->aspect_ratio_extra_size();
 | |
| 
 | |
|     double titleBarHeight = windowSize.height() - contentSize.height();
 | |
|     double extraWidthPlusFrame =
 | |
|         windowSize.width() - contentSize.width() + extraSize.width();
 | |
|     double extraHeightPlusFrame = titleBarHeight + extraSize.height();
 | |
| 
 | |
|     newSize.width =
 | |
|         roundf((frameSize.height - extraHeightPlusFrame) * aspectRatio +
 | |
|                extraWidthPlusFrame);
 | |
|     newSize.height =
 | |
|         roundf((newSize.width - extraWidthPlusFrame) / aspectRatio +
 | |
|                extraHeightPlusFrame);
 | |
| 
 | |
|     // Clamp to minimum width/height while ensuring aspect ratio remains.
 | |
|     NSSize minSize = [window minSize];
 | |
|     NSSize zeroSize =
 | |
|         shell_->has_frame() ? NSMakeSize(0, titleBarHeight) : NSZeroSize;
 | |
|     if (!NSEqualSizes(minSize, zeroSize)) {
 | |
|       double minWidthForAspectRatio =
 | |
|           (minSize.height - titleBarHeight) * aspectRatio;
 | |
|       bool atMinHeight =
 | |
|           minSize.height > zeroSize.height && newSize.height <= minSize.height;
 | |
|       newSize.width = atMinHeight ? minWidthForAspectRatio
 | |
|                                   : std::max(newSize.width, minSize.width);
 | |
| 
 | |
|       double minHeightForAspectRatio = minSize.width / aspectRatio;
 | |
|       bool atMinWidth =
 | |
|           minSize.width > zeroSize.width && newSize.width <= minSize.width;
 | |
|       newSize.height = atMinWidth ? minHeightForAspectRatio
 | |
|                                   : std::max(newSize.height, minSize.height);
 | |
|     }
 | |
| 
 | |
|     // Clamp to maximum width/height while ensuring aspect ratio remains.
 | |
|     NSSize maxSize = [window maxSize];
 | |
|     if (!NSEqualSizes(maxSize, NSMakeSize(FLT_MAX, FLT_MAX))) {
 | |
|       double maxWidthForAspectRatio = maxSize.height * aspectRatio;
 | |
|       bool atMaxHeight =
 | |
|           maxSize.height < FLT_MAX && newSize.height >= maxSize.height;
 | |
|       newSize.width = atMaxHeight ? maxWidthForAspectRatio
 | |
|                                   : std::min(newSize.width, maxSize.width);
 | |
| 
 | |
|       double maxHeightForAspectRatio = maxSize.width / aspectRatio;
 | |
|       bool atMaxWidth =
 | |
|           maxSize.width < FLT_MAX && newSize.width >= maxSize.width;
 | |
|       newSize.height = atMaxWidth ? maxHeightForAspectRatio
 | |
|                                   : std::min(newSize.height, maxSize.height);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!resizingHorizontally_) {
 | |
|     const auto widthDelta = frameSize.width - [window frame].size.width;
 | |
|     const auto heightDelta = frameSize.height - [window frame].size.height;
 | |
|     resizingHorizontally_ = std::abs(widthDelta) > std::abs(heightDelta);
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     bool prevent_default = false;
 | |
|     NSRect new_bounds = NSMakeRect(sender.frame.origin.x, sender.frame.origin.y,
 | |
|                                    newSize.width, newSize.height);
 | |
|     shell_->NotifyWindowWillResize(gfx::ScreenRectFromNSRect(new_bounds),
 | |
|                                    *resizingHorizontally_
 | |
|                                        ? gfx::ResizeEdge::kRight
 | |
|                                        : gfx::ResizeEdge::kBottom,
 | |
|                                    &prevent_default);
 | |
|     if (prevent_default) {
 | |
|       return sender.frame.size;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return newSize;
 | |
| }
 | |
| 
 | |
| - (void)windowDidResize:(NSNotification*)notification {
 | |
|   [super windowDidResize:notification];
 | |
|   shell_->NotifyWindowResize();
 | |
|   shell_->RedrawTrafficLights();
 | |
| }
 | |
| 
 | |
| - (void)windowWillMove:(NSNotification*)notification {
 | |
|   NSWindow* window = [notification object];
 | |
|   NSSize size = [[window contentView] frame].size;
 | |
|   NSRect new_bounds = NSMakeRect(window.frame.origin.x, window.frame.origin.y,
 | |
|                                  size.width, size.height);
 | |
|   bool prevent_default = false;
 | |
| 
 | |
|   // prevent_default has no effect
 | |
|   shell_->NotifyWindowWillMove(gfx::ScreenRectFromNSRect(new_bounds),
 | |
|                                &prevent_default);
 | |
| }
 | |
| 
 | |
| - (void)windowDidMove:(NSNotification*)notification {
 | |
|   [super windowDidMove:notification];
 | |
|   // TODO(zcbenz): Remove the alias after figuring out a proper
 | |
|   // way to dispatch move.
 | |
|   shell_->NotifyWindowMove();
 | |
|   shell_->NotifyWindowMoved();
 | |
| }
 | |
| 
 | |
| - (void)windowWillMiniaturize:(NSNotification*)notification {
 | |
|   NSWindow* window = shell_->GetNativeWindow().GetNativeNSWindow();
 | |
|   // store the current status window level to be restored in
 | |
|   // windowDidDeminiaturize
 | |
|   level_ = [window level];
 | |
|   shell_->SetWindowLevel(NSNormalWindowLevel);
 | |
|   shell_->UpdateWindowOriginalFrame();
 | |
|   shell_->DetachChildren();
 | |
| }
 | |
| 
 | |
| - (void)windowDidMiniaturize:(NSNotification*)notification {
 | |
|   [super windowDidMiniaturize:notification];
 | |
|   is_minimized_ = true;
 | |
| 
 | |
|   shell_->set_wants_to_be_visible(false);
 | |
|   shell_->NotifyWindowMinimize();
 | |
| }
 | |
| 
 | |
| - (void)windowDidDeminiaturize:(NSNotification*)notification {
 | |
|   [super windowDidDeminiaturize:notification];
 | |
|   is_minimized_ = false;
 | |
| 
 | |
|   shell_->set_wants_to_be_visible(true);
 | |
|   shell_->AttachChildren();
 | |
|   shell_->SetWindowLevel(level_);
 | |
|   shell_->NotifyWindowRestore();
 | |
| }
 | |
| 
 | |
| - (BOOL)windowShouldZoom:(NSWindow*)window toFrame:(NSRect)newFrame {
 | |
|   is_zooming_ = true;
 | |
|   return YES;
 | |
| }
 | |
| 
 | |
| - (void)windowDidEndLiveResize:(NSNotification*)notification {
 | |
|   resizingHorizontally_.reset();
 | |
|   shell_->NotifyWindowResized();
 | |
|   if (is_zooming_) {
 | |
|     if (shell_->IsMaximized())
 | |
|       shell_->NotifyWindowMaximize();
 | |
|     else
 | |
|       shell_->NotifyWindowUnmaximize();
 | |
|     is_zooming_ = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| - (void)windowWillEnterFullScreen:(NSNotification*)notification {
 | |
|   // Store resizable mask so it can be restored after exiting fullscreen.
 | |
|   is_resizable_ = shell_->HasStyleMask(NSWindowStyleMaskResizable);
 | |
| 
 | |
|   shell_->set_fullscreen_transition_state(FullScreenTransitionState::kEntering);
 | |
| 
 | |
|   shell_->NotifyWindowWillEnterFullScreen();
 | |
| 
 | |
|   // Set resizable to true before entering fullscreen.
 | |
|   shell_->SetResizable(true);
 | |
| }
 | |
| 
 | |
| - (void)windowDidEnterFullScreen:(NSNotification*)notification {
 | |
|   shell_->set_fullscreen_transition_state(FullScreenTransitionState::kNone);
 | |
| 
 | |
|   shell_->NotifyWindowEnterFullScreen();
 | |
| 
 | |
|   if (shell_->HandleDeferredClose())
 | |
|     return;
 | |
| 
 | |
|   shell_->HandlePendingFullscreenTransitions();
 | |
| }
 | |
| 
 | |
| - (void)windowWillExitFullScreen:(NSNotification*)notification {
 | |
|   shell_->set_fullscreen_transition_state(FullScreenTransitionState::kExiting);
 | |
| 
 | |
|   shell_->NotifyWindowWillLeaveFullScreen();
 | |
| }
 | |
| 
 | |
| - (void)windowDidExitFullScreen:(NSNotification*)notification {
 | |
|   shell_->set_fullscreen_transition_state(FullScreenTransitionState::kNone);
 | |
| 
 | |
|   shell_->SetResizable(is_resizable_);
 | |
|   shell_->NotifyWindowLeaveFullScreen();
 | |
| 
 | |
|   if (shell_->HandleDeferredClose())
 | |
|     return;
 | |
| 
 | |
|   shell_->HandlePendingFullscreenTransitions();
 | |
| }
 | |
| 
 | |
| - (void)windowWillClose:(NSNotification*)notification {
 | |
|   shell_->Cleanup();
 | |
|   shell_->NotifyWindowClosed();
 | |
| 
 | |
|   // Something called -[NSWindow close] on a sheet rather than calling
 | |
|   // -[NSWindow endSheet:] on its parent. If the modal session is not ended
 | |
|   // then the parent will never be able to show another sheet. But calling
 | |
|   // -endSheet: here will block the thread with an animation, so post a task.
 | |
|   if (shell_->is_modal() && shell_->parent() && shell_->IsVisible()) {
 | |
|     NSWindow* window = shell_->GetNativeWindow().GetNativeNSWindow();
 | |
|     NSWindow* sheetParent = [window sheetParent];
 | |
|     base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
 | |
|         FROM_HERE, base::BindOnce(^{
 | |
|           [sheetParent endSheet:window];
 | |
|         }));
 | |
|   }
 | |
| 
 | |
|   // Clears the delegate when window is going to be closed, since EL Capitan it
 | |
|   // is possible that the methods of delegate would get called after the window
 | |
|   // has been closed.
 | |
|   auto* bridge_host = views::NativeWidgetMacNSWindowHost::GetFromNativeWindow(
 | |
|       shell_->GetNativeWindow());
 | |
|   auto* bridged_view = bridge_host->GetInProcessNSWindowBridge();
 | |
|   bridged_view->OnWindowWillClose();
 | |
| }
 | |
| 
 | |
| - (BOOL)windowShouldClose:(id)window {
 | |
|   shell_->NotifyWindowCloseButtonClicked();
 | |
|   return NO;
 | |
| }
 | |
| 
 | |
| - (NSRect)window:(NSWindow*)window
 | |
|     willPositionSheet:(NSWindow*)sheet
 | |
|             usingRect:(NSRect)rect {
 | |
|   NSView* view = window.contentView;
 | |
| 
 | |
|   rect.origin.x = shell_->GetSheetOffsetX();
 | |
|   rect.origin.y = view.frame.size.height - shell_->GetSheetOffsetY();
 | |
|   return rect;
 | |
| }
 | |
| 
 | |
| - (void)windowWillBeginSheet:(NSNotification*)notification {
 | |
|   shell_->NotifyWindowSheetBegin();
 | |
| }
 | |
| 
 | |
| - (void)windowDidEndSheet:(NSNotification*)notification {
 | |
|   shell_->NotifyWindowSheetEnd();
 | |
| }
 | |
| 
 | |
| - (IBAction)newWindowForTab:(id)sender {
 | |
|   shell_->NotifyNewWindowForTab();
 | |
|   electron::Browser::Get()->NewWindowForTab();
 | |
| }
 | |
| 
 | |
| #pragma mark - NSTouchBarDelegate
 | |
| 
 | |
| - (NSTouchBarItem*)touchBar:(NSTouchBar*)touchBar
 | |
|       makeItemForIdentifier:(NSTouchBarItemIdentifier)identifier {
 | |
|   if (touchBar && shell_->touch_bar())
 | |
|     return [shell_->touch_bar() makeItemForIdentifier:identifier];
 | |
|   else
 | |
|     return nil;
 | |
| }
 | |
| 
 | |
| #pragma mark - QLPreviewPanelDataSource
 | |
| 
 | |
| - (NSInteger)numberOfPreviewItemsInPreviewPanel:(QLPreviewPanel*)panel {
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| - (id<QLPreviewItem>)previewPanel:(QLPreviewPanel*)panel
 | |
|                previewItemAtIndex:(NSInteger)index {
 | |
|   return shell_->preview_item();
 | |
| }
 | |
| 
 | |
| @end
 |