bumblebee-status/modules/core/battery.py

139 lines
5 KiB
Python
Raw Normal View History

2020-03-06 19:34:30 +00:00
# pylint: disable=C0111,R0903
'''Displays battery status, remaining percentage and charging information.
Parameters:
* battery.device : Comma-separated list of battery devices to read information from (defaults to auto for auto-detection)
* battery.warning : Warning threshold in % of remaining charge (defaults to 20)
* battery.critical : Critical threshold in % of remaining charge (defaults to 10)
* battery.showdevice : If set to 'true', add the device name to the widget (defaults to False)
* battery.decorate : If set to 'false', hides additional icons (charging, etc.) (defaults to True)
* battery.showpowerconsumption: If set to 'true', show current power consumption (defaults to False)
'''
import os
import glob
import logging
log = logging.getLogger(__name__)
try:
import power
except ImportError:
log.warning('unable to import module "power": Time estimates will not be available')
2020-03-06 19:34:30 +00:00
2020-03-06 19:37:46 +00:00
import core.module
import core.widget
import core.input
2020-03-06 19:34:30 +00:00
2020-03-06 19:37:46 +00:00
import util.format
2020-03-06 19:34:30 +00:00
2020-03-06 19:37:46 +00:00
class Module(core.module.Module):
def __init__(self, config=None):
2020-03-06 19:34:30 +00:00
widgets = []
2020-03-06 19:37:46 +00:00
super().__init__(config, widgets)
self._batteries = util.format.aslist(self.parameter('device', 'auto'))
2020-03-06 19:34:30 +00:00
if self._batteries[0] == 'auto':
self._batteries = glob.glob('/sys/class/power_supply/BAT*')
else:
self._batteries = ['/sys/class/power_supply/{}'.format(b) for b in self._batteries]
if len(self._batteries) == 0:
2020-03-06 19:37:46 +00:00
raise Exceptions('no batteries configured/found')
core.input.register(self, button=core.input.LEFT_MOUSE,
2020-03-06 19:34:30 +00:00
cmd='gnome-power-statistics')
for path in self._batteries:
log.debug('adding new widget for {}'.format(path))
widget = core.widget.Widget(full_text=self.capacity, name=path, module=self)
widgets.append(widget)
2020-03-06 19:34:30 +00:00
self.capacity(widget)
if util.format.asbool(self.parameter('decorate', True)) == False:
2020-03-06 19:34:30 +00:00
widget.set('theme.exclude', 'suffix')
def remaining(self):
estimate = 0.0
try:
estimate = power.PowerManagement().get_time_remaining_estimate()
# do not show remaining if on AC
if estimate == power.common.TIME_REMAINING_UNLIMITED:
return None
if estimate == power.common.TIME_REMAINING_UNKNOWN:
return ''
except Exception:
return ''
2020-03-07 12:33:28 +00:00
return util.format.duration(estimate*60, compact=True, unit=True) # estimate is in minutes
2020-03-06 19:34:30 +00:00
def capacity(self, widget):
widget.set('capacity', -1)
widget.set('ac', False)
if not os.path.exists(widget.name()):
2020-03-06 19:34:30 +00:00
widget.set('capacity', 100)
widget.set('ac', True)
return 'ac'
capacity = 100
try:
with open('{}/capacity'.format(widget.name())) as f:
2020-03-06 19:34:30 +00:00
capacity = int(f.read())
except IOError:
return 'n/a'
capacity = capacity if capacity < 100 else 100
widget.set('capacity', capacity)
# Read power conumption
if util.format.asbool(self.parameter('showpowerconsumption', False)):
r=open(widget.name() + '/power_now', 'r')
2020-03-06 19:34:30 +00:00
output = '{}% ({})'.format(capacity,str(int(r.read())/1000000) + 'W')
else:
output = '{}%'.format(capacity)
widget.set('theme.minwidth', '100%')
if util.format.asbool(self.parameter('showremaining', True))\
2020-03-06 19:34:30 +00:00
and self.getCharge(widget) == 'Discharging':
output = '{} {}'.format(output, self.remaining())
if util.format.asbool(self.parameter('showdevice', False)):
2020-03-06 19:34:30 +00:00
output = '{} ({})'.format(output, os.path.basename(widget.name))
return output
def state(self, widget):
state = []
capacity = widget.get('capacity')
if capacity < 0:
log.debug('battery state: {}'.format(state))
2020-03-06 19:34:30 +00:00
return ['critical', 'unknown']
if capacity < int(self.parameter('critical', 10)):
state.append('critical')
elif capacity < int(self.parameter('warning', 20)):
state.append('warning')
if widget.get('ac'):
state.append('AC')
else:
charge = self.getCharge(widget)
if charge == 'Discharging':
state.append('discharging-{}'.format(min([10, 25, 50, 80, 100], key=lambda i: abs(i-capacity))))
elif charge == 'Unknown':
state.append('unknown-{}'.format(min([10, 25, 50, 80, 100], key=lambda i: abs(i-capacity))))
else:
if capacity > 95:
state.append('charged')
else:
state.append('charging')
return state
def getCharge(self, widget):
charge = ''
try:
with open('{}/status'.format(widget.name())) as f:
2020-03-06 19:34:30 +00:00
charge = f.read().strip()
except IOError:
pass
return charge
2020-03-06 19:34:30 +00:00
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4