Cocoa: Make sure main window gets focus after closing modal dialog

As explained in the code, this is due to our event dispatcher. This could
also make applications crash due to the way we bind windows to the menubar
in the Cocoa QPA plugin.

Task-number: QTBUG-30451
Change-Id: I6dd190d5b3b83f1216caec0b14414efa771634a5
Reviewed-by: Shawn Rutledge <shawn.rutledge@digia.com>
This commit is contained in:
Gabriel de Dietrich 2013-05-15 14:47:27 +02:00 committed by The Qt Project
parent ef8ece3f93
commit 27f68818c8

View File

@ -118,7 +118,7 @@ static bool isMouseEvent(NSEvent *ev)
// Windows with a transient parent (such as combobox popup windows) // Windows with a transient parent (such as combobox popup windows)
// cannot become the main window: // cannot become the main window:
if (m_cocoaPlatformWindow->window()->transientParent()) if (!m_cocoaPlatformWindow || m_cocoaPlatformWindow->window()->transientParent())
canBecomeMain = NO; canBecomeMain = NO;
return canBecomeMain; return canBecomeMain;
@ -156,7 +156,8 @@ static bool isMouseEvent(NSEvent *ev)
- (BOOL)canBecomeKeyWindow - (BOOL)canBecomeKeyWindow
{ {
// Only tool or dialog windows should become key: // Only tool or dialog windows should become key:
if (m_cocoaPlatformWindow->window()->type() == Qt::Tool || m_cocoaPlatformWindow->window()->type() == Qt::Dialog) if (m_cocoaPlatformWindow
&& (m_cocoaPlatformWindow->window()->type() == Qt::Tool || m_cocoaPlatformWindow->window()->type() == Qt::Dialog))
return YES; return YES;
return NO; return NO;
} }
@ -362,11 +363,22 @@ void QCocoaWindow::setVisible(bool visible)
QCocoaEventDispatcherPrivate *cocoaEventDispatcherPrivate = static_cast<QCocoaEventDispatcherPrivate *>(QObjectPrivate::get(cocoaEventDispatcher)); QCocoaEventDispatcherPrivate *cocoaEventDispatcherPrivate = static_cast<QCocoaEventDispatcherPrivate *>(QObjectPrivate::get(cocoaEventDispatcher));
cocoaEventDispatcherPrivate->endModalSession(window()); cocoaEventDispatcherPrivate->endModalSession(window());
m_hasModalSession = false; m_hasModalSession = false;
[m_nsWindow orderOut:m_nsWindow];
if (m_nsWindow == [NSApp keyWindow] && !cocoaEventDispatcherPrivate->currentModalSession()) {
// Probably because we call runModalSession: outside [NSApp run] in QCocoaEventDispatcher
// (e.g., when show()-ing a modal QDialog instead of exec()-ing it), it can happen that
// the current NSWindow is still key after being ordered out. Then, after checking we
// don't have any other modal session left, it's safe to make the main window key again.
NSWindow *mainWindow = [NSApp mainWindow];
if (mainWindow && [mainWindow canBecomeKeyWindow])
[mainWindow makeKeyWindow];
}
} else { } else {
if ([m_nsWindow isSheet]) if ([m_nsWindow isSheet])
[NSApp endSheet:m_nsWindow]; [NSApp endSheet:m_nsWindow];
}
[m_nsWindow orderOut:m_nsWindow]; [m_nsWindow orderOut:m_nsWindow];
}
} else { } else {
[m_contentView setHidden:YES]; [m_contentView setHidden:YES];
} }