PyQt: как обрабатывать событие без наследования

Как я могу обрабатывать событие мыши без наследования, usecase можно описать следующим образом:

предположим, что я хочу, чтобы объект QLabel был Генделем MouseMoveEvent, путь в учебнике часто идет так, как мы создаем новый класс, унаследованный от QLabel. Но могу ли я просто использовать лямбда-выражение, чтобы передать событие без наследования так же, как

ql = QLabel()
ql.mouseMoveEvent = lambda e : print e.x(), e.y()

поэтому мне не нужно писать целый класс и просто использовать простое лямбда-выражение для реализации некоторых простое событие.

2 ответов


самый гибкий способ сделать это-установить фильтр событий который может получать события от имени объекта:

from PyQt4 import QtGui, QtCore

class Window(QtGui.QWidget):
    def __init__(self):
        QtGui.QWidget.__init__(self)
        self.label = QtGui.QLabel(self)
        self.label.setText('Hello World')
        self.label.setAlignment(QtCore.Qt.AlignCenter)
        self.label.setFrameStyle(QtGui.QFrame.Box | QtGui.QFrame.Plain)
        self.label.setMouseTracking(True)
        self.label.installEventFilter(self)
        layout = QtGui.QVBoxLayout(self)
        layout.addWidget(self.label)

    def eventFilter(self, source, event):
        if (event.type() == QtCore.QEvent.MouseMove and
            source is self.label):
            pos = event.pos()
            print('mouse move: (%d, %d)' % (pos.x(), pos.y()))
        return QtGui.QWidget.eventFilter(self, source, event)

if __name__ == '__main__':

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

Да, вы можете это сделать, но в python2 вы не можете использовать print в вашей лямбде, так как это оператор, а не функция и не возвращает значение.

попробуйте это:

ql = QLabel()
def event_handler(e):
    print e.x(), e.y()
ql.mouseMoveEvent = event_handler