我一直在尝试解析复选框 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()
将返回预期结果。
替代方案:如果您不想将项目的状态传播到其祖先并保持 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/