【发布时间】:2019-08-27 20:22:24
【问题描述】:
我有一个 PyQt5 应用程序,它具有可选功能(比如说一个按钮),只有在明确告知应用程序“打开此选项”时才会显示该功能。调用此按钮时,应用程序会进行一些计算,然后发出带有结果的信号。
我希望程序在有人连接到信号时显示功能,而不是显式传递构造函数选项。
工作示例:
class ExampleApp(QWidget):
do_something = pyqtSignal(str)
def __init__(self, something_enabled: bool = False):
super().__init__()
self.button = QPushButton("Do something")
self.button.pressed.connect(self.do_work)
self.button.setVisible(something_enabled)
self.setLayout(QVBoxLayout())
self.layout().addWidget(self.button)
self.show()
def do_work(self):
self.do_something.emit("Something!")
if __name__ == '__main__':
import sys
app = QApplication(sys.argv)
window = ExampleApp(something_enabled=True)
window.do_something.connect(lambda result: print(result))
app.exec()
我在文档中找不到任何可用选项,也找不到此功能的实现。在我自己的代码中,我尝试直接更改连接函数,但我得到一个只读的 AttributeError。
包装函数:
self.original_connect = self.do_something.connect
def new_connect(slot, type=None, no_receiver_check=False):
print("New connection!")
self.original_connect(slot, type, no_receiver_check)
self.do_something.connect = new_connect
属性错误:
AttributeError: 'PyQt5.QtCore.pyqtBoundSignal' object attribute 'connect' is read-only
我还创建了一个工作包装类,但我担心未知的后果。
包装类:
class SignalWrapper(QObject):
new_connection = pyqtSignal(object)
def __init__(self, signal_to_wrap):
super().__init__()
self.wrapped_signal = signal_to_wrap
self.signal = self.wrapped_signal.signal
def connect(self, slot, type=None, no_receiver_check=False):
self.new_connection.emit(slot)
return self.wrapped_signal.connect(slot)
def disconnect(self, slot=None): # real signature unknown; restored from __doc__
return self.wrapped_signal.disconnect(slot)
def emit(self, *args): # real signature unknown; restored from __doc__
return self.wrapped_signal.emit(*args)
def __call__(self, *args, **kwargs): # real signature unknown
return self.wrapped_signal.__call__(*args, **kwargs)
def __getitem__(self, *args, **kwargs): # real signature unknown
return self.wrapped_signal.__getitem__(*args, **kwargs)
def __repr__(self, *args, **kwargs): # real signature unknown
return self.wrapped_signal.__repr__(*args, **kwargs)
ExampleApp 的修改部分:
class ExampleApp(QWidget):
do_something = pyqtSignal(str)
def __init__(self):
super().__init__()
self.do_something = SignalWrapper(self.do_something)
我想知道在 PyQt 中是否有一种优雅的方式来拦截信号连接。
【问题讨论】:
标签: python pyqt pyqt5 signals-slots