2013-03-07 186 views
2

感谢this thread,我可以将小部件添加到view的顶级项目的第二个或更后一列QAbstractItemView(在我的示例中为QTreeView)。在QTreeView上设置小部件对象

但是可以将小部件添加到子项目吗?

这是我已经试过这部分进展顺利:

#!/usr/bin/env python 
import os 

from PyQt4.QtCore import QModelIndex, Qt 
from PyQt4.QtGui import QApplication, QItemSelectionModel, \ 
         QPushButton, QStandardItem, \ 
         QStandardItemModel, QTreeView 
from PyQt4.uic import loadUi 


class PrvTreeviewNest(QTreeView): 
    def __init__(self): 
     super(PrvTreeviewNest, self).__init__() 

     loadUi('/home/user/yourproject/resource/treeview_nest.ui') 

     # row can be 0 even when it's more than 0. 
     self._datamodel = QStandardItemModel(0, 2) 
     self.setModel(self._datamodel) 

     for i in range(4): 
      self._add_widget(i + 1) 

     self.show() 

    def _add_widget(self, n): 
     std_item = QStandardItem('{}th item'.format(n)) 
     self._datamodel.setItem(n, 0, std_item) 

     node_widget = QPushButton('{}th button'.format(n)) 
     qindex_widget = self._datamodel.index(n, 1, QModelIndex()) 
     self.setIndexWidget(qindex_widget, node_widget) 

     if n == 2: 
      std_item_child = QStandardItem('child') 
      std_item.appendRow(std_item_child) 

      node_widget_child = QPushButton('petit button') 
      qindex_widget_child = self._datamodel.index(n, 1, QModelIndex()) 
      self.setIndexWidget(qindex_widget_child, node_widget_child) 

if __name__ == '__main__': 
    import sys 
    app = QApplication(sys.argv) 
    # window = TreeviewWidgetSelectProve() 
    window = PrvTreeviewNest() 
    # window = TreeviewWidgetSelectProve() 
    window.resize(320, 240) 
    # window.show(); 
    window.setWindowTitle(
     QApplication.translate("toplevel", "Top-level widget")) 
    # window.add_cols() 

    sys.exit(app.exec_()) 

treeview_nest.ui可用。

您在下图中看到,项目child未显示按钮,并且其父按钮被覆盖。显然我不知道如何为它编写代码。

enter image description here


更新)我想出如何小部件添加到孩子。狡猾,使用QStandardItem.insertRow结合索引做的工作。在我上面的示例代码,替换为以下_add_widget

def _add_widget(self, n): 
     item_toplevel = QStandardItem('{}th item'.format(n)) 
     self._datamodel.setItem(n, 0, item_toplevel) 

     widget_toplevel = QPushButton('{}th button'.format(n)) 
     qindex_toplevel = self._datamodel.index(n, 1, QModelIndex()) 
     self.setIndexWidget(qindex_toplevel, widget_toplevel) 

     if n == 2: 
      item_child_col0 = QStandardItem('child col0') 
      item_child_col1 = QStandardItem('child col1') 
      #item_toplevel.appendRow(item_child_col0) 

      item_toplevel.insertRow(0, [item_child_col0, item_child_col1]) 

      widget_child = QPushButton('child widget') 
      qindex_child = item_child_col1.index() 
      self.setIndexWidget(qindex_child, widget_child) 

我敢肯定,这是不是最好的/理想的设计方式,但似乎为我工作。我在@Plulucious的回答之后提出了我的想法。谢谢!

enter image description here

回答

1

你的错误是在这条线:

qindex_widget_child = self._datamodel.index(n, 1, QModelIndex()) 
self.setIndexWidget(qindex_widget_child, node_widget_child) 

它给你排索引2列的模型,这是“帖后项目”的1 ,而不是你的孩子。您的孩子在std_item的第1行第1列。在Qt中,尤其是在QStandardItemModel中,儿童是相对于他们的父亲存储的,而不是相对于模型。

我不够熟悉PyQt的给你确切的代码,但你应该能够做这样的事情:

qindex_widget_child = std_item_child.index(QModelIndex()) 
self.setIndexWidget(qindex_widget_child, node_widget_child) 

或像这样:

qindex_widget_child = std_item.child(0, 1).index(QModelIndex()) 
self.setIndexWidget(qindex_widget_child, node_widget_child) 
+0

谢谢,我最终选择你作为答案,因为我得到了启发,并提出了一个解决方法,因为我修改了我的问题。评论你的想法,不幸的是,前者在列0上添加小部件。后者中的方法不存在(pyqt主要遵循C++ api)。 – IsaacS 2013-03-08 22:26:50

+0

我再次查看代码并将第二个选项修改为希望更有可能工作的东西。我对那个人有点sl。。很高兴你能够得到一些有用的东西! – Phlucious 2013-03-08 23:04:47

+0

你的解决方案或多或少是我在C++中所做的,所以我不会说它是马虎。对于是否可以先调用孩子(0,1)函数而不先指定孩子,我有点朦胧! – Phlucious 2013-03-08 23:07:36