我遇到了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)毫秒然后,所有突然出现的每个输入框都是黄色的!
答案 0 :(得分:1)
问题在于:
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))
你正在调用changebg
立即在double for循环中 - 然后你将返回值(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)