Создать меню PyQt из списка строк - PullRequest
10 голосов
/ 09 июля 2009

У меня есть список строк и я хочу создать пункт меню для каждой из этих строк. Когда пользователь нажимает на одну из записей, всегда должна вызываться одна и та же функция со строкой в ​​качестве аргумента. После некоторых попыток и исследований я придумал что-то вроде этого:

import sys
from PyQt4 import QtGui, QtCore

class MainWindow(QtGui.QMainWindow):
    def __init__(self):
        QtGui.QMainWindow.__init__(self)
        self.menubar = self.menuBar()
        menuitems = ["Item 1","Item 2","Item 3"]
        menu = self.menubar.addMenu('&Stuff')
        for item in menuitems:
            entry = menu.addAction(item)
            self.connect(entry,QtCore.SIGNAL('triggered()'), lambda: self.doStuff(item))
            menu.addAction(entry)
        print "init done"

    def doStuff(self, item):
        print item

app = QtGui.QApplication(sys.argv)
main = MainWindow()
main.show()
sys.exit(app.exec_())

Теперь проблема в том, что каждый из пунктов меню напечатает один и тот же вывод: «Элемент 3» вместо соответствующего. Я благодарен за любые идеи о том, как я могу сделать это правильно. Спасибо.

Ответы [ 2 ]

23 голосов
/ 09 июля 2009

Вы встречаете то, что часто упоминается (может быть, не совсем педантически-правильно ;-) как «проблема с ограничением объема» в Python - привязка запаздывает (лексический поиск во время вызова), в то время как вы хотели бы рано (по умолчанию). Итак, где у вас сейчас есть:

    for item in menuitems:
        entry = menu.addAction(item)
        self.connect(entry,QtCore.SIGNAL('triggered()'), lambda: self.doStuff(item))

попробуйте вместо:

    for item in menuitems:
        entry = menu.addAction(item)
        self.connect(entry,QtCore.SIGNAL('triggered()'), lambda item=item: self.doStuff(item))

Это «предвосхищает» привязку, поскольку значения по умолчанию (как здесь item) вычисляются один раз для всех в def-time. Добавление одного уровня вложенности функций (например, двойной лямбды) тоже работает, но это немного излишне! -)

Вы могли бы альтернативно использовать functools.partial(self.doStuff, item)import functools наверху, конечно), что является еще одним хорошим решением, но я думаю, что я бы выбрал самое простое (и наиболее распространенное) "поддельное значение по умолчанию для аргумента "идиома.

2 голосов
/ 09 июля 2009

Это должно сработать, но я уверен, что сейчас есть лучший способ, который я не могу вспомнить.

def do_stuff_caller(self, item):
    return lambda: self.doStuff(item)

...
self.connect(entry, QtCore.SIGNAL('triggered()'), self.do_stuff_caller(item))

Редактировать : Короче версия, это все еще не то, о чем я думаю ... или, может быть, это было на другом языке? :)

(lambda x: lambda self.do_stuff(x))(item)
...