2017-10-07 15:10:31 +02: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/>.
|
|
|
|
|
|
|
|
import datetime
|
|
|
|
import logging
|
|
|
|
import subprocess
|
|
|
|
import threading
|
|
|
|
|
|
|
|
from safeeyes import Utility
|
|
|
|
from safeeyes.model import State
|
|
|
|
|
|
|
|
"""
|
|
|
|
Safe Eyes smart pause plugin
|
|
|
|
"""
|
|
|
|
|
|
|
|
context = None
|
|
|
|
idle_condition = threading.Condition()
|
|
|
|
lock = threading.Lock()
|
|
|
|
active = False
|
|
|
|
idle_time = 0
|
|
|
|
enable_safe_eyes = None
|
|
|
|
disable_safe_eyes = None
|
|
|
|
smart_pause_activated = False
|
|
|
|
idle_start_time = None
|
|
|
|
next_break_time = None
|
2017-10-17 20:50:57 +02:00
|
|
|
next_break_duration = 0
|
2017-10-07 15:10:31 +02:00
|
|
|
break_interval = 0
|
2017-10-14 15:30:11 +02:00
|
|
|
waiting_time = 2
|
2017-10-17 20:50:57 +02:00
|
|
|
interpret_idle_as_break = False
|
2017-10-07 15:10:31 +02:00
|
|
|
|
2017-11-11 16:53:24 +01:00
|
|
|
|
2017-10-07 15:10:31 +02:00
|
|
|
def __system_idle_time():
|
|
|
|
"""
|
|
|
|
Get system idle time in minutes.
|
|
|
|
Return the idle time if xprintidle is available, otherwise return 0.
|
|
|
|
"""
|
|
|
|
try:
|
2018-10-31 04:41:25 +01:00
|
|
|
return int(subprocess.check_output(['xprintidle']).decode('utf-8')) / 1000 # Convert to seconds
|
2017-10-07 15:10:31 +02:00
|
|
|
except BaseException:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
def __is_active():
|
|
|
|
"""
|
|
|
|
Thread safe function to see if this plugin is active or not.
|
|
|
|
"""
|
|
|
|
is_active = False
|
|
|
|
with lock:
|
|
|
|
is_active = active
|
|
|
|
return is_active
|
|
|
|
|
|
|
|
|
|
|
|
def __set_active(is_active):
|
|
|
|
"""
|
|
|
|
Thread safe function to change the state of the plugin.
|
|
|
|
"""
|
|
|
|
global active
|
|
|
|
with lock:
|
|
|
|
active = is_active
|
|
|
|
|
|
|
|
|
|
|
|
def init(ctx, safeeyes_config, plugin_config):
|
|
|
|
"""
|
|
|
|
Initialize the plugin.
|
|
|
|
"""
|
|
|
|
global context
|
|
|
|
global enable_safe_eyes
|
|
|
|
global disable_safe_eyes
|
2018-10-31 04:41:25 +01:00
|
|
|
global postpone
|
2017-10-07 15:10:31 +02:00
|
|
|
global idle_time
|
|
|
|
global break_interval
|
2017-10-14 15:30:11 +02:00
|
|
|
global waiting_time
|
2017-10-17 20:50:57 +02:00
|
|
|
global interpret_idle_as_break
|
2018-10-31 04:41:25 +01:00
|
|
|
global postpone_if_active
|
2017-10-07 15:10:31 +02:00
|
|
|
logging.debug('Initialize Smart Pause plugin')
|
|
|
|
context = ctx
|
|
|
|
enable_safe_eyes = context['api']['enable_safeeyes']
|
|
|
|
disable_safe_eyes = context['api']['disable_safeeyes']
|
2018-10-31 04:41:25 +01:00
|
|
|
postpone = context['api']['postpone']
|
2017-10-07 15:10:31 +02:00
|
|
|
idle_time = plugin_config['idle_time']
|
2017-10-17 20:50:57 +02:00
|
|
|
interpret_idle_as_break = plugin_config['interpret_idle_as_break']
|
2018-10-31 04:41:25 +01:00
|
|
|
postpone_if_active = plugin_config['postpone_if_active']
|
2017-11-11 16:53:24 +01:00
|
|
|
break_interval = safeeyes_config.get('break_interval') * 60 # Convert to seconds
|
2018-10-31 04:41:25 +01:00
|
|
|
waiting_time = min(2, idle_time) # If idle time is 1 sec, wait only 1 sec
|
2017-10-07 15:10:31 +02:00
|
|
|
|
|
|
|
|
|
|
|
def __start_idle_monitor():
|
|
|
|
"""
|
|
|
|
Continuously check the system idle time and pause/resume Safe Eyes based on it.
|
|
|
|
"""
|
|
|
|
global smart_pause_activated
|
|
|
|
global idle_start_time
|
|
|
|
while __is_active():
|
2017-10-14 15:30:11 +02:00
|
|
|
# Wait for waiting_time seconds
|
2017-10-07 15:10:31 +02:00
|
|
|
idle_condition.acquire()
|
2017-10-14 15:30:11 +02:00
|
|
|
idle_condition.wait(waiting_time)
|
2017-10-07 15:10:31 +02:00
|
|
|
idle_condition.release()
|
|
|
|
|
|
|
|
if __is_active():
|
|
|
|
# Get the system idle time
|
|
|
|
system_idle_time = __system_idle_time()
|
|
|
|
if system_idle_time >= idle_time and context['state'] == State.WAITING:
|
|
|
|
smart_pause_activated = True
|
|
|
|
idle_start_time = datetime.datetime.now()
|
|
|
|
logging.info('Pause Safe Eyes due to system idle')
|
2018-02-09 16:09:08 +01:00
|
|
|
disable_safe_eyes(None)
|
2017-10-07 15:10:31 +02:00
|
|
|
elif system_idle_time < idle_time and context['state'] == State.STOPPED:
|
|
|
|
logging.info('Resume Safe Eyes due to user activity')
|
|
|
|
smart_pause_activated = False
|
|
|
|
idle_period = (datetime.datetime.now() - idle_start_time)
|
2017-10-17 20:50:57 +02:00
|
|
|
idle_seconds = idle_period.total_seconds()
|
|
|
|
if interpret_idle_as_break and idle_seconds >= next_break_duration:
|
|
|
|
# User is idle for break duration and wants to consider it as a break
|
|
|
|
enable_safe_eyes()
|
|
|
|
elif idle_seconds < break_interval:
|
|
|
|
# Credit back the idle time
|
2017-10-07 15:10:31 +02:00
|
|
|
next_break = next_break_time + idle_period
|
|
|
|
enable_safe_eyes(next_break.timestamp())
|
|
|
|
else:
|
2017-10-17 20:50:57 +02:00
|
|
|
# User is idle for more than the time between two breaks
|
2017-10-07 15:10:31 +02:00
|
|
|
enable_safe_eyes()
|
|
|
|
|
|
|
|
|
|
|
|
def on_start():
|
|
|
|
"""
|
|
|
|
Start a thread to continuously call xprintidle.
|
|
|
|
"""
|
|
|
|
global active
|
|
|
|
if not __is_active():
|
|
|
|
# If SmartPause is already started, do not start it again
|
|
|
|
logging.debug('Start Smart Pause plugin')
|
|
|
|
__set_active(True)
|
|
|
|
Utility.start_thread(__start_idle_monitor)
|
|
|
|
|
|
|
|
|
|
|
|
def on_stop():
|
|
|
|
"""
|
|
|
|
Stop the thread from continuously calling xprintidle.
|
|
|
|
"""
|
|
|
|
global active
|
|
|
|
global smart_pause_activated
|
|
|
|
if smart_pause_activated:
|
|
|
|
# Safe Eyes is stopped due to system idle
|
|
|
|
smart_pause_activated = False
|
|
|
|
return
|
|
|
|
logging.debug('Stop Smart Pause plugin')
|
|
|
|
__set_active(False)
|
|
|
|
idle_condition.acquire()
|
|
|
|
idle_condition.notify_all()
|
|
|
|
idle_condition.release()
|
|
|
|
|
2017-11-11 16:53:24 +01:00
|
|
|
|
2017-10-17 20:50:57 +02:00
|
|
|
def update_next_break(break_obj, dateTime):
|
2017-10-07 15:10:31 +02:00
|
|
|
"""
|
|
|
|
Update the next break time.
|
|
|
|
"""
|
|
|
|
global next_break_time
|
2017-10-17 20:50:57 +02:00
|
|
|
global next_break_duration
|
2017-10-07 15:10:31 +02:00
|
|
|
next_break_time = dateTime
|
2017-10-17 20:50:57 +02:00
|
|
|
next_break_duration = break_obj.time
|
2018-10-31 04:41:25 +01:00
|
|
|
|
|
|
|
|
|
|
|
def on_start_break(break_obj):
|
|
|
|
"""
|
|
|
|
Lifecycle method executes just before the break.
|
|
|
|
"""
|
|
|
|
if postpone_if_active:
|
|
|
|
# Postpone this break if the user is active
|
|
|
|
system_idle_time = __system_idle_time()
|
|
|
|
if system_idle_time < 2:
|
|
|
|
postpone(2) # Postpone for 2 seconds
|