使用XLookupString处理C-q

时间:2017-03-30 09:35:23

标签: c x11

这是X11代码的一个工作示例,它处理Ctrl-q事件以退出应用程序:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xos.h>

void exitOnCondition(char cond, const char *msg, int exitCode, Display *dpy, Window *w, GC *gc) {
    if(cond) {
        printf("%s\n", msg);
        if(dpy && gc) XFreeGC(dpy, *gc);
        if(dpy && w) XDestroyWindow(dpy, *w);
        if(dpy) XCloseDisplay(dpy);
        exit(exitCode);
    }
}

int main(int argc, char **argv) {
      Display *dpy = XOpenDisplay(0);
      exitOnCondition(dpy == 0, "Error: XOpenDisplay failed", -1, dpy, 0, 0);

      int blackColor = BlackPixel(dpy, DefaultScreen(dpy));
      int whiteColor = WhitePixel(dpy, DefaultScreen(dpy));

      Window w = XCreateSimpleWindow(dpy, DefaultRootWindow(dpy), 0, 0, 
                     200, 100, 0, blackColor, blackColor);

      //Tell X Server to send MapNotify events
      XSelectInput(dpy, w, StructureNotifyMask | KeyPressMask);

      //Make window appear
      XMapWindow(dpy, w);

      //Graphics Context
      GC gc = XCreateGC(dpy, w, 0, 0);

      //Set white color for drawing
      XSetForeground(dpy, gc, whiteColor);

      //Wait for the MapNotify event
      for(;;) {
        XEvent e;
        XNextEvent(dpy, &e);
        if (e.type == MapNotify) {
          break;
        }
      }

      //Draw the line
      XDrawLine(dpy, w, gc, 10, 60, 180, 20);

      //Send the "DrawLine" request to the server
      XFlush(dpy);

      char text[255];
      XEvent e;
      KeySym key;
      int numKeys = 0;
      for(;;) {
          XNextEvent(dpy, &e);
          if(e.type == KeyPress) {
              //With modifier XLookupString will return garbage(?) in text[0] and key as latin1
              if((numKeys = XLookupString(&e.xkey, text, 255, &key, 0))) {
                  printf("lookup returned:\n");
                  for(int i = 0; i < numKeys; i++) {
                      printf("text[%d]=%x\n", i, text[i]);
                  }

                  if(e.xkey.state == ControlMask && key == XK_q) {
                      exitOnCondition(1, "C-Q pressed", 0, dpy, &w, &gc);
                  }
              }
          }
      }

      XFreeGC(dpy, gc);
      XDestroyWindow(dpy,w);
      XCloseDisplay(dpy);   
}

此代码是否会在任何系统上正确处理Ctrl-q事件?

即使Ctrl被反弹到CAPS Lock(或其他任何东西),我可以使用e.xkey.state在XLookupString之后检查任何键盘布局的Ctrl修饰符吗?

为什么XLookupString为Ctrl-q事件返回一个符号文本[0] == 0x11而不是text [0] == CtrlModifierCode text [1] =='q'?

1 个答案:

答案 0 :(得分:0)

XLookupString返回一系列ISO-8859-1字符(至少根据我的手册;我不知道它是如何更新的)。 ISO-8859-1中没有用于“ctrl”键的字符代码。严格来说,ctrl-q组合也没有,但是tradition dictates ctrl +(A ... Z)映射到1 ... 26范围。