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

S.l*_*ous 7 c++ qt qmenu qaction

我有一个QActionQMenu.当QActiontriggered()我想知道哪个按钮做到了.

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

void MainWindow::actionclicked(QMouseEvent *e)
{
    if (e->buttons() == Qt::RightButton) 
}
Run Code Online (Sandbox Code Playgroud)

我不能做这样的事情,因为triggered()没有这样的论点.

Ser*_*gey 5

正如@mvidelgauz所注意到的,它QAction是从可能触发该动作的输入设备中抽象出来的。但是,如果在GUI中使用了该操作,则该操作具有一个或多个关联的小部件:工具栏中的工具按钮,菜单栏中的条目等。这些小部件的行为与其他任何小部件一样,因此它们接收的事件可以使用installEventFiltereventFilter进行过滤。这两种方法均继承自QObject,因此它们几乎存在于所有Qt类中。例如,让我们创建一个名为QMainWindow和QAction的应用程序actionTest。然后,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);
}
Run Code Online (Sandbox Code Playgroud)

然后,我们需要为所有监视的对象安装事件过滤器对象,在本例中为小部件。让我们在主窗口构造函数中进行操作:

MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    for(auto wgtPtr : ui->actionTest->associatedWidgets()) {
        wgtPtr->installEventFilter(this);
    }
}
Run Code Online (Sandbox Code Playgroud)

最后,添加一个用于triggered()信号处理的插槽:

void on_actionTest_triggered() {
    qDebug() << "Action triggered!";
}
Run Code Online (Sandbox Code Playgroud)

现在,如果您用鼠标左键单击操作菜单项,它将打印

Left button!
Action triggered!
Run Code Online (Sandbox Code Playgroud)

而对于鼠标右键,结果将是

Right button!
Action triggered!
Run Code Online (Sandbox Code Playgroud)

请注意,小部件事件过滤始终在triggered()信号发射之前执行。

上面的代码只是一个示例,而MainWindow类并不是托管eventFilter方法的最佳位置。在真实代码中,您可以:

  1. 创建QObject用于QAction窗口小部件事件过滤的专用子类。
  2. 子类化QAction并覆盖它的eventFilter方法。在这种情况下,您可以将结果保存QMouseEvent::button()QAction子类对象中,然后在triggered()信号处理程序中使用它。Qt创建者(至少在v3.2.1版之前)不允许您QAction在其表单设计器中“升级” ,这带来了一点不便,因此您需要在窗口构造器中手动向菜单添加操作。
  3. 子类QMenuQToolBar等等。,并使其成为动作过滤器?我不知道怎么能比以前的两个变种更好。

另请参阅有关Qt事件系统的文档

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

for(auto wgtPtr : ui->myActionObject->associatedWidgets()) {
    wgtPtr->installEventFilter(ui->myActionObject);
}
Run Code Online (Sandbox Code Playgroud)

ui->setupUi(this)调用后转到窗口构造函数。此代码类似于上面的示例,但是我们使用ui->myActionObject代替thisobject作为过滤器。