-
在漫谈QWidget及其派生类(二)一文的最后我们简单提到了QMainWindow的一些东西。但是内容太少了,本文中我们换个角度看看QMainWindow,希望大家能了解一点:与QWidget相比,其派生类QMainWindow也不过如此
例子一
上一篇太乱了,应该主要是缺少例子。我们这次改一改,直接上个例子看看:
#include <QtGui/QtGui> class MainWindow:public QMainWindow { public: MainWindow() { menuBar()->addMenu("&File"); menuBar()->addMenu("&Help"); statusBar()->addWidget(new QLabel("Hello from Dbzhang800...")); statusBar()->addPermanentWidget(new QLabel("2011-09-03")); setCentralWidget(new QTextEdit); } }; int main(int argc, char *argv[]) { QApplication a(argc, argv); MainWindow w; w.show(); return a.exec(); }
这个例子实在没什么可说的,对不??
- 菜单栏
- 状态栏
- 中心窗体
这么常规的东西,有必要写在这儿么?如果只是这样,肯定没有必要了。可是,如果我说:不用QMainWindow,直接用QWidget可以轻易实现同样的效果。是不是值得聊聊了?
HMainWindow
其实上面你看到的截图,是我用的HMainWindow生成的。使用HMainWindow的话,你只需要将前面的 QMainWindow ==> HMainWindow,是不是很简单?
- HMainWindow 的定义:
class HMainWindow:public QWidget { public: HMainWindow(QWidget * parent=0) :QWidget(parent, Qt::Window), m_menuBar(0), m_statusBar(0), m_central(0) { m_vbox = new QVBoxLayout(this); m_vbox->setContentsMargins(0,0,0,0); m_vbox->setSpacing(0); } void setCentralWidget(QWidget * w) { if (!m_central) { m_central = w; m_vbox->insertWidget(0, w, 1); } } QMenuBar * menuBar() { if (!m_menuBar) { m_menuBar = new QMenuBar(this); m_vbox->setMenuBar(m_menuBar); } return m_menuBar; } QStatusBar * statusBar() { if (!m_statusBar) { m_statusBar = new QStatusBar(this); m_vbox->addWidget(m_statusBar); } return m_statusBar; } private: QMenuBar * m_menuBar; QStatusBar * m_statusBar; QWidget * m_central; QVBoxLayout * m_vbox; };
我们用这个来模拟一个QMainWindow,这个类很简单:
HMainWindow() |
构造函数。创建了一个layout(用来放置菜单栏、状态栏、中心窗体);传递给基类QWidget一个Qt::Window标记,当然,对我们这个例子这个东西有没有无所谓。 |
menuBar() |
第一次调用它时,会生成一个QMenuBar,并加入layout |
statusBar() |
同上,生成状态栏并加入layout |
setCentralWidget() |
除状态栏、菜单栏外的区域,总要放个东西吧? |
和 QMainWindow比起来,我们这个HMainWindow实在是太简易了。因为QMainWindow所使用的QMainWindowLayout(对,有这么一个东西,是个私有类) 比 QVBoxLayout复杂太多了。
不过呢,思想是一样的。一旦理解了这个,也就掌握 QMainWindow 的那点小把戏了。
例子二
不少网友抱怨:
- 覆盖基类的paintEvent函数,结果画的东西全都看不到
- 覆盖基类的mousePressEvent函数,结果收不到鼠标事件
- 覆盖基类的****Event函数,结果...
- ...
//class MainWindow:public QMainWindow class MainWindow:public HMainWindow { public: MainWindow() { //... } protected: protected: void mousePressEvent(QMouseEvent *) { //... } void paintEvent(QPaintEvent *) { //... } };
有了前面的基础,想想是不是很简单?QMainWindow只不过是一个带layout的Widget,上面放置了菜单栏、状态栏、中心窗体这些子Widget。挡住了我们的QMainWindow
例子三
如果没有意识到QMainWindow的中心窗体的作用,很容易犯下面的错误,你能找到答案么?
- 创建一个子Widget,比如按钮。不设置为(或添加到)中心窗体
//class MainWindow:public QMainWindow class MainWindow:public HMainWindow { public: MainWindow() { //... QPushButton * btn = new QPushButton(this); } };
什么现象?哈哈,其实 很有意思:
创建了一个按钮,回想上一节?几何尺寸是如何改变的?只能通过setGeometry或resize或move。这些我们都没使用。于是默认大小、默认位置(0,0)。于是,左上角出现一个按钮!!
可是,左上角一般是什么东西呢?菜单栏嘛?
菜单栏和按钮同时出现在左上角??可能么?
有何不可呢,只不过两个东西必然有一个在上一个在下!!谁上谁下,和什么有关?你可以自己试试看(考虑到文章长度,本文不涉及widget的堆放层次的控制)。
例子四
如果没有意识到QMainWindow这个东西,其实已经有了一个layout,很容易犯下面的问题,你能找到答案么?
- layout 不起作用,按钮依然出现在左上角!!
//class MainWindow:public QMainWindow class MainWindow:public HMainWindow { public: MainWindow() { QHBoxLayout * hbox = new QHBoxLayout; hbox->addWidget(new QPushButton(this)); setLayout(hbox); } };
似乎不少人对此不解?我用其他Widget都是这么用的啊?创建layout,添加其他widget,设置layout,
怎么失败了呢??
看看Manual,知道答案了吧?
void QWidget::setLayout ( QLayout * layout ) If there already is a layout manager installed on this widget, QWidget won't let you install another.
例子五
真不想写了,给大家个链接,感兴趣的可以看看。QMainWindow一旦概念不清(或用法不对),还会有什么问题
本文完。
希望本文的内容对大家有帮助。但本文不能取代Manual,用QMainWindow,一定要认真看QMainWindow的Manual。
作者:dbzhang800