如何将slot作为参数传递

How to pass a slot as an argument

本文关键字:参数传递 slot      更新时间:2023-10-16

我使用Qt 5.7和我有一个组件的行为像QMenuBar。该组件有两个方法,一个称为addMenu(QPushButton* menu),用于添加一个栏菜单,另一个称为addMenuItem(QPushButton* menu, QString item, const char *slot),用于添加组成一个栏菜单的项。在第二个方法中,我调用QMenu的addAction方法,并且必须向它传递两个参数:项目名称和当项目被触发时将运行的插槽。我的问题是怎么做。我做了一个搜索(见这里),我发现QT把槽当作const char *,但是当我尝试使用这个方法签名编译我的程序时,它给出了错误,指出没有为这种类型注册的操作符()。

我代码:

GMInternalMenu::addMenuItem(QPushButton *menu, QString menuItem, const char* slot){
    QMenu *dropdown = mapa->value(menu);
    dropdown->addAction(menuItem, slot);
}

使用

方法
ui->imMenu->addMenuItem(someButton, "Some action", SLOT(close()));

输出错误:

In file included from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtGui/qwindowdefs.h:43:0,
                 from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/qwidget.h:43,
                 from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/qmenu.h:43,
                 from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/QMenu:1,
                 from ../Project/mymenu.h:4,
                 from ../Project/mymenu.cpp:1:
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobject.h: In instantiation of 'static typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type QObject::connect(const typename QtPrivate::FunctionPointer<Func>::Object*, Func1, const QObject*, Func2, Qt::ConnectionType) [with Func1 = void (QAction::*)(bool); Func2 = const char*; typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type = QMetaObject::Connection; typename QtPrivate::FunctionPointer<Func>::Object = QAction]':
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobject.h:293:74:   required from 'static typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type QObject::connect(const typename QtPrivate::FunctionPointer<Func>::Object*, Func1, Func2) [with Func1 = void (QAction::*)(bool); Func2 = const char*; typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type = QMetaObject::Connection; typename QtPrivate::FunctionPointer<Func>::Object = QAction]'
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/qmenu.h:120:9:   required from 'QAction* QMenu::addAction(const QString&, Func1, const QKeySequence&) [with Func1 = const char*]'
../Project/mymenu.cpp:35:39:   required from here
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qglobal.h:746:47: error: static assertion failed: Signal and slot arguments are not compatible.
 #define Q_STATIC_ASSERT_X(Condition, Message) static_assert(bool(Condition), Message)
                                               ^
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobject.h:306:9: note: in expansion of macro 'Q_STATIC_ASSERT_X'
         Q_STATIC_ASSERT_X((FunctorArgumentCount >= 0),
         ^
In file included from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobjectdefs.h:50:0,
                 from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtGui/qwindowdefs.h:44,
                 from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/qwidget.h:43,
                 from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/qmenu.h:43,
                 from C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/QMenu:1,
                 from ../Project/mymenu.h:4,
                 from ../Project/mymenu.cpp:1:
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobjectdefs_impl.h: In instantiation of 'struct QtPrivate::FunctorReturnType<const char*, QtPrivate::List<> >':
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobject.h:309:158:   required from 'static typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type QObject::connect(const typename QtPrivate::FunctionPointer<Func>::Object*, Func1, const QObject*, Func2, Qt::ConnectionType) [with Func1 = void (QAction::*)(bool); Func2 = const char*; typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type = QMetaObject::Connection; typename QtPrivate::FunctionPointer<Func>::Object = QAction]'
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobject.h:293:74:   required from 'static typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type QObject::connect(const typename QtPrivate::FunctionPointer<Func>::Object*, Func1, Func2) [with Func1 = void (QAction::*)(bool); Func2 = const char*; typename QtPrivate::QEnableIf<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::Type = QMetaObject::Connection; typename QtPrivate::FunctionPointer<Func>::Object = QAction]'
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtWidgets/qmenu.h:120:9:   required from 'QAction* QMenu::addAction(const QString&, Func1, const QKeySequence&) [with Func1 = const char*]'
../Project/mymenu.cpp:35:39:   required from here
C:/Qt/Qt5.7.0/5.7/mingw53_32/include/QtCore/qobjectdefs_impl.h:632:78: error: request for member 'operator()' in 'QtPrivate::FunctorReturnType<Functor, QtPrivate::List<Tail ...> >::dummy<const char*>()', which is of non-class type 'const char*'
         typedef decltype(dummy<Functor>().operator()((dummy<ArgList>())...)) Value;

这是您试图使用的QMenu::addAction方法的声明:

QAction* addAction(const QString &text, 
                   const QObject *receiver, 
                   const char *member, 
                   const QKeySequence &shortcut = 0);

如您所见,您缺少receiver参数,该参数是将接收信号的对象。

从你的代码中,我猜你应该有这样的东西:
GMInternalMenu::addMenuItem(QPushButton *menu, QString menuItem, 
                            QObject *receiver, const char* slot){
    QMenu *dropdown = mapa->value(menu);
    dropdown->addAction(menuItem, receiver, slot);
}

然后当你使用它时:

ui->imMenu->addMenuItem(someButton, "Some action", this, SLOT(close()));
//                                                 ^^^^
//                 Maybe something else... I do not know
//                 which object should receive the signal.