在 PyQt 5.5 中,以下代码有效,但在 PyQt 5.7 中则不然(列表显示“示例”而不是“新示例”,并且确实调试显示该插槽从未被命中)。有谁知道它有什么问题:
from PyQt5.QtWidgets import QListWidgetItem, QListWidget, QApplication
from PyQt5.QtCore import QObject, pyqtSlot, pyqtSignal, Qt
class MyListItemData(QObject):
def __init__(self, list_widget_item, obj):
super().__init__()
self.list_widget_item = list_widget_item
obj.sig_name_changed.connect(self.__on_list_item_name_changed)
# @pyqtSlot(str)
def __on_list_item_name_changed(self, new_name: str):
self.list_widget_item.setText(new_name)
class Data(QObject):
sig_name_changed = pyqtSignal(str)
class SearchPanel2(QListWidget):
def __init__(self, parent=None):
QListWidget.__init__(self, parent)
obj = Data()
hit_item = QListWidgetItem('example')
hit_item.setData(Qt.UserRole, MyListItemData(hit_item, obj))
self.addItem(hit_item)
obj.sig_name_changed.emit('new_example')
app = QApplication([])
search = SearchPanel2()
search.show()
app.exec
虽然这可能不是应该完成的方式,但在 PyQt 5.5 中,对于 PyQt 5.5 错误(这阻止我们简单地从 QListWidgetItem 派生,因此项目可以直接连接到信号),这是一个可接受的解决方法。
回复后编辑
在 Ekhumoro 回答之后,我面临着一个严酷的现实:这修复了发布的示例代码,但不是我的应用程序,因为我的应用程序完全按照解决方案所说的去做。所以我重新审视:在真实的应用程序中,项目是稍后创建的,名称更改的信号是稍后发出的。因此,重现我的问题的更好的最小示例将具有以下内容:
class SearchPanel2(QListWidget):
def __init__(self, obj, parent=None):
QListWidget.__init__(self, parent)
hit_item = QListWidgetItem('example')
data = MyListItemData(hit_item, obj)
hit_item.setData(Qt.UserRole, data) # slot not called
self.addItem(hit_item)
# self.data = data
def emit(self):
obj.sig_name_changed.emit('new_example')
app = QApplication([])
obj = Data()
search = SearchPanel2(obj)
search.show()
QTimer.singleShot(2000, search.emit)
app.exec()
assert search.item(0).text() == 'new_example'
这使断言失败。如果数据由强引用保存(取消注释最后一行 init),则断言通过。所以很可能 setData() 只保留对其第二个参数的弱引用,导致数据在 init 结束时被删除,除非它存储在某个地方。