QWidget
查帮助文档可知,QWidget类存在一个名叫contextMenuPolicy的成员,这是跟右键菜单有关的成员。由于继承,QWidget及其子类都能拥有右键菜单。
contextMenuPolicy的可取值如下,这些值定义了组件何时显示右键菜单:
Qt::NoContextMenu | 没有右键菜单,右键事件被组件的父组件处理 |
Qt::PreventContextMenu | 没有右键菜单,与NoContextMenu不同的是,组件上所有的右键事件都通过mousePressEvent()与mouseReleaseEvent()递交给自己, |
Qt::DefaultContextMen | 组件的 QWidget::contextMenuEvent() 被调用 |
Qt::ActionsContextMenu | 组件显示它的QWidget::actions() 来作为上下文菜单 |
Qt::CustomContextMenu | 组件发送QWidget::customContextMenuRequested() 信号 |
其中,能够有右键菜单的是Qt::DefaultContextMen(这也是默认值)、Qt::ActionsContextMenu、Qt::CustomContextMenu
为了实现右键菜单,先调用成员函数
void QWidget::setContextMenuPolicy (Qt::ContextMenuPolicy policy )
将成员变量contextMenuPolicy设为合适的值
通过事件处理
如果设置的值是Qt::DefaultContextMen(这也是默认值),这个值设置后组件用事件处理函数 QWidget::contextMenuEvent()来处理QContextMenuEvent事件,而QContextMenuEvent事件在用户执行与打开菜单有关的操作时触发(触发操作与平台相关,在windows中,鼠标右键的点击、菜单按钮的按下都会触发QContextMenuEvent事件),而事件处理函数 QWidget::contextMenuEvent()的默认实现是忽略QContextMenuEvent事件,因此,在此种情况下,我们应该重写 QWidget::contextMenuEvent()函数
//widget.h
#ifndefWIDGET_H
#defineWIDGET_H
#include<QtGui/QWidget>
#include<QMenu>
#include<QContextMenuEvent>
classWidget : public QWidget
{
Q_OBJECT
private:
QMenu menu;
privateslots:
void display();
protected:
void contextMenuEvent(QContextMenuEvent*e);
public:
Widget(QWidget *parent = 0);
~Widget();
};
#endif //WIDGET_H
//widget.cpp
#include"Widget.h"
#include<QCursor>
#include<QDebug>
Widget::Widget(QWidget*parent)
: QWidget(parent),menu(this)
{
menu.addAction("Evening");
connect(menu.actions()[0],SIGNAL(triggered()), this, SLOT(display()));
}
voidWidget::contextMenuEvent(QContextMenuEvent *e)
{
//menu.exec(e->globalPos());
menu.exec(QCursor::pos());
e->accept(); //事件已处理
}
voidWidget::display()
{
qDebug() << "Good Evening";
}
Widget::~Widget()
{
}
通过actions
如果设置的值是Qt::ActionsContextMenu,组件的actions会被作为上下文菜单,我们只需要为组件将action加入到组件的actions列表中
Widget::Widget(QWidget*parent)
:QWidget(parent)
{
addAction(new QAction("Evening",this));
setContextMenuPolicy(Qt::ActionsContextMenu);
}
通过槽函数
如果设置的值是Qt::CustomContextMenu,组件会发送QWidget::customContextMenuRequested() 信号,我们应该定义相应此信号的槽函数,在槽函数中实现上下文菜单,信号的原型: voidcustomContextMenuRequested (const QPoint & pos ),参数pos 表示组件收到的QContextMenuEvent事件的位置,这个位置一般在组件的坐标系内(位置为相对于组件右上角(0, 0)的坐标),但对于QAbstractScrollArea 及其子类,对应的是视口(viewport() )坐标
voidWidget::displaySlot(const QPoint & pos )
{
menu.exec(mapToGlobal(pos)); //需转化为对应的屏幕坐标,函数exec()需求
//menu.exec(QCursor::pos());
}