aboutsummaryrefslogtreecommitdiff
path: root/sway/commands/input
diff options
context:
space:
mode:
Diffstat (limited to 'sway/commands/input')
-rw-r--r--sway/commands/input/xkb_switch_layout.c51
1 files changed, 51 insertions, 0 deletions
diff --git a/sway/commands/input/xkb_switch_layout.c b/sway/commands/input/xkb_switch_layout.c
new file mode 100644
index 00000000..fdf21452
--- /dev/null
+++ b/sway/commands/input/xkb_switch_layout.c
@@ -0,0 +1,51 @@
+#define _POSIX_C_SOURCE 200809L
+#include "sway/config.h"
+#include "sway/commands.h"
+#include "sway/input/input-manager.h"
+#include "log.h"
+
+static void switch_layout(struct wlr_keyboard *kbd, xkb_layout_index_t idx) {
+ xkb_layout_index_t num_layouts = xkb_keymap_num_layouts(kbd->keymap);
+ if (idx >= num_layouts) {
+ return;
+ }
+ wlr_keyboard_notify_modifiers(kbd, kbd->modifiers.depressed,
+ kbd->modifiers.latched, kbd->modifiers.locked, idx);
+}
+
+struct cmd_results *input_cmd_xkb_switch_layout(int argc, char **argv) {
+ struct cmd_results *error = NULL;
+ if ((error = checkarg(argc, "xkb_switch_layout", EXPECTED_EQUAL_TO, 1))) {
+ return error;
+ }
+ struct input_config *ic = config->handler_context.input_config;
+ if (!ic) {
+ return cmd_results_new(CMD_FAILURE, "No input device defined.");
+ }
+
+ if (config->reading || !config->active) {
+ return cmd_results_new(CMD_DEFER, NULL);
+ }
+
+ const char *layout_str = argv[0];
+
+ char *end;
+ int layout = strtol(layout_str, &end, 10);
+ if (layout_str[0] == '\0' || end[0] != '\0' || layout < 0) {
+ return cmd_results_new(CMD_FAILURE, "Invalid layout index.");
+ }
+
+ struct sway_input_device *dev;
+ wl_list_for_each(dev, &server.input->devices, link) {
+ if (strcmp(ic->identifier, "*") != 0 &&
+ strcmp(ic->identifier, dev->identifier) != 0) {
+ continue;
+ }
+ if (dev->wlr_device->type != WLR_INPUT_DEVICE_KEYBOARD) {
+ continue;
+ }
+ switch_layout(dev->wlr_device->keyboard, layout);
+ }
+
+ return cmd_results_new(CMD_SUCCESS, NULL);
+}