2012-01-11 223 views
7

我在工具欄中有大約10個QAction(這個數字在運行時會有所不同),它們都會做同樣的事情,但使用不同的參數。我在考慮將參數作爲屬性添加到QAction對象,然後,QAction的觸發信號也會將對象本身發送到回調函數,以便我可以獲取該函數所需的參數。我其實有兩個問題:如何將參數傳遞給PyQt中的回調函數

  • 可以這樣做嗎?
  • 有沒有更好的方法來做到這一點?

回答

4

您可以使用signal mapper發送動作對象本身。但是,簡單地發送標識符並執行信號處理程序中的所有工作可能會更好。

這裏有一個簡單的演示腳本:

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_()) 
+0

這個答案似乎不是很受歡迎。 QSignalMapper沒有被廣泛使用嗎? – 101 2015-07-11 08:33:37

+0

@ figs。我不知道爲什麼OP接受了這個答案,因爲已經給出了更傳統的「lambda/partial」解決方案。曾經有一次'QSignalMapper'更有用,因爲PyQt並不總是支持使用'lambda/partial'作爲插槽 - 但那是很久以前的事了。然而,目前,我想不出一個更有效的場景(當然,除了C++之外)。 – ekhumoro 2015-07-11 09:45:49

35

如何參數傳遞給回調函數在PyQt的

您可以使用functools.partial從非標準Python庫。例如與QAction

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

非常簡單和有效的。恕我直言,這應該是公認的答案。 – reima 2013-08-22 17:03:06

+0

我只是使用這個解決方案。效果很好! – swdev 2014-06-28 03:57:42

1

仍有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 
9

您可以使用與GUI控件的插槽關聯的lambda函數將額外的參數傳遞給您要執行的方法。

# 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 - Using Lambda Function To Pass Extra Argument in PyQt4

+1

謝謝!正是我在找什麼! – 2014-01-07 08:34:57

3

來傳遞參數的最佳方式是無法通過它們。您可以使用python的動態特性,並將所需的任何數據設置爲您自己的小部件本身的屬性,使用self.sender()獲取處理程序中的目標小部件,然後直接從小部件獲取所需的任何屬性。

在這個例子中被創建五個按鈕和所需的狀態被設置在按鈕上widget作爲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_()) 
+0

這種方法有一些缺點,詳見[docs for sender](http://doc.qt.io/qt-4.8/qobject.html#sender)。正如你寫的那樣,'click_handler'不能被連接到它的特定按鈕之外的任何東西安全地調用。 – ekhumoro 2015-12-02 18:04:35

+0

@ekhumoro:同意;添加了可選的數據參數,這樣''click_handler()'可以以更通用的方式使用。 – ccpizza 2015-12-02 19:25:53