PyQt5 QListWidget删除鼠标悬停QListItem

时间:2018-05-21 18:34:44

标签: python pyqt pyqt5 qlistwidget qlistwidgetitem

我陷入了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. 为每个项目小部件分配一个名称标签,然后在单击按钮后循环遍历listwidget并使用takeItem()删除具有相同标签的那个。我只是觉得这种做法并不优雅......

  4. 我认为如果我能够以某种方式获取小部件的行号,哪个按钮被点击并使用takeItem()上的self.list_widget.row(row_number)来删除,那将是最有意义的。我无法弄清楚如何。

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

    PyQt QListWidget custom items

    重新编写自定义小部件到QListWidget

1 个答案:

答案 0 :(得分:1)

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

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

  • 使用该职位我们通过itemAt()获取该项目。

  • 使用该项目,我们可以获得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