从未调用列表小部件项目数据对象上的槽(在 PyQt 5.7 中)

Slot on list widget item data object never called (in PyQt 5.7)

在 PyQt 5.5 中,以下代码有效,但在 PyQt 5.7 中却不是这样(列表显示 'example' 而不是 'new example',实际上调试显示永远不会命中插槽)。有谁知道它有什么问题:

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 派生,因此该项目可以直接连接到信号)。

Post-答案编辑

在 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() 很可能只保留对其第二个参数的弱引用,导致数据在初始化结束时被删除,除非它存储在某个地方。

似乎存在某种垃圾收集问题。试试这个:

    hit_item = QListWidgetItem('example')
    data = MyListItemData(hit_item, obj)
    hit_item.setData(Qt.UserRole, data)