尝试从 ttkwidgets 解析 CheckboxTreeview 并获取树中所有项目的复选框状态

Trying to parse a CheckboxTreeview from ttkwidgets and get the checkbox status of all the items in the tree

我一直在尝试将复选框树视图解析为 return 字典,其中项目的 ID 作为键,复选框状态作为值 "checked","unchecked" 和"tristate"。但是,我尝试使用 CheckboxTreview.get_checked() 对项目进行排序。我不知道我是不是用错了,或者这只是包装中的一个缺陷,但它只是 return 最高级别('' 作为父级)的选中项目。

from tkinter import *
from tkinter import ttk
from ttkwidgets import CheckboxTreeview

def parse_Tree(tree, parent):

    children = list(tree.get_children(parent))
    checkedList = tree.get_checked()
    itemDic = {}
    #print(checkedList)

    for item in children:
        if tree.get_children(item) == () and item in checkedList:
            itemDic[item] = "checked"
        elif tree.get_children(item) != () and item in checkedList:
            itemDic[item] = "checked"
            itemDic.update(parse_Tree(tree, item))
        elif tree.get_children(item) != () and item not in checkedList:
            for boxStatus in parse_Tree(tree, item).values():
                if boxStatus == "checked" or boxStatus == "tristate":
                    itemDic[item] = "tristate"
                else:
                    itemDic[item] = "unchecked"
                    itemDic.update(parse_Tree(tree,item))
        else:
            itemDic[item] = "unchecked"

    return itemDic

def listTreeview(textFile):

    list = []
    file = open(textFile, "r")
    treeview = file.read().split("\n")
    file.close()
    for item in treeview:
        list += [item.split(",")]
    root = Tk()
    master = ''
    level = []
    tree = CheckboxTreeview(root)
    for index,i in enumerate(list):
        indent = 0

        while i[0][indent] == ' ': indent += 1

        if indent%4:
            print("wrong indentation")
            break
        else:
            i[0] = i[0].replace(' ','')

        level.append(int(indent/4))

        if len(level)==1:
            tree.insert(master,'end',i[0], text = i[0])
        elif level[index]-level[index-1] == 1:
            master = list[index - 1][0]
            tree.insert(master, 'end', i[0], text=i[0])
        elif level[index]-level[index-1] < 0:
            prev = index-1
            while level[index] != level[prev]:
                prev -= 1
            master = tree.parent(list[prev][0])
            tree.insert(master,'end',i[0], text = i[0])
        elif level[index] - level[index - 1] > 1:
            print('wrong indentation')
        else: #level hasnt change
            tree.insert(master, 'end', i[0], text=i[0])
        if i[1] == '1':
            tree.change_state(i[0], "checked")


    tree.expand_all()
    dic = parse_Tree(tree,'')
    print(dic)
    tree.pack()
    root.mainloop()

listTreeview("Treeview.txt")

我解析了下面的文本文件来处理,缩进表示级别,最后一个数字表示是否被选中或者not.In这种情况例如,item4.1.1应该显示为item4.1.1: "checked" 但它没有...

有没有其他方法可以通过复选框树视图并获取每个项目的状态?

item0,1
item1,0
    item1.1,0
    item1.2,0
item2,0
    item2.1,1
    item2.2,0
        item2.2.1,1
        item2.2.2,0
            item2.2.2.1,0
    item2.3,0
        item2.3.1,1
item3,1
item4,0
    item4.1,1
    item4.2,0
    item4.2.1,0

之所以tree.get_checked()没有return预期的结果,是因为下层checked items的parents没有被checked,代码假定未checked parent的所有children都没有被checked .

小部件的缺点是它仅在用户单击时传播状态更改,例如如果用户选中了一个项目,则该项目的父项变为已选中或三态。但是当您从代码中更改项目的状态时,这不会发生。

您可以做的是创建 check/uncheck 项目并传播状态更改的方法:

from tkinter import Tk
from ttkwidgets import CheckboxTreeview as Tree


class CheckboxTreeview(Tree):

    def item_check(self, item):
        """Check item and propagate the state change to ancestors and descendants."""
        self._check_ancestor(item)
        self._check_descendant(item)

    def item_uncheck(self, item):
        """Uncheck item and propagate the state change to ancestors and descendants."""
        self._uncheck_descendant(item)
        self._uncheck_ancestor(item)

_(un)check_ancestor()_(un)check_descendant()CheckboxTreeview 的内部方法,当用户点击某个项目时使用。

所以现在,在 listTreeview() 中您可以使用 item_check() 而不是 change_state(),但是,您需要替换

if i[1] == '1':
    tree.change_state(i[0], "checked")

来自

if i[1] == '1':
    tree.item_check(i[0])
else:
    tree.item_uncheck(i[0])

因为选中新创建的项目的父级时,默认情况下也会选中该项目。

现在,树中选中项目的祖先具有正确的状态,因此 tree.get_checked() 将 return 预期结果。

备选方案:如果您不想将项目的状态传播到它们的祖先并保持您的树视图与代码中的一样,您可以递归地查找选中的项目在整棵树中(不仅在检查祖先/三态时):

def get_checked(tree):
    checked = []

    def rec_get_checked(item):
        if tree.tag_has('checked', item):
            checked.append(item)
        for ch in tree.get_children(item):
            rec_get_checked(ch)

    rec_get_checked('')
    return checked