我希望用户能够在Spinbox小部件中输入一个整数值。如果输入的值不是整数或超出Spinbox限制的整数,则一旦Spinbox失去焦点,Spinbox内容中的值就必须恢复为默认值。
在示例代码中,我将Entry窗口小部件仅用于Spinbox可能会失去焦点。
如果用户回到Spinbox输入新值,则他的输入无效。
我确认马尔科姆(Malcolm)在Interactively validating Entry widget content in tkinter中的评论,即该命令更新小部件的值后,validatecommand=command
功能将被清除。
有没有办法使输入到Spinbox中的值反复验证而不仅仅是一次?
from tkinter import *
class GUI:
def __init__(self):
# root window of the whole program
self.root = Tk()
self.root.title('Validate Spinbox')
# registering validate and invalid commands
validate_cmd = (self.root.register(self.validate), '%P')
invalid_cmd = (self.root.register(self.invalid))
# creating a Label
items_lbl = Label(self.root, text="# of items (5-10):")
items_lbl.grid(row=0, column=0)
# creating a Spinbox widget
self.items_var = StringVar()
self.items_var.set(7)
items_count = Spinbox(self.root, textvariable=self.items_var,
from_=5, to=10, width=4, validate='focusout',
validatecommand=validate_cmd,
invalidcommand=invalid_cmd)
items_count.grid(row=0, column=1)
# creating an Entry widget
self.entry_var = StringVar()
self.entry_var.set("Input some text here")
text_entry = Entry(self.root, textvariable=self.entry_var)
text_entry.grid(row=1, column=0)
def validate(self, entry):
try:
value = int(entry)
valid = value in range(5, 11)
except ValueError:
valid = False
if not valid:
self.root.bell()
return valid
def invalid(self):
self.items_var.set(7)
if __name__ == '__main__':
main_window = GUI()
mainloop()
最佳答案
我在这里(在“验证”一章的最后一段)找到了一个很好的解释:
http://stupidpythonideas.blogspot.fr/2013/12/tkinter-validation.html
如果您的validatecommand(或invalidcommand)直接或间接修改了Entry(例如,通过调用StringVar上的set),则函数返回后,验证将被禁用。 (这是Tk防止验证无限循环触发另一个验证的方式。)您必须将其重新打开(通过调用config)。但是您不能从函数内部执行此操作,因为在函数返回后它将被禁用。
但是您需要进行一些更改才能使用此技巧。
您需要使用Spinbox
将self
设置为实例属性:
self.items_count = Spinbox(self.root, textvariable=self.items_var,
from_=5, to=10, width=4, validate='focusout',
validatecommand=validate_cmd,
invalidcommand=invalid_cmd)
self.items_count.grid(row=0, column=1)
然后您可以在
self.items_count.after_idle(...)
方法内调用validate
:def validate(self, entry):
try:
value = int(entry)
valid = value in range(5, 11)
except ValueError:
valid = False
if not valid:
self.root.bell()
self.items_count.after_idle(lambda: self.items_count.config(validate='focusout'))
return valid