如何使用键盘布局获取XLib键的字符?

时间:2016-02-13 09:11:59

标签: c++ linux internationalization x11 xlib

使用XKeycodeToKeysym(在C ++中)我可以获得按键事件的KeySym。从我的理解,这已经应该尊重键盘布局。但是,当我切换键盘布局(英语和希伯来语之间)时,我得到了相同的KeySym。我怀疑Xlib Keysym只尊重X11级别中定义的键盘布局?在我的系统中,键盘布局仅在桌面环境级别(Mate)中定义。如果是这样,有没有办法在不使用像Qt / GTK这样的工具包的情况下获得正确的角色?我必须单独处理每个桌面环境吗?

[编辑]

我已经尝试了以下(根据安德烈的建议),但这并不起作用:

#include <X11/XKBlib.h>

#include <cstring>
#include <cassert>
#include <iostream>

int main() {
    Display *display = XOpenDisplay(nullptr);
    Window root, window;
    XSetWindowAttributes swa;
    root = DefaultRootWindow( display );
    XSelectInput( display, root, MappingNotify );
    std::memset(&swa, 0, sizeof swa );
    swa.event_mask = MappingNotify | KeyPressMask;
    window = XCreateWindow( display, root, 50, 200, 1024, 768, 0, CopyFromParent, InputOutput, CopyFromParent,
                             CWEventMask, &swa );
    XMapRaised( display, window );
    int xkbEventCode=0, n0=0, n1=0, n2=0, n3=0;
    bool isOk = XkbQueryExtension(  display, &n0, &xkbEventCode, &n1, &n2, &n3 ) ==True;
    assert( isOk );
    isOk = XkbSelectEvents( display, XkbUseCoreKbd, XkbAllEventsMask, XkbAllEventsMask ) ==True;
    assert( isOk );
    while (true) {
        XEvent event;
        std::memset( &event, 0, sizeof event );
        XNextEvent( display, &event );
        if (event.type == xkbEventCode)
            switch (reinterpret_cast<XkbEvent*>(&event)->any.xkb_type) {
              case XkbNewKeyboardNotify:
              case XkbMapNotify: {
                std::cout << "Keyboard mapping has changed." << std::endl;
                break;
                }
            default:  break;
            }
        else
            switch (event.type) {
              case KeyPress: {
                KeyCode keyCode = event.xkey.keycode;
                int keySymsPerkeyCode=0;
                KeySym *keySyms( XGetKeyboardMapping(display, keyCode, 1, &keySymsPerkeyCode)),
                       *keySym = keySyms;
                while (keySymsPerkeyCode--  &&  *keySym != NoSymbol) {
                    std::cout << *keySym << std::endl;
                    ++keySym;
                    }
                XFree( keySyms );
                break;
                }
              case MappingNotify: {
                std::cout << "Keyboard mapping has changed." << std::endl;
                break;
                }
              default:  break;
              }
        }
    return 0;
    }

1 个答案:

答案 0 :(得分:1)

我只能在较低级别回答,不知道XKeycodeToKeysym做了什么。

您需要收听MappingNotify事件,并且每次收到该事件时都会使用XGetKeyboardMapping

重新阅读新映射(通常这意味着键盘布局/语言已更改)

这是核心X协议,桌面环境/ WM独立的一部分

<强>更新

看起来xlib默认启用XKeyboard扩展。启用后,您需要明确表达对映射更改事件的兴趣(与核心协议不同,其中MappingNotify始终发送给所有客户端) - 请参阅http://www.x.org/archive/X11R7.5/doc/man/man3/XkbSelectEvents.3.html

 XkbSelectEvents (display, XkbUseCoreKbd, XkbAllEventsMask, XkbAllEventsMask);

之后,您回复XkbMapNotify事件并执行XkbGetMap()以获得新的布局