免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1457 | 回复: 1
打印 上一主题 下一主题

为什么我能跟XIM连接,但只能输入英文。(见代码) [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2009-11-11 08:53 |只看该作者 |倒序浏览

  1. #include <stdio.h>
  2. #include <stdlib.h>

  3. #include <X11/Xlib.h>
  4. #include <X11/keysym.h>
  5. #include <X11/Xutil.h>
  6. #include <X11/extensions/XShm.h>
  7. #include <X11/Xlocale.h>

  8. #include <time.h>
  9. #include <sys/time.h>

  10. #define W 300
  11. #define H 300

  12. XFontSet createFontSet (Display* dpy, XIM im)
  13. {
  14.     XFontSet fontset;
  15.     char **missing_charsets;
  16.     int num_missing_charsets = 0;
  17.     char *default_string;
  18.     int i;

  19.     fontset = XCreateFontSet(dpy,
  20.         "-adobe-helvetica-medium-r-normal--14-*-*-*-*-*-iso8859-*,\
  21.             -*-song-medium-r-normal--14-*-*-*-*-*-gb2312.1980-0",
  22.             &missing_charsets, &num_missing_charsets,
  23.             &default_string);
  24.     /*
  25.      * if there are charsets for which no fonts can
  26.      * be found, print a warning message.
  27.      */
  28.     if (num_missing_charsets > 0) {
  29.         fprintf(stderr, "The following charsets are missing:\n");

  30.         for(i=0; i < num_missing_charsets; i++)
  31.         {
  32.             fprintf(stderr, "%s\n", missing_charsets[i]);
  33.         }

  34.         fprintf(stderr, "The string %s will be used in place of "
  35.                 "any characters from those sets.", default_string);

  36.         XFreeStringList(missing_charsets);
  37.     }
  38.     return fontset;
  39. }

  40. XIMStyle ChooseBetterStyle(XIMStyle style1, XIMStyle style2)
  41. {
  42.     XIMStyle s,t;
  43.     XIMStyle preedit = XIMPreeditArea | XIMPreeditCallbacks |
  44.             XIMPreeditPosition | XIMPreeditNothing | XIMPreeditNone;
  45.     XIMStyle status = XIMStatusArea | XIMStatusCallbacks |
  46.             XIMStatusNothing | XIMStatusNone;

  47.     if (style1 == 0) return style2;
  48.     if (style2 == 0) return style1;
  49.     if ((style1 & (preedit | status)) == (style2 & (preedit | status)))
  50.         return style1;

  51.     s = style1 & preedit;
  52.     t = style2 & preedit;

  53.     if (s != t) {
  54.         int st = (s | t);
  55.         if (st & XIMPreeditCallbacks)
  56.             return (s == XIMPreeditCallbacks)?style1:style2;
  57.         else if (st & XIMPreeditPosition)
  58.             return (s == XIMPreeditPosition)?style1:style2;
  59.         else if (st & XIMPreeditArea)
  60.             return (s == XIMPreeditArea)?style1:style2;
  61.         else if (st & XIMPreeditNothing)
  62.             return (s == XIMPreeditNothing)?style1:style2;
  63.     }
  64.     else { /* if preedit flags are the same, compare status flags */
  65.         s = style1 & status;
  66.         t = style2 & status;
  67.         int st = (s | t);
  68.         if (st & XIMStatusCallbacks)
  69.             return (s == XIMStatusCallbacks)?style1:style2;
  70.         else if (st & XIMStatusArea)
  71.             return (s == XIMStatusArea)?style1:style2;
  72.         else if (st & XIMStatusNothing)
  73.             return (s == XIMStatusNothing)?style1:style2;
  74.     }
  75.     return 0;
  76. }

  77. XIC createInputContext (Display* dpy, Window win, XIM im)
  78. {
  79.     XFontSet fontset;
  80.     XIC ic;
  81.     XIMStyles *im_supported_styles;
  82.     XIMStyle app_supported_styles;
  83.     XIMStyle style;
  84.     XIMStyle best_style;
  85.     XVaNestedList list;
  86.     int i;

  87.     fontset = createFontSet (dpy, im);
  88.     /* figure out which styles the IM can support */
  89.     XGetIMValues (im, XNQueryInputStyle, &im_supported_styles, NULL);
  90.     /* set flags for the styles our application can support */
  91.     app_supported_styles = XIMPreeditNone | XIMPreeditNothing | XIMPreeditArea;
  92.     app_supported_styles |= XIMStatusNone | XIMStatusNothing | XIMStatusArea;
  93.     /*
  94.      * now look at each of the IM supported styles, and
  95.      * chose the "best" one that we can support.
  96.      */
  97.     best_style = 0;
  98.     for(i=0; i < im_supported_styles->count_styles; i++) {
  99.         style = im_supported_styles->supported_styles[i];
  100.         if ((style & app_supported_styles) == style) /* if we can handle it */
  101.             best_style = ChooseBetterStyle(style, best_style);
  102.     }
  103.     /* if we couldn't support any of them, print an error and exit */
  104.     if (best_style == 0) {
  105.         (void)fprintf(stderr, "application and program do not share a "
  106.                 "commonly supported interaction style.\n");
  107.         exit(1);
  108.     }

  109.     XFree(im_supported_styles);

  110.     /*
  111.      * Now go create an IC using the style we chose.
  112.      * Also set the window and fontset attributes now.
  113.      */
  114.     list = XVaCreateNestedList(0, XNFontSet, fontset, NULL);
  115.     printf ("list = %p\n", list);
  116.     ic = XCreateIC(im, XNInputStyle, best_style,
  117.             XNClientWindow, win,
  118.             XNPreeditAttributes, list,
  119.            // XNPreeditAttributes, NULL,
  120.             XNStatusAttributes, list,
  121.             //XNStatusAttributes, NULL,
  122.             NULL);

  123.     printf ("ic = %p\n", ic);

  124.     XFree(list);
  125.     return ic;
  126. }

  127. #define FALSE 0
  128. #define TRUE 1
  129. int main (int argc, char **argv)
  130. {
  131.     Display *dpy;
  132.     int screen;
  133.     Window w;
  134.     fd_set fdset;

  135.     XIM im;
  136.     XIC ic;

  137.     struct timeval now, timePoint, waitTime;
  138.     int delay;
  139.     XEvent xev;
  140.     int status;
  141.     KeySym keysym;

  142.     if (setlocale(LC_ALL, "zh_CN.utf-8") == NULL) {
  143.         fprintf(stderr, "cannot set locale.\n");
  144.         exit(1);
  145.     }

  146.     if (!XSupportsLocale()) {
  147.         fprintf(stderr, "X does not support locale %s.\n", setlocale(LC_ALL, NULL));
  148.         exit(1);
  149.     }


  150.     dpy = XOpenDisplay (NULL);

  151.     if (XSetLocaleModifiers("") == NULL) {
  152.         fprintf(stderr, "Warning: cannot set locale modifiers.\n");
  153.         exit(1);
  154.     }

  155.     if ((im = XOpenIM(dpy, NULL, NULL, NULL)) == NULL) {
  156.         fprintf(stderr, "Couldn't open input method\n");
  157.         exit(1);
  158.     }

  159.     screen = DefaultScreen (dpy);
  160.     w = XCreateSimpleWindow (dpy, RootWindow (dpy, screen),
  161.             0, 0, W, H, 0,
  162.             BlackPixel (dpy, screen), WhitePixel (dpy, screen));

  163.     XSelectInput (dpy, w,
  164.             PointerMotionMask | ButtonReleaseMask
  165.             | ExposureMask | StructureNotifyMask
  166.             | ButtonPressMask | KeyPressMask);
  167.     XMapWindow (dpy, w);

  168.     ic = createInputContext (dpy, w, im);
  169.     printf ("ic = %p\n", ic);


  170.     FD_ZERO(&fdset);

  171.     XSync(dpy, FALSE);

  172.     gettimeofday (&timePoint, NULL);

  173.     while (1)
  174.     {
  175.         FD_SET (ConnectionNumber(dpy),&fdset);

  176.         gettimeofday (&now, NULL);
  177.         delay = (timePoint.tv_sec-now.tv_sec)*1000 + (timePoint.tv_usec-now.tv_usec)/1000;

  178.         if (delay < 0) {
  179.             delay = 0;
  180.         }

  181.         waitTime.tv_sec = delay / 1000;
  182.         waitTime.tv_usec = (delay % 1000) * 1000;

  183.         status = XPending (dpy);

  184.         if (status == 0)
  185.         {
  186.             status = select (ConnectionNumber(dpy)+1, &fdset, 0, 0, &waitTime);
  187.         }

  188.         if (status == 0) {

  189.             timePoint.tv_usec += 83*1000;
  190.             if (timePoint.tv_usec > 1000000)
  191.             {
  192.                 timePoint.tv_usec -= 1000000;
  193.                 timePoint.tv_sec ++;
  194.             }

  195.             /*TODO: ....*/
  196.         }
  197.         else if (status > 0)
  198.         {
  199.             XNextEvent (dpy, &xev);
  200.             if (xev.xany.window == w)
  201.             {
  202.                 switch (xev.xany.type)
  203.                 {
  204.                     case ButtonPress:
  205.                         printf("    butun press\n");
  206.                         break;

  207.                     case MotionNotify:
  208.                         printf("    mouse move press\n");
  209.                         break;
  210.                     case KeyPress:
  211.                         {
  212.                             printf("    key press\n");
  213.                             int strBufLen = 100;
  214.                             char strBuf [strBufLen];
  215.                             Status statRet;
  216.                             /* Get characters until you encounter a
  217.                              * carriage return; deal with backspaces, etc. */
  218.                             if (XFilterEvent (&xev, w))
  219.                             {
  220.                                 printf ("filtered key event\n");
  221.                                 continue;
  222.                             }

  223.                             XmbLookupString (ic, &(xev.xkey), strBuf, strBufLen,
  224.                                     &keysym, &statRet);
  225.                             printf ("statRet = %d\n", statRet);
  226.                             printf ("char is :%s\n", strBuf);
  227.                                 //XLookupString(&(xev.xkey), NULL, 0, &keysym, 0);
  228.                         }
  229.                         break;

  230.                     default:
  231.                         printf("    other event\n");
  232.                         break;
  233.                 }
  234.             }
  235.         }
  236.     }

  237.     XCloseDisplay (dpy);
  238.     return 0;
  239. }
复制代码


运行后,用ctrl+space可以控制 fcitx的输入提示框显示/隐藏,但只能输入英文。不知道怎么回事,大家帮帮忙!

论坛徽章:
0
2 [报告]
发表于 2009-11-11 16:04 |只看该作者
自问自答以下:
1: 把

  1. if (XFilterEvent (&xev, None))  //去掉 XFilterEvent (&xev, w) 那个if块。
  2. {
  3.        printf ("filtered key event\n");
  4.         continue;
  5. }
复制代码

放到 XNextEvent 后面
2 :把
  1. if (xev.xany.window == w)
复制代码
去掉。

可能是这样: 有些消息是xserver通过与应用程序的连接,发给输入法窗口的。不能用 if (xev.xany.window == w)略过。
XFilterEvent ()也不能指定特定的窗口。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP