我在使用Tkinter after((方法时遇到问题。事实上,我想做的是随着时间的推移尽快改变一些输入框的背景颜色。让我们来看这段代码(它与我正在编写的脚本不同,但描述的情况相同(:
import Tkinter as tk
root = tk.Tk()
root.option_add("*Entry.Font","Arial 32 bold")
emptyLabel=tk.Label()
emptyLabel.grid(row=4) #Empty label for geometry purpose
entryList=[]
for x in range(4):
entryList.append([])
for y in range(4):
entryList[x].append('')
entryList[x][y]=tk.Entry(root, bg="white",width=2,justify="center",
takefocus=True,insertofftime=True)
entryList[x][y].grid(row=x,column=y)
solvebt=tk.Button(root,text='Solve').grid(row=5,column=2)
newgamebt=tk.Button(root,text='New').grid(row=5,column=1)
#BROKEN PART STARTS HERE
def changebg(x,y):
entryList[x][y]['bg']='yellow'
for x in range(4):
for y in range(4):
entryList[x][y].after(300,changebg(x,y))
#Same result with root.after(300,changebg(x,y))
root.mainloop()
问题是,当我开始这个程序时,我希望它能向我显示它"绘制"的样子,一次一个,所有的输入框都是黄色的。相反,发生的是,程序冻结(300*16(毫秒,然后,突然,每个输入框都是黄色的!
问题就在这里:
def changebg(x,y):
entryList[x][y]['bg']='yellow'
for x in range(4):
for y in range(4):
entryList[x][y].after(300,changebg(x,y))
#Same result with root.after(300,changebg(x,y))
在double-for循环中立即调用changebg
——然后将返回值(None
(传递给root.after
。这不会导致你所描述的延迟。也许你的实际代码看起来像:
for x in range(4):
for y in range(4):
entryList[x][y].after(300,lambda x=x,y=y : changebg(x,y))
这将导致你实际描述的行为。最终,您需要的是使小部件列表变平,然后一次通过一个——如果存在,则注册下一个:
import itertools
all_entries = itertools.chain.from_iterable(entryList)
def changebg(ientries):
ientries = iter(ientries) #allow passing a list in as well ...
entry = next(ientries,None)
if entry is not None:
entry['bg'] = 'yellow' #change the color of this widget
root.after(300,lambda : changebg(ientries)) #wait 300ms and change color of next one.
changebg(all_entries)