Files
allhaileris afb81b8278
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Close stale issues and PRs / stale (push) Has been cancelled
init
2026-02-16 15:50:16 +03:00

538 lines
16 KiB
C

/* -*- Mode: C; indent-tabs-mode: nil; c-basic-offset: 2; tab-width: 2 -*- */
/******************************************************************
Copyright 1994, 1995 by Sun Microsystems, Inc.
Copyright 1993, 1994 by Hewlett-Packard Company
Permission to use, copy, modify, distribute, and sell this software
and its documentation for any purpose is hereby granted without fee,
provided that the above copyright notice appear in all copies and
that both that copyright notice and this permission notice appear
in supporting documentation, and that the name of Sun Microsystems, Inc.
and Hewlett-Packard not be used in advertising or publicity pertaining to
distribution of the software without specific, written prior permission.
Sun Microsystems, Inc. and Hewlett-Packard make no representations about
the suitability of this software for any purpose. It is provided "as is"
without express or implied warranty.
SUN MICROSYSTEMS INC. AND HEWLETT-PACKARD COMPANY DISCLAIMS ALL
WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
SUN MICROSYSTEMS, INC. AND HEWLETT-PACKARD COMPANY BE LIABLE FOR ANY
SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
Author: Hidetoshi Tajima(tajima@Eng.Sun.COM) Sun Microsystems, Inc.
This version tidied and debugged by Steve Underwood May 1999
******************************************************************/
#include "i18nMethod.h"
#include "FrameMgr.h"
#include "Xi18n.h"
#include "XimFunc.h"
#include <glib.h>
/* How to generate SUPPORTED_LOCALES
#!/usr/bin/ruby
# -*- coding: utf-8 -*-
require 'set'
set = Set.new
File.open("/usr/share/i18n/SUPPORTED", "r").each do |line|
set << line.split(/_| /)[0]
end
puts set.to_a.join(",")
*/
#define SUPPORTED_LOCALES \
"aa,af,agr,ak,am,an,anp,ar,ayc,az,as,ast,be,bem,ber,bg,bhb,bho,bi,bn,bo," \
"br,brx,bs,byn,ca,ce,chr,cmn,crh,cs,csb,cv,cy,da,de,doi,dsb,dv,dz,el,en," \
"eo,es,et,eu,fa,ff,fi,fil,fo,fr,fur,fy,ga,gd,gez,gl,gu,gv,ha,hak,he,hi," \
"hif,hne,hr,hsb,ht,hu,hy,ia,id,ig,ik,is,it,iu,ja,ka,kab,kk,kl,km,kn,ko," \
"kok,ks,ku,kw,ky,lb,lg,li,lij,ln,lo,lt,lv,lzh,mag,mai,mfe,mg,mhr,mi,miq," \
"mjw,mk,ml,mn,mni,mr,ms,mt,my,nan,nb,nds,ne,nhn,niu,nl,nn,nr,nso,oc,om,or," \
"os,pa,pap,pl,ps,pt,quz,raj,ro,ru,rw,sa,sah,sat,sc,sd,se,sgs,shn,shs,si," \
"sid,sk,sl,sm,so,sq,sr,ss,st,sv,sw,szl,ta,tcy,te,tg,th,the,ti,tig,tk,tl," \
"tn,to,tpi,tr,ts,tt,ug,uk,unm,ur,uz,ve,vi,wa,wae,wal,wo,xh,yi,yo,yue,yuw," \
"zh,zu"
extern Xi18nClient *_Xi18nFindClient (NimfXim *, CARD16);
#ifndef XIM_SERVERS
#define XIM_SERVERS "XIM_SERVERS"
#endif
static Atom XIM_Servers = None;
static int SetXi18nSelectionOwner (NimfXim *xim, Window im_window)
{
Window root = RootWindow (xim->display, DefaultScreen (xim->display));
Atom realtype;
int realformat;
unsigned long bytesafter;
long *data=NULL;
unsigned long length;
Atom atom;
int i;
int found;
int forse = False;
if ((atom = XInternAtom (xim->display, "@server=nimf", False)) == 0)
return False;
xim->address.selection = atom;
if (XIM_Servers == None)
XIM_Servers = XInternAtom (xim->display, XIM_SERVERS, False);
/*endif*/
XGetWindowProperty (xim->display,
root,
XIM_Servers,
0L,
1000000L,
False,
XA_ATOM,
&realtype,
&realformat,
&length,
&bytesafter,
(unsigned char **) (&data));
if (realtype != None && (realtype != XA_ATOM || realformat != 32)) {
if (data != NULL)
XFree ((char *) data);
return False;
}
found = False;
for (i = 0; i < length; i++) {
if (data[i] == atom) {
Window owner;
found = True;
if ((owner = XGetSelectionOwner (xim->display, atom)) != im_window) {
if (owner == None || forse == True)
XSetSelectionOwner (xim->display, atom, im_window, CurrentTime);
else
return False;
}
break;
}
}
if (found == False) {
XSetSelectionOwner (xim->display, atom, im_window, CurrentTime);
XChangeProperty (xim->display,
root,
XIM_Servers,
XA_ATOM,
32,
PropModePrepend,
(unsigned char *) &atom,
1);
}
else {
/*
* We always need to generate the PropertyNotify to the Root Window
*/
XChangeProperty (xim->display,
root,
XIM_Servers,
XA_ATOM,
32,
PropModePrepend,
(unsigned char *) data,
0);
}
if (data != NULL)
XFree ((char *) data);
/* Intern "LOCALES" and "TRANSOPORT" Target Atoms */
xim->address.Localename = XInternAtom (xim->display, LOCALES, False);
xim->address.Transportname = XInternAtom (xim->display, TRANSPORT, False);
return (XGetSelectionOwner (xim->display, atom) == im_window);
}
static int DeleteXi18nAtom (NimfXim *xim)
{
Window root = RootWindow (xim->display, DefaultScreen (xim->display));
Atom realtype;
int realformat;
unsigned long bytesafter;
long *data=NULL;
unsigned long length;
Atom atom;
int i, ret;
int found;
if ((atom = XInternAtom (xim->display, "@server=nimf", False)) == 0)
return False;
xim->address.selection = atom;
if (XIM_Servers == None)
XIM_Servers = XInternAtom (xim->display, XIM_SERVERS, False);
XGetWindowProperty (xim->display,
root,
XIM_Servers,
0L,
1000000L,
False,
XA_ATOM,
&realtype,
&realformat,
&length,
&bytesafter,
(unsigned char **) (&data));
if (realtype != XA_ATOM || realformat != 32) {
if (data != NULL)
XFree ((char *) data);
return False;
}
found = False;
for (i = 0; i < length; i++) {
if (data[i] == atom) {
found = True;
break;
}
}
if (found == True) {
for (i=i+1; i<length; i++)
data[i-1] = data[i];
XChangeProperty (xim->display,
root,
XIM_Servers,
XA_ATOM,
32,
PropModeReplace,
(unsigned char *)data,
length-1);
ret = True;
}
else {
XChangeProperty (xim->display,
root,
XIM_Servers,
XA_ATOM,
32,
PropModePrepend,
(unsigned char *)data,
0);
ret = False;
}
if (data != NULL)
XFree ((char *) data);
return ret;
}
static void
ReturnSelectionNotify (NimfXim *xim, XSelectionRequestEvent *ev)
{
XEvent event;
const char *data = NULL;
event.type = SelectionNotify;
event.xselection.requestor = ev->requestor;
event.xselection.selection = ev->selection;
event.xselection.target = ev->target;
event.xselection.time = ev->time;
event.xselection.property = ev->property;
if (ev->target == xim->address.Localename)
data = "@locale=C" SUPPORTED_LOCALES;
else if (ev->target == xim->address.Transportname)
data = "@transport=X/";
XChangeProperty (xim->display,
event.xselection.requestor,
ev->target,
ev->target,
8,
PropModeReplace,
(unsigned char *) data,
strlen (data));
XSendEvent (xim->display, event.xselection.requestor, False, NoEventMask, &event);
XFlush (xim->display);
}
Bool
WaitXSelectionRequest (NimfXim *xim,
XEvent *ev)
{
if (((XSelectionRequestEvent *) ev)->selection == xim->address.selection)
{
ReturnSelectionNotify (xim, (XSelectionRequestEvent *) ev);
return True;
}
return False;
}
Status
xi18n_openIM (NimfXim *xim, Window im_window)
{
if (!SetXi18nSelectionOwner (xim, im_window))
return False;
xim->_protocol = XInternAtom (xim->display, "_XIM_PROTOCOL", False);
xim->_xconnect = XInternAtom (xim->display, "_XIM_XCONNECT", False);
XFlush (xim->display);
return True;
}
Status xi18n_closeIM (NimfXim *xim)
{
DeleteXi18nAtom (xim);
return True;
}
static void EventToWireEvent (XEvent *ev, xEvent *event,
CARD16 *serial, Bool byte_swap)
{
FrameMgr fm;
extern XimFrameRec wire_keyevent_fr[];
extern XimFrameRec short_fr[];
BYTE b;
CARD16 c16;
CARD32 c32;
*serial = (CARD16)(ev->xany.serial >> 16);
switch (ev->type) {
case KeyPress:
case KeyRelease:
{
XKeyEvent *kev = (XKeyEvent*)ev;
/* create FrameMgr */
fm = FrameMgrInit(wire_keyevent_fr, (char *)(&(event->u)), byte_swap);
/* set values */
b = (BYTE)kev->type; FrameMgrPutToken(fm, b);
b = (BYTE)kev->keycode; FrameMgrPutToken(fm, b);
c16 = (CARD16)(kev->serial & (unsigned long)0xffff);
FrameMgrPutToken(fm, c16);
c32 = (CARD32)kev->time; FrameMgrPutToken(fm, c32);
c32 = (CARD32)kev->root; FrameMgrPutToken(fm, c32);
c32 = (CARD32)kev->window; FrameMgrPutToken(fm, c32);
c32 = (CARD32)kev->subwindow; FrameMgrPutToken(fm, c32);
c16 = (CARD16)kev->x_root; FrameMgrPutToken(fm, c16);
c16 = (CARD16)kev->y_root; FrameMgrPutToken(fm, c16);
c16 = (CARD16)kev->x; FrameMgrPutToken(fm, c16);
c16 = (CARD16)kev->y; FrameMgrPutToken(fm, c16);
c16 = (CARD16)kev->state; FrameMgrPutToken(fm, c16);
b = (BYTE)kev->same_screen; FrameMgrPutToken(fm, b);
}
break;
default:
/* create FrameMgr */
fm = FrameMgrInit(short_fr, (char *)(&(event->u.u.sequenceNumber)),
byte_swap);
c16 = (CARD16)(ev->xany.serial & (unsigned long)0xffff);
FrameMgrPutToken(fm, c16);
break;
}
/* free FrameMgr */
FrameMgrFree(fm);
}
Status xi18n_forwardEvent (NimfXim *xim, XPointer xp)
{
IMForwardEventStruct *call_data = (IMForwardEventStruct *)xp;
FrameMgr fm;
extern XimFrameRec forward_event_fr[];
register int total_size;
unsigned char *reply = NULL;
unsigned char *replyp;
CARD16 serial;
int event_size;
Xi18nClient *client;
client = (Xi18nClient *) _Xi18nFindClient (xim, call_data->connect_id);
/* create FrameMgr */
fm = FrameMgrInit (forward_event_fr,
NULL,
_Xi18nNeedSwap (xim, call_data->connect_id));
total_size = FrameMgrGetTotalSize (fm);
event_size = sizeof (xEvent);
reply = (unsigned char *) malloc (total_size + event_size);
if (!reply)
{
_Xi18nSendMessage (xim,
call_data->connect_id,
XIM_ERROR,
0,
0,
0);
return False;
}
/*endif*/
memset (reply, 0, total_size + event_size);
FrameMgrSetBuffer (fm, reply);
replyp = reply;
call_data->sync_bit = 1; /* always sync */
client->sync = True;
FrameMgrPutToken (fm, call_data->connect_id);
FrameMgrPutToken (fm, call_data->icid);
FrameMgrPutToken (fm, call_data->sync_bit);
replyp += total_size;
EventToWireEvent (&(call_data->event),
(xEvent *) replyp,
&serial,
_Xi18nNeedSwap (xim, call_data->connect_id));
FrameMgrPutToken (fm, serial);
_Xi18nSendMessage (xim,
call_data->connect_id,
XIM_FORWARD_EVENT,
0,
reply,
total_size + event_size);
XFree (reply);
FrameMgrFree (fm);
return True;
}
Status xi18n_commit (NimfXim *xim, XPointer xp)
{
IMCommitStruct *call_data = (IMCommitStruct *)xp;
FrameMgr fm;
extern XimFrameRec commit_chars_fr[];
extern XimFrameRec commit_both_fr[];
register int total_size;
unsigned char *reply = NULL;
CARD16 str_length;
call_data->flag |= XimSYNCHRONUS; /* always sync */
if (!(call_data->flag & XimLookupKeySym)
&&
(call_data->flag & XimLookupChars))
{
fm = FrameMgrInit (commit_chars_fr,
NULL,
_Xi18nNeedSwap (xim, call_data->connect_id));
/* set length of STRING8 */
str_length = strlen (call_data->commit_string);
FrameMgrSetSize (fm, str_length);
total_size = FrameMgrGetTotalSize (fm);
reply = (unsigned char *) malloc (total_size);
if (!reply)
{
_Xi18nSendMessage (xim,
call_data->connect_id,
XIM_ERROR,
0,
0,
0);
return False;
}
/*endif*/
memset (reply, 0, total_size);
FrameMgrSetBuffer (fm, reply);
str_length = FrameMgrGetSize (fm);
FrameMgrPutToken (fm, call_data->connect_id);
FrameMgrPutToken (fm, call_data->icid);
FrameMgrPutToken (fm, call_data->flag);
FrameMgrPutToken (fm, str_length);
FrameMgrPutToken (fm, call_data->commit_string);
}
else
{
fm = FrameMgrInit (commit_both_fr,
NULL,
_Xi18nNeedSwap (xim, call_data->connect_id));
/* set length of STRING8 */
str_length = strlen (call_data->commit_string);
if (str_length > 0)
FrameMgrSetSize (fm, str_length);
/*endif*/
total_size = FrameMgrGetTotalSize (fm);
reply = (unsigned char *) malloc (total_size);
if (!reply)
{
_Xi18nSendMessage (xim,
call_data->connect_id,
XIM_ERROR,
0,
0,
0);
return False;
}
/*endif*/
FrameMgrSetBuffer (fm, reply);
FrameMgrPutToken (fm, call_data->connect_id);
FrameMgrPutToken (fm, call_data->icid);
FrameMgrPutToken (fm, call_data->flag);
FrameMgrPutToken (fm, call_data->keysym);
if (str_length > 0)
{
str_length = FrameMgrGetSize (fm);
FrameMgrPutToken (fm, str_length);
FrameMgrPutToken (fm, call_data->commit_string);
}
/*endif*/
}
/*endif*/
_Xi18nSendMessage (xim,
call_data->connect_id,
XIM_COMMIT,
0,
reply,
total_size);
FrameMgrFree (fm);
XFree (reply);
return True;
}
int nimf_xim_call_callback (NimfXim *xim, XPointer xp)
{
IMProtocol *call_data = (IMProtocol *)xp;
switch (call_data->major_code)
{
case XIM_GEOMETRY:
return _Xi18nGeometryCallback (xim, call_data);
case XIM_PREEDIT_START:
return _Xi18nPreeditStartCallback (xim, call_data);
case XIM_PREEDIT_DRAW:
return _Xi18nPreeditDrawCallback (xim, call_data);
case XIM_PREEDIT_CARET:
return _Xi18nPreeditCaretCallback (xim, call_data);
case XIM_PREEDIT_DONE:
return _Xi18nPreeditDoneCallback (xim, call_data);
case XIM_STATUS_START:
return _Xi18nStatusStartCallback (xim, call_data);
case XIM_STATUS_DRAW:
return _Xi18nStatusDrawCallback (xim, call_data);
case XIM_STATUS_DONE:
return _Xi18nStatusDoneCallback (xim, call_data);
case XIM_STR_CONVERSION:
return _Xi18nStringConversionCallback (xim, call_data);
}
/*endswitch*/
return False;
}