#python #tkinter #python-datetime
Вопрос:
Я пытаюсь составить список дел, в который вы можете ввести задачи в формате «вывести собаку в 13:45» через TKinter. Затем я обрабатываю время, указанное в задаче, как объект datetime, который я сравниваю с текущим временем, чтобы узнать, должна ли еще выполняться задача. Однако у меня возникли некоторые проблемы со структурой этого.
Я хочу, чтобы функция sound_alarm работала непрерывно, чтобы проверить, должны ли еще быть выполнены задачи. Но я не могу сделать это с помощью цикла true:, так как это мешает root.mainloop() из TKinter.
Есть какие-нибудь идеи, как сделать так, чтобы при вводе задач функция sound_alarm запускалась с этого момента, чтобы проверить, выполняется ли еще какая-либо из задач?
Вот код, как он у меня есть сейчас:
import tkinter as tk
from tkinter import messagebox
import pickle
from datetime import datetime
root = tk.Tk()
root.title("To Do list")
def add_task():
task = entry_task.get()
if task != "":
listbox_task.insert(tk.END, task)
print(task)
entry_task.delete(0, tk.END)
else:
tk.messagebox.showwarning(title="Warning", message="Enter a task first")
def del_task():
try:
task_index = listbox_task.curselection()[0]
listbox_task.delete(task_index)
except:
tk.messagebox.showwarning(title="Warning", message="Select a task first")
def load_tasks():
try:
tasks = pickle.load(open("tasks.dat", "rb"))
listbox_task.delete(0, tk.END)
for task in tasks:
listbox_task.insert(tk.END, task)
except:
tk.messagebox.showwarning(title="Warning", message="Cannot find task file")
def save_tasks():
tasks = listbox_task.get(0, listbox_task.size())
pickle.dump(tasks, open("tasks.dat", "wb"))
def get_time_from_task(task):
time = task.split("at ", 1)[1]
datetime_time = datetime.strptime(time, '%H:%M').time()
return datetime_time
def sound_alarm():
tasks = listbox_task.get(0, listbox_task.size())
now = datetime.now().time()
for task in tasks:
datetime_time = get_time_from_task(task)
print(datetime_time, now)
if now > datetime_time:
print(task, "is due!")
frame_tasks = tk.Frame(root)
frame_tasks.pack()
listbox_task = tk.Listbox(frame_tasks, height=20, width=50)
listbox_task.pack(side=tk.LEFT)
scrollbar_tasks = tk.Scrollbar(frame_tasks)
scrollbar_tasks.pack(side=tk.RIGHT, fill=tk.Y)
listbox_task.config(yscrollcommand=scrollbar_tasks.set)
scrollbar_tasks.config(command=listbox_task.yview)
entry_task = tk.Entry(root, width=50)
entry_task.pack()
btn_add = tk.Button(root, text="Add task", width=48, command=add_task)
btn_add.pack()
btn_del = tk.Button(root, text="Delete task", width=48, command=del_task)
btn_del.pack()
btn_load = tk.Button(root, text="Load tasks", width=48, command=load_tasks)
btn_load.pack()
btn_save = tk.Button(root, text="Save tasks", width=48, command=save_tasks)
btn_save.pack()
root.mainloop()
Комментарии:
1. Вы можете использовать
.after()
для выполненияsound_alarm()
каждую минуту.2. вы должны использовать
.after(milliseconds, function)
для повторения кода безwhile True
3. Так что я думаю, ты имеешь в виду что-то вроде этого, верно?
root.after(60000, sound_alarm())
. Где должно быть размещено это заявление? До или после root.mainloop()? Потому что, когда я помещаю его в любом месте перед root.mainloop (), программа, естественно, ждет, пока функция after() не будет завершена, и только тогда открывает стартовое окно. Тем не менее, я хочу, чтобы начальное окно немедленно открывалось при запуске для ввода задач.4. Обычно добавляйте
root.after(60000, sound_alarm)
в концеsound_alarm()
и выполняйтеsound_alarm()
раньшеroot.mainloop()
.5. Вы должны поместить это
root.after(60000, sound_alarm)
в конце своей функцииsound_alarm()
внутри него
Ответ №1:
Вы можете использовать .after()
для выполнения sound_alarm()
каждую минуту.
Ниже изменено sound_alarm()
:
def sound_alarm():
tasks = listbox_task.get(0, "end")
now = datetime.now()
current_time = now.time()
for task in tasks:
task_time = get_time_from_task(task)
print(task_time, current_time)
if current_time > task_time:
print(task, "is due!")
# try to schedule next check at 0 second of next minute
delay = 60 - now.second
root.after(delay*1000, sound_alarm)
...
sound_alarm() # start the checking task
root.mainloop()
Комментарии:
1. Это сработало как заклинание! Спасибо за предложение