2017-10-07 15:10:31 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# Safe Eyes is a utility to remind you to take break frequently
|
|
|
|
# to protect your eyes from eye strain.
|
|
|
|
|
|
|
|
# Copyright (C) 2017 Gobinath
|
|
|
|
|
|
|
|
# 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 <http://www.gnu.org/licenses/>.
|
|
|
|
"""
|
|
|
|
SafeEyes connects all the individual components and provide the complete application.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import atexit
|
|
|
|
import logging
|
2019-02-23 15:57:53 +01:00
|
|
|
import os
|
2017-10-07 15:10:31 +02:00
|
|
|
from threading import Timer
|
|
|
|
|
|
|
|
import dbus
|
|
|
|
import gi
|
|
|
|
from dbus.mainloop.glib import DBusGMainLoop
|
2020-03-18 13:33:11 +01:00
|
|
|
from safeeyes import utility
|
|
|
|
from safeeyes.ui.about_dialog import AboutDialog
|
|
|
|
from safeeyes.ui.break_screen import BreakScreen
|
2017-10-07 15:10:31 +02:00
|
|
|
from safeeyes.model import State
|
|
|
|
from safeeyes.rpc import RPCServer
|
2020-03-18 13:33:11 +01:00
|
|
|
from safeeyes.plugin_manager import PluginManager
|
|
|
|
from safeeyes.core import SafeEyesCore
|
|
|
|
from safeeyes.ui.settings_dialog import SettingsDialog
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
gi.require_version('Gtk', '3.0')
|
|
|
|
from gi.repository import Gtk
|
|
|
|
|
2021-04-20 03:32:50 +02:00
|
|
|
SAFE_EYES_VERSION = "2.1.3"
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
|
2020-03-18 13:33:11 +01:00
|
|
|
class SafeEyes:
|
2017-10-07 15:10:31 +02:00
|
|
|
"""
|
|
|
|
This class represents a runnable Safe Eyes instance.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, system_locale, config):
|
|
|
|
self.active = False
|
|
|
|
self.break_screen = None
|
|
|
|
self.safe_eyes_core = None
|
|
|
|
self.config = config
|
|
|
|
self.context = {}
|
|
|
|
self.plugins_manager = None
|
|
|
|
self.settings_dialog_active = False
|
|
|
|
self.rpc_server = None
|
2018-01-29 02:16:02 +01:00
|
|
|
self._status = ''
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
# Initialize the Safe Eyes Context
|
|
|
|
self.context['version'] = SAFE_EYES_VERSION
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context['desktop'] = utility.desktop_environment()
|
|
|
|
self.context['is_wayland'] = utility.is_wayland()
|
2017-10-07 15:10:31 +02:00
|
|
|
self.context['locale'] = system_locale
|
|
|
|
self.context['api'] = {}
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context['api']['show_settings'] = lambda: utility.execute_main_thread(
|
2019-02-21 15:02:58 +01:00
|
|
|
self.show_settings)
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context['api']['show_about'] = lambda: utility.execute_main_thread(
|
2019-02-21 15:02:58 +01:00
|
|
|
self.show_about)
|
2020-10-25 13:44:29 +01:00
|
|
|
self.context['api']['enable_safeeyes'] = lambda next_break_time=-1, reset_breaks=False: \
|
|
|
|
utility.execute_main_thread(self.enable_safeeyes, next_break_time, reset_breaks)
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context['api']['disable_safeeyes'] = lambda status: utility.execute_main_thread(
|
2019-02-21 15:02:58 +01:00
|
|
|
self.disable_safeeyes, status)
|
2018-01-29 02:16:02 +01:00
|
|
|
self.context['api']['status'] = self.status
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context['api']['quit'] = lambda: utility.execute_main_thread(
|
2019-02-21 15:02:58 +01:00
|
|
|
self.quit)
|
2017-10-07 15:10:31 +02:00
|
|
|
if self.config.get('persist_state'):
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context['session'] = utility.open_session()
|
2017-10-07 15:10:31 +02:00
|
|
|
else:
|
|
|
|
self.context['session'] = {'plugin': {}}
|
|
|
|
|
2019-02-21 15:02:58 +01:00
|
|
|
self.break_screen = BreakScreen(
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context, self.on_skipped, self.on_postponed, utility.STYLE_SHEET_PATH)
|
2017-10-07 15:10:31 +02:00
|
|
|
self.break_screen.initialize(self.config)
|
2020-04-15 23:52:14 +02:00
|
|
|
self.plugins_manager = PluginManager()
|
2017-10-07 15:10:31 +02:00
|
|
|
self.safe_eyes_core = SafeEyesCore(self.context)
|
|
|
|
self.safe_eyes_core.on_pre_break += self.plugins_manager.pre_break
|
2018-10-31 04:41:25 +01:00
|
|
|
self.safe_eyes_core.on_start_break += self.on_start_break
|
|
|
|
self.safe_eyes_core.start_break += self.start_break
|
2017-10-07 15:10:31 +02:00
|
|
|
self.safe_eyes_core.on_count_down += self.countdown
|
|
|
|
self.safe_eyes_core.on_stop_break += self.stop_break
|
|
|
|
self.safe_eyes_core.on_update_next_break += self.update_next_break
|
|
|
|
self.safe_eyes_core.initialize(self.config)
|
2020-03-18 13:33:11 +01:00
|
|
|
self.context['api']['take_break'] = lambda: utility.execute_main_thread(
|
2019-02-21 15:02:58 +01:00
|
|
|
self.safe_eyes_core.take_break)
|
2017-10-07 15:10:31 +02:00
|
|
|
self.context['api']['has_breaks'] = self.safe_eyes_core.has_breaks
|
2018-10-31 04:41:25 +01:00
|
|
|
self.context['api']['postpone'] = self.safe_eyes_core.postpone
|
2017-10-07 15:10:31 +02:00
|
|
|
self.plugins_manager.init(self.context, self.config)
|
|
|
|
atexit.register(self.persist_session)
|
|
|
|
|
|
|
|
def start(self):
|
|
|
|
"""
|
|
|
|
Start Safe Eyes
|
|
|
|
"""
|
2019-11-25 02:39:28 +01:00
|
|
|
if self.config.get('use_rpc_server', True):
|
|
|
|
self.__start_rpc_server()
|
|
|
|
|
2017-10-07 15:10:31 +02:00
|
|
|
if self.safe_eyes_core.has_breaks():
|
|
|
|
self.active = True
|
|
|
|
self.context['state'] = State.START
|
|
|
|
self.plugins_manager.start() # Call the start method of all plugins
|
|
|
|
self.safe_eyes_core.start()
|
|
|
|
self.handle_system_suspend()
|
|
|
|
|
|
|
|
def show_settings(self):
|
|
|
|
"""
|
|
|
|
Listen to tray icon Settings action and send the signal to Settings dialog.
|
|
|
|
"""
|
|
|
|
if not self.settings_dialog_active:
|
|
|
|
logging.info("Show Settings dialog")
|
|
|
|
self.settings_dialog_active = True
|
2019-02-21 15:02:58 +01:00
|
|
|
settings_dialog = SettingsDialog(
|
|
|
|
self.config.clone(), self.save_settings)
|
2017-10-07 15:10:31 +02:00
|
|
|
settings_dialog.show()
|
|
|
|
|
|
|
|
def show_about(self):
|
|
|
|
"""
|
|
|
|
Listen to tray icon About action and send the signal to About dialog.
|
|
|
|
"""
|
|
|
|
logging.info("Show About dialog")
|
|
|
|
about_dialog = AboutDialog(SAFE_EYES_VERSION)
|
|
|
|
about_dialog.show()
|
|
|
|
|
|
|
|
def quit(self):
|
|
|
|
"""
|
|
|
|
Listen to the tray menu quit action and stop the core, notification and the app itself.
|
|
|
|
"""
|
|
|
|
logging.info("Quit Safe Eyes")
|
|
|
|
self.context['state'] = State.QUIT
|
|
|
|
self.plugins_manager.stop()
|
|
|
|
self.safe_eyes_core.stop()
|
|
|
|
self.plugins_manager.exit()
|
2019-11-25 02:39:28 +01:00
|
|
|
self.__stop_rpc_server()
|
2019-03-02 21:56:27 +01:00
|
|
|
self.persist_session()
|
2017-10-07 15:10:31 +02:00
|
|
|
Gtk.main_quit()
|
2019-02-23 15:57:53 +01:00
|
|
|
# Exit all threads
|
|
|
|
os._exit(0)
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
def handle_suspend_callback(self, sleeping):
|
|
|
|
"""
|
|
|
|
If the system goes to sleep, Safe Eyes stop the core if it is already active.
|
|
|
|
If it was active, Safe Eyes will become active after wake up.
|
|
|
|
"""
|
|
|
|
if sleeping:
|
|
|
|
# Sleeping / suspending
|
|
|
|
if self.active:
|
|
|
|
logging.info("Stop Safe Eyes due to system suspend")
|
|
|
|
self.plugins_manager.stop()
|
|
|
|
self.safe_eyes_core.stop()
|
|
|
|
else:
|
|
|
|
# Resume from sleep
|
|
|
|
if self.active and self.safe_eyes_core.has_breaks():
|
|
|
|
logging.info("Resume Safe Eyes after system wakeup")
|
|
|
|
self.plugins_manager.start()
|
|
|
|
self.safe_eyes_core.start()
|
|
|
|
|
|
|
|
def handle_system_suspend(self):
|
|
|
|
"""
|
|
|
|
Setup system suspend listener.
|
|
|
|
"""
|
|
|
|
DBusGMainLoop(set_as_default=True)
|
|
|
|
bus = dbus.SystemBus()
|
2019-02-21 15:02:58 +01:00
|
|
|
bus.add_signal_receiver(self.handle_suspend_callback, 'PrepareForSleep',
|
|
|
|
'org.freedesktop.login1.Manager', 'org.freedesktop.login1')
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
def on_skipped(self):
|
|
|
|
"""
|
|
|
|
Listen to break screen Skip action and send the signal to core.
|
|
|
|
"""
|
|
|
|
logging.info("User skipped the break")
|
|
|
|
self.safe_eyes_core.skip()
|
|
|
|
self.plugins_manager.stop_break()
|
|
|
|
|
|
|
|
def on_postponed(self):
|
|
|
|
"""
|
|
|
|
Listen to break screen Postpone action and send the signal to core.
|
|
|
|
"""
|
|
|
|
logging.info("User postponed the break")
|
|
|
|
self.safe_eyes_core.postpone()
|
|
|
|
self.plugins_manager.stop_break()
|
|
|
|
|
|
|
|
def save_settings(self, config):
|
|
|
|
"""
|
|
|
|
Listen to Settings dialog Save action and write to the config file.
|
|
|
|
"""
|
|
|
|
self.settings_dialog_active = False
|
|
|
|
|
2019-02-21 15:02:58 +01:00
|
|
|
if self.config == config:
|
|
|
|
# Config is not modified
|
|
|
|
return
|
|
|
|
|
|
|
|
logging.info("Saving settings to safeeyes.json")
|
2017-10-07 15:10:31 +02:00
|
|
|
# Stop the Safe Eyes core
|
|
|
|
if self.active:
|
|
|
|
self.plugins_manager.stop()
|
|
|
|
self.safe_eyes_core.stop()
|
|
|
|
|
|
|
|
# Write the configuration to file
|
|
|
|
config.save()
|
|
|
|
self.persist_session()
|
|
|
|
|
|
|
|
logging.info("Initialize SafeEyesCore with modified settings")
|
|
|
|
|
2019-11-25 02:39:28 +01:00
|
|
|
if self.rpc_server is None and config.get('use_rpc_server'):
|
|
|
|
# RPC server wasn't running but now enabled
|
|
|
|
self.__start_rpc_server()
|
|
|
|
elif self.rpc_server is not None and not config.get('use_rpc_server'):
|
|
|
|
# RPC server was running but now disabled
|
|
|
|
self.__stop_rpc_server()
|
|
|
|
|
2021-01-27 04:55:59 +01:00
|
|
|
# Restart the core and initialize the components
|
2017-10-07 15:10:31 +02:00
|
|
|
self.config = config
|
|
|
|
self.safe_eyes_core.initialize(config)
|
|
|
|
self.break_screen.initialize(config)
|
|
|
|
self.plugins_manager.init(self.context, self.config)
|
|
|
|
if self.active and self.safe_eyes_core.has_breaks():
|
|
|
|
# 1 sec delay is required to give enough time for core to be stopped
|
|
|
|
Timer(1.0, self.safe_eyes_core.start).start()
|
|
|
|
self.plugins_manager.start()
|
|
|
|
|
2020-10-25 13:44:29 +01:00
|
|
|
def enable_safeeyes(self, scheduled_next_break_time=-1, reset_breaks=False):
|
2017-10-07 15:10:31 +02:00
|
|
|
"""
|
|
|
|
Listen to tray icon enable action and send the signal to core.
|
|
|
|
"""
|
|
|
|
if not self.active and self.safe_eyes_core.has_breaks():
|
|
|
|
self.active = True
|
2020-10-25 13:44:29 +01:00
|
|
|
self.safe_eyes_core.start(scheduled_next_break_time, reset_breaks)
|
2017-10-07 15:10:31 +02:00
|
|
|
self.plugins_manager.start()
|
|
|
|
|
2018-01-29 02:16:02 +01:00
|
|
|
def disable_safeeyes(self, status=None):
|
2017-10-07 15:10:31 +02:00
|
|
|
"""
|
|
|
|
Listen to tray icon disable action and send the signal to core.
|
|
|
|
"""
|
|
|
|
if self.active:
|
|
|
|
self.active = False
|
|
|
|
self.plugins_manager.stop()
|
|
|
|
self.safe_eyes_core.stop()
|
2018-01-29 02:16:02 +01:00
|
|
|
if status is None:
|
|
|
|
status = _('Disabled until restart')
|
|
|
|
self._status = status
|
2017-10-07 15:10:31 +02:00
|
|
|
|
2018-10-31 04:41:25 +01:00
|
|
|
def on_start_break(self, break_obj):
|
2017-10-07 15:10:31 +02:00
|
|
|
"""
|
2018-10-31 04:41:25 +01:00
|
|
|
Pass the break information to plugins.
|
2017-10-07 15:10:31 +02:00
|
|
|
"""
|
|
|
|
if not self.plugins_manager.start_break(break_obj):
|
|
|
|
return False
|
2018-10-31 04:41:25 +01:00
|
|
|
return True
|
|
|
|
|
|
|
|
def start_break(self, break_obj):
|
|
|
|
"""
|
|
|
|
Pass the break information to break screen.
|
|
|
|
"""
|
2017-10-07 15:10:31 +02:00
|
|
|
# Get the HTML widgets content from plugins
|
|
|
|
widget = self.plugins_manager.get_break_screen_widgets(break_obj)
|
2019-01-12 22:12:50 +01:00
|
|
|
actions = self.plugins_manager.get_break_screen_tray_actions(break_obj)
|
|
|
|
self.break_screen.show_message(break_obj, widget, actions)
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
def countdown(self, countdown, seconds):
|
|
|
|
"""
|
|
|
|
Pass the countdown to plugins and break screen.
|
|
|
|
"""
|
|
|
|
self.break_screen.show_count_down(countdown, seconds)
|
|
|
|
self.plugins_manager.countdown(countdown, seconds)
|
|
|
|
return True
|
|
|
|
|
2017-10-17 20:50:57 +02:00
|
|
|
def update_next_break(self, break_obj, break_time):
|
2017-10-07 15:10:31 +02:00
|
|
|
"""
|
|
|
|
Update the next break to plugins and save the session.
|
|
|
|
"""
|
2017-10-17 20:50:57 +02:00
|
|
|
self.plugins_manager.update_next_break(break_obj, break_time)
|
2019-02-21 15:02:58 +01:00
|
|
|
self._status = _('Next break at %s') % (
|
2020-03-18 13:33:11 +01:00
|
|
|
utility.format_time(break_time))
|
2017-10-07 15:10:31 +02:00
|
|
|
if self.config.get('persist_state'):
|
2020-03-18 13:33:11 +01:00
|
|
|
utility.write_json(utility.SESSION_FILE_PATH,
|
2019-02-21 15:02:58 +01:00
|
|
|
self.context['session'])
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
def stop_break(self):
|
|
|
|
"""
|
|
|
|
Stop the current break.
|
|
|
|
"""
|
|
|
|
self.break_screen.close()
|
|
|
|
self.plugins_manager.stop_break()
|
|
|
|
return True
|
|
|
|
|
|
|
|
def take_break(self):
|
|
|
|
"""
|
|
|
|
Take a break now.
|
|
|
|
"""
|
|
|
|
self.safe_eyes_core.take_break()
|
|
|
|
|
2018-01-29 02:16:02 +01:00
|
|
|
def status(self):
|
|
|
|
"""
|
|
|
|
Return the status of Safe Eyes.
|
|
|
|
"""
|
|
|
|
return self._status
|
2018-10-31 04:41:25 +01:00
|
|
|
|
2017-10-07 15:10:31 +02:00
|
|
|
def persist_session(self):
|
|
|
|
"""
|
|
|
|
Save the session object to the session file.
|
|
|
|
"""
|
|
|
|
if self.config.get('persist_state'):
|
2020-03-18 13:33:11 +01:00
|
|
|
utility.write_json(utility.SESSION_FILE_PATH,
|
2019-02-21 15:02:58 +01:00
|
|
|
self.context['session'])
|
2017-10-07 15:10:31 +02:00
|
|
|
else:
|
2020-03-18 13:33:11 +01:00
|
|
|
utility.delete(utility.SESSION_FILE_PATH)
|
2019-11-25 02:39:28 +01:00
|
|
|
|
|
|
|
def __start_rpc_server(self):
|
|
|
|
if self.rpc_server is None:
|
|
|
|
self.rpc_server = RPCServer(self.config.get('rpc_port'), self.context)
|
|
|
|
self.rpc_server.start()
|
|
|
|
|
|
|
|
def __stop_rpc_server(self):
|
|
|
|
if self.rpc_server is not None:
|
|
|
|
self.rpc_server.stop()
|
|
|
|
self.rpc_server = None
|