;; pomodoro-mode.el --- Introduces a pomodoro timer to your mode-line -*- lexical-binding: t -*- ;; Copyright (C) 2025 jjanzen ;; ;; This program is free software: you can redistribute it and/or modify ;; it under the terms of the GNU General Public License as published by ;; the Free Software Foundation, either version 3 of the License, or ;; (at your option) any later version ;; ;; This program is distributed in the hope that it will be useful, ;; but WITHOUT ANY WARRANTY; without even the implied warranty of ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;; GNU General Public License for more details. ;; ;; You should have received a copy of the GNU General Public License ;; along with this program. If not, see . ;; Author: j. janzen ;; Version: 0.1 ;; Keywords: productivity, convenience ;;; Commentary: ;; This package provides a minor mode to add a simple pomodoro timer to your mode-line. ;; To activate it, run `M-x pomodoro-mode`. ;; To turn it off, run `M-x pomodoro-mode` once again. ;;;###autoload (defgroup pomodoro nil "Pomodoro timer" :group 'applications) ;;;###autoload (defcustom pomodoro-start-time 0 "When the current pomodoro break/work session has begun" :type '(integer) :group 'pomodoro) ;;;###autoload (defcustom pomodoro-state 'working "The current pomodoro state" :type '(working break) :group 'pomodoro) ;;;###autoload (defcustom pomodoro-work-time (* 25 60) "The length of a pomodoro work session in seconds" :type '(integer) :group 'pomodoro) ;;;###autoload (defcustom pomodoro-break-time (* 5 60) "The length of a pomodoro break session in seconds" :type '(integer) :group 'pomodoro) ;;;###autoload (defcustom pomodoro-work-symbol "🍅" "The symbol to display while in a work session" :type '(string) :group 'pomodoro) ;;;###autoload (defcustom pomodoro-break-symbol "🥫" "The symbol to display while in a break session" :type '(string) :group 'pomodoro) ;;;###autoload (defun pomodoro-toggle-state () "Restart the pomodoro timer and switch to the other state." (interactive) (cond ((equal pomodoro-state 'working) (setq pomodoro-state 'break)) ((equal pomodoro-state 'break) (setq pomodoro-state 'working)) (t (user-error "Invalid pomodoro state"))) (setq pomodoro-start-time (floor (float-time)))) (defun pomodoro-get-status-string () "Get the status string for the pomodoro timer. May update the state if necessary." (let* ((curr-time (floor (float-time))) (delta (- curr-time pomodoro-start-time)) (symbol (if (equal pomodoro-state 'working) pomodoro-work-symbol pomodoro-break-symbol)) (max-time (if (equal pomodoro-state 'working) pomodoro-work-time pomodoro-break-time)) (delta-trunc (if (>= delta max-time) max-time delta)) (time-value (- max-time delta-trunc)) (minutes (/ time-value 60)) (seconds (% time-value 60))) (if (<= time-value 0) (pomodoro-toggle-state)) (format " %s (%02d:%02d)" symbol minutes seconds))) (define-minor-mode pomodoro-local-mode "Run a pomodoro timer. Use pomodoro-mode instead of this." :init-value nil :group 'pomodoro :lighter (:eval (pomodoro-get-status-string))) ;;;###autoload (define-globalized-minor-mode pomodoro-mode pomodoro-local-mode pomodoro-local-mode "Initialize pomodoro-mode in all buffers" :group 'pomodoro (setq pomodoro-state 'working) (setq pomodoro-start-time (floor (float-time)))) (provide 'pomodoro-mode)