【问题标题】:How to Drag and Drop from ListWidget onto ComboBox如何从 ListWidget 拖放到 ComboBox
【发布时间】:2014-05-11 12:51:05
【问题描述】:

目标是能够将 ListWidget 项拖放到组合框上。丢弃的项目应添加到组合框中。理想情况下,我们希望避免阅读 listWidget 的 .currentItem() 或 .selectedItems() 等任何技巧......想法?

from PyQt4 import QtGui, QtCore
import sys, os

class MyClass(object):
    def __init__(self):
        super(MyClass, self).__init__()
        self.name=None     
    def setName(self, arg):
        self.name=arg
    def getName(self):
        return self.name       

class DropableComboBox(QtGui.QComboBox):
    def __init__(self):
        self.model_mime_type = 'application/x-qabstractitemmodeldatalist'
        super(DropableComboBox, self).__init__()
        self.setAcceptDrops(True)

    def dragEnterEvent(self, event):
        if event.mimeData().hasFormat(self.model_mime_type) or event.mimeData().hasFormat('text/plain'):
            event.accept()
        else:
            event.ignore()

    def dropEvent(self, event):
        if event.mimeData().hasUrls():
            event.setDropAction(QtCore.Qt.CopyAction)
            event.accept()
            links = []
            for url in event.mimeData().urls():
                links.append(str(url.toLocalFile()))
            self.emit(QtCore.SIGNAL("dropped"), links)
        else:
            super(DropableComboBox, self).dropEvent(event)
            self.emit(QtCore.SIGNAL("dropped"))

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

        myQWidget = QtGui.QWidget()
        myBoxLayout = QtGui.QVBoxLayout()
        myQWidget.setLayout(myBoxLayout)
        self.setCentralWidget(myQWidget)

        self.listWidget = QtGui.QListWidget()
        self.listWidget.setDragDropMode(QtGui.QAbstractItemView.DragDrop)
        self.listWidget.setSelectionMode(QtGui.QAbstractItemView.ExtendedSelection)
        self.listWidget.currentItemChanged.connect(self.item_clicked)

        for i in range(3):
            my_item=QtGui.QListWidgetItem()
            name='ListWidget Item '+str(i)
            my_item.setText(name)
            self.listWidget.addItem(my_item)

            myObject=MyClass()
            myObject.setName(name) 

            my_item.setData(QtCore.Qt.UserRole, myObject)

        myBoxLayout.addWidget(self.listWidget)

        self.ComboBox = DropableComboBox()
        for i in range(3):
            self.ComboBox.addItem("Combobox Item " + str(i))

        self.ComboBox.currentIndexChanged.connect(self.combobox_selected)
        self.connect(self.ComboBox, QtCore.SIGNAL("dropped"), self.droppedOnCombobox)

        myBoxLayout.addWidget(self.ComboBox)

    def item_clicked(self, arg=None):
        print arg.data(QtCore.Qt.UserRole).toPyObject().getName()

    def combobox_selected(self, index):
        myObject=self.ComboBox.itemData(index).toPyObject()
        if hasattr(myObject, 'getName'): print myObject.getName()

    def droppedOnCombobox(self):
        print "Drop!"


if __name__ == '__main__':
    app = QtGui.QApplication(sys.argv)
    dialog_1 = Dialog_01()
    dialog_1.show()
    dialog_1.resize(480,320)
    sys.exit(app.exec_())

【问题讨论】:

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


    【解决方案1】:

    执行此操作的“正确”方法是使用 QDataStream 解压缩 mimedata。但是,这似乎需要使用 QMap,而这在 PyQt 中不可用。因此,相反,它可以通过为我们的脏工作获取代理模型,以一种有点 hacky(或者这应该是“棘手”?)的方式来完成:

    class DropableComboBox(QtGui.QComboBox):
        def __init__(self):
            super(DropableComboBox, self).__init__()
            self.model_mime_type = 'application/x-qabstractitemmodeldatalist'
            self.setAcceptDrops(True)
            self._proxymodel = QtGui.QStandardItemModel(self)
    
        def dropEvent(self, event):
            if event.mimeData().hasUrls():
                event.setDropAction(QtCore.Qt.CopyAction)
                event.accept()
                links = []
                for url in event.mimeData().urls():
                    links.append(str(url.toLocalFile()))
                self.emit(QtCore.SIGNAL("dropped"), links)
            elif event.mimeData().hasFormat(self.model_mime_type):
                self._proxymodel.setRowCount(0)
                self._proxymodel.dropMimeData(
                    event.mimeData(), QtCore.Qt.CopyAction,
                    0, 0, QtCore.QModelIndex())
                for index in range(self._proxymodel.rowCount()):
                    item = self._proxymodel.item(index, 0)                  
                    self.addItem(item.text())
                # no point calling the base-class dropEvent here,
                # because it's a NO-OP in QComboBox
                self.emit(QtCore.SIGNAL("dropped"))
    

    注意:

    这将复制列表小部件中的项目,而不是移动它们(您没有要求)。此外,如果您想防止添加重复项,请使用setDuplicatesEnabled。如果您想更改添加项目的方式,请使用setInsertPolicy

    【讨论】:

      猜你喜欢
      • 2020-06-25
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-08-12
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多