2017-02-01 01:20:17 +01:00
|
|
|
# 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/>.
|
|
|
|
|
2017-02-06 03:09:04 +01:00
|
|
|
import gi
|
|
|
|
gi.require_version('Gdk', '3.0')
|
|
|
|
from gi.repository import Gdk, GLib
|
2017-03-13 14:30:44 +01:00
|
|
|
import babel.dates, os, errno, re, subprocess, threading, logging, locale, json
|
2017-02-20 20:17:50 +01:00
|
|
|
import pyaudio, wave
|
2017-02-01 01:20:17 +01:00
|
|
|
|
2017-02-12 14:36:09 +01:00
|
|
|
bin_directory = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
home_directory = os.path.expanduser('~')
|
2017-03-13 14:30:44 +01:00
|
|
|
system_language_directory = os.path.join(bin_directory, "config/lang")
|
2017-04-07 22:20:23 +02:00
|
|
|
config_directory = os.path.join(home_directory, '.config/safeeyes')
|
2017-02-12 14:36:09 +01:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
|
2017-02-01 01:20:17 +01:00
|
|
|
def play_notification():
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Play the alert.wav
|
|
|
|
"""
|
2017-02-06 03:09:04 +01:00
|
|
|
logging.info("Playing audible alert")
|
2017-02-20 20:17:50 +01:00
|
|
|
CHUNK = 1024
|
|
|
|
|
2017-02-01 01:20:17 +01:00
|
|
|
try:
|
2017-02-20 20:17:50 +01:00
|
|
|
# Open the sound file
|
2017-04-07 22:20:23 +02:00
|
|
|
path = get_resource_path('alert.wav')
|
|
|
|
if path is None:
|
|
|
|
return
|
2017-02-20 20:17:50 +01:00
|
|
|
sound = wave.open(path, 'rb')
|
|
|
|
|
|
|
|
# Create a sound stream
|
|
|
|
wrapper = pyaudio.PyAudio()
|
|
|
|
stream = wrapper.open(format=wrapper.get_format_from_width(sound.getsampwidth()),
|
|
|
|
channels=sound.getnchannels(),
|
|
|
|
rate=sound.getframerate(),
|
|
|
|
output=True)
|
|
|
|
|
|
|
|
# Write file data into the sound stream
|
|
|
|
data = sound.readframes(CHUNK)
|
2017-03-23 01:07:51 +01:00
|
|
|
while data != b'':
|
2017-02-20 20:17:50 +01:00
|
|
|
stream.write(data)
|
|
|
|
data = sound.readframes(CHUNK)
|
|
|
|
|
|
|
|
# Close steam
|
|
|
|
stream.stop_stream()
|
|
|
|
stream.close()
|
2017-03-23 01:07:51 +01:00
|
|
|
sound.close()
|
2017-02-20 20:17:50 +01:00
|
|
|
wrapper.terminate()
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Unable to play audible alert')
|
|
|
|
logging.exception(e)
|
2017-02-06 03:09:04 +01:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
|
|
|
|
def get_resource_path(resource_name):
|
|
|
|
"""
|
2017-04-07 22:20:23 +02:00
|
|
|
Return the user-defined resource if a system resource is overridden by the user.
|
|
|
|
Otherwise, return the system resource. Return None if the specified resource does not exist.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-04-07 22:20:23 +02:00
|
|
|
if resource_name is None:
|
|
|
|
return None
|
|
|
|
resource_location = os.path.join(config_directory, 'resource', resource_name)
|
|
|
|
if not os.path.isfile(resource_location):
|
|
|
|
resource_location = os.path.join(bin_directory, 'resource', resource_name)
|
|
|
|
if not os.path.isfile(resource_location):
|
|
|
|
logging.error('Resource not found: ' + resource_name)
|
|
|
|
resource_location = None
|
|
|
|
|
|
|
|
return resource_location
|
|
|
|
|
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def system_idle_time():
|
|
|
|
"""
|
2017-02-06 03:09:04 +01:00
|
|
|
Get system idle time in minutes.
|
|
|
|
Return the idle time if xprintidle is available, otherwise return 0.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-02-06 03:09:04 +01:00
|
|
|
try:
|
|
|
|
return int(subprocess.check_output(['xprintidle']).decode('utf-8')) / 60000 # Convert to minutes
|
|
|
|
except:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
2017-02-16 00:29:14 +01:00
|
|
|
def start_thread(target_function, **args):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Execute the function in a separate thread.
|
|
|
|
"""
|
2017-02-09 03:39:27 +01:00
|
|
|
thread = threading.Thread(target=target_function, kwargs=args)
|
2017-02-06 03:09:04 +01:00
|
|
|
thread.start()
|
|
|
|
|
|
|
|
|
2017-02-10 02:06:04 +01:00
|
|
|
def execute_main_thread(target_function, args=None):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Execute the given function in main thread.
|
|
|
|
"""
|
2017-02-10 02:06:04 +01:00
|
|
|
if args:
|
|
|
|
GLib.idle_add(lambda: target_function(args))
|
|
|
|
else:
|
|
|
|
GLib.idle_add(lambda: target_function())
|
2017-02-06 03:09:04 +01:00
|
|
|
|
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def is_active_window_skipped(skip_break_window_classes, take_break_window_classes, unfullscreen_allowed=False):
|
|
|
|
"""
|
2017-02-09 03:39:27 +01:00
|
|
|
Check for full-screen applications.
|
2017-02-16 00:29:14 +01:00
|
|
|
This method must be executed by the main thread. If not, it will cause to random failure.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-02-06 03:09:04 +01:00
|
|
|
logging.info("Searching for full-screen application")
|
|
|
|
screen = Gdk.Screen.get_default()
|
2017-03-13 14:30:44 +01:00
|
|
|
|
2017-04-07 22:20:23 +02:00
|
|
|
active_window = screen.get_active_window()
|
|
|
|
if active_window:
|
|
|
|
active_xid = str(active_window.get_xid())
|
|
|
|
cmdlist = ['xprop', '-root', '-notype','-id',active_xid, 'WM_CLASS', '_NET_WM_STATE']
|
|
|
|
|
|
|
|
try:
|
|
|
|
stdout = subprocess.check_output(cmdlist).decode('utf-8')
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
logging.warning("Error in finding full-screen application")
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
if stdout:
|
|
|
|
is_fullscreen = 'FULLSCREEN' in stdout
|
|
|
|
# Extract the process name
|
|
|
|
process_names = re.findall('"(.+?)"', stdout)
|
|
|
|
if process_names:
|
|
|
|
process = process_names[1].lower()
|
|
|
|
if process in skip_break_window_classes:
|
|
|
|
return True
|
|
|
|
elif process in take_break_window_classes:
|
|
|
|
if is_fullscreen and unfullscreen_allowed:
|
|
|
|
try:
|
|
|
|
active_window.unfullscreen()
|
|
|
|
except:
|
|
|
|
logging.error('Error in unfullscreen the window ' + process)
|
|
|
|
pass
|
|
|
|
return False
|
|
|
|
|
|
|
|
return is_fullscreen
|
|
|
|
|
|
|
|
return False
|
2017-02-09 03:39:27 +01:00
|
|
|
|
|
|
|
|
2017-03-13 14:30:44 +01:00
|
|
|
def __system_locale():
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Return the system locale. If not available, return en_US.UTF-8.
|
|
|
|
"""
|
2017-02-13 15:55:08 +01:00
|
|
|
locale.setlocale(locale.LC_ALL, '')
|
|
|
|
system_locale = locale.getlocale(locale.LC_TIME)[0]
|
2017-02-09 03:39:27 +01:00
|
|
|
if not system_locale:
|
|
|
|
system_locale = 'en_US.UTF-8'
|
2017-03-13 14:30:44 +01:00
|
|
|
return system_locale
|
|
|
|
|
|
|
|
|
|
|
|
def format_time(time):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Format time based on the system time.
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
system_locale = __system_locale()
|
2017-02-09 03:39:27 +01:00
|
|
|
return babel.dates.format_time(time, format='short', locale=system_locale)
|
2017-02-10 02:45:49 +01:00
|
|
|
|
|
|
|
|
|
|
|
def mkdir(path):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Create directory if not exists.
|
|
|
|
"""
|
2017-02-10 02:45:49 +01:00
|
|
|
try:
|
|
|
|
os.makedirs(path)
|
|
|
|
except OSError as exc:
|
|
|
|
if exc.errno == errno.EEXIST and os.path.isdir(path):
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
logging.error('Error while creating ' + str(path))
|
2017-02-12 17:44:48 +01:00
|
|
|
raise
|
2017-03-13 14:30:44 +01:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def parse_language_code(lang_code):
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
Convert the user defined language code to a valid one.
|
|
|
|
This includes converting to lower case and finding system locale language,
|
|
|
|
if the given lang_code code is 'system'.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
# Convert to lower case
|
|
|
|
lang_code = str(lang_code).lower()
|
|
|
|
|
|
|
|
# If it is system, use the system language
|
|
|
|
if lang_code == 'system':
|
|
|
|
logging.info('Use system language for Safe Eyes')
|
|
|
|
system_locale = __system_locale()
|
|
|
|
lang_code = system_locale[0:2].lower()
|
|
|
|
|
|
|
|
# Check whether translation is available for this language.
|
|
|
|
# If not available, use English by default.
|
|
|
|
language_file_path = os.path.join(system_language_directory, lang_code + '.json')
|
|
|
|
if not os.path.exists(language_file_path):
|
|
|
|
logging.warn('The language {} does not exist. Use English instead'.format(lang_code))
|
|
|
|
lang_code = 'en'
|
|
|
|
|
|
|
|
return lang_code
|
|
|
|
|
|
|
|
|
|
|
|
def load_language(lang_code):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Load the desired language from the available list based on the preference.
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
# Convert the user defined language code to a valid one
|
|
|
|
lang_code = parse_language_code(lang_code)
|
|
|
|
|
|
|
|
# Construct the translation file path
|
|
|
|
language_file_path = os.path.join(system_language_directory, lang_code + '.json')
|
|
|
|
|
|
|
|
language = None
|
|
|
|
# Read the language file and construct the json object
|
|
|
|
with open(language_file_path) as language_file:
|
|
|
|
language = json.load(language_file)
|
|
|
|
|
|
|
|
return language
|
|
|
|
|
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def read_lang_files():
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
Read all the language translations and build a key-value mapping of language names
|
|
|
|
in English and ISO 639-1 (Filename without extension).
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
languages = {}
|
|
|
|
for lang_file_name in os.listdir(system_language_directory):
|
|
|
|
lang_file_path = os.path.join(system_language_directory, lang_file_name)
|
|
|
|
if os.path.isfile(lang_file_path):
|
|
|
|
with open(lang_file_path) as lang_file:
|
|
|
|
lang = json.load(lang_file)
|
|
|
|
languages[lang_file_name.lower().replace('.json', '')] = lang['meta_info']['language_name']
|
|
|
|
|
|
|
|
return languages
|
2017-04-08 15:30:44 +02:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def lock_screen_command():
|
2017-04-08 15:30:44 +02:00
|
|
|
"""
|
2017-04-09 02:29:51 +02:00
|
|
|
Function tries to detect the screensaver command based on the current envinroment
|
|
|
|
Possible results:
|
|
|
|
Gnome, Unity: ["gnome-screensaver-command", "--lock"]
|
|
|
|
Cinnamon: ["cinnamon-screensaver-command", "--lock"]
|
|
|
|
None if nothing detected
|
2017-04-08 15:30:44 +02:00
|
|
|
"""
|
2017-04-09 02:29:51 +02:00
|
|
|
# TODO: Add the command-line tools for other desktop environments (Atleast for KDE, XFCE, LXDE and MATE)
|
|
|
|
desktop_session = os.environ.get("DESKTOP_SESSION")
|
|
|
|
if desktop_session is not None:
|
|
|
|
desktop_session = desktop_session.lower()
|
|
|
|
# if desktop_session in ["gnome","unity", "cinnamon", "mate", "xfce4", "lxde", "fluxbox", "blackbox", "openbox", "icewm", "jwm", "afterstep", "trinity", "kde"]:
|
|
|
|
if desktop_session in ["gnome","unity"] or desktop_session.startswith("ubuntu"):
|
|
|
|
return ["gnome-screensaver-command", "--lock"]
|
|
|
|
elif desktop_session == "cinnamon":
|
|
|
|
return ["cinnamon-screensaver-command", "--lock"]
|
|
|
|
# elif desktop_session.startswith("lubuntu"):
|
|
|
|
# return "lxde"
|
|
|
|
# elif desktop_session.startswith("kubuntu"):
|
|
|
|
# return "kde"
|
|
|
|
# elif "xfce" in desktop_session or desktop_session.startswith("xubuntu"):
|
|
|
|
# return "xfce4"
|
|
|
|
# elif desktop_session.startswith("razor"):
|
|
|
|
# return "razor-qt"
|
|
|
|
# elif desktop_session.startswith("wmaker"):
|
|
|
|
# return "windowmaker"
|
|
|
|
# if os.environ.get('KDE_FULL_SESSION') == 'true':
|
|
|
|
# return "kde"
|
|
|
|
elif os.environ.get('GNOME_DESKTOP_SESSION_ID'):
|
|
|
|
if not "deprecated" in os.environ.get('GNOME_DESKTOP_SESSION_ID'):
|
|
|
|
return ["gnome-screensaver-command", "--lock"]
|
|
|
|
# elif self.is_running("xfce-mcs-manage"):
|
|
|
|
# return "xfce4"
|
|
|
|
# elif self.is_running("ksmserver"):
|
|
|
|
# return "kde"
|
2017-04-08 15:30:44 +02:00
|
|
|
return None
|
|
|
|
|
|
|
|
def is_desktop_lock_supported():
|
2017-04-09 02:29:51 +02:00
|
|
|
return lock_screen_command() is not None
|
2017-04-08 15:30:44 +02:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def lock_desktop():
|
|
|
|
"""
|
|
|
|
Lock the screen using the predefined commands
|
|
|
|
"""
|
|
|
|
command = lock_screen_command()
|
|
|
|
if command is not None:
|
|
|
|
try:
|
|
|
|
subprocess.Popen(command)
|
|
|
|
except Exception as e:
|
|
|
|
logging.error("Error in executing the commad" + str(command) + " to lock screen", e)
|