bumblebee-status/modules/contrib/pomodoro.py

123 lines
5.1 KiB
Python
Raw Normal View History

2020-04-25 16:15:46 +02:00
# pylint: disable=C0111,R0903
"""Display and run a Pomodoro timer.
Left click to start timer, left click again to pause.
Right click will cancel the timer.
Parameters:
* pomodoro.work: The work duration of timer in minutes (defaults to 25)
* pomodoro.break: The break duration of timer in minutes (defaults to 5)
2020-04-25 16:16:20 +02:00
* pomodoro.format: Timer display format with '%m' and '%s' for minutes and seconds (defaults to '%m:%s')
Examples: '%m min %s sec', '%mm', '', 'timer'
2020-04-25 16:15:46 +02:00
* pomodoro.notify: Notification command to run when timer ends/starts (defaults to nothing)
2020-04-25 16:16:20 +02:00
Example: 'notify-send 'Time up!''. If you want to chain multiple commands,
2020-04-25 16:15:46 +02:00
please use an external wrapper script and invoke that. The module itself does
not support command chaining (see https://github.com/tobi-wan-kenobi/bumblebee-status/issues/532
for a detailled explanation)
"""
from __future__ import absolute_import
import datetime
from math import ceil
import bumblebee.input
import bumblebee.output
import bumblebee.engine
class Module(bumblebee.engine.Module):
def __init__(self, engine, config):
widgets = bumblebee.output.Widget(full_text=self.text)
super(Module, self).__init__(engine, config, widgets)
# Parameters
2020-04-25 16:16:20 +02:00
self._work_period = int(self.parameter('work', 25))
self._break_period = int(self.parameter('break', 5))
self._time_format = self.parameter('format', '%m:%s')
self._notify_cmd = self.parameter('notify', '')
2020-04-25 16:15:46 +02:00
# TODO: Handle time formats more gracefully. This is kludge.
self.display_seconds_p = False
self.display_minutes_p = False
2020-04-25 16:16:20 +02:00
if '%s' in self._time_format:
2020-04-25 16:15:46 +02:00
self.display_seconds_p = True
2020-04-25 16:16:20 +02:00
if '%m' in self._time_format:
2020-04-25 16:15:46 +02:00
self.display_minutes_p = True
self.remaining_time = datetime.timedelta(minutes=self._work_period)
self.time = None
2020-04-25 16:16:20 +02:00
self.pomodoro = { 'state':'OFF', 'type': ''}
self._text = self.remaining_time_str() + self.pomodoro['type']
2020-04-25 16:15:46 +02:00
engine.input.register_callback(self, button=bumblebee.input.LEFT_MOUSE,
cmd=self.timer_play_pause)
engine.input.register_callback(self, button=bumblebee.input.RIGHT_MOUSE,
cmd=self.timer_reset)
def remaining_time_str(self):
if self.display_seconds_p and self.display_minutes_p:
minutes, seconds = divmod(self.remaining_time.seconds, 60)
if not self.display_seconds_p:
minutes = ceil(self.remaining_time.seconds / 60)
seconds = 0
if not self.display_minutes_p:
minutes = 0
seconds = self.remaining_time.seconds
2020-04-25 16:16:20 +02:00
minutes = '{:2d}'.format(minutes)
seconds = '{:02d}'.format(seconds)
return self._time_format.replace('%m',minutes).replace('%s',seconds)+' '
2020-04-25 16:15:46 +02:00
def text(self, widget):
2020-04-25 16:16:20 +02:00
return '{}'.format(self._text)
2020-04-25 16:15:46 +02:00
def update(self, widget):
2020-04-25 16:16:20 +02:00
if self.pomodoro['state'] == 'ON':
2020-04-25 16:15:46 +02:00
timediff = (datetime.datetime.now() - self.time)
if timediff.seconds >= 0:
self.remaining_time -= timediff
self.time = datetime.datetime.now()
if self.remaining_time.total_seconds() <= 0:
self.notify()
2020-04-25 16:16:20 +02:00
if self.pomodoro['type'] == 'Work':
self.pomodoro['type'] = 'Break'
2020-04-25 16:15:46 +02:00
self.remaining_time = datetime.timedelta(minutes=self._break_period)
2020-04-25 16:16:20 +02:00
elif self.pomodoro['type'] == 'Break':
self.pomodoro['type'] = 'Work'
2020-04-25 16:15:46 +02:00
self.remaining_time = datetime.timedelta(minutes=self._work_period)
2020-04-25 16:16:20 +02:00
self._text = self.remaining_time_str() + self.pomodoro['type']
2020-04-25 16:15:46 +02:00
def notify(self):
if self._notify_cmd:
bumblebee.util.execute(self._notify_cmd)
def timer_play_pause(self, widget):
2020-04-25 16:16:20 +02:00
if self.pomodoro['state'] == 'OFF':
self.pomodoro = {'state': 'ON', 'type': 'Work'}
2020-04-25 16:15:46 +02:00
self.remaining_time = datetime.timedelta(minutes=self._work_period)
self.time = datetime.datetime.now()
2020-04-25 16:16:20 +02:00
elif self.pomodoro['state'] == 'ON':
self.pomodoro['state'] = 'PAUSED'
2020-04-25 16:15:46 +02:00
self.remaining_time -= (datetime.datetime.now() - self.time)
self.time = datetime.datetime.now()
2020-04-25 16:16:20 +02:00
elif self.pomodoro['state'] == 'PAUSED':
self.pomodoro['state'] = 'ON'
2020-04-25 16:15:46 +02:00
self.time = datetime.datetime.now()
def timer_reset(self, widget):
2020-04-25 16:16:20 +02:00
if self.pomodoro['state'] == 'ON' or self.pomodoro['state'] == 'PAUSED':
self.pomodoro = {'state':'OFF', 'type': '' }
2020-04-25 16:15:46 +02:00
self.remaining_time = datetime.timedelta(minutes=self._work_period)
def state(self, widget):
state = [];
2020-04-25 16:16:20 +02:00
state.append(self.pomodoro['state'].lower())
if self.pomodoro['state'] == 'ON' or self.pomodoro['state'] == 'OFF':
state.append(self.pomodoro['type'].lower())
2020-04-25 16:15:46 +02:00
return state