Domanda
Sto iniziando a imparare Qt4 e Python, seguendo lungo alcuni tutorial che ho trovato su interwebs. Ho il seguente due file:
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_())
Quando ho eseguito questo io ottenere i seguenti errori:
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)
Ho letto che gli slot PyQt non sono altro che metodi, che ho definito, così che cosa sto facendo di sbagliato?
Sono anche imparando Qt4 con Ruby che è dove questo codice proviene da, ho tradotto da Ruby a Python. Nella versione di Ruby la classe LCDRange è definita come questo:
class LCDRange < Qt::Widget
signals 'valueChanged(int)'
slots 'setValue(int)'
def initialize(parent = nil)
...
Quindi, la mia ipotesi è che devo dichiarare in qualche modo l'esistenza dello slot personalizzato?
Soluzione
Prova questo:
self.connect(lcdRange, QtCore.SIGNAL('valueChanged'), previousRange.setValue)
Qual è la differenza?
La documentazione PyQt ha una sezione sui segnali / slot in PyQt, funzionano in modo leggermente diverso.
SIGNAL
SIGNAL('valueChanged')
è qualcosa che si chiama un breve segnale del circuito . Lavorano solo per i metodi Python-to-Python, ma sono più veloci e più facili da implementare.
SLOT
Se si dispone di uno slot di pitone, è possibile specificare semplicemente inclinando il metodo: previousRange.setValue
. Questo funziona per tutti i metodi accessibili da Python.
Se le slot dovrebbero essere accessibili come C ++ Qt slot, come si è tentato nel codice, è necessario utilizzare una sintassi speciale. È possibile trovare informazioni sulla pyqtSignature decoratore sul sito PyQt.
Altri suggerimenti
si è dimenticato di mettere
@Qt.pyqtSlot()
sopra il metodo che si sta utilizzando come slot.
Per esempio il codice dovrebbe assomigliare a questo
@Qt.pyqtSlot('const QPoint&')
def setValue(self,value):
self.slider.setValue(value)
Ecco un buona pagina sulla PyQt Slot decorator:
Ciao
NOTE
Il "testo" all'interno del segnale deve corrispondere al C ++ documentazione 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 &