XDefaultIMIF.c revision 2d67cb4f
11ab64890Smrg/* 21ab64890SmrgCopyright 1985, 1986, 1987, 1991, 1998 The Open Group 31ab64890Smrg 41ab64890SmrgPermission is hereby granted, free of charge, to any person obtaining a 51ab64890Smrgcopy of this software and associated documentation files (the 61ab64890Smrg"Software"), to deal in the Software without restriction, including 71ab64890Smrgwithout limitation the rights to use, copy, modify, merge, publish, 81ab64890Smrgdistribute, sublicense, and/or sell copies of the Software, and to 91ab64890Smrgpermit persons to whom the Software is furnished to do so, subject to 101ab64890Smrgthe following conditions: The above copyright notice and this 111ab64890Smrgpermission notice shall be included in all copies or substantial 121ab64890Smrgportions of the Software. 131ab64890Smrg 141ab64890Smrg 15b4ee4795SmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16b4ee4795SmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17b4ee4795SmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18b4ee4795SmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN 19b4ee4795SmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 20b4ee4795SmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE 21b4ee4795SmrgEVEN IF ADVISED IN ADVANCE OF THE POSSIBILITY OF SUCH DAMAGES. 221ab64890Smrg 231ab64890Smrg 24b4ee4795SmrgExcept as contained in this notice, the name of The Open Group shall not be 25b4ee4795Smrgused in advertising or otherwise to promote the sale, use or other dealings 26b4ee4795Smrgin this Software without prior written authorization from The Open Group. 271ab64890Smrg 281ab64890Smrg 291ab64890SmrgX Window System is a trademark of The Open Group 301ab64890Smrg 311ab64890SmrgOSF/1, OSF/Motif and Motif are registered trademarks, and OSF, the OSF 321ab64890Smrglogo, LBX, X Window System, and Xinerama are trademarks of the Open 331ab64890SmrgGroup. All other trademarks and registered trademarks mentioned herein 341ab64890Smrgare the property of their respective owners. No right, title or 351ab64890Smrginterest in or to any trademark, service mark, logo or trade name of 361ab64890SmrgSun Microsystems, Inc. or its licensors is granted. 371ab64890Smrg 381ab64890Smrg*/ 39b4ee4795Smrg/* 40e9fcaa8aSmrg * Copyright 2000 Oracle and/or its affiliates. All rights reserved. 41b4ee4795Smrg * 42b4ee4795Smrg * Permission is hereby granted, free of charge, to any person obtaining a 43b4ee4795Smrg * copy of this software and associated documentation files (the "Software"), 44b4ee4795Smrg * to deal in the Software without restriction, including without limitation 45b4ee4795Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 46b4ee4795Smrg * and/or sell copies of the Software, and to permit persons to whom the 47b4ee4795Smrg * Software is furnished to do so, subject to the following conditions: 48b4ee4795Smrg * 49b4ee4795Smrg * The above copyright notice and this permission notice (including the next 50b4ee4795Smrg * paragraph) shall be included in all copies or substantial portions of the 51b4ee4795Smrg * Software. 52b4ee4795Smrg * 53b4ee4795Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 54b4ee4795Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 55b4ee4795Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 56b4ee4795Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 57b4ee4795Smrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 58b4ee4795Smrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 59b4ee4795Smrg * DEALINGS IN THE SOFTWARE. 60b4ee4795Smrg */ 61b4ee4795Smrg 621ab64890Smrg 631ab64890Smrg#ifdef HAVE_CONFIG_H 641ab64890Smrg#include <config.h> 651ab64890Smrg#endif 661ab64890Smrg#include <stdio.h> 671ab64890Smrg#include "Xlibint.h" 681ab64890Smrg#include "Xlcint.h" 691ab64890Smrg#include "XlcGeneric.h" 701ab64890Smrg 711ab64890Smrg#ifndef MAXINT 721ab64890Smrg#define MAXINT (~((unsigned int)1 << (8 * sizeof(int)) - 1)) 731ab64890Smrg#endif /* !MAXINT */ 741ab64890Smrg 751ab64890Smrgtypedef struct _StaticXIM *StaticXIM; 761ab64890Smrg 771ab64890Smrgtypedef struct _XIMStaticXIMRec { 781ab64890Smrg /* for CT => MB,WC converter */ 791ab64890Smrg XlcConv ctom_conv; 801ab64890Smrg XlcConv ctow_conv; 811ab64890Smrg} XIMStaticXIMRec; 821ab64890Smrg 831ab64890Smrgtypedef enum { 841ab64890Smrg CREATE_IC = 1, 851ab64890Smrg SET_ICVAL = 2, 861ab64890Smrg GET_ICVAL = 3 871ab64890Smrg} XICOp_t; 881ab64890Smrg 891ab64890Smrgtypedef struct _StaticXIM { 901ab64890Smrg XIMMethods methods; 911ab64890Smrg XIMCoreRec core; 921ab64890Smrg XIMStaticXIMRec *private; 931ab64890Smrg} StaticXIMRec; 941ab64890Smrg 951ab64890Smrgstatic Status _CloseIM( 961ab64890Smrg XIM 971ab64890Smrg); 981ab64890Smrg 991ab64890Smrgstatic char *_SetIMValues( 1001ab64890Smrg XIM, XIMArg * 1011ab64890Smrg); 1021ab64890Smrg 1031ab64890Smrgstatic char *_GetIMValues( 1041ab64890Smrg XIM, XIMArg* 1051ab64890Smrg); 1061ab64890Smrg 1071ab64890Smrgstatic XIC _CreateIC( 1081ab64890Smrg XIM, XIMArg* 1091ab64890Smrg); 1101ab64890Smrg 1111ab64890Smrgstatic _Xconst XIMMethodsRec local_im_methods = { 1121ab64890Smrg _CloseIM, /* close */ 1131ab64890Smrg _SetIMValues, /* set_values */ 1141ab64890Smrg _GetIMValues, /* get_values */ 1151ab64890Smrg _CreateIC, /* create_ic */ 1161ab64890Smrg NULL, /* ctstombs */ 1171ab64890Smrg NULL /* ctstowcs */ 1181ab64890Smrg}; 1191ab64890Smrg 1201ab64890Smrgstatic void _DestroyIC( 1211ab64890Smrg XIC 1221ab64890Smrg); 1231ab64890Smrgstatic void _SetFocus( 1241ab64890Smrg XIC 1251ab64890Smrg); 1261ab64890Smrgstatic void _UnsetFocus( 1271ab64890Smrg XIC 1281ab64890Smrg); 1291ab64890Smrgstatic char* _SetICValues( 1301ab64890Smrg XIC, XIMArg * 1311ab64890Smrg); 1321ab64890Smrgstatic char* _GetICValues( 1331ab64890Smrg XIC, XIMArg * 1341ab64890Smrg); 1351ab64890Smrgstatic char *_MbReset( 1361ab64890Smrg XIC 1371ab64890Smrg); 1381ab64890Smrgstatic wchar_t *_WcReset( 1391ab64890Smrg XIC 1401ab64890Smrg); 1411ab64890Smrgstatic int _MbLookupString( 1421ab64890Smrg XIC, XKeyEvent *, char *, int, KeySym *, Status * 1431ab64890Smrg); 1441ab64890Smrgstatic int _WcLookupString( 1451ab64890Smrg XIC, XKeyEvent *, wchar_t *, int, KeySym *, Status * 1461ab64890Smrg); 1471ab64890Smrg 1481ab64890Smrgstatic _Xconst XICMethodsRec local_ic_methods = { 1491ab64890Smrg _DestroyIC, /* destroy */ 1501ab64890Smrg _SetFocus, /* set_focus */ 1511ab64890Smrg _UnsetFocus, /* unset_focus */ 1521ab64890Smrg _SetICValues, /* set_values */ 1531ab64890Smrg _GetICValues, /* get_values */ 1541ab64890Smrg _MbReset, /* mb_reset */ 1551ab64890Smrg _WcReset, /* wc_reset */ 1561ab64890Smrg NULL, /* utf8_reset */ /* ??? */ 1571ab64890Smrg _MbLookupString, /* mb_lookup_string */ 1581ab64890Smrg _WcLookupString, /* wc_lookup_string */ 1591ab64890Smrg NULL /* utf8_lookup_string */ /* ??? */ 1601ab64890Smrg}; 1611ab64890Smrg 1621ab64890SmrgXIM 1631ab64890Smrg_XDefaultOpenIM( 1641ab64890Smrg XLCd lcd, 1651ab64890Smrg Display *dpy, 1661ab64890Smrg XrmDatabase rdb, 1671ab64890Smrg char *res_name, 1681ab64890Smrg char *res_class) 1691ab64890Smrg{ 1701ab64890Smrg StaticXIM im; 1711ab64890Smrg int i; 1721ab64890Smrg char *mod; 1731ab64890Smrg char buf[BUFSIZ]; 1741ab64890Smrg 1752d67cb4fSmrg if ((im = Xcalloc(1, sizeof(StaticXIMRec))) == NULL) 1762d67cb4fSmrg return NULL; 17761b2299dSmrg 1782d67cb4fSmrg if ((im->private = Xcalloc(1, sizeof(XIMStaticXIMRec))) == NULL) 1792d67cb4fSmrg goto Error; 1801ab64890Smrg 1812d67cb4fSmrg if ((im->private->ctom_conv = _XlcOpenConverter(lcd, XlcNCompoundText, 1822d67cb4fSmrg lcd, XlcNMultiByte)) 1832d67cb4fSmrg == NULL) 1842d67cb4fSmrg goto Error; 1852d67cb4fSmrg 1862d67cb4fSmrg if ((im->private->ctow_conv = _XlcOpenConverter(lcd, XlcNCompoundText, 1872d67cb4fSmrg lcd, XlcNWideChar)) 1882d67cb4fSmrg == NULL) 1892d67cb4fSmrg goto Error; 1901ab64890Smrg 1911ab64890Smrg buf[0] = '\0'; 1921ab64890Smrg i = 0; 1931ab64890Smrg if ((lcd->core->modifiers) && (*lcd->core->modifiers)) { 1941ab64890Smrg#define MODIFIER "@im=" 1951ab64890Smrg mod = strstr(lcd->core->modifiers, MODIFIER); 1961ab64890Smrg if (mod) { 1971ab64890Smrg mod += strlen(MODIFIER); 1981ab64890Smrg while (*mod && *mod != '@' && i < BUFSIZ - 1) { 1991ab64890Smrg buf[i++] = *mod++; 2001ab64890Smrg } 2011ab64890Smrg buf[i] = '\0'; 2021ab64890Smrg } 2031ab64890Smrg } 2041ab64890Smrg#undef MODIFIER 2052d67cb4fSmrg if ((im->core.im_name = strdup(buf)) == NULL) 2062d67cb4fSmrg goto Error; 2071ab64890Smrg 2081ab64890Smrg im->methods = (XIMMethods)&local_im_methods; 2091ab64890Smrg im->core.lcd = lcd; 2101ab64890Smrg im->core.ic_chain = (XIC)NULL; 2111ab64890Smrg im->core.display = dpy; 2121ab64890Smrg im->core.rdb = rdb; 2131ab64890Smrg im->core.res_name = NULL; 2141ab64890Smrg im->core.res_class = NULL; 2151ab64890Smrg 2161ab64890Smrg if ((res_name != NULL) && (*res_name != '\0')){ 2176cc2b21fSmrg im->core.res_name = strdup(res_name); 2181ab64890Smrg } 2191ab64890Smrg if ((res_class != NULL) && (*res_class != '\0')){ 2206cc2b21fSmrg im->core.res_class = strdup(res_class); 2211ab64890Smrg } 2221ab64890Smrg 2231ab64890Smrg return (XIM)im; 2242d67cb4fSmrg 2252d67cb4fSmrg Error: 2262d67cb4fSmrg _CloseIM((XIM)im); 2271ab64890Smrg Xfree(im); 2281ab64890Smrg return(NULL); 2291ab64890Smrg} 2301ab64890Smrg 2311ab64890Smrgstatic Status 2321ab64890Smrg_CloseIM(XIM xim) 2331ab64890Smrg{ 2341ab64890Smrg StaticXIM im = (StaticXIM)xim; 2352d67cb4fSmrg 2362d67cb4fSmrg if (im->private->ctom_conv != NULL) 2372d67cb4fSmrg _XlcCloseConverter(im->private->ctom_conv); 2382d67cb4fSmrg if (im->private->ctow_conv != NULL) 2392d67cb4fSmrg _XlcCloseConverter(im->private->ctow_conv); 2401ab64890Smrg XFree(im->private); 2411ab64890Smrg XFree(im->core.im_name); 2422d67cb4fSmrg XFree(im->core.res_name); 2432d67cb4fSmrg XFree(im->core.res_class); 2442d67cb4fSmrg return 1; 2451ab64890Smrg} 2461ab64890Smrg 2471ab64890Smrgstatic char * 2481ab64890Smrg_SetIMValues( 2491ab64890Smrg XIM xim, 2501ab64890Smrg XIMArg *arg) 2511ab64890Smrg{ 2521ab64890Smrg return(arg->name); /* evil */ 2531ab64890Smrg} 2541ab64890Smrg 2551ab64890Smrgstatic char * 2561ab64890Smrg_GetIMValues( 2571ab64890Smrg XIM xim, 2581ab64890Smrg XIMArg *values) 2591ab64890Smrg{ 2601ab64890Smrg XIMArg *p; 2611ab64890Smrg XIMStyles *styles; 2621ab64890Smrg 2631ab64890Smrg for (p = values; p->name != NULL; p++) { 2641ab64890Smrg if (strcmp(p->name, XNQueryInputStyle) == 0) { 265818534a1Smrg styles = Xmalloc(sizeof(XIMStyles)); 2661ab64890Smrg *(XIMStyles **)p->value = styles; 2671ab64890Smrg styles->count_styles = 1; 2681ab64890Smrg styles->supported_styles = 269818534a1Smrg Xmalloc(styles->count_styles * sizeof(XIMStyle)); 2701ab64890Smrg styles->supported_styles[0] = (XIMPreeditNone | XIMStatusNone); 2711ab64890Smrg } else { 2721ab64890Smrg break; 2731ab64890Smrg } 2741ab64890Smrg } 2751ab64890Smrg return (p->name); 2761ab64890Smrg} 2771ab64890Smrg 2781ab64890Smrgstatic char* 2791ab64890Smrg_SetICValueData(XIC ic, XIMArg *values, XICOp_t mode) 2801ab64890Smrg{ 2811ab64890Smrg XIMArg *p; 2821ab64890Smrg char *return_name = NULL; 2831ab64890Smrg 2841ab64890Smrg for (p = values; p != NULL && p->name != NULL; p++) { 2851ab64890Smrg if(strcmp(p->name, XNInputStyle) == 0) { 2861ab64890Smrg if (mode == CREATE_IC) 2871ab64890Smrg ic->core.input_style = (XIMStyle)p->value; 2881ab64890Smrg } else if (strcmp(p->name, XNClientWindow) == 0) { 2891ab64890Smrg ic->core.client_window = (Window)p->value ; 2901ab64890Smrg } else if (strcmp(p->name, XNFocusWindow) == 0) { 2911ab64890Smrg ic->core.focus_window = (Window)p->value ; 2921ab64890Smrg } else if (strcmp(p->name, XNPreeditAttributes) == 0 2931ab64890Smrg || strcmp(p->name, XNStatusAttributes) == 0) { 2941ab64890Smrg return_name = _SetICValueData(ic, (XIMArg*)p->value, mode); 2951ab64890Smrg if (return_name) break; 2961ab64890Smrg } else { 2971ab64890Smrg return_name = p->name; 2981ab64890Smrg break; 2991ab64890Smrg } 3001ab64890Smrg } 3011ab64890Smrg return(return_name); 3021ab64890Smrg} 3031ab64890Smrg 3041ab64890Smrgstatic char* 3051ab64890Smrg_GetICValueData(XIC ic, XIMArg *values, XICOp_t mode) 3061ab64890Smrg{ 3071ab64890Smrg XIMArg *p; 3081ab64890Smrg char *return_name = NULL; 3091ab64890Smrg 3101ab64890Smrg for (p = values; p->name != NULL; p++) { 3111ab64890Smrg if(strcmp(p->name, XNInputStyle) == 0) { 3121ab64890Smrg *((XIMStyle *)(p->value)) = ic->core.input_style; 3131ab64890Smrg } else if (strcmp(p->name, XNClientWindow) == 0) { 3141ab64890Smrg *((Window *)(p->value)) = ic->core.client_window; 3151ab64890Smrg } else if (strcmp(p->name, XNFocusWindow) == 0) { 3161ab64890Smrg *((Window *)(p->value)) = ic->core.focus_window; 3171ab64890Smrg } else if (strcmp(p->name, XNFilterEvents) == 0) { 3181ab64890Smrg *((unsigned long *)(p->value))= ic->core.filter_events; 3191ab64890Smrg } else if (strcmp(p->name, XNPreeditAttributes) == 0 3201ab64890Smrg || strcmp(p->name, XNStatusAttributes) == 0) { 3211ab64890Smrg return_name = _GetICValueData(ic, (XIMArg*)p->value, mode); 3221ab64890Smrg if (return_name) break; 3231ab64890Smrg } else { 3241ab64890Smrg return_name = p->name; 3251ab64890Smrg break; 3261ab64890Smrg } 3271ab64890Smrg } 3281ab64890Smrg return(return_name); 3291ab64890Smrg} 3301ab64890Smrg 3311ab64890Smrgstatic XIC 3321ab64890Smrg_CreateIC(XIM im, XIMArg *arg) 3331ab64890Smrg{ 3341ab64890Smrg XIC ic; 3351ab64890Smrg 3362d67cb4fSmrg if ((ic = Xcalloc(1, sizeof(XICRec))) == (XIC)NULL) { 3371ab64890Smrg return ((XIC)NULL); 3381ab64890Smrg } 3391ab64890Smrg 3401ab64890Smrg ic->methods = (XICMethods)&local_ic_methods; 3411ab64890Smrg ic->core.im = im; 3421ab64890Smrg ic->core.filter_events = KeyPressMask; 3431ab64890Smrg 3441ab64890Smrg if (_SetICValueData(ic, arg, CREATE_IC) != NULL) 3451ab64890Smrg goto err_return; 3461ab64890Smrg if (!(ic->core.input_style)) 3471ab64890Smrg goto err_return; 3481ab64890Smrg 3491ab64890Smrg return (XIC)ic; 3501ab64890Smrgerr_return: 3511ab64890Smrg XFree(ic); 3521ab64890Smrg return ((XIC)NULL); 3531ab64890Smrg} 3541ab64890Smrg 3551ab64890Smrgstatic void 3561ab64890Smrg_DestroyIC(XIC ic) 3571ab64890Smrg{ 3581ab64890Smrg/*BugId4255571. This Xfree() should be removed because XDestroyIC() still need ic after invoking _DestroyIC() and there is a XFree(ic) at the end of XDestroyIC() already. 3591ab64890Smrg if(ic) 3601ab64890Smrg XFree(ic); */ 3611ab64890Smrg} 3621ab64890Smrg 3631ab64890Smrgstatic void 3641ab64890Smrg_SetFocus(XIC ic) 3651ab64890Smrg{ 3661ab64890Smrg} 3671ab64890Smrg 3681ab64890Smrgstatic void 3691ab64890Smrg_UnsetFocus(XIC ic) 3701ab64890Smrg{ 3711ab64890Smrg} 3721ab64890Smrg 3731ab64890Smrgstatic char* 3741ab64890Smrg_SetICValues(XIC ic, XIMArg *args) 3751ab64890Smrg{ 3761ab64890Smrg char *ret = NULL; 3771ab64890Smrg if (!ic) { 3781ab64890Smrg return (args->name); 3791ab64890Smrg } 3801ab64890Smrg ret = _SetICValueData(ic, args, SET_ICVAL); 3811ab64890Smrg return(ret); 3821ab64890Smrg} 3831ab64890Smrg 3841ab64890Smrgstatic char* 3851ab64890Smrg_GetICValues(XIC ic, XIMArg *args) 3861ab64890Smrg{ 3871ab64890Smrg char *ret = NULL; 3881ab64890Smrg if (!ic) { 3891ab64890Smrg return (args->name); 3901ab64890Smrg } 3911ab64890Smrg ret = _GetICValueData(ic, args, GET_ICVAL); 3921ab64890Smrg return(ret); 3931ab64890Smrg} 3941ab64890Smrg 3951ab64890Smrgstatic char * 3961ab64890Smrg_MbReset(XIC xic) 3971ab64890Smrg{ 3981ab64890Smrg return(NULL); 3991ab64890Smrg} 4001ab64890Smrg 4011ab64890Smrgstatic wchar_t * 4021ab64890Smrg_WcReset(XIC xic) 4031ab64890Smrg{ 4041ab64890Smrg return(NULL); 4051ab64890Smrg} 4061ab64890Smrg 4071ab64890Smrgstatic int 4081ab64890Smrg_MbLookupString( 4091ab64890Smrg XIC xic, 4101ab64890Smrg XKeyEvent *ev, 4111ab64890Smrg char * buffer, 4121ab64890Smrg int bytes, 4131ab64890Smrg KeySym *keysym, 4141ab64890Smrg Status *status) 4151ab64890Smrg{ 4161ab64890Smrg XComposeStatus NotSupportedYet ; 4171ab64890Smrg int length; 41861b2299dSmrg 4191ab64890Smrg length = XLookupString(ev, buffer, bytes, keysym, &NotSupportedYet); 4201ab64890Smrg 4211ab64890Smrg if (keysym && *keysym == NoSymbol){ 4221ab64890Smrg *status = XLookupNone; 4231ab64890Smrg } else if (length > 0) { 4241ab64890Smrg *status = XLookupBoth; 4251ab64890Smrg } else { 4261ab64890Smrg *status = XLookupKeySym; 4271ab64890Smrg } 4281ab64890Smrg return(length); 4291ab64890Smrg} 4301ab64890Smrg 4311ab64890Smrgstatic int 4321ab64890Smrg_WcLookupString( 4331ab64890Smrg XIC xic, 4341ab64890Smrg XKeyEvent *ev, 4351ab64890Smrg wchar_t * buffer, 4361ab64890Smrg int wlen, 4371ab64890Smrg KeySym *keysym, 4381ab64890Smrg Status *status) 4391ab64890Smrg{ 4401ab64890Smrg XComposeStatus NotSupportedYet ; 4411ab64890Smrg int length; 4421ab64890Smrg /* In single-byte, mb_len = wc_len */ 443818534a1Smrg char *mb_buf = Xmalloc(wlen); 4441ab64890Smrg 4451ab64890Smrg length = XLookupString(ev, mb_buf, wlen, keysym, &NotSupportedYet); 4461ab64890Smrg 4471ab64890Smrg if (keysym && *keysym == NoSymbol){ 4481ab64890Smrg *status = XLookupNone; 4491ab64890Smrg } else if (length > 0) { 4501ab64890Smrg *status = XLookupBoth; 4511ab64890Smrg } else { 4521ab64890Smrg *status = XLookupKeySym; 4531ab64890Smrg } 4541ab64890Smrg mbstowcs(buffer, mb_buf, length); 4551ab64890Smrg XFree(mb_buf); 4561ab64890Smrg return(length); 4571ab64890Smrg} 458