Правильный ответ - используйте validatecommand
атрибут виджета. К сожалению, эта функция плохо документирована в мире Tkinter, хотя в мире Tk она достаточно документирована. Несмотря на то, что он плохо документирован, в нем есть все, что вам нужно для проверки, не прибегая к привязкам или отслеживанию переменных или изменению виджета изнутри процедуры проверки.
Хитрость заключается в том, чтобы знать, что вы можете заставить Tkinter передавать специальные значения вашей команде проверки. Эти значения предоставляют вам всю информацию, которую вам нужно знать, чтобы решить, действительны ли данные или нет: значение до редактирования, значение после редактирования, если редактирование допустимо, и несколько других битов информации. Однако, чтобы использовать их, вам нужно немного поработать, чтобы передать эту информацию вашей команде проверки.
Примечание: важно, чтобы команда проверки возвращала либо, True
либоFalse
. Все остальное приведет к отключению проверки для виджета.
Вот пример, который допускает только строчные буквы (и выводит все эти забавные значения):
import tkinter as tk
class Example(tk.Frame):
def __init__(self, parent):
tk.Frame.__init__(self, parent)
vcmd = (self.register(self.onValidate),
'%d', '%i', '%P', '%s', '%S', '%v', '%V', '%W')
self.entry = tk.Entry(self, validate="key", validatecommand=vcmd)
self.text = tk.Text(self, height=10, width=40)
self.entry.pack(side="top", fill="x")
self.text.pack(side="bottom", fill="both", expand=True)
def onValidate(self, d, i, P, s, S, v, V, W):
self.text.delete("1.0", "end")
self.text.insert("end","OnValidate:\n")
self.text.insert("end","d='%s'\n" % d)
self.text.insert("end","i='%s'\n" % i)
self.text.insert("end","P='%s'\n" % P)
self.text.insert("end","s='%s'\n" % s)
self.text.insert("end","S='%s'\n" % S)
self.text.insert("end","v='%s'\n" % v)
self.text.insert("end","V='%s'\n" % V)
self.text.insert("end","W='%s'\n" % W)
if S == S.lower():
return True
else:
self.bell()
return False
if __name__ == "__main__":
root = tk.Tk()
Example(root).pack(fill="both", expand=True)
root.mainloop()
Дополнительные сведения о том, что происходит под капотом при вызове register
метода, см. В разделе Проверка ввода tkinter.