1426 lines
39 KiB
C
1426 lines
39 KiB
C
|
/************************************************************
|
||
|
Copyright (c) 1993 by Silicon Graphics Computer Systems, Inc.
|
||
|
|
||
|
Permission to use, copy, modify, and distribute this
|
||
|
software and its documentation for any purpose and without
|
||
|
fee is hereby granted, 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 Silicon Graphics not be
|
||
|
used in advertising or publicity pertaining to distribution
|
||
|
of the software without specific prior written permission.
|
||
|
Silicon Graphics makes no representation about the suitability
|
||
|
of this software for any purpose. It is provided "as is"
|
||
|
without any express or implied warranty.
|
||
|
|
||
|
SILICON GRAPHICS DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS
|
||
|
SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
||
|
AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL SILICON
|
||
|
GRAPHICS 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.
|
||
|
|
||
|
********************************************************/
|
||
|
|
||
|
#ifdef HAVE_DIX_CONFIG_H
|
||
|
#include <dix-config.h>
|
||
|
#endif
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <math.h>
|
||
|
#define NEED_EVENTS 1
|
||
|
#include <X11/X.h>
|
||
|
#include <X11/Xproto.h>
|
||
|
#include <X11/keysym.h>
|
||
|
#include "misc.h"
|
||
|
#include "inputstr.h"
|
||
|
#include <X11/extensions/XKBsrv.h>
|
||
|
#include "xkb.h"
|
||
|
#include <ctype.h>
|
||
|
|
||
|
static unsigned int _xkbServerGeneration;
|
||
|
int xkbDevicePrivateIndex = -1;
|
||
|
|
||
|
void
|
||
|
xkbUnwrapProc(DeviceIntPtr device, DeviceHandleProc proc,
|
||
|
pointer data)
|
||
|
{
|
||
|
xkbDeviceInfoPtr xkbPrivPtr = XKBDEVICEINFO(device);
|
||
|
ProcessInputProc tmp = device->public.processInputProc;
|
||
|
if(xkbPrivPtr->unwrapProc)
|
||
|
xkbPrivPtr->unwrapProc = NULL;
|
||
|
|
||
|
UNWRAP_PROCESS_INPUT_PROC(device,xkbPrivPtr);
|
||
|
proc(device,data);
|
||
|
WRAP_PROCESS_INPUT_PROC(device,xkbPrivPtr,
|
||
|
tmp,xkbUnwrapProc);
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
XkbSetExtension(DeviceIntPtr device, ProcessInputProc proc)
|
||
|
{
|
||
|
xkbDeviceInfoPtr xkbPrivPtr;
|
||
|
|
||
|
if (serverGeneration != _xkbServerGeneration) {
|
||
|
if ((xkbDevicePrivateIndex = AllocateDevicePrivateIndex()) == -1)
|
||
|
return;
|
||
|
_xkbServerGeneration = serverGeneration;
|
||
|
}
|
||
|
if (!AllocateDevicePrivate(device, xkbDevicePrivateIndex))
|
||
|
return;
|
||
|
|
||
|
xkbPrivPtr = (xkbDeviceInfoPtr) xalloc(sizeof(xkbDeviceInfoRec));
|
||
|
if (!xkbPrivPtr)
|
||
|
return;
|
||
|
xkbPrivPtr->unwrapProc = NULL;
|
||
|
|
||
|
device->devPrivates[xkbDevicePrivateIndex].ptr = xkbPrivPtr;
|
||
|
WRAP_PROCESS_INPUT_PROC(device,xkbPrivPtr,
|
||
|
proc,xkbUnwrapProc);
|
||
|
}
|
||
|
|
||
|
#ifdef XINPUT
|
||
|
extern void ProcessOtherEvent(
|
||
|
xEvent * /* xE */,
|
||
|
DeviceIntPtr /* dev */,
|
||
|
int /* count */
|
||
|
);
|
||
|
#endif
|
||
|
|
||
|
/***====================================================================***/
|
||
|
|
||
|
static XkbAction
|
||
|
_FixUpAction(XkbDescPtr xkb,XkbAction *act)
|
||
|
{
|
||
|
static XkbAction fake;
|
||
|
|
||
|
if (XkbIsPtrAction(act)&&(!(xkb->ctrls->enabled_ctrls&XkbMouseKeysMask))) {
|
||
|
fake.type = XkbSA_NoAction;
|
||
|
return fake;
|
||
|
}
|
||
|
if (XkbDisableLockActions) {
|
||
|
switch (act->type) {
|
||
|
case XkbSA_LockMods:
|
||
|
fake.mods.type = XkbSA_SetMods;
|
||
|
fake.mods.flags = 0;
|
||
|
fake.mods.mask = act->mods.mask;
|
||
|
return fake;
|
||
|
case XkbSA_LatchMods:
|
||
|
fake.mods.type = XkbSA_SetMods;
|
||
|
fake.mods.flags = 0;
|
||
|
fake.mods.mask = act->mods.mask;
|
||
|
return fake;
|
||
|
case XkbSA_ISOLock:
|
||
|
if (act->iso.flags&XkbSA_ISODfltIsGroup) {
|
||
|
fake.group.type = XkbSA_SetGroup;
|
||
|
fake.group.flags = act->iso.flags&XkbSA_GroupAbsolute;
|
||
|
XkbSASetGroup(&fake.group,XkbSAGroup(&act->iso));
|
||
|
}
|
||
|
else {
|
||
|
fake.mods.type = XkbSA_SetMods;
|
||
|
fake.mods.flags = 0;
|
||
|
fake.mods.mask = act->iso.mask;
|
||
|
}
|
||
|
return fake;
|
||
|
case XkbSA_LockGroup:
|
||
|
case XkbSA_LatchGroup:
|
||
|
/* We want everything from the latch/lock action except the
|
||
|
* type should be changed to set.
|
||
|
*/
|
||
|
fake = *act;
|
||
|
fake.group.type = XkbSA_SetGroup;
|
||
|
return fake;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
if (xkb->ctrls->enabled_ctrls&XkbStickyKeysMask) {
|
||
|
if (act->any.type==XkbSA_SetMods) {
|
||
|
fake.mods.type = XkbSA_LatchMods;
|
||
|
fake.mods.mask = act->mods.mask;
|
||
|
if (XkbAX_NeedOption(xkb->ctrls,XkbAX_LatchToLockMask))
|
||
|
fake.mods.flags= XkbSA_ClearLocks|XkbSA_LatchToLock;
|
||
|
else fake.mods.flags= XkbSA_ClearLocks;
|
||
|
return fake;
|
||
|
}
|
||
|
if (act->any.type==XkbSA_SetGroup) {
|
||
|
fake.group.type = XkbSA_LatchGroup;
|
||
|
if (XkbAX_NeedOption(xkb->ctrls,XkbAX_LatchToLockMask))
|
||
|
fake.group.flags= XkbSA_ClearLocks|XkbSA_LatchToLock;
|
||
|
else fake.group.flags= XkbSA_ClearLocks;
|
||
|
XkbSASetGroup(&fake.group,XkbSAGroup(&act->group));
|
||
|
return fake;
|
||
|
}
|
||
|
}
|
||
|
return *act;
|
||
|
}
|
||
|
|
||
|
static XkbAction
|
||
|
XkbGetKeyAction(XkbSrvInfoPtr xkbi,XkbStatePtr xkbState,CARD8 key)
|
||
|
{
|
||
|
int effectiveGroup;
|
||
|
int col;
|
||
|
XkbDescPtr xkb;
|
||
|
XkbKeyTypePtr type;
|
||
|
XkbAction * pActs;
|
||
|
static XkbAction fake;
|
||
|
|
||
|
xkb= xkbi->desc;
|
||
|
if (!XkbKeyHasActions(xkb,key) || !XkbKeycodeInRange(xkb,key)) {
|
||
|
fake.type = XkbSA_NoAction;
|
||
|
return fake;
|
||
|
}
|
||
|
pActs= XkbKeyActionsPtr(xkb,key);
|
||
|
col= 0;
|
||
|
effectiveGroup= xkbState->group;
|
||
|
if (effectiveGroup!=XkbGroup1Index) {
|
||
|
if (XkbKeyNumGroups(xkb,key)>(unsigned)1) {
|
||
|
if (effectiveGroup>=XkbKeyNumGroups(xkb,key)) {
|
||
|
unsigned gi= XkbKeyGroupInfo(xkb,key);
|
||
|
switch (XkbOutOfRangeGroupAction(gi)) {
|
||
|
default:
|
||
|
case XkbWrapIntoRange:
|
||
|
effectiveGroup %= XkbKeyNumGroups(xkb,key);
|
||
|
break;
|
||
|
case XkbClampIntoRange:
|
||
|
effectiveGroup = XkbKeyNumGroups(xkb,key)-1;
|
||
|
break;
|
||
|
case XkbRedirectIntoRange:
|
||
|
effectiveGroup= XkbOutOfRangeGroupInfo(gi);
|
||
|
if (effectiveGroup>=XkbKeyNumGroups(xkb,key))
|
||
|
effectiveGroup= 0;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else effectiveGroup= XkbGroup1Index;
|
||
|
col+= (effectiveGroup*XkbKeyGroupsWidth(xkb,key));
|
||
|
}
|
||
|
type= XkbKeyKeyType(xkb,key,effectiveGroup);
|
||
|
if (type->map!=NULL) {
|
||
|
register unsigned i,mods;
|
||
|
register XkbKTMapEntryPtr entry;
|
||
|
mods= xkbState->mods&type->mods.mask;
|
||
|
for (entry= type->map,i=0;i<type->map_count;i++,entry++) {
|
||
|
if ((entry->active)&&(entry->mods.mask==mods)) {
|
||
|
col+= entry->level;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (pActs[col].any.type==XkbSA_NoAction)
|
||
|
return pActs[col];
|
||
|
fake= _FixUpAction(xkb,&pActs[col]);
|
||
|
return fake;
|
||
|
}
|
||
|
|
||
|
XkbAction
|
||
|
XkbGetButtonAction(DeviceIntPtr kbd,DeviceIntPtr dev,int button)
|
||
|
{
|
||
|
XkbAction fake;
|
||
|
if ((dev->button)&&(dev->button->xkb_acts)) {
|
||
|
if (dev->button->xkb_acts[button-1].any.type!=XkbSA_NoAction) {
|
||
|
fake= _FixUpAction(kbd->key->xkbInfo->desc,
|
||
|
&dev->button->xkb_acts[button-1]);
|
||
|
return fake;
|
||
|
}
|
||
|
}
|
||
|
fake.any.type= XkbSA_NoAction;
|
||
|
return fake;
|
||
|
}
|
||
|
|
||
|
/***====================================================================***/
|
||
|
|
||
|
#define SYNTHETIC_KEYCODE 1
|
||
|
#define BTN_ACT_FLAG 0x100
|
||
|
|
||
|
typedef struct _XkbFilter {
|
||
|
CARD16 keycode;
|
||
|
CARD8 what;
|
||
|
CARD8 active;
|
||
|
CARD8 filterOthers;
|
||
|
CARD32 priv;
|
||
|
XkbAction upAction;
|
||
|
int (*filter)(
|
||
|
XkbSrvInfoPtr /* xkbi */,
|
||
|
struct _XkbFilter * /* filter */,
|
||
|
unsigned /* keycode */,
|
||
|
XkbAction * /* action */
|
||
|
);
|
||
|
struct _XkbFilter *next;
|
||
|
} XkbFilterRec,*XkbFilterPtr;
|
||
|
|
||
|
static int
|
||
|
_XkbFilterSetState( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction *pAction)
|
||
|
{
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = ((pAction->mods.mask&XkbSA_ClearLocks)!=0);
|
||
|
filter->priv = 0;
|
||
|
filter->filter = _XkbFilterSetState;
|
||
|
if (pAction->type==XkbSA_SetMods) {
|
||
|
filter->upAction = *pAction;
|
||
|
xkbi->setMods= pAction->mods.mask;
|
||
|
}
|
||
|
else {
|
||
|
xkbi->groupChange = XkbSAGroup(&pAction->group);
|
||
|
if (pAction->group.flags&XkbSA_GroupAbsolute)
|
||
|
xkbi->groupChange-= xkbi->state.base_group;
|
||
|
filter->upAction= *pAction;
|
||
|
XkbSASetGroup(&filter->upAction.group,xkbi->groupChange);
|
||
|
}
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
if (filter->upAction.type==XkbSA_SetMods) {
|
||
|
xkbi->clearMods = filter->upAction.mods.mask;
|
||
|
if (filter->upAction.mods.flags&XkbSA_ClearLocks) {
|
||
|
xkbi->state.locked_mods&= ~filter->upAction.mods.mask;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
if (filter->upAction.group.flags&XkbSA_ClearLocks) {
|
||
|
xkbi->state.locked_group = 0;
|
||
|
}
|
||
|
xkbi->groupChange = -XkbSAGroup(&filter->upAction.group);
|
||
|
}
|
||
|
filter->active = 0;
|
||
|
}
|
||
|
else {
|
||
|
filter->upAction.mods.flags&= ~XkbSA_ClearLocks;
|
||
|
filter->filterOthers = 0;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
#define LATCH_KEY_DOWN 1
|
||
|
#define LATCH_PENDING 2
|
||
|
#define NO_LATCH 3
|
||
|
|
||
|
static int
|
||
|
_XkbFilterLatchState( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 1;
|
||
|
filter->priv = LATCH_KEY_DOWN;
|
||
|
filter->filter = _XkbFilterLatchState;
|
||
|
if (pAction->type==XkbSA_LatchMods) {
|
||
|
filter->upAction = *pAction;
|
||
|
xkbi->setMods = pAction->mods.mask;
|
||
|
}
|
||
|
else {
|
||
|
xkbi->groupChange = XkbSAGroup(&pAction->group);
|
||
|
if (pAction->group.flags&XkbSA_GroupAbsolute)
|
||
|
xkbi->groupChange-= xkbi->state.base_group;
|
||
|
filter->upAction= *pAction;
|
||
|
XkbSASetGroup(&filter->upAction.group,xkbi->groupChange);
|
||
|
}
|
||
|
}
|
||
|
else if ( pAction && (filter->priv==LATCH_PENDING) ) {
|
||
|
if (((1<<pAction->type)&XkbSA_BreakLatch)!=0) {
|
||
|
filter->active = 0;
|
||
|
if (filter->upAction.type==XkbSA_LatchMods)
|
||
|
xkbi->state.latched_mods&= ~filter->upAction.mods.mask;
|
||
|
else xkbi->state.latched_group-=XkbSAGroup(&filter->upAction.group);
|
||
|
}
|
||
|
else if ((pAction->type==filter->upAction.type)&&
|
||
|
(pAction->mods.flags==filter->upAction.mods.flags)&&
|
||
|
(pAction->mods.mask==filter->upAction.mods.mask)) {
|
||
|
if (filter->upAction.mods.flags&XkbSA_LatchToLock) {
|
||
|
XkbControlsPtr ctrls= xkbi->desc->ctrls;
|
||
|
if (filter->upAction.type==XkbSA_LatchMods)
|
||
|
pAction->mods.type= XkbSA_LockMods;
|
||
|
else pAction->group.type= XkbSA_LockGroup;
|
||
|
if (XkbAX_NeedFeedback(ctrls,XkbAX_StickyKeysFBMask)&&
|
||
|
(ctrls->enabled_ctrls&XkbStickyKeysMask)) {
|
||
|
XkbDDXAccessXBeep(xkbi->device,_BEEP_STICKY_LOCK,
|
||
|
XkbStickyKeysMask);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
if (filter->upAction.type==XkbSA_LatchMods)
|
||
|
pAction->mods.type= XkbSA_SetMods;
|
||
|
else pAction->group.type= XkbSA_SetGroup;
|
||
|
}
|
||
|
if (filter->upAction.type==XkbSA_LatchMods)
|
||
|
xkbi->state.latched_mods&= ~filter->upAction.mods.mask;
|
||
|
else xkbi->state.latched_group-=XkbSAGroup(&filter->upAction.group);
|
||
|
filter->active = 0;
|
||
|
}
|
||
|
}
|
||
|
else if (filter->keycode==keycode) { /* release */
|
||
|
XkbControlsPtr ctrls= xkbi->desc->ctrls;
|
||
|
int needBeep;
|
||
|
int beepType= _BEEP_NONE;
|
||
|
|
||
|
needBeep= ((ctrls->enabled_ctrls&XkbStickyKeysMask)&&
|
||
|
XkbAX_NeedFeedback(ctrls,XkbAX_StickyKeysFBMask));
|
||
|
if (filter->upAction.type==XkbSA_LatchMods) {
|
||
|
xkbi->clearMods = filter->upAction.mods.mask;
|
||
|
if ((filter->upAction.mods.flags&XkbSA_ClearLocks)&&
|
||
|
(xkbi->clearMods&xkbi->state.locked_mods)==xkbi->clearMods) {
|
||
|
xkbi->state.locked_mods&= ~xkbi->clearMods;
|
||
|
filter->priv= NO_LATCH;
|
||
|
beepType= _BEEP_STICKY_UNLOCK;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
xkbi->groupChange = -XkbSAGroup(&filter->upAction.group);
|
||
|
if ((filter->upAction.group.flags&XkbSA_ClearLocks)&&
|
||
|
(xkbi->state.locked_group)) {
|
||
|
xkbi->state.locked_group = 0;
|
||
|
filter->priv = NO_LATCH;
|
||
|
beepType= _BEEP_STICKY_UNLOCK;
|
||
|
}
|
||
|
}
|
||
|
if (filter->priv==NO_LATCH) {
|
||
|
filter->active= 0;
|
||
|
}
|
||
|
else {
|
||
|
filter->priv= LATCH_PENDING;
|
||
|
if (filter->upAction.type==XkbSA_LatchMods) {
|
||
|
xkbi->state.latched_mods |= filter->upAction.mods.mask;
|
||
|
needBeep = xkbi->state.latched_mods ? needBeep : 0;
|
||
|
xkbi->state.latched_mods |= filter->upAction.mods.mask;
|
||
|
}
|
||
|
else {
|
||
|
xkbi->state.latched_group+= XkbSAGroup(&filter->upAction.group);
|
||
|
}
|
||
|
if (needBeep && (beepType==_BEEP_NONE))
|
||
|
beepType= _BEEP_STICKY_LATCH;
|
||
|
}
|
||
|
if (needBeep && (beepType!=_BEEP_NONE))
|
||
|
XkbDDXAccessXBeep(xkbi->device,beepType,XkbStickyKeysMask);
|
||
|
}
|
||
|
else if (filter->priv==LATCH_KEY_DOWN) {
|
||
|
filter->priv= NO_LATCH;
|
||
|
filter->filterOthers = 0;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterLockState( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
|
||
|
if (pAction&&(pAction->type==XkbSA_LockGroup)) {
|
||
|
if (pAction->group.flags&XkbSA_GroupAbsolute)
|
||
|
xkbi->state.locked_group= XkbSAGroup(&pAction->group);
|
||
|
else xkbi->state.locked_group+= XkbSAGroup(&pAction->group);
|
||
|
return 1;
|
||
|
}
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->priv = 0;
|
||
|
filter->filter = _XkbFilterLockState;
|
||
|
filter->upAction = *pAction;
|
||
|
xkbi->state.locked_mods^= pAction->mods.mask;
|
||
|
xkbi->setMods = pAction->mods.mask;
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
filter->active = 0;
|
||
|
xkbi->clearMods = filter->upAction.mods.mask;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
#define ISO_KEY_DOWN 0
|
||
|
#define NO_ISO_LOCK 1
|
||
|
|
||
|
static int
|
||
|
_XkbFilterISOLock( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
CARD8 flags= pAction->iso.flags;
|
||
|
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 1;
|
||
|
filter->priv = ISO_KEY_DOWN;
|
||
|
filter->upAction = *pAction;
|
||
|
filter->filter = _XkbFilterISOLock;
|
||
|
if (flags&XkbSA_ISODfltIsGroup) {
|
||
|
xkbi->groupChange = XkbSAGroup(&pAction->iso);
|
||
|
xkbi->setMods = 0;
|
||
|
}
|
||
|
else {
|
||
|
xkbi->setMods = pAction->iso.mask;
|
||
|
xkbi->groupChange = 0;
|
||
|
}
|
||
|
if ((!(flags&XkbSA_ISONoAffectMods))&&(xkbi->state.base_mods)) {
|
||
|
filter->priv= NO_ISO_LOCK;
|
||
|
xkbi->state.locked_mods^= xkbi->state.base_mods;
|
||
|
}
|
||
|
if ((!(flags&XkbSA_ISONoAffectGroup))&&(xkbi->state.base_group)) {
|
||
|
/* 6/22/93 (ef) -- lock groups if group key is down first */
|
||
|
}
|
||
|
if (!(flags&XkbSA_ISONoAffectPtr)) {
|
||
|
/* 6/22/93 (ef) -- lock mouse buttons if they're down */
|
||
|
}
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
CARD8 flags= filter->upAction.iso.flags;
|
||
|
|
||
|
if (flags&XkbSA_ISODfltIsGroup) {
|
||
|
xkbi->groupChange = -XkbSAGroup(&filter->upAction.iso);
|
||
|
xkbi->clearMods = 0;
|
||
|
if (filter->priv==ISO_KEY_DOWN)
|
||
|
xkbi->state.locked_group+= XkbSAGroup(&filter->upAction.iso);
|
||
|
}
|
||
|
else {
|
||
|
xkbi->clearMods= filter->upAction.iso.mask;
|
||
|
xkbi->groupChange= 0;
|
||
|
if (filter->priv==ISO_KEY_DOWN)
|
||
|
xkbi->state.locked_mods^= filter->upAction.iso.mask;
|
||
|
}
|
||
|
filter->active = 0;
|
||
|
}
|
||
|
else if (pAction) {
|
||
|
CARD8 flags= filter->upAction.iso.flags;
|
||
|
|
||
|
switch (pAction->type) {
|
||
|
case XkbSA_SetMods: case XkbSA_LatchMods:
|
||
|
if (!(flags&XkbSA_ISONoAffectMods)) {
|
||
|
pAction->type= XkbSA_LockMods;
|
||
|
filter->priv= NO_ISO_LOCK;
|
||
|
}
|
||
|
break;
|
||
|
case XkbSA_SetGroup: case XkbSA_LatchGroup:
|
||
|
if (!(flags&XkbSA_ISONoAffectGroup)) {
|
||
|
pAction->type= XkbSA_LockGroup;
|
||
|
filter->priv= NO_ISO_LOCK;
|
||
|
}
|
||
|
break;
|
||
|
case XkbSA_PtrBtn:
|
||
|
if (!(flags&XkbSA_ISONoAffectPtr)) {
|
||
|
pAction->type= XkbSA_LockPtrBtn;
|
||
|
filter->priv= NO_ISO_LOCK;
|
||
|
}
|
||
|
break;
|
||
|
case XkbSA_SetControls:
|
||
|
if (!(flags&XkbSA_ISONoAffectCtrls)) {
|
||
|
pAction->type= XkbSA_LockControls;
|
||
|
filter->priv= NO_ISO_LOCK;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
static CARD32
|
||
|
_XkbPtrAccelExpire(OsTimerPtr timer,CARD32 now,pointer arg)
|
||
|
{
|
||
|
XkbSrvInfoPtr xkbi= (XkbSrvInfoPtr)arg;
|
||
|
XkbControlsPtr ctrls= xkbi->desc->ctrls;
|
||
|
int dx,dy;
|
||
|
|
||
|
if (xkbi->mouseKey==0)
|
||
|
return 0;
|
||
|
|
||
|
if (xkbi->mouseKeysAccel) {
|
||
|
if ((xkbi->mouseKeysCounter)<ctrls->mk_time_to_max) {
|
||
|
double step;
|
||
|
xkbi->mouseKeysCounter++;
|
||
|
step= xkbi->mouseKeysCurveFactor*
|
||
|
pow((double)xkbi->mouseKeysCounter,xkbi->mouseKeysCurve);
|
||
|
if (xkbi->mouseKeysDX<0)
|
||
|
dx= floor( ((double)xkbi->mouseKeysDX)*step );
|
||
|
else dx= ceil( ((double)xkbi->mouseKeysDX)*step );
|
||
|
if (xkbi->mouseKeysDY<0)
|
||
|
dy= floor( ((double)xkbi->mouseKeysDY)*step );
|
||
|
else dy= ceil( ((double)xkbi->mouseKeysDY)*step );
|
||
|
}
|
||
|
else {
|
||
|
dx= xkbi->mouseKeysDX*ctrls->mk_max_speed;
|
||
|
dy= xkbi->mouseKeysDY*ctrls->mk_max_speed;
|
||
|
}
|
||
|
if (xkbi->mouseKeysFlags&XkbSA_MoveAbsoluteX)
|
||
|
dx= xkbi->mouseKeysDX;
|
||
|
if (xkbi->mouseKeysFlags&XkbSA_MoveAbsoluteY)
|
||
|
dy= xkbi->mouseKeysDY;
|
||
|
}
|
||
|
else {
|
||
|
dx= xkbi->mouseKeysDX;
|
||
|
dy= xkbi->mouseKeysDY;
|
||
|
}
|
||
|
XkbDDXFakePointerMotion(xkbi->mouseKeysFlags,dx,dy);
|
||
|
return xkbi->desc->ctrls->mk_interval;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterPointerMove( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
int x,y;
|
||
|
Bool accel;
|
||
|
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->priv=0;
|
||
|
filter->filter = _XkbFilterPointerMove;
|
||
|
filter->upAction= *pAction;
|
||
|
xkbi->mouseKeysCounter= 0;
|
||
|
xkbi->mouseKey= keycode;
|
||
|
accel= ((pAction->ptr.flags&XkbSA_NoAcceleration)==0);
|
||
|
x= XkbPtrActionX(&pAction->ptr);
|
||
|
y= XkbPtrActionY(&pAction->ptr);
|
||
|
XkbDDXFakePointerMotion(pAction->ptr.flags,x,y);
|
||
|
AccessXCancelRepeatKey(xkbi,keycode);
|
||
|
xkbi->mouseKeysAccel= accel&&
|
||
|
(xkbi->desc->ctrls->enabled_ctrls&XkbMouseKeysAccelMask);
|
||
|
xkbi->mouseKeysFlags= pAction->ptr.flags;
|
||
|
xkbi->mouseKeysDX= XkbPtrActionX(&pAction->ptr);
|
||
|
xkbi->mouseKeysDY= XkbPtrActionY(&pAction->ptr);
|
||
|
xkbi->mouseKeyTimer= TimerSet(xkbi->mouseKeyTimer, 0,
|
||
|
xkbi->desc->ctrls->mk_delay,
|
||
|
_XkbPtrAccelExpire,(pointer)xkbi);
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
filter->active = 0;
|
||
|
if (xkbi->mouseKey==keycode) {
|
||
|
xkbi->mouseKey= 0;
|
||
|
xkbi->mouseKeyTimer= TimerSet(xkbi->mouseKeyTimer, 0, 0,
|
||
|
NULL, NULL);
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterPointerBtn( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
int button= pAction->btn.button;
|
||
|
|
||
|
if (button==XkbSA_UseDfltButton)
|
||
|
button = xkbi->desc->ctrls->mk_dflt_btn;
|
||
|
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->priv=0;
|
||
|
filter->filter = _XkbFilterPointerBtn;
|
||
|
filter->upAction= *pAction;
|
||
|
filter->upAction.btn.button= button;
|
||
|
switch (pAction->type) {
|
||
|
case XkbSA_LockPtrBtn:
|
||
|
if (((xkbi->lockedPtrButtons&(1<<button))==0)&&
|
||
|
((pAction->btn.flags&XkbSA_LockNoLock)==0)) {
|
||
|
xkbi->lockedPtrButtons|= (1<<button);
|
||
|
AccessXCancelRepeatKey(xkbi,keycode);
|
||
|
XkbDDXFakePointerButton(ButtonPress,button);
|
||
|
filter->upAction.type= XkbSA_NoAction;
|
||
|
}
|
||
|
break;
|
||
|
case XkbSA_PtrBtn:
|
||
|
{
|
||
|
register int i,nClicks;
|
||
|
AccessXCancelRepeatKey(xkbi,keycode);
|
||
|
if (pAction->btn.count>0) {
|
||
|
nClicks= pAction->btn.count;
|
||
|
for (i=0;i<nClicks;i++) {
|
||
|
XkbDDXFakePointerButton(ButtonPress,button);
|
||
|
XkbDDXFakePointerButton(ButtonRelease,button);
|
||
|
}
|
||
|
filter->upAction.type= XkbSA_NoAction;
|
||
|
}
|
||
|
else XkbDDXFakePointerButton(ButtonPress,button);
|
||
|
}
|
||
|
break;
|
||
|
case XkbSA_SetPtrDflt:
|
||
|
{
|
||
|
XkbControlsPtr ctrls= xkbi->desc->ctrls;
|
||
|
XkbControlsRec old;
|
||
|
xkbControlsNotify cn;
|
||
|
|
||
|
old= *ctrls;
|
||
|
AccessXCancelRepeatKey(xkbi,keycode);
|
||
|
switch (pAction->dflt.affect) {
|
||
|
case XkbSA_AffectDfltBtn:
|
||
|
if (pAction->dflt.flags&XkbSA_DfltBtnAbsolute)
|
||
|
ctrls->mk_dflt_btn=
|
||
|
XkbSAPtrDfltValue(&pAction->dflt);
|
||
|
else {
|
||
|
ctrls->mk_dflt_btn+=
|
||
|
XkbSAPtrDfltValue(&pAction->dflt);
|
||
|
if (ctrls->mk_dflt_btn>5)
|
||
|
ctrls->mk_dflt_btn= 5;
|
||
|
else if (ctrls->mk_dflt_btn<1)
|
||
|
ctrls->mk_dflt_btn= 1;
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
ErrorF(
|
||
|
"Attempt to change unknown pointer default (%d) ignored\n",
|
||
|
pAction->dflt.affect);
|
||
|
break;
|
||
|
}
|
||
|
if (XkbComputeControlsNotify(xkbi->device,
|
||
|
&old,xkbi->desc->ctrls,
|
||
|
&cn,False)) {
|
||
|
cn.keycode = keycode;
|
||
|
cn.eventType = KeyPress;
|
||
|
cn.requestMajor = 0;
|
||
|
cn.requestMinor = 0;
|
||
|
XkbSendControlsNotify(xkbi->device,&cn);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
int button= filter->upAction.btn.button;
|
||
|
|
||
|
switch (filter->upAction.type) {
|
||
|
case XkbSA_LockPtrBtn:
|
||
|
if (((filter->upAction.btn.flags&XkbSA_LockNoUnlock)!=0)||
|
||
|
((xkbi->lockedPtrButtons&(1<<button))==0)) {
|
||
|
break;
|
||
|
}
|
||
|
xkbi->lockedPtrButtons&= ~(1<<button);
|
||
|
case XkbSA_PtrBtn:
|
||
|
XkbDDXFakePointerButton(ButtonRelease,button);
|
||
|
break;
|
||
|
}
|
||
|
filter->active = 0;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterControls( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
XkbControlsRec old;
|
||
|
XkbControlsPtr ctrls;
|
||
|
DeviceIntPtr kbd;
|
||
|
unsigned int change;
|
||
|
XkbEventCauseRec cause;
|
||
|
|
||
|
kbd= xkbi->device;
|
||
|
ctrls= xkbi->desc->ctrls;
|
||
|
old= *ctrls;
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
change= XkbActionCtrls(&pAction->ctrls);
|
||
|
filter->priv = change;
|
||
|
filter->filter = _XkbFilterControls;
|
||
|
filter->upAction = *pAction;
|
||
|
|
||
|
if (pAction->type==XkbSA_LockControls) {
|
||
|
filter->priv= (ctrls->enabled_ctrls&change);
|
||
|
change&= ~ctrls->enabled_ctrls;
|
||
|
}
|
||
|
|
||
|
if (change) {
|
||
|
xkbControlsNotify cn;
|
||
|
XkbSrvLedInfoPtr sli;
|
||
|
|
||
|
ctrls->enabled_ctrls|= change;
|
||
|
if (XkbComputeControlsNotify(kbd,&old,ctrls,&cn,False)) {
|
||
|
cn.keycode = keycode;
|
||
|
cn.eventType = KeyPress;
|
||
|
cn.requestMajor = 0;
|
||
|
cn.requestMinor = 0;
|
||
|
XkbSendControlsNotify(kbd,&cn);
|
||
|
}
|
||
|
|
||
|
XkbSetCauseKey(&cause,keycode,KeyPress);
|
||
|
|
||
|
/* If sticky keys were disabled, clear all locks and latches */
|
||
|
if ((old.enabled_ctrls&XkbStickyKeysMask)&&
|
||
|
(!(ctrls->enabled_ctrls&XkbStickyKeysMask))) {
|
||
|
XkbClearAllLatchesAndLocks(kbd,xkbi,False,&cause);
|
||
|
}
|
||
|
sli= XkbFindSrvLedInfo(kbd,XkbDfltXIClass,XkbDfltXIId,0);
|
||
|
XkbUpdateIndicators(kbd,sli->usesControls,True,NULL,&cause);
|
||
|
if (XkbAX_NeedFeedback(ctrls,XkbAX_FeatureFBMask))
|
||
|
XkbDDXAccessXBeep(kbd,_BEEP_FEATURE_ON,change);
|
||
|
}
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
change= filter->priv;
|
||
|
if (change) {
|
||
|
xkbControlsNotify cn;
|
||
|
XkbSrvLedInfoPtr sli;
|
||
|
|
||
|
ctrls->enabled_ctrls&= ~change;
|
||
|
if (XkbComputeControlsNotify(kbd,&old,ctrls,&cn,False)) {
|
||
|
cn.keycode = keycode;
|
||
|
cn.eventType = KeyRelease;
|
||
|
cn.requestMajor = 0;
|
||
|
cn.requestMinor = 0;
|
||
|
XkbSendControlsNotify(kbd,&cn);
|
||
|
}
|
||
|
|
||
|
XkbSetCauseKey(&cause,keycode,KeyRelease);
|
||
|
/* If sticky keys were disabled, clear all locks and latches */
|
||
|
if ((old.enabled_ctrls&XkbStickyKeysMask)&&
|
||
|
(!(ctrls->enabled_ctrls&XkbStickyKeysMask))) {
|
||
|
XkbClearAllLatchesAndLocks(kbd,xkbi,False,&cause);
|
||
|
}
|
||
|
sli= XkbFindSrvLedInfo(kbd,XkbDfltXIClass,XkbDfltXIId,0);
|
||
|
XkbUpdateIndicators(kbd,sli->usesControls,True,NULL,&cause);
|
||
|
if (XkbAX_NeedFeedback(ctrls,XkbAX_FeatureFBMask))
|
||
|
XkbDDXAccessXBeep(kbd,_BEEP_FEATURE_OFF,change);
|
||
|
}
|
||
|
filter->keycode= 0;
|
||
|
filter->active= 0;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterActionMessage(XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
XkbMessageAction * pMsg;
|
||
|
DeviceIntPtr kbd;
|
||
|
|
||
|
kbd= xkbi->device;
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
pMsg= &pAction->msg;
|
||
|
if ((pMsg->flags&XkbSA_MessageOnRelease)||
|
||
|
((pMsg->flags&XkbSA_MessageGenKeyEvent)==0)) {
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->priv = 0;
|
||
|
filter->filter = _XkbFilterActionMessage;
|
||
|
filter->upAction = *pAction;
|
||
|
}
|
||
|
if (pMsg->flags&XkbSA_MessageOnPress) {
|
||
|
xkbActionMessage msg;
|
||
|
|
||
|
msg.keycode= keycode;
|
||
|
msg.press= 1;
|
||
|
msg.keyEventFollows=((pMsg->flags&XkbSA_MessageGenKeyEvent)!=0);
|
||
|
memcpy((char *)msg.message,
|
||
|
(char *)pMsg->message,XkbActionMessageLength);
|
||
|
XkbSendActionMessage(kbd,&msg);
|
||
|
}
|
||
|
return ((pAction->msg.flags&XkbSA_MessageGenKeyEvent)!=0);
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
pMsg= &filter->upAction.msg;
|
||
|
if (pMsg->flags&XkbSA_MessageOnRelease) {
|
||
|
xkbActionMessage msg;
|
||
|
|
||
|
msg.keycode= keycode;
|
||
|
msg.press= 0;
|
||
|
msg.keyEventFollows=((pMsg->flags&XkbSA_MessageGenKeyEvent)!=0);
|
||
|
memcpy((char *)msg.message,(char *)pMsg->message,
|
||
|
XkbActionMessageLength);
|
||
|
XkbSendActionMessage(kbd,&msg);
|
||
|
}
|
||
|
filter->keycode= 0;
|
||
|
filter->active= 0;
|
||
|
return ((pMsg->flags&XkbSA_MessageGenKeyEvent)!=0);
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterRedirectKey( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
unsigned realMods;
|
||
|
xEvent ev;
|
||
|
int x,y;
|
||
|
XkbStateRec old;
|
||
|
unsigned mods,mask,oldCoreState = 0,oldCorePrevState = 0;
|
||
|
xkbDeviceInfoPtr xkbPrivPtr = XKBDEVICEINFO(xkbi->device);
|
||
|
|
||
|
if ((filter->keycode!=0)&&(filter->keycode!=keycode))
|
||
|
return 1;
|
||
|
|
||
|
GetSpritePosition(&x,&y);
|
||
|
ev.u.keyButtonPointer.time = GetTimeInMillis();
|
||
|
ev.u.keyButtonPointer.rootX = x;
|
||
|
ev.u.keyButtonPointer.rootY = y;
|
||
|
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
if ((pAction->redirect.new_key<xkbi->desc->min_key_code)||
|
||
|
(pAction->redirect.new_key>xkbi->desc->max_key_code)) {
|
||
|
return 1;
|
||
|
}
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->priv = 0;
|
||
|
filter->filter = _XkbFilterRedirectKey;
|
||
|
filter->upAction = *pAction;
|
||
|
|
||
|
ev.u.u.type = KeyPress;
|
||
|
ev.u.u.detail = pAction->redirect.new_key;
|
||
|
|
||
|
mask= XkbSARedirectVModsMask(&pAction->redirect);
|
||
|
mods= XkbSARedirectVMods(&pAction->redirect);
|
||
|
if (mask) XkbVirtualModsToReal(xkbi->desc,mask,&mask);
|
||
|
if (mods) XkbVirtualModsToReal(xkbi->desc,mods,&mods);
|
||
|
mask|= pAction->redirect.mods_mask;
|
||
|
mods|= pAction->redirect.mods;
|
||
|
|
||
|
if ( mask || mods ) {
|
||
|
old= xkbi->state;
|
||
|
oldCoreState= xkbi->device->key->state;
|
||
|
oldCorePrevState= xkbi->device->key->prev_state;
|
||
|
xkbi->state.base_mods&= ~mask;
|
||
|
xkbi->state.base_mods|= (mods&mask);
|
||
|
xkbi->state.latched_mods&= ~mask;
|
||
|
xkbi->state.latched_mods|= (mods&mask);
|
||
|
xkbi->state.locked_mods&= ~mask;
|
||
|
xkbi->state.locked_mods|= (mods&mask);
|
||
|
XkbComputeDerivedState(xkbi);
|
||
|
xkbi->device->key->state= xkbi->device->key->prev_state=
|
||
|
xkbi->state.mods;
|
||
|
}
|
||
|
|
||
|
realMods = xkbi->device->key->modifierMap[ev.u.u.detail];
|
||
|
xkbi->device->key->modifierMap[ev.u.u.detail] = 0;
|
||
|
UNWRAP_PROCESS_INPUT_PROC(xkbi->device,xkbPrivPtr);
|
||
|
xkbi->device->public.processInputProc(&ev,xkbi->device,1);
|
||
|
COND_WRAP_PROCESS_INPUT_PROC(xkbi->device, xkbPrivPtr,
|
||
|
ProcessKeyboardEvent,xkbUnwrapProc);
|
||
|
xkbi->device->key->modifierMap[ev.u.u.detail] = realMods;
|
||
|
|
||
|
if ( mask || mods ) {
|
||
|
xkbi->device->key->state= oldCoreState;
|
||
|
xkbi->device->key->prev_state= oldCorePrevState;
|
||
|
xkbi->state= old;
|
||
|
}
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
|
||
|
ev.u.u.type = KeyRelease;
|
||
|
ev.u.u.detail = filter->upAction.redirect.new_key;
|
||
|
|
||
|
mask= XkbSARedirectVModsMask(&filter->upAction.redirect);
|
||
|
mods= XkbSARedirectVMods(&filter->upAction.redirect);
|
||
|
if (mask) XkbVirtualModsToReal(xkbi->desc,mask,&mask);
|
||
|
if (mods) XkbVirtualModsToReal(xkbi->desc,mods,&mods);
|
||
|
mask|= filter->upAction.redirect.mods_mask;
|
||
|
mods|= filter->upAction.redirect.mods;
|
||
|
|
||
|
if ( mask || mods ) {
|
||
|
old= xkbi->state;
|
||
|
oldCoreState= xkbi->device->key->state;
|
||
|
oldCorePrevState= xkbi->device->key->prev_state;
|
||
|
xkbi->state.base_mods&= ~mask;
|
||
|
xkbi->state.base_mods|= (mods&mask);
|
||
|
xkbi->state.latched_mods&= ~mask;
|
||
|
xkbi->state.latched_mods|= (mods&mask);
|
||
|
xkbi->state.locked_mods&= ~mask;
|
||
|
xkbi->state.locked_mods|= (mods&mask);
|
||
|
XkbComputeDerivedState(xkbi);
|
||
|
xkbi->device->key->state= xkbi->device->key->prev_state=
|
||
|
xkbi->state.mods;
|
||
|
}
|
||
|
|
||
|
realMods = xkbi->device->key->modifierMap[ev.u.u.detail];
|
||
|
xkbi->device->key->modifierMap[ev.u.u.detail] = 0;
|
||
|
UNWRAP_PROCESS_INPUT_PROC(xkbi->device,xkbPrivPtr);
|
||
|
xkbi->device->public.processInputProc(&ev,xkbi->device,1);
|
||
|
COND_WRAP_PROCESS_INPUT_PROC(xkbi->device, xkbPrivPtr,
|
||
|
ProcessKeyboardEvent,xkbUnwrapProc);
|
||
|
xkbi->device->key->modifierMap[ev.u.u.detail] = realMods;
|
||
|
|
||
|
if ( mask || mods ) {
|
||
|
xkbi->device->key->state= oldCoreState;
|
||
|
xkbi->device->key->prev_state= oldCorePrevState;
|
||
|
xkbi->state= old;
|
||
|
}
|
||
|
|
||
|
filter->keycode= 0;
|
||
|
filter->active= 0;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterSwitchScreen( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
DeviceIntPtr dev = xkbi->device;
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->filter = _XkbFilterSwitchScreen;
|
||
|
AccessXCancelRepeatKey(xkbi, keycode);
|
||
|
XkbDDXSwitchScreen(dev,keycode,pAction);
|
||
|
return 0;
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
filter->active= 0;
|
||
|
return 0;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbFilterXF86Private( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
DeviceIntPtr dev = xkbi->device;
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->filter = _XkbFilterXF86Private;
|
||
|
XkbDDXPrivate(dev,keycode,pAction);
|
||
|
return 0;
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
filter->active= 0;
|
||
|
return 0;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
#ifdef XINPUT
|
||
|
|
||
|
static int
|
||
|
_XkbFilterDeviceBtn( XkbSrvInfoPtr xkbi,
|
||
|
XkbFilterPtr filter,
|
||
|
unsigned keycode,
|
||
|
XkbAction * pAction)
|
||
|
{
|
||
|
DeviceIntPtr dev;
|
||
|
int button;
|
||
|
|
||
|
if (filter->keycode==0) { /* initial press */
|
||
|
dev= _XkbLookupButtonDevice(pAction->devbtn.device,NULL);
|
||
|
if ((!dev)||(!dev->public.on)||(&dev->public==LookupPointerDevice()))
|
||
|
return 1;
|
||
|
|
||
|
button= pAction->devbtn.button;
|
||
|
if ((button<1)||(button>dev->button->numButtons))
|
||
|
return 1;
|
||
|
|
||
|
filter->keycode = keycode;
|
||
|
filter->active = 1;
|
||
|
filter->filterOthers = 0;
|
||
|
filter->priv=0;
|
||
|
filter->filter = _XkbFilterDeviceBtn;
|
||
|
filter->upAction= *pAction;
|
||
|
switch (pAction->type) {
|
||
|
case XkbSA_LockDeviceBtn:
|
||
|
if ((pAction->devbtn.flags&XkbSA_LockNoLock)||
|
||
|
(dev->button->down[button/8]&(1L<<(button%8))))
|
||
|
return 0;
|
||
|
XkbDDXFakeDeviceButton(dev,True,button);
|
||
|
filter->upAction.type= XkbSA_NoAction;
|
||
|
break;
|
||
|
case XkbSA_DeviceBtn:
|
||
|
if (pAction->devbtn.count>0) {
|
||
|
int nClicks,i;
|
||
|
nClicks= pAction->btn.count;
|
||
|
for (i=0;i<nClicks;i++) {
|
||
|
XkbDDXFakeDeviceButton(dev,True,button);
|
||
|
XkbDDXFakeDeviceButton(dev,False,button);
|
||
|
}
|
||
|
filter->upAction.type= XkbSA_NoAction;
|
||
|
}
|
||
|
else XkbDDXFakeDeviceButton(dev,True,button);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
else if (filter->keycode==keycode) {
|
||
|
int button;
|
||
|
|
||
|
filter->active= 0;
|
||
|
dev= _XkbLookupButtonDevice(filter->upAction.devbtn.device,NULL);
|
||
|
if ((!dev)||(!dev->public.on)||(&dev->public==LookupPointerDevice()))
|
||
|
return 1;
|
||
|
|
||
|
button= filter->upAction.btn.button;
|
||
|
switch (filter->upAction.type) {
|
||
|
case XkbSA_LockDeviceBtn:
|
||
|
if ((filter->upAction.devbtn.flags&XkbSA_LockNoUnlock)||
|
||
|
((dev->button->down[button/8]&(1L<<(button%8)))==0))
|
||
|
return 0;
|
||
|
XkbDDXFakeDeviceButton(dev,False,button);
|
||
|
break;
|
||
|
case XkbSA_DeviceBtn:
|
||
|
XkbDDXFakeDeviceButton(dev,False,button);
|
||
|
break;
|
||
|
}
|
||
|
filter->active = 0;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int szFilters = 0;
|
||
|
static XkbFilterPtr filters = NULL;
|
||
|
|
||
|
static XkbFilterPtr
|
||
|
_XkbNextFreeFilter(
|
||
|
void
|
||
|
)
|
||
|
{
|
||
|
register int i;
|
||
|
|
||
|
if (szFilters==0) {
|
||
|
szFilters = 4;
|
||
|
filters = _XkbTypedCalloc(szFilters,XkbFilterRec);
|
||
|
/* 6/21/93 (ef) -- XXX! deal with allocation failure */
|
||
|
}
|
||
|
for (i=0;i<szFilters;i++) {
|
||
|
if (!filters[i].active) {
|
||
|
filters[i].keycode = 0;
|
||
|
return &filters[i];
|
||
|
}
|
||
|
}
|
||
|
szFilters*=2;
|
||
|
filters= _XkbTypedRealloc(filters,szFilters,XkbFilterRec);
|
||
|
/* 6/21/93 (ef) -- XXX! deal with allocation failure */
|
||
|
bzero(&filters[szFilters/2],(szFilters/2)*sizeof(XkbFilterRec));
|
||
|
return &filters[szFilters/2];
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
_XkbApplyFilters(XkbSrvInfoPtr xkbi,unsigned kc,XkbAction *pAction)
|
||
|
{
|
||
|
register int i,send;
|
||
|
|
||
|
send= 1;
|
||
|
for (i=0;i<szFilters;i++) {
|
||
|
if ((filters[i].active)&&(filters[i].filter))
|
||
|
send= ((*filters[i].filter)(xkbi,&filters[i],kc,pAction)&&send);
|
||
|
}
|
||
|
return send;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
XkbHandleActions(DeviceIntPtr dev,DeviceIntPtr kbd,xEvent *xE,int count)
|
||
|
{
|
||
|
int key,bit,i;
|
||
|
CARD8 realMods;
|
||
|
XkbSrvInfoPtr xkbi;
|
||
|
KeyClassPtr keyc;
|
||
|
int changed,sendEvent;
|
||
|
Bool genStateNotify;
|
||
|
XkbStateRec oldState;
|
||
|
XkbAction act;
|
||
|
XkbFilterPtr filter;
|
||
|
Bool keyEvent;
|
||
|
Bool pressEvent;
|
||
|
#ifdef XINPUT
|
||
|
Bool xiEvent;
|
||
|
#endif
|
||
|
|
||
|
xkbDeviceInfoPtr xkbPrivPtr = XKBDEVICEINFO(dev);
|
||
|
|
||
|
keyc= kbd->key;
|
||
|
xkbi= keyc->xkbInfo;
|
||
|
key= xE->u.u.detail;
|
||
|
if ((xkbi->flags&_XkbStateNotifyInProgress)==0) {
|
||
|
oldState= xkbi->state;
|
||
|
xkbi->flags|= _XkbStateNotifyInProgress;
|
||
|
genStateNotify= True;
|
||
|
}
|
||
|
else genStateNotify= False;
|
||
|
|
||
|
xkbi->clearMods = xkbi->setMods = 0;
|
||
|
xkbi->groupChange = 0;
|
||
|
|
||
|
sendEvent = 1;
|
||
|
#ifdef XINPUT
|
||
|
keyEvent= ((xE->u.u.type==KeyPress)||(xE->u.u.type==DeviceKeyPress)||
|
||
|
(xE->u.u.type==KeyRelease)||(xE->u.u.type==DeviceKeyRelease));
|
||
|
pressEvent= (xE->u.u.type==KeyPress)||(xE->u.u.type==DeviceKeyPress)||
|
||
|
(xE->u.u.type==ButtonPress)||(xE->u.u.type==DeviceButtonPress);
|
||
|
xiEvent= (xE->u.u.type==DeviceKeyPress)||(xE->u.u.type==DeviceKeyRelease)||
|
||
|
(xE->u.u.type==DeviceButtonPress)||
|
||
|
(xE->u.u.type==DeviceButtonRelease);
|
||
|
#else
|
||
|
keyEvent= (xE->u.u.type==KeyPress)||(xE->u.u.type==KeyRelease);
|
||
|
pressEvent= (xE->u.u.type==KeyPress)||(xE->u.u.type==ButtonPress);
|
||
|
#endif
|
||
|
|
||
|
if (pressEvent) {
|
||
|
if (keyEvent)
|
||
|
act = XkbGetKeyAction(xkbi,&xkbi->state,key);
|
||
|
else {
|
||
|
act = XkbGetButtonAction(kbd,dev,key);
|
||
|
key|= BTN_ACT_FLAG;
|
||
|
}
|
||
|
sendEvent = _XkbApplyFilters(xkbi,key,&act);
|
||
|
if (sendEvent) {
|
||
|
switch (act.type) {
|
||
|
case XkbSA_SetMods:
|
||
|
case XkbSA_SetGroup:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent = _XkbFilterSetState(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_LatchMods:
|
||
|
case XkbSA_LatchGroup:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent=_XkbFilterLatchState(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_LockMods:
|
||
|
case XkbSA_LockGroup:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent=_XkbFilterLockState(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_ISOLock:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent=_XkbFilterISOLock(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_MovePtr:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent= _XkbFilterPointerMove(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_PtrBtn:
|
||
|
case XkbSA_LockPtrBtn:
|
||
|
case XkbSA_SetPtrDflt:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent= _XkbFilterPointerBtn(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_Terminate:
|
||
|
sendEvent= XkbDDXTerminateServer(dev,key,&act);
|
||
|
break;
|
||
|
case XkbSA_SwitchScreen:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent=_XkbFilterSwitchScreen(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_SetControls:
|
||
|
case XkbSA_LockControls:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent=_XkbFilterControls(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_ActionMessage:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent=_XkbFilterActionMessage(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
case XkbSA_RedirectKey:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent= _XkbFilterRedirectKey(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
#ifdef XINPUT
|
||
|
case XkbSA_DeviceBtn:
|
||
|
case XkbSA_LockDeviceBtn:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent= _XkbFilterDeviceBtn(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
#endif
|
||
|
case XkbSA_XFree86Private:
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
sendEvent= _XkbFilterXF86Private(xkbi,filter,key,&act);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
if (!keyEvent)
|
||
|
key|= BTN_ACT_FLAG;
|
||
|
sendEvent = _XkbApplyFilters(xkbi,key,NULL);
|
||
|
}
|
||
|
|
||
|
if (xkbi->groupChange!=0)
|
||
|
xkbi->state.base_group+= xkbi->groupChange;
|
||
|
if (xkbi->setMods) {
|
||
|
for (i=0,bit=1; xkbi->setMods; i++,bit<<=1 ) {
|
||
|
if (xkbi->setMods&bit) {
|
||
|
keyc->modifierKeyCount[i]++;
|
||
|
xkbi->state.base_mods|= bit;
|
||
|
xkbi->setMods&= ~bit;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (xkbi->clearMods) {
|
||
|
for (i=0,bit=1; xkbi->clearMods; i++,bit<<=1 ) {
|
||
|
if (xkbi->clearMods&bit) {
|
||
|
keyc->modifierKeyCount[i]--;
|
||
|
if (keyc->modifierKeyCount[i]<=0) {
|
||
|
xkbi->state.base_mods&= ~bit;
|
||
|
keyc->modifierKeyCount[i] = 0;
|
||
|
}
|
||
|
xkbi->clearMods&= ~bit;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (sendEvent) {
|
||
|
#ifdef XINPUT
|
||
|
if (xiEvent)
|
||
|
ProcessOtherEvent(xE,dev,count);
|
||
|
else
|
||
|
#endif
|
||
|
if (keyEvent) {
|
||
|
realMods = keyc->modifierMap[key];
|
||
|
keyc->modifierMap[key] = 0;
|
||
|
UNWRAP_PROCESS_INPUT_PROC(dev,xkbPrivPtr);
|
||
|
dev->public.processInputProc(xE,dev,count);
|
||
|
COND_WRAP_PROCESS_INPUT_PROC(dev, xkbPrivPtr,
|
||
|
ProcessKeyboardEvent,xkbUnwrapProc);
|
||
|
keyc->modifierMap[key] = realMods;
|
||
|
}
|
||
|
else CoreProcessPointerEvent(xE,dev,count);
|
||
|
}
|
||
|
else if (keyEvent)
|
||
|
FixKeyState(xE,dev);
|
||
|
|
||
|
xkbi->prev_state= oldState;
|
||
|
XkbComputeDerivedState(xkbi);
|
||
|
keyc->prev_state= keyc->state;
|
||
|
keyc->state= XkbStateFieldFromRec(&xkbi->state);
|
||
|
changed = XkbStateChangedFlags(&oldState,&xkbi->state);
|
||
|
if (genStateNotify) {
|
||
|
if (changed) {
|
||
|
xkbStateNotify sn;
|
||
|
sn.keycode= key;
|
||
|
sn.eventType= xE->u.u.type;
|
||
|
sn.requestMajor = sn.requestMinor = 0;
|
||
|
sn.changed= changed;
|
||
|
XkbSendStateNotify(dev,&sn);
|
||
|
}
|
||
|
xkbi->flags&= ~_XkbStateNotifyInProgress;
|
||
|
}
|
||
|
changed= XkbIndicatorsToUpdate(dev,changed,False);
|
||
|
if (changed) {
|
||
|
XkbEventCauseRec cause;
|
||
|
XkbSetCauseKey(&cause,key,xE->u.u.type);
|
||
|
XkbUpdateIndicators(dev,changed,True,NULL,&cause);
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
XkbLatchModifiers(DeviceIntPtr pXDev,CARD8 mask,CARD8 latches)
|
||
|
{
|
||
|
XkbSrvInfoPtr xkbi;
|
||
|
XkbFilterPtr filter;
|
||
|
XkbAction act;
|
||
|
unsigned clear;
|
||
|
|
||
|
if ( pXDev && pXDev->key && pXDev->key->xkbInfo ) {
|
||
|
xkbi = pXDev->key->xkbInfo;
|
||
|
clear= (mask&(~latches));
|
||
|
xkbi->state.latched_mods&= ~clear;
|
||
|
/* Clear any pending latch to locks.
|
||
|
*/
|
||
|
act.type = XkbSA_NoAction;
|
||
|
_XkbApplyFilters(xkbi,SYNTHETIC_KEYCODE,&act);
|
||
|
act.type = XkbSA_LatchMods;
|
||
|
act.mods.flags = 0;
|
||
|
act.mods.mask = mask&latches;
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
_XkbFilterLatchState(xkbi,filter,SYNTHETIC_KEYCODE,&act);
|
||
|
_XkbFilterLatchState(xkbi,filter,SYNTHETIC_KEYCODE,(XkbAction *)NULL);
|
||
|
return Success;
|
||
|
}
|
||
|
return BadValue;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
XkbLatchGroup(DeviceIntPtr pXDev,int group)
|
||
|
{
|
||
|
XkbSrvInfoPtr xkbi;
|
||
|
XkbFilterPtr filter;
|
||
|
XkbAction act;
|
||
|
|
||
|
if ( pXDev && pXDev->key && pXDev->key->xkbInfo ) {
|
||
|
xkbi = pXDev->key->xkbInfo;
|
||
|
act.type = XkbSA_LatchGroup;
|
||
|
act.group.flags = 0;
|
||
|
XkbSASetGroup(&act.group,group);
|
||
|
filter = _XkbNextFreeFilter();
|
||
|
_XkbFilterLatchState(xkbi,filter,SYNTHETIC_KEYCODE,&act);
|
||
|
_XkbFilterLatchState(xkbi,filter,SYNTHETIC_KEYCODE,(XkbAction *)NULL);
|
||
|
return Success;
|
||
|
}
|
||
|
return BadValue;
|
||
|
}
|
||
|
|
||
|
/***====================================================================***/
|
||
|
|
||
|
void
|
||
|
XkbClearAllLatchesAndLocks( DeviceIntPtr dev,
|
||
|
XkbSrvInfoPtr xkbi,
|
||
|
Bool genEv,
|
||
|
XkbEventCausePtr cause)
|
||
|
{
|
||
|
XkbStateRec os;
|
||
|
xkbStateNotify sn;
|
||
|
|
||
|
sn.changed= 0;
|
||
|
os= xkbi->state;
|
||
|
if (os.latched_mods) { /* clear all latches */
|
||
|
XkbLatchModifiers(dev,~0,0);
|
||
|
sn.changed|= XkbModifierLatchMask;
|
||
|
}
|
||
|
if (os.latched_group) {
|
||
|
XkbLatchGroup(dev,0);
|
||
|
sn.changed|= XkbGroupLatchMask;
|
||
|
}
|
||
|
if (os.locked_mods) {
|
||
|
xkbi->state.locked_mods= 0;
|
||
|
sn.changed|= XkbModifierLockMask;
|
||
|
}
|
||
|
if (os.locked_group) {
|
||
|
xkbi->state.locked_group= 0;
|
||
|
sn.changed|= XkbGroupLockMask;
|
||
|
}
|
||
|
if ( genEv && sn.changed) {
|
||
|
CARD32 changed;
|
||
|
|
||
|
XkbComputeDerivedState(xkbi);
|
||
|
sn.keycode= cause->kc;
|
||
|
sn.eventType= cause->event;
|
||
|
sn.requestMajor= cause->mjr;
|
||
|
sn.requestMinor= cause->mnr;
|
||
|
sn.changed= XkbStateChangedFlags(&os,&xkbi->state);
|
||
|
XkbSendStateNotify(dev,&sn);
|
||
|
changed= XkbIndicatorsToUpdate(dev,sn.changed,False);
|
||
|
if (changed) {
|
||
|
XkbUpdateIndicators(dev,changed,True,NULL,cause);
|
||
|
}
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|