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