编写 native 风格的 Qt 程序(5)

首先先来看QDialog的一副截图(出自 Qt Developer Day, 2009):

在这里,我们要注意的是不同平台之上对话框的按钮的不同。其实这是同一段代码编译的,没有使用条件编译技术。那么是如何做到的呢?答案是使用QDialogButtonBox这个类。

QDialogButtonBox用于管理对话框按钮的顺序、布局、文本和图标等,以保证这些在不同平台能够具有不同表现。如果我们没有 Mac 系统,我们怎么知道该如何布局按钮呢?我们怎么获得这些按钮在不同平台上的图标呢?这些都不会成为我们实现程序的障碍,因为使用QDialogButtonBox就足够了。例如,要实现上面的效果,我们只需要一行代码:

QDialogButtonBox box(QDialogButtonBox::Save | 
                     QDalogButtonBox::Discard | 
                     QDialogButtonBox::Cancel);

这样,Qt 就会在不同的平台进行自动适应。

QDialogButtonBox为不同的按钮分配不同的角色来实现这一功能。因此,我们在上面的QDialogButtonBox::Save这些实际都是一个简单的 enum,用于标记按钮的角色。如果你需要使用自己的按钮,并且为之附加角色,那么可以使用下面的代码:

QDialogButtonBox box;
box.addButton(myButton, QDialogButtonBox::AcceptRole);

这样,myButton的角色就是QDialogButtonBox::AcceptRole,而QDialogButtonBox也能够根据这个角色为之分配合适的图标和位置等等。

QDialogButtonBox先告一段落,下面来说说模态对话框。什么是模态对话框?所谓模态,就是在对话框弹出来之后,能够阻塞后面的窗口。Windows 上一般在退出时会弹出来一个问你是否保存的对话框,就是一个模态对话框。当它出现的时候,后面的窗口是不能点击的,必须要你关闭这个对话框之后才可以。在 Qt 实现模态对话框很简单:

MyQDialogSubclass dialog;
// Various bits of initialization
if (dialog.exec() == QDialog::Accept)  {  // HERE
    // Set new values or do extra work
    // based on results.
}

这段代码在运行时,会在标记 HERE 注释这行阻塞,具体是QDialog::exec()这个函数。这之后的代码在你关闭 dialog 对话框之后才会被执行。利用这一技术,你就可以在 if 里面获取依赖于对话框返回值的数据。例如,对话框用于收集用户数据等。

class MyDialog : public QDialog
{
public:
    QString name;
};
// .... 
MyDialog d;
if(d.exec() == QDialog::Accept) {
    QString name = d.name;
    // do something with name...
}

上面的代码,MyDialog用于用户输入 name 的值。我们使用模态对话框,就可以在 if 里面获取这个值了。

不过,不同平台上的模态对话框的使用方式是不一样的。比如,Windows 平台上,模态对话框用于严重错误的提示,或者是在继续之前必须完成的任务;KDE 上,模态对话框用于可能造成数据丢失或者严重后果的交互。

Qt 中,对话框的打开具有三种方式:

  1. QDialog::show(): 非模态
  2. QDialog::exec(): 模态
  3. QDialog::open(): 窗口模态

前两种我们很容易理解,下面来看看什么是窗口模态。比如,我们有两个窗口(以下图示来自 Qt Developer Day, 2009):

使用如下代码,我们用open()函数打开一个对话框:

if(!messageBox) { 
    messageBox = new QMessageBox("SDI", 
                                 "The document has been modified. \n" 
                                 "Do you want to save your changes?", 
                                 QMessageBox::Warning, 
                                 QMessageBox::Yes | QMessageBox::Default, 
                                 QMessageBox::No, 
                                 QMessageBox::Cancel | QMessageBox::Escape, 
                                 this); 
    connect(messageBox, SIGNAL(finished(int)), SLOT(handleDialogClose(int))); 
} 
messageBox->open();

这就是open()函数的效果:它类似模态对话框,但是只会阻塞一个窗口,而不是将整个系统阻塞掉。

最后一个要说的技术是QFormLayout。这个布局用于展示表单。来看一下下面的截图(出自 Qt Developer Day, 2009):

注意这里的文本对齐方式和按钮的顺序。前面已经说过用QDialogButtonBox实现不同平台下按钮的顺序,而上面的对齐方式则是使用QFormLayout实现,例如:

QFormLayout *layout = new QFormLayout;
layout->addRow(tr("Name:"), nameLineEdit);
// more...

这样,我们又能够使用简单的代码实现不同平台的不同布局表现。

2 Comments

  1. Ryan 2011年8月22日
    • DevBean 2011年8月23日

Leave a Reply