PyQt:从 QListWidget 中删除自定义小部件

PyQt: removing custom widgets from a QListWidget

我正在将图片加载到 QGraphicsScene 然后当您单击照片时,它会在单击的区域放置圆圈。然后我添加一个自定义小部件,它将保留点列表。这个小部件有几个按钮。其中一个可以移动圆圈,另一个可以删除圆圈。我目前卡在删除部分。

问题

我可以很好地删除圆圈,也可以从列表中删除小部件。问题是列表中仍然有一个 space 来自小部件曾经所在的位置,并且由于我使用的是小部件中的按钮而不是选择项目,所以我不确定如何删除该位置。另外,如果我删除了一堆然后尝试添加一些 Python 它自己会崩溃,我不知道为什么。

我不确定是否可以完成我想要的,因为确实没有参考,或者我是否最好将按钮移至主要 window 并将它们从自定义小部件中删除。如果可以的话,我想保持原样。

有几个文件,所以我会把它放在 GitHub 上,所以我不会占用大量 space。非常感谢任何帮助。

GitHub Link

Link to GitHub Project

相关代码(来自Main.py):

class MainWindow(QMainWindow, Ui_MainWindow):
    ...

    def Add_History(self,pos):
        self.HistoryWidget = HistoryList()
        self.HistoryWidget.setObjectName("HistoryWidget_"+ str(Constents.analysisCount))
        myQListWidgetItem = QListWidgetItem(self.History_List)
        myQListWidgetItem.setSizeHint(self.HistoryWidget.sizeHint())
        self.History_List.addItem(myQListWidgetItem)
        self.History_List.setItemWidget(myQListWidgetItem, self.HistoryWidget)
        self.HistoryWidget.buttonPushed.connect(self.deleteObject)
        self.HistoryWidget.setXpoint(str(pos.x()))
        self.HistoryWidget.setYpoint(str(pos.y()))
        self.HistoryWidget.setHistoryName("Point "+ str(Constents.analysisCount))
        Constents.analysisCount = Constents.analysisCount + 1

    def deleteObject(self):
        sender = self.sender() #var of object that sent the request
        row_number = sender.objectName().split("_")
        number = row_number[1]
        x,y = Constents.analysisDetails[str(number)]# getting xy for object
        self.loadPicture.findAndRemoveAnalysisPoint(x,y) #calling the class with the scense to delete it
        Constents.analysisDetails.pop(str(number)) # get rid of the object in the variables
        HistoryWidget = self.findChildren(HistoryList, "HistoryWidget_"+number)[0] #find the actual object

        HistoryWidget.deleteLater() #delete that object
        #Simport pdb; pdb.set_trace()
        #self.History_List.takeItem(HistoryWidget)

图片

您必须同时删除项目小部件和项目本身。为此,需要一种从项目小部件(或其子小部件之一)获取项目的方法。一个干净的方法是使用列表小部件的 itemAt 方法,它可以从屏幕上的一个点获取一个项目。这样做的主要好处是它不需要知道项目的索引,当其他项目被删除时索引当然会改变。这也意味着项目小部件不需要了解与它们关联的特定列表小部件项目的任何信息。

这是您的 deleteObject 方法的重写,它实现了:

def deleteObject(self):
    sender = self.sender() #var of object that sent the request
    row_number = sender.objectName().split("_")
    number = row_number[1]
    x,y = Constents.analysisDetails[str(number)]# getting xy for object
    self.loadPicture.findAndRemoveAnalysisPoint(x,y) #calling the class with the scense to delete it
    Constents.analysisDetails.pop(str(number)) # get rid of the object in the variables

    # get the viewport coords of the item-widget
    pos = self.History_List.viewport().mapFromGlobal(
        sender.mapToGlobal(QtCore.QPoint(1, 1)))
    if not pos.isNull():
        # get the item from the coords
        item = self.History_List.itemAt(pos)
        if item is not None:
            # delete both the widget and the item
            widget = self.History_List.itemWidget(item)
            self.History_List.removeItemWidget(item)
            self.History_List.takeItem(self.History_List.row(item))
            widget.deleteLater()