2012-03-12 91 views
2

我有一個QtreeView作爲QComboBox中的視圖。在我的應用程序中,根項目是類別標籤,不能被選中。當我創建視圖,我想預先選擇其中一個子項目(第一個根項目是默認選中),但我無法弄清楚如何。這方面的例子(特別是蟒蛇)在地面上很薄。如何在QComboBox中的QTreeView中選擇項目

這裏是我的簡單的例子:

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

data = [ (("Cat A",False), [(("Thing 1",True), []),(("Thing 2",True), [])]), 
    (("Cat B",False), [(("Thing 3",True), []), (("Thing 4",True), [])])] 

class MyComboBox(QComboBox): 
    def __init__(self): 
     super(QComboBox,self).__init__() 
     self.setView(QTreeView()) 

     self.view().setHeaderHidden(True) 
     self.view().setItemsExpandable(False) 
     self.view().setRootIsDecorated(False) 

    def showPopup(self): 
     self.view().expandAll() 
     QComboBox.showPopup(self) 

class Window(QWidget): 
    def __init__(self): 

     QWidget.__init__(self) 

     self.model = QStandardItemModel() 
     self.addItems(self.model, data) 

     self.combo = MyComboBox() 
     self.combo.setModel(self.model) 

     layout = QVBoxLayout() 
     layout.addWidget(self.combo) 
     self.setLayout(layout) 

     # I can choose which combobox item to select here, but I am unable to 
     #choose child items 
     #self.combo.setCurrentIndex(1) 

    def addItems(self, parent, elements): 
     for text, children in elements: 
      item = QStandardItem(text[0]) 
      # root items are not selectable, users pick from child items 
      item.setSelectable(text[1]) 
      parent.appendRow(item) 
      if children: 
       self.addItems(item, children) 

if __name__ == "__main__": 
    app = QApplication(sys.argv) 
    window = Window() 
    window.show() 
    sys.exit(app.exec_()) 

我從實例工作herehere

的問題已經非常有人問before,而不是蟒蛇,和解決方案發布沒有按不爲我工作。

回答

2

這是一個替代和更通用的方式爲您的當前代碼。它將用於額外級別的嵌套項目以及任何可選項目的配置。

class MyComboBox(QComboBox): 
    def __init__(self): 
     super(MyComboBox,self).__init__() # your super was wrong. 
              # you need to pass the _current_ class name 
     self.setView(QTreeView()) 

     self.view().setHeaderHidden(True) 
     self.view().setItemsExpandable(False) 
     self.view().setRootIsDecorated(False) 

    def showPopup(self): 
     self.setRootModelIndex(QModelIndex()) # you need to add this 
     self.view().expandAll() 
     QComboBox.showPopup(self) 

    def setModel(self, model): 
     super(MyComboBox, self).setModel(model) 
     parent, row = self._firstSelectableItem() 
     if row is not None: 
      self.setRootModelIndex(parent) 
      self.setCurrentIndex(row) 

    def _firstSelectableItem(self, parent=QModelIndex()): 
     """ 
     Internal recursive function for finding the first selectable item. 
     """ 
     for i in range(self.model().rowCount(parent)): 
      itemIndex = self.model().index(i,0,parent) 
      if self.model().itemFromIndex(itemIndex).isSelectable(): 
       return parent, i 
      else: 
       itemIndex, row = self._firstSelectableItem(itemIndex) 
       if row is not None: 
        return itemIndex, row 
     return parent, None 
+0

多數民衆贊成在很好,適合我更好。感謝(兩者)花時間! (感謝超級筆記) – tom 2012-03-13 09:22:33

0

,如果你對你的組合框使用QTreeWidget同時作爲視圖和模型即

self.tree = QTreeWidget() 
self.combo.setModel(self.tree.model()) 
self.combo.setView(self.tree) 

此外,您將需要更改爲addItems()函數來構建一個QTreeWidget與QTreeWidgetItem的兒童和存儲這將工作您要選擇的子項目的父項(PARENTOFITEMSELECTED)。一旦你做到了這一點,下面將在您的treewidget選擇一個項目:

self.tree.setCurrentItem(PARENTOFITEMSELECTED, 0) 
self.combo.setRootModelIndex(self.tree.currentIndex()) 
self.combo.setCurrentIndex(CHILDITEMINDEX) 
self.tree.setCurrentItem(self.tree.invisibleRootItem(), 0) 
self.combo.setRootModelIndex(self.tree.currentIndex()) 

這是基於例如發現here

希望這會有所幫助。

+0

是的,這對我很有用 - 我曾經看過那個例子,但沒有向QTreeWidget做出飛躍。非常感謝! – tom 2012-03-12 18:00:27