diff options
author | ben <ben@nagy.contact> | 2025-05-03 14:59:49 -0700 |
---|---|---|
committer | ben <ben@nagy.contact> | 2025-05-03 14:59:49 -0700 |
commit | ed80e5df8c718fc2321904db1b81604d1ed33444 (patch) | |
tree | 04bc58346a40cd696f5840bcd3c724684bd5f591 /config/suckless/slstatus/components/keyboard_indicators.c | |
parent | ea687c769521b3139c8547a2d4a839842fa7b0fe (diff) |
suckless
Diffstat (limited to 'config/suckless/slstatus/components/keyboard_indicators.c')
-rw-r--r-- | config/suckless/slstatus/components/keyboard_indicators.c | 48 |
1 files changed, 48 insertions, 0 deletions
diff --git a/config/suckless/slstatus/components/keyboard_indicators.c b/config/suckless/slstatus/components/keyboard_indicators.c new file mode 100644 index 0000000..b35eba1 --- /dev/null +++ b/config/suckless/slstatus/components/keyboard_indicators.c @@ -0,0 +1,48 @@ +/* See LICENSE file for copyright and license details. */ +#include <ctype.h> +#include <stdio.h> +#include <string.h> +#include <X11/Xlib.h> + +#include "../util.h" + +/* + * fmt consists of uppercase or lowercase 'c' for caps lock and/or 'n' for num + * lock, each optionally followed by '?', in the order of indicators desired. + * If followed by '?', the letter with case preserved is included in the output + * if the corresponding indicator is on. Otherwise, the letter is always + * included, lowercase when off and uppercase when on. + */ +const char * +keyboard_indicators(const char *fmt) +{ + Display *dpy; + XKeyboardState state; + size_t fmtlen, i, n; + int togglecase, isset; + char key; + + if (!(dpy = XOpenDisplay(NULL))) { + warn("XOpenDisplay: Failed to open display"); + return NULL; + } + XGetKeyboardControl(dpy, &state); + XCloseDisplay(dpy); + + fmtlen = strnlen(fmt, 4); + for (i = n = 0; i < fmtlen; i++) { + key = tolower(fmt[i]); + if (key != 'c' && key != 'n') { + continue; + } + togglecase = (i + 1 >= fmtlen || fmt[i + 1] != '?'); + isset = (state.led_mask & (1 << (key == 'n'))); + if (togglecase) { + buf[n++] = isset ? toupper(key) : key; + } else if (isset) { + buf[n++] = fmt[i]; + } + } + buf[n] = 0; + return buf; +} |