i18nMethod.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537
  1. /* -*- Mode: C; indent-tabs-mode: nil; c-basic-offset: 2; tab-width: 2 -*- */
  2. /******************************************************************
  3. Copyright 1994, 1995 by Sun Microsystems, Inc.
  4. Copyright 1993, 1994 by Hewlett-Packard Company
  5. Permission to use, copy, modify, distribute, and sell this software
  6. and its documentation for any purpose is hereby granted without fee,
  7. provided that the above copyright notice appear in all copies and
  8. that both that copyright notice and this permission notice appear
  9. in supporting documentation, and that the name of Sun Microsystems, Inc.
  10. and Hewlett-Packard not be used in advertising or publicity pertaining to
  11. distribution of the software without specific, written prior permission.
  12. Sun Microsystems, Inc. and Hewlett-Packard make no representations about
  13. the suitability of this software for any purpose. It is provided "as is"
  14. without express or implied warranty.
  15. SUN MICROSYSTEMS INC. AND HEWLETT-PACKARD COMPANY DISCLAIMS ALL
  16. WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED
  17. WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
  18. SUN MICROSYSTEMS, INC. AND HEWLETT-PACKARD COMPANY BE LIABLE FOR ANY
  19. SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
  20. RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
  21. CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
  22. IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  23. Author: Hidetoshi Tajima(tajima@Eng.Sun.COM) Sun Microsystems, Inc.
  24. This version tidied and debugged by Steve Underwood May 1999
  25. ******************************************************************/
  26. #include "i18nMethod.h"
  27. #include "FrameMgr.h"
  28. #include "Xi18n.h"
  29. #include "XimFunc.h"
  30. #include <glib.h>
  31. /* How to generate SUPPORTED_LOCALES
  32. #!/usr/bin/ruby
  33. # -*- coding: utf-8 -*-
  34. require 'set'
  35. set = Set.new
  36. File.open("/usr/share/i18n/SUPPORTED", "r").each do |line|
  37. set << line.split(/_| /)[0]
  38. end
  39. puts set.to_a.join(",")
  40. */
  41. #define SUPPORTED_LOCALES \
  42. "aa,af,agr,ak,am,an,anp,ar,ayc,az,as,ast,be,bem,ber,bg,bhb,bho,bi,bn,bo," \
  43. "br,brx,bs,byn,ca,ce,chr,cmn,crh,cs,csb,cv,cy,da,de,doi,dsb,dv,dz,el,en," \
  44. "eo,es,et,eu,fa,ff,fi,fil,fo,fr,fur,fy,ga,gd,gez,gl,gu,gv,ha,hak,he,hi," \
  45. "hif,hne,hr,hsb,ht,hu,hy,ia,id,ig,ik,is,it,iu,ja,ka,kab,kk,kl,km,kn,ko," \
  46. "kok,ks,ku,kw,ky,lb,lg,li,lij,ln,lo,lt,lv,lzh,mag,mai,mfe,mg,mhr,mi,miq," \
  47. "mjw,mk,ml,mn,mni,mr,ms,mt,my,nan,nb,nds,ne,nhn,niu,nl,nn,nr,nso,oc,om,or," \
  48. "os,pa,pap,pl,ps,pt,quz,raj,ro,ru,rw,sa,sah,sat,sc,sd,se,sgs,shn,shs,si," \
  49. "sid,sk,sl,sm,so,sq,sr,ss,st,sv,sw,szl,ta,tcy,te,tg,th,the,ti,tig,tk,tl," \
  50. "tn,to,tpi,tr,ts,tt,ug,uk,unm,ur,uz,ve,vi,wa,wae,wal,wo,xh,yi,yo,yue,yuw," \
  51. "zh,zu"
  52. extern Xi18nClient *_Xi18nFindClient (NimfXim *, CARD16);
  53. #ifndef XIM_SERVERS
  54. #define XIM_SERVERS "XIM_SERVERS"
  55. #endif
  56. static Atom XIM_Servers = None;
  57. static int SetXi18nSelectionOwner (NimfXim *xim, Window im_window)
  58. {
  59. Window root = RootWindow (xim->display, DefaultScreen (xim->display));
  60. Atom realtype;
  61. int realformat;
  62. unsigned long bytesafter;
  63. long *data=NULL;
  64. unsigned long length;
  65. Atom atom;
  66. int i;
  67. int found;
  68. int forse = False;
  69. if ((atom = XInternAtom (xim->display, "@server=nimf", False)) == 0)
  70. return False;
  71. xim->address.selection = atom;
  72. if (XIM_Servers == None)
  73. XIM_Servers = XInternAtom (xim->display, XIM_SERVERS, False);
  74. /*endif*/
  75. XGetWindowProperty (xim->display,
  76. root,
  77. XIM_Servers,
  78. 0L,
  79. 1000000L,
  80. False,
  81. XA_ATOM,
  82. &realtype,
  83. &realformat,
  84. &length,
  85. &bytesafter,
  86. (unsigned char **) (&data));
  87. if (realtype != None && (realtype != XA_ATOM || realformat != 32)) {
  88. if (data != NULL)
  89. XFree ((char *) data);
  90. return False;
  91. }
  92. found = False;
  93. for (i = 0; i < length; i++) {
  94. if (data[i] == atom) {
  95. Window owner;
  96. found = True;
  97. if ((owner = XGetSelectionOwner (xim->display, atom)) != im_window) {
  98. if (owner == None || forse == True)
  99. XSetSelectionOwner (xim->display, atom, im_window, CurrentTime);
  100. else
  101. return False;
  102. }
  103. break;
  104. }
  105. }
  106. if (found == False) {
  107. XSetSelectionOwner (xim->display, atom, im_window, CurrentTime);
  108. XChangeProperty (xim->display,
  109. root,
  110. XIM_Servers,
  111. XA_ATOM,
  112. 32,
  113. PropModePrepend,
  114. (unsigned char *) &atom,
  115. 1);
  116. }
  117. else {
  118. /*
  119. * We always need to generate the PropertyNotify to the Root Window
  120. */
  121. XChangeProperty (xim->display,
  122. root,
  123. XIM_Servers,
  124. XA_ATOM,
  125. 32,
  126. PropModePrepend,
  127. (unsigned char *) data,
  128. 0);
  129. }
  130. if (data != NULL)
  131. XFree ((char *) data);
  132. /* Intern "LOCALES" and "TRANSOPORT" Target Atoms */
  133. xim->address.Localename = XInternAtom (xim->display, LOCALES, False);
  134. xim->address.Transportname = XInternAtom (xim->display, TRANSPORT, False);
  135. return (XGetSelectionOwner (xim->display, atom) == im_window);
  136. }
  137. static int DeleteXi18nAtom (NimfXim *xim)
  138. {
  139. Window root = RootWindow (xim->display, DefaultScreen (xim->display));
  140. Atom realtype;
  141. int realformat;
  142. unsigned long bytesafter;
  143. long *data=NULL;
  144. unsigned long length;
  145. Atom atom;
  146. int i, ret;
  147. int found;
  148. if ((atom = XInternAtom (xim->display, "@server=nimf", False)) == 0)
  149. return False;
  150. xim->address.selection = atom;
  151. if (XIM_Servers == None)
  152. XIM_Servers = XInternAtom (xim->display, XIM_SERVERS, False);
  153. XGetWindowProperty (xim->display,
  154. root,
  155. XIM_Servers,
  156. 0L,
  157. 1000000L,
  158. False,
  159. XA_ATOM,
  160. &realtype,
  161. &realformat,
  162. &length,
  163. &bytesafter,
  164. (unsigned char **) (&data));
  165. if (realtype != XA_ATOM || realformat != 32) {
  166. if (data != NULL)
  167. XFree ((char *) data);
  168. return False;
  169. }
  170. found = False;
  171. for (i = 0; i < length; i++) {
  172. if (data[i] == atom) {
  173. found = True;
  174. break;
  175. }
  176. }
  177. if (found == True) {
  178. for (i=i+1; i<length; i++)
  179. data[i-1] = data[i];
  180. XChangeProperty (xim->display,
  181. root,
  182. XIM_Servers,
  183. XA_ATOM,
  184. 32,
  185. PropModeReplace,
  186. (unsigned char *)data,
  187. length-1);
  188. ret = True;
  189. }
  190. else {
  191. XChangeProperty (xim->display,
  192. root,
  193. XIM_Servers,
  194. XA_ATOM,
  195. 32,
  196. PropModePrepend,
  197. (unsigned char *)data,
  198. 0);
  199. ret = False;
  200. }
  201. if (data != NULL)
  202. XFree ((char *) data);
  203. return ret;
  204. }
  205. static void
  206. ReturnSelectionNotify (NimfXim *xim, XSelectionRequestEvent *ev)
  207. {
  208. XEvent event;
  209. const char *data = NULL;
  210. event.type = SelectionNotify;
  211. event.xselection.requestor = ev->requestor;
  212. event.xselection.selection = ev->selection;
  213. event.xselection.target = ev->target;
  214. event.xselection.time = ev->time;
  215. event.xselection.property = ev->property;
  216. if (ev->target == xim->address.Localename)
  217. data = "@locale=C" SUPPORTED_LOCALES;
  218. else if (ev->target == xim->address.Transportname)
  219. data = "@transport=X/";
  220. XChangeProperty (xim->display,
  221. event.xselection.requestor,
  222. ev->target,
  223. ev->target,
  224. 8,
  225. PropModeReplace,
  226. (unsigned char *) data,
  227. strlen (data));
  228. XSendEvent (xim->display, event.xselection.requestor, False, NoEventMask, &event);
  229. XFlush (xim->display);
  230. }
  231. Bool
  232. WaitXSelectionRequest (NimfXim *xim,
  233. XEvent *ev)
  234. {
  235. if (((XSelectionRequestEvent *) ev)->selection == xim->address.selection)
  236. {
  237. ReturnSelectionNotify (xim, (XSelectionRequestEvent *) ev);
  238. return True;
  239. }
  240. return False;
  241. }
  242. Status
  243. xi18n_openIM (NimfXim *xim, Window im_window)
  244. {
  245. if (!SetXi18nSelectionOwner (xim, im_window))
  246. return False;
  247. xim->_protocol = XInternAtom (xim->display, "_XIM_PROTOCOL", False);
  248. xim->_xconnect = XInternAtom (xim->display, "_XIM_XCONNECT", False);
  249. XFlush (xim->display);
  250. return True;
  251. }
  252. Status xi18n_closeIM (NimfXim *xim)
  253. {
  254. DeleteXi18nAtom (xim);
  255. return True;
  256. }
  257. static void EventToWireEvent (XEvent *ev, xEvent *event,
  258. CARD16 *serial, Bool byte_swap)
  259. {
  260. FrameMgr fm;
  261. extern XimFrameRec wire_keyevent_fr[];
  262. extern XimFrameRec short_fr[];
  263. BYTE b;
  264. CARD16 c16;
  265. CARD32 c32;
  266. *serial = (CARD16)(ev->xany.serial >> 16);
  267. switch (ev->type) {
  268. case KeyPress:
  269. case KeyRelease:
  270. {
  271. XKeyEvent *kev = (XKeyEvent*)ev;
  272. /* create FrameMgr */
  273. fm = FrameMgrInit(wire_keyevent_fr, (char *)(&(event->u)), byte_swap);
  274. /* set values */
  275. b = (BYTE)kev->type; FrameMgrPutToken(fm, b);
  276. b = (BYTE)kev->keycode; FrameMgrPutToken(fm, b);
  277. c16 = (CARD16)(kev->serial & (unsigned long)0xffff);
  278. FrameMgrPutToken(fm, c16);
  279. c32 = (CARD32)kev->time; FrameMgrPutToken(fm, c32);
  280. c32 = (CARD32)kev->root; FrameMgrPutToken(fm, c32);
  281. c32 = (CARD32)kev->window; FrameMgrPutToken(fm, c32);
  282. c32 = (CARD32)kev->subwindow; FrameMgrPutToken(fm, c32);
  283. c16 = (CARD16)kev->x_root; FrameMgrPutToken(fm, c16);
  284. c16 = (CARD16)kev->y_root; FrameMgrPutToken(fm, c16);
  285. c16 = (CARD16)kev->x; FrameMgrPutToken(fm, c16);
  286. c16 = (CARD16)kev->y; FrameMgrPutToken(fm, c16);
  287. c16 = (CARD16)kev->state; FrameMgrPutToken(fm, c16);
  288. b = (BYTE)kev->same_screen; FrameMgrPutToken(fm, b);
  289. }
  290. break;
  291. default:
  292. /* create FrameMgr */
  293. fm = FrameMgrInit(short_fr, (char *)(&(event->u.u.sequenceNumber)),
  294. byte_swap);
  295. c16 = (CARD16)(ev->xany.serial & (unsigned long)0xffff);
  296. FrameMgrPutToken(fm, c16);
  297. break;
  298. }
  299. /* free FrameMgr */
  300. FrameMgrFree(fm);
  301. }
  302. Status xi18n_forwardEvent (NimfXim *xim, XPointer xp)
  303. {
  304. IMForwardEventStruct *call_data = (IMForwardEventStruct *)xp;
  305. FrameMgr fm;
  306. extern XimFrameRec forward_event_fr[];
  307. register int total_size;
  308. unsigned char *reply = NULL;
  309. unsigned char *replyp;
  310. CARD16 serial;
  311. int event_size;
  312. Xi18nClient *client;
  313. client = (Xi18nClient *) _Xi18nFindClient (xim, call_data->connect_id);
  314. /* create FrameMgr */
  315. fm = FrameMgrInit (forward_event_fr,
  316. NULL,
  317. _Xi18nNeedSwap (xim, call_data->connect_id));
  318. total_size = FrameMgrGetTotalSize (fm);
  319. event_size = sizeof (xEvent);
  320. reply = (unsigned char *) malloc (total_size + event_size);
  321. if (!reply)
  322. {
  323. _Xi18nSendMessage (xim,
  324. call_data->connect_id,
  325. XIM_ERROR,
  326. 0,
  327. 0,
  328. 0);
  329. return False;
  330. }
  331. /*endif*/
  332. memset (reply, 0, total_size + event_size);
  333. FrameMgrSetBuffer (fm, reply);
  334. replyp = reply;
  335. call_data->sync_bit = 1; /* always sync */
  336. client->sync = True;
  337. FrameMgrPutToken (fm, call_data->connect_id);
  338. FrameMgrPutToken (fm, call_data->icid);
  339. FrameMgrPutToken (fm, call_data->sync_bit);
  340. replyp += total_size;
  341. EventToWireEvent (&(call_data->event),
  342. (xEvent *) replyp,
  343. &serial,
  344. _Xi18nNeedSwap (xim, call_data->connect_id));
  345. FrameMgrPutToken (fm, serial);
  346. _Xi18nSendMessage (xim,
  347. call_data->connect_id,
  348. XIM_FORWARD_EVENT,
  349. 0,
  350. reply,
  351. total_size + event_size);
  352. XFree (reply);
  353. FrameMgrFree (fm);
  354. return True;
  355. }
  356. Status xi18n_commit (NimfXim *xim, XPointer xp)
  357. {
  358. IMCommitStruct *call_data = (IMCommitStruct *)xp;
  359. FrameMgr fm;
  360. extern XimFrameRec commit_chars_fr[];
  361. extern XimFrameRec commit_both_fr[];
  362. register int total_size;
  363. unsigned char *reply = NULL;
  364. CARD16 str_length;
  365. call_data->flag |= XimSYNCHRONUS; /* always sync */
  366. if (!(call_data->flag & XimLookupKeySym)
  367. &&
  368. (call_data->flag & XimLookupChars))
  369. {
  370. fm = FrameMgrInit (commit_chars_fr,
  371. NULL,
  372. _Xi18nNeedSwap (xim, call_data->connect_id));
  373. /* set length of STRING8 */
  374. str_length = strlen (call_data->commit_string);
  375. FrameMgrSetSize (fm, str_length);
  376. total_size = FrameMgrGetTotalSize (fm);
  377. reply = (unsigned char *) malloc (total_size);
  378. if (!reply)
  379. {
  380. _Xi18nSendMessage (xim,
  381. call_data->connect_id,
  382. XIM_ERROR,
  383. 0,
  384. 0,
  385. 0);
  386. return False;
  387. }
  388. /*endif*/
  389. memset (reply, 0, total_size);
  390. FrameMgrSetBuffer (fm, reply);
  391. str_length = FrameMgrGetSize (fm);
  392. FrameMgrPutToken (fm, call_data->connect_id);
  393. FrameMgrPutToken (fm, call_data->icid);
  394. FrameMgrPutToken (fm, call_data->flag);
  395. FrameMgrPutToken (fm, str_length);
  396. FrameMgrPutToken (fm, call_data->commit_string);
  397. }
  398. else
  399. {
  400. fm = FrameMgrInit (commit_both_fr,
  401. NULL,
  402. _Xi18nNeedSwap (xim, call_data->connect_id));
  403. /* set length of STRING8 */
  404. str_length = strlen (call_data->commit_string);
  405. if (str_length > 0)
  406. FrameMgrSetSize (fm, str_length);
  407. /*endif*/
  408. total_size = FrameMgrGetTotalSize (fm);
  409. reply = (unsigned char *) malloc (total_size);
  410. if (!reply)
  411. {
  412. _Xi18nSendMessage (xim,
  413. call_data->connect_id,
  414. XIM_ERROR,
  415. 0,
  416. 0,
  417. 0);
  418. return False;
  419. }
  420. /*endif*/
  421. FrameMgrSetBuffer (fm, reply);
  422. FrameMgrPutToken (fm, call_data->connect_id);
  423. FrameMgrPutToken (fm, call_data->icid);
  424. FrameMgrPutToken (fm, call_data->flag);
  425. FrameMgrPutToken (fm, call_data->keysym);
  426. if (str_length > 0)
  427. {
  428. str_length = FrameMgrGetSize (fm);
  429. FrameMgrPutToken (fm, str_length);
  430. FrameMgrPutToken (fm, call_data->commit_string);
  431. }
  432. /*endif*/
  433. }
  434. /*endif*/
  435. _Xi18nSendMessage (xim,
  436. call_data->connect_id,
  437. XIM_COMMIT,
  438. 0,
  439. reply,
  440. total_size);
  441. FrameMgrFree (fm);
  442. XFree (reply);
  443. return True;
  444. }
  445. int nimf_xim_call_callback (NimfXim *xim, XPointer xp)
  446. {
  447. IMProtocol *call_data = (IMProtocol *)xp;
  448. switch (call_data->major_code)
  449. {
  450. case XIM_GEOMETRY:
  451. return _Xi18nGeometryCallback (xim, call_data);
  452. case XIM_PREEDIT_START:
  453. return _Xi18nPreeditStartCallback (xim, call_data);
  454. case XIM_PREEDIT_DRAW:
  455. return _Xi18nPreeditDrawCallback (xim, call_data);
  456. case XIM_PREEDIT_CARET:
  457. return _Xi18nPreeditCaretCallback (xim, call_data);
  458. case XIM_PREEDIT_DONE:
  459. return _Xi18nPreeditDoneCallback (xim, call_data);
  460. case XIM_STATUS_START:
  461. return _Xi18nStatusStartCallback (xim, call_data);
  462. case XIM_STATUS_DRAW:
  463. return _Xi18nStatusDrawCallback (xim, call_data);
  464. case XIM_STATUS_DONE:
  465. return _Xi18nStatusDoneCallback (xim, call_data);
  466. case XIM_STR_CONVERSION:
  467. return _Xi18nStringConversionCallback (xim, call_data);
  468. }
  469. /*endswitch*/
  470. return False;
  471. }