1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195
|
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/browser/renderer_host/popup_menu_helper_mac.h"
#import "base/mac/scoped_sending_event.h"
#import "base/message_loop/message_pump_apple.h"
#include "base/task/current_thread.h"
#import "components/remote_cocoa/app_shim/native_widget_mac_nswindow.h"
#import "content/app_shim_remote_cocoa/render_widget_host_view_cocoa.h"
#include "content/browser/permissions/permission_controller_impl.h"
#include "content/browser/renderer_host/frame_tree.h"
#include "content/browser/renderer_host/frame_tree_node.h"
#include "content/browser/renderer_host/render_frame_host_impl.h"
#include "content/browser/renderer_host/render_view_host_impl.h"
#include "content/browser/renderer_host/render_widget_host_view_mac.h"
#include "content/browser/renderer_host/web_menu_runner_mac.h"
#include "content/public/browser/web_contents.h"
#import "ui/base/cocoa/base_view.h"
namespace content {
namespace {
bool g_allow_showing_popup_menus = true;
} // namespace
struct PopupMenuHelper::ObjCStorage {
WebMenuRunner* __weak menu_runner;
};
PopupMenuHelper::PopupMenuHelper(
Delegate* delegate,
RenderFrameHost* render_frame_host,
mojo::PendingRemote<blink::mojom::PopupMenuClient> popup_client)
: delegate_(delegate),
render_frame_host_(
static_cast<RenderFrameHostImpl*>(render_frame_host)->GetWeakPtr()),
popup_client_(std::move(popup_client)),
objc_storage_(std::make_unique<ObjCStorage>()) {
RenderWidgetHost* widget_host =
render_frame_host->GetRenderViewHost()->GetWidget();
observation_.Observe(widget_host);
popup_client_.set_disconnect_handler(
base::BindOnce(&PopupMenuHelper::Hide, weak_ptr_factory_.GetWeakPtr()));
}
PopupMenuHelper::~PopupMenuHelper() {
Hide();
}
void PopupMenuHelper::ShowPopupMenu(
const gfx::Rect& bounds,
int item_height,
double item_font_size,
int selected_item,
std::vector<blink::mojom::MenuItemPtr> items,
bool right_aligned,
bool allow_multiple_selection) {
// Only single selection list boxes show a popup on Mac.
DCHECK(!allow_multiple_selection);
if (!g_allow_showing_popup_menus)
return;
RenderWidgetHostViewMac* rwhvm = GetRenderWidgetHostView();
auto* web_contents = rwhvm->GetWebContents();
// Convert element_bounds to be in screen.
gfx::Rect client_area = web_contents->GetContainerBounds();
gfx::Rect bounds_in_screen = bounds + client_area.OffsetFromOrigin();
// The new popup menu would overlap the permission prompt, which could lead to
// users making decisions based on incorrect information. We should close the
// popup if it intersects with the permission prompt.
auto permission_exclusion_area_bounds =
PermissionControllerImpl::FromBrowserContext(
web_contents->GetBrowserContext())
->GetExclusionAreaBoundsInScreen(web_contents);
if (permission_exclusion_area_bounds &&
permission_exclusion_area_bounds->Intersects(bounds_in_screen)) {
popup_client_->DidCancel();
delegate_->OnMenuClosed(); // May delete |this|.
return;
}
// Retain the Cocoa view for the duration of the pop-up so that it can't be
// dealloced if my Destroy() method is called while the pop-up's up (which
// would in turn delete me, causing a crash once the -runMenuInView
// call returns. That's what was happening in <http://crbug.com/33250>).
RenderWidgetHostViewCocoa* cocoa_view = rwhvm->GetInProcessNSView();
// Check if the underlying native window is headless and if so, return early
// to avoid showing the popup menu.
NativeWidgetMacNSWindow* ns_window =
base::apple::ObjCCastStrict<NativeWidgetMacNSWindow>([cocoa_view window]);
if (ns_window && [ns_window isHeadless]) {
return;
}
// Display the menu.
WebMenuRunner* runner = [[WebMenuRunner alloc] initWithItems:items
fontSize:item_font_size
rightAligned:right_aligned];
// Take a weak reference so that Hide() can close the menu.
objc_storage_->menu_runner = runner;
base::WeakPtr<PopupMenuHelper> weak_ptr(weak_ptr_factory_.GetWeakPtr());
{
// Make sure events can be pumped while the menu is up.
base::CurrentThread::ScopedAllowApplicationTasksInNativeNestedLoop allow;
// One of the events that could be pumped is |window.close()|.
// User-initiated event-tracking loops protect against this by
// setting flags in -[CrApplication sendEvent:], but since
// web-content menus are initiated by IPC message the setup has to
// be done manually.
base::mac::ScopedSendingEvent sending_event_scoper;
// Ensure the UI can update while the menu is fading out.
pump_in_fade_ = std::make_unique<base::ScopedPumpMessagesInPrivateModes>();
// Now run a NESTED EVENT LOOP until the pop-up is finished.
[runner runMenuInView:cocoa_view
withBounds:[cocoa_view flipRectToNSRect:bounds]
initialIndex:selected_item];
}
if (!weak_ptr)
return; // Handle |this| being deleted.
pump_in_fade_ = nullptr;
objc_storage_->menu_runner = nil;
// The RenderFrameHost may be deleted while running the menu, or it may have
// requested the close. Don't notify in these cases.
if (popup_client_ && !popup_was_hidden_) {
if ([runner menuItemWasChosen]) {
int index = [runner indexOfSelectedItem];
if (index < 0)
popup_client_->DidCancel();
else
popup_client_->DidAcceptIndices({index});
} else {
popup_client_->DidCancel();
}
}
delegate_->OnMenuClosed(); // May delete |this|.
}
void PopupMenuHelper::Hide() {
// Blink core reuses the PopupMenu of an element and first invokes Hide() over
// IPC if a menu is already showing. Attempting to show a new menu while the
// old menu is fading out confuses AppKit, since we're still in the NESTED
// EVENT LOOP of ShowPopupMenu(). Disable pumping of events in the fade
// animation of the old menu in this case so that it closes synchronously.
// See http://crbug.com/812260.
pump_in_fade_ = nullptr;
if (objc_storage_->menu_runner) {
[objc_storage_->menu_runner hide];
}
popup_was_hidden_ = true;
popup_client_.reset();
}
// static
void PopupMenuHelper::DontShowPopupMenuForTesting() {
g_allow_showing_popup_menus = false;
}
RenderWidgetHostViewMac* PopupMenuHelper::GetRenderWidgetHostView() const {
return static_cast<RenderWidgetHostViewMac*>(
render_frame_host_->GetOutermostMainFrameOrEmbedder()->GetView());
}
void PopupMenuHelper::RenderWidgetHostVisibilityChanged(
RenderWidgetHost* widget_host,
bool became_visible) {
if (!became_visible)
Hide();
}
void PopupMenuHelper::RenderWidgetHostDestroyed(RenderWidgetHost* widget_host) {
DCHECK(observation_.IsObservingSource(widget_host));
observation_.Reset();
}
} // namespace content
|