i3-config-wizard: Bugfix: Check the modifier mapping to decide whether to use Mod1 or Mod4

Fixes #491

Before, we assumed XK_Alt_L was in Mod1 and XK_Super_L was in Mod4 (which is
true on most systems). Now we just check if the keycode is in Mod1 or Mod4 and
write the config accordingly.
This commit is contained in:
Michael Stapelberg 2011-08-28 16:16:53 +02:00
parent c96b5a12a7
commit 0dc11b5d22

View File

@ -54,10 +54,11 @@ while (0)
#include "ipc.h" #include "ipc.h"
enum { STEP_WELCOME, STEP_GENERATE } current_step = STEP_WELCOME; enum { STEP_WELCOME, STEP_GENERATE } current_step = STEP_WELCOME;
enum { MOD_ALT, MOD_SUPER } modifier = MOD_SUPER; enum { MOD_Mod1, MOD_Mod4 } modifier = MOD_Mod4;
static char *config_path; static char *config_path;
static xcb_connection_t *conn; static xcb_connection_t *conn;
static xcb_get_modifier_mapping_reply_t *modmap_reply;
static uint32_t font_id; static uint32_t font_id;
static uint32_t font_bold_id; static uint32_t font_bold_id;
static char *socket_path; static char *socket_path;
@ -208,13 +209,13 @@ static int handle_expose() {
txt(85, 10, "to abort"); txt(85, 10, "to abort");
/* the not-selected modifier */ /* the not-selected modifier */
if (modifier == MOD_SUPER) if (modifier == MOD_Mod4)
txt(31, 5, "<Alt>"); txt(31, 5, "<Alt>");
else txt(31, 4, "<Win>"); else txt(31, 4, "<Win>");
/* the selected modifier */ /* the selected modifier */
xcb_change_gc_single(conn, pixmap_gc, XCB_GC_FONT, font_bold_id); xcb_change_gc_single(conn, pixmap_gc, XCB_GC_FONT, font_bold_id);
if (modifier == MOD_SUPER) if (modifier == MOD_Mod4)
txt(31, 4, "<Win>"); txt(31, 4, "<Win>");
else txt(31, 5, "<Alt>"); else txt(31, 5, "<Alt>");
@ -271,11 +272,34 @@ static int handle_key_press(void *ignored, xcb_connection_t *conn, xcb_key_press
if (sym == XK_Escape) if (sym == XK_Escape)
exit(0); exit(0);
if (sym == XK_Alt_L) /* Check if this is Mod1 or Mod4. The modmap contains Shift, Lock, Control,
modifier = MOD_ALT; * Mod1, Mod2, Mod3, Mod4, Mod5 (in that order) */
xcb_keycode_t *modmap = xcb_get_modifier_mapping_keycodes(modmap_reply);
/* Mod1? */
int mask = 3;
for (int i = 0; i < modmap_reply->keycodes_per_modifier; i++) {
xcb_keycode_t code = modmap[(mask * modmap_reply->keycodes_per_modifier) + i];
if (code == XCB_NONE)
continue;
printf("Modifier keycode for Mod1: 0x%02x\n", code);
if (code == event->detail) {
modifier = MOD_Mod1;
printf("This is Mod1!\n");
}
}
if (sym == XK_Super_L) /* Mod4? */
modifier = MOD_SUPER; mask = 6;
for (int i = 0; i < modmap_reply->keycodes_per_modifier; i++) {
xcb_keycode_t code = modmap[(mask * modmap_reply->keycodes_per_modifier) + i];
if (code == XCB_NONE)
continue;
printf("Modifier keycode for Mod4: 0x%02x\n", code);
if (code == event->detail) {
modifier = MOD_Mod4;
printf("This is Mod4!\n");
}
}
handle_expose(); handle_expose();
return 1; return 1;
@ -291,13 +315,13 @@ static void handle_button_press(xcb_button_press_event_t* event) {
if (event->event_x >= 32 && event->event_x <= 68 && if (event->event_x >= 32 && event->event_x <= 68 &&
event->event_y >= 45 && event->event_y <= 54) { event->event_y >= 45 && event->event_y <= 54) {
modifier = MOD_SUPER; modifier = MOD_Mod4;
handle_expose(); handle_expose();
} }
if (event->event_x >= 32 && event->event_x <= 68 && if (event->event_x >= 32 && event->event_x <= 68 &&
event->event_y >= 56 && event->event_y <= 70) { event->event_y >= 56 && event->event_y <= 70) {
modifier = MOD_ALT; modifier = MOD_Mod1;
handle_expose(); handle_expose();
} }
@ -356,7 +380,7 @@ static void finish() {
/* Set the modifier the user chose */ /* Set the modifier the user chose */
if (strncmp(walk, "set $mod ", strlen("set $mod ")) == 0) { if (strncmp(walk, "set $mod ", strlen("set $mod ")) == 0) {
if (modifier == MOD_ALT) if (modifier == MOD_Mod1)
fputs("set $mod Mod1\n", ks_config); fputs("set $mod Mod1\n", ks_config);
else fputs("set $mod Mod4\n", ks_config); else fputs("set $mod Mod4\n", ks_config);
continue; continue;
@ -459,6 +483,9 @@ int main(int argc, char *argv[]) {
xcb_connection_has_error(conn)) xcb_connection_has_error(conn))
errx(1, "Cannot open display\n"); errx(1, "Cannot open display\n");
xcb_get_modifier_mapping_cookie_t modmap_cookie;
modmap_cookie = xcb_get_modifier_mapping(conn);
/* Place requests for the atoms we need as soon as possible */ /* Place requests for the atoms we need as soon as possible */
#define xmacro(atom) \ #define xmacro(atom) \
xcb_intern_atom_cookie_t atom ## _cookie = xcb_intern_atom(conn, 0, strlen(#atom), #atom); xcb_intern_atom_cookie_t atom ## _cookie = xcb_intern_atom(conn, 0, strlen(#atom), #atom);
@ -468,6 +495,11 @@ int main(int argc, char *argv[]) {
xcb_screen_t *root_screen = xcb_aux_get_screen(conn, screens); xcb_screen_t *root_screen = xcb_aux_get_screen(conn, screens);
root = root_screen->root; root = root_screen->root;
if (!(modmap_reply = xcb_get_modifier_mapping_reply(conn, modmap_cookie, NULL)))
errx(EXIT_FAILURE, "Could not get modifier mapping\n");
/* XXX: we should refactor xcb_get_numlock_mask so that it uses the
* modifier mapping we already have */
xcb_get_numlock_mask(conn); xcb_get_numlock_mask(conn);
symbols = xcb_key_symbols_alloc(conn); symbols = xcb_key_symbols_alloc(conn);