Проверка того, прошло ли время для задания ввода

#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. Это сработало как заклинание! Спасибо за предложение