188 lines
4.7 KiB
Python
188 lines
4.7 KiB
Python
import tkinter as tk
|
|
import os
|
|
from datetime import datetime
|
|
|
|
STORAGE_PATH = os.path.expanduser("~/.config/focus-timer/data.txt")
|
|
|
|
STATE_IDLE = "IDLE"
|
|
STATE_FOCUS = "FOCUS"
|
|
STATE_WORK = "WORK"
|
|
STATE_OVERTIME = "OVERTIME"
|
|
STATE_BREAK = "BREAK"
|
|
|
|
current_state = STATE_IDLE
|
|
seconds_left = 0
|
|
distractions = 0
|
|
|
|
width = 120
|
|
height = 140
|
|
|
|
app = tk.Tk(className="focustimer")
|
|
app.title("Focus-Timer")
|
|
app.configure(background="#17153B")
|
|
app.geometry(f"{width}x{height}")
|
|
app.minsize(width, height)
|
|
app.maxsize(width, height)
|
|
app.update()
|
|
app.resizable(False, False)
|
|
|
|
|
|
def tick():
|
|
global seconds_left, current_state
|
|
|
|
match current_state:
|
|
case "IDLE":
|
|
return
|
|
case "OVERTIME":
|
|
seconds_left += 1
|
|
case _:
|
|
seconds_left -= 1
|
|
|
|
update_timer_label()
|
|
|
|
if current_state in ("FOCUS","WORK","BREAK") and seconds_left <= 0:
|
|
handle_transition()
|
|
|
|
app.after(1000, tick)
|
|
|
|
def update_timer_label():
|
|
global seconds_left
|
|
mins, secs = divmod(seconds_left, 60)
|
|
timer_label.config(text=f"{mins}:{secs}")
|
|
|
|
def handle_transition():
|
|
global current_state, seconds_left, distraction
|
|
|
|
match current_state:
|
|
case "FOCUS":
|
|
current_state = STATE_WORK
|
|
seconds_left = 4200
|
|
distractions = 0
|
|
|
|
case "WORK":
|
|
current_state = STATE_OVERTIME
|
|
seconds_left = 0
|
|
|
|
case "BREAK":
|
|
current_state = STATE_IDLE
|
|
seconds_left = 0
|
|
|
|
update_timer_label()
|
|
update_ui()
|
|
|
|
def start_button_f():
|
|
global current_state, seconds_left, distractions
|
|
distractions = 0
|
|
current_state = STATE_FOCUS
|
|
seconds_left = 1200
|
|
|
|
tick()
|
|
update_ui()
|
|
|
|
def distracted_button_f():
|
|
global current_state, seconds_left, distractions
|
|
|
|
if current_state != STATE_FOCUS:
|
|
return
|
|
|
|
distractions += 1
|
|
|
|
if distractions > 2:
|
|
seconds_left = 600
|
|
else:
|
|
seconds_left = 1200 - (300 * distractions)
|
|
|
|
update_timer_label()
|
|
|
|
def break_button_f():
|
|
global current_state, seconds_left
|
|
|
|
current_state = STATE_BREAK
|
|
seconds_left = 1500
|
|
update_timer_label()
|
|
|
|
create_post()
|
|
update_ui()
|
|
|
|
def complete_task_button_f():
|
|
create_post()
|
|
break_button_f()
|
|
|
|
def create_post():
|
|
os.makedirs(os.path.dirname(STORAGE_PATH), exist_ok=True)
|
|
history = {}
|
|
|
|
if os.path.exists(STORAGE_PATH):
|
|
with open(STORAGE_PATH, "r", encoding="utf-8") as f:
|
|
for line in f:
|
|
line = line.strip()
|
|
if " : " in line and line.endswith(" cycles."):
|
|
date_part, cycles_part = line.split(" : ", maxsplit=1)
|
|
count = int(cycles_part.replace(" cycles.", ""))
|
|
history[date_part] = count
|
|
|
|
date = datetime.now().strftime("%Y-%m-%d")
|
|
if date in history:
|
|
history[date] += 1
|
|
else:
|
|
history[date] = 1
|
|
|
|
with open(STORAGE_PATH, "w", encoding="utf-8") as f:
|
|
for dates, counts in history.items():
|
|
f.write(f"{dates} : {counts} cycles.\n")
|
|
|
|
def update_ui():
|
|
global current_state
|
|
|
|
start_button.pack_forget()
|
|
distracted_button.pack_forget()
|
|
complete_task_button.pack_forget()
|
|
break_button.pack_forget()
|
|
|
|
match current_state:
|
|
case "IDLE":
|
|
text_label.config(text="Wanna start?", fg="#C8ACD6")
|
|
timer_label.config(text="00:20:00", fg="#C8ACD6")
|
|
|
|
start_button.config(fg="#c8acd6", activebackground="#c8acd6")
|
|
start_button.pack(side="bottom",pady="15")
|
|
|
|
case "FOCUS":
|
|
text_label.config(text="Time to focus!", fg="red")
|
|
timer_label.config(fg="red")
|
|
distracted_button.pack(side="bottom", pady="15")
|
|
|
|
case "WORK":
|
|
text_label.config(text="Work.", fg="aqua")
|
|
timer_label.config(fg="aqua")
|
|
complete_task_button.pack(side="bottom", pady="15")
|
|
|
|
case "OVERTIME":
|
|
text_label.config(text="Maybe a break?", fg="orange")
|
|
timer_label.config(fg="orange")
|
|
break_button.pack(side="bottom", pady="15")
|
|
|
|
case "BREAK":
|
|
text_label.config(text="Relax..", fg="yellow")
|
|
timer_label.config(fg="yellow")
|
|
|
|
start_button.config(fg="yellow", activebackground="yellow")
|
|
start_button.pack(side="bottom", pady="15")
|
|
|
|
text_label = tk.Label(app, font=("Helvetica", 14, "bold"), bg="#17153b")
|
|
text_label.pack(side="top", pady="15")
|
|
|
|
timer_label = tk.Label(app, text="hh:mm:ss",fg="#C8ACD6",background="#17153B",font=("Helvetica", 25) , anchor="center")
|
|
timer_label.pack(expand=True,fill="y")
|
|
|
|
start_button = tk.Button(app, text="Start.", command=start_button_f, activeforeground="black", background="#17153b", relief="flat", bd="0")
|
|
|
|
distracted_button = tk.Button(app, text="I got distracted.", command=distracted_button_f, fg="red", bg="#17153b", relief="flat", bd="0", activebackground="red", activeforeground="black")
|
|
|
|
complete_task_button = tk.Button(app, text="Complete the task.", command=complete_task_button_f, fg="aqua", bg="#17153b", relief="flat", bd="0", activebackground="aqua", activeforeground="black")
|
|
|
|
break_button = tk.Button(app, text="Go break :3", command=break_button_f, fg="orange", bg="#17153b", relief="flat", bd="0", activebackground="orange", activeforeground="black")
|
|
|
|
update_ui()
|
|
app.mainloop()
|