[all] Refactor module <-> output communication

Modules now return "bumblebee.output.Widget" objects, so that they can
actually define a list of items to be drawn in the bar.
This commit is contained in:
Tobias Witek 2016-11-05 08:59:43 +01:00
parent 7f91b8298f
commit bab7821607
6 changed files with 223 additions and 131 deletions

View file

@ -43,18 +43,15 @@ class Engine:
self.register_event(e) self.register_event(e)
def run(self): def run(self):
print self._output.start() self._output.start()
while True: while True:
# improve this self._theme.begin()
self._theme.reset()
for m in self._modules: for m in self._modules:
self._output.add(m, self._theme) self._output.draw(m.widgets(), self._theme)
self._theme.next() self._output.flush()
print self._output.get() self._output.wait()
sys.stdout.flush()
self._output.wait(self._args.interval)
print self._output.stop() self._output.stop()
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4 # vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4

View file

@ -31,7 +31,11 @@ class Module(bumblebee.module.Module):
param_name = "{}.format".format(module) param_name = "{}.format".format(module)
self._fmt = config.parameter(param_name, default) self._fmt = config.parameter(param_name, default)
def data(self): def widgets(self):
return datetime.datetime.now().strftime(self._fmt) return [
bumblebee.output.Widget(
datetime.datetime.now().strftime(self._fmt)
)
]
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4 # vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4

View file

@ -1,11 +1,39 @@
import inspect
import threading import threading
def output(args): def output(args):
import bumblebee.outputs.i3 import bumblebee.outputs.i3
return bumblebee.outputs.i3.Output(args) return bumblebee.outputs.i3.Output(args)
class Widget(object):
def __init__(self, text, warning=False, critical=False, state=None):
self._name = inspect.getmodule(inspect.stack()[1][0]).__name__
self._text = text
self._warning = warning
self._critical = critical
self._state = state
def state(self):
return self._state
def warning(self):
return self._warning
def critical(self):
return self._critical
def module(self):
return self._name.split(".")[-1]
def name(self):
return self._name
def text(self):
return self._text
class Output(object): class Output(object):
def __init__(self, args): def __init__(self, config):
self._config = config
self._callbacks = {} self._callbacks = {}
self._wait = threading.Condition() self._wait = threading.Condition()
self._wait.acquire() self._wait.acquire()
@ -35,16 +63,16 @@ class Output(object):
), None) ), None)
return cb return cb
def wait(self, interval): def wait(self):
self._wait.wait(interval) self._wait.wait(self._config.parameter("interval", 1))
def start(self): def start(self):
pass pass
def add(self, obj, theme): def draw(self, widgets, theme):
pass pass
def get(self): def flush(self):
pass pass
def stop(self): def stop(self):

View file

@ -32,51 +32,47 @@ class Output(bumblebee.output.Output):
super(Output, self).__init__(args) super(Output, self).__init__(args)
self._data = [] self._data = []
self.add_callback("i3-msg workspace prev_on_output", 4) # TODO
self.add_callback("i3-msg workspace next_on_output", 5) # self.add_callback("i3-msg workspace prev_on_output", 4)
# self.add_callback("i3-msg workspace next_on_output", 5)
self._thread = threading.Thread(target=read_input, args=(self,)) # self._thread = threading.Thread(target=read_input, args=(self,))
self._thread.start() # self._thread.start()
def start(self): def start(self):
return json.dumps({ "version": 1, "click_events": True }) + "[" print json.dumps({ "version": 1, "click_events": True }) + "["
def add(self, obj, theme): def draw(self, widgets, theme):
for widget in widgets:
while True: if theme.separator(widget):
d = obj.data()
data = {
u"full_text": "{}{}{}".format(theme.prefix(obj), d, theme.suffix(obj)),
"color": theme.color(obj),
"background": theme.background(obj),
"name": obj.__module__.replace("bumblebee.modules.",""),
"instance": obj.instance() if hasattr(obj, "instance") else None,
}
if theme.urgent(obj) and obj.critical():
data["urgent"] = True
if theme.default_separators(obj) == False:
data["separator"] = False
data["separator_block_width"] = 0
if theme.separator(obj):
self._data.append({ self._data.append({
u"full_text": theme.separator(obj), u"full_text": theme.separator(obj),
"color": theme.background(obj), "color": theme.separator_color(obj),
"background": theme.previous_background(), "background": theme.separator_background(),
"separator": False, "separator": False,
"separator_block_width": 0, "separator_block_width": 0,
}) })
self._data.append(data) self._data.append({
if obj.next() == False: u"full_text": "{}{}{}".format(
break theme.prefix(widget),
theme.next() widget.text(),
theme.suffix(widget)
),
"color": theme.color(widget),
"background": theme.background(widget),
"name": widget.name(),
"instance": getattr(widget, "instance", None),
"separator": theme.default_separators(widget),
"separator_block_width": theme.separator_block_width(widget),
})
theme.next_widget()
def get(self): def flush(self):
data = json.dumps(self._data) data = json.dumps(self._data)
self._data = [] self._data = []
return data + "," print(data + ",")
sys.stdout.flush()
def stop(self): def stop(self):
return "]" return "]"

View file

@ -10,96 +10,162 @@ def themes():
return [ os.path.basename(f).replace(".json", "") for f in glob.iglob("{}/*.json".format(d)) ] return [ os.path.basename(f).replace(".json", "") for f in glob.iglob("{}/*.json".format(d)) ]
class Theme: class Theme:
_cycle_index = 0 # TODO: cycle handling
_cycle = None
def __init__(self, config): def __init__(self, config):
self._data = None
path = os.path.dirname(os.path.realpath(__file__))
with open("{}/{}.json".format(getpath(), config.theme())) as f: with open("{}/{}.json".format(getpath(), config.theme())) as f:
self._data = json.load(f) self._data = json.load(f)
self._defaults = self._data.get("defaults", {}) self._defaults = self._data.get("defaults", {})
self._cycle = self._defaults.get("cycle", []) self._cycle = {}
self._modules = {}
self.reset() def begin(self):
pass
def _gettheme(self, obj, key): def next_widget(self):
module = obj.__module__.split(".")[-1] pass
def prefix(self, widget):
pass
def suffix(self, widget):
pass
def color(self, widget):
result = self._get(widget, "fg")
if widget.warning():
result = self._get(widget, "fg-warning")
if widget.critical():
result = self._get(widget, "fg-critical")
return result
def background(self, widget):
pass
def separator(self, widget):
pass
def separator_color(self, widget):
pass
def separator_background(self, widget):
pass
def default_separators(self, widget):
pass
def separator_block_width(self, widget):
pass
def _get(self, widget, name):
module = widget.module()
state = widget.state()
module_theme = self._data.get(module, {}) module_theme = self._data.get(module, {})
value = getattr(obj, key)() if hasattr(obj, key) else None
value = self._defaults.get(key, value)
if len(self._cycle) > 0:
value = self._defaults["cycle"][self._cycle_index].get(key, value)
value = module_theme.get(key, value)
if hasattr(obj, "state"):
state = getattr(obj, "state")()
state_theme = module_theme.get("states", {}).get(state, {}) state_theme = module_theme.get("states", {}).get(state, {})
value = state_theme.get(key, value) value = None
value = self._defaults.get(name, value)
value = self._cycle.get(name, value)
value = module_theme.get(name, value)
value = state_theme.get(name, value)
if type(value) is list: if type(value) is list:
# cycle through the values # if the value is a list, cycle through it
if not obj in self._modules: # TODO
self._modules[obj] = { "idx": 0 } pass
else:
self._modules[obj]["idx"] += 1
if self._modules[obj]["idx"] >= len(value):
self._modules[obj]["idx"] = 0
value = value[self._modules[obj]["idx"]]
return value return value
def reset(self):
self._cycle_index = 0
self._previous_background = None
self._background = None
def urgent(self, obj): #class Theme:
self._gettheme(obj, "urgent") # _cycle_index = 0
# _cycle = None
def next(self): # def __init__(self, config):
self._cycle_index += 1 # self._data = None
self._previous_background = self._background # path = os.path.dirname(os.path.realpath(__file__))
if self._cycle_index >= len(self._cycle): # with open("{}/{}.json".format(getpath(), config.theme())) as f:
self._cycle_index = 0 # self._data = json.load(f)
# self._defaults = self._data.get("defaults", {})
def color(self, obj): # self._cycle = self._defaults.get("cycle", [])
fg = None # self._modules = {}
if obj.warning(): #
fg = self._gettheme(obj, "fg-warning") # self.reset()
if obj.critical(): #
fg = self._gettheme(obj, "fg-critical") # def _gettheme(self, obj, key):
if fg == None: # module = obj.__module__.split(".")[-1]
fg = self._gettheme(obj, "fg") # module_theme = self._data.get(module, {})
return fg #
# value = getattr(obj, key)() if hasattr(obj, key) else None
def background(self, obj): # value = self._defaults.get(key, value)
self._background = None # if len(self._cycle) > 0:
if obj.warning(): # value = self._defaults["cycle"][self._cycle_index].get(key, value)
self._background = self._gettheme(obj, "bg-warning") # value = module_theme.get(key, value)
if obj.critical(): #
self._background = self._gettheme(obj, "bg-critical") # if hasattr(obj, "state"):
# state = getattr(obj, "state")()
if self._background == None: # state_theme = module_theme.get("states", {}).get(state, {})
self._background = self._gettheme(obj, "bg") #
# value = state_theme.get(key, value)
return self._background #
# if type(value) is list:
def previous_background(self): # # cycle through the values
return self._previous_background # if not obj in self._modules:
# self._modules[obj] = { "idx": 0 }
def separator(self, obj): # else:
return self._gettheme(obj, "separator") # self._modules[obj]["idx"] += 1
# if self._modules[obj]["idx"] >= len(value):
def default_separators(self, obj): # self._modules[obj]["idx"] = 0
return self._gettheme(obj, "default_separators") # value = value[self._modules[obj]["idx"]]
#
def prefix(self, obj): # return value
return self._gettheme(obj, "prefix") #
# def reset(self):
def suffix(self, obj): # self._cycle_index = 0
return self._gettheme(obj, "suffix") # self._previous_background = None
# self._background = None
#
# def urgent(self, obj):
# self._gettheme(obj, "urgent")
#
# def next(self):
# self._cycle_index += 1
# self._previous_background = self._background
# if self._cycle_index >= len(self._cycle):
# self._cycle_index = 0
#
# def color(self, obj):
# fg = None
# if obj.warning():
# fg = self._gettheme(obj, "fg-warning")
# if obj.critical():
# fg = self._gettheme(obj, "fg-critical")
# if fg == None:
# fg = self._gettheme(obj, "fg")
# return fg
#
# def background(self, obj):
# self._background = None
# if obj.warning():
# self._background = self._gettheme(obj, "bg-warning")
# if obj.critical():
# self._background = self._gettheme(obj, "bg-critical")
#
# if self._background == None:
# self._background = self._gettheme(obj, "bg")
#
# return self._background
#
# def previous_background(self):
# return self._previous_background
#
# def separator(self, obj):
# return self._gettheme(obj, "separator")
#
# def default_separators(self, obj):
# return self._gettheme(obj, "default_separators")
#
# def prefix(self, obj):
# return self._gettheme(obj, "prefix")
#
# def suffix(self, obj):
# return self._gettheme(obj, "suffix")
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4 # vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4

View file

@ -2,7 +2,8 @@
"defaults": { "defaults": {
"prefix": " ", "prefix": " ",
"suffix" : " ", "suffix" : " ",
"urgent": true "urgent": true,
"fg": "#aabbcc"
}, },
"date": { "date": {
"prefix": " " "prefix": " "