TMgrab.c revision 0568f49b
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    }
1050568f49bSmrg    careOn = (careOn | (Modifiers)modMatch->modifiers);
1060568f49bSmrg    careMask = (careMask | (Modifiers)modMatch->modifierMask);
107444c061aSmrg
1080568f49bSmrg    keycodes = NULL;
109444c061aSmrg    XtKeysymToKeycodeList(
110444c061aSmrg	    dpy,
111444c061aSmrg	    (KeySym)typeMatch->eventCode,
112444c061aSmrg	    &keycodes,
113444c061aSmrg	    &keycount
114444c061aSmrg			 );
1150568f49bSmrg    if (keycount == 0) {
1160568f49bSmrg	XtFree((char *)keycodes);
1170568f49bSmrg	return;
1180568f49bSmrg    }
119444c061aSmrg    for (keycodeP = keycodes; keycount--; keycodeP++) {
120444c061aSmrg	if (modMatch->standard) {
121444c061aSmrg	    /* find standard modifiers that produce this keysym */
122444c061aSmrg	    KeySym keysym;
123444c061aSmrg	    int std_mods, least_mod;
124444c061aSmrg	    Modifiers modifiers_return;
125444c061aSmrg	    XtTranslateKeycode( dpy, *keycodeP, (Modifiers)0,
126444c061aSmrg			        &modifiers_return, &keysym );
1270568f49bSmrg	    if (careOn & modifiers_return) {
1280568f49bSmrg		XtFree((char *)keycodes);
129444c061aSmrg		return;
1300568f49bSmrg	    }
131444c061aSmrg	    if (keysym == typeMatch->eventCode) {
132444c061aSmrg		XtGrabKey(widget, *keycodeP, careOn,
133444c061aSmrg			  grabP->owner_events,
134444c061aSmrg			  grabP->pointer_mode,
135444c061aSmrg			  grabP->keyboard_mode
136444c061aSmrg			);
137444c061aSmrg		/* continue; */		/* grab all modifier combinations */
138444c061aSmrg	    }
1390568f49bSmrg	    least_mod = (int) (modifiers_return & (~modifiers_return + 1));
1400568f49bSmrg	    for (std_mods = (int) modifiers_return;
141444c061aSmrg		 std_mods >= least_mod; std_mods--) {
142444c061aSmrg		Modifiers dummy;
143444c061aSmrg		 /* check all useful combinations of modifier bits */
1440568f49bSmrg		if ((modifiers_return & (Modifiers)std_mods) &&
1450568f49bSmrg		    !(~modifiers_return & (Modifiers)std_mods)) {
146444c061aSmrg		    XtTranslateKeycode( dpy, *keycodeP,
147444c061aSmrg					(Modifiers)std_mods,
148444c061aSmrg					&dummy, &keysym );
149444c061aSmrg		    if (keysym == typeMatch->eventCode) {
150444c061aSmrg			XtGrabKey(widget, *keycodeP,
151444c061aSmrg				  careOn | (Modifiers) std_mods,
152444c061aSmrg				  grabP->owner_events,
153444c061aSmrg				  grabP->pointer_mode,
154444c061aSmrg				  grabP->keyboard_mode
155444c061aSmrg				);
156444c061aSmrg			/* break; */	/* grab all modifier combinations */
157444c061aSmrg		    }
158444c061aSmrg		}
159444c061aSmrg	    }
160444c061aSmrg	} else /* !event->standard */ {
161444c061aSmrg	    XtGrabKey(widget, *keycodeP, careOn,
162444c061aSmrg		      grabP->owner_events,
163444c061aSmrg		      grabP->pointer_mode,
164444c061aSmrg		      grabP->keyboard_mode
165444c061aSmrg		    );
166444c061aSmrg	}
167444c061aSmrg    }
168444c061aSmrg    XtFree((char *)keycodes);
169444c061aSmrg}
170444c061aSmrg
171444c061aSmrgtypedef struct {
172444c061aSmrg    TMShortCard count;
173444c061aSmrg    Widget	widget;
174444c061aSmrg    GrabActionRec *grabP;
175444c061aSmrg}DoGrabRec;
176444c061aSmrg
177444c061aSmrgstatic Boolean DoGrab(
178444c061aSmrg    StatePtr		state,
179444c061aSmrg    XtPointer		data)
180444c061aSmrg{
181444c061aSmrg    DoGrabRec		*doGrabP = (DoGrabRec *)data;
182444c061aSmrg    GrabActionRec* 	grabP = doGrabP->grabP;
183444c061aSmrg    Widget		widget = doGrabP->widget;
184444c061aSmrg    TMShortCard		count = doGrabP->count;
185444c061aSmrg    TMShortCard		typeIndex = state->typeIndex;
186444c061aSmrg    TMShortCard		modIndex = state->modIndex;
187444c061aSmrg    ActionRec		*action;
188444c061aSmrg    TMTypeMatch		typeMatch;
189444c061aSmrg    TMModifierMatch	modMatch;
190444c061aSmrg    Modifiers		careOn = 0;
191444c061aSmrg    Modifiers		careMask = 0;
192444c061aSmrg
193444c061aSmrg    LOCK_PROCESS;
194444c061aSmrg    typeMatch = TMGetTypeMatch(typeIndex);
195444c061aSmrg    modMatch = TMGetModifierMatch(modIndex);
196444c061aSmrg
197444c061aSmrg    for (action = state->actions; action; action = action->next)
198444c061aSmrg      if (count == action->idx) break;
199444c061aSmrg    if (!action) {
200444c061aSmrg	UNLOCK_PROCESS;
201444c061aSmrg	return False;
202444c061aSmrg    }
203444c061aSmrg
204444c061aSmrg    switch (typeMatch->eventType) {
205444c061aSmrg      case ButtonPress:
206444c061aSmrg      case ButtonRelease:
207444c061aSmrg	if (modMatch->lateModifiers) {
2080568f49bSmrg	    Boolean resolved = _XtComputeLateBindings(XtDisplay(widget),
2090568f49bSmrg						      modMatch->lateModifiers,
2100568f49bSmrg						      &careOn, &careMask);
211444c061aSmrg	    if (!resolved) break;
212444c061aSmrg	}
2130568f49bSmrg	careOn = (careOn | (Modifiers) modMatch->modifiers);
214444c061aSmrg	XtGrabButton(
215444c061aSmrg		     widget,
2160568f49bSmrg		     (int) typeMatch->eventCode,
217444c061aSmrg		     careOn,
218444c061aSmrg		     grabP->owner_events,
219444c061aSmrg		     grabP->event_mask,
220444c061aSmrg		     grabP->pointer_mode,
221444c061aSmrg		     grabP->keyboard_mode,
222444c061aSmrg		     None,
223444c061aSmrg		     None
224444c061aSmrg		     );
225444c061aSmrg	break;
226444c061aSmrg
227444c061aSmrg      case KeyPress:
228444c061aSmrg      case KeyRelease:
229444c061aSmrg	GrabAllCorrectKeys(widget, typeMatch, modMatch, grabP);
230444c061aSmrg	break;
231444c061aSmrg
232444c061aSmrg      case EnterNotify:
233444c061aSmrg	break;
234444c061aSmrg
235444c061aSmrg      default:
236444c061aSmrg	XtAppWarningMsg(XtWidgetToApplicationContext(widget),
237444c061aSmrg			"invalidPopup","unsupportedOperation",XtCXtToolkitError,
238444c061aSmrg			"Pop-up menu creation is only supported on Button, Key or EnterNotify events.",
2390568f49bSmrg			NULL, NULL);
240444c061aSmrg	break;
241444c061aSmrg    }
242444c061aSmrg    UNLOCK_PROCESS;
243444c061aSmrg    return False;
244444c061aSmrg}
245444c061aSmrg
246444c061aSmrgvoid _XtRegisterGrabs(
247444c061aSmrg    Widget widget)
248444c061aSmrg{
249444c061aSmrg    XtTranslations 	xlations = widget->core.tm.translations;
250444c061aSmrg    TMComplexStateTree 	*stateTreePtr;
251444c061aSmrg    unsigned int 	count;
252444c061aSmrg    TMShortCard		i;
253444c061aSmrg    TMBindData   	bindData = (TMBindData) widget->core.tm.proc_table;
254444c061aSmrg    XtActionProc	*procs;
255444c061aSmrg
256444c061aSmrg    if (! XtIsRealized(widget) || widget->core.being_destroyed)
257444c061aSmrg	return;
258444c061aSmrg
259444c061aSmrg    /* walk the widget instance action bindings table looking for */
260444c061aSmrg    /* actions registered as grab actions. */
261444c061aSmrg    /* when you find one, do a grab on the triggering event */
262444c061aSmrg
263444c061aSmrg    if (xlations == NULL) return;
264444c061aSmrg    stateTreePtr = (TMComplexStateTree *) xlations->stateTreeTbl;
265444c061aSmrg    if (*stateTreePtr == NULL) return;
266444c061aSmrg    for (i = 0; i < xlations->numStateTrees; i++, stateTreePtr++) {
267444c061aSmrg	if (bindData->simple.isComplex)
268444c061aSmrg	  procs = TMGetComplexBindEntry(bindData, i)->procs;
269444c061aSmrg	else
270444c061aSmrg	  procs = TMGetSimpleBindEntry(bindData, i)->procs;
271444c061aSmrg	for (count=0; count < (*stateTreePtr)->numQuarks; count++) {
272444c061aSmrg	    GrabActionRec* grabP;
273444c061aSmrg	    DoGrabRec      doGrab;
274444c061aSmrg
275444c061aSmrg	    LOCK_PROCESS;
276444c061aSmrg	    for (grabP = grabActionList; grabP != NULL; grabP = grabP->next) {
277444c061aSmrg		if (grabP->action_proc == procs[count]) {
278444c061aSmrg		    /* we've found a "grabber" in the action table. Find the
279444c061aSmrg		     * states that call this action.  Note that if there is
280444c061aSmrg		     * more than one "grabber" in the action table, we end
281444c061aSmrg		     * up searching all of the states multiple times.
282444c061aSmrg		     */
283444c061aSmrg		    doGrab.widget = widget;
284444c061aSmrg		    doGrab.grabP = grabP;
2850568f49bSmrg		    doGrab.count = (TMShortCard) count;
286444c061aSmrg		    _XtTraverseStateTree((TMStateTree)*stateTreePtr,
287444c061aSmrg					 DoGrab,
288444c061aSmrg					 (XtPointer)&doGrab);
289444c061aSmrg		}
290444c061aSmrg	    }
291444c061aSmrg	    UNLOCK_PROCESS;
292444c061aSmrg	}
293444c061aSmrg    }
294444c061aSmrg}
295444c061aSmrg
296444c061aSmrgvoid XtRegisterGrabAction(
297444c061aSmrg    XtActionProc action_proc,
298444c061aSmrg    _XtBoolean owner_events,
299444c061aSmrg    unsigned int event_mask,
300444c061aSmrg    int pointer_mode,
301444c061aSmrg    int keyboard_mode)
302444c061aSmrg{
303444c061aSmrg    GrabActionRec* actionP;
304444c061aSmrg
305444c061aSmrg    LOCK_PROCESS;
306444c061aSmrg    for (actionP = grabActionList; actionP != NULL; actionP = actionP->next) {
307444c061aSmrg	if (actionP->action_proc == action_proc) break;
308444c061aSmrg    }
309444c061aSmrg    if (actionP == NULL) {
310444c061aSmrg	actionP = XtNew(GrabActionRec);
311444c061aSmrg	actionP->action_proc = action_proc;
312444c061aSmrg	actionP->next = grabActionList;
313444c061aSmrg	grabActionList = actionP;
314444c061aSmrg    }
315444c061aSmrg#ifdef DEBUG
316444c061aSmrg    else
317444c061aSmrg	if (   actionP->owner_events != owner_events
318444c061aSmrg	    || actionP->event_mask != event_mask
319444c061aSmrg	    || actionP->pointer_mode != pointer_mode
320444c061aSmrg	    || actionP->keyboard_mode != keyboard_mode) {
321444c061aSmrg	    Cardinal n = 0;
322444c061aSmrg	    XtWarningMsg(
323444c061aSmrg		"argsReplaced", "xtRegisterGrabAction", XtCXtToolkitError,
324444c061aSmrg		"XtRegisterGrabAction called on same proc with different args",
325444c061aSmrg			 NULL, &n);
326444c061aSmrg	}
327444c061aSmrg#endif /*DEBUG*/
328444c061aSmrg
3290568f49bSmrg    actionP->owner_events = (Boolean) owner_events;
330444c061aSmrg    actionP->event_mask = event_mask;
331444c061aSmrg    actionP->pointer_mode = pointer_mode;
332444c061aSmrg    actionP->keyboard_mode = keyboard_mode;
333444c061aSmrg    UNLOCK_PROCESS;
334444c061aSmrg}
335444c061aSmrg
336444c061aSmrg/*ARGSUSED*/
337444c061aSmrgvoid _XtGrabInitialize(
338444c061aSmrg    XtAppContext	app)
339444c061aSmrg{
340444c061aSmrg    LOCK_PROCESS;
341444c061aSmrg    if (grabActionList == NULL)
342444c061aSmrg	XtRegisterGrabAction( XtMenuPopupAction, True,
343444c061aSmrg			      (unsigned)(ButtonPressMask | ButtonReleaseMask),
344444c061aSmrg			      GrabModeAsync,
345444c061aSmrg			      GrabModeAsync
346444c061aSmrg			    );
347444c061aSmrg    UNLOCK_PROCESS;
348444c061aSmrg
349444c061aSmrg}
350