TMgrab.c revision 249c3046
1444c061aSmrg/*********************************************************** 2249c3046SmrgCopyright (c) 1993, Oracle and/or its affiliates. All rights reserved. 31477040fSmrg 41477040fSmrgPermission is hereby granted, free of charge, to any person obtaining a 51477040fSmrgcopy of this software and associated documentation files (the "Software"), 61477040fSmrgto deal in the Software without restriction, including without limitation 71477040fSmrgthe rights to use, copy, modify, merge, publish, distribute, sublicense, 81477040fSmrgand/or sell copies of the Software, and to permit persons to whom the 91477040fSmrgSoftware is furnished to do so, subject to the following conditions: 101477040fSmrg 111477040fSmrgThe above copyright notice and this permission notice (including the next 121477040fSmrgparagraph) shall be included in all copies or substantial portions of the 131477040fSmrgSoftware. 141477040fSmrg 151477040fSmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 161477040fSmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 171477040fSmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 181477040fSmrgTHE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 191477040fSmrgLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 201477040fSmrgFROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 211477040fSmrgDEALINGS IN THE SOFTWARE. 221477040fSmrg 231477040fSmrgCopyright 1987, 1988 by Digital Equipment Corporation, Maynard, Massachusetts. 24444c061aSmrg 25444c061aSmrg All Rights Reserved 26444c061aSmrg 27444c061aSmrgPermission to use, copy, modify, and distribute this software and its 28444c061aSmrgdocumentation for any purpose and without fee is hereby granted, 29444c061aSmrgprovided that the above copyright notice appear in all copies and that 30444c061aSmrgboth that copyright notice and this permission notice appear in 311477040fSmrgsupporting documentation, and that the name of Digital not be 32444c061aSmrgused in advertising or publicity pertaining to distribution of the 33444c061aSmrgsoftware without specific, written prior permission. 34444c061aSmrg 35444c061aSmrgDIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING 36444c061aSmrgALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL 37444c061aSmrgDIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR 38444c061aSmrgANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, 39444c061aSmrgWHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, 40444c061aSmrgARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS 41444c061aSmrgSOFTWARE. 42444c061aSmrg 43444c061aSmrg******************************************************************/ 44444c061aSmrg 45444c061aSmrg/* 46444c061aSmrg 47444c061aSmrgCopyright 1987, 1988, 1998 The Open Group 48444c061aSmrg 49444c061aSmrgPermission to use, copy, modify, distribute, and sell this software and its 50444c061aSmrgdocumentation for any purpose is hereby granted without fee, provided that 51444c061aSmrgthe above copyright notice appear in all copies and that both that 52444c061aSmrgcopyright notice and this permission notice appear in supporting 53444c061aSmrgdocumentation. 54444c061aSmrg 55444c061aSmrgThe above copyright notice and this permission notice shall be included in 56444c061aSmrgall copies or substantial portions of the Software. 57444c061aSmrg 58444c061aSmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 59444c061aSmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 60444c061aSmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 61444c061aSmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN 62444c061aSmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 63444c061aSmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 64444c061aSmrg 65444c061aSmrgExcept as contained in this notice, the name of The Open Group shall not be 66444c061aSmrgused in advertising or otherwise to promote the sale, use or other dealings 67444c061aSmrgin this Software without prior written authorization from The Open Group. 68444c061aSmrg 69444c061aSmrg*/ 70444c061aSmrg 71444c061aSmrg/*LINTLIBRARY*/ 72444c061aSmrg#ifdef HAVE_CONFIG_H 73444c061aSmrg#include <config.h> 74444c061aSmrg#endif 75444c061aSmrg#include "IntrinsicI.h" 76444c061aSmrg 77444c061aSmrgtypedef struct _GrabActionRec { 78444c061aSmrg struct _GrabActionRec* next; 79444c061aSmrg XtActionProc action_proc; 80444c061aSmrg Boolean owner_events; 81444c061aSmrg unsigned int event_mask; 82444c061aSmrg int pointer_mode, keyboard_mode; 83444c061aSmrg} GrabActionRec; 84444c061aSmrg 85444c061aSmrgstatic GrabActionRec *grabActionList = NULL; 86444c061aSmrg 87444c061aSmrgstatic void GrabAllCorrectKeys( 88444c061aSmrg Widget widget, 89444c061aSmrg TMTypeMatch typeMatch, 90444c061aSmrg TMModifierMatch modMatch, 91444c061aSmrg GrabActionRec* grabP) 92444c061aSmrg{ 93444c061aSmrg Display *dpy = XtDisplay(widget); 94444c061aSmrg KeyCode *keycodes, *keycodeP; 95444c061aSmrg Cardinal keycount; 96444c061aSmrg Modifiers careOn = 0; 97444c061aSmrg Modifiers careMask = 0; 98444c061aSmrg 99444c061aSmrg if (modMatch->lateModifiers) { 100444c061aSmrg Boolean resolved; 101444c061aSmrg resolved = _XtComputeLateBindings(dpy, modMatch->lateModifiers, 102444c061aSmrg &careOn, &careMask); 103444c061aSmrg if (!resolved) return; 104444c061aSmrg } 105444c061aSmrg careOn |= modMatch->modifiers; 106444c061aSmrg careMask |= modMatch->modifierMask; 107444c061aSmrg 108444c061aSmrg XtKeysymToKeycodeList( 109444c061aSmrg dpy, 110444c061aSmrg (KeySym)typeMatch->eventCode, 111444c061aSmrg &keycodes, 112444c061aSmrg &keycount 113444c061aSmrg ); 114444c061aSmrg if (keycount == 0) return; 115444c061aSmrg for (keycodeP = keycodes; keycount--; keycodeP++) { 116444c061aSmrg if (modMatch->standard) { 117444c061aSmrg /* find standard modifiers that produce this keysym */ 118444c061aSmrg KeySym keysym; 119444c061aSmrg int std_mods, least_mod; 120444c061aSmrg Modifiers modifiers_return; 121444c061aSmrg XtTranslateKeycode( dpy, *keycodeP, (Modifiers)0, 122444c061aSmrg &modifiers_return, &keysym ); 123444c061aSmrg if (careOn & modifiers_return) 124444c061aSmrg return; 125444c061aSmrg if (keysym == typeMatch->eventCode) { 126444c061aSmrg XtGrabKey(widget, *keycodeP, careOn, 127444c061aSmrg grabP->owner_events, 128444c061aSmrg grabP->pointer_mode, 129444c061aSmrg grabP->keyboard_mode 130444c061aSmrg ); 131444c061aSmrg /* continue; */ /* grab all modifier combinations */ 132444c061aSmrg } 133444c061aSmrg least_mod = modifiers_return & (~modifiers_return + 1); 134444c061aSmrg for (std_mods = modifiers_return; 135444c061aSmrg std_mods >= least_mod; std_mods--) { 136444c061aSmrg Modifiers dummy; 137444c061aSmrg /* check all useful combinations of modifier bits */ 138444c061aSmrg if (modifiers_return & std_mods && 139444c061aSmrg !(~modifiers_return & std_mods)) { 140444c061aSmrg XtTranslateKeycode( dpy, *keycodeP, 141444c061aSmrg (Modifiers)std_mods, 142444c061aSmrg &dummy, &keysym ); 143444c061aSmrg if (keysym == typeMatch->eventCode) { 144444c061aSmrg XtGrabKey(widget, *keycodeP, 145444c061aSmrg careOn | (Modifiers) std_mods, 146444c061aSmrg grabP->owner_events, 147444c061aSmrg grabP->pointer_mode, 148444c061aSmrg grabP->keyboard_mode 149444c061aSmrg ); 150444c061aSmrg /* break; */ /* grab all modifier combinations */ 151444c061aSmrg } 152444c061aSmrg } 153444c061aSmrg } 154444c061aSmrg } else /* !event->standard */ { 155444c061aSmrg XtGrabKey(widget, *keycodeP, careOn, 156444c061aSmrg grabP->owner_events, 157444c061aSmrg grabP->pointer_mode, 158444c061aSmrg grabP->keyboard_mode 159444c061aSmrg ); 160444c061aSmrg } 161444c061aSmrg } 162444c061aSmrg XtFree((char *)keycodes); 163444c061aSmrg} 164444c061aSmrg 165444c061aSmrgtypedef struct { 166444c061aSmrg TMShortCard count; 167444c061aSmrg Widget widget; 168444c061aSmrg GrabActionRec *grabP; 169444c061aSmrg}DoGrabRec; 170444c061aSmrg 171444c061aSmrgstatic Boolean DoGrab( 172444c061aSmrg StatePtr state, 173444c061aSmrg XtPointer data) 174444c061aSmrg{ 175444c061aSmrg DoGrabRec *doGrabP = (DoGrabRec *)data; 176444c061aSmrg GrabActionRec* grabP = doGrabP->grabP; 177444c061aSmrg Widget widget = doGrabP->widget; 178444c061aSmrg TMShortCard count = doGrabP->count; 179444c061aSmrg TMShortCard typeIndex = state->typeIndex; 180444c061aSmrg TMShortCard modIndex = state->modIndex; 181444c061aSmrg ActionRec *action; 182444c061aSmrg TMTypeMatch typeMatch; 183444c061aSmrg TMModifierMatch modMatch; 184444c061aSmrg Modifiers careOn = 0; 185444c061aSmrg Modifiers careMask = 0; 186444c061aSmrg Boolean resolved; 187444c061aSmrg 188444c061aSmrg LOCK_PROCESS; 189444c061aSmrg typeMatch = TMGetTypeMatch(typeIndex); 190444c061aSmrg modMatch = TMGetModifierMatch(modIndex); 191444c061aSmrg 192444c061aSmrg for (action = state->actions; action; action = action->next) 193444c061aSmrg if (count == action->idx) break; 194444c061aSmrg if (!action) { 195444c061aSmrg UNLOCK_PROCESS; 196444c061aSmrg return False; 197444c061aSmrg } 198444c061aSmrg 199444c061aSmrg switch (typeMatch->eventType) { 200444c061aSmrg case ButtonPress: 201444c061aSmrg case ButtonRelease: 202444c061aSmrg if (modMatch->lateModifiers) { 203444c061aSmrg resolved = _XtComputeLateBindings(XtDisplay(widget), 204444c061aSmrg modMatch->lateModifiers, 205444c061aSmrg &careOn, &careMask); 206444c061aSmrg if (!resolved) break; 207444c061aSmrg } 208444c061aSmrg careOn |= modMatch->modifiers; 209444c061aSmrg XtGrabButton( 210444c061aSmrg widget, 211444c061aSmrg (unsigned) typeMatch->eventCode, 212444c061aSmrg careOn, 213444c061aSmrg grabP->owner_events, 214444c061aSmrg grabP->event_mask, 215444c061aSmrg grabP->pointer_mode, 216444c061aSmrg grabP->keyboard_mode, 217444c061aSmrg None, 218444c061aSmrg None 219444c061aSmrg ); 220444c061aSmrg break; 221444c061aSmrg 222444c061aSmrg case KeyPress: 223444c061aSmrg case KeyRelease: 224444c061aSmrg GrabAllCorrectKeys(widget, typeMatch, modMatch, grabP); 225444c061aSmrg break; 226444c061aSmrg 227444c061aSmrg case EnterNotify: 228444c061aSmrg break; 229444c061aSmrg 230444c061aSmrg default: 231444c061aSmrg XtAppWarningMsg(XtWidgetToApplicationContext(widget), 232444c061aSmrg "invalidPopup","unsupportedOperation",XtCXtToolkitError, 233444c061aSmrg "Pop-up menu creation is only supported on Button, Key or EnterNotify events.", 234444c061aSmrg (String *)NULL, (Cardinal *)NULL); 235444c061aSmrg break; 236444c061aSmrg } 237444c061aSmrg UNLOCK_PROCESS; 238444c061aSmrg return False; 239444c061aSmrg} 240444c061aSmrg 241444c061aSmrgvoid _XtRegisterGrabs( 242444c061aSmrg Widget widget) 243444c061aSmrg{ 244444c061aSmrg XtTranslations xlations = widget->core.tm.translations; 245444c061aSmrg TMComplexStateTree *stateTreePtr; 246444c061aSmrg unsigned int count; 247444c061aSmrg TMShortCard i; 248444c061aSmrg TMBindData bindData = (TMBindData) widget->core.tm.proc_table; 249444c061aSmrg XtActionProc *procs; 250444c061aSmrg 251444c061aSmrg if (! XtIsRealized(widget) || widget->core.being_destroyed) 252444c061aSmrg return; 253444c061aSmrg 254444c061aSmrg /* walk the widget instance action bindings table looking for */ 255444c061aSmrg /* actions registered as grab actions. */ 256444c061aSmrg /* when you find one, do a grab on the triggering event */ 257444c061aSmrg 258444c061aSmrg if (xlations == NULL) return; 259444c061aSmrg stateTreePtr = (TMComplexStateTree *) xlations->stateTreeTbl; 260444c061aSmrg if (*stateTreePtr == NULL) return; 261444c061aSmrg for (i = 0; i < xlations->numStateTrees; i++, stateTreePtr++) { 262444c061aSmrg if (bindData->simple.isComplex) 263444c061aSmrg procs = TMGetComplexBindEntry(bindData, i)->procs; 264444c061aSmrg else 265444c061aSmrg procs = TMGetSimpleBindEntry(bindData, i)->procs; 266444c061aSmrg for (count=0; count < (*stateTreePtr)->numQuarks; count++) { 267444c061aSmrg GrabActionRec* grabP; 268444c061aSmrg DoGrabRec doGrab; 269444c061aSmrg 270444c061aSmrg LOCK_PROCESS; 271444c061aSmrg for (grabP = grabActionList; grabP != NULL; grabP = grabP->next) { 272444c061aSmrg if (grabP->action_proc == procs[count]) { 273444c061aSmrg /* we've found a "grabber" in the action table. Find the 274444c061aSmrg * states that call this action. Note that if there is 275444c061aSmrg * more than one "grabber" in the action table, we end 276444c061aSmrg * up searching all of the states multiple times. 277444c061aSmrg */ 278444c061aSmrg doGrab.widget = widget; 279444c061aSmrg doGrab.grabP = grabP; 280444c061aSmrg doGrab.count = count; 281444c061aSmrg _XtTraverseStateTree((TMStateTree)*stateTreePtr, 282444c061aSmrg DoGrab, 283444c061aSmrg (XtPointer)&doGrab); 284444c061aSmrg } 285444c061aSmrg } 286444c061aSmrg UNLOCK_PROCESS; 287444c061aSmrg } 288444c061aSmrg } 289444c061aSmrg} 290444c061aSmrg 291444c061aSmrgvoid XtRegisterGrabAction( 292444c061aSmrg XtActionProc action_proc, 293444c061aSmrg _XtBoolean owner_events, 294444c061aSmrg unsigned int event_mask, 295444c061aSmrg int pointer_mode, 296444c061aSmrg int keyboard_mode) 297444c061aSmrg{ 298444c061aSmrg GrabActionRec* actionP; 299444c061aSmrg 300444c061aSmrg LOCK_PROCESS; 301444c061aSmrg for (actionP = grabActionList; actionP != NULL; actionP = actionP->next) { 302444c061aSmrg if (actionP->action_proc == action_proc) break; 303444c061aSmrg } 304444c061aSmrg if (actionP == NULL) { 305444c061aSmrg actionP = XtNew(GrabActionRec); 306444c061aSmrg actionP->action_proc = action_proc; 307444c061aSmrg actionP->next = grabActionList; 308444c061aSmrg grabActionList = actionP; 309444c061aSmrg } 310444c061aSmrg#ifdef DEBUG 311444c061aSmrg else 312444c061aSmrg if ( actionP->owner_events != owner_events 313444c061aSmrg || actionP->event_mask != event_mask 314444c061aSmrg || actionP->pointer_mode != pointer_mode 315444c061aSmrg || actionP->keyboard_mode != keyboard_mode) { 316444c061aSmrg Cardinal n = 0; 317444c061aSmrg XtWarningMsg( 318444c061aSmrg "argsReplaced", "xtRegisterGrabAction", XtCXtToolkitError, 319444c061aSmrg "XtRegisterGrabAction called on same proc with different args", 320444c061aSmrg NULL, &n); 321444c061aSmrg } 322444c061aSmrg#endif /*DEBUG*/ 323444c061aSmrg 324444c061aSmrg actionP->owner_events = owner_events; 325444c061aSmrg actionP->event_mask = event_mask; 326444c061aSmrg actionP->pointer_mode = pointer_mode; 327444c061aSmrg actionP->keyboard_mode = keyboard_mode; 328444c061aSmrg UNLOCK_PROCESS; 329444c061aSmrg} 330444c061aSmrg 331444c061aSmrg/*ARGSUSED*/ 332444c061aSmrgvoid _XtGrabInitialize( 333444c061aSmrg XtAppContext app) 334444c061aSmrg{ 335444c061aSmrg LOCK_PROCESS; 336444c061aSmrg if (grabActionList == NULL) 337444c061aSmrg XtRegisterGrabAction( XtMenuPopupAction, True, 338444c061aSmrg (unsigned)(ButtonPressMask | ButtonReleaseMask), 339444c061aSmrg GrabModeAsync, 340444c061aSmrg GrabModeAsync 341444c061aSmrg ); 342444c061aSmrg UNLOCK_PROCESS; 343444c061aSmrg 344444c061aSmrg} 345