关于Qt模态与非模态对话框

来源:互联网 发布:软件性能测试分类 编辑:程序博客网 时间:2024/05/22 03:08

模态对话框(Modal Dialog)与非模态对话框(Modeless Dialog)的概念不是Qt所独有的,在各种不同的平台下都存在。又有叫法是称为模式对话框,无模式对话框等。所谓模态对话框就是在其没有被关闭之前,用户不能与同一个应用程序的其他窗口进行交互,直到该对话框关闭。对于非模态对话框,当被打开时,用户既可选择和该对话框进行交互,也可以选择同应用程序的其他窗口交互。

Qt中,显示一个对话框一般有两种方式,一种是使用exec()方法,它总是以模态来显示对话框;另一种是使用show()方法,它使得对话框既可以模态显示,也可以非模态显示,决定它是模态还是非模态的是对话框的modal属性。

Qt中,Qt模态非模态对话框选择是通过其属性modal来确定的。我们来看看modal属性,其定义如下:

modal : bool默认情况下,对话框的该属性值是false,这时通过show()方法显示的对话框就是非模态的。而如果将该属性值设置为true,就设置成了模态对话框,其作用于把QWidget::windowModality属性设置为Qt::ApplicationModal。

而使用exec()方法显示对话框的话,将忽略modal属性值的设置并把对话框设置为模态对话框。

一般使用setModal()方法来设置对话框的modal属性。

我们总结一下设置对话框为模态的方法。

◆ 如果要设置为模态对话框,最简单的就是使用exec()方法,示例代码如下:

MyDialog myDlg;   myDlg.exec();也可以使用show()方法,示例代码如下:

MyDialog myDlg;   myDlg.setModal(true);   myDlg.show();

如果要设置为非模态对话框,必须使用show()方法,示例代码如下:

MyDialog myDlg;   myDlg.setModal(false);
//或者
myDlg.setModal();
myDlg.show();
再次强调,目前有的朋友对于模态对话框和非模态对话框的认识有误解,认为使用show()方法显示的就是非模态对话框,这是不正确的。

小贴士:有时候,我们需要一个对话框以非模态的形式显示,但又需要它总在所有窗口的最前面,这时可以通过如下代码设置:

MyDialog myDlg; myDlg.setModal(false);
//或者
myDlg.setModal();   myDlg.show();
//关键是下面这行
myDlg.setWindowFlags(Qt::WindowStaysOnTopHint);

Qt中创建模态对话框,主要用到了QDialog的exec函数:
SonDialog dlg(this);
int res = dlg.exec();
if (res == QDialog::Accepted)
{
QMessageBox::information(this, “INFORMATION”, “You clicked OK button!”);
}
if (res == QDialog::Rejected)
{
QMessageBox::information(this, “INFORMATION”, “You clicked CANCEL button!”);
}
正如上面代码所显示的,可以通过exec函数的返回值来判断用户点击了哪个按钮使得模态对话框退出的,这可以使得我们能够根据用户的不同行为在退出模态对话框之后采取不同的处理方法。


细心的读者可能就会问了,既然new了,如果不delete,那么内存不就存在了泄露的问题了吗?确实如此!所以,我们希望该Qt窗口在退出时自动能够delete掉自己,因此,我们在SonDialog的构造函数里,添加这样的一句代码:
setAttribute (Qt::WA_DeleteOnClose);
这样,我们的SonDialog就能够在它退出时自动的delete掉自己了,不会再造成内存泄漏问题
 
---------------------------------------------------------------分割线--------------------------------------------------
补充:
设有两个窗体分别是A和B,通过点击A中的一个按钮来弹出B窗体,并保持B窗体置顶,且如果B不退出无法操作A,这样可以采取两种方法
方法1:B窗体设置为模式窗体,就如上文所述,然后在A中执行B.show();
方法2:B不设置模式窗体,直接在A中执行B.exec();
这两种方法是有区别的,方法1,A不会在B.show()这句话处阻塞,而方法2会在B.exec()处阻塞A自己,等B退出后才会继续执行。
区别的表现之一是
如果显示B的代码是放在A的构造函数,B窗体不设置为模式窗体,在A的构造函数执行B.show(),那么B窗体会显示,接着A窗体显示,将B覆盖
如果B窗体设置为模式窗体,在A的构造函数中执行B.show(),AB都会显示,B在顶端,一些正常
如果在上一步的假设基础上,在显示B时候不显示B的标题栏,则这时B窗体中的按钮可以用,但是QLineEdit之类的输入控件就无法输入了
所以在弹出登录窗口之类的窗体时,推荐使用方法2
原创粉丝点击