从未调用列表小部件项目数据对象上的槽(在 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)
在 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)