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

模态对话框(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

你可能感兴趣的:(关于Qt模态与非模态对话框)