2.3快速设计对话框(Rapid Dialog Design)

2.3快速设计对话框
通常程序员们都是用c++源代码编写Qt应用程序,Qt也是很容易用来编写的。然而,许多程序员更喜欢用可视化的方法设计对话框,这样能更快速更容易对对话框进行修改。
Qt Designer满足了程序员的这一要求,提供了可视化设计对话框的方法。它可以给一个应用程序提供全部或者部分对话框。用Qt Designer设计的对话框和用c++代码写成的对话框是一样的,可以用做一个常用的工具,并不对编辑器产生影响。
在这一节中,我们使用Qt Designer创建Go-to-Cell对话框,无论用编写代码的方式还是用Qt Designer,创建对话框都有如下基本的步骤:
1、创建和初始化子控件。
2、把子控件放到布局管理器中。
3、设置tab顺序。
4、创建信号和槽。
5、实现对话框的自己的槽函数。
在windows平台Qt的安装目录的bin目录下,点击desinger.exe,或者在unix平台,在命令行上输入designer。当Qt Designer启动后,它会列出一个控件模板的列表,选择一个模板,进入设计。
**************************
原文中对Qt Designer的介绍略去不想翻译了,只要稍有点界面编程基础的都可以轻松使用。如果确实需要,以后再补上。
我个人不喜欢使用这个东东,因为要多一个文件要维护,当然如果要频繁修改所设计的对话框,那这种方法还是很方便的。但不管怎么样,最终都要修改源代码。所以我还是比较喜欢用源代码的方式把控件手工写出来。
我想主要介绍把对话框设计好以后,保存为.ui文件后的处理。
**************************
假如设计好的文件保存在gotocell目录中,命名为gotocelldialog.ui中,然后在同一个目录下创建一个main.cpp文件,编码如下:
#include <QApplication>
#include <QDialog>
#include "ui_gotocelldialog.h"
int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    Ui::GoToCellDialog ui;
    QDialog *dialog = new QDialog;
    ui.setupUi(dialog);
    dialog->show();
    return app.exec();
}
保存后,在该目录下运行qmake,创建.pro文件,然后运行qmake -project生成makefile文件。qmake可以发现gotocelldialog.ui文件,然后就会调用uic(Qt的用户界面编译器),uic工具把gotocelldialog.ui转换成c++代码,保存在ui_gotocelldialog.h中。
在ui_gotocelldialog.h中,包含了Ui::GoToCellDialog类的定义,这个类和gotocelldialog.ui等价。这个类声明成员变量存储对话框的子控件和布局管理器,setupUi()函数初始化对话框。
这个类的定义看起来有点象下面这个样子:
class Ui::GoToCellDialog
{
public:
    QLabel *label;
    QLineEdit *lineEdit;
    QSpacerItem *spacerItem;
    QPushButton *okButton;
    QPushButton *cancelButton;
    ...
    void setupUi(QWidget *widget) {
        ...
    }
};
这个类没有父类。使用时创建一个QDialog,把它传递给setupUi()函数。
运行这个程序,对话框将会显示出来,但是有些功能它还不能实现:
1、Ok按钮是不可用状态的
2、Cancel按钮不作任何事情
3、编辑框除可以输入许可的字符或者数字外,还可以输入任何文本
我们可以编写代码,让这个对话框变得有用起来。最直接的方法是创建一个新类,继承QDialog和Ui::GoToCell-Dialog,补上缺少的功能。(这说明任何软件问题可以通过添加一层间接包装来简单解决)。通常命名新类规则是把去掉uic生成的类名去掉Ui::前缀。
创建gotocelldialog.h头文件,写下如下代码:
#ifndef GOTOCELLDIALOG_H
#define GOTOCELLDIALOG_H
#include <QDialog>
#include "ui_gotocelldialog.h"
class GoToCellDialog : public QDialog, public Ui::GoToCellDialog
{
    Q_OBJECT
public:
    GoToCellDialog(QWidget *parent = 0);
private slots:
    void on_lineEdit_textChanged();
};
#endif
新建gotocelldialog.cpp源文件,实现这个类:
#include <QtGui>
#include "gotocelldialog.h"
GoToCellDialog::GoToCellDialog(QWidget *parent)
    : QDialog(parent)
{
    setupUi(this);
    QRegExp regExp("[A-Za-z][1-9][0-9]{0,2}");
    lineEdit->setValidator(new QRegExpValidator(regExp, this));
    connect(okButton, SIGNAL(clicked()), this, SLOT(accept()));
    connect(cancelButton, SIGNAL(clicked()), this, SLOT(reject()));
}
void GoToCellDialog::on_lineEdit_textChanged()
{
    okButton->setEnabled(lineEdit->hasAcceptableInput());
}
在构造函数中,我们调用setupUi()初始化这个对话框。由于多继承,我们可以直接使Ui::GoToCellDialog的成员。创建了用户界面以后,我们可以把子控件的信号和槽函数连接起来。
在构造函数中,我们还创建一个许可器(validator)限制编辑框输入的范围。Qt提供了三个许可器类:QIntValidator,QDoubleValidator和QRegExpValidator。这里我们使用了QRegExpValidator,使用的表达式为“[A-Za-z][1-9][0-9]{0,2}”这个表达式的意思是第一个字符输入为大写或者小写字母,第二个字符为一个数字范围是1到9,第三个字符是一个数字范围为0到9。在QRegExpValidator的构造函数中,第二个参数为this,把当前类作为它的父控件,这样就可以不用删除它,父控件析构时可以被自动删除。
Qt的父子机制在QObject中实现的。当我们创建一个带有父的对象(如一个子控件,一个许可器,布局管理器等)时,父对象把子对象放到自己的子对象列表中。父对象被删除时,它查找自己的子对象并把每一个删除掉。这些子对象再把自己的子对象删除掉,如此递归,知道删除所有对象。
这种父子对象的机制简化了内存管理,减少了内存泄漏的危险。需要程序员删除的对象就是我们使用new创建的没有父对象的对象。如果我们在父对象存在时删除了它的一个子对象,Qt将会在父列表中自动删除。(需要记住的是Qt只是删除有父的对象,父对象还是需要手动删除的,还有就是那些用new申请的没有指定父的内存,一般情况下,在对话框里的子控件,许可器和布局管理器由Qt自己管理,其他还要程序员小心删除)
对于控件来讲,父对象还有一个意义:子控件在父对象的显示区域内显示。当父控件删除后,子控件不但在内存中被删除,它也同时在屏幕上消失。
在构造函数的最后两行,把QDialog的accept()函数连接到OK按钮的点击信号,把Cancel按钮的点击信号连接到reject()函数。这两个槽函数都关闭这个对话框,但是accept()返回 QDialog::Accepted(值为1),reject()返回值为QDialog::Rejected(值为0)。不同的返回值可以判断用户点击了那个按钮。
on_lineEdit_textChanged()槽函数控制Ok按钮的可用状态,通过编辑框中的输入字符,如果字符有效Ok按钮则有效,否则为不可用状态。QLineEdit::hasAcceptableInput()根据我们在构造函数中设置的许可器返回bool值。
这就完成了这个对话框,现在重写这个main.cpp文件:
#include <QApplication>
#include "gotocelldialog.h"
int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    GoToCellDialog *dialog = new GoToCellDialog;
    dialog->show();
    return app.exec();
}
编译这个程序(qmake -project; qmake gotocell.pro)然后运行。输入“A12”,Ok按钮变为可用。试着输入一行随意字符,观察许可器的反映。点击Cancel按钮关闭这个对话框。
使用qt Designer可以不改变源程序的情况下改变对话框的设计。如果对话框用C++代码编写,改变它将会很费力的。使用Qt Designer,uic自动重新生成源文件。不会浪费任何时间。 

你可能感兴趣的:(C++,UI,dialog,qt,makefile,Signal)