]> git.sur5r.net Git - i3/i3/commitdiff
Translate bindsym bindings upon ISO_Next_Group
authorMichael Stapelberg <michael@stapelberg.de>
Thu, 6 Aug 2015 07:32:22 +0000 (09:32 +0200)
committerMichael Stapelberg <michael@stapelberg.de>
Thu, 6 Aug 2015 07:32:22 +0000 (09:32 +0200)
With commit c738b2e454bb8b096dd99d44e9e51030f8355b90 we changed i3 so
that the default keybindings can be used when ISO_Next_Group is enabled,
but bindings which explicitly use Mode_switch have precedence. This
behavior required the use of bindcode instead of bindsym.

With this commit, when switching from group 1 to group 2 using
ISO_Next_Group, i3 will re-translate all keybindings (looking at column
2/3, regardless of whether the keybinding itself specifies Mode_switch)
and re-grab them.

That way, the keybinding “bindsym $mod+x nop foo” will work when
pressing $mod+x without Mode_switch and when pressing the corresponding
$mod+x (different key) with Mode_switch. A binding such as “bindsym
Mode_switch+$mod+x nop bar” will still have precedence.

The intention here is to make bindsym keybindings work well with dual
keyboard layouts (such as {dvorak, us} or {us, ru}), so that users can
switch between groups and still have their (logical) keybindings behave
the same way.

fixes #1775

src/bindings.c
src/handlers.c

index 8e8e9febe8bee0f97926ec690090f4cc6f3e2b36..adca55464d6076da40f981d6a782d59126a3a2bd 100644 (file)
@@ -111,9 +111,7 @@ static void grab_keycode_for_binding(xcb_connection_t *conn, Binding *bind, uint
 void grab_all_keys(xcb_connection_t *conn, bool bind_mode_switch) {
     Binding *bind;
     TAILQ_FOREACH(bind, bindings, bindings) {
-        if (bind->input_type != B_KEYBOARD ||
-            (bind_mode_switch && (bind->mods & BIND_MODE_SWITCH) == 0) ||
-            (!bind_mode_switch && (bind->mods & BIND_MODE_SWITCH) != 0))
+        if (bind->input_type != B_KEYBOARD)
             continue;
 
         /* The easy case: the user specified a keycode directly. */
@@ -225,7 +223,7 @@ Binding *get_binding_from_xcb_event(xcb_generic_event_t *event) {
     /* No match? Then the user has Mode_switch enabled but does not have a
      * specific keybinding. Fall back to the default keybindings (without
      * Mode_switch). Makes it much more convenient for users of a hybrid
-     * layout (like ru). */
+     * layout (like {us, ru} or {dvorak, us}, see e.g. ticket #1775). */
     if (bind == NULL) {
         state_filtered &= ~(BIND_MODE_SWITCH);
         DLOG("no match, new state_filtered = %d\n", state_filtered);
@@ -252,6 +250,8 @@ void translate_keysyms(void) {
     int col;
     xcb_keycode_t i, min_keycode, max_keycode;
 
+    const bool mode_switch = (xkb_current_group == XCB_XKB_GROUP_2);
+
     min_keycode = xcb_get_setup(conn)->min_keycode;
     max_keycode = xcb_get_setup(conn)->max_keycode;
 
@@ -282,7 +282,7 @@ void translate_keysyms(void) {
          * the base column and the corresponding shift column, so without
          * mode_switch, we look in 0 and 1, with mode_switch we look in 2 and
          * 3. */
-        col = (bind->mods & BIND_MODE_SWITCH ? 2 : 0);
+        col = (bind->mods & BIND_MODE_SWITCH || mode_switch ? 2 : 0);
 
         FREE(bind->translated_to);
         bind->number_keycodes = 0;
index 49636543e1e63359674320f9f2191e41e29ca5c4..714e183b3e19bbc6f25755c4b9d440dd62d8e730 100644 (file)
@@ -1361,14 +1361,10 @@ void handle_event(int type, xcb_generic_event_t *event) {
             if (xkb_current_group == state->group)
                 return;
             xkb_current_group = state->group;
-            if (state->group == XCB_XKB_GROUP_1) {
-                DLOG("Mode_switch disabled\n");
-                ungrab_all_keys(conn);
-                grab_all_keys(conn, false);
-            } else {
-                DLOG("Mode_switch enabled\n");
-                grab_all_keys(conn, true);
-            }
+            DLOG("Mode_switch %s\n", (xkb_current_group == XCB_XKB_GROUP_1 ? "disabled" : "enabled"));
+            ungrab_all_keys(conn);
+            translate_keysyms();
+            grab_all_keys(conn, (xkb_current_group == XCB_XKB_GROUP_2));
         }
 
         return;