Как создавать пользовательские события в цикле событий в PyQt - PullRequest
1 голос
/ 07 апреля 2019

Я пытаюсь генерировать пользовательские события в PyQt.Один виджет будет излучать, а другой будет прослушивать события, но эти два виджета не нужно будет связывать.

В JavaScript я бы достиг этого, выполнив

// Component 1
document.addEventListener('Hello', () => console.log('Got it'))

// Component 2
document.dispatchEvent(new Event("Hello"))

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

1 Ответ

2 голосов
/ 07 апреля 2019

В PyQt следующая инструкция:

document.addEventListener('Hello', () => console.log('Got it'))

эквивалентна

document.hello_signal.connect(lambda: print('Got it'))

Аналогичным образом:

document.dispatchEvent(new Event("Hello"))

эквивалентна

document.hello_signal.emit()

Но большая разница заключается в объеме объекта «документ», поскольку связь между глобальным элементом.Но в PyQt этот элемент не существует.

Один из способов эмулировать поведение, на которое вы указываете, - создать глобальный объект:

globalobject.py

from PyQt5 import QtCore
import functools

@functools.lru_cache()
class GlobalObject(QtCore.QObject):
    def __init__(self):
        super().__init__()
        self._events = {}

    def addEventListener(self, name, func):
        if name not in self._events:
            self._events[name] = [func]
        else:
            self._events[name].append(func)

    def dispatchEvent(self, name):
        functions = self._events.get(name, [])
        for func in functions:
            QtCore.QTimer.singleShot(0, func)

main.py

from PyQt5 import QtCore, QtWidgets
from globalobject import GlobalObject


class MainWindow(QtWidgets.QMainWindow):
    def __init__(self, parent=None):
        super().__init__(parent)
        button = QtWidgets.QPushButton(text="Press me", clicked=self.on_clicked)
        self.setCentralWidget(button)

    @QtCore.pyqtSlot()
    def on_clicked(self):
        GlobalObject().dispatchEvent("hello")


class Widget(QtWidgets.QWidget):
    def __init__(self, parent=None):
        super().__init__(parent)
        GlobalObject().addEventListener("hello", self.foo)
        self._label = QtWidgets.QLabel()
        lay = QtWidgets.QVBoxLayout(self)
        lay.addWidget(self._label)

    @QtCore.pyqtSlot()
    def foo(self):
        self._label.setText("foo")


if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)
    w1 = MainWindow()
    w2 = Widget()
    w1.show()
    w2.show()
    sys.exit(app.exec_())
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...