QListWidgetItem仅在按下按钮时可拖动

时间:2015-11-24 23:30:59

标签: python qt drag-and-drop pyqt qlistwidget

我正在尝试创建一个包含自定义小部件的QListWidget,每个小部件都包含一个拖动按钮。我们的想法是,当按下其中名为“拖动”的按钮时,只能拖放QListWidgetItem。释放鼠标/按钮后,QListWidgetItem将落入其位置。整个事情应该像普通的QListWidget一样,除了鼠标按下的位置必须在名为“拖动”的按钮上。

我当前的方法是在初始化时将所有QListWidgetItem标志设置为~Qt.ItemIsDragEnabled,然后在单击按钮时将其重新打开。然后在离开小部件后,我再向后拖动。

以下是工作代码,用于演示我的位置。显然这不是我真正想要的,但我无法弄清楚如何将“拖动”按钮内的鼠标按下事件一次性传递到QListWidgetItem后面。

import sys
import os
from PyQt4 import QtGui
from PyQt4.QtCore import *
from PyQt4.QtGui import *


class DragArea(QFrame):
    """docstring for DragButton"""
    def __init__(self, parent=None, text='Test', index=None):
        super(DragArea, self).__init__()
        self.parent = parent
        self.index = index
        self.layout = QHBoxLayout()
        self.setLayout(self.layout)

        self.label = QLabel()
        self.button = QPushButton()

        self.label.setText(text)
        self.button.setText('Drag')

        self.layout.addWidget(self.label)
        self.layout.addWidget(self.button)

        self.button.pressed.connect(self.drag)

    def drag(self):
        item = self.parent.bodyMap[self.index]
        item.setFlags(
            Qt.ItemIsSelectable |
            Qt.ItemIsEnabled |
            Qt.ItemIsDragEnabled
            )

    def leaveEvent(self, event):
        item = self.parent.bodyMap[self.index]
        item.setFlags(
            item.flags() &
            ~Qt.ItemIsSelectable &
            ~Qt.ItemIsEnabled &
            ~Qt.ItemIsDragEnabled
            )


class QuoteArea(QListWidget):
    def __init__(self, parent=None):
        super(QuoteArea, self).__init__()

        self.bodyMap = {}
        self.setStyleSheet("""
            QListWidget {
                background: gray;
            }

            QListWidget::item:selected {
                background: None;
                }

            """)
        self.setDragDropMode(QAbstractItemView.InternalMove)
        self.setDefaultDropAction(Qt.MoveAction)
        self.setDragDropOverwriteMode(False)
        self.setAcceptDrops(True)
        self.setDropIndicatorShown(True)
        self.setDragEnabled(True)
        self.setSelectionRectVisible(False)
        self.setResizeMode(QListView.Adjust)
        self.setGeometry(30, 40, 400, 500)
        for a in xrange(5):
            self.addQuote(a)


    def addQuote(self, index):
        text = 'Lorum ipsum ' + str(index)
        widget = DragArea(text=text, parent=self, index=index)
        item = QListWidgetItem()
        item.setFlags(item.flags() &
            ~Qt.ItemIsSelectable &
            ~Qt.ItemIsEnabled &
            ~Qt.ItemIsDragEnabled
            )
        self.bodyMap[index] = item
        item.setSizeHint(
            QSize(item.sizeHint().width(), widget.sizeHint().height())
            )
        self.addItem(item)
        self.setItemWidget(item, widget)


def main():
    app = QtGui.QApplication(sys.argv)
    ex = QuoteArea()
    ex.show()
    sys.exit(app.exec_())

if __name__ == '__main__':
    main()

1 个答案:

答案 0 :(得分:0)

我认为按钮会占用所有鼠标事件。新闻和动作。我想唯一的解决方案是从QPushButton派生一个新的按钮类并重新实现mouseEvents,因为您需要按钮将mouseEvents报告给List。

您的按钮可以引用需要接收事件的对象。

像这样:

class MyPushButton(QPushButton): # derive from QPushButton

   # .. some more code here

   def mouseEvent(self, event):
      QPushButton.mouseEvent(event) # call mouse event of base class to get the button animation
      self.draggerObject.mouseEvent(event) # call the mouse event of the dragging classes using the event.

这只是一个非常基本的示例,您可能需要进行一些鼠标坐标校正才能使其正常工作。但重要的是控制按钮接收的鼠标事件。