问题描述
在过去的几个小时里,我一直被困在这个问题上,似乎无法弄清楚为什么代码的行为如此,因此寻求任何帮助。基本上我在 tkinter python 3.6 中有一个树视图小部件,有 60000 多个项目。我创建了一个 onTripleClick 函数,“理想情况下”应该突出显示具有特定样式(使用标签)的第一个点击项目,并且当下一个项目是三重点击时,先前样式的项目应该回到原始状态(使用另一个标签)而新点击应该呈现样式状态。出于某种原因,只有第一个 tag_configure 被执行 - 如果我注释掉第一个 tag_configure,那么第二个 tag_configure 也会被执行 - 当第二个 tag_configure 满足 IF 条件时,我需要它们都被执行。非常感谢任何建议!
编辑:根据要求 - 我已经包含了一个最小版本的工作代码,上面解释的想法是用一种样式(trClicked 标签)突出显示三个左击行,当另一行三次单击时,先前单击的项目返回到原始状态(普通标签),而新项目采用样式(trClicked 标签)。按原样运行代码,然后注释掉第 9 行的第一个 tree.tag_configure() 位 - 并重新运行。这次第二个 tree.tag_configure(){line14} 被执行——我需要它们同时执行。谢谢!
from tkinter import *
from tkinter import ttk
def onTripleClick(event):
global selectedSigName,selectedSigPath,trClickedItemiid,toBeClearedItemiid
itemSelection = tree.selection()[0]
trClickedItemiid = tree.focus()
tree.tag_configure('trClicked'+str(trClickedItemiid),background='light green',foreground='black',font=( 'Helvetica',8,'bold','italic')) # font=(family,size,weight,slant,underline,overstrike)
# tree.tag_bind('trClicked'+str(trClickedItemiid),'<1>',trClickedItemiid)
prevTrClicked.append(trClickedItemiid)
if len(prevTrClicked) > 1:
toBeClearedItemiid = prevTrClicked.pop(0) # pop the 0th index and pass it to normal
tree.tag_configure('normal'+str(toBeClearedItemiid),background='pink',font=( 'Courier','normal','roman'))
# tree.tag_bind('normal'+str(toBeClearedItemiid),toBeClearedItemiid)
# Create main root object of TK class
root = Tk()
root.title('MyTreeview')
root.geometry("700x500")
# create frame to house treeview AND scrollbar
frame = Frame(root)
frame.pack(pady=5)
tree = ttk.Treeview(frame,height=20,selectmode="browse")
tree.pack(side=LEFT)
tree['columns'] = ("Column1","Column2","Column3")
#Format Columns
tree.column("#0",width=10,minwidth=10) # this is where the plus icon will live
tree.column("Column1",anchor=W,width=150)
tree.column("Column2",width=300)
tree.column("Column3",width=120)
# Create headings
tree.heading("#0",text="",anchor=W)
tree.heading("Column1",text="Column1",anchor=W)
tree.heading("Column2",text="Column2",anchor=W)
tree.heading("Column3",text="Column3",anchor=W)
# to be used by TrClicked Function for helping clear older clicked items
global prevTrClicked
prevTrClicked = [] #
rows= [
['TopMostParent1','2ndParent-ColE','ColF-3rdParent'],['TopMostParent2','5thParent-ColE','ColF-1stParent'],['TopMostParent3','4thParent-ColE','ColF-2ndParent'],['TopMostParent4',['TopMostParent6','3rdParent-ColE',['TopMostParent5','ColF-3rdParent']
]
count=0
for row in rows:
tree.insert(parent='',index='end',iid=count,text='',tags=('trClicked'+str(count),'normal'+str(count)),values=(row[0],row[1],row[2]))
count+=1
tree.bind("<Triple-1>",onTripleClick)
# tree.tag_bind('trClicked'+str(trClickedItemiid),trClickedItemiid)
# tree.tag_bind('normal'+str(toBeClearedItemiid),toBeClearedItemiid)
root.mainloop()
解决方法
就您而言,列表中标签的顺序很重要
如果我们改变这个
tags=('trClicked'+str(count),'normal'+str(count))
到这里
tags=('normal'+str(count),'trClicked'+str(count))
这次替换后,之前的元素变成粉红色。 你会看到相反的结果:变成粉红色的元素将不再变成绿色。
解决方案
您对两个标签(背景、前景、字体)使用相同的样式属性。
在这种情况下,单个 trClicked
标签就足够了。
您将仅根据事件更改样式属性值。
如果您使用一个标签('trClicked'):
tree.tag_configure('trClicked'+str(trClickedItemiid),background='light green',foreground='black',font=( 'Helvetica',8,'bold','italic'))
...
tree.tag_configure('trClicked'+str(toBeClearedItemiid),background='pink',font=( 'Courier','normal','roman'))
然后,一般来说,一切正常。
完整示例
from tkinter import *
from tkinter import ttk
# to be used by TrClicked Function for helping clear older clicked items
global clicked_queue
clicked_queue = [] # last 3 items
def onTripleClick(event):
global clicked_queue
# 'trClicked'+iid - a unique tag for each item
# the iid (string) of the item that currently has focus,or '' if no item has focus
current_item_iid = event.widget.focus()
if not current_item_iid:
return
print(event.widget.item(current_item_iid))
print("current:",current_item_iid) # iid=count
if clicked_queue:
# get the previous item,the last one in the queue
prev_item_iid = clicked_queue[-1]
print("prev item:",prev_item_iid)
if prev_item_iid == current_item_iid:
return
tree.tag_configure('trClicked'+prev_item_iid,'roman'))
tree.tag_configure('trClicked'+current_item_iid,'italic'))
# add current item to the right side of the queue
clicked_queue.append(current_item_iid)
print("new prev item:",current_item_iid)
if len(clicked_queue) == 3:
# get and remove an element from the left side of the queue
prev_prev_item_iid = clicked_queue[0]
clicked_queue.remove(prev_prev_item_iid)
if prev_prev_item_iid == current_item_iid:
return
# return to the default style
tree.tag_configure('trClicked'+prev_prev_item_iid,background='white','roman'))
# Create main root object of TK class
root = Tk()
root.title('MyTreeview')
root.geometry("700x500")
# create a consistent style for the background and font
style = ttk.Style()
style.configure('Treeview','roman'))
# create frame to house treeview AND scrollbar
frame = Frame(root)
frame.pack(pady=5)
tree = ttk.Treeview(frame,height=20,selectmode="browse")
tree.pack(side=LEFT)
tree['columns'] = ("Column1","Column2","Column3")
#Format Columns
tree.column("#0",width=10,minwidth=10) # this is where the plus icon will live
tree.column("Column1",anchor=W,width=150)
tree.column("Column2",width=300)
tree.column("Column3",width=120)
# Create headings
tree.heading("#0",text="",anchor=W)
tree.heading("Column1",text="Column1",anchor=W)
tree.heading("Column2",text="Column2",anchor=W)
tree.heading("Column3",text="Column3",anchor=W)
rows= [
['TopMostParent1','2ndParent-ColE','ColF-3rdParent'],['TopMostParent2','5thParent-ColE','ColF-1stParent'],['TopMostParent3','4thParent-ColE','ColF-2ndParent'],['TopMostParent4',['TopMostParent6','3rdParent-ColE',['TopMostParent5','ColF-3rdParent']
]
count = 1 # <- start at 1 or set iid=str(count),otherwise iid=0 will default to "I001"
for row in rows:
tree.insert(parent='',index='end',iid=count,text='',tags=('trClicked'+str(count),),values=(row[0],row[1],row[2]))
count += 1
tree.bind("<Triple-1>",onTripleClick)
root.mainloop()