- QMenu和QMenuBar都是QWidget的派生类,按理说也没有什么特别的。只是真的想理清它,似乎还真的不是那么简单...
QMenuBar
在漫谈QWidget及其派生类(三)中,我们已经知道:QMenuBar和QStatusBar都是QWidget的派生类,所以我们可以像添加一个QPushButton一样将它们任意放置到另一个QWidget中。
继续看个别的例子:
#include <QtGui> int main(int argc, char *argv[]) { QApplication app(argc, argv); QMenuBar menubar; menubar.addAction("Action1"); menubar.addAction("Action1"); menubar.show(); return app.exec(); }
我们知道,QWidget本来就有一个成员函数:
void QWidget::addAction ( QAction * action )
而此处只不过是QMenuBar提供的重载版本:
QAction *QMenuBar::addAction(const QString &text) { QAction *ret = new QAction(text, this); addAction(ret); return ret; }
可是同样的东西,到了QMenuBar中,就显示出一种菜单栏的效果了呢?
不同的Widget,为什么不一样?我们都会自定义控件,所以都熟悉:
void QMenuBar::paintEvent(QPaintEvent *e) { ... for (int i = 0; i < d->actions.count(); ++i) { QAction *action = d->actions.at(i); QRect adjustedActionRect = d->actionRect(action); ...
便利它自身的actions列表,然后依次将它们paint出来。
然后当触发快捷键,或者在某个区域点击鼠标时:
void QMenuBar::mousePressEvent(QMouseEvent *e) { ... QAction *action = d->actionAt(e->pos()); .... d->setCurrentAction(action, true); ...
addMenu()
在继续之前,我们需要简单了解一下addAction和addMenu
QMenu *QMenuBar::addMenu(const QString &title) { QMenu *menu = new QMenu(title, this); addAction(menu->menuAction()); return menu; }
看,无非是创建一个QMenu,然后将该菜单对应的QAction添加进来了。而这个QMenu,稍候可以通过对应的QAction的成员函数
QMenu * QAction::menu () const
来获得
先看看QMenu,稍候再回来
QMenu
QMenu 也是一个QWidget的派生类
- 一个规规矩矩的矩形窗口(窗口类别 Qt::Popup)
- 通过setVisible(true)或show()使其显示出来
但是,为什么QMenu长成这个样子呢,一条一条的?
其实看了前面的QMenuBar,我们对QMenu为什么长这个样子,应该不会感到任何惊奇了。
先添加一些QAction,然后在paintEvent中遍历Action列表,并画出一个一个矩形条。
void QMenu::paintEvent(QPaintEvent *e) { //draw the items that need updating.. for (int i = 0; i < d->actions.count(); ++i) { QAction *action = d->actions.at(i); ...
当然,为了有点立体效果,或者说为了让QMenu和其他东西分开,还需要画边框。
上下文菜单
我们在一个Widget的某个位置上点击右键,或者通过键盘上某个键。会弹出上下文菜单。
这个菜单和我们在菜单栏看到的菜单其实没有区别,要说区别的话,就是需要我们自己:
- 设置菜单的位置
- 并让菜单显示出来
考虑:一般我们是如何使用上下文菜单的?
void Widget::contextMenuEvent(QContextMenuEvent *event) { QMenu menu(this); menu.addAction("A1"); menu.addAction("A2"); filemenu->exec(this->mapToGlobal(event.pos()));
- 创建菜单
- 添加action
- 显示菜单
如何显示?
- 记得刚接触这个东西时,在这个地方都是犯糊涂。这儿为什么不能直接用event中的位置,而是要map到全局坐标。
还是要从QMenu是QWidget的派生类说起。在前面我们提到了这样的内容:
- QWidget分:窗口(Window)和部件(Widget)两类。前者的位置是全局的或者说相对屏幕的,后者坐标是相对父部件的
QMenu 是 Window,所以位置坐标是全局的。而event坐标是当前窗口部件的。
- 显示一个QWidget,我们必须调用setVisible(true)/show()才行!!这儿用的确实exec(),我们类比QDialog的话,很容易才到答案。
其内部肯定调用了show(),而且还使用QEventLoop开启了局部的事件循环。
popup()
既然直接调用show()就可以让菜单显示出来,这个popup()又有何用?
void QMenu::popup(const QPoint &p, QAction *atAction) { ... }
它做的工作:
- 确保样式和字体被正确设置
- 调整几何尺寸(位置和大小)
- 调用show() 显示
而我们刚提到的exec()就是在popup的基础上加了个事件循环
QAction *QMenu::exec(const QPoint &p, QAction *action) { QEventLoop eventLoop; d->eventLoop = &eventLoop; popup(p, action); ...
回到菜单栏
了解了上下文菜单,我们可以回头看看菜单栏是如何控制菜单显示的。
一开始说了,
- 菜单对应的action被加入到了菜单栏
- 菜单栏在paintEvent绘制我们熟悉的那一栏东西
- 当我们点击鼠标时,菜单栏mousePressEvent中判断位置对应哪个action
- 这样就到了...
void QMenuBarPrivate::popupAction(QAction *action, bool activateFirst) { if(!action || !action->menu() || losePopupMode) return; activeMenu = action->menu(); .... activeMenu->popup(pos);
源码中,最长的一段就是在计算和调整菜单的位置(以确保菜单不会跑到屏幕外),即那个pos
相关阅读
作者:dbzhang800