我想知道QAction是被左键还是右键点击的。

7

我在 QMenu 中有一个 QAction。当 QAction 被触发时,我想知道是哪个按钮触发了它。

connect(YourAction, SIGNAL(triggered()), this, SLOT(actionclicked()));

void MainWindow::actionclicked(QMouseEvent *e)
{
    if (e->buttons() == Qt::RightButton) 
}

我无法像这样做,因为triggered()没有这样的参数。

2个回答

4

正如@mvidelgauz所指出的那样,QAction是从可能触发该操作的输入设备中抽象出来的。尽管如此,如果该操作在您的GUI中使用,则会有一个或多个关联小部件:工具栏中的工具按钮,在菜单栏中的条目等等。这些小部件像任何其他小部件一样,因此它们接收事件,这些事件可以通过使用installEventFiltereventFilter进行过滤。这两种方法都继承自QObject,因此它们几乎存在于任何Qt类中。例如,让我们创建一个带有QMainWindow和QAction的应用程序,称为actionTest。然后,通过重写主窗口的eventFilter方法,将主窗口本身转换为actionTest关联小部件的操作过滤器:

bool eventFilter(QObject *obj, QEvent *ev) {
    //Catch only mouse press events.
    if(ev->type() == QEvent::MouseButtonPress) {
        // Cast general event to mouse event.
        QMouseEvent *mev = static_cast<QMouseEvent*>(ev);
        // Show which button was clicked.
        if(mev->button() == Qt::LeftButton) {
            qDebug() << "Left button!";
        }
        if(mev->button() == Qt::RightButton) {
            qDebug() << "Right button!";
        }
    }
    // In this example we just showed the clicked button. Pass the event
    // for further processing to make QAction slots work.
    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 方法的最佳位置。在实际代码中,您可以选择:
  1. QAction 小部件创建专用的 QObject 子类进行事件过滤。
  2. QAction 进行子类化并重写其 eventFilter 方法。在这种情况下,您可以仅在 QAction 子类对象中保存 QMouseEvent::button() 的结果,并稍后在 triggered() 信号处理程序中使用它。有一个小问题是 Qt creator(至少到 v3.2.1)不允许您在其表单设计器中“推广” QAction,因此您需要在窗口构造函数中手动将操作添加到菜单中。
  3. QMenuQToolBar 等进行子类化,使它们成为操作过滤器?我不知道这比前两种变体更好在哪里。

另请参阅Qt事件系统文档

让我们澄清第二种情况。假设继承自QAction的类名为MyAction。为了使其工作,您需要将MyAction对象安装为它们自己(更具体地说是它们的小部件)的过滤器。您需要在小部件创建后进行安装,因此在MyAction构造函数中安装过滤器可能过早并导致崩溃。更好的过滤器安装位置是拥有MyAction对象的类的构造函数。通常是窗口或小部件类。所以只需添加:

for(auto wgtPtr : ui->myActionObject->associatedWidgets()) {
    wgtPtr->installEventFilter(ui->myActionObject);
}

在调用ui->setupUi(this)后,将代码添加到窗口构造函数中。此代码与上面的示例类似,但我们使用ui->myActionObject而不是this对象作为过滤器。

我可以在我使用作为QAction的类中创建eventFilter吗?我已经有了一个继承自QAction的类,并在我的应用程序中使用它。我重写了eventFilter方法,但是没有任何反应。我需要在某个地方安装它吗? - user6285273
@S.llous 当然可以。我在我的答案中为情况2添加了澄清,这应该可以使您的代码正常工作。尝试从您的eventFilter方法中打印一些内容到qDebug(),以确保它正常工作。 - Sergey
当我想在QSystemTrayIcon上使用它时,这个能用吗?我的应用程序没有GUI,只有在QSystemTrayIcon中的QActions - user6285273
@S.llous 这些操作是否在菜单中,当您单击托盘图标时会出现?这应该可以工作。 - Sergey

3
按设计,triggered()不能有这个参数,因为它本身不一定是鼠标事件的结果:
当用户激活操作时,例如单击菜单选项、工具栏按钮或按下操作的快捷键组合,或者调用了trigger()时,会发出此信号。
如果需要QMouseEvent作为参数,则需要连接到鼠标事件。实际上,在Qt中,当(但不仅限于我在文档中强调的情况)框架从菜单接收到鼠标事件时,Qt本身会发出triggered()。因此,看起来您需要在代码中执行类似的操作并添加自己的逻辑。
P.S. 这个讨论可能对您有趣

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接