Python:使用Tkinter动态更改条目颜色

1 投票
1 回答
4195 浏览
提问于 2025-04-17 16:57

我在使用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)毫秒,然后突然间,所有输入框都变成了黄色!

1 个回答

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 函数,然后把它的返回值(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)

撰写回答