如何在 PyQt 中向事件循环发出自定义事件

Posted

技术标签:

【中文标题】如何在 PyQt 中向事件循环发出自定义事件【英文标题】:How to emit custom Events to the Event Loop in PyQt 【发布时间】:2019-04-06 21:32:22 【问题描述】:

我正在尝试在 PyQt 中发出自定义事件。一个小部件会发出,另一个小部件会监听事件,但这两个小部件不需要相关。

javascript 中,我会这样做

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

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

编辑:我知道信号和插槽,但只知道如何在父子之间使用它们。我将如何在任意不相关的小部件之间使用这种机制(或其他机制)?

【问题讨论】:

我知道信号和插槽,但只知道如何在父子之间使用它们。我将如何在任意、不相关的小部件之间使用它们? 您这样做的真正目的是什么?您要解决哪些具体问题?在 qt 中可以发送自定义事件,但很少需要这种功能。也很难理解为什么你只能知道如何在父母和孩子之间使用信号和槽。 我有一个小部件 A,其中包含一个按钮。当我按下那个按钮时,我想让灯变绿。灯离按钮很远(分层)。我必须去 Button -> Widget A -> Parent -> Parent -> Child -> Child -> Child -> Child -> Light。 也许我试图与我已经拥有的知识联系起来是错误的。在 React 中,我只需从 Button 更新 Redux Store 并从 light 订阅状态​​更改。我应该如何在 Qt 中考虑它? 只需将按钮的点击信号连接到改变灯光颜色的插槽即可。小部件的层次结构应该是无关紧要的。如果不是,您可能需要重组您的类。如果没有看到一些实际的代码,真的不能说更多。 【参考方案1】:

在 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)

ma​​in.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_())

【讨论】:

像魔术一样工作。不能足够支持!也很清楚的解决方案,完全理解了,谢谢! @eyllanesc,我们如何使用这种方法将参数传递给函数? 使用QApplication 作为全局对象怎么样?你可以在任何地方通过QApplication.instance()...

以上是关于如何在 PyQt 中向事件循环发出自定义事件的主要内容,如果未能解决你的问题,请参考以下文章

用JavaScript编写自定义事件调度程序?

js怎么自定义事件,并能让用on事件监听

将自定义 QEvent 传播到 Qt/PyQt 中的父小部件

java自定义事件,线程a如何每一秒钟触发一个事件,然后另一个线程b监听之,并作出反应?

Spring-自定义事件发布

Python+selenium自动循环发邮件