Close widgets properly from session management.

Introduce new virtual QGuiApplicationPrivate::tryCloseAllWindows()
which allows overriding the behavior in QApplication to properly close
the widgets first.

Without this, QGuiApplication closes the widget windows leaving a stale
window handle behind in the associated QWidget which then causes the
application not to terminate since QApplication::shouldQuit() stills
finds the affected widgets to be visible.

Task-number: QTBUG-35986

Change-Id: I19ac4b5a19250ee68d09e461c03dbace458c98e4
Reviewed-by: David Faure <david.faure@kdab.com>
This commit is contained in:
Friedemann Kleint 2014-02-11 13:00:59 +01:00 committed by The Qt Project
parent ff11af4fbc
commit 9835a63dde
4 changed files with 64 additions and 35 deletions

View File

@ -2696,6 +2696,27 @@ bool QGuiApplicationPrivate::shouldQuitInternal(const QWindowList &processedWind
return true;
}
bool QGuiApplicationPrivate::tryCloseAllWindows()
{
return tryCloseRemainingWindows(QWindowList());
}
bool QGuiApplicationPrivate::tryCloseRemainingWindows(QWindowList processedWindows)
{
QWindowList list = QGuiApplication::topLevelWindows();
for (int i = 0; i < list.size(); ++i) {
QWindow *w = list.at(i);
if (w->isVisible() && !processedWindows.contains(w)) {
if (!w->close())
return false;
processedWindows.append(w);
list = QGuiApplication::topLevelWindows();
i = -1;
}
}
return true;
}
/*!
\since 5.2
\fn Qt::ApplicationState QGuiApplication::applicationState()
@ -2902,23 +2923,8 @@ void QGuiApplicationPrivate::commitData()
Q_Q(QGuiApplication);
is_saving_session = true;
emit q->commitDataRequest(*session_manager);
if (session_manager->allowsInteraction()) {
QWindowList done;
QWindowList list = QGuiApplication::topLevelWindows();
bool cancelled = false;
for (int i = 0; !cancelled && i < list.size(); ++i) {
QWindow* w = list.at(i);
if (w->isVisible() && !done.contains(w)) {
cancelled = !w->close();
if (!cancelled)
done.append(w);
list = QGuiApplication::topLevelWindows();
i = -1;
}
}
if (cancelled)
session_manager->cancel();
}
if (session_manager->allowsInteraction() && !tryCloseAllWindows())
session_manager->cancel();
is_saving_session = false;
}

View File

@ -92,6 +92,7 @@ public:
virtual bool shouldQuit();
bool shouldQuitInternal(const QWindowList &processedWindows);
virtual bool tryCloseAllWindows();
static Qt::KeyboardModifiers modifier_buttons;
static Qt::MouseButtons mouse_buttons;
@ -291,6 +292,7 @@ public:
protected:
virtual void notifyThemeChanged();
bool tryCloseRemainingWindows(QWindowList processedWindows);
#ifndef QT_NO_DRAGANDDROP
virtual void notifyDragStarted(const QDrag *);
#endif // QT_NO_DRAGANDDROP

View File

@ -1733,6 +1733,41 @@ QFontMetrics QApplication::fontMetrics()
return desktop()->fontMetrics();
}
bool QApplicationPrivate::tryCloseAllWidgetWindows(QWindowList *processedWindows)
{
Q_ASSERT(processedWindows);
while (QWidget *w = QApplication::activeModalWidget()) {
if (!w->isVisible() || w->data->is_closing)
break;
QWindow *window = w->windowHandle();
if (!w->close()) // Qt::WA_DeleteOnClose may cause deletion.
return false;
if (window)
processedWindows->append(window);
}
QWidgetList list = QApplication::topLevelWidgets();
for (int i = 0; i < list.size(); ++i) {
QWidget *w = list.at(i);
if (w->isVisible() && w->windowType() != Qt::Desktop && !w->data->is_closing) {
QWindow *window = w->windowHandle();
if (!w->close()) // Qt::WA_DeleteOnClose may cause deletion.
return false;
if (window)
processedWindows->append(window);
list = QApplication::topLevelWidgets();
i = -1;
}
}
return true;
}
bool QApplicationPrivate::tryCloseAllWindows()
{
QWindowList processedWindows;
return QApplicationPrivate::tryCloseAllWidgetWindows(&processedWindows)
&& QGuiApplicationPrivate::tryCloseRemainingWindows(processedWindows);
}
/*!
Closes all top-level windows.
@ -1754,24 +1789,8 @@ QFontMetrics QApplication::fontMetrics()
*/
void QApplication::closeAllWindows()
{
bool did_close = true;
QWidget *w;
while ((w = activeModalWidget()) && did_close) {
if (!w->isVisible() || w->data->is_closing)
break;
did_close = w->close();
}
QWidgetList list = QApplication::topLevelWidgets();
for (int i = 0; did_close && i < list.size(); ++i) {
w = list.at(i);
if (w->isVisible()
&& w->windowType() != Qt::Desktop
&& !w->data->is_closing) {
did_close = w->close();
list = QApplication::topLevelWidgets();
i = -1;
}
}
QWindowList processedWindows;
QApplicationPrivate::tryCloseAllWidgetWindows(&processedWindows);
}
/*!

View File

@ -120,6 +120,7 @@ public:
virtual void notifyActiveWindowChange(QWindow *);
virtual bool shouldQuit();
bool tryCloseAllWindows() Q_DECL_OVERRIDE;
#if defined(Q_WS_X11)
#ifndef QT_NO_SETTINGS
@ -294,6 +295,7 @@ public:
QPixmap applyQIconStyleHelper(QIcon::Mode mode, const QPixmap& base) const;
private:
static QApplicationPrivate *self;
static bool tryCloseAllWidgetWindows(QWindowList *processedWindows);
static void giveFocusAccordingToFocusPolicy(QWidget *w, QEvent *event, QPoint localPos);
static bool shouldSetFocus(QWidget *w, Qt::FocusPolicy policy);