正如@mvidelgauz所指出的那样,QAction
是从可能触发该操作的输入设备中抽象出来的。尽管如此,如果该操作在您的GUI中使用,则会有一个或多个关联小部件:工具栏中的工具按钮,在菜单栏中的条目等等。这些小部件像任何其他小部件一样,因此它们接收事件,这些事件可以通过使用installEventFilter和eventFilter进行过滤。这两种方法都继承自QObject
,因此它们几乎存在于任何Qt类中。例如,让我们创建一个带有QMainWindow和QAction的应用程序,称为actionTest
。然后,通过重写主窗口的eventFilter
方法,将主窗口本身转换为actionTest
关联小部件的操作过滤器:
bool eventFilter(QObject *obj, QEvent *ev) {
if(ev->type() == QEvent::MouseButtonPress) {
QMouseEvent *mev = static_cast<QMouseEvent*>(ev);
if(mev->button() == Qt::LeftButton) {
qDebug() << "Left button!";
}
if(mev->button() == Qt::RightButton) {
qDebug() << "Right button!";
}
}
return QMainWindow::eventFilter(obj, ev);
}
接下来,我们需要为所有被观察的对象(在这种情况下是小部件)安装事件过滤器对象。让我们在主窗口构造函数中完成这个步骤:
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
ui->setupUi(this);
for(auto wgtPtr : ui->actionTest->associatedWidgets()) {
wgtPtr->installEventFilter(this);
}
}
最后,添加一个用于处理 triggered()
信号的插槽:
void on_actionTest_triggered() {
qDebug() << "Action triggered!";
}
现在,如果您使用鼠标左键单击操作菜单条目,它将打印。
Left button!
Action triggered!
对于右键,结果将是
Right button!
Action triggered!
请注意,小部件事件过滤总是在触发
triggered()
信号之前执行。
上面的代码只是一个例子,
MainWindow
类不是托管
eventFilter
方法的最佳位置。在实际代码中,您可以选择:
- 为
QAction
小部件创建专用的 QObject
子类进行事件过滤。
- 对
QAction
进行子类化并重写其 eventFilter
方法。在这种情况下,您可以仅在 QAction
子类对象中保存 QMouseEvent::button()
的结果,并稍后在 triggered()
信号处理程序中使用它。有一个小问题是 Qt creator(至少到 v3.2.1)不允许您在其表单设计器中“推广” QAction
,因此您需要在窗口构造函数中手动将操作添加到菜单中。
- 对
QMenu
、QToolBar
等进行子类化,使它们成为操作过滤器?我不知道这比前两种变体更好在哪里。
另请参阅Qt事件系统文档。
让我们澄清第二种情况。假设继承自QAction
的类名为MyAction
。为了使其工作,您需要将MyAction
对象安装为它们自己(更具体地说是它们的小部件)的过滤器。您需要在小部件创建后进行安装,因此在MyAction
构造函数中安装过滤器可能过早并导致崩溃。更好的过滤器安装位置是拥有MyAction
对象的类的构造函数。通常是窗口或小部件类。所以只需添加:
for(auto wgtPtr : ui->myActionObject->associatedWidgets()) {
wgtPtr->installEventFilter(ui->myActionObject);
}
在调用
ui->setupUi(this)
后,将代码添加到窗口构造函数中。此代码与上面的示例类似,但我们使用
ui->myActionObject
而不是
this
对象作为过滤器。
eventFilter
方法中打印一些内容到qDebug()
,以确保它正常工作。 - SergeyQSystemTrayIcon
上使用它时,这个能用吗?我的应用程序没有GUI,只有在QSystemTrayIcon
中的QActions
。 - user6285273