QTreeWidget類中的常用方法
方法 | 描述 |
---|---|
setColumnWidth(int column,int width) | 將指定列的寬度設(shè)置為給定的值width |
insertTopLevelItems() | 在視圖的頂層索引中插入項(xiàng)目列表 |
expandAll() | 展開所有的樹形節(jié)點(diǎn) |
invisibleRootItem() | 返回樹形控件中不可見的根選項(xiàng) |
selectedItems() | 返回所有選定的非隱藏項(xiàng)目的列表 |
QTreeWidgetItem類中的常用方法
方法 | 描述 |
---|---|
addChild() | 將子項(xiàng)追加到子列表中 |
setText() | 設(shè)置顯示的節(jié)點(diǎn)文本 |
Text() | 返回顯示的節(jié)點(diǎn)文本 |
setCheckState(column,state) | 設(shè)置指定列的選中狀態(tài):Qt.Checked(節(jié)點(diǎn)選中),Qt.Unchecked(節(jié)點(diǎn)未選中) |
setIcon(column,icon) | 在指定的列中顯示圖標(biāo) |
準(zhǔn)備一個(gè)小demo
常見小demo
簡介:商品種類和商品的樹形嵌套結(jié)構(gòu)demo
功能:點(diǎn)擊按鍵獲取當(dāng)前所有被選中的商品(夾帶了一點(diǎn)私貨〃‘▽'〃)
注意:篇幅有限,沒有寫子節(jié)點(diǎn)和父節(jié)點(diǎn)聯(lián)動(dòng)選中,所以在選擇子節(jié)點(diǎn)時(shí)麻煩自行將父節(jié)點(diǎn)選上,不然會(huì)跳過。
代碼塊兒:
import sys from PyQt5.QtWidgets import QTreeWidgetItem, QTreeWidget, QWidget, QVBoxLayout, QPushButton, QApplication from PyQt5.QtCore import Qt class Demo(QWidget): def __init__(self): super().__init__() # 實(shí)例化一個(gè)樹形結(jié)構(gòu),隱藏了header self.tree = QTreeWidget() self.tree.setHeaderHidden(True) # 頂級分支 self.tree_main = QTreeWidgetItem(self.tree) self.tree_main.setText(0, '商品種類') # 設(shè)置一些二級分支 tree_second = ['電子產(chǎn)品', '水果', '日用品', '喜歡的人'] self.gen_branch(self.tree_main, tree_second) # 設(shè)置一些三級分支 tree_fruit = ['蘋果', '香蕉', '梨'] tree_daily_use = ['紙巾', '毛巾'] tree_lovers = ['迪迪1號(hào)', '迪迪2號(hào)'] # child(1) 意思是分支的第1個(gè)節(jié)點(diǎn), 序號(hào)從0算起 self.gen_branch(self.tree_main.child(1), tree_fruit) self.gen_branch(self.tree_main.child(2), tree_daily_use) self.gen_branch(self.tree_main.child(3), tree_lovers) # 一個(gè)按鈕 self.pushButton = QPushButton('選好了') # 顯示出來 self.qvl = QVBoxLayout() self.qvl.addWidget(self.tree) self.qvl.addWidget(self.pushButton) self.setLayout(self.qvl) # 綁定一下槽函數(shù),傳入主要的分支節(jié)點(diǎn) self.pushButton.clicked.connect(lambda: self.get_checked(self.tree_main)) @staticmethod def gen_branch(node: QTreeWidgetItem, texts: list): """ 給定某個(gè)節(jié)點(diǎn)和列表 在該節(jié)點(diǎn)生成列表內(nèi)分支""" for text in texts: item = QTreeWidgetItem() item.setText(0, text) item.setCheckState(0, Qt.Unchecked) node.addChild(item) def get_checked(self, node: QTreeWidgetItem)->list: """ 得到當(dāng)前節(jié)點(diǎn)選中的所有分支, 返回一個(gè) list """ temp_list = [] # 此處看下方注釋 1 for item in node.takeChildren(): # 判斷是否選中 if item.checkState(0) == Qt.Checked: temp_list.append(item.text(0)) # 判斷是否還有子分支 if item.childCount(): temp_list.extend(self.get_checked(item)) node.addChild(item) print(temp_list) return temp_list if __name__ == '__main__': app = QApplication(sys.argv) win = Demo() win.show() sys.exit(app.exec_())
注釋01:在這個(gè)函數(shù)中,我傳入了一個(gè) node 節(jié)點(diǎn),takeChildren() 這個(gè)方法會(huì)將該node節(jié)點(diǎn)的所有一級子分支拿出來(刪除),并返回節(jié)點(diǎn)的所有一級分支的列表,如下所示。該方法只能返回一級的節(jié)點(diǎn)信息,利用 childCount() 來判斷是否有子分支,有則遞歸,一直到最底部的節(jié)點(diǎn)。因?yàn)樵讷@取的時(shí)候 takeChildren() 刪除了所有節(jié)點(diǎn),所以在操作結(jié)束后重新加入到 node 節(jié)點(diǎn)中
[ <PyQt5.QtWidgets.QTreeWidgetItem object at 0x0000000008464708>, <PyQt5.QtWidgets.QTreeWidgetItem object at 0x0000000008464798>, ]
這樣的做法有什么好處和壞處?
最大的好處無疑是不需要?jiǎng)?chuàng)建額外的變量去存儲(chǔ)子節(jié)點(diǎn)的信息,子節(jié)點(diǎn)的信息和順序都是實(shí)時(shí)獲取的而非前期就定好了的。壞處,我設(shè)想這個(gè)方法用多了可能會(huì)存在節(jié)點(diǎn)順序改變的情況。比如“蘋果、香蕉”變成了“香蕉、蘋果”,目前未出現(xiàn)。
對比網(wǎng)上的方法
有一個(gè)關(guān)于 QTreeWidgetItemIterator 的辦法,這是Qt中自帶的遍歷器,大概如下
item = QtWidgets.QTreeWidgetItemIterator(self.treeWidget),
用 item.value() 來定位到一個(gè)節(jié)點(diǎn),item.value() 的實(shí)例就是上文列表中的那種對象,個(gè)人感覺差不太多。
還有一種比較暴力做法。在生成子節(jié)點(diǎn)的時(shí)候?qū)⑺凶庸?jié)點(diǎn)放到當(dāng)前類的作用域中,也就是作為屬性存在。
self.item1 = QTreeWidgetItem()
或是生成的時(shí)候保存在一個(gè)定義在作用域的列表中,這么做有一個(gè)壞處,節(jié)點(diǎn)的信息都是提前定好了的。但實(shí)際上遇到的情況更多應(yīng)該是未知的。
self.item_list = []
self.item_list.append([... ... ])
到此這篇關(guān)于PyQt5 QTreeWidget 樹形結(jié)構(gòu)遞歸遍歷當(dāng)前所有節(jié)點(diǎn)的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)PyQt5 QTreeWidget 樹形結(jié)構(gòu)遍歷內(nèi)容請搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://blog.csdn.net/qq_39177678/article/details/109179729