1
0
mirror of https://github.com/slgobinath/SafeEyes.git synced 2025-01-01 00:47:31 +01:00
SafeEyes/safeeyes/Utility.py

297 lines
9.0 KiB
Python
Raw Normal View History

# 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-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
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
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))
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)