1

在 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 结束时被删除,除非它存储在某个地方。

4

1 回答 1

1

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

    hit_item = QListWidgetItem('example')
    data = MyListItemData(hit_item, obj)
    hit_item.setData(Qt.UserRole, data)
于 2016-09-02T17:50:26.167 回答