2016-12-04 11:26:20 +00:00
|
|
|
# pylint: disable=R0201
|
|
|
|
|
|
|
|
"""Output classes"""
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import json
|
2016-12-09 18:29:16 +00:00
|
|
|
import uuid
|
2016-12-04 11:26:20 +00:00
|
|
|
|
2016-12-10 10:25:02 +00:00
|
|
|
import bumblebee.store
|
2019-10-03 15:36:26 +00:00
|
|
|
import bumblebee.util
|
2019-10-02 20:30:55 +00:00
|
|
|
|
2017-04-22 11:07:50 +00:00
|
|
|
def scrollable(func):
|
|
|
|
def wrapper(module, widget):
|
|
|
|
text = func(module, widget)
|
2019-11-10 19:30:46 +00:00
|
|
|
if not text:
|
|
|
|
return text
|
|
|
|
width = widget.get("theme.width", int(module.parameter("width", 30)))
|
2019-10-03 15:36:26 +00:00
|
|
|
if bumblebee.util.asbool(module.parameter("scrolling.makewide", "true")):
|
2019-10-02 20:30:55 +00:00
|
|
|
widget.set("theme.minwidth", "A"*width)
|
2019-11-10 19:30:46 +00:00
|
|
|
if width < 0:
|
|
|
|
return text
|
2017-04-22 11:07:50 +00:00
|
|
|
if len(text) <= width:
|
|
|
|
return text
|
|
|
|
# we need to shorten
|
2019-11-10 19:30:46 +00:00
|
|
|
|
2018-12-28 21:27:38 +00:00
|
|
|
try:
|
|
|
|
bounce = int(module.parameter("scrolling.bounce", 1))
|
|
|
|
except ValueError:
|
|
|
|
bounce = 1
|
|
|
|
try:
|
|
|
|
scroll_speed = int(module.parameter("scrolling.speed", 1))
|
|
|
|
except ValueError:
|
|
|
|
scroll_speed = 1
|
2017-04-22 11:07:50 +00:00
|
|
|
start = widget.get("scrolling.start", -1)
|
|
|
|
direction = widget.get("scrolling.direction", "right")
|
2018-12-28 21:27:38 +00:00
|
|
|
start += scroll_speed if direction == "right" else -(scroll_speed)
|
2019-11-10 19:30:46 +00:00
|
|
|
|
2018-12-28 21:27:38 +00:00
|
|
|
if width + start > len(text) + (scroll_speed -1):
|
|
|
|
if bounce:
|
|
|
|
widget.set("scrolling.direction", "left")
|
|
|
|
else:
|
|
|
|
start = 0
|
|
|
|
elif start <= 0:
|
|
|
|
if bounce:
|
|
|
|
widget.set("scrolling.direction", "right")
|
|
|
|
else:
|
|
|
|
start = len(text)
|
2017-04-22 11:07:50 +00:00
|
|
|
widget.set("scrolling.start", start)
|
|
|
|
text = text[start:width+start]
|
|
|
|
|
|
|
|
return text
|
|
|
|
return wrapper
|
|
|
|
|
2016-12-10 10:25:02 +00:00
|
|
|
class Widget(bumblebee.store.Store):
|
2016-12-04 16:45:42 +00:00
|
|
|
"""Represents a single visible block in the status bar"""
|
2016-12-10 06:47:24 +00:00
|
|
|
def __init__(self, full_text="", name=""):
|
2016-12-10 10:25:02 +00:00
|
|
|
super(Widget, self).__init__()
|
2016-12-04 16:45:42 +00:00
|
|
|
self._full_text = full_text
|
2016-12-09 10:49:59 +00:00
|
|
|
self.module = None
|
2016-12-10 07:09:13 +00:00
|
|
|
self._module = None
|
2018-09-15 12:40:52 +00:00
|
|
|
self._minimized = False
|
2016-12-10 06:47:24 +00:00
|
|
|
self.name = name
|
2016-12-09 18:29:16 +00:00
|
|
|
self.id = str(uuid.uuid4())
|
2016-12-08 11:44:52 +00:00
|
|
|
|
2017-08-12 15:03:04 +00:00
|
|
|
def get_module(self):
|
|
|
|
return self._module
|
|
|
|
|
2018-09-15 12:40:52 +00:00
|
|
|
def toggle_minimize(self):
|
|
|
|
self._minimized = not self._minimized
|
|
|
|
|
2016-12-09 10:49:59 +00:00
|
|
|
def link_module(self, module):
|
2016-12-09 06:11:23 +00:00
|
|
|
"""Set the module that spawned this widget
|
|
|
|
|
|
|
|
This is done outside the constructor to avoid having to
|
|
|
|
pass in the module name in every concrete module implementation"""
|
2016-12-09 10:49:59 +00:00
|
|
|
self.module = module.name
|
2016-12-10 07:09:13 +00:00
|
|
|
self._module = module
|
2016-12-04 16:45:42 +00:00
|
|
|
|
2017-05-10 18:01:29 +00:00
|
|
|
def cls(self):
|
|
|
|
if not self._module:
|
|
|
|
return None
|
|
|
|
return self._module.__module__.replace("bumblebee.modules.", "")
|
|
|
|
|
2016-12-09 12:32:22 +00:00
|
|
|
def state(self):
|
2016-12-09 15:33:29 +00:00
|
|
|
"""Return the widget's state"""
|
2016-12-10 07:09:13 +00:00
|
|
|
if self._module and hasattr(self._module, "state"):
|
2016-12-10 10:25:02 +00:00
|
|
|
states = self._module.state(self)
|
|
|
|
if not isinstance(states, list):
|
|
|
|
return [states]
|
|
|
|
return states
|
|
|
|
return []
|
2016-12-09 12:32:22 +00:00
|
|
|
|
2016-12-10 10:25:02 +00:00
|
|
|
def full_text(self, value=None):
|
|
|
|
"""Set or retrieve the full text to display in the widget"""
|
|
|
|
if value:
|
|
|
|
self._full_text = value
|
2016-12-08 07:44:54 +00:00
|
|
|
else:
|
2018-09-15 12:40:52 +00:00
|
|
|
if self._minimized:
|
|
|
|
return u"\u2026"
|
2016-12-10 10:25:02 +00:00
|
|
|
if callable(self._full_text):
|
2016-12-11 10:37:24 +00:00
|
|
|
return self._full_text(self)
|
2016-12-10 10:25:02 +00:00
|
|
|
else:
|
|
|
|
return self._full_text
|
2016-12-04 16:45:42 +00:00
|
|
|
|
2016-12-04 11:26:20 +00:00
|
|
|
class I3BarOutput(object):
|
|
|
|
"""Manage output according to the i3bar protocol"""
|
2018-09-22 12:40:32 +00:00
|
|
|
def __init__(self, theme, config=None):
|
2016-12-08 10:31:20 +00:00
|
|
|
self._theme = theme
|
2016-12-08 11:44:52 +00:00
|
|
|
self._widgets = []
|
2017-04-02 06:31:23 +00:00
|
|
|
self._started = False
|
2018-09-22 12:40:32 +00:00
|
|
|
self._config = config
|
2017-04-02 06:31:23 +00:00
|
|
|
|
|
|
|
def started(self):
|
|
|
|
return self._started
|
2016-12-04 11:26:20 +00:00
|
|
|
|
|
|
|
def start(self):
|
|
|
|
"""Print start preamble for i3bar protocol"""
|
2017-04-02 06:31:23 +00:00
|
|
|
self._started = True
|
2017-04-26 05:37:47 +00:00
|
|
|
sys.stdout.write(json.dumps({"version": 1, "click_events": True}) + "\n[\n")
|
2016-12-04 11:26:20 +00:00
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
"""Finish i3bar protocol"""
|
|
|
|
sys.stdout.write("]\n")
|
|
|
|
|
2019-01-26 18:40:08 +00:00
|
|
|
def draw(self, widget, module=None, engine=None):
|
|
|
|
"""Draw a single widget"""
|
2016-12-08 10:31:20 +00:00
|
|
|
full_text = widget.full_text()
|
2019-01-26 18:40:08 +00:00
|
|
|
if widget.get_module() and widget.get_module().hidden():
|
|
|
|
return
|
|
|
|
if widget.get_module() and widget.get_module().name in self._config.autohide():
|
|
|
|
if not any(state in widget.state() for state in ["warning", "critical"]):
|
|
|
|
return
|
2016-12-09 12:06:08 +00:00
|
|
|
padding = self._theme.padding(widget)
|
|
|
|
prefix = self._theme.prefix(widget, padding)
|
|
|
|
suffix = self._theme.suffix(widget, padding)
|
2019-01-26 18:40:08 +00:00
|
|
|
|
2016-12-08 10:52:47 +00:00
|
|
|
if prefix:
|
2016-12-08 11:44:52 +00:00
|
|
|
full_text = u"{}{}".format(prefix, full_text)
|
2016-12-08 10:52:47 +00:00
|
|
|
if suffix:
|
2016-12-08 11:44:52 +00:00
|
|
|
full_text = u"{}{}".format(full_text, suffix)
|
2019-01-26 18:40:08 +00:00
|
|
|
|
2016-12-09 11:55:16 +00:00
|
|
|
separator = self._theme.separator(widget)
|
2019-01-26 18:40:08 +00:00
|
|
|
if separator:
|
|
|
|
self._widgets.append({
|
|
|
|
u"full_text": separator,
|
|
|
|
"separator": False,
|
|
|
|
"color": self._theme.separator_fg(widget),
|
|
|
|
"background": self._theme.separator_bg(widget),
|
|
|
|
"separator_block_width": self._theme.separator_block_width(widget),
|
|
|
|
})
|
2017-04-22 11:07:50 +00:00
|
|
|
width = self._theme.minwidth(widget)
|
2018-11-25 16:50:05 +00:00
|
|
|
|
|
|
|
if width:
|
|
|
|
full_text = full_text.ljust(len(width) + len(prefix) + len(suffix))
|
2019-01-26 18:40:08 +00:00
|
|
|
|
|
|
|
self._widgets.append({
|
2018-01-11 19:31:48 +00:00
|
|
|
u"full_text": full_text,
|
2016-12-09 07:58:45 +00:00
|
|
|
"color": self._theme.fg(widget),
|
|
|
|
"background": self._theme.bg(widget),
|
2016-12-09 11:55:16 +00:00
|
|
|
"separator_block_width": self._theme.separator_block_width(widget),
|
|
|
|
"separator": True if separator is None else False,
|
2018-11-25 16:50:05 +00:00
|
|
|
"min_width": None,
|
2019-01-26 18:40:08 +00:00
|
|
|
# "min_width": width + "A"*(len(prefix) + len(suffix)) if width else None,
|
2017-04-22 06:24:52 +00:00
|
|
|
"align": self._theme.align(widget),
|
2016-12-09 18:29:16 +00:00
|
|
|
"instance": widget.id,
|
|
|
|
"name": module.id,
|
2019-12-24 12:54:41 +00:00
|
|
|
"markup": "none" if not self._config else self._config.markup(),
|
2019-01-26 18:40:08 +00:00
|
|
|
})
|
2016-12-08 08:04:47 +00:00
|
|
|
|
2016-12-08 11:44:52 +00:00
|
|
|
def begin(self):
|
|
|
|
"""Start one output iteration"""
|
|
|
|
self._widgets = []
|
2016-12-09 11:28:39 +00:00
|
|
|
self._theme.reset()
|
2016-12-04 11:53:18 +00:00
|
|
|
|
2016-12-04 15:14:43 +00:00
|
|
|
def flush(self):
|
|
|
|
"""Flushes output"""
|
2018-09-22 12:40:32 +00:00
|
|
|
widgets = self._widgets
|
|
|
|
if self._config and self._config.reverse():
|
|
|
|
widgets = list(reversed(widgets))
|
|
|
|
sys.stdout.write(json.dumps(widgets))
|
2016-12-08 11:44:52 +00:00
|
|
|
|
|
|
|
def end(self):
|
|
|
|
"""Finalizes output"""
|
2016-12-04 15:14:43 +00:00
|
|
|
sys.stdout.write(",\n")
|
|
|
|
sys.stdout.flush()
|
|
|
|
|
2016-12-04 11:26:20 +00:00
|
|
|
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4
|