2016-12-11 07:51:56 +00:00
|
|
|
# pylint: disable=C0111,R0903
|
|
|
|
|
|
|
|
"""Displays volume and mute status of PulseAudio devices.
|
|
|
|
|
|
|
|
Aliases: pasink, pasource
|
2017-01-05 03:55:14 +00:00
|
|
|
|
|
|
|
Requires the following executable:
|
|
|
|
* pactl
|
2016-12-11 07:51:56 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
import re
|
|
|
|
|
|
|
|
import bumblebee.util
|
|
|
|
import bumblebee.input
|
|
|
|
import bumblebee.output
|
|
|
|
import bumblebee.engine
|
|
|
|
|
|
|
|
ALIASES = [ "pasink", "pasource" ]
|
|
|
|
|
|
|
|
class Module(bumblebee.engine.Module):
|
|
|
|
def __init__(self, engine, config):
|
|
|
|
super(Module, self).__init__(engine, config,
|
|
|
|
bumblebee.output.Widget(full_text=self.volume)
|
|
|
|
)
|
|
|
|
|
|
|
|
self._left = 0
|
|
|
|
self._right = 0
|
|
|
|
self._mono = 0
|
|
|
|
self._mute = False
|
|
|
|
channel = "sink" if self.name == "pasink" else "source"
|
|
|
|
|
2016-12-17 07:06:58 +00:00
|
|
|
self._patterns = [
|
|
|
|
{ "expr": "Name:", "callback": (lambda line: False) },
|
|
|
|
{ "expr": "Mute:", "callback": (lambda line: self.mute(False if " no" in line.lower() else True)) },
|
|
|
|
{ "expr": "Volume:", "callback": self.getvolume },
|
|
|
|
]
|
|
|
|
|
2016-12-11 07:51:56 +00:00
|
|
|
engine.input.register_callback(self, button=bumblebee.input.RIGHT_MOUSE, cmd="pavucontrol")
|
2016-12-17 07:16:10 +00:00
|
|
|
|
|
|
|
events = [
|
|
|
|
{ "type": "mute", "action": "toggle", "button": bumblebee.input.LEFT_MOUSE },
|
|
|
|
{ "type": "volume", "action": "+2%", "button": bumblebee.input.WHEEL_UP },
|
|
|
|
{ "type": "volume", "action": "-2%", "button": bumblebee.input.WHEEL_DOWN },
|
|
|
|
]
|
|
|
|
|
|
|
|
for event in events:
|
|
|
|
engine.input.register_callback(self, button=event["button"],
|
|
|
|
cmd="pactl set-{}-{} @DEFAULT_{}@ {}".format(channel, event["type"],
|
|
|
|
channel.upper(), event["action"]))
|
2016-12-11 07:51:56 +00:00
|
|
|
|
2016-12-17 07:06:58 +00:00
|
|
|
def mute(self, value):
|
|
|
|
self._mute = value
|
|
|
|
|
|
|
|
def getvolume(self, line):
|
|
|
|
if "mono" in line:
|
|
|
|
m = re.search(r'mono:.*\s*\/\s*(\d+)%', line)
|
|
|
|
if m:
|
|
|
|
self._mono = m.group(1)
|
|
|
|
else:
|
|
|
|
m = re.search(r'left:.*\s*\/\s*(\d+)%.*right:.*\s*\/\s*(\d+)%', line)
|
|
|
|
if m:
|
|
|
|
self._left = m.group(1)
|
|
|
|
self._right = m.group(2)
|
|
|
|
return True
|
|
|
|
|
2016-12-11 07:51:56 +00:00
|
|
|
def _default_device(self):
|
|
|
|
output = bumblebee.util.execute("pactl info")
|
|
|
|
pattern = "Default Sink: " if self.name == "pasink" else "Default Source: "
|
|
|
|
for line in output.split("\n"):
|
|
|
|
if line.startswith(pattern):
|
|
|
|
return line.replace(pattern, "")
|
|
|
|
return "n/a"
|
|
|
|
|
2016-12-11 10:37:24 +00:00
|
|
|
def volume(self, widget):
|
2016-12-11 07:51:56 +00:00
|
|
|
if int(self._mono) > 0:
|
|
|
|
return "{}%".format(self._mono)
|
|
|
|
elif self._left == self._right:
|
|
|
|
return "{}%".format(self._left)
|
|
|
|
else:
|
|
|
|
return "{}%/{}%".format(self._left, self._right)
|
|
|
|
return "n/a"
|
|
|
|
|
|
|
|
def update(self, widgets):
|
|
|
|
channel = "sinks" if self.name == "pasink" else "sources"
|
|
|
|
device = self._default_device()
|
|
|
|
|
|
|
|
result = bumblebee.util.execute("pactl list {}".format(channel))
|
|
|
|
found = False
|
2016-12-17 07:06:58 +00:00
|
|
|
|
2016-12-11 07:51:56 +00:00
|
|
|
for line in result.split("\n"):
|
|
|
|
if device in line:
|
|
|
|
found = True
|
2016-12-20 06:49:21 +00:00
|
|
|
continue
|
2016-12-17 07:06:58 +00:00
|
|
|
if found == False:
|
|
|
|
continue
|
|
|
|
for pattern in self._patterns:
|
|
|
|
if not pattern["expr"] in line:
|
|
|
|
continue
|
2016-12-20 06:49:21 +00:00
|
|
|
if pattern["callback"](line) == False and found == True:
|
|
|
|
return
|
2016-12-11 07:51:56 +00:00
|
|
|
|
|
|
|
def state(self, widget):
|
|
|
|
if self._mute:
|
|
|
|
return [ "warning", "muted" ]
|
|
|
|
return [ "unmuted" ]
|
|
|
|
|
|
|
|
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4
|