PyQt5 QTreeWidget 樹形結構遞歸遍歷當前所有節點的實現

QTreeWidget類中的常用方法

方法 描述
setColumnWidth(int column,int width) 將指定列的寬度設置為給定的值width
insertTopLevelItems() 在視圖的頂層索引中插入項目列表
expandAll() 展開所有的樹形節點
invisibleRootItem() 返回樹形控件中不可見的根選項
selectedItems() 返回所有選定的非隱藏項目的列表

QTreeWidgetItem類中的常用方法

方法 描述
addChild() 將子項追加到子列表中
setText() 設置顯示的節點文本
Text() 返回顯示的節點文本
setCheckState(column,state) 設置指定列的選中狀態:Qt.Checked(節點選中),Qt.Unchecked(節點未選中)
setIcon(column,icon) 在指定的列中顯示圖標

準備一個小demo

常見小demo
簡介:商品種類和商品的樹形嵌套結構demo
功能:點擊按鍵獲取當前所有被選中的商品(夾帶瞭一點私貨〃‘▽’〃)
註意:篇幅有限,沒有寫子節點和父節點聯動選中,所以在選擇子節點時麻煩自行將父節點選上,不然會跳過。

在這裡插入圖片描述

在這裡插入圖片描述

代碼塊兒:

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__()
        # 實例化一個樹形結構,隱藏瞭header
        self.tree = QTreeWidget()
        self.tree.setHeaderHidden(True)
        # 頂級分支
        self.tree_main = QTreeWidgetItem(self.tree)
        self.tree_main.setText(0, '商品種類')
        # 設置一些二級分支
        tree_second = ['電子產品', '水果', '日用品', '喜歡的人']
        self.gen_branch(self.tree_main, tree_second)
        # 設置一些三級分支
        tree_fruit = ['蘋果', '香蕉', '梨']
        tree_daily_use = ['紙巾', '毛巾']
        tree_lovers = ['迪迪1號', '迪迪2號']
        # child(1) 意思是分支的第1個節點, 序號從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)
        # 一個按鈕
        self.pushButton = QPushButton('選好瞭')
        # 顯示出來
        self.qvl = QVBoxLayout()
        self.qvl.addWidget(self.tree)
        self.qvl.addWidget(self.pushButton)
        self.setLayout(self.qvl)

        # 綁定一下槽函數,傳入主要的分支節點
        self.pushButton.clicked.connect(lambda: self.get_checked(self.tree_main))

    @staticmethod
    def gen_branch(node: QTreeWidgetItem, texts: list):
        """ 給定某個節點和列表 在該節點生成列表內分支"""
        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:
        """ 得到當前節點選中的所有分支, 返回一個 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:在這個函數中,我傳入瞭一個 node 節點,takeChildren() 這個方法會將該node節點的所有一級子分支拿出來(刪除),並返回節點的所有一級分支的列表,如下所示。該方法隻能返回一級的節點信息,利用 childCount() 來判斷是否有子分支,有則遞歸,一直到最底部的節點。因為在獲取的時候 takeChildren() 刪除瞭所有節點,所以在操作結束後重新加入到 node 節點中

[
<PyQt5.QtWidgets.QTreeWidgetItem object at 0x0000000008464708>, 
<PyQt5.QtWidgets.QTreeWidgetItem object at 0x0000000008464798>, 
]

這樣的做法有什麼好處和壞處?

最大的好處無疑是不需要創建額外的變量去存儲子節點的信息,子節點的信息和順序都是實時獲取的而非前期就定好瞭的。壞處,我設想這個方法用多瞭可能會存在節點順序改變的情況。比如“蘋果、香蕉”變成瞭“香蕉、蘋果”,目前未出現。

對比網上的方法

有一個關於 QTreeWidgetItemIterator 的辦法,這是Qt中自帶的遍歷器,大概如下

item = QtWidgets.QTreeWidgetItemIterator(self.treeWidget),

用 item.value() 來定位到一個節點,item.value() 的實例就是上文列表中的那種對象,個人感覺差不太多。

還有一種比較暴力做法。在生成子節點的時候將所有子節點放到當前類的作用域中,也就是作為屬性存在。

self.item1 = QTreeWidgetItem()

或是生成的時候保存在一個定義在作用域的列表中,這麼做有一個壞處,節點的信息都是提前定好瞭的。但實際上遇到的情況更多應該是未知的。

self.item_list = []
self.item_list.append([... ... ]) 

到此這篇關於PyQt5 QTreeWidget 樹形結構遞歸遍歷當前所有節點的實現的文章就介紹到這瞭,更多相關PyQt5 QTreeWidget 樹形結構遍歷內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: