首页 > 解决方案 > 尝试从 ttkwidgets 解析 CheckboxTreeview 并获取树中所有项目的复选框状态

问题描述

我一直在尝试解析一个复选框树视图,以返回一个字典,其中项目的 ID 作为键,复选框状态作为“已选中”、“未选中”和“三态”的值。但是,我尝试使用 CheckboxTreview.get_checked() 对项目进行排序。我不知道我是不是用错了,或者这只是包装的一个缺陷,但它只返回最高级别的检查项目(''作为父级)。

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")

我解析了以下文本文件以使用,缩进表示级别,最后一个数字表示是否选中。在这种情况下,例如 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

标签: pythontkintercheckboxtreeview

解决方案


没有返回预期结果的原因tree.get_checked()是下级选中项的父项未选中,代码假定未选中父项的所有子项均未选中。

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

您可以做的是创建检查/取消选中项目并传播状态更改的方法:

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()将返回预期结果。

截屏

替代方案:如果您不想将项目的状态传播给它们的祖先并保持您的树视图就像在您的代码中一样,您可以递归地在整个树中查找已检查的项目(而不仅仅是在检查祖先时/ 三态):

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

推荐阅读