Как передать аргументы в функции обратного вызова в PyQt

У меня есть около 10 QAction (это число будет варьироваться во время выполнения) на панели инструментов, которая все будет делать то же самое, но с использованием разных параметров. Я думаю добавить параметр в качестве атрибута к объекту QAction, а затем, сработавший сигнал QAction также отправит сам объект в функцию обратного вызова, чтобы я мог получить необходимые параметры для функции. У меня есть 2 вопроса по этому поводу:

  • Это можно сделать?
  • Есть ли лучший способ сделать это?

5 ответов

Решение

Вы можете отправить сам объект действия, используя средство отображения сигналов. Однако может быть лучше просто отправить идентификатор и выполнить всю работу в обработчике сигнала.

Вот простой демонстрационный скрипт:

from PyQt4 import QtGui, QtCore

class Window(QtGui.QMainWindow):
    def __init__(self):
        QtGui.QMainWindow.__init__(self)
        self.mapper = QtCore.QSignalMapper(self)
        self.toolbar = self.addToolBar('Foo')
        self.toolbar.setToolButtonStyle(QtCore.Qt.ToolButtonTextOnly)
        for text in 'One Two Three'.split():
            action = QtGui.QAction(text, self)
            self.mapper.setMapping(action, text)
            action.triggered.connect(self.mapper.map)
            self.toolbar.addAction(action)
        self.mapper.mapped['QString'].connect(self.handleButton)
        self.edit = QtGui.QLineEdit(self)
        self.setCentralWidget(self.edit)

    def handleButton(self, identifier):
        if identifier == 'One':
            text = 'Do This'
        elif identifier == 'Two':
            text = 'Do That'
        elif identifier == 'Three':
            text = 'Do Other'
        self.edit.setText(text)

if __name__ == '__main__':

    import sys
    app = QtGui.QApplication(sys.argv)
    window = Window()
    window.resize(300, 60)
    window.show()
    sys.exit(app.exec_())

Как передать аргументы в функции обратного вызова в PyQt

Ты можешь использовать functools.partial из стандартной библиотеки Python. Пример с QAction:

some_action.triggered.connect(functools.partial(some_callback, param1, param2))

Вы можете использовать лямбда-функцию, связанную со слотом элемента управления GUI, чтобы передать дополнительные аргументы методу, который вы хотите выполнить.

# Create the build button with its caption
self.build_button = QPushButton('&Build Greeting', self)
# Connect the button's clicked signal to AddControl
self.build_button.clicked.connect(lambda: self.AddControl('fooData'))
def AddControl(self, name):
    print name

Источник: snip2code - Использование лямбда-функции для передачи дополнительного аргумента в PyQt4

Лучший способ передать аргументы - не передавать их вообще. Вы можете использовать динамическую природу python и устанавливать любые нужные вам данные в качестве своих собственных свойств на самих виджетах, получить целевой виджет в обработчике, используя self.sender(), а затем получить все необходимые свойства прямо из виджета.

В этом примере пять кнопок созданы, и требуемое состояние установлено на виджете кнопки как my_own_data имущество:

import sys
from PyQt4 import QtGui, QtCore

class Main(QtGui.QMainWindow):

    def __init__(self):
        QtGui.QMainWindow.__init__(self)
        self.centralwidget = QtGui.QWidget()
        self.vboxlayout = QtGui.QVBoxLayout()

        for idx in range(5):
            button = QtGui.QPushButton('button ' + str(idx), None)
            button.my_own_data = str(idx)  # <<< set your own property
            button.clicked.connect(self.click_handler)  # <<< no args needed
            self.vboxlayout.addWidget(button)

        self.centralwidget.setLayout(self.vboxlayout)
        self.setCentralWidget(self.centralwidget)
        self.show()

    def click_handler(self, data=None):
        if not data:
            target = self.sender()  # <<< get the event target, i.e. the button widget
            data = target.my_own_data  # <<< get your own property
        QtGui.QMessageBox.information(self, 
                                      "you clicked me!", 
                                      "my index is {0}".format(data))

app = QtGui.QApplication(sys.argv)
main = Main()
main.show()

sys.exit(app.exec_())

Все еще есть ошибка в том, как PyQt использует QSignalMapper. Я нашел обходной путь здесь:

http://www.riverbankcomputing.com/pipermail/pyqt/2010-March/026113.html

Чтобы исправить ответ на вопрос, внесите следующие изменения:

        action.triggered[()].connect(self.mapper.map)

Это было необходимо для следующей версии Python:

Python 2.6.6 (r266:84292, Feb 21 2013, 19:26:11)
[GCC 4.4.7 20120313 (Red Hat 4.4.7-3)] on linux2
Другие вопросы по тегам