summaryrefslogtreecommitdiff
path: root/components/keymap.c
diff options
context:
space:
mode:
authorxAlpharax <42233094+xAlpharax@users.noreply.github.com>2023-08-14 20:40:34 +0300
committerxAlpharax <42233094+xAlpharax@users.noreply.github.com>2023-08-14 20:40:34 +0300
commit810cb8655fc97ce19db5f9190cb390f655d0f0d2 (patch)
tree72a2fc906bb1219257b5cd02ba46a9f77c60c762 /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.c86
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;
+}