我希望用户能够在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)。但是您不能从函数内部执行此操作,因为在函数返回后它将被禁用。


但是您需要进行一些更改才能使用此技巧。

您需要使用Spinboxself设置为实例属性:

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

10-04 20:33