tkinter中的小部件验证

时间:2017-10-21 07:53:18

标签: python validation tkinter widget

我希望用户能够在Spinbox小部件中输入整数值。如果输入的值不是整数或者是Spinbox限制之外的整数,则只要Spinbox失去焦点,Spinbox内容中的值就必须恢复为默认值。

在示例代码中,我使用Entry小部件只是因为Spinbox可能会失去焦点。

如果用户返回Spinbox输入新值,则不会验证其条目。 我在Interactively validating Entry widget content in tkinter中确认了Malcolm的评论,一旦此命令更新了小部件的值,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()

1 个答案:

答案 0 :(得分:1)

我在这里找到了一个很好的解释(在验证一章的最后一段):

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