qt 中Dialog mainwindow widget 之间的联系和区别

本是古典 何须时尚 2022-05-28 11:15 315阅读 0赞
dialog、widget、mainwindow的区别
  1. 1. dialogexec函数,如果是dialog窗口,后边的窗口时不可选的;
  2. 2. widgetdialog都有show函数,如果通过这个函数显示这两种类型的窗口,则两个窗口都是可选的;
  3. 3. widget主要是在上面放置布局和控件;
  4. 4. mainwindow可以显示菜单,工具栏,状态栏、托盘等功能。

注意:

1. 查看帮助文档可以发现,setMenuBar、addToolBar、setCentralWidget、setStatusBar几种行为只有QMainWindow类具有。

因此,在继承自QWidget类的用户类中无法创建菜单栏等几种行为。

这就是QMainWindow和QWidget的主要区别。

2. QWidget继承于QObject和QPaintDevice,QDialog和QMainWindow则继承于QWidget,QDialog、QMainWindow两者之间没有直接关系。

QWidget

QWidget类是所有用户界面对象的基类。

窗口部件是用户界面的一个原子:它从窗口系统接收鼠标、键盘和其它事件,并且将自己的表现形式绘制在屏幕上。每一个窗口部件都是矩形,并且它们按Z轴顺序排列。一个窗口部件可以被它的父窗口部件或者它前面的窗口部件盖住一部分。

QWidget有很多成员函数,但是它们中的一些有少量的直接功能:例如,QWidget有字体属性,但是自己从来不用。为很多继承它的子类提供了实际的功能,比如QLabel、QPushButton、QCheckBox等等。

没有父窗体的小部件始终是一个独立的窗口(顶级窗口部件)。非窗口的小部件为子部件,它们在父窗口中显示。Qt中大多数部件主要被用作子部件。例如:可以显示一个按钮作为顶层窗口,但大多数人更喜欢将按钮内置于其它部件,如QDialog。

QDialog

QDialog类是对话框窗口的基类。

对话框窗口是一个顶级窗体,主要用于短期任务以及和用户进行简要通讯。QDialog可以是模式的也可以是非模式的。QDialog支持扩展性并且可以提供返回值。它们可以有默认按钮。QDialog也可以有一个QSizeGrip在它的右下角,使用setSizeGripEnabled()。

注意:QDialog(以及其它使用Qt::Dialog类型的widget)使用父窗口部件的方法和Qt中其它类稍微不同。对话框总是顶级窗口部件,但是如果它有一个父对象,它的默认位置就是父对象的中间。它也将和父对象共享工具条条目

  1. #include "mydialog.h"
  2. mydialog::mydialog(QDialog *parent):QDialog(parent) {
  3. QPushButton *button = new QPushButton( "button" , this ) ;
  4. connect( button , SIGNAL(clicked()) , this , SLOT(slotButtonClick()) ) ;
  5. }
  6. void mydialog::slotButtonClick() {
  7. #if 0
  8. /*dialog和widget的区别,exec和show的区别而已*/
  9. QDialog *dlg = new QDialog ;
  10. QPushButton *button = new QPushButton("close" , dlg ) ;
  11. connect( button , SIGNAL(clicked()) , dlg , SLOT(reject()) ) ;
  12. int ret = dlg->exec( ) ;
  13. //通过exec显示出来的窗口称为,模块对话框
  14. // 在模块对话框中,exec有自己的消息循环,并且把app的消息循环接管了
  15. // 如果Dialog是通过exec来显示,那么可以通过accept或者reject来关闭窗口
  16. // 如果Dialog是通过show来显示,那么可以通过close来关闭窗口,这个和QWidget一样的
  17. // 有许多特殊的dailog:文件选择,MessageBox,颜色选择,字体选择,打印预览,打印
  18. if( ret == QDialog::Accepted )
  19. qDebug()<<"accepted" ;
  20. else if( ret== QDialog::Rejected )
  21. qDebug()<<"rejected" ;
  22. #endif
  23. #if 0
  24. /*文件选择:这个窗口可以选择保存文件的名称,然后将路径+名称返回,我们就可以根据返回路径名来保存文件。*/
  25. QString strFilename = QFileDialog::getSaveFileName(NULL,
  26. "Select file for save",
  27. _strDir,
  28. "pic file (*.png *.jpg)");
  29. #endif
  30. #if 0
  31. /*文件选择:选择要打开的文件名(绝对路劲);我们就可以根据这个文件路径来打开相应的文件*/
  32. QString strFilename = QFileDialog::getOpenFileName(NULL,
  33. "Select file for open",
  34. _strDir,
  35. "pic file (*.png *.jpg)");
  36. #endif
  37. #if 0
  38. QString strFilename = QFileDialog::getExistingDirectory();
  39. if(strFilename.isEmpty())
  40. {
  41. qDebug() << "select none";
  42. return;
  43. }
  44. qDebug() << strFilename;
  45. QFileInfo fileInfo(strFilename);
  46. _strDir = fileInfo.filePath();
  47. qDebug() << _strDir;
  48. #endif
  49. //do something for io ... ...
  50. //上面的选择将绝对路径名都给拿下来了,如果要进行保存,不是很容易吗!
  51. QPixmap pixmap(this->size()) ;
  52. QPainter painter(&pixmap) ;
  53. this->render( &painter ) ;
  54. pixmap.save(strFilename) ;
  55. #if 0
  56. /*颜色选择对话框:可以选择相应的颜色;然后将选择的颜色返回,这样我们就可以操作了*/
  57. QColorDialog colorDia ;
  58. colorDia.exec() ;
  59. QColor c = colorDia.selectedColor() ;
  60. #endif
  61. #if 0
  62. /*字体选择对话框:可以选择字体;然后将选择的字体信息返回,我们同样可以用这些信息来设置相应的值*/
  63. QFontDialog fontDia ;
  64. fontDia.exec() ;
  65. QFont font = fontDia.selectedFont() ;
  66. #endif
  67. #if 0
  68. /*这个也是弹窗对话框,不过只是简单的选择以下枚举中的值,可以尝试下效果*/
  69. int ret = QMessageBox::question(this, "????", "realy do .......",
  70. QMessageBox::Yes| QMessageBox::No|
  71. QMessageBox::YesAll| QMessageBox::NoAll);
  72. if(ret == QMessageBox::Yes)
  73. {
  74. qDebug() << "user select yes";
  75. }
  76. if(ret == QMessageBox::No)
  77. {
  78. qDebug() << "user select no";
  79. }
  80. #endif
  81. }
  82. void mydialog::paintEvent( QPaintEvent *ev ) {
  83. }
  84. mydialog::~mydialog(void) { }
模式对话框

阻塞同一应用程序中其它可视窗口输入的对话框。模式对话框有自己的事件循环,用户必须完成这个对话框中的交互操作,并且关闭了它之后才能访问应用程序中的其它任何窗口。模式对话框仅阻止访问与对话相关联的窗口,允许用户继续使用其它窗口中的应用程序。

显示模态对话框最常见的方法是调用其exec()函数,当用户关闭对话框,exec()将提供一个有用的返回值,并且这时流程控制继续从调用exec()的地方进行。通常情况下,要获得对话框关闭并返回相应的值,我们连接默认按钮,例如:”确定”按钮连接到accept()槽,”取消”按钮连接到reject()槽。另外我们也可以连接done()槽,传递给它Accepted或Rejected。

非模式对话框

和同一个程序中其它窗口操作无关的对话框。在文字处理中的查找和替换对话框通常是非模式的,允许用户同时与应用程序的主窗口和对话框进行交互。调用show()来显示非模式对话框,并立即将控制返回给调用者。

如果隐藏对话框后调用show()函数,对话框将显示在其原始位置,这是因为窗口管理器决定的窗户位置没有明确由程序员指定,为了保持被用户移动的对话框位置,在closeEvent()中进行处理,然后在显示之前,将对话框移动到该位置。

半模式对话框

调用setModal(true)或者setWindowModality(),然后show()。有别于exec(),show() 立即返回给控制调用者。

对于进度对话框来说,调用setModal(true)是非常有用的,用户必须拥有与其交互的能力,例如:取消长时间运行的操作。如果使用show()和setModal(true)共同执行一个长时间操作,则必须定期在执行过程中调用QApplication::processEvents(),以使用户能够与对话框交互(可以参考QProgressDialog)。

QMainWindow

QMainWindow类提供一个有菜单条、工具栏、状态条的主应用程序窗口(例如:开发Qt常用的IDE-Visual Studio、Qt Creator等)。

一个主窗口提供了构建应用程序的用户界面框架。Qt拥有QMainWindow及其相关类来管理主窗口。

QMainWindow拥有自己的布局,我们可以使用QMenuBar(菜单栏)、QToolBar(工具栏)、QStatusBar(状态栏)以及QDockWidget(悬浮窗体),布局有一个可由任何种类小窗口所占据的中心区域。

  1. #include "MyWindow.h"
  2. MyWindow::MyWindow(QMainWindow *parent):QMainWindow( parent ) {
  3. /*menuBar菜单栏,菜单menu*/
  4. QMenuBar *menuBar = this->menuBar() ;
  5. _menu = menuBar->addMenu( "&File" ) ;
  6. QMenu *edit = menuBar->addMenu( "&Edit" ) ;
  7. /*menu菜单中的选项action*/
  8. QAction *openAction = _menu->addAction( "&Open"
  9. , this , SLOT(slotOpen()) , QKeySequence::Open ) ;
  10. QAction *saveAction = _menu->addAction( "&Save"
  11. , this , SLOT(slotOpen()) , QKeySequence::Save ) ;
  12. _menu->addSeparator() ;//添加分界线
  13. QAction *closeAction = _menu->addAction( "&Exit"
  14. , this , SLOT(close()) , QKeySequence::Close ) ;
  15. /*toolBar工具栏*/
  16. QToolBar *toolBar = this->addToolBar( "mimi" ) ;
  17. toolBar->addAction( openAction ) ;
  18. toolBar->addAction( saveAction ) ;
  19. toolBar->addAction( closeAction ) ;
  20. /*statusBar状态栏*/
  21. QStatusBar *statusBar = this->statusBar() ;
  22. QLabel *label ;
  23. statusBar->addWidget( label = new QLabel("Ok") ) ;
  24. label->setText( "<font color=blue>XXXXX... ...</font>" ) ;
  25. /*上面的三种栏介绍完之后,剩下的窗口区域就是CentralWidget
  26. **如果将widget直接add到mainwindow这个窗口的话,
  27. **toolbar是会跟添加进来的widget重叠的*/
  28. MyView *view = new MyView ;
  29. this->setCentralWidget( view ) ;
  30. /*最后就是window系统右下脚的托盘:system tray icon*/
  31. QSystemTrayIcon *icon = new QSystemTrayIcon ;
  32. icon->setIcon( QIcon("./1.png") ) ;//图标
  33. icon->setToolTip( "luck dog" ) ;//鼠标滑过提示文字
  34. icon->show() ;//展示在右下角
  35. icon->setContextMenu( _menu ) ;//右击出现的菜单
  36. this->connect( icon , SIGNAL( slotActivated(QSystemTrayIcon::ActivationReason) )
  37. , this , SLOT(slotActivated(QSystemTrayIcon::QSystemTrayIcon::ActivationReason)) ) ;
  38. this->installEventFilter(this);
  39. }
  40. void MyWindow::slotActivated(QSystemTrayIcon::ActivationReason reason){
  41. /*这个没成功*/
  42. if( reason==QSystemTrayIcon::Trigger ) {
  43. if( this->isHidden() )
  44. this->show() ;
  45. else
  46. this->hide() ;
  47. }
  48. }
  49. bool MyWindow::eventFilter(QObject *o, QEvent *e)
  50. {
  51. /*实现什么功能呢?*/
  52. if(o == (QObject*)this && e->type() == QEvent::Close)
  53. {
  54. return true;
  55. }
  56. return QMainWindow::eventFilter(o, e);
  57. }
  58. void MyWindow::slotOpen() {
  59. QString fileName = QFileDialog::getOpenFileName() ;
  60. qDebug()<<fileName ;
  61. /*将打开的文件中的内容显示在窗口上... ...*/
  62. }
  63. bool MyWindow::event(QEvent *ev)
  64. {
  65. qDebug() << ev;
  66. if(ev->type() == QEvent::Close) {
  67. return false;
  68. }
  69. /*怎么弄才能实现窗口关闭,托盘还在?*/
  70. return QMainWindow::event(ev);
  71. }
  72. void MyWindow::paintEvent( QPaintEvent * ) {
  73. QPainter painter(this) ;
  74. painter.drawPixmap( QPoint(0,0) , QPixmap("./1.png") ) ;
  75. }
  76. void MyWindow::mousePressEvent( QMouseEvent *ev ) {
  77. if( ev->button() == Qt::RightButton ) {
  78. _menu->exec( QCursor::pos() ) ;
  79. }
  80. }
  81. MyWindow::~MyWindow(void) { }
使用原则

如果需要嵌入到其他窗体中,则基于QWidget创建。
如果是顶级对话框,则基于QDialog创建。

如果是主窗体,则基于QMainWindow创建。

参考链接:

https://blog.csdn.net/liang19890820/article/details/50533262

https://blog.csdn.net/a2604539133/article/details/73920696

发表评论

表情:
评论列表 (有 0 条评论,315人围观)

还没有评论,来说两句吧...

相关阅读

    相关 DPL,RPL,CPL 之间联系区别

    这篇文章写的太好了!!这是保护模式完整的权限检查,其实当今的操作系统也不会弄的这么复杂…… 特权级是保护模式下一个重要的概念,CPL,RPL和DPL是其中的核心概念,查阅资料