Repeater.c revision 7a84e134
17a84e134Smrg/*
27a84e134Smrg * $Xorg: Repeater.c,v 1.4 2001/02/09 02:03:45 xorgcvs Exp $
37a84e134Smrg *
47a84e134SmrgCopyright 1990, 1994, 1998  The Open Group
57a84e134Smrg
67a84e134SmrgPermission to use, copy, modify, distribute, and sell this software and its
77a84e134Smrgdocumentation for any purpose is hereby granted without fee, provided that
87a84e134Smrgthe above copyright notice appear in all copies and that both that
97a84e134Smrgcopyright notice and this permission notice appear in supporting
107a84e134Smrgdocumentation.
117a84e134Smrg
127a84e134SmrgThe above copyright notice and this permission notice shall be included in
137a84e134Smrgall copies or substantial portions of the Software.
147a84e134Smrg
157a84e134SmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
167a84e134SmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
177a84e134SmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
187a84e134SmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
197a84e134SmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
207a84e134SmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
217a84e134Smrg
227a84e134SmrgExcept as contained in this notice, the name of The Open Group shall not be
237a84e134Smrgused in advertising or otherwise to promote the sale, use or other dealings
247a84e134Smrgin this Software without prior written authorization from The Open Group.
257a84e134Smrg *
267a84e134Smrg * Author:  Jim Fulton, MIT X Consortium
277a84e134Smrg *
287a84e134Smrg * This widget is used for press-and-hold style buttons.
297a84e134Smrg */
307a84e134Smrg/* $XFree86: xc/lib/Xaw/Repeater.c,v 1.6 2001/01/17 19:42:29 dawes Exp $ */
317a84e134Smrg
327a84e134Smrg#ifdef HAVE_CONFIG_H
337a84e134Smrg#include <config.h>
347a84e134Smrg#endif
357a84e134Smrg#include <X11/IntrinsicP.h>
367a84e134Smrg#include <X11/StringDefs.h>
377a84e134Smrg#include <X11/Xaw/RepeaterP.h>
387a84e134Smrg#include <X11/Xaw/XawInit.h>
397a84e134Smrg
407a84e134Smrg#define DO_CALLBACK(rw) \
417a84e134SmrgXtCallCallbackList((Widget)rw, rw->command.callbacks, NULL)
427a84e134Smrg
437a84e134Smrg#define ADD_TIMEOUT(rw, delay)					\
447a84e134SmrgXtAppAddTimeOut(XtWidgetToApplicationContext((Widget)rw),	\
457a84e134Smrg		delay, tic, (XtPointer)rw)
467a84e134Smrg
477a84e134Smrg#define CLEAR_TIMEOUT(rw) \
487a84e134Smrgif ((rw)->repeater.timer) {			\
497a84e134Smrg    XtRemoveTimeOut((rw)->repeater.timer);	\
507a84e134Smrg    (rw)->repeater.timer = 0;			\
517a84e134Smrg}
527a84e134Smrg
537a84e134Smrg/*
547a84e134Smrg * Class Methods
557a84e134Smrg */
567a84e134Smrgstatic void XawRepeaterInitialize(Widget, Widget, ArgList, Cardinal*);
577a84e134Smrgstatic void XawRepeaterDestroy(Widget);
587a84e134Smrgstatic Boolean XawRepeaterSetValues(Widget, Widget, Widget,
597a84e134Smrg				    ArgList, Cardinal*);
607a84e134Smrg
617a84e134Smrg/*
627a84e134Smrg * Prototypes
637a84e134Smrg */
647a84e134Smrgstatic void tic(XtPointer, XtIntervalId*);
657a84e134Smrg
667a84e134Smrg/*
677a84e134Smrg * Actions
687a84e134Smrg */
697a84e134Smrgstatic void ActionStart(Widget, XEvent*, String*, Cardinal*);
707a84e134Smrgstatic void ActionStop(Widget, XEvent*, String*, Cardinal*);
717a84e134Smrg
727a84e134Smrg/*
737a84e134Smrg * Initialization
747a84e134Smrg */
757a84e134Smrgstatic char defaultTranslations[] =
767a84e134Smrg"<Enter>:"	"highlight()\n"
777a84e134Smrg"<Leave>:"	"unhighlight()\n"
787a84e134Smrg"<Btn1Down>:"	"set() start()\n"
797a84e134Smrg"<Btn1Up>:"	"stop() unset()\n"
807a84e134Smrg;
817a84e134Smrg
827a84e134Smrgstatic XtActionsRec actions[] = {
837a84e134Smrg  {"start",	ActionStart},
847a84e134Smrg  {"stop",	ActionStop},
857a84e134Smrg};
867a84e134Smrg
877a84e134Smrg#define offset(field)	XtOffsetOf(RepeaterRec, repeater.field)
887a84e134Smrgstatic XtResource resources[] = {
897a84e134Smrg  {
907a84e134Smrg    XtNdecay,
917a84e134Smrg    XtCDecay,
927a84e134Smrg    XtRInt,
937a84e134Smrg    sizeof(int),
947a84e134Smrg    offset(decay),
957a84e134Smrg    XtRImmediate,
967a84e134Smrg    (XtPointer)REP_DEF_DECAY
977a84e134Smrg  },
987a84e134Smrg  {
997a84e134Smrg    XtNinitialDelay,
1007a84e134Smrg    XtCDelay,
1017a84e134Smrg    XtRInt,
1027a84e134Smrg    sizeof(int),
1037a84e134Smrg    offset(initial_delay),
1047a84e134Smrg    XtRImmediate,
1057a84e134Smrg    (XtPointer)REP_DEF_INITIAL_DELAY
1067a84e134Smrg  },
1077a84e134Smrg  {
1087a84e134Smrg    XtNminimumDelay,
1097a84e134Smrg    XtCMinimumDelay,
1107a84e134Smrg    XtRInt,
1117a84e134Smrg    sizeof(int),
1127a84e134Smrg    offset(minimum_delay),
1137a84e134Smrg    XtRImmediate,
1147a84e134Smrg    (XtPointer)REP_DEF_MINIMUM_DELAY
1157a84e134Smrg  },
1167a84e134Smrg  {
1177a84e134Smrg    XtNrepeatDelay,
1187a84e134Smrg    XtCDelay,
1197a84e134Smrg    XtRInt,
1207a84e134Smrg    sizeof(int),
1217a84e134Smrg    offset(repeat_delay),
1227a84e134Smrg    XtRImmediate,
1237a84e134Smrg    (XtPointer)REP_DEF_REPEAT_DELAY
1247a84e134Smrg  },
1257a84e134Smrg  {
1267a84e134Smrg    XtNflash,
1277a84e134Smrg    XtCBoolean,
1287a84e134Smrg    XtRBoolean,
1297a84e134Smrg    sizeof(Boolean),
1307a84e134Smrg    offset(flash),
1317a84e134Smrg    XtRImmediate,
1327a84e134Smrg    (XtPointer)False
1337a84e134Smrg  },
1347a84e134Smrg  {
1357a84e134Smrg    XtNstartCallback,
1367a84e134Smrg    XtCStartCallback,
1377a84e134Smrg    XtRCallback,
1387a84e134Smrg    sizeof(XtPointer),
1397a84e134Smrg    offset(start_callbacks),
1407a84e134Smrg    XtRImmediate,
1417a84e134Smrg    NULL
1427a84e134Smrg  },
1437a84e134Smrg  {
1447a84e134Smrg    XtNstopCallback,
1457a84e134Smrg    XtCStopCallback,
1467a84e134Smrg    XtRCallback,
1477a84e134Smrg    sizeof(XtPointer),
1487a84e134Smrg    offset(stop_callbacks),
1497a84e134Smrg    XtRImmediate,
1507a84e134Smrg    NULL
1517a84e134Smrg  },
1527a84e134Smrg};
1537a84e134Smrg#undef offset
1547a84e134Smrg
1557a84e134Smrg#define Superclass	(&commandClassRec)
1567a84e134SmrgRepeaterClassRec repeaterClassRec = {
1577a84e134Smrg  /* core */
1587a84e134Smrg  {
1597a84e134Smrg    (WidgetClass)Superclass,		/* superclass */
1607a84e134Smrg    "Repeater",				/* class_name */
1617a84e134Smrg    sizeof(RepeaterRec),		/* widget_size */
1627a84e134Smrg    XawInitializeWidgetSet,		/* class_initialize */
1637a84e134Smrg    NULL,				/* class_part_initialize */
1647a84e134Smrg    False,				/* class_inited */
1657a84e134Smrg    XawRepeaterInitialize,		/* initialize */
1667a84e134Smrg    NULL,				/* initialize_hook */
1677a84e134Smrg    XtInheritRealize,			/* realize */
1687a84e134Smrg    actions,				/* actions */
1697a84e134Smrg    XtNumber(actions),			/* num_actions */
1707a84e134Smrg    resources,				/* resources */
1717a84e134Smrg    XtNumber(resources),		/* num_resources */
1727a84e134Smrg    NULLQUARK,				/* xrm_class */
1737a84e134Smrg    True,				/* compress_motion */
1747a84e134Smrg    True,				/* compress_exposure */
1757a84e134Smrg    True,				/* compress_enterleave */
1767a84e134Smrg    False,				/* visible_interest */
1777a84e134Smrg    XawRepeaterDestroy,			/* destroy */
1787a84e134Smrg    XtInheritResize,			/* resize */
1797a84e134Smrg    XtInheritExpose,			/* expose */
1807a84e134Smrg    XawRepeaterSetValues,		/* set_values */
1817a84e134Smrg    NULL,				/* set_values_hook */
1827a84e134Smrg    XtInheritSetValuesAlmost,		/* set_values_almost */
1837a84e134Smrg    NULL,				/* get_values_hook */
1847a84e134Smrg    NULL,				/* accept_focus */
1857a84e134Smrg    XtVersion,				/* version */
1867a84e134Smrg    NULL,				/* callback_private */
1877a84e134Smrg    defaultTranslations,		/* tm_table */
1887a84e134Smrg    XtInheritQueryGeometry,		/* query_geometry */
1897a84e134Smrg    XtInheritDisplayAccelerator,	/* display_accelerator */
1907a84e134Smrg    NULL,				/* extension */
1917a84e134Smrg  },
1927a84e134Smrg  /* simple */
1937a84e134Smrg  {
1947a84e134Smrg    XtInheritChangeSensitive,		/* change_sensitive */
1957a84e134Smrg  },
1967a84e134Smrg  /* label */
1977a84e134Smrg  {
1987a84e134Smrg    NULL,				/* extension */
1997a84e134Smrg  },
2007a84e134Smrg  /* command */
2017a84e134Smrg  {
2027a84e134Smrg    NULL,				/* extension */
2037a84e134Smrg  },
2047a84e134Smrg  /* repeater */
2057a84e134Smrg  {
2067a84e134Smrg    NULL,				/* extension */
2077a84e134Smrg  },
2087a84e134Smrg};
2097a84e134Smrg
2107a84e134SmrgWidgetClass repeaterWidgetClass = (WidgetClass) &repeaterClassRec;
2117a84e134Smrg
2127a84e134Smrg
2137a84e134Smrg/*
2147a84e134Smrg * Implementation
2157a84e134Smrg */
2167a84e134Smrg/*ARGSUSED*/
2177a84e134Smrgstatic void
2187a84e134Smrgtic(XtPointer client_data, XtIntervalId *id)
2197a84e134Smrg{
2207a84e134Smrg    RepeaterWidget rw = (RepeaterWidget)client_data;
2217a84e134Smrg
2227a84e134Smrg    rw->repeater.timer = 0;		/* timer is removed */
2237a84e134Smrg    if (rw->repeater.flash) {
2247a84e134Smrg	Widget w = (Widget)rw;
2257a84e134Smrg
2267a84e134Smrg	XClearWindow(XtDisplay(w), XtWindow(w));
2277a84e134Smrg	XtCallActionProc(w, "reset", NULL, NULL, 0);
2287a84e134Smrg	XClearWindow(XtDisplay(w), XtWindow(w));
2297a84e134Smrg	XtCallActionProc(w, "set", NULL, NULL, 0);
2307a84e134Smrg    }
2317a84e134Smrg    DO_CALLBACK(rw);
2327a84e134Smrg
2337a84e134Smrg    rw->repeater.timer = ADD_TIMEOUT(rw, rw->repeater.next_delay);
2347a84e134Smrg
2357a84e134Smrg    if (rw->repeater.decay) {
2367a84e134Smrg	rw->repeater.next_delay -= rw->repeater.decay;
2377a84e134Smrg	if (rw->repeater.next_delay < rw->repeater.minimum_delay)
2387a84e134Smrg	    rw->repeater.next_delay = rw->repeater.minimum_delay;
2397a84e134Smrg    }
2407a84e134Smrg}
2417a84e134Smrg
2427a84e134Smrg/*ARGSUSED*/
2437a84e134Smrgstatic void
2447a84e134SmrgXawRepeaterInitialize(Widget greq, Widget gnew,
2457a84e134Smrg		      ArgList args, Cardinal *num_args)
2467a84e134Smrg{
2477a84e134Smrg    RepeaterWidget cnew = (RepeaterWidget)gnew;
2487a84e134Smrg
2497a84e134Smrg    if (cnew->repeater.minimum_delay < 0)
2507a84e134Smrg	cnew->repeater.minimum_delay = 0;
2517a84e134Smrg    cnew->repeater.timer = 0;
2527a84e134Smrg}
2537a84e134Smrg
2547a84e134Smrgstatic void
2557a84e134SmrgXawRepeaterDestroy(Widget gw)
2567a84e134Smrg{
2577a84e134Smrg    CLEAR_TIMEOUT((RepeaterWidget)gw);
2587a84e134Smrg}
2597a84e134Smrg
2607a84e134Smrg/*ARGSUSED*/
2617a84e134Smrgstatic Boolean
2627a84e134SmrgXawRepeaterSetValues(Widget gcur, Widget greq, Widget gnew,
2637a84e134Smrg		     ArgList args, Cardinal *num_args)
2647a84e134Smrg{
2657a84e134Smrg    RepeaterWidget cur = (RepeaterWidget)gcur;
2667a84e134Smrg    RepeaterWidget cnew = (RepeaterWidget)gnew;
2677a84e134Smrg
2687a84e134Smrg    if (cur->repeater.minimum_delay != cnew->repeater.minimum_delay) {
2697a84e134Smrg	if (cnew->repeater.next_delay < cnew->repeater.minimum_delay)
2707a84e134Smrg	    cnew->repeater.next_delay = cnew->repeater.minimum_delay;
2717a84e134Smrg    }
2727a84e134Smrg
2737a84e134Smrg    return (False);
2747a84e134Smrg}
2757a84e134Smrg
2767a84e134Smrg/*ARGSUSED*/
2777a84e134Smrgstatic void
2787a84e134SmrgActionStart(Widget gw, XEvent *event, String *params, Cardinal *num_params)
2797a84e134Smrg{
2807a84e134Smrg    RepeaterWidget rw = (RepeaterWidget)gw;
2817a84e134Smrg
2827a84e134Smrg    CLEAR_TIMEOUT(rw);
2837a84e134Smrg    if (rw->repeater.start_callbacks)
2847a84e134Smrg	XtCallCallbackList(gw, rw->repeater.start_callbacks, NULL);
2857a84e134Smrg
2867a84e134Smrg    DO_CALLBACK(rw);
2877a84e134Smrg    rw->repeater.timer = ADD_TIMEOUT(rw, rw->repeater.initial_delay);
2887a84e134Smrg    rw->repeater.next_delay = rw->repeater.repeat_delay;
2897a84e134Smrg}
2907a84e134Smrg
2917a84e134Smrg/*ARGSUSED*/
2927a84e134Smrgstatic void
2937a84e134SmrgActionStop(Widget gw, XEvent *event, String *params, Cardinal *num_params)
2947a84e134Smrg{
2957a84e134Smrg    RepeaterWidget rw = (RepeaterWidget)gw;
2967a84e134Smrg
2977a84e134Smrg    CLEAR_TIMEOUT((RepeaterWidget)gw);
2987a84e134Smrg    if (rw->repeater.stop_callbacks)
2997a84e134Smrg	XtCallCallbackList(gw, rw->repeater.stop_callbacks, NULL);
3007a84e134Smrg}
301