首页 > 解决方案 > PyQt5 QListWidget 删除鼠标悬停 QListItem

问题描述

我陷入了 PyQt5 Qlistwidget 问题。以下是我拥有的当前代码:

from PyQt5 import QtWidgets, QtCore, QtGui
import sys

class FlatUIListWidget(QtWidgets.QListWidget):
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        self.setStyleSheet("""
                QScrollBar:vertical {              
            border: none;
            background:white;
            width:3px;
            margin: 0px 0px 0px 0px;
        }
        QScrollBar::handle:vertical {
            background: qlineargradient(x1:0, y1:0, x2:1, y2:0,
            stop: 0 rgb(32, 47, 130), stop: 0.5 rgb(32, 47, 130), stop:1 rgb(32, 47, 130));
            min-height: 0px;
        }
        QScrollBar::add-line:vertical {
            background: qlineargradient(x1:0, y1:0, x2:1, y2:0,
            stop: 0 rgb(32, 47, 130), stop: 0.5 rgb(32, 47, 130),  stop:1 rgb(32, 47, 130));
            height: 0px;
            subcontrol-position: bottom;
            subcontrol-origin: margin;
        }
        QScrollBar::sub-line:vertical {
            background: qlineargradient(x1:0, y1:0, x2:1, y2:0,
            stop: 0  rgb(32, 47, 130), stop: 0.5 rgb(32, 47, 130),  stop:1 rgb(32, 47, 130));
            height: 0 px;
            subcontrol-position: top;
            subcontrol-origin: margin;
        }
    """)


    def add_custom_widget(self, widget):
        myQListWidgetItem = QtWidgets.QListWidgetItem(self)
        # Set size hint
        myQListWidgetItem.setSizeHint(widget.sizeHint())
        # Add QListWidgetItem into QListWidget
        self.addItem(myQListWidgetItem)
        self.setItemWidget(myQListWidgetItem, widget)


class DummyWidget(QtWidgets.QWidget):
    delete_signal = QtCore.pyqtSignal(str)
    def __init__(self, text):
        super().__init__()
        self.text = text
        self.layout = QtWidgets.QVBoxLayout(self)
        self.delete_button = QtWidgets.QPushButton(text)
        self.delete_button.clicked.connect(self.on_delte_button_clicked)
        self.layout.addWidget(self.delete_button)


    def on_delte_button_clicked(self):
        self.delete_signal.emit(self.text)


class Container(QtWidgets.QWidget):
    def __init__(self):
        super().__init__()
        self.layout = QtWidgets.QVBoxLayout(self)
        self.list_widget = FlatUIListWidget()
        self.layout.addWidget(self.list_widget)
        for i in range(10):
            self.add_widget(str(i))

    def add_widget(self, text):
        item = DummyWidget(text)
        item.delete_signal.connect(self.on_item_delete)
        self.list_widget.add_custom_widget(item)

    def on_item_delete(self, string):
        try:
            # cursor = QtGui.QCursor()
            # print(cursor.pos())
            # print(self.list_widget.itemAt(cursor.pos()))
            # print(self.list_widget.selectedItems())
            # print(self.sender().parent().parent())
            # print(self.list_widget.currentRow())
            # self.list_widget.setCurrentItem(self.sender().parent())
            # self.list_widget.takeItem(self.sender().parent())
            print(string)
            self.list_widget.takeItem(self.list_widget.currentRow())
        except Exception as e:
            print(e)


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    mw = Container()
    mw.show()
    sys.exit(app.exec_())

基本上,我想创建一个QListWidget包含所有自定义项目的项目。在这些项目中的每一个中,都有一个允许用户自行删除的按钮。我尝试了以下方法:

  1. 将按钮点击信号连接到self.sender().deleteLater(). 这不起作用,因为它只会删除 . 中的小部件,QListWidgetItem并且会在QListWidget.

  2. 单击按钮后,向小部件发送信号Container并删除当前行。这也不起作用,因为您实际上需要先在小部件上选择并单击删除,否则它将删除第一行。

  3. 给每个项目小部件一个名称标签,然后单击按钮,循环遍历列表小部件并用于takeItem()删除具有相同标签的小部件。我只是认为这种方法并不优雅......

我认为如果我能以某种方式获取小部件的行号,则最有意义的是,该按钮已被单击并用于takeItem()删除self.list_widget.row(row_number)。我只是想不通怎么做。

我已阅读 and 的官方文档QListWidgetQListWidgetItems但没有给我答案。如果有人可以提供一些想法,我将不胜感激。

QListWidget从这篇文章中重写了添加自定义小部件: PyQt QListWidget custom items

标签: pythonpyqtpyqt5qlistwidgetqlistwidgetitem

解决方案


消除项目使用QWidget过程如下:

  • 获取小部件相对于 的位置viewport(),在这种情况下很容易,因为小部件(DummyWidget)是 的孩子viewport(),所以我们应该只使用self.sender().pos().

  • 使用我们通过 获取项目的位置itemAt()

  • 使用 item 我们通过row().

  • 然后我们用 消除它takeItem()

实现如下:

def on_item_delete(self, string):
    it = self.list_widget.itemAt(self.sender().pos())
    row = self.list_widget.row(it)
    item = self.list_widget.takeItem(row)
    del item

推荐阅读