Вопрос
Я начинаю изучать Qt4 и Python, следуя некоторому руководству, которое я нашел в Интернете.У меня есть следующие два файла:
lcdrange.py:
from PyQt4 import QtGui, QtCore
class LCDRange(QtGui.QWidget):
def __init__(self, parent=None):
QtGui.QWidget.__init__(self, parent)
lcd = QtGui.QLCDNumber(2)
self.slider = QtGui.QSlider()
self.slider.setRange(0,99)
self.slider.setValue(0)
self.connect(self.slider, QtCore.SIGNAL('valueChanged(int)'),
lcd, QtCore.SLOT('display(int)'))
self.connect(self.slider, QtCore.SIGNAL('valueChanged(int)'),
self, QtCore.SIGNAL('valueChanged(int)'))
layout = QtGui.QVBoxLayout()
layout.addWidget(lcd)
layout.addWidget(self.slider)
self.setLayout(layout)
def value(self):
self.slider.value()
def setValue(self,value):
self.slider.setValue(value)
main.py:
import sys
from PyQt4 import QtGui, QtCore
from lcdrange import LCDRange
class MyWidget(QtGui.QWidget):
def __init__(self, parent=None):
QtGui.QWidget.__init__(self, parent)
quit = QtGui.QPushButton('Quit')
quit.setFont(QtGui.QFont('Times', 18, QtGui.QFont.Bold))
self.connect(quit, QtCore.SIGNAL('clicked()'), QtGui.qApp, QtCore.SLOT('quit()'))
grid = QtGui.QGridLayout()
previousRange = None
for row in range(0,3):
for column in range(0,3):
lcdRange = LCDRange()
grid.addWidget(lcdRange, row, column)
if not previousRange == None:
self.connect(lcdRange, QtCore.SIGNAL('valueChanged(int)'),
previousRange, QtCore.SLOT('setValue(int)'))
previousRange = lcdRange
layout = QtGui.QVBoxLayout()
layout.addWidget(quit)
layout.addLayout(grid)
self.setLayout(layout)
app = QtGui.QApplication(sys.argv)
widget = MyWidget()
widget.show()
sys.exit(app.exec_())
Когда я запускаю это, я получаю следующие ошибки:
Object::connect: No such slot LCDRange::setValue(int)
Object::connect: No such slot LCDRange::setValue(int)
Object::connect: No such slot LCDRange::setValue(int)
Object::connect: No such slot LCDRange::setValue(int)
Object::connect: No such slot LCDRange::setValue(int)
Object::connect: No such slot LCDRange::setValue(int)
Object::connect: No such slot LCDRange::setValue(int)
Object::connect: No such slot LCDRange::setValue(int)
Я читал, что слоты PyQt - это не что иное, как методы, которые я определил, так что же я делаю не так?
Я также изучаю Qt4 с Ruby, откуда взят этот код, я перевел его с Ruby на Python.В версии Ruby класс LCDRange определяется следующим образом:
class LCDRange < Qt::Widget
signals 'valueChanged(int)'
slots 'setValue(int)'
def initialize(parent = nil)
...
Итак, мое предположение состояло в том, что я должен каким-то образом объявить о существовании пользовательского слота?
Решение
Попробуй это:
self.connect(lcdRange, QtCore.SIGNAL('valueChanged'), previousRange.setValue)
В чем разница?
Документация PyQt в PyQt есть раздел о СИГНАЛАХ / слотах, они работают немного по-другому.
СИГНАЛ
SIGNAL('valueChanged')
это то, что называется сигнал короткого замыкания.Они работают только для методов Python-to-Python, но они быстрее и проще в реализации.
СЛОТ
Если у вас есть слот python, вы можете указать его, просто указав метод: previousRange.setValue
.Это работает для всех методов, доступных в Python.
Если ваши слоты должны быть доступны подобно слотам C ++ Qt, как вы пытались сделать в своем коде, вы должны использовать специальный синтаксис.Вы можете найти информацию о pyqtSignature декоратор на веб-сайте PyQt.
Другие советы
вы забыли поставить
@Qt.pyqtSlot()
описанный выше метод вы используете в качестве слота.
Например, ваш код должен выглядеть следующим образом
@Qt.pyqtSlot('const QPoint&')
def setValue(self,value):
self.slider.setValue(value)
Вот одна хорошая страница о pyqt slot decorator:
Пока
ПРИМЕЧАНИЕ
"Текст" внутри СИГНАЛА должен соответствовать документации c ++ API.
# This will work - its IDENTICAL to the documentation
QtCore.SIGNAL('customContextMenuRequested(const QPoint&)')
# this wont
QtCore.SIGNAL('customContextMenuRequested(QPoint&)')
# and this wont
QtCore.SIGNAL('customContextMenuRequested(const QPoint)')
# Spot the bug
QtCore.SIGNAL('selectionChanged(const QItemSelection,const QItemSelection&)')
^ < missing &