QMetaObject :: invokeMethod找不到带参数的方法

时间:2014-05-08 10:15:57

标签: python qt pyside qtcore qt-signals

这是QMetaObject::invokeMethod doesn't find the method的后续行动。调用没有参数的方法有效。但是将前一个问题扩展到带参数的方法会让我再次失败。

请参阅Python中的以下示例脚本:

from PySide import QtCore

class Example(QtCore.QObject):
    def __init__(self):
        super().__init__()

    @QtCore.Slot()
    def dup(self):
        beep('dup-class')

    @QtCore.Slot(str)
    def beep(self, text):
        print(text)

@QtCore.Slot()
def dup(self):
    beep('dup-local')

@QtCore.Slot(str)
def beep(text):
    print(text)

if __name__ == '__main__':
    QtCore.QMetaObject.invokeMethod(None, 'dup')
    QtCore.QMetaObject.invokeMethod(None, 'beep', QtCore.Qt.AutoConnection, QtCore.QGenericArgument('text', 'beep-local'))

    print('now some classy trials')
    t = Example()
    QtCore.QMetaObject.invokeMethod(t, 'dup')
    QtCore.QMetaObject.invokeMethod(t, 'beep', QtCore.Qt.AutoConnection, QtCore.QGenericArgument('text', 'beep-class'))
    QtCore.QMetaObject.invokeMethod(t, 'beep', QtCore.Qt.AutoConnection, QtCore.QGenericArgument('self', t), QtCore.QGenericArgument('text', 'beep-class-b'))

在Windows 7和Ubuntu 14.04上使用PySide 1.2.1和Python 3.3的输出也是:

now some classy trials
dup-class
QMetaObject::invokeMethod: No such method Example::beep(text)
QMetaObject::invokeMethod: No such method Example::beep(self,text)

这意味着对本地方法的invokeMethod调用以静默方式失败。只有对Example:dup()的调用给出了预期的输出。获得示例的两个试验:beep(str)工作失败,尽管失败消息给出了实际应该存在的方法签名。

我在PySide mailing list上提出了这个问题的早期版本,但没有回答。

问题:如何使QMetaObject::invokeMethod使用Python Qt绑定中的参数调用local和class方法(最好是在PySide中)?

编辑:顺便说一句:如果有人知道幕后的Signal:emit(...)QtCore.QTimer.singleShot(0, ...)做了什么,那么这也有帮助。毕竟所有这些不同的方法都有非常相似的效果。


EDIT2:

'QString'作为参数名称,警告消息消失,但Python在整个段落失败时失败。它可能是PySide的实现错误。规则似乎是必须在invokeMethod中提供Qt-C ++类型的参数,在插槽中提供Python类型。

from PySide import QtCore

class Example(QtCore.QObject):
    def __init__(self):
        super().__init__()

    @QtCore.Slot(str)
    def beep(self, text='default'):
        print(text)

if __name__ == '__main__':
    app = QtCore.QCoreApplication([])

    e = Example()
    QtCore.QMetaObject.invokeMethod(e, 'beep', QtCore.Qt.AutoConnection, QtCore.QGenericArgument('QString', 'beep'))

    QtCore.QTimer.singleShot(1000, app.quit)
    app.exec_()

1 个答案:

答案 0 :(得分:5)

对于仍然对此感兴趣的人:

从版本1.2.4开始,PySide存在错误,因为它没有包装Q_ARGQ_RETURN_ARG宏。相反,它不明智地包装QGenericArgumentQGenericReturnArgument类,它们是内部帮助程序类,并不意味着直接实例化。因此,使用参数调用插槽总是会导致分段错误。

相比之下,PyQt 确实包装宏而不是类,并且不会遇到同样的问题。