gpt4 book ai didi

python - 奇怪的发射行为。执行emit时无法调用slot

转载 作者:行者123 更新时间:2023-11-30 23:45:46 25 4
gpt4 key购买 nike

我在使用 PySide/Qt 时遇到了这种奇怪的情况。

编辑:最后我添加了一个小的“可以运行”代码来测试所经历的行为。如果你愿意,你可以跳过胡言乱语,通过实际代码来看看我在说什么。根据信号/槽机制,它应该表现出意外

快速演示设计。我有一个 MainWindow,它包含两个 View 、两个 Controller 和两个模型。

这两个 View 是在主窗口中创建的,如下所示

    ui.listView = views.ListView(ui.hSplitter)
ui.threeDView = views.ThreeDView(ui.hSplitter)

没什么特别的。模型是在 ui 之后创建的

    listModel = models.ListModel()
listSelectionModel = models.ListSelectionModel()

第一个包含实际数据。第二个包含选择,以便 3D View 可以检查选择是否已更改(通常通过用户单击 listView)并以 3d 形式绘制实体。

最后 Controller 就这样创建了

threeDController = ThreeDController(threeDView=self._ui.threeDView,
listModel=listModel,
listSelectionModel=listSelectionModel)

listController = ListController(listView = self._ui.listView,
listModel = listModel,
listSelectionModel=listSelectionModel)

然后我在 listModel 中添加一些模拟项目以单击某些内容。

这个想法是列表应该显示项目,当您单击时,3d View 将显示它们。我做了如下:ListView 定义如下

class ListView(QtGui.QWidget):
itemActivated = QtCore.Signal()

def __init__(self, parent):
super(ListView, self).__init__(parent)

self._ui = self._createUi()
self._ui.qListWidget.itemActivated.connect(self._itemActivatedSlot)

def _createUi(self):

ui = UIElems()

hLayout = QtGui.QHBoxLayout(self)
ui.qListWidget = QtGui.QListWidget(self)
hLayout.addWidget(ui.qListWidget)

return ui

def _itemActivatedSlot(self, listitem):
# I CONFIRM THIS LINE IS EXECUTED
# SO THE EMIT IS CORRECTLY PERFORMED
self.itemActivated.emit()

请注意我如何为 ListView 定义 itemActivated 信号。我拦截 QListWidget(注意 Q)itemActivated,将其重定向到 ListView 的 protected 插槽,然后发出 ListView 信号,该信号将被外部监听。我这样做是因为我希望 ListView 完全包裹其内部控件。

谁在监听此 ListView itemActivated 事件? ListController 应该获取此信号,并相应地设置选择,以便 3d Controller 可以更新 3d View 。

这是列表 Controller

class ListController(QtCore.QObject):
def __init__(self, listView, listModel, selectionModel):
super(ListController, self).__init__()

self._listView = listView
self._listModel = listModel
self._selectionModel = selectionModel

self._listModel.changed.connect(self._listModelChanged)

# HERE IS THE PROBLEM
self._listView.itemActivated.connect(self._listViewItemActivated)

# PLACEHOLDER

def _listModelChanged(self):
self._listView.setItems(self._listModel.items())

def _listViewItemActivated(self): ### ... AND HERE
identifier = self._listView.currentSelectedId()
self._selectionModel.setSelected(identifier)

如您所见,在 Controller 中,我连接到 ListView 的信号 itemActivated,并期望看到调用的正确插槽。不幸的是这并没有发生。但是,如果我在占位符处添加以下行

 self._listView.itemActivated.emit()
也就是说,我在“从外部”连接后强制发出信号(我在 Controller 中,并且发出 View 中定义的信号),我开始接收事件并且一切按预期工作:ListController。每次我单击项目时都会调用 _listViewItemActivated ,并且 3D View 也会随之更新。

我尝试在一个较小的程序中复制这个案例,但我总是得到一些有用的东西。此时此刻,你是我唯一的希望,因为我所经历的行为根本没有意义。

编辑

此代码显示了相同的行为,只是现在即使强制发射也无法解决。尝试单击 ListView 中的“hello”元素。它将打印消息,但不会随之发出“胜利!”行将不会被打印。

import sys 
from PySide import QtGui, QtCore

class ListView(QtGui.QWidget):
itemActivated = QtCore.Signal()

def __init__(self, parent):
super(ListView, self).__init__(parent)

qlistview = QtGui.QListWidget(self)
qlistview.addItem("hello")
qlistview.itemActivated.connect(self._itemActivatedSlot)

def _itemActivatedSlot(self, listitem):
print "Activated "+ listitem.text() # this is printed.

self.itemActivated.emit() # Why not emitted or received ?!?!

class ListController(QtCore.QObject):
def __init__(self, listView):
super(ListController, self).__init__()

self._listView = listView
self._listView.itemActivated.connect(self._listViewItemActivated)
# self._listView.itemActivated.emit() # uncomment to see that the signal is actually connected and the mechanism works

def _listViewItemActivated(self):
print "_listViewItemActivated activated!!! Victory!"

class MainWindow(QtGui.QMainWindow):
def __init__(self):
super(MainWindow, self).__init__()

listView = ListView(self)
listController = ListController(listView)

if __name__ == "__main__":
app = QtGui.QApplication(sys.argv)
app.setApplicationName("Test")
mw = MainWindow()
mw.show()
sys.exit(app.exec_())

最佳答案

我可能是错的,但尝试添加插槽装饰器,例如:

@QtCore.Slot()
def _listViewItemActivated(self):
identifier = self._listView.currentSelectedId()
self._selectionModel.setSelected(identifier)

或者也许只是添加一些虚拟参数来传递?

@QtCore.Slot(int)
...

itemActivated = QtCore.Signal(int)

关于python - 奇怪的发射行为。执行emit时无法调用slot,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9386928/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com