我想知道是否用鼠标左键或右键单击QAction

时间:2016-08-06 11:36:52

标签: c++ qt qmenu qaction

QAction中有QMenu。当QActiontriggered()时,我想知道是哪个按钮。

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

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

我无法做到这一点,因为triggered()没有这样的论点。

2 个答案:

答案 0 :(得分:5)

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

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. QObject小部件事件过滤创建专用的QAction子类。
  2. 子类QAction并覆盖它的eventFilter方法。在这种情况下,您可以将QMouseEvent::button()的结果保存在QAction子类对象中,然后在triggered()信号处理程序中使用它。 Qt创建者(至少高达v3.2.1)不允许您在其表单设计器中“提升”QAction,因此您需要在窗口构造函数中手动向菜单添加操作,这是一个小小的不便。
  3. 子类QMenuQToolBar等等,并让它们成为动作过滤器?我不知道它怎么能比两个以前的变种更好。
  4. 另见documentation关于Qt事件系统。

    让我们澄清案例2.假设从QAction继承的类称为MyAction。为了使其工作,您需要安装MyAction个对象作为自己的过滤器(它们的小部件,更具体)。您需要在创建窗口小部件后执行此操作,因此在MyAction构造函数中安装过滤器可能为时过早并导致崩溃。过滤器安装的最佳位置是拥有MyAction对象的类的构造函数。通常它是一个小部件或窗口类。所以只需添加

    for(auto wgtPtr : ui->myActionObject->associatedWidgets()) {
        wgtPtr->installEventFilter(ui->myActionObject);
    }
    
    ui->setupUi(this)调用后

    到您的窗口构造函数。此代码与上面的示例类似,但我们使用ui->myActionObject代替this对象作为过滤器。

答案 1 :(得分:3)

triggered()不能设计这个参数,因为它本身不一定是鼠标事件的结果:

  

当用户激活动作时发出此信号;例如,当用户单击菜单选项,工具栏按钮,或按动作的快捷键组合,或者调用trigger()时

如果需要QMouseEvent作为参数,则需要连接鼠标事件。事实上Qt本身发出triggered()时(但不仅仅是我在doc引用中突出显示)框架从菜单接收鼠标事件。因此,您似乎需要在代码中执行类似的操作并添加自己的逻辑。

P.S。 This discussion对你来说可能很有意思