QT:对话框Dialog

1 对话框的基本概念

  • 对话框是 GUI 程序中不可或缺的组成部分。很多不能或者不适合放入主窗口的 功能组件都必须放在对话框中设置。对话框通常会是一个顶层窗口,出现在程序最上层,用于实现短期任务或者简洁的用户交互。
  • Qt 中使用QDialog类实现对话框。就像主窗口一样,我们通常会设计一个类继承 QDialog。QDialog(及其子类,以及所有Qt::Dialog类型的类)的对于其 parent 指针都有额外的解释:如果 parent 为 NULL,则该对话框会作为一个 顶层窗口,否则则作为其父组件的子对话框(此时,其默认出现的位置是 parent 的中 心)。顶层窗口与非顶层窗口的区别在于,顶层窗口在任务栏会 有自己的位置,而非顶层窗口则会共享其父组件的位置。
  • 对话框分为模态对话框和非模态对话框.
    1.模态对话框,就是会阻塞同一应用程序中其它窗口的输入。 模态对话框很常见,比如“打开文件”功能。你可以尝试一下记事本的打开文 件,当打开文件对话框出现时,我们是不能对除此对话框之外的窗口部分进行操 作的。
    2.与此相反的是非模态对话框,例如查找对话框,我们可以在显示着查找对话框 的同时,继续对记事本的内容进行编辑。

2 标准对话框

  • 所谓标准对话框,是 Qt 内置的一系列对话框,用于简化开发。事实上,有很多对话框都是通用的,比如打开文件、设置颜色、打印设置等。这些对话框在所有
  • 程序中几乎相同,因此没有必要在每一个程序中都自己实现这么一个对话框。 Qt 的内置对话框大致分为以下几类:
    QColorDialog:选择颜色;
    QFileDialog: 选择文件或者目录;
    QFontDialog: 选择字体;
    QInputDialog:允许用户输入一个值,并将其值返回;
    QMessageBox: 模态对话框,用于显示信息、询问问题等; QPageSetupDialog:为打印机提供纸张相关的选项;
    QPrintDialog: 打印机配置;
    QPrintPreviewDialog:打印预览;
    QProgressDialog: 显示操作过程。

代码示例:

#include "widget.h"
#include <QFileDialog>
#include <QString>
#include <QDebug>
#include <QStringList>
#include <QDialog>
Widget::Widget(QWidget *parent)
    : QWidget(parent)
{
    button = new QPushButton("选择文件",this);
    this->resize(600,500);
    button->resize(100,100);
    button->move(250,100);
    connect(button,&QPushButton::clicked,this,[](){
        //QDebug str = QFileDialog::getOpenFileName();
        QStringList str = QFileDialog::getOpenFileNames();
        qDebug()<<str;
    });   //标准对话框
}

Widget::~Widget()
{
}

3 模态对话框

  • 模态对话框 lQt 有两种级别的模态对话框:
    应用程序级别的模态 当该种模态的对话框出现时,用户必须首先对对话框进行交互,直到关闭对话 框,然后才能访问程序中其他的窗口。

    窗口级别的模态 该模态仅仅阻塞与对话框关联的窗口,但是依然允许用户与程序中其它窗口交 互。窗口级别的模态尤其适用于多窗口模式。 一般默认是应用程序级别的模态。

  • 在下面的示例中,我们调用了exec()将对话框显示出来,因此这就是一个模态对话框。当对话框出现时,我们不能与主窗口进行任何交互,直到我们关闭了该对话框。
    QDialog dialog;
    dialog.setWindowTitle(tr(“Hello, dialog!”));
    dialog.exec();

代码示例:

#include "widget.h"
#include <QFileDialog>
#include <QString>
#include <QDebug>
#include <QStringList>
#include <QDialog>
Widget::Widget(QWidget *parent)
    : QWidget(parent)
{
    button1 = new QPushButton("模块对话框按钮",this);
    this->resize(600,500);
    button1->resize(100,100);
    button1->move(250,200);
    connect(button1,&QPushButton::clicked,this,[](){
       QDialog dialog;//定义一个对话框
       dialog.setWindowTitle("模态对话框");//设置对话框标题
       dialog.exec();//显示对话框,并阻塞
    });   //模态对话框
}

Widget::~Widget()
{
}

4 非模态对话框

下面我们试着将exec()修改为show(),看看非模态对话框:

QDialog dialog(this); 
dialog.setWindowTitle(tr("Hello, dialog!")); 
dialog.show();

是不是事与愿违?对话框竟然一闪而过!这是因为,show()函数不会阻塞当前线程,对话框会显示出来,然后函数立即返回,代码继续执行。注意,dialog 是建立在栈上的,show()函数执行完,dialog 超 出作用域被析构,因此对话框消失了。知道了原因就好改了,我们将 dialog 改 成堆上建立,当然就没有这个问题了:

QDialog *dialog = new QDialog; 
dialog­>setWindowTitle(tr("Hello, dialog!")); 
dialog­>show();

如果你足够细心,应该发现上面的代码是有问题的:当你频繁点击按钮的时候,dialog 存在内存泄露! dialog 使用 new 在堆上分配空间,却一直没有 delete。解决方案也很简单:

dialog­>setAttribute(Qt::WA_DeleteOnClose);

setAttribute()函数设置对话框关闭时,自动销毁对话框。

代码示例:

#include "widget.h"
#include <QFileDialog>
#include <QString>
#include <QDebug>
#include <QStringList>
#include <QDialog>
Widget::Widget(QWidget *parent)
    : QWidget(parent)
{
    button2 = new QPushButton("非模块对话框按钮",this);
    this->resize(600,500);
    button2->resize(100,100);
    button2->move(350,100);
    connect(button2,&QPushButton::clicked,this,[](){
       QDialog *dialog = new QDialog;//定义一个对话框
       dialog->setWindowTitle("非模态对话框");//设置对话框标题
       dialog->setAttribute(Qt::WA_DeleteOnClose);//关闭窗口后自动释放dialog
       dialog->show();//显示对话框
    });   //非模态对话框
}

Widget::~Widget()
{
}

5 消息对话框

QMessageBox用于显示消息提示。我们一般会使用其提供的几个 static 函数:

  • 显示关于对话框。
 void about(QWidget * parent, const QString & title, const QString & text)  
这是一个最简单的对话框,其标题是 title,内容是 text,父窗口是 parent。对话框只有一个 OK 按钮。
  • 显示关于 Qt 对话框。该对话框用于显示有关 Qt 的信息。
void aboutQt(QWidget * parent, const QString & title = QString())
  • 显示严重错误对话框。
StandardButton critical(QWidget * parent, const QString
   & title, const QString & text, StandardButtons buttons = Ok,
   StandardButton defaultButton = NoButton)
  这个对话框将显示一个红色的错误符号。我们可以通过buttons 参数指明其显 示的按钮。默认情况下只有一个 Ok 按钮,我们可以使用StandardButtons类型 指定多种按钮。
  • 与QMessageBox::critical()类似,不同之处在于这个对话框提供一个普通信 息图标。
StandardButton information(
          QWidget * parent, 
          const QString & title, 
          const QString & text, 
          StandardButtons buttons = Ok, 
          StandardButton defaultButton = NoButton)
  • 与QMessageBox::critical ()类似,不同之处在于这个对话框提供一个问号图 标,并且其显示的按钮是“是”和“否”。
StandardButton question(
             QWidget * parent, 
             const QString & title, 
             const QString & text, 
             StandardButtons buttons = StandardButtons( Yes | No ),              StandardButton defaultButton = NoButton)
  • 与QMessageBox::critical()类似,不同之处在于这个对话框提供一个黄色叹 号图标。
StandardButton warning(
            QWidget * parent, 
            const QString & title, 
            const QString & text, 
            StandardButtons buttons = Ok, 
            StandardButton defaultButton = NoButton)

我们使用QMessageBox::question()来询问一个问题。

  • 这个对话框的父窗口是 this。 QMessageBox是QDialog的子类,这意味着它的初始显示位置将会是在 parent 窗
    口的中央。
  • 第二个参数是对话框
  • 第三个参数是我们想要显示的内容。
  • 第四个参数是关联的按键类型,我们可以使用或运算符(|)指定对话框应该 出现的按钮。比如我们希望是一个 Yes 和一个 No。
  • 最后一个参数指定默认选择的按钮。

这个函数有一个返回值,用于确定用户点击的是哪一个按钮。按照我们的写法, 应该很容易的看出,这是一个模态对话框,因此我们可以直接获取其返回值。 QMessageBox类的 static 函数优点是方便使用,缺点也很明显:非常不灵活。 我们只能使用简单的几种形式。为了能够定制QMessageBox细节,我们必须使用 QMessageBox的属性设置 API。如果我们希望制作一个询问是否保存的对话框, 我们可以使用如下的代码:

     QMessageBox msgBox; 
     msgBox.setText(tr("The document has been modified.")); 
     msgBox.setInformativeText(tr("Do you want to save your changes?"));
     msgBox.setDetailedText(tr("Differences here..."));
     msgBox.setStandardButtons(QMessageBox::Save |QMessageBox::Discard | QMessageBox::Cancel);
     msgBox.setDefaultButton(QMessageBox::Save); 
     int ret = msgBox.exec(); 
     switch (ret)
      {
       case QMessageBox::Save:qDebug() << "Save document!"; break; 
       case QMessageBox::Discard: qDebug() << "Discard changes!"; break; 
       case QMessageBox::Cancel: qDebug() << "Close document!"; break; 
      } 
   msgBox 是一个建立在栈上的QMessageBox实例。我们设置其主要文本信息 为“The document has been modified.”,informativeText 则是会在对话框 中显示的简单说明文字。下面我们使用了一个detailedText,也就是详细信息, 当我们点击了详细信息按钮时,对话框可以自动显示更多信息。我们自己定义的 对话框的按钮有三个:保存、丢弃和取消。然后我们使用了exec()是其成为一个 模态对话框,根据其返回值进行相应的操作。

代码示例:

#include "widget.h"
#include <QFileDialog>
#include <QString>
#include <QDebug>
#include <QStringList>
#include <QDialog>
#include <QMessageBox>
Widget::Widget(QWidget *parent)
    : QWidget(parent)
{
    button3 = new QPushButton("关于对话框按钮",this);
    this->resize(600,500);
    button3->resize(100,100);
    button3->move(350,200);
    connect(button3,&QPushButton::clicked,this,[=](){
       QMessageBox::about(this,"标题","内容");
    });   //关于对话框
    button4 = new QPushButton("询问对话框按钮",this);
    this->resize(600,500);
    button4->resize(100,100);
    button4->move(350,300);
    connect(button4,&QPushButton::clicked,this,[=](){
       QMessageBox::question(this,"惠大每日问答","zex是憨批?");
    });   //询问对话框
}

Widget::~Widget()
{
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值