summaryrefslogtreecommitdiffstats
path: root/chrome
diff options
context:
space:
mode:
authoravi@chromium.org <avi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-07-30 16:24:35 +0000
committeravi@chromium.org <avi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-07-30 16:24:35 +0000
commit5ba0cd1cb49ac8e94f2ad9e836a440e640663f52 (patch)
tree95ed9c11b626829f1cdae4247d705f743011adce /chrome
parentc8e2f18ee89786c6a626e181a7b98d7e73e0f378 (diff)
downloadchromium_src-5ba0cd1cb49ac8e94f2ad9e836a440e640663f52.zip
chromium_src-5ba0cd1cb49ac8e94f2ad9e836a440e640663f52.tar.gz
chromium_src-5ba0cd1cb49ac8e94f2ad9e836a440e640663f52.tar.bz2
Properly destroy popups when their parent views are told to destroy.
BUG=http://crbug.com/12970 TEST=as in bug Review URL: http://codereview.chromium.org/160353 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@22058 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome')
-rw-r--r--chrome/browser/renderer_host/render_widget_host_view_mac.mm53
1 files changed, 38 insertions, 15 deletions
diff --git a/chrome/browser/renderer_host/render_widget_host_view_mac.mm b/chrome/browser/renderer_host/render_widget_host_view_mac.mm
index 21345e3..6a8c60b 100644
--- a/chrome/browser/renderer_host/render_widget_host_view_mac.mm
+++ b/chrome/browser/renderer_host/render_widget_host_view_mac.mm
@@ -23,6 +23,7 @@ using WebKit::WebMouseWheelEvent;
@interface RenderWidgetHostViewCocoa (Private)
- (id)initWithRenderWidgetHostViewMac:(RenderWidgetHostViewMac*)r;
+- (void)cancelChildPopups;
@end
namespace {
@@ -250,6 +251,20 @@ void RenderWidgetHostViewMac::RenderViewGone() {
}
void RenderWidgetHostViewMac::Destroy() {
+ // On Windows, popups are implemented with a popup window style, so that when
+ // an event comes in that would "cancel" it, it receives the OnCancelMode
+ // message and can kill itself. Alas, on the Mac, views cannot capture events
+ // outside of themselves. On Windows, if Destroy is being called on a view,
+ // then the event causing the destroy had also cancelled any popups by the
+ // time Destroy() was called. On the Mac we have to destroy all the popups
+ // ourselves.
+
+ // Depth-first destroy all popups. Use ShutdownHost() to enforce deepest-first
+ // ordering.
+ for (RenderWidgetHostViewCocoa* subview in [cocoa_view_ subviews]) {
+ [subview renderWidgetHostViewMac]->ShutdownHost();
+ }
+
// We've been told to destroy.
[cocoa_view_ retain];
[cocoa_view_ removeFromSuperview];
@@ -423,6 +438,12 @@ void RenderWidgetHostViewMac::SetActive(bool active) {
}
- (void)mouseEvent:(NSEvent *)theEvent {
+ // Don't cancel child popups; killing them on a mouse click would prevent the
+ // user from positioning the insertion point in the text field spawning the
+ // popup. A click outside the text field would cause the text field to drop
+ // the focus, and then EditorClientImpl::textFieldDidEndEditing() would cancel
+ // the popup anyway, so we're OK.
+
const WebMouseEvent& event =
WebInputEventFactory::mouseEvent(theEvent, self);
if (renderWidgetHostView_->render_widget_host_)
@@ -433,27 +454,16 @@ void RenderWidgetHostViewMac::SetActive(bool active) {
// TODO(avi): Possibly kill self? See RenderWidgetHostViewWin::OnKeyEvent and
// http://b/issue?id=1192881 .
+ // Don't cancel child popups; the key events are probably what's triggering
+ // the popup in the first place.
+
NativeWebKeyboardEvent event(theEvent);
if (renderWidgetHostView_->render_widget_host_)
renderWidgetHostView_->render_widget_host_->ForwardKeyboardEvent(event);
}
- (void)scrollWheel:(NSEvent *)theEvent {
- // On Windows, popups are implemented with a popup window style, so that when
- // an event comes in that would "cancel" it, it receives the OnCancelMode
- // message and can kill itself. Alas, on the Mac, views cannot capture events
- // outside of themselves. While a click outside a popup kills it, that's just
- // a defocusing of the text field in WebCore and our editor client kills the
- // popup. But a scroll wheel event outside a popup must kill it.
- //
- // Thus this lovely case of filicide. If this view can be the key view, it is
- // not a popup. Therefore, if it has any children, they are popups that need
- // to be canceled.
- if (canBeKeyView_) {
- for (NSView* subview in [self subviews]) {
- ((RenderWidgetHostViewCocoa*)subview)->renderWidgetHostView_->KillSelf();
- }
- }
+ [self cancelChildPopups];
const WebMouseWheelEvent& event =
WebInputEventFactory::mouseWheelEvent(theEvent, self);
@@ -461,6 +471,19 @@ void RenderWidgetHostViewMac::SetActive(bool active) {
renderWidgetHostView_->render_widget_host_->ForwardWheelEvent(event);
}
+// See the comment in RenderWidgetHostViewMac::Destroy() about cancellation
+// events. On the Mac we must kill popups on outside events, thus this lovely
+// case of filicide caused by events on parent views.
+- (void)cancelChildPopups {
+ // If this view can be the key view, it is not a popup. Therefore, if it has
+ // any children, they are popups that need to be canceled.
+ if (canBeKeyView_) {
+ for (RenderWidgetHostViewCocoa* subview in [self subviews]) {
+ subview->renderWidgetHostView_->KillSelf();
+ }
+ }
+}
+
- (void)setFrame:(NSRect)frameRect {
[super setFrame:frameRect];
if (renderWidgetHostView_->render_widget_host_)