mirror of
https://github.com/crystalidea/qt6windows7.git
synced 2025-07-03 15:55:27 +08:00
qt 6.5.1 original
This commit is contained in:
4
tests/auto/widgets/dialogs/qdialog/BLACKLIST
Normal file
4
tests/auto/widgets/dialogs/qdialog/BLACKLIST
Normal file
@ -0,0 +1,4 @@
|
||||
[snapToDefaultButton]
|
||||
macos
|
||||
[showFullScreen]
|
||||
macos ci
|
17
tests/auto/widgets/dialogs/qdialog/CMakeLists.txt
Normal file
17
tests/auto/widgets/dialogs/qdialog/CMakeLists.txt
Normal file
@ -0,0 +1,17 @@
|
||||
# Copyright (C) 2022 The Qt Company Ltd.
|
||||
# SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
#####################################################################
|
||||
## tst_qdialog Test:
|
||||
#####################################################################
|
||||
|
||||
qt_internal_add_test(tst_qdialog
|
||||
SOURCES
|
||||
tst_qdialog.cpp
|
||||
LIBRARIES
|
||||
Qt::CorePrivate
|
||||
Qt::Gui
|
||||
Qt::GuiPrivate
|
||||
Qt::Widgets
|
||||
Qt::WidgetsPrivate
|
||||
)
|
772
tests/auto/widgets/dialogs/qdialog/tst_qdialog.cpp
Normal file
772
tests/auto/widgets/dialogs/qdialog/tst_qdialog.cpp
Normal file
@ -0,0 +1,772 @@
|
||||
// Copyright (C) 2016 The Qt Company Ltd.
|
||||
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
||||
|
||||
#include "../../../shared/highdpi.h"
|
||||
|
||||
#include <QTest>
|
||||
#include <QTestEventLoop>
|
||||
|
||||
#include <qdialog.h>
|
||||
#include <qapplication.h>
|
||||
#include <qlineedit.h>
|
||||
#include <qpushbutton.h>
|
||||
#include <qstyle.h>
|
||||
#include <QVBoxLayout>
|
||||
#include <QSignalSpy>
|
||||
#include <QSizeGrip>
|
||||
#include <QTimer>
|
||||
#include <QGraphicsProxyWidget>
|
||||
#include <QGraphicsView>
|
||||
#include <QWindow>
|
||||
#include <private/qguiapplication_p.h>
|
||||
#include <qpa/qplatformtheme.h>
|
||||
#include <qpa/qplatformtheme_p.h>
|
||||
|
||||
#include <QtWidgets/private/qapplication_p.h>
|
||||
|
||||
QT_FORWARD_DECLARE_CLASS(QDialog)
|
||||
|
||||
// work around function being protected
|
||||
class DummyDialog : public QDialog
|
||||
{
|
||||
public:
|
||||
DummyDialog(): QDialog() {}
|
||||
};
|
||||
|
||||
class tst_QDialog : public QObject
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
tst_QDialog();
|
||||
|
||||
private slots:
|
||||
void cleanup();
|
||||
void getSetCheck();
|
||||
void defaultButtons();
|
||||
void showMaximized();
|
||||
void showMinimized();
|
||||
void showFullScreen();
|
||||
void showAsTool();
|
||||
void showWithoutActivating_data();
|
||||
void showWithoutActivating();
|
||||
void toolDialogPosition();
|
||||
void deleteMainDefault();
|
||||
void deleteInExec();
|
||||
#if QT_CONFIG(sizegrip)
|
||||
void showSizeGrip();
|
||||
#endif
|
||||
void setVisible();
|
||||
void reject();
|
||||
void snapToDefaultButton();
|
||||
void transientParent_data();
|
||||
void transientParent();
|
||||
void dialogInGraphicsView();
|
||||
void keepPositionOnClose();
|
||||
void virtualsOnClose();
|
||||
void deleteOnDone();
|
||||
void quitOnDone();
|
||||
void focusWidgetAfterOpen();
|
||||
};
|
||||
|
||||
// Testing get/set functions
|
||||
void tst_QDialog::getSetCheck()
|
||||
{
|
||||
QDialog obj1;
|
||||
// int QDialog::result()
|
||||
// void QDialog::setResult(int)
|
||||
obj1.setResult(0);
|
||||
QCOMPARE(0, obj1.result());
|
||||
obj1.setResult(INT_MIN);
|
||||
QCOMPARE(INT_MIN, obj1.result());
|
||||
obj1.setResult(INT_MAX);
|
||||
QCOMPARE(INT_MAX, obj1.result());
|
||||
}
|
||||
|
||||
class ToolDialog : public QDialog
|
||||
{
|
||||
public:
|
||||
ToolDialog(QWidget *parent = nullptr)
|
||||
: QDialog(parent, Qt::Tool), mWasActive(false), mWasModalWindow(false), tId(-1) {}
|
||||
|
||||
bool wasActive() const { return mWasActive; }
|
||||
bool wasModalWindow() const { return mWasModalWindow; }
|
||||
|
||||
int exec() override
|
||||
{
|
||||
tId = startTimer(300);
|
||||
return QDialog::exec();
|
||||
}
|
||||
protected:
|
||||
void timerEvent(QTimerEvent *event) override
|
||||
{
|
||||
if (tId == event->timerId()) {
|
||||
killTimer(tId);
|
||||
mWasActive = isActiveWindow();
|
||||
mWasModalWindow = QGuiApplication::modalWindow() == windowHandle();
|
||||
reject();
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
int mWasActive;
|
||||
bool mWasModalWindow;
|
||||
int tId;
|
||||
};
|
||||
|
||||
tst_QDialog::tst_QDialog()
|
||||
{
|
||||
}
|
||||
|
||||
void tst_QDialog::cleanup()
|
||||
{
|
||||
QVERIFY(QApplication::topLevelWidgets().isEmpty());
|
||||
}
|
||||
|
||||
|
||||
void tst_QDialog::defaultButtons()
|
||||
{
|
||||
DummyDialog testWidget;
|
||||
testWidget.resize(200, 200);
|
||||
testWidget.setWindowTitle(QTest::currentTestFunction());
|
||||
QLineEdit *lineEdit = new QLineEdit(&testWidget);
|
||||
QPushButton *push = new QPushButton("Button 1", &testWidget);
|
||||
QPushButton *pushTwo = new QPushButton("Button 2", &testWidget);
|
||||
QPushButton *pushThree = new QPushButton("Button 3", &testWidget);
|
||||
pushThree->setAutoDefault(false);
|
||||
|
||||
testWidget.show();
|
||||
QApplicationPrivate::setActiveWindow(&testWidget);
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&testWidget));
|
||||
|
||||
push->setDefault(true);
|
||||
QVERIFY(push->isDefault());
|
||||
|
||||
pushTwo->setFocus();
|
||||
QVERIFY(pushTwo->isDefault());
|
||||
pushThree->setFocus();
|
||||
QVERIFY(push->isDefault());
|
||||
lineEdit->setFocus();
|
||||
QVERIFY(push->isDefault());
|
||||
|
||||
pushTwo->setDefault(true);
|
||||
QVERIFY(pushTwo->isDefault());
|
||||
|
||||
pushTwo->setFocus();
|
||||
QVERIFY(pushTwo->isDefault());
|
||||
lineEdit->setFocus();
|
||||
QVERIFY(pushTwo->isDefault());
|
||||
}
|
||||
|
||||
void tst_QDialog::showMaximized()
|
||||
{
|
||||
QDialog dialog(0);
|
||||
dialog.setSizeGripEnabled(true);
|
||||
#if QT_CONFIG(sizegrip)
|
||||
QSizeGrip *sizeGrip = dialog.findChild<QSizeGrip *>();
|
||||
QVERIFY(sizeGrip);
|
||||
#endif
|
||||
|
||||
dialog.showMaximized();
|
||||
QVERIFY(dialog.isMaximized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
#if QT_CONFIG(sizegrip) && !defined(Q_OS_DARWIN) && !defined(Q_OS_HPUX)
|
||||
QVERIFY(!sizeGrip->isVisible());
|
||||
#endif
|
||||
|
||||
dialog.showNormal();
|
||||
QVERIFY(!dialog.isMaximized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
#if QT_CONFIG(sizegrip)
|
||||
QVERIFY(sizeGrip->isVisible());
|
||||
#endif
|
||||
|
||||
dialog.showMaximized();
|
||||
QVERIFY(dialog.isMaximized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.hide();
|
||||
QVERIFY(dialog.isMaximized());
|
||||
QVERIFY(!dialog.isVisible());
|
||||
|
||||
dialog.setVisible(true);
|
||||
QVERIFY(dialog.isMaximized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.hide();
|
||||
QVERIFY(dialog.isMaximized());
|
||||
QVERIFY(!dialog.isVisible());
|
||||
|
||||
dialog.showMaximized();
|
||||
QVERIFY(dialog.isMaximized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
}
|
||||
|
||||
void tst_QDialog::showMinimized()
|
||||
{
|
||||
QDialog dialog(0);
|
||||
|
||||
dialog.showMinimized();
|
||||
QVERIFY(dialog.isMinimized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.showNormal();
|
||||
QVERIFY(!dialog.isMinimized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.showMinimized();
|
||||
QVERIFY(dialog.isMinimized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.hide();
|
||||
QVERIFY(dialog.isMinimized());
|
||||
QVERIFY(!dialog.isVisible());
|
||||
|
||||
dialog.setVisible(true);
|
||||
QVERIFY(dialog.isMinimized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.hide();
|
||||
QVERIFY(dialog.isMinimized());
|
||||
QVERIFY(!dialog.isVisible());
|
||||
|
||||
dialog.showMinimized();
|
||||
QVERIFY(dialog.isMinimized());
|
||||
QVERIFY(dialog.isVisible());
|
||||
}
|
||||
|
||||
void tst_QDialog::showFullScreen()
|
||||
{
|
||||
QDialog dialog(0, Qt::X11BypassWindowManagerHint);
|
||||
dialog.setSizeGripEnabled(true);
|
||||
#if QT_CONFIG(sizegrip)
|
||||
QSizeGrip *sizeGrip = dialog.findChild<QSizeGrip *>();
|
||||
QVERIFY(sizeGrip);
|
||||
#endif
|
||||
|
||||
dialog.showFullScreen();
|
||||
QVERIFY(dialog.isFullScreen());
|
||||
QVERIFY(dialog.isVisible());
|
||||
#if QT_CONFIG(sizegrip)
|
||||
QVERIFY(!sizeGrip->isVisible());
|
||||
#endif
|
||||
|
||||
dialog.showNormal();
|
||||
QVERIFY(!dialog.isFullScreen());
|
||||
QVERIFY(dialog.isVisible());
|
||||
#if QT_CONFIG(sizegrip)
|
||||
QVERIFY(sizeGrip->isVisible());
|
||||
#endif
|
||||
|
||||
dialog.showFullScreen();
|
||||
QVERIFY(dialog.isFullScreen());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.hide();
|
||||
QVERIFY(dialog.isFullScreen());
|
||||
QVERIFY(!dialog.isVisible());
|
||||
|
||||
dialog.setVisible(true);
|
||||
QVERIFY(dialog.isFullScreen());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.hide();
|
||||
QVERIFY(dialog.isFullScreen());
|
||||
QVERIFY(!dialog.isVisible());
|
||||
|
||||
dialog.showFullScreen();
|
||||
QVERIFY(dialog.isFullScreen());
|
||||
QVERIFY(dialog.isVisible());
|
||||
|
||||
dialog.hide();
|
||||
QVERIFY(dialog.isFullScreen());
|
||||
QVERIFY(!dialog.isVisible());
|
||||
}
|
||||
|
||||
void tst_QDialog::showAsTool()
|
||||
{
|
||||
if (QStringList{"xcb", "offscreen"}.contains(QGuiApplication::platformName()))
|
||||
QSKIP("activeWindow() is not respected by all Xcb window managers and the offscreen plugin");
|
||||
|
||||
if (!QGuiApplicationPrivate::platformIntegration()->hasCapability(QPlatformIntegration::WindowActivation))
|
||||
QSKIP("QWindow::requestActivate() is not supported.");
|
||||
|
||||
DummyDialog testWidget;
|
||||
testWidget.resize(200, 200);
|
||||
testWidget.setWindowTitle(QTest::currentTestFunction());
|
||||
ToolDialog dialog(&testWidget);
|
||||
testWidget.show();
|
||||
testWidget.activateWindow();
|
||||
QVERIFY(QTest::qWaitForWindowActive(&testWidget));
|
||||
dialog.exec();
|
||||
if (testWidget.style()->styleHint(QStyle::SH_Widget_ShareActivation, 0, &testWidget)) {
|
||||
QCOMPARE(dialog.wasActive(), true);
|
||||
} else {
|
||||
QCOMPARE(dialog.wasActive(), false);
|
||||
}
|
||||
}
|
||||
|
||||
void tst_QDialog::showWithoutActivating_data()
|
||||
{
|
||||
QTest::addColumn<bool>("showWithoutActivating");
|
||||
QTest::addColumn<int>("focusInCount");
|
||||
|
||||
QTest::addRow("showWithoutActivating") << true << 0;
|
||||
QTest::addRow("showWithActivating") << false << 1;
|
||||
}
|
||||
|
||||
void tst_QDialog::showWithoutActivating()
|
||||
{
|
||||
QFETCH(bool, showWithoutActivating);
|
||||
QFETCH(int, focusInCount);
|
||||
|
||||
struct Dialog : public QDialog
|
||||
{
|
||||
int focusInCount = 0;
|
||||
protected:
|
||||
void focusInEvent(QFocusEvent *) override { ++focusInCount; }
|
||||
} dialog;
|
||||
dialog.setAttribute(Qt::WA_ShowWithoutActivating, showWithoutActivating);
|
||||
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
QCOMPARE(dialog.focusInCount, focusInCount);
|
||||
}
|
||||
|
||||
// Verify that pos() returns the same before and after show()
|
||||
// for a dialog with the Tool window type.
|
||||
void tst_QDialog::toolDialogPosition()
|
||||
{
|
||||
QDialog dialog(0, Qt::Tool);
|
||||
dialog.move(QPoint(100,100));
|
||||
const QPoint beforeShowPosition = dialog.pos();
|
||||
dialog.show();
|
||||
const int fuzz = int(dialog.devicePixelRatio());
|
||||
const QPoint afterShowPosition = dialog.pos();
|
||||
QVERIFY2(HighDpi::fuzzyCompare(afterShowPosition, beforeShowPosition, fuzz),
|
||||
HighDpi::msgPointMismatch(afterShowPosition, beforeShowPosition).constData());
|
||||
}
|
||||
|
||||
class Dialog : public QDialog
|
||||
{
|
||||
public:
|
||||
Dialog(QPushButton *&button)
|
||||
{
|
||||
button = new QPushButton(this);
|
||||
}
|
||||
};
|
||||
|
||||
void tst_QDialog::deleteMainDefault()
|
||||
{
|
||||
QPushButton *button;
|
||||
Dialog dialog(button);
|
||||
button->setDefault(true);
|
||||
delete button;
|
||||
dialog.show();
|
||||
QTestEventLoop::instance().enterLoop(2);
|
||||
}
|
||||
|
||||
void tst_QDialog::deleteInExec()
|
||||
{
|
||||
QDialog *dialog = new QDialog(0);
|
||||
QMetaObject::invokeMethod(dialog, "deleteLater", Qt::QueuedConnection);
|
||||
QCOMPARE(dialog->exec(), int(QDialog::Rejected));
|
||||
}
|
||||
|
||||
#if QT_CONFIG(sizegrip)
|
||||
|
||||
void tst_QDialog::showSizeGrip()
|
||||
{
|
||||
QDialog dialog(nullptr);
|
||||
dialog.show();
|
||||
new QWidget(&dialog);
|
||||
QVERIFY(!dialog.isSizeGripEnabled());
|
||||
|
||||
dialog.setSizeGripEnabled(true);
|
||||
QPointer<QSizeGrip> sizeGrip = dialog.findChild<QSizeGrip *>();
|
||||
QVERIFY(sizeGrip);
|
||||
QVERIFY(sizeGrip->isVisible());
|
||||
QVERIFY(dialog.isSizeGripEnabled());
|
||||
|
||||
dialog.setSizeGripEnabled(false);
|
||||
QVERIFY(!dialog.isSizeGripEnabled());
|
||||
|
||||
dialog.setSizeGripEnabled(true);
|
||||
sizeGrip = dialog.findChild<QSizeGrip *>();
|
||||
QVERIFY(sizeGrip);
|
||||
QVERIFY(sizeGrip->isVisible());
|
||||
sizeGrip->hide();
|
||||
dialog.hide();
|
||||
dialog.show();
|
||||
QVERIFY(!sizeGrip->isVisible());
|
||||
}
|
||||
|
||||
#endif // QT_CONFIG(sizegrip)
|
||||
|
||||
void tst_QDialog::setVisible()
|
||||
{
|
||||
QWidget topLevel;
|
||||
topLevel.show();
|
||||
|
||||
QDialog *dialog = new QDialog;
|
||||
dialog->setLayout(new QVBoxLayout);
|
||||
dialog->layout()->addWidget(new QPushButton("dialog button"));
|
||||
|
||||
QWidget *widget = new QWidget(&topLevel);
|
||||
widget->setLayout(new QVBoxLayout);
|
||||
widget->layout()->addWidget(dialog);
|
||||
|
||||
QVERIFY(!dialog->isVisible());
|
||||
QVERIFY(!dialog->isHidden());
|
||||
|
||||
widget->show();
|
||||
QVERIFY(dialog->isVisible());
|
||||
QVERIFY(!dialog->isHidden());
|
||||
|
||||
widget->hide();
|
||||
dialog->hide();
|
||||
widget->show();
|
||||
QVERIFY(!dialog->isVisible());
|
||||
QVERIFY(dialog->isHidden());
|
||||
}
|
||||
|
||||
class TestRejectDialog : public QDialog
|
||||
{
|
||||
public:
|
||||
TestRejectDialog() : cancelReject(false), called(0) {}
|
||||
void reject() override
|
||||
{
|
||||
called++;
|
||||
if (!cancelReject)
|
||||
QDialog::reject();
|
||||
}
|
||||
bool cancelReject;
|
||||
int called;
|
||||
};
|
||||
|
||||
void tst_QDialog::reject()
|
||||
{
|
||||
TestRejectDialog dialog;
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
QVERIFY(dialog.isVisible());
|
||||
dialog.reject();
|
||||
QTRY_VERIFY(!dialog.isVisible());
|
||||
QCOMPARE(dialog.called, 1);
|
||||
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
QVERIFY(dialog.isVisible());
|
||||
QVERIFY(dialog.close());
|
||||
QTRY_VERIFY(!dialog.isVisible());
|
||||
QCOMPARE(dialog.called, 2);
|
||||
|
||||
dialog.cancelReject = true;
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
QVERIFY(dialog.isVisible());
|
||||
dialog.reject();
|
||||
QTRY_VERIFY(dialog.isVisible());
|
||||
QCOMPARE(dialog.called, 3);
|
||||
QVERIFY(!dialog.close());
|
||||
QTRY_VERIFY(dialog.isVisible());
|
||||
QCOMPARE(dialog.called, 4);
|
||||
}
|
||||
|
||||
static QByteArray formatPoint(QPoint p)
|
||||
{
|
||||
return QByteArray::number(p.x()) + ", " + QByteArray::number(p.y());
|
||||
}
|
||||
|
||||
void tst_QDialog::snapToDefaultButton()
|
||||
{
|
||||
#ifdef QT_NO_CURSOR
|
||||
QSKIP("Test relies on there being a cursor");
|
||||
#else
|
||||
if (!QGuiApplication::platformName().compare(QLatin1String("wayland"), Qt::CaseInsensitive))
|
||||
QSKIP("This platform does not support setting the cursor position.");
|
||||
#ifdef Q_OS_ANDROID
|
||||
QSKIP("Android does not support cursor");
|
||||
#endif
|
||||
|
||||
const QRect dialogGeometry(QGuiApplication::primaryScreen()->availableGeometry().topLeft()
|
||||
+ QPoint(100, 100), QSize(200, 200));
|
||||
const QPoint startingPos = dialogGeometry.bottomRight() + QPoint(100, 100);
|
||||
QCursor::setPos(startingPos);
|
||||
#ifdef Q_OS_MACOS
|
||||
// On OS X we use CGEventPost to move the cursor, it needs at least
|
||||
// some time before the event handled and the position really set.
|
||||
QTest::qWait(100);
|
||||
#endif
|
||||
QCOMPARE(QCursor::pos(), startingPos);
|
||||
QDialog dialog;
|
||||
QPushButton *button = new QPushButton(&dialog);
|
||||
button->setDefault(true);
|
||||
dialog.setGeometry(dialogGeometry);
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
const QPoint localPos = button->mapFromGlobal(QCursor::pos());
|
||||
if (QGuiApplicationPrivate::platformTheme()->themeHint(QPlatformTheme::DialogSnapToDefaultButton).toBool())
|
||||
QVERIFY2(button->rect().contains(localPos), formatPoint(localPos).constData());
|
||||
else
|
||||
QVERIFY2(!button->rect().contains(localPos), formatPoint(localPos).constData());
|
||||
#endif // !QT_NO_CURSOR
|
||||
}
|
||||
|
||||
void tst_QDialog::transientParent_data()
|
||||
{
|
||||
QTest::addColumn<bool>("nativewidgets");
|
||||
QTest::newRow("Non-native") << false;
|
||||
QTest::newRow("Native") << true;
|
||||
}
|
||||
|
||||
void tst_QDialog::transientParent()
|
||||
{
|
||||
QFETCH(bool, nativewidgets);
|
||||
QWidget topLevel;
|
||||
topLevel.resize(200, 200);
|
||||
topLevel.move(QGuiApplication::primaryScreen()->availableGeometry().center() - QPoint(100, 100));
|
||||
QVBoxLayout *layout = new QVBoxLayout(&topLevel);
|
||||
QWidget *innerWidget = new QWidget(&topLevel);
|
||||
layout->addWidget(innerWidget);
|
||||
if (nativewidgets)
|
||||
innerWidget->winId();
|
||||
topLevel.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&topLevel));
|
||||
QDialog dialog(innerWidget);
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
// Transient parent should always be the top level, also when using
|
||||
// native child widgets.
|
||||
QCOMPARE(dialog.windowHandle()->transientParent(), topLevel.windowHandle());
|
||||
}
|
||||
|
||||
void tst_QDialog::dialogInGraphicsView()
|
||||
{
|
||||
// QTBUG-49124: A dialog embedded into QGraphicsView has Qt::WA_DontShowOnScreen
|
||||
// set (as has a native dialog). It must not trigger the modal handling though
|
||||
// as not to lock up.
|
||||
QGraphicsScene scene;
|
||||
QGraphicsView view(&scene);
|
||||
view.setWindowTitle(QTest::currentTestFunction());
|
||||
const QRect availableGeometry = QGuiApplication::primaryScreen()->availableGeometry();
|
||||
view.resize(availableGeometry.size() / 2);
|
||||
view.move(availableGeometry.left() + availableGeometry.width() / 4,
|
||||
availableGeometry.top() + availableGeometry.height() / 4);
|
||||
ToolDialog *dialog = new ToolDialog;
|
||||
scene.addWidget(dialog);
|
||||
view.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&view));
|
||||
for (int i = 0; i < 3; ++i) {
|
||||
dialog->exec();
|
||||
QVERIFY(!dialog->wasModalWindow());
|
||||
}
|
||||
}
|
||||
|
||||
// QTBUG-79147 (Windows): Closing a dialog by clicking the 'X' in the title
|
||||
// bar would offset the dialog position when shown next time.
|
||||
void tst_QDialog::keepPositionOnClose()
|
||||
{
|
||||
QDialog dialog;
|
||||
dialog.setWindowTitle(QTest::currentTestFunction());
|
||||
const QRect availableGeometry = QGuiApplication::primaryScreen()->availableGeometry();
|
||||
dialog.resize(availableGeometry.size() / 4);
|
||||
QPoint pos = availableGeometry.topLeft() + QPoint(100, 100);
|
||||
dialog.move(pos);
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
pos = dialog.pos();
|
||||
dialog.close();
|
||||
dialog.windowHandle()->destroy(); // Emulate a click on close by destroying the window.
|
||||
QTest::qWait(50);
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
QTest::qWait(50);
|
||||
QCOMPARE(dialog.pos(), pos);
|
||||
}
|
||||
|
||||
/*!
|
||||
Verify that the virtual functions related to closing a dialog are
|
||||
called exactly once, no matter how the dialog gets closed.
|
||||
*/
|
||||
void tst_QDialog::virtualsOnClose()
|
||||
{
|
||||
class Dialog : public QDialog
|
||||
{
|
||||
public:
|
||||
using QDialog::QDialog;
|
||||
int closeEventCount = 0;
|
||||
int acceptCount = 0;
|
||||
int rejectCount = 0;
|
||||
int doneCount = 0;
|
||||
|
||||
void accept() override
|
||||
{
|
||||
++acceptCount;
|
||||
QDialog::accept();
|
||||
}
|
||||
void reject() override
|
||||
{
|
||||
++rejectCount;
|
||||
QDialog::reject();
|
||||
}
|
||||
void done(int result) override
|
||||
{
|
||||
++doneCount;
|
||||
QDialog::done(result);
|
||||
}
|
||||
|
||||
protected:
|
||||
void closeEvent(QCloseEvent *e) override
|
||||
{
|
||||
++closeEventCount;
|
||||
QDialog::closeEvent(e);
|
||||
}
|
||||
};
|
||||
|
||||
{
|
||||
Dialog dialog;
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
dialog.accept();
|
||||
// we used to only hide the dialog, and we still don't want a
|
||||
// closeEvent call for application-triggered calls to QDialog::done
|
||||
QCOMPARE(dialog.closeEventCount, 0);
|
||||
QCOMPARE(dialog.acceptCount, 1);
|
||||
QCOMPARE(dialog.rejectCount, 0);
|
||||
QCOMPARE(dialog.doneCount, 1);
|
||||
}
|
||||
|
||||
{
|
||||
Dialog dialog;
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
dialog.reject();
|
||||
QCOMPARE(dialog.closeEventCount, 0);
|
||||
QCOMPARE(dialog.acceptCount, 0);
|
||||
QCOMPARE(dialog.rejectCount, 1);
|
||||
QCOMPARE(dialog.doneCount, 1);
|
||||
}
|
||||
|
||||
{
|
||||
Dialog dialog;
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
dialog.close();
|
||||
QCOMPARE(dialog.closeEventCount, 1);
|
||||
QCOMPARE(dialog.acceptCount, 0);
|
||||
QCOMPARE(dialog.rejectCount, 1);
|
||||
QCOMPARE(dialog.doneCount, 1);
|
||||
}
|
||||
|
||||
{
|
||||
// user clicks close button in title bar
|
||||
Dialog dialog;
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
QWindowSystemInterface::handleCloseEvent(dialog.windowHandle());
|
||||
QApplication::processEvents();
|
||||
QCOMPARE(dialog.closeEventCount, 1);
|
||||
QCOMPARE(dialog.acceptCount, 0);
|
||||
QCOMPARE(dialog.rejectCount, 1);
|
||||
QCOMPARE(dialog.doneCount, 1);
|
||||
}
|
||||
|
||||
{
|
||||
struct EventFilter : QObject {
|
||||
EventFilter(Dialog *dialog)
|
||||
{ dialog->installEventFilter(this); }
|
||||
int closeEventCount = 0;
|
||||
bool eventFilter(QObject *r, QEvent *e) override
|
||||
{
|
||||
if (e->type() == QEvent::Close) {
|
||||
++closeEventCount;
|
||||
}
|
||||
return QObject::eventFilter(r, e);
|
||||
}
|
||||
};
|
||||
// dialog gets destroyed while shown
|
||||
Dialog *dialog = new Dialog;
|
||||
QSignalSpy rejectedSpy(dialog, &QDialog::rejected);
|
||||
EventFilter filter(dialog);
|
||||
|
||||
dialog->show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(dialog));
|
||||
delete dialog;
|
||||
// Qt doesn't deliver events to QWidgets closed during destruction
|
||||
QCOMPARE(filter.closeEventCount, 0);
|
||||
// QDialog doesn't emit signals when closed by destruction
|
||||
QCOMPARE(rejectedSpy.size(), 0);
|
||||
}
|
||||
}
|
||||
|
||||
/*!
|
||||
QDialog::done is documented to respect Qt::WA_DeleteOnClose.
|
||||
*/
|
||||
void tst_QDialog::deleteOnDone()
|
||||
{
|
||||
{
|
||||
std::unique_ptr<QDialog> dialog(new QDialog);
|
||||
QPointer<QDialog> watcher(dialog.get());
|
||||
dialog->setAttribute(Qt::WA_DeleteOnClose);
|
||||
dialog->show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(dialog.get()));
|
||||
|
||||
dialog->accept();
|
||||
QTRY_COMPARE(watcher.isNull(), true);
|
||||
dialog.release(); // if we get here, the dialog is destroyed
|
||||
}
|
||||
|
||||
// it is still safe to delete the dialog explicitly as long as events
|
||||
// have not yet been processed
|
||||
{
|
||||
std::unique_ptr<QDialog> dialog(new QDialog);
|
||||
dialog->setAttribute(Qt::WA_DeleteOnClose);
|
||||
dialog->show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(dialog.get()));
|
||||
|
||||
dialog->accept();
|
||||
dialog.reset();
|
||||
QApplication::processEvents();
|
||||
}
|
||||
}
|
||||
|
||||
/*!
|
||||
QDialog::done is documented to make QApplication emit lastWindowClosed if
|
||||
the dialog was the last window.
|
||||
*/
|
||||
void tst_QDialog::quitOnDone()
|
||||
{
|
||||
QSignalSpy quitSpy(qApp, &QGuiApplication::lastWindowClosed);
|
||||
|
||||
QDialog dialog;
|
||||
dialog.show();
|
||||
QVERIFY(QTest::qWaitForWindowExposed(&dialog));
|
||||
|
||||
// QGuiApplication::lastWindowClosed is documented to only be emitted
|
||||
// when we are in exec()
|
||||
QTimer::singleShot(0, &dialog, &QDialog::accept);
|
||||
// also quit with a timer in case the test fails
|
||||
QTimer::singleShot(1000, QApplication::instance(), &QApplication::quit);
|
||||
QApplication::exec();
|
||||
QCOMPARE(quitSpy.size(), 1);
|
||||
}
|
||||
|
||||
void tst_QDialog::focusWidgetAfterOpen()
|
||||
{
|
||||
QDialog dialog;
|
||||
dialog.setLayout(new QVBoxLayout);
|
||||
|
||||
QPushButton *pb1 = new QPushButton;
|
||||
QPushButton *pb2 = new QPushButton;
|
||||
dialog.layout()->addWidget(pb1);
|
||||
dialog.layout()->addWidget(pb2);
|
||||
|
||||
pb2->setFocus();
|
||||
QCOMPARE(dialog.focusWidget(), static_cast<QWidget *>(pb2));
|
||||
|
||||
dialog.open();
|
||||
QCOMPARE(dialog.focusWidget(), static_cast<QWidget *>(pb2));
|
||||
}
|
||||
|
||||
QTEST_MAIN(tst_QDialog)
|
||||
#include "tst_qdialog.moc"
|
Reference in New Issue
Block a user