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

标签 python tkinter checkbox treeview

我一直在尝试解析复选框 TreeView 以返回一个字典,其中项目的 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"但它不...

还有其他方法可以浏览复选框 TreeView 并获取每个项目的状态吗?

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() 没有返回预期结果的原因是未检查较低级别检查项的父项,并且代码假定未检查父项的所有子项都已检查未选中。

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

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

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

screenshot

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

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

关于python - 尝试从 ttkwidgets 解析 CheckboxTreeview 并获取树中所有项目的复选框状态,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59976372/

相关文章:

python - Tkinter 从外部函数更改类中的小部件?

android - 如何将复选框绑定(bind)到图像

python - Tensorflow:不支持的可调用

python - 函数适用于数据框的每一行,但不使用 df.apply

python - 使用 Python 脚本包装在 Bash-Shell 中输入的所有命令

python - 在 Tkinter 中找到相同的图像?

python - 如何创建一个可以使用或不使用参数的装饰器?

使用拖放的 Python GUI 编程,还包含标准输出重定向

android - 复选框首选项屏幕

Android 在同一垂直线上对齐复选框