// 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 "atom/browser/api/atom_api_browser_window.h" #include #include #import #include "atom/browser/native_browser_view.h" #include "atom/browser/native_window_mac.h" #include "atom/browser/ui/inspectable_web_contents_view.h" #include "base/mac/scoped_nsobject.h" @interface NSView (WebContentsView) - (void)setMouseDownCanMoveWindow:(BOOL)can_move; @end @interface ControlRegionView : NSView @end @implementation ControlRegionView - (BOOL)mouseDownCanMoveWindow { return NO; } - (NSView*)hitTest:(NSPoint)aPoint { return nil; } @end namespace atom { namespace api { namespace { // Return a vector of non-draggable regions that fill a window of size // |width| by |height|, but leave gaps where the window should be draggable. std::vector CalculateNonDraggableRegions( std::unique_ptr draggable, int width, int height) { std::vector result; SkRegion non_draggable; non_draggable.op(0, 0, width, height, SkRegion::kUnion_Op); non_draggable.op(*draggable, SkRegion::kDifference_Op); for (SkRegion::Iterator it(non_draggable); !it.done(); it.next()) { result.push_back(gfx::SkIRectToRect(it.rect())); } return result; } } // namespace void BrowserWindow::OverrideNSWindowContentView(InspectableWebContents* iwc) { // Make NativeWindow use a NSView as content view. static_cast(window())->OverrideNSWindowContentView(); // Add webview to contentView. NSView* webView = iwc->GetView()->GetNativeView().GetNativeNSView(); NSView* contentView = [window()->GetNativeWindow().GetNativeNSWindow() contentView]; [webView setFrame:[contentView bounds]]; // ensure that buttons view is floated to top of view hierarchy NSArray* subviews = [contentView subviews]; NSView* last = subviews.lastObject; [contentView addSubview:webView positioned:NSWindowBelow relativeTo:last]; [contentView viewDidMoveToWindow]; } void BrowserWindow::UpdateDraggableRegions( const std::vector& regions) { if (window_->has_frame()) return; // All ControlRegionViews should be added as children of the WebContentsView, // because WebContentsView will be removed and re-added when entering and // leaving fullscreen mode. NSView* webView = web_contents()->GetNativeView().GetNativeNSView(); NSInteger webViewWidth = NSWidth([webView bounds]); NSInteger webViewHeight = NSHeight([webView bounds]); if ([webView respondsToSelector:@selector(setMouseDownCanMoveWindow:)]) { [webView setMouseDownCanMoveWindow:YES]; } // Remove all ControlRegionViews that are added last time. // Note that [webView subviews] returns the view's mutable internal array and // it should be copied to avoid mutating the original array while enumerating // it. base::scoped_nsobject subviews([[webView subviews] copy]); for (NSView* subview in subviews.get()) if ([subview isKindOfClass:[ControlRegionView class]]) [subview removeFromSuperview]; // Draggable regions is implemented by having the whole web view draggable // (mouseDownCanMoveWindow) and overlaying regions that are not draggable. draggable_regions_.clear(); for (const auto& r : regions) draggable_regions_.push_back(r.Clone()); std::vector drag_exclude_rects; if (regions.empty()) { drag_exclude_rects.push_back(gfx::Rect(0, 0, webViewWidth, webViewHeight)); } else { drag_exclude_rects = CalculateNonDraggableRegions( DraggableRegionsToSkRegion(regions), webViewWidth, webViewHeight); } auto browser_views = window_->browser_views(); for (NativeBrowserView* view : browser_views) { view->UpdateDraggableRegions(drag_exclude_rects); } // Create and add a ControlRegionView for each region that needs to be // excluded from the dragging. for (const auto& rect : drag_exclude_rects) { base::scoped_nsobject controlRegion( [[ControlRegionView alloc] initWithFrame:NSZeroRect]); [controlRegion setFrame:NSMakeRect(rect.x(), webViewHeight - rect.bottom(), rect.width(), rect.height())]; [webView addSubview:controlRegion]; } // AppKit will not update its cache of mouseDownCanMoveWindow unless something // changes. Previously we tried adding an NSView and removing it, but for some // reason it required reposting the mouse-down event, and didn't always work. // Calling the below seems to be an effective solution. [[webView window] setMovableByWindowBackground:NO]; [[webView window] setMovableByWindowBackground:YES]; } } // namespace api } // namespace atom