diff options
-rw-r--r-- | bin/.gitignore | 1 | ||||
-rw-r--r-- | bin/Makefile | 3 | ||||
-rwxr-xr-x | bin/i3status.py | 63 | ||||
-rw-r--r-- | bin/kbdlayout.c | 57 | ||||
-rw-r--r-- | i3.symlink/config | 8 | ||||
-rw-r--r-- | system/i3status.conf.symlink | 12 |
6 files changed, 139 insertions, 5 deletions
diff --git a/bin/.gitignore b/bin/.gitignore new file mode 100644 index 0000000..95d9967 --- /dev/null +++ b/bin/.gitignore @@ -0,0 +1 @@ +kbdlayout diff --git a/bin/Makefile b/bin/Makefile new file mode 100644 index 0000000..221dde7 --- /dev/null +++ b/bin/Makefile @@ -0,0 +1,3 @@ +kbdlayout: kbdlayout.c + $(CC) $(CCFLAGS) -lX11 $< -o $@ + diff --git a/bin/i3status.py b/bin/i3status.py new file mode 100755 index 0000000..421317e --- /dev/null +++ b/bin/i3status.py @@ -0,0 +1,63 @@ +#!/usr/bin/env python +import json +from select import select +from subprocess import Popen, PIPE +import sys + +def remove_empty_outputs(obj): + return [o for o in obj if o['full_text'].strip() != ''] + +first = True +def print_i3stat(j): + global first + j = remove_empty_outputs(j) + sys.stdout.write(('' if first else ',') + json.dumps(j) + '\n') + sys.stdout.flush() + first = False + +def parse_i3stat(s): + if s[0] == ',': + s = s[1:] + j = json.loads(s) + return j + +def parse_kbdlayout(s): + colors = { 'us(intl)' : '#66ccff', + 'us(dvorak-intl)' : '#dc68fc', + 'ru(phonetic)' : '#ff3300', + 'il(biblicalSIL)' : '#66ff66' + } + return [{'full_text': s, 'color': colors[s]}] + +def merge_status_items(*args): + return [item for sublist in args for item in sublist] + +if __name__ == '__main__': + timeout = 0.1 + + # open subprocesses + i3stat = Popen(['i3status'], stdout=PIPE, bufsize=1, close_fds=True) + kbdlayout = Popen(['unbuffer', 'kbdlayout'], + stdout=PIPE, bufsize=1, close_fds=True) + + # skip i3status header + sys.stdout.write(i3stat.stdout.readline()) + sys.stdout.write(i3stat.stdout.readline()) + + stat, kbd = [], [] + try: + while True: + fs = select([p.stdout for p in [i3stat, kbdlayout]], [], [])[0] + for f in fs: + line = f.readline()[:-1] + if f == kbdlayout.stdout: + kbd = parse_kbdlayout(line) + else: + stat = parse_i3stat(line) + + print_i3stat(merge_status_items(kbd, stat)) + + except: + i3stat.kill() + kbdlayout.kill() + diff --git a/bin/kbdlayout.c b/bin/kbdlayout.c new file mode 100644 index 0000000..7cdd97e --- /dev/null +++ b/bin/kbdlayout.c @@ -0,0 +1,57 @@ +#include <X11/XKBlib.h> +#include <stdlib.h> +#include <stdio.h> +#include <string.h> + +char* getActiveLayout(char* names, unsigned char active) { + int i; + for (i = 0; names[i]; i++) { + if (names[i] == '+') { + if (active <= 0) { + int j = ++i; + for (; names[j] && names[j] != '+' && names[j] != ':'; j++); + char* ret = malloc(j - i + 1); + strncpy(ret, names + i, j - i); + ret[j - i] = '\0'; + return ret; + } else { + active--; + } + } + } + return NULL; +} + +int main(void) { + + int evCode, errRet, rsnRet; + int maj = XkbMajorVersion; + int min = XkbMinorVersion; + + Display* disp = XkbOpenDisplay("", &evCode, &errRet, &maj, &min, &rsnRet); + + // State + XkbStatePtr state = calloc(1, sizeof(XkbStateRec)); + XkbGetState(disp, 0x100, state); + + // Names + XkbDescPtr desc = XkbAllocKeyboard(); + XkbGetNames(disp, XkbSymbolsNameMask, desc); + + Atom symNameAtom = desc->names->symbols; + char* layouts = XGetAtomName(disp, symNameAtom); + + printf("%s\n", getActiveLayout(layouts, state->group)); + + unsigned int mask = XkbStateNotifyMask; + XkbSelectEvents(disp, XkbUseCoreKbd, mask, mask); + + XkbEvent event; + while (1) { + XNextEvent(disp, &event.core); + if (event.state.changed & 0x90) + printf("%s\n", getActiveLayout(layouts, event.state.group)); + } + +} + diff --git a/i3.symlink/config b/i3.symlink/config index 462a82f..5ee2128 100644 --- a/i3.symlink/config +++ b/i3.symlink/config @@ -4,9 +4,6 @@ floating_modifier $mod font pango:DejaVu Sans Mono 8 focus_follows_mouse no -# Bar -exec xmobar -d - # Autolock exec xautolock -time 10 -locker 'i3lock -d' bindsym $mod+Control+Shift+l exec xautolock -locknow @@ -113,3 +110,8 @@ mode "resize" { bindsym $mod+r mode "default" } +bar { + font pango:DejaVu Sans Mono, 9 + status_command i3status.py +} + diff --git a/system/i3status.conf.symlink b/system/i3status.conf.symlink index 211b6e8..37406ea 100644 --- a/system/i3status.conf.symlink +++ b/system/i3status.conf.symlink @@ -1,6 +1,7 @@ general { colors = true interval = 5 + output_format = "i3bar" } order += "cmd nowplaying" @@ -8,6 +9,7 @@ order += "volume master" order += "wireless wlan0" order += "battery 0" order += "cpu_usage" +order += "load" order += "tztime local" volume master { @@ -28,17 +30,23 @@ wireless wlan0 { } battery 0 { - format = "%status %percentage %remaining" + format = "%status %percentage %remaining %consumption" + hide_seconds = true status_chr = "⚇" status_bat = "⚡" status_full = "☻" - low_threshold = 20 + low_threshold = 30 + threshold_type = "percentage" } cpu_usage { format = "⚙️ %usage" } +load { + max_threshold = "0.7" +} + tztime local { format = "%Y-%m-%d %H:%M:%S" } |