7e946f68d9
1) it can handle more user cases. 2) it can handle binding more modifier keys with their keysym names like Hyper and Super. W is a shortcut for Super, cuz the windows key is generally bound to it. 3) if you change your keymap bindings and reconfigure openbox it will learn your changes. and i dunno.. its just nice and stuff now.. you can actually read it I think..
185 lines
5.5 KiB
C
185 lines
5.5 KiB
C
/* -*- indent-tabs-mode: nil; tab-width: 4; c-basic-offset: 4; -*-
|
|
|
|
translate.c for the Openbox window manager
|
|
Copyright (c) 2006 Mikael Magnusson
|
|
Copyright (c) 2003-2007 Dana Jansens
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
See the COPYING file for a copy of the GNU General Public License.
|
|
*/
|
|
|
|
#include "openbox.h"
|
|
#include "mouse.h"
|
|
#include "modkeys.h"
|
|
#include "gettext.h"
|
|
#include <glib.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
static guint translate_modifier(gchar *str)
|
|
{
|
|
guint mask = 0;
|
|
|
|
if (!g_ascii_strcasecmp("Mod1", str)) mask = Mod1Mask;
|
|
else if (!g_ascii_strcasecmp("Mod2", str)) mask = Mod2Mask;
|
|
else if (!g_ascii_strcasecmp("Mod3", str)) mask = Mod3Mask;
|
|
else if (!g_ascii_strcasecmp("Mod4", str)) mask = Mod4Mask;
|
|
else if (!g_ascii_strcasecmp("Mod5", str)) mask = Mod5Mask;
|
|
|
|
else if (!g_ascii_strcasecmp("Control", str) ||
|
|
!g_ascii_strcasecmp("C", str))
|
|
mask = modkeys_key_to_mask(OB_MODKEY_KEY_CONTROL);
|
|
else if (!g_ascii_strcasecmp("Alt", str) ||
|
|
!g_ascii_strcasecmp("A", str))
|
|
mask = modkeys_key_to_mask(OB_MODKEY_KEY_ALT);
|
|
else if (!g_ascii_strcasecmp("Meta", str) ||
|
|
!g_ascii_strcasecmp("M", str))
|
|
mask = modkeys_key_to_mask(OB_MODKEY_KEY_META);
|
|
/* W = windows key, is linked to the Super_L/R buttons */
|
|
else if (!g_ascii_strcasecmp("Super", str) ||
|
|
!g_ascii_strcasecmp("W", str))
|
|
mask = modkeys_key_to_mask(OB_MODKEY_KEY_SUPER);
|
|
else if (!g_ascii_strcasecmp("Shift", str) ||
|
|
!g_ascii_strcasecmp("S", str))
|
|
mask = modkeys_key_to_mask(OB_MODKEY_KEY_SHIFT);
|
|
else if (!g_ascii_strcasecmp("Hyper", str) ||
|
|
!g_ascii_strcasecmp("H", str))
|
|
mask = modkeys_key_to_mask(OB_MODKEY_KEY_HYPER);
|
|
else
|
|
g_message(_("Invalid modifier key '%s' in key/pointer binding"), str);
|
|
|
|
return mask;
|
|
}
|
|
|
|
gboolean translate_button(const gchar *str, guint *state, guint *button)
|
|
{
|
|
gchar **parsed;
|
|
gchar *l;
|
|
gint i;
|
|
gboolean ret = FALSE;
|
|
|
|
parsed = g_strsplit(str, "-", -1);
|
|
|
|
/* first, find the button (last token) */
|
|
l = NULL;
|
|
for (i = 0; parsed[i] != NULL; ++i)
|
|
l = parsed[i];
|
|
if (l == NULL)
|
|
goto translation_fail;
|
|
|
|
/* figure out the mod mask */
|
|
*state = 0;
|
|
for (i = 0; parsed[i] != l; ++i) {
|
|
guint m = translate_modifier(parsed[i]);
|
|
if (!m) goto translation_fail;
|
|
*state |= m;
|
|
}
|
|
|
|
/* figure out the button */
|
|
if (!g_ascii_strcasecmp("Left", l)) *button = 1;
|
|
else if (!g_ascii_strcasecmp("Middle", l)) *button = 2;
|
|
else if (!g_ascii_strcasecmp("Right", l)) *button = 3;
|
|
else if (!g_ascii_strcasecmp("Up", l)) *button = 4;
|
|
else if (!g_ascii_strcasecmp("Down", l)) *button = 5;
|
|
else if (!g_ascii_strncasecmp("Button", l, 6)) *button = atoi(l+6);
|
|
if (!*button) {
|
|
g_message(_("Invalid button '%s' in pointer binding"), l);
|
|
goto translation_fail;
|
|
}
|
|
|
|
ret = TRUE;
|
|
|
|
translation_fail:
|
|
g_strfreev(parsed);
|
|
return ret;
|
|
}
|
|
|
|
gboolean translate_key(const gchar *str, guint *state, guint *keycode)
|
|
{
|
|
gchar **parsed;
|
|
gchar *l;
|
|
gint i;
|
|
gboolean ret = FALSE;
|
|
KeySym sym;
|
|
|
|
parsed = g_strsplit(str, "-", -1);
|
|
|
|
/* first, find the key (last token) */
|
|
l = NULL;
|
|
for (i = 0; parsed[i] != NULL; ++i)
|
|
l = parsed[i];
|
|
if (l == NULL)
|
|
goto translation_fail;
|
|
|
|
/* figure out the mod mask */
|
|
*state = 0;
|
|
for (i = 0; parsed[i] != l; ++i) {
|
|
guint m = translate_modifier(parsed[i]);
|
|
if (!m) goto translation_fail;
|
|
*state |= m;
|
|
}
|
|
|
|
if (!g_ascii_strncasecmp("0x", l, 2)) {
|
|
gchar *end;
|
|
|
|
/* take it directly */
|
|
*keycode = strtol(l, &end, 16);
|
|
if (*l == '\0' || *end != '\0') {
|
|
g_message(_("Invalid key code '%s' in key binding"), l);
|
|
goto translation_fail;
|
|
}
|
|
} else {
|
|
/* figure out the keycode */
|
|
sym = XStringToKeysym(l);
|
|
if (sym == NoSymbol) {
|
|
g_message(_("Invalid key name '%s' in key binding"), l);
|
|
goto translation_fail;
|
|
}
|
|
*keycode = XKeysymToKeycode(ob_display, sym);
|
|
}
|
|
if (!*keycode) {
|
|
g_message(_("Requested key '%s' does not exist on the display"), l);
|
|
goto translation_fail;
|
|
}
|
|
|
|
ret = TRUE;
|
|
|
|
translation_fail:
|
|
g_strfreev(parsed);
|
|
return ret;
|
|
}
|
|
|
|
const gchar *translate_keycode(guint keycode)
|
|
{
|
|
KeySym sym;
|
|
const gchar *ret = NULL;
|
|
|
|
if ((sym = XKeycodeToKeysym(ob_display, keycode, 0)) != NoSymbol)
|
|
ret = XKeysymToString(sym);
|
|
return g_locale_to_utf8(ret, -1, NULL, NULL, NULL);
|
|
}
|
|
|
|
gunichar translate_unichar(guint keycode)
|
|
{
|
|
gunichar unikey = 0;
|
|
|
|
const char *key;
|
|
if ((key = translate_keycode(keycode)) != NULL &&
|
|
/* don't accept keys that aren't a single letter, like "space" */
|
|
key[1] == '\0')
|
|
{
|
|
unikey = g_utf8_get_char_validated(key, -1);
|
|
if (unikey == (gunichar)-1 || unikey == (gunichar)-2 || unikey == 0)
|
|
unikey = 0;
|
|
}
|
|
return unikey;
|
|
}
|