diff options
author | xAlpharax <42233094+xAlpharax@users.noreply.github.com> | 2023-08-14 20:40:34 +0300 |
---|---|---|
committer | xAlpharax <42233094+xAlpharax@users.noreply.github.com> | 2023-08-14 20:40:34 +0300 |
commit | 810cb8655fc97ce19db5f9190cb390f655d0f0d2 (patch) | |
tree | 72a2fc906bb1219257b5cd02ba46a9f77c60c762 /components/keymap.c |
Initial fork from https://git.suckless.org/slstatus
Changes to be committed:
new file: LICENSE
new file: Makefile
new file: README
new file: arg.h
new file: components/battery.c
new file: components/cat.c
new file: components/cpu.c
new file: components/datetime.c
new file: components/disk.c
new file: components/entropy.c
new file: components/hostname.c
new file: components/ip.c
new file: components/kernel_release.c
new file: components/keyboard_indicators.c
new file: components/keymap.c
new file: components/load_avg.c
new file: components/netspeeds.c
new file: components/num_files.c
new file: components/ram.c
new file: components/run_command.c
new file: components/swap.c
new file: components/temperature.c
new file: components/uptime.c
new file: components/user.c
new file: components/volume.c
new file: components/wifi.c
new file: config.def.h
new file: config.mk
new file: slstatus.1
new file: slstatus.c
new file: slstatus.h
new file: util.c
new file: util.h
Diffstat (limited to 'components/keymap.c')
-rw-r--r-- | components/keymap.c | 86 |
1 files changed, 86 insertions, 0 deletions
diff --git a/components/keymap.c b/components/keymap.c new file mode 100644 index 0000000..f8a2a47 --- /dev/null +++ b/components/keymap.c @@ -0,0 +1,86 @@ +/* See LICENSE file for copyright and license details. */ +#include <ctype.h> +#include <stdlib.h> +#include <string.h> +#include <X11/XKBlib.h> +#include <X11/Xlib.h> + +#include "../slstatus.h" +#include "../util.h" + +static int +valid_layout_or_variant(char *sym) +{ + size_t i; + /* invalid symbols from xkb rules config */ + static const char *invalid[] = { "evdev", "inet", "pc", "base" }; + + for (i = 0; i < LEN(invalid); i++) + if (!strncmp(sym, invalid[i], strlen(invalid[i]))) + return 0; + + return 1; +} + +static char * +get_layout(char *syms, int grp_num) +{ + char *tok, *layout; + int grp; + + layout = NULL; + tok = strtok(syms, "+:"); + for (grp = 0; tok && grp <= grp_num; tok = strtok(NULL, "+:")) { + if (!valid_layout_or_variant(tok)) { + continue; + } else if (strlen(tok) == 1 && isdigit(tok[0])) { + /* ignore :2, :3, :4 (additional layout groups) */ + continue; + } + layout = tok; + grp++; + } + + return layout; +} + +const char * +keymap(const char *unused) +{ + Display *dpy; + XkbDescRec *desc; + XkbStateRec state; + char *symbols; + const char *layout; + + layout = NULL; + + if (!(dpy = XOpenDisplay(NULL))) { + warn("XOpenDisplay: Failed to open display"); + return NULL; + } + if (!(desc = XkbAllocKeyboard())) { + warn("XkbAllocKeyboard: Failed to allocate keyboard"); + goto end; + } + if (XkbGetNames(dpy, XkbSymbolsNameMask, desc)) { + warn("XkbGetNames: Failed to retrieve key symbols"); + goto end; + } + if (XkbGetState(dpy, XkbUseCoreKbd, &state)) { + warn("XkbGetState: Failed to retrieve keyboard state"); + goto end; + } + if (!(symbols = XGetAtomName(dpy, desc->names->symbols))) { + warn("XGetAtomName: Failed to get atom name"); + goto end; + } + layout = bprintf("%s", get_layout(symbols, state.group)); + XFree(symbols); +end: + XkbFreeKeyboard(desc, XkbSymbolsNameMask, 1); + if (XCloseDisplay(dpy)) + warn("XCloseDisplay: Failed to close display"); + + return layout; +} |