Как очистить поле ввода в Tkinter?
Здравствуйте, нужна помощь. Изучаю Python и решил написать небольшую программу, чтобы немного разобраться во всем.
- Получать данные от пользователя и вносить их в файл.
- При запросе пользователя прочитать файл и вывести данные в табличном виде.
- Удалить ненужные данные (ошибка или потеря актуальности).
Изначально попробовал написать чисто консольный скрипт, теперь хочу добавить немного «красивостей». Использую для этого Tkinter.
На данном этапе готова функция с получением данных и записью в файл. Также есть функция прочтения и вывода в табличной форме (только в консоли, пока не придумал как их вывести в окно программы). Сейчас это весь готовый функционал.
Проблема заключается в следующем: пользователь вводит данные, нажимает кнопку, всё отлично записывается и выскакивает уведомление, дальше нужно очистить поля и ждать следующего действия пользователя.
Почитал про то, как можно очистить поле ввода, перепробовал несколько вариантов, но не один не сработал. Кто может подсказать как это исправить?
Если будут какие-то замечания по коду, пишите в комментариях.
from tkinter import * from tkinter import messagebox import pandas as pd import os def clear (): NameUserEntry.delete("0", END) NumbersEntry.delete("0", END) CarEntry.delete("0", END) VolumeSEntry.delete("0", END) PriceSEntry.delete("0", END) VolumeFEntry.delete("0", END) PriceFEntry.delete("0", END) ExpensesEntry.delete("0", END) def data_user(): os.system('CLS') print("Получение данных. ", end="") name_driver = NameUserEntry.get() number_driver = NumbersEntry.get() car_driver = CarEntry.get() volume_start = int(VolumeSEntry.get()) price_start = int(PriceSEntry.get()) volume_finish = int(VolumeFEntry.get()) price_finish = int(PriceFEntry.get()) expenses = int(ExpensesEntry.get()) profit = volume_finish * price_finish - volume_start * price_start - expenses print('Ок') os.system('CLS') print("Формируется словарь. ", end="") data_driver = < 'name':name_driver, 'numbers':number_driver, 'car':car_driver, 'Vs':volume_start, 'Ps':price_start, 'Vf':volume_finish, 'Pf':price_finish, 'Ex':expenses, 'Prof':profit, >print("Ок") print("Собираем фрейм. ", end="") columns = ['ФИО', 'Телефон', 'Машина', 'Объем на старте', 'Цена покупки', 'Объем на выходе', 'Цена продажи', 'Затраты', 'Профит',] data = [[data_driver['name'], data_driver['numbers'], data_driver['car'], data_driver['Vs'], data_driver['Ps'], data_driver['Vf'], data_driver['Pf'], data_driver['Ex'], data_driver['Prof'] ]] df = pd.DataFrame(data, columns=columns) print('Ок') print('Делаем запись. ', end='') df.to_csv(r'data.csv', mode='a', sep='/', header=False, index=False, encoding='utf-8') print('Ок') positive = 'Запись успешно добалена!' messagebox.showinfo('Уведомление', positive) def read_data(): df =pd.read_csv(r'data.csv', sep='/', names = ['ФИО', 'Телефон', 'Машина', 'Объем на старте', 'Цена покупки', 'Объем на выходе', 'Цена продажи', 'Затраты', 'Профит']) print(df) root = Tk() root.title('F1') root.geometry('500x600') button_padding = header_padding = #ФИО водителя NameUserLabel = Label(root, text='ФИО водителя: ', **header_padding) NameUserLabel.grid(row = 1, rowspan=1, column=1, sticky='w') NameUserEntry = Entry(root, bg='#fff', fg='#444') NameUserEntry.grid(row = 2, rowspan=1, column=1,sticky='w') #Номер телефона NumbersLabel = Label(root, text='Номер телефона: ') NumbersLabel.grid(row = 1, rowspan=1, column=3, sticky='w') NumbersEntry = Entry(root, bg='#fff', fg='#444') NumbersEntry.grid(row = 2, rowspan=1, column=3,sticky='w') #марка машины, номер CarLabel = Label(root, text='Машина(марка,г.с.з.: ', **header_padding) CarLabel.grid(row = 3, rowspan=1, column=1, sticky='w') CarEntry = Entry(root, bg='#fff', fg='#444') CarEntry.grid(row = 4, rowspan=1, column=1,sticky='w') #объем при покупке VolumeSLabel = Label(root, text='Объем (покупка): ', **header_padding) VolumeSLabel.grid(row = 6, rowspan=1, column=1, sticky='w') VolumeSEntry = Entry(root, bg='#fff', fg='#444') VolumeSEntry.grid(row = 7, rowspan=1, column=1,sticky='w') #цена покупки PriceSLabel = Label(root, text='Цена (покупка): ') PriceSLabel.grid(row = 6, rowspan=1, column=3, sticky='w') PriceSEntry = Entry(root, bg='#fff', fg='#444') PriceSEntry.grid(row = 7, rowspan=1, column=3,sticky='w') #объем при продаже VolumeFLabel = Label(root, text='Объем (продажа): ', **header_padding) VolumeFLabel.grid(row = 9, rowspan=1, column=1, sticky='w') VolumeFEntry = Entry(root, bg='#fff', fg='#444') VolumeFEntry.grid(row = 10, rowspan=1, column=1,sticky='w') #цена при продаже PriceFLabel = Label(root, text='Цена (продажа): ') PriceFLabel.grid(row = 9, rowspan=1, column=3, sticky='w',) PriceFEntry = Entry(root, bg='#fff', fg='#444') PriceFEntry.grid(row = 10, rowspan=1, column=3,sticky='w') #прочие затраты ExpensesLabel = Label(root, text='Прочие затраты: ', **header_padding) ExpensesLabel.grid(row = 12, rowspan=1, column=1, sticky='w') ExpensesEntry = Entry(root, bg='#fff', fg='#444', ) ExpensesEntry.grid(row = 13, rowspan=1, column=1,sticky='w') #профит, высчитывается и вносится в таблицу # кнопка сделать запись ButtonWrite = Button(root, text='Сделать запись', bg='#fff', fg='#444') ButtonWrite.grid(row = 15, rowspan=1, column=1, sticky='ws') ButtonWrite.config(command=data_user) # кнопка очистки данных ButtonDelete = Button(root, text='очистить данные', bg='#fff', fg='#444') ButtonDelete.grid(row = 15, rowspan=1, column=3, sticky='w') ButtonDelete.config(command=clear) root.mainloop()
Python tkinter entry delete
Элемент Entry представляет поле для ввода текста. С помощью конструктора Entry можно установить ряд параметров, основные из них:
- background : фоновый цвет
- cursor : курсор указателя мыши при наведении на текстовое поле
- foreground : цвет текста
- font : шрифт текста
- justify : устанавливает выравнивание текста. Значение LEFT выравнивает текст по левому краю, CENTER — по центру, RIGHT — по правому краю
- show : задает маску для вводимых символов
- state : состояние элемента, может принимать значения NORMAL (по умолчанию) и DISABLED
- textvariable : устанавливает привязку к элементу StringVar
- width : ширина элемента
Элемент Entry имеет ряд методов. Основные из них:
- insert(index, str) : вставляет в текстовое поле строку по определенному индексу
- get() : возвращает введенный в текстовое поле текст
- delete(first, last=None) : удаляет символ по индексу first. Если указан параметр last, то удаление производится до индекса last. Чтобы удалить до конца, в качестве второго параметра можно использовать значение END.
- focus() : установить фокус на текстовое поле
Простейшее текстовое поле:
from tkinter import * from tkinter import ttk root = Tk() root.title("METANIT.COM") root.geometry("250x200") ttk.Entry().pack(anchor=NW, padx=8, pady= 8) root.mainloop()
Получение введенного текста
Для получения текста из Entry, можно использовать его метод get() . Так, определим элемент Entry и по нажатию на кнопку выведем введенный текст на текстовую метку:
from tkinter import * from tkinter import ttk def show_message(): label["text"] = entry.get() # получаем введенный текст root = Tk() root.title("METANIT.COM") root.geometry("250x200") entry = ttk.Entry() entry.pack(anchor=NW, padx=6, pady=6) btn = ttk.Button(text="Click", command=show_message) btn.pack(anchor=NW, padx=6, pady=6) label = ttk.Label() label.pack(anchor=NW, padx=6, pady=6) root.mainloop()
Вставка и удаление текста
Рассмотрим вставку и удаление текста в Entry:
from tkinter import * from tkinter import ttk def clear(): entry.delete(0, END) # удаление введенного текста def display(): label["text"] = entry.get() # получение введенного текста root = Tk() root.title("METANIT.COM") root.geometry("250x150") label = ttk.Label() label.pack(anchor=NW, padx=6, pady=6) entry = ttk.Entry() entry.pack(anchor=NW, padx=6, pady=6) # вставка начальных данных entry.insert(0, "Hello World") display_button = ttk.Button(text="Display", command=display) display_button.pack(side=LEFT, anchor=N, padx=6, pady=6) clear_button = ttk.Button(text="Clear", command=clear) clear_button.pack(side=LEFT, anchor=N, padx=6, pady=6) root.mainloop()
При запуске программы в текстовое поле сразу же добавляется текст по умолчанию:
Кнопка Clear очищает оба поля, вызывая метод delete:
def clear(): entry.delete(0, END)
Вторая кнопка, используя метод get, получает введенный текст:
def display(): label["text"] = entry.get()
Валидация
С помощью параметра validate конструктора Entry можно задать, когда проводить валидацию введенного значения. Этот параметр может принимать следующие значения:
- none : отсутствие валидации, значение по умолчанию
- focus : валидация при получении фокуса
- focusin : валидация при изменении фокуса
- focusout : валидация при потере фокуса
- key : валидация при каждом вводе нового символа
- all : валидация при измении фокуса и вводе символов в поле
Параметр validatecommand позволяет установить команду валидации.
Рассмотрим небольшой пример. Допустим, пользовтаель должен ввести номер телефона в формете +xxxxxxxxxxx. То есть сначала должен идти знак +, а затем 11 цифр, например, +12345678901:
from tkinter import * from tkinter import ttk import re def is_valid(newval): return re.match("^\+\d$", newval) is not None root = Tk() root.title("METANIT.COM") root.geometry("250x200") check = (root.register(is_valid), "%P") phone_entry = ttk.Entry(validate="key", validatecommand=check) phone_entry.pack(padx=5, pady=5, anchor=NW) root.mainloop()
Итак, параметр validate=»key» указывает, что мы будем валидировать ввод при каждом нажати на клавиатуру. Параметр validatecommand=check говорит, что валидировать ввод будет команда «check». Эта команда представляет кортеж из двух элементов:
check = (root.register(is_valid), "%P")
Первый элемент — вызов метода root.register(is_valid) регистрирует функцию, которая собственно будет производить валидацию — это функция is_valid() . Второй элемент — подстановка «%P» представляет новое значение, которое передается в функцию валидации.
Собственно саму валидацию выполняет функция is_valid() . Она принимает один параметр — текущее значение Entry, которое надо валидировать. Она возвращает True, если значение прошло валидацию, и False, если не прошло. Сама логика валидации представляет проверку строки на регулярное выражение «^\+\d*$» . Если новое значение соответствует этому выражению, и в нем не больше 12 символов, то оно прошло валидацию.
В итоге мы сможем ввести в текстовое поле только символ + и затем только 11 цифр.
Теперь немного изменим код и добавим вывод ошибок валидации:
from tkinter import * from tkinter import ttk import re def is_valid(newval): result= re.match("^\+\d$", newval) is not None if not result and len(newval)Здесь для вывода ошибок валидации добавлен виджет Label. Если введенное значение не соответствует регулярному выражению (например, пользователь попытался ввести нецифровой символ), и длина ввода меньше и равно 12 символов (проверять ввод больше 12 символов нет смысла, так как номер телефона содержит только 12 символов), то в метке выводим сообщение об ошибке
Также мы можем передать значение параметра validate, чтобы в функции валидации в зависимости от того, нажал пользователь на клавишу или убрал фокус с поля, производить те или иные действия. В этом случае необходимо передать команде валидации дополнительный аргумент:
check = (root.register(is_valid), "%P", "%V")Здесь значение "%V" представляет событие, которое вызывает валидацию (focus/focusin/focusout/key). Тогда в функции валидации с помощью второго параметра мы сможем получить это значение:
def is_valid(newval, op): result= re.match("^\+\d$", newval) is not None if op=="key": # некоторые действия elif op=="focus": # некоторые действия return result