X-Git-Url: https://git.dogcows.com/gitweb?a=blobdiff_plain;f=obt%2Fkeyboard.c;h=4e84f4819e24a8b872cbbbcf52a4b72a477461aa;hb=HEAD;hp=699fa8baccbd55accf1d32f7680e6b9495936301;hpb=c230443c6596039f426542f5a286120891a745c5;p=chaz%2Fopenbox diff --git a/obt/keyboard.c b/obt/keyboard.c index 699fa8ba..4e84f481 100644 --- a/obt/keyboard.c +++ b/obt/keyboard.c @@ -22,6 +22,14 @@ #include #include +struct _ObtIC +{ + guint ref; + XIC xic; + Window client; + Window focus; +}; + /* These masks are constants and the modifier keys are bound to them as anyone sees fit: ShiftMask (1<<0), LockMask (1<<1), ControlMask (1<<2), Mod1Mask (1<<3), @@ -34,12 +42,14 @@ #define nth_mask(n) (1 << n) static void set_modkey_mask(guchar mask, KeySym sym); +static void xim_init(void); void obt_keyboard_shutdown(); +void obt_keyboard_context_renew(ObtIC *ic); static XModifierKeymap *modmap; static KeySym *keymap; static gint min_keycode, max_keycode, keysyms_per_keycode; -/* This is a bitmask of the different masks for each modifier key */ +/*! This is a bitmask of the different masks for each modifier key */ static guchar modkeys_keys[OBT_KEYBOARD_NUM_MODKEYS]; static gboolean alt_l = FALSE; @@ -49,6 +59,10 @@ static gboolean hyper_l = FALSE; static gboolean started = FALSE; +static XIM xim = NULL; +static XIMStyle xim_style = 0; +static GSList *xic_all = NULL; + void obt_keyboard_reload(void) { gint i, j, k; @@ -56,12 +70,15 @@ void obt_keyboard_reload(void) if (started) obt_keyboard_shutdown(); /* free stuff */ started = TRUE; + xim_init(); + /* reset the keys to not be bound to any masks */ for (i = 0; i < OBT_KEYBOARD_NUM_MODKEYS; ++i) modkeys_keys[i] = 0; modmap = XGetModifierMapping(obt_display); - g_assert(modmap->max_keypermod > 0); + /* note: modmap->max_keypermod can be 0 when there is no valid key layout + available */ XDisplayKeycodes(obt_display, &min_keycode, &max_keycode); keymap = XGetKeyboardMapping(obt_display, min_keycode, @@ -99,30 +116,94 @@ void obt_keyboard_reload(void) void obt_keyboard_shutdown(void) { + GSList *it; + XFreeModifiermap(modmap); modmap = NULL; XFree(keymap); keymap = NULL; + for (it = xic_all; it; it = g_slist_next(it)) { + ObtIC* ic = it->data; + if (ic->xic) { + XDestroyIC(ic->xic); + ic->xic = NULL; + } + } + if (xim) XCloseIM(xim); + xim = NULL; + xim_style = 0; started = FALSE; } -guint obt_keyboard_keycode_to_modmask(guint keycode) +void xim_init(void) { - gint i, j; - guint mask = 0; + GSList *it; + gchar *aname, *aclass; + + aname = g_strdup(g_get_prgname()); + if (!aname) aname = g_strdup("obt"); + + aclass = g_strdup(aname); + if (g_ascii_islower(aclass[0])) + aclass[0] = g_ascii_toupper(aclass[0]); + + xim = XOpenIM(obt_display, NULL, aname, aclass); + + if (!xim) + g_message("Failed to open an Input Method"); + else { + XIMStyles *xim_styles = NULL; + char *r; + + /* get the input method styles */ + r = XGetIMValues(xim, XNQueryInputStyle, &xim_styles, NULL); + if (r || !xim_styles) + g_message("Input Method does not support any styles"); + if (xim_styles) { + int i; + + /* find a style that doesnt need preedit or status */ + for (i = 0; i < xim_styles->count_styles; ++i) { + if (xim_styles->supported_styles[i] == + (XIMPreeditNothing | XIMStatusNothing)) + { + xim_style = xim_styles->supported_styles[i]; + break; + } + } + XFree(xim_styles); + } - if (keycode == NoSymbol) return 0; + if (!xim_style) { + g_message("Input Method does not support a usable style"); - /* go through each of the modifier masks (eg ShiftMask, CapsMask...) */ - for (i = 0; i < NUM_MASKS; ++i) { - /* go through each keycode that is bound to the mask */ - for (j = 0; j < modmap->max_keypermod; ++j) { - /* compare with a keycode that is bound to the mask (i) */ - if (modmap->modifiermap[i*modmap->max_keypermod + j] == keycode) - mask |= nth_mask(i); + XCloseIM(xim); + xim = NULL; } } - return mask; + + /* any existing contexts need to be recreated for the new input method */ + for (it = xic_all; it; it = g_slist_next(it)) + obt_keyboard_context_renew(it->data); + + g_free(aclass); + g_free(aname); +} + +guint obt_keyboard_keyevent_to_modmask(XEvent *e) +{ + gint i, masknum; + + g_return_val_if_fail(e->type == KeyPress || e->type == KeyRelease, + OBT_KEYBOARD_MODKEY_NONE); + + for (masknum = 0; masknum < NUM_MASKS; ++masknum) + for (i = 0; i < modmap->max_keypermod; ++i) { + KeyCode c = modmap->modifiermap[masknum*modmap->max_keypermod + i]; + if (c == e->xkey.keycode) + return 1<type == KeyPress, 0); + + if (!ic) + g_warning("Using obt_keyboard_keypress_to_unichar() without an " + "Input Context. No i18n support!"); + + if (ic && ic->xic) { + buf = fixbuf; + bufsz = sizeof(fixbuf); + +#ifdef X_HAVE_UTF8_STRING + len = Xutf8LookupString(ic->xic, &ev->xkey, buf, bufsz, &sym, &status); +#else + len = XmbLookupString(ic->xic, &ev->xkey, buf, bufsz, &sym, &status); +#endif + + if (status == XBufferOverflow) { + buf = g_new(char, len); + bufsz = len; + +#ifdef X_HAVE_UTF8_STRING + len = Xutf8LookupString(ic->xic, &ev->xkey, buf, bufsz, &sym, + &status); +#else + len = XmbLookupString(ic->xic, &ev->xkey, buf, bufsz, &sym, + &status); +#endif + } + + if ((status == XLookupChars || status == XLookupBoth)) { + if ((guchar)buf[0] >= 32) { /* not an ascii control character */ +#ifndef X_HAVE_UTF8_STRING + /* convert to utf8 */ + gchar *buf2 = buf; + buf = g_locale_to_utf8(buf2, r, NULL, NULL, NULL); + g_free(buf2); +#endif - if ((sym = XKeycodeToKeysym(obt_display, keycode, 0)) != NoSymbol) - return g_locale_to_utf8(XKeysymToString(sym), -1, NULL, NULL, NULL); - return NULL; + got_string = TRUE; + } + } + else if (status == XLookupKeySym) + /* this key doesn't have a text representation, it is a command + key of some sort */; + else + g_message("Bad keycode lookup. Keysym 0x%x Status: %s\n", + (guint) sym, + (status == XBufferOverflow ? "BufferOverflow" : + status == XLookupNone ? "XLookupNone" : + status == XLookupKeySym ? "XLookupKeySym" : + "Unknown status")); + } + else { + buf = fixbuf; + bufsz = sizeof(fixbuf); + len = XLookupString(&ev->xkey, buf, bufsz, &sym, NULL); + if ((guchar)buf[0] >= 32) /* not an ascii control character */ + got_string = TRUE; + } + + if (got_string) { + gunichar u = g_utf8_get_char_validated(buf, len); + if (u && u != (gunichar)-1 && u != (gunichar)-2) + unikey = u; + } + + if (buf != fixbuf) g_free(buf); + + return unikey; } -gunichar obt_keyboard_keycode_to_unichar(guint keycode) +KeySym obt_keyboard_keypress_to_keysym(XEvent *ev) { - gunichar unikey = 0; - char *key; - - if ((key = obt_keyboard_keycode_to_string(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; + KeySym sym; + + g_return_val_if_fail(ev->type == KeyPress, None); + + sym = None; + XLookupString(&ev->xkey, NULL, 0, &sym, NULL); + return sym; +} + +void obt_keyboard_context_renew(ObtIC *ic) +{ + if (xim) { + ic->xic = XCreateIC(xim, + XNInputStyle, xim_style, + XNClientWindow, ic->client, + XNFocusWindow, ic->focus, + NULL); + if (!ic->xic) + g_message("Error creating Input Context for window 0x%x 0x%x\n", + (guint)ic->client, (guint)ic->focus); + } +} + +ObtIC* obt_keyboard_context_new(Window client, Window focus) +{ + ObtIC *ic; + + g_return_val_if_fail(client != None && focus != None, NULL); + + ic = g_slice_new(ObtIC); + ic->ref = 1; + ic->client = client; + ic->focus = focus; + ic->xic = NULL; + + obt_keyboard_context_renew(ic); + xic_all = g_slist_prepend(xic_all, ic); + + return ic; +} + +void obt_keyboard_context_ref(ObtIC *ic) +{ + ++ic->ref; +} + +void obt_keyboard_context_unref(ObtIC *ic) +{ + if (--ic->ref < 1) { + xic_all = g_slist_remove(xic_all, ic); + if (ic->xic) + XDestroyIC(ic->xic); + g_slice_free(ObtIC, ic); } - g_free(key); - return unikey; }