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

标签 python signals-slots pyqt5 qlistwidgetitem

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

最佳答案

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

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

关于python - 列表小部件项数据对象上的槽从未被调用(在 PyQt 5.7 中),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39292449/

相关文章:

python - Pandas - 合并数据框

python - 列表操作,跟踪旧列表

c++ - Qt 插槽测试

c++ - 与(静态)库的两种方式通信

python - 如何知道 QTextEdit 上的 Enter 键何时被按下

python - 使用拖放时如何拖动图标

python - 转移概率的递归函数(查普曼-柯尔莫哥洛夫方程)

python - PyQt4 @pyqtSlot : what is the result kwarg for?

python - Pyqt5 鼠标事件不适用于我的自定义选项卡栏

python - Pandas 与平均值的百分比差异