Qt之模态、非模态、半模态窗口的介绍及实现QDialog的exec()方法

scorlw 发布于

Qt 之模态、非模态、半模态窗口的介绍及实现QDialog的exec()方法

一、简述

先简单介绍一下模态与非模态对话框。

模态对话框

简单一点讲就是在弹出模态对话框时,除了该对话框整个应用程序窗口都无法接受用户响应,处于等待状态,直到模态对话框被关闭。这时一般需要点击对话框中的确定或者取消等按钮关闭该对话框,程序得到对话框的返回值(即点击了确定还是取消),并根据返回值进行相应的操作,之后将操作权返回给用户。这个时候用户可以点击或者拖动程序其他窗口。

说白了就相当于阻塞同一应用程序中其它可视窗口的输入的对话框,用户必须完成这个对话框中的交互操作并且关闭了它之后才能访问应用程序中的其它窗口。

其实模态对话框的作用就是得到用户选择的结果,根据结果来进行下面的操作。

非模态对话框

又叫做无模式对话框,即弹出非模态对话框时,用户仍然可以对其他窗口进行操作,不会因为这个对话框未关闭就不能操作其他窗口。

半模态对话框

半模态对话框区别于模态与非模态对话框,或者说是介于两者之间,也就是说半模态对话框会阻塞窗口的响应,但是不会影响后续代码的执行。

二、Qt中的模态&非模态&半模态

QWidget

QWidget提供了setWindowModality()方法设置窗口半模态or非模态;

Qt::NonModal非模态对话框

Qt::WindowModal窗口级模态对话框,即只会阻塞父窗口、父窗口的父窗口及兄弟窗口。(半模态对话框)

Qt::ApplicationModal应用程序级模态对话框,即会阻塞整个应用程序的所有窗口。(半模态对话框)

show()方法——非模态对话框

显示窗口以及他的子窗口。

setWindowModality()方法

setWindowModality()方法可以设置窗口是否是模态窗口,Qt::WindowModality的默认值为Qt::NonModal,也就是非模态窗口。

所以,如果没有设置Qt::WindowModality属性值,我们每次用show()方法显示出的窗口都是非模态窗口。

QDialog

我们知道QWidget是大部分 控件的父类,也就是说QWidget是控件的始祖类,处于最上层,而QDialog也继承自QWidget。

在Qt助手中我们发现在QDialog除了继承QWidget的show()方法外,多了两个方法用来显示窗口,分别是open() 和 exec()方法。

open()方法——半模态对话框

使用open()方法显示出的对话框为窗口级模态对话框,并且立即返回,这样open()方法后的代码将会继续执行。open()方法就相当于如下代码。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
void showWindow()
{
QWidget* pWindow = new QWidget();

QWidget* childWindow = new QWidget(pWindow);
childWindow->setWindowModality(Qt::WindowModal);
childWindow->show();

// 上面三行代码相当于下面两行代码;
//QDialog* childDialog = new QDialog(pWindow);
//childDialog->open();

// 下面的代码可以执行;
qDebug() << "这是一个半模态窗口";
}

exec()方法——模态对话框

exec()方法显示出的对话框为模态对话框,同时会阻塞之前窗口的响应直到用户关闭这个对话框,并且返回DialogCode(包括Accepted和Rejected两个值)结果。

如果没有设置Qt::WindowModality属性值,使用exec()方法显示出的对话框默认为应用程序级模态对话框。所有使用exec()方法显示对话框在窗口关闭前会阻塞整个程序所有窗口的响应。同时调用exec()方法后的代码也不会执行直到对话框关闭才会继续执行。在关闭对话框后exec()方法会返回Accepted或者Rejected,一般程序根据返回不同的结果进行相应的操作。

用QDialog的exec()方法来显示一个模态对话框

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
void showModalWindow()
{
QWidget* pWindow = new QWidget();

QDialog* childDialog = new QDialog(pWindow);
int resutl = childDialog ->exec();
if (resutl == QDialog::Accepted)
{
qDebug() << "You Choose Ok";
}
else
{
qDebug() << "You Choose Cancel";
}


// 在关闭对话框之后,下面的代码才可以执行;
qDebug() << "这是一个模态窗口";
}

模式对话框有自己的事件循环。实际上 exec() 方法是先设置modal属性为Qt::ApplicationModal,然后调用 show() 显示对话框,最后启用事件循环来阻止exec() 方法的结束。直到窗口关闭,得到返回结果(DialogCode),退出事件循环,最后exec()方法调用结束,exec()方法后的代码将继续执行。

QDialog的exec() 方法的实现 整体上就是按照上方所讲的思路进行实现的。关于exec() 方法返回的结果可以通过对界面上的按钮绑定相应的槽,比如确定按钮绑定accept()槽,取消按钮绑定reject()槽,这样在点击确定或者取消按钮时exec()方法就会返回Accepted 或者 Rejected,可以根据返回的值做出相应的操作。

三、 Qt事件循环的一些理解(exec、eventloop)

1、事件循环一般用exec()函数开启。QApplicaion::exec()QMessageBox::exec()都是事件循环。其中前者又被称为主事件循环。

事件循环首先是一个无限“循环”,程序在exec()里面无限循环,能让跟在exec()后面的代码得不到运行机会,直至程序从exec()跳出。从exec()跳出时,事件循环即被终止。QEventLoop::quit()能够终止事件循环。

其次,之所以被称为“事件”循环,是因为它能接收事件,并处理之。当事件太多而不能马上处理完的时候,待处理事件被放在一个“队列”里,称为“事件循环队列”。当事件循环处理完一个事件后,就从“事件循环队列”中取出下一个事件处理之。当事件循环队列为空的时候,它和一个啥事也不做的永真循环有点类似,但是和永真循环不同的是,事件循环不会大量占用CPU资源。

事件循环的本质就是以队列的方式再次分配线程时间片。

2、事件循环是可以嵌套的,一层套一层,子层的事件循环执行exec()的时候,父层事件循环就处于中断状态;当子层事件循环跳出exec()后,父层事件循环才能继续循环下去。
另外,子层事件循环具有父层事件循环的几乎所有功能。Qt会把事件送到当前生效的那个事件循环队列中去,其中包括Gui的各种事件。所以用户在主线程中执行各种exec()(如QMessageBox::exec()QEventLoop::exec())的时候,虽然这些exec()打断了main()中的QApplication::exec(),但是Gui界面仍然能够正常响应。

3、如果某个子事件循环仍然有效,但其父循环被强制跳出,此时父循环不会立即执行跳出,而是等待子事件循环跳出后,父循环才会跳出。


版权声明:本文为CSDN博主「前行中的小猪」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/GoForwardToStep/article/details/53667566