1


0

pyqt QTreeWidget setItemWidgetがドラッグ/ドロップ後に消える

QTreeWidget.setItemWidget()を使用して、親(ドラッグアンドドロップ)の後、ウィジェットをQTreeWidgetItemに入れておくようにしています。

しかし、次のコードをコンパイルすると、結果はQTreeWidgetItem内のウィジェットが消えることです。 何かアイディアは? どのコードがこれを修正しますか(QTreeWidgetItemに私が推測するウィジェットを再投入しますか?)

from PyQt4.QtCore import *
from PyQt4.QtGui import *


class InlineEditor (QWidget):
    _MUTE = 'MUTE'

    def __init__ (self, parent):
        QWidget.__init__ (self, parent)

        self.setAutoFillBackground (True)
        lo = QHBoxLayout()
        lo.setSpacing(4)

        self._cbFoo = QComboBox()
        for x in ["ABC", "DEF", "GHI", "JKL"]:
            self._cbFoo.addItem(x)

        self._leBar = QLineEdit('', self)
        lo.addWidget (self._cbFoo, 3)
        lo.addSpacing (5)
        lo.addWidget (QLabel ( 'Bar:'))
        lo.addWidget (self._leBar, 3)
        lo.addStretch (5)
        self.setLayout (lo)

class Form (QDialog):
    def __init__(self,parent=None):
        QDialog.__init__(self, parent)

        grid = QGridLayout ()
        tree = QTreeWidget ()

        # Here is the issue?
        tree.setDragDropMode(QAbstractItemView.InternalMove)

        tree.setColumnCount(3)

        for n in range (2):
            i = QTreeWidgetItem (tree) # create QTreeWidget the sub i
            i.setText (0, "first" + str (n)) # set the text of the first 0
            i.setText (1, "second")
            for m in range (2):
                j = QTreeWidgetItem(i)
                j.setText (0, "child first" + str (m))

        #b1 = QCheckBox("push me 0", tree) # this wont work w/ drag by itself either
        #tree.setItemWidget (tree.topLevelItem(0).child(1), 1, b1)

        item = InlineEditor(tree) # deal with a combination of multiple controls
        tree.setItemWidget(tree.topLevelItem(0).child(1), 1, item)

        grid.addWidget (tree)
        self.setLayout (grid)

app = QApplication ([])
form = Form ()
form.show ()
app.exec_ ()

2 Answer


2


私自身のtreeDropEventを書くことで、比較的「働く」修正を得ることができました…​ ただし、誰かがよりエレガントなソリューションを持っている場合は、気軽に共有してください。 以下のコードは、ツリー内のsetItemWidgetsを使用してドラッグ/ドロップする他の人の頭痛の種を解決します。

from PyQt4.QtCore import *
from PyQt4.QtGui import *

class InlineEditor (QWidget):
    _MUTE = 'MUTE'

    def __init__ (self, parent):
        QWidget.__init__ (self, parent)

        self.setAutoFillBackground (True)
        lo = QHBoxLayout()
        lo.setSpacing(4)

        self._cbFoo = QComboBox()
        for x in ["ABC", "DEF", "GHI", "JKL"]:
            self._cbFoo.addItem(x)

        self._leBar = QLineEdit('', self)
        lo.addWidget (self._cbFoo, 3)
        lo.addSpacing (5)
        lo.addWidget (QLabel ( 'Bar:'))
        lo.addWidget (self._leBar, 3)
        lo.addStretch (5)
        self.setLayout (lo)

class Tree(QTreeWidget):
    def __init__(self, parent=None):
        QTreeWidget.__init__(self, parent)

        # Here is the issue?
        self.setDragDropMode(QAbstractItemView.InternalMove)
        self.installEventFilter(self)
        self.setColumnCount(3)
        self.dropEvent = self.treeDropEvent

        for n in range (2):
            i = QTreeWidgetItem (self) # create QTreeWidget the sub i
            i.setText (0, "first" + str (n)) # set the text of the first 0
            i.setText (1, "second")
            for m in range (2):
                j = QTreeWidgetItem(i)
                j.setText (0, "child first" + str (m))

        self.item = InlineEditor(self) # deal with a combination of multiple controls
        self.setItemWidget(self.topLevelItem(0).child(1), 1, self.item)

    def treeDropEvent(self, event):
        dragItem = self.currentItem()

        QTreeWidget.dropEvent(self, event)
        # rebuild widget (grabbing it doesnt seem to work from self.itemWidget?)
        self.item = InlineEditor(self)
        self.setItemWidget(dragItem, 1, self.item)

class Form (QDialog):
    def __init__(self,parent=None):
        QDialog.__init__(self, parent)
    grid = QGridLayout ()
    tree = Tree ()
    grid.addWidget (tree)
    self.setLayout (grid)

app = QApplication ([])
form = Form ()
form.show ()
app.exec_ ()


0


私もそれがなぜ起こるかを知りたいです。

itemWidgetの「基になるC / C ++オブジェクトが削除された」ようです。 とにかく、それが修正されることを期待して、ウィジェットが消えた後にもう一度setItemWidget()を試みると、それが得られます。

QTreeWidgetItemがドロップされたときに呼び出されるイベントを入れましたが、ドロップされるとすぐにオブジェクトが削除されるようです

from PyQt4.QtCore import *
from PyQt4.QtGui import *


class InlineEditor (QWidget):
    _MUTE = 'MUTE'

    def __init__ (self, parent):
        QWidget.__init__ (self, parent)

        self.setAutoFillBackground (True)
        lo = QHBoxLayout()
        lo.setSpacing(4)

        self._cbFoo = QComboBox()
        for x in ["ABC", "DEF", "GHI", "JKL"]:
            self._cbFoo.addItem(x)

        self._leBar = QLineEdit('', self)
        lo.addWidget (self._cbFoo, 3)
        lo.addSpacing (5)
        lo.addWidget (QLabel ( 'Bar:'))
        lo.addWidget (self._leBar, 3)
        lo.addStretch (5)
        self.setLayout (lo)

class Tree(QTreeWidget):
    def __init__(self, parent=None):
        QTreeWidget.__init__(self, parent)

        # Here is the issue?
        self.setDragDropMode(QAbstractItemView.InternalMove)
        self.installEventFilter(self)
        self.setColumnCount(3)

        for n in range (2):
            i = QTreeWidgetItem (self) # create QTreeWidget the sub i
            i.setText (0, "first" + str (n)) # set the text of the first 0
            i.setText (1, "second")
            for m in range (2):
                j = QTreeWidgetItem(i)
                j.setText (0, "child first" + str (m))

        #b1 = QCheckBox("push me 0", tree) # this wont work w/ drag by itself either
        #tree.setItemWidget (tree.topLevelItem(0).child(1), 1, b1)

        self.item = InlineEditor(self) # deal with a combination of multiple controls
        self.setItemWidget(self.topLevelItem(0).child(1), 1, self.item)

    def eventFilter(self, sender, event):
        if event.type() == QEvent.ChildRemoved:
            print self.item._cbFoo # looks like this remains
            print self.item._cbFoo.currentText() # CRASH! but the data is gone
            #self.setItemWidget(self.topLevelItem(0).child(1), 1, self.item)
        return False


class Form (QDialog):
    def __init__(self,parent=None):
        QDialog.__init__(self, parent)

        grid = QGridLayout ()
        tree = Tree ()

        grid.addWidget (tree)
        self.setLayout (grid)

app = QApplication ([])
form = Form ()
form.show ()
app.exec_ ()