1 GTK输入法回顾
在GTK中,每个GtkEntry对象里都有一个指向输入法上下文对象的指针(GtkIMContext *)。在初始化时,这个指针指向一个GtkIMMulticontext对象。
entry->im_context = gtk_im_multicontext_new ();
在gtk_entry_set_visibility函数中,先解引用当前对象,然后根据visible属性,决定要创建的输入法上下文对象类型:
if (visible)
entry->im_context = gtk_im_multicontext_new ();
else
entry->im_context = gtk_im_context_simple_new ();
当visible为真时创建GtkIMMulticontext对象,当visible为假时创建GtkIMContextSimple对象。 GtkIMMulticontext和GtkIMContextSimple都是GtkIMContext的派生类。 GtkIMContextSimple实现了一个简单的表映射,不能动态加载输入法模块。GtkIMMulticontext可以动态加载输入法模块。
在实验一中我们实现了一个派生自GtkIMContext的输入法模块。GtkEntry通过GtkIMMulticontext对象加载了我们的输入法模块。这个输入法模块与输入法服务器通信,使用GtkIMContext接口实现了中文输入。
如果一个编辑框不是GtkEntry对象,显然就不会加载GTK输入法模块,这时基于GTK版本的输入法就无法使用。例如在poky的浏览器中,网页上的编辑框就不是基于GtkEntry的,无法使用GTK版本的输入法。
2 基于XIM的输入法
XIM(X Input Method)是X Window的输入法协议。如果系统的GUI方案是GTK+X,那么基于XIM的输入法就有更好的兼容性。所谓“更好”也是相对的,如果系统的GUI方案是GTK+DirectFB,自然就用不着XIM。
XIM规定了输入法服务器和XIM客户程序的通信规程。只要通信双方都遵守规程,输入法就能正常运作。有一个叫IMdkit的程序库可以简化服务器的开发。基于XIM的输入法通常都会使用这个开发于1994年的程序库。
GTK里面有一个叫imxim的输入法模块。这个模块在GtkIMContext接口的基础上实现XIM客户程序。即它实现了GTK IM和XIM之间的接口转换。只要加载了这个模块,基于GtkEntry的编辑框也可以使用基于XIM的输入法。 imxim也是一个不错的XIM客户程序实例。
2.1 XIM运作机制
和基于gtk的输入法一样,要实现输入法总是要从编辑器里截获按键事件传递给服务器,服务器组字后把输入文本再送回编辑器。在gtk输入法中,gtk im模块是客户,客户与服务器的通信协议是我们自己定义的。在XIM中,客户和服务器的协议是XIM规范规定的。
上图来自谢东翰先生的《Xi18n 程式设计简介》。图中的序号标示出中文输入的典型步骤:
- 在基于X的系统中,键盘和显示都是X Server控制的。X Server检测到按键。
- X Server向客户窗口发送按键事件。
-
客户程序在X事件循环里用XFilterEvent函数过滤XIM服务器需要的事件。例如:
for (;;) {
XNextEvent (dpy, &event);
if (XFilterEvent (&event, None) == True) {
continue;
}
//处理X事件
......
}
- 如果Xlib找到了locale匹配的XIM服务器,而且服务器是激活的,XFilterEvent就会把相关事件交给XIM服务器。如果用了IMdkit库,服务器就会收到XIM_FORWARD_EVENT等XIM呼叫。
- 用户在服务器的输入窗口完成组字,服务器调用IMdkit库的IMCommitString函数将输入文本送回客户程序。服务器只处理自己需要的按键,其它按键可以用IMForwardEvent函数送回客户程序。
2.2 实现XIM输入法
在IMdkit库的帮助下,在服务器侧实现XIM接口还是比较简单的。用IMOpenIM创建XIM服务,登记回调函数(例如MyProtoHandler)。当服务器接收到XIM呼叫时,回调函数会被调用。下面是典型的回调函数:
static Bool MyProtoHandler (XIMS ims, IMProtocol * call_data)
{
switch (call_data->major_code) {
case XIM_OPEN:
return MyOpenHandler ((IMOpenStruct *) call_data);
case XIM_CLOSE:
return MyCloseHandler ((IMOpenStruct *) call_data);
break;
case XIM_FORWARD_EVENT:
ProcessKey (ims, (IMForwardEventStruct *) call_data);
return True;
// IC管理
case XIM_CREATE_IC:
return MyCreateICHandler ((IMChangeICStruct *) call_data);
case XIM_DESTROY_IC:
return MyDestroyICHandler ((IMChangeICStruct *) call_data);
case XIM_GET_IC_VALUES:
return MyGetICValuesHandler ((IMChangeICStruct *) call_data);
case XIM_SET_IC_VALUES:
return MySetICValuesHandler ((IMChangeICStruct *) call_data);
break;
// 焦点
case XIM_SET_IC_FOCUS:
return MySetFocusHandler (ims, (IMChangeFocusStruct *) call_data);
case XIM_UNSET_IC_FOCUS:
return MyUnsetFocusHandler (ims, (IMChangeICStruct *) call_data);;
case XIM_TRIGGER_NOTIFY:
return MyTriggerNotifyHandler((IMTriggerNotifyStruct *) call_data);
default:
DEBUGP("major_code=%d\n", call_data->major_code);
break;
}
return True;
}
应用程序打开和关闭连接时,服务器收到XIM_OPEN和XIM_CLOSE。IC代表输入上下文,即客户程序中的一个编辑框。只有先处理好IC,服务器才能收到XIM_FORWARD_EVENT,即按键事件。 XIM呼叫的处理方法可以参照IMdkit例程或其它输入法程序。
在实验一中,我们已经实现了基于gtk的输入法。现在只要将服务器接口部分用XIM实现就可以了。服务器接口部分使用输入法引擎的以下接口:
-
打开和关闭输入法。
void ime_enable(void);
void ime_disable(void);
在打开/关闭连接和获得/失去焦点时调用。在基于GTK的输入法中,我把输入法开关状态保存在gtk im模块里面。在XIM版本中,我们要在服务器用链表保存每个客户的输入法开关状态。
-
输入按键。
boolean ime_mgr_input(void *user, int c);
第一个参数是可以标识客户的句柄,第二个参数是输入的字符。返回值表示按键是否已经处理。如果按键需要进一步处理,该函数返回FALSE,服务器接口调用IMForwardEvent将按键事件送回客户程序。
-
根据光标位置移动输入窗口。
void ime_move_input(int cursor_x, int cursor_y, int cursor_h);
在XIM_SET_IC_VALUES呼叫中得到光标位置后调用ime_move_input。在gtk版本中,我们可以得到光标高度。在XIM版本,我不知道怎么获取光标高度,只好用固定值。
服务器接口部分要对外提供两个接口:
-
由主程序调用的初始化接口。
int ime_svr_init(void);
返回值小于0表示初始化失败,否则成功。
-
由输入法引擎调用的文本提交接口。
void ime_svr_send(void *user, const char *str);
输入法引擎使用这个接口提交输入文本,第一个参数是由ime_mgr_input传入的客户句柄。 ime_svr_send用IMCommitString将输入文本送到客户程序。
改写服务器接口部分后,XIM版本的输入法就可以运行了。不算IMdkit库,XIM版本服务器接口部分有539行代码。gtk版本的服务器接口部分有265行代码。相对而言,还是XIM版本麻烦一些。
3 webkit上的测试问题
在poky环境测试,GTK的编辑框里当然可以正常使用,例如:
在网页的编辑框里可以输入中文,但是光标跟随无效,例如:
poky里的浏览器是一个叫web2的小程序,只有740行代码。这个小程序使用了webkit引擎。我打开打印后发现,在连接webkit的网页编辑框时,服务器可以收到:
- XIM_OPEN
- XIM_CREATE_IC
- XIM_GET_IC_VALUES
- XIM_TRIGGER_NOTIFY
- XIM_FORWARD_EVENT
- XIM_SET_IC_FOCUS
- XIM_DESTROY_IC
但是收不到
- XIM_SET_IC_VALUES
- XIM_UNSET_IC_FOCUS
光标跟随是根据XIM_SET_IC_VALUES呼叫中的光标位置实现的,收不到XIM_SET_IC_VALUES,自然就不会移动输入窗口。收不到XIM_UNSET_IC_FOCUS对输入法状态切换也会有影响。
前面说过,XIM是客户和服务器之间的协议,必须双方都遵守协议,输入法才能正常工作。从现象看,webkit上的输入法问题是webkit的编辑器控件没有很好地实现XIM协议引起的。
4 结束语
我重新构建poky,恢复了对XIM的支持,然后为XIM接口写了一个服务器接口,实现了基于XIM接口的输入法。 XIM版本的输入法适合于使用X的环境,它要求客户程序遵守XIM协议。从现象看,浏览器引擎webkit的编辑器控件没有很好地实现XIM协议。
分享到:
相关推荐
基于POKY的嵌入式Linux根文件系统的构建.pdf
Yocto:Yocto是这个开源项目的名称,该项目旨在帮助我们自定义...此外Poky还有另外一层意思,使用Poky系统得到的默认参考 Linux 发行版也叫Poky(当然,我们可以对此发行版随意命名)。 很有用的yocto教程,看完包会
meta-debian是用于poky构建系统的一组配方(元数据),它允许使用Debian源代码包交叉构建GNU / Linux映像。 通过启用meta-debian,您可以使用Debian源代码为多种体系结构交叉构建一个小的GNU / Linux映像。 meta-...
poky:git的镜子
POKY WooCommerce版本允许您一键将产品从多个电子商务平台导入/复制到WooCommerce商店,这是一个节省您复制和粘贴时间的好应用程序。 您只需在商店上安装POKY扩展程序,然后在我们的应用程序支持的平台上访问产品...
poky 允许您将多电子商务平台从多次电子商务平台导入/复制产品,只需单击一下,它是保存您的时间复制和粘贴的良好应用程序。 您只需在商店安装Poky应用程序并访问我们的应用支持的平台上的产品页面,并启动少量点击...
poky:Poky Distro元数据
Poky 编译器是Yocto 项目编译出的,常用于编译Uboot和 Linux 出厂源码及快速编译 Qt 应用程序到开发板上运行
qtcreator-plugin-pokysdk QtCreator的Poky插件 该插件将查找安装在/ opt中的Poky SDK并自动将其导入。
龙芯平台yocto工程
The following instructions describe how to install, update, and configure an Ubuntu 12.04 or Ubuntu 14 (64-bit) system. You must be able to log in as root or use sudo to have root permissions during ...
M6708-T系列核心板交叉编译所需工具,包括交叉编译工具poky-glibc-x86_64-meta-toolchain-qt5-cortexa9hf-vfp-neon-toolchain-1.7.sh,触摸屏驱动tslib-1.21-new.tar.gz,文件系统rootfs.tar.gz
该uimage是基于Ubuntu下在arm-linux-gcc 的环境下交叉编译生成的,帮助一些朋友不用经过繁琐的编译
arm32可用的ssh环境,基于fsl imx6 arm-poky-linux-gnueabi-gcc 交叉编译工具编译出来的openssh环境,可以直接拷贝至嵌入式板子/usr根目录下配置后使用。避免自己重新去编译
现场直播 现场扩声通知警察保卫警察保卫住所! 支持语言:Français
meta-chaos:用于poky构建系统的一组食谱,该系统可为家庭自动化项目交叉构建GNULinux映像
Yocto项目快速构建文档,中文,版本 Yocto 3.0,用于Yocto初学者快速构建Poky发行版。
设置 Edison 工具链环境 source /opt/poky-edison/1.5.1/environment-setup-i586-poky-linux 制作调试版 make 制作发布版本 make Release=1 构建测试应用程序 make test Javascript 支持 如果您不熟悉 node-gyp,...