Scrollbar.c revision 7a84e134
17a84e134Smrg/* $Xorg: Scrollbar.c,v 1.4 2001/02/09 02:03:45 xorgcvs Exp $ */
27a84e134Smrg
37a84e134Smrg/***********************************************************
47a84e134Smrg
57a84e134SmrgCopyright 1987, 1988, 1994, 1998  The Open Group
67a84e134Smrg
77a84e134SmrgPermission to use, copy, modify, distribute, and sell this software and its
87a84e134Smrgdocumentation for any purpose is hereby granted without fee, provided that
97a84e134Smrgthe above copyright notice appear in all copies and that both that
107a84e134Smrgcopyright notice and this permission notice appear in supporting
117a84e134Smrgdocumentation.
127a84e134Smrg
137a84e134SmrgThe above copyright notice and this permission notice shall be included in
147a84e134Smrgall copies or substantial portions of the Software.
157a84e134Smrg
167a84e134SmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
177a84e134SmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
187a84e134SmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
197a84e134SmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
207a84e134SmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
217a84e134SmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
227a84e134Smrg
237a84e134SmrgExcept as contained in this notice, the name of The Open Group shall not be
247a84e134Smrgused in advertising or otherwise to promote the sale, use or other dealings
257a84e134Smrgin this Software without prior written authorization from The Open Group.
267a84e134Smrg
277a84e134Smrg
287a84e134SmrgCopyright 1987, 1988 by Digital Equipment Corporation, Maynard, Massachusetts.
297a84e134Smrg
307a84e134Smrg                        All Rights Reserved
317a84e134Smrg
327a84e134SmrgPermission to use, copy, modify, and distribute this software and its
337a84e134Smrgdocumentation for any purpose and without fee is hereby granted,
347a84e134Smrgprovided that the above copyright notice appear in all copies and that
357a84e134Smrgboth that copyright notice and this permission notice appear in
367a84e134Smrgsupporting documentation, and that the name of Digital not be
377a84e134Smrgused in advertising or publicity pertaining to distribution of the
387a84e134Smrgsoftware without specific, written prior permission.
397a84e134Smrg
407a84e134SmrgDIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
417a84e134SmrgALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
427a84e134SmrgDIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
437a84e134SmrgANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
447a84e134SmrgWHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
457a84e134SmrgARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
467a84e134SmrgSOFTWARE.
477a84e134Smrg
487a84e134Smrg******************************************************************/
497a84e134Smrg/* $XFree86: xc/lib/Xaw/Scrollbar.c,v 1.12 2001/01/17 19:42:29 dawes Exp $ */
507a84e134Smrg
517a84e134Smrg#ifdef HAVE_CONFIG_H
527a84e134Smrg#include <config.h>
537a84e134Smrg#endif
547a84e134Smrg#include <X11/IntrinsicP.h>
557a84e134Smrg#include <X11/StringDefs.h>
567a84e134Smrg#include <X11/Xmu/Drawing.h>
577a84e134Smrg#include <X11/Xaw/ScrollbarP.h>
587a84e134Smrg#include <X11/Xaw/XawInit.h>
597a84e134Smrg#include "Private.h"
607a84e134Smrg
617a84e134Smrg#define NoButton	-1
627a84e134Smrg#define PICKLENGTH(widget, x, y)					\
637a84e134Smrg(((widget)->scrollbar.orientation == XtorientHorizontal) ? (x) : (y))
647a84e134Smrg
657a84e134Smrg/*
667a84e134Smrg * Class Methods
677a84e134Smrg */
687a84e134Smrgstatic void XawScrollbarClassInitialize(void);
697a84e134Smrgstatic void XawScrollbarDestroy(Widget);
707a84e134Smrgstatic void XawScrollbarInitialize(Widget, Widget, ArgList, Cardinal*_args);
717a84e134Smrgstatic void XawScrollbarRealize(Widget, Mask*, XSetWindowAttributes*);
727a84e134Smrgstatic void XawScrollbarRedisplay(Widget, XEvent*, Region);
737a84e134Smrgstatic void XawScrollbarResize(Widget);
747a84e134Smrgstatic Boolean XawScrollbarSetValues(Widget, Widget, Widget,
757a84e134Smrg				     ArgList, Cardinal*);
767a84e134Smrg
777a84e134Smrg/*
787a84e134Smrg * Prototypes
797a84e134Smrg */
807a84e134Smrgstatic Boolean CompareEvents(XEvent*, XEvent*);
817a84e134Smrgstatic void CreateGC(Widget);
827a84e134Smrgstatic float FloatInRange(float, float, float);
837a84e134Smrgstatic float FractionLoc(ScrollbarWidget, int, int);
847a84e134Smrgstatic void ExtractPosition(XEvent*, Position*, Position*);
857a84e134Smrgstatic int InRange(int, int, int);
867a84e134Smrgstatic void FillArea(ScrollbarWidget, int, int, int);
877a84e134Smrgstatic Bool LookAhead(Widget, XEvent*);
887a84e134Smrgstatic void PaintThumb(ScrollbarWidget);
897a84e134Smrgstatic Bool PeekNotifyEvent(Display*, XEvent*, char*);
907a84e134Smrgstatic void SetDimensions(ScrollbarWidget);
917a84e134Smrg
927a84e134Smrg/*
937a84e134Smrg * Actions
947a84e134Smrg */
957a84e134Smrgstatic void EndScroll(Widget, XEvent*, String*, Cardinal*);
967a84e134Smrgstatic void MoveThumb(Widget, XEvent*, String*, Cardinal*);
977a84e134Smrgstatic void NotifyScroll(Widget, XEvent*, String*, Cardinal*);
987a84e134Smrgstatic void NotifyThumb(Widget, XEvent*, String*, Cardinal*);
997a84e134Smrgstatic void StartScroll(Widget, XEvent*, String*, Cardinal*);
1007a84e134Smrg
1017a84e134Smrg/*
1027a84e134Smrg * Initialization
1037a84e134Smrg */
1047a84e134Smrgstatic char defaultTranslations[] =
1057a84e134Smrg"<Btn1Down>:"	"StartScroll(Forward)\n"
1067a84e134Smrg"<Btn2Down>:"	"StartScroll(Continuous) MoveThumb() NotifyThumb()\n"
1077a84e134Smrg"<Btn3Down>:"	"StartScroll(Backward)\n"
1087a84e134Smrg"<Btn2Motion>:"	"MoveThumb() NotifyThumb()\n"
1097a84e134Smrg"<BtnUp>:"	"NotifyScroll(Proportional) EndScroll()\n";
1107a84e134Smrg
1117a84e134Smrgstatic float floatZero = 0.0;
1127a84e134Smrg
1137a84e134Smrg#define Offset(field) XtOffsetOf(ScrollbarRec, field)
1147a84e134Smrg
1157a84e134Smrgstatic XtResource resources[] = {
1167a84e134Smrg  {
1177a84e134Smrg    XtNlength,
1187a84e134Smrg    XtCLength,
1197a84e134Smrg    XtRDimension,
1207a84e134Smrg    sizeof(Dimension),
1217a84e134Smrg    Offset(scrollbar.length),
1227a84e134Smrg    XtRImmediate,
1237a84e134Smrg    (XtPointer)1
1247a84e134Smrg  },
1257a84e134Smrg  {
1267a84e134Smrg    XtNthickness,
1277a84e134Smrg    XtCThickness,
1287a84e134Smrg    XtRDimension,
1297a84e134Smrg    sizeof(Dimension),
1307a84e134Smrg    Offset(scrollbar.thickness),
1317a84e134Smrg    XtRImmediate,
1327a84e134Smrg    (XtPointer)14
1337a84e134Smrg  },
1347a84e134Smrg  {
1357a84e134Smrg    XtNorientation,
1367a84e134Smrg    XtCOrientation,
1377a84e134Smrg    XtROrientation,
1387a84e134Smrg    sizeof(XtOrientation),
1397a84e134Smrg    Offset(scrollbar.orientation),
1407a84e134Smrg    XtRImmediate,
1417a84e134Smrg    (XtPointer)XtorientVertical
1427a84e134Smrg  },
1437a84e134Smrg  {
1447a84e134Smrg    XtNscrollProc,
1457a84e134Smrg    XtCCallback,
1467a84e134Smrg    XtRCallback,
1477a84e134Smrg    sizeof(XtPointer),
1487a84e134Smrg    Offset(scrollbar.scrollProc),
1497a84e134Smrg    XtRCallback,
1507a84e134Smrg    NULL
1517a84e134Smrg  },
1527a84e134Smrg  {
1537a84e134Smrg    XtNthumbProc,
1547a84e134Smrg    XtCCallback,
1557a84e134Smrg    XtRCallback,
1567a84e134Smrg    sizeof(XtPointer),
1577a84e134Smrg    Offset(scrollbar.thumbProc),
1587a84e134Smrg    XtRCallback,
1597a84e134Smrg    NULL
1607a84e134Smrg  },
1617a84e134Smrg  {
1627a84e134Smrg    XtNjumpProc,
1637a84e134Smrg    XtCCallback,
1647a84e134Smrg    XtRCallback,
1657a84e134Smrg    sizeof(XtPointer),
1667a84e134Smrg    Offset(scrollbar.jumpProc),
1677a84e134Smrg    XtRCallback,
1687a84e134Smrg    NULL
1697a84e134Smrg  },
1707a84e134Smrg  {
1717a84e134Smrg    XtNthumb,
1727a84e134Smrg    XtCThumb,
1737a84e134Smrg    XtRBitmap,
1747a84e134Smrg    sizeof(Pixmap),
1757a84e134Smrg    Offset(scrollbar.thumb),
1767a84e134Smrg    XtRImmediate,
1777a84e134Smrg    (XtPointer)XtUnspecifiedPixmap
1787a84e134Smrg  },
1797a84e134Smrg  {
1807a84e134Smrg    XtNforeground,
1817a84e134Smrg    XtCForeground,
1827a84e134Smrg    XtRPixel,
1837a84e134Smrg    sizeof(Pixel),
1847a84e134Smrg    Offset(scrollbar.foreground),
1857a84e134Smrg    XtRString,
1867a84e134Smrg    XtDefaultForeground
1877a84e134Smrg  },
1887a84e134Smrg  {
1897a84e134Smrg    XtNshown,
1907a84e134Smrg    XtCShown,
1917a84e134Smrg    XtRFloat,
1927a84e134Smrg    sizeof(float),
1937a84e134Smrg    Offset(scrollbar.shown),
1947a84e134Smrg    XtRFloat,
1957a84e134Smrg    (XtPointer)&floatZero
1967a84e134Smrg  },
1977a84e134Smrg  {
1987a84e134Smrg    XtNtopOfThumb,
1997a84e134Smrg    XtCTopOfThumb,
2007a84e134Smrg    XtRFloat,
2017a84e134Smrg    sizeof(float),
2027a84e134Smrg    Offset(scrollbar.top),
2037a84e134Smrg    XtRFloat,
2047a84e134Smrg    (XtPointer)&floatZero
2057a84e134Smrg  },
2067a84e134Smrg  {
2077a84e134Smrg    XtNscrollVCursor,
2087a84e134Smrg    XtCCursor,
2097a84e134Smrg    XtRCursor,
2107a84e134Smrg    sizeof(Cursor),
2117a84e134Smrg    Offset(scrollbar.verCursor),
2127a84e134Smrg    XtRString,
2137a84e134Smrg    "sb_v_double_arrow"
2147a84e134Smrg  },
2157a84e134Smrg  {
2167a84e134Smrg    XtNscrollHCursor,
2177a84e134Smrg    XtCCursor,
2187a84e134Smrg    XtRCursor,
2197a84e134Smrg    sizeof(Cursor),
2207a84e134Smrg    Offset(scrollbar.horCursor),
2217a84e134Smrg    XtRString,
2227a84e134Smrg    "sb_h_double_arrow"
2237a84e134Smrg  },
2247a84e134Smrg  {
2257a84e134Smrg    XtNscrollUCursor,
2267a84e134Smrg    XtCCursor,
2277a84e134Smrg    XtRCursor,
2287a84e134Smrg    sizeof(Cursor),
2297a84e134Smrg    Offset(scrollbar.upCursor),
2307a84e134Smrg    XtRString,
2317a84e134Smrg    "sb_up_arrow"
2327a84e134Smrg  },
2337a84e134Smrg  {
2347a84e134Smrg    XtNscrollDCursor,
2357a84e134Smrg    XtCCursor,
2367a84e134Smrg    XtRCursor,
2377a84e134Smrg    sizeof(Cursor),
2387a84e134Smrg    Offset(scrollbar.downCursor),
2397a84e134Smrg    XtRString,
2407a84e134Smrg    "sb_down_arrow"
2417a84e134Smrg  },
2427a84e134Smrg  {
2437a84e134Smrg    XtNscrollLCursor,
2447a84e134Smrg    XtCCursor,
2457a84e134Smrg    XtRCursor,
2467a84e134Smrg    sizeof(Cursor),
2477a84e134Smrg    Offset(scrollbar.leftCursor),
2487a84e134Smrg    XtRString,
2497a84e134Smrg    "sb_left_arrow"
2507a84e134Smrg  },
2517a84e134Smrg  {
2527a84e134Smrg    XtNscrollRCursor,
2537a84e134Smrg    XtCCursor,
2547a84e134Smrg    XtRCursor,
2557a84e134Smrg    sizeof(Cursor),
2567a84e134Smrg    Offset(scrollbar.rightCursor),
2577a84e134Smrg    XtRString,
2587a84e134Smrg    "sb_right_arrow"
2597a84e134Smrg  },
2607a84e134Smrg  {
2617a84e134Smrg    XtNminimumThumb,
2627a84e134Smrg    XtCMinimumThumb,
2637a84e134Smrg    XtRDimension,
2647a84e134Smrg    sizeof(Dimension),
2657a84e134Smrg    Offset(scrollbar.min_thumb),
2667a84e134Smrg    XtRImmediate,
2677a84e134Smrg    (XtPointer)7
2687a84e134Smrg  },
2697a84e134Smrg};
2707a84e134Smrg#undef Offset
2717a84e134Smrg
2727a84e134Smrgstatic XtActionsRec actions[] = {
2737a84e134Smrg	{"StartScroll",		StartScroll},
2747a84e134Smrg	{"MoveThumb",		MoveThumb},
2757a84e134Smrg	{"NotifyThumb",		NotifyThumb},
2767a84e134Smrg	{"NotifyScroll",	NotifyScroll},
2777a84e134Smrg	{"EndScroll",		EndScroll},
2787a84e134Smrg};
2797a84e134Smrg
2807a84e134Smrg#define Superclass (&simpleClassRec)
2817a84e134SmrgScrollbarClassRec scrollbarClassRec = {
2827a84e134Smrg  /* core */
2837a84e134Smrg  {
2847a84e134Smrg    (WidgetClass)&simpleClassRec,	/* superclass */
2857a84e134Smrg    "Scrollbar",			/* class_name */
2867a84e134Smrg    sizeof(ScrollbarRec),		/* widget_size */
2877a84e134Smrg    XawScrollbarClassInitialize,	/* class_initialize */
2887a84e134Smrg    NULL,				/* class_part_init */
2897a84e134Smrg    False,				/* class_inited */
2907a84e134Smrg    XawScrollbarInitialize,		/* initialize */
2917a84e134Smrg    NULL,				/* initialize_hook */
2927a84e134Smrg    XawScrollbarRealize,		/* realize */
2937a84e134Smrg    actions,				/* actions */
2947a84e134Smrg    XtNumber(actions),			/* num_actions */
2957a84e134Smrg    resources,				/* resources */
2967a84e134Smrg    XtNumber(resources),		/* num_resources */
2977a84e134Smrg    NULLQUARK,				/* xrm_class */
2987a84e134Smrg    True,				/* compress_motion */
2997a84e134Smrg    True,				/* compress_exposure */
3007a84e134Smrg    True,				/* compress_enterleave */
3017a84e134Smrg    False,				/* visible_interest */
3027a84e134Smrg    XawScrollbarDestroy,		/* destroy */
3037a84e134Smrg    XawScrollbarResize,			/* resize */
3047a84e134Smrg    XawScrollbarRedisplay,		/* expose */
3057a84e134Smrg    XawScrollbarSetValues,		/* set_values */
3067a84e134Smrg    NULL,				/* set_values_hook */
3077a84e134Smrg    XtInheritSetValuesAlmost,		/* set_values_almost */
3087a84e134Smrg    NULL,				/* get_values_hook */
3097a84e134Smrg    NULL,				/* accept_focus */
3107a84e134Smrg    XtVersion,				/* version */
3117a84e134Smrg    NULL,				/* callback_private */
3127a84e134Smrg    defaultTranslations,		/* tm_table */
3137a84e134Smrg    XtInheritQueryGeometry,		/* query_geometry */
3147a84e134Smrg    XtInheritDisplayAccelerator,	/* display_accelerator */
3157a84e134Smrg    NULL,				/* extension */
3167a84e134Smrg  },
3177a84e134Smrg  /* simple */
3187a84e134Smrg  {
3197a84e134Smrg    XtInheritChangeSensitive,		/* change_sensitive */
3207a84e134Smrg  },
3217a84e134Smrg  /* scrollbar */
3227a84e134Smrg  {
3237a84e134Smrg    NULL,				/* extension */
3247a84e134Smrg  },
3257a84e134Smrg};
3267a84e134Smrg
3277a84e134SmrgWidgetClass scrollbarWidgetClass = (WidgetClass)&scrollbarClassRec;
3287a84e134Smrg
3297a84e134Smrg/*
3307a84e134Smrg * Implementation
3317a84e134Smrg */
3327a84e134Smrgstatic void
3337a84e134SmrgXawScrollbarClassInitialize(void)
3347a84e134Smrg{
3357a84e134Smrg    XawInitializeWidgetSet();
3367a84e134Smrg    XtAddConverter(XtRString, XtROrientation, XmuCvtStringToOrientation,
3377a84e134Smrg		   NULL, 0);
3387a84e134Smrg    XtSetTypeConverter(XtROrientation, XtRString, XmuCvtOrientationToString,
3397a84e134Smrg		       NULL, 0, XtCacheNone, NULL);
3407a84e134Smrg}
3417a84e134Smrg
3427a84e134Smrg/*
3437a84e134Smrg * Make sure the first number is within the range specified by the other
3447a84e134Smrg * two numbers.
3457a84e134Smrg */
3467a84e134Smrgstatic int
3477a84e134SmrgInRange(int num, int small, int big)
3487a84e134Smrg{
3497a84e134Smrg    return ((num < small) ? small : ((num > big) ? big : num));
3507a84e134Smrg}
3517a84e134Smrg
3527a84e134Smrg/*
3537a84e134Smrg * Same as above, but for floating numbers
3547a84e134Smrg */
3557a84e134Smrgstatic float
3567a84e134SmrgFloatInRange(float num, float small, float big)
3577a84e134Smrg{
3587a84e134Smrg    return ((num < small) ? small : ((num > big) ? big : num));
3597a84e134Smrg}
3607a84e134Smrg
3617a84e134Smrg/* Fill the area specified by top and bottom with the given pattern */
3627a84e134Smrgstatic float
3637a84e134SmrgFractionLoc(ScrollbarWidget w, int x, int y)
3647a84e134Smrg{
3657a84e134Smrg    float   result;
3667a84e134Smrg
3677a84e134Smrg    result = PICKLENGTH(w, x / (float)XtWidth(w), y / (float)XtHeight(w));
3687a84e134Smrg
3697a84e134Smrg    return (FloatInRange(result, 0.0, 1.0));
3707a84e134Smrg}
3717a84e134Smrg
3727a84e134Smrgstatic void
3737a84e134SmrgFillArea(ScrollbarWidget w, int top, int bottom, int thumb)
3747a84e134Smrg{
3757a84e134Smrg    Dimension length;
3767a84e134Smrg
3777a84e134Smrg    top = XawMax(1, top);
3787a84e134Smrg    if (w->scrollbar.orientation == XtorientHorizontal)
3797a84e134Smrg    bottom = XawMin(bottom, XtWidth(w) - 1);
3807a84e134Smrg    else
3817a84e134Smrg    bottom = XawMin(bottom, XtHeight(w) - 1);
3827a84e134Smrg
3837a84e134Smrg    if (bottom <= top)
3847a84e134Smrg	return;
3857a84e134Smrg
3867a84e134Smrg    length = bottom - top;
3877a84e134Smrg
3887a84e134Smrg    switch(thumb) {
3897a84e134Smrg	/* Fill the new Thumb location */
3907a84e134Smrg	case 1:
3917a84e134Smrg	    if (w->scrollbar.orientation == XtorientHorizontal)
3927a84e134Smrg		XFillRectangle(XtDisplay(w), XtWindow(w), w->scrollbar.gc,
3937a84e134Smrg			       top, 1, length, XtHeight(w) - 2);
3947a84e134Smrg	    else
3957a84e134Smrg		XFillRectangle(XtDisplay(w), XtWindow(w), w->scrollbar.gc,
3967a84e134Smrg			       1, top, XtWidth(w) - 2, length);
3977a84e134Smrg	    break;
3987a84e134Smrg	/* Clear the old Thumb location */
3997a84e134Smrg	case 0:
4007a84e134Smrg	    if (w->scrollbar.orientation == XtorientHorizontal)
4017a84e134Smrg		XClearArea(XtDisplay(w), XtWindow(w),
4027a84e134Smrg			   top, 1, length, XtHeight(w) - 2, False);
4037a84e134Smrg	    else
4047a84e134Smrg		XClearArea(XtDisplay(w), XtWindow(w),
4057a84e134Smrg			   1, top, XtWidth(w) - 2, length, False);
4067a84e134Smrg	    break;
4077a84e134Smrg    }
4087a84e134Smrg}
4097a84e134Smrg
4107a84e134Smrg
4117a84e134Smrg/* Paint the thumb in the area specified by w->top and
4127a84e134Smrg   w->shown.  The old area is erased.  The painting and
4137a84e134Smrg   erasing is done cleverly so that no flickering will occur. */
4147a84e134Smrgstatic void
4157a84e134SmrgPaintThumb(ScrollbarWidget w)
4167a84e134Smrg{
4177a84e134Smrg    Position oldtop, oldbot, newtop, newbot;
4187a84e134Smrg
4197a84e134Smrg    oldtop = w->scrollbar.topLoc;
4207a84e134Smrg    oldbot = oldtop + w->scrollbar.shownLength;
4217a84e134Smrg    newtop = w->scrollbar.length * w->scrollbar.top;
4227a84e134Smrg    newbot = newtop + (int)(w->scrollbar.length * w->scrollbar.shown);
4237a84e134Smrg    if (newbot < newtop + (int)w->scrollbar.min_thumb)
4247a84e134Smrg	newbot = newtop + w->scrollbar.min_thumb;
4257a84e134Smrg    w->scrollbar.topLoc = newtop;
4267a84e134Smrg    w->scrollbar.shownLength = newbot - newtop;
4277a84e134Smrg
4287a84e134Smrg    if (XtIsRealized((Widget)w)) {
4297a84e134Smrg	if (newtop < oldtop)
4307a84e134Smrg	    FillArea(w, newtop, XawMin(newbot, oldtop), 1);
4317a84e134Smrg	if (newtop > oldtop)
4327a84e134Smrg	    FillArea(w, oldtop, XawMin(newtop, oldbot), 0);
4337a84e134Smrg	if (newbot < oldbot)
4347a84e134Smrg	    FillArea(w, XawMax(newbot, oldtop), oldbot, 0);
4357a84e134Smrg	if (newbot > oldbot)
4367a84e134Smrg	    FillArea(w, XawMax(newtop, oldbot), newbot, 1);
4377a84e134Smrg    }
4387a84e134Smrg}
4397a84e134Smrg
4407a84e134Smrgstatic void
4417a84e134SmrgSetDimensions(ScrollbarWidget w)
4427a84e134Smrg{
4437a84e134Smrg    if (w->scrollbar.orientation == XtorientVertical) {
4447a84e134Smrg	w->scrollbar.length = XtHeight(w);
4457a84e134Smrg	w->scrollbar.thickness = XtWidth(w);
4467a84e134Smrg    }
4477a84e134Smrg    else {
4487a84e134Smrg	w->scrollbar.length = XtWidth(w);
4497a84e134Smrg	w->scrollbar.thickness = XtHeight(w);
4507a84e134Smrg    }
4517a84e134Smrg}
4527a84e134Smrg
4537a84e134Smrgstatic void
4547a84e134SmrgXawScrollbarDestroy(Widget w)
4557a84e134Smrg{
4567a84e134Smrg    ScrollbarWidget sbw = (ScrollbarWidget)w;
4577a84e134Smrg
4587a84e134Smrg    XtReleaseGC(w, sbw->scrollbar.gc);
4597a84e134Smrg}
4607a84e134Smrg
4617a84e134Smrgstatic void
4627a84e134SmrgCreateGC(Widget w)
4637a84e134Smrg{
4647a84e134Smrg    ScrollbarWidget sbw = (ScrollbarWidget)w;
4657a84e134Smrg    XGCValues gcValues;
4667a84e134Smrg    XtGCMask mask;
4677a84e134Smrg    unsigned int depth = 1;
4687a84e134Smrg
4697a84e134Smrg    if (sbw->scrollbar.thumb == XtUnspecifiedPixmap)
4707a84e134Smrg	sbw->scrollbar.thumb = XmuCreateStippledPixmap(XtScreen(w),
4717a84e134Smrg						       (Pixel)1, (Pixel)0,
4727a84e134Smrg						       depth);
4737a84e134Smrg    else if (sbw->scrollbar.thumb != None) {
4747a84e134Smrg	Window root;
4757a84e134Smrg	int x, y;
4767a84e134Smrg	unsigned int width, height, bw;
4777a84e134Smrg
4787a84e134Smrg	XGetGeometry(XtDisplay(w), sbw->scrollbar.thumb, &root, &x, &y,
4797a84e134Smrg		     &width, &height, &bw, &depth);
4807a84e134Smrg    }
4817a84e134Smrg
4827a84e134Smrg    gcValues.foreground = sbw->scrollbar.foreground;
4837a84e134Smrg    gcValues.background = sbw->core.background_pixel;
4847a84e134Smrg    mask = GCForeground | GCBackground;
4857a84e134Smrg
4867a84e134Smrg    if (sbw->scrollbar.thumb != None) {
4877a84e134Smrg	if (depth == 1) {
4887a84e134Smrg	    gcValues.fill_style = FillOpaqueStippled;
4897a84e134Smrg	    gcValues.stipple = sbw->scrollbar.thumb;
4907a84e134Smrg	    mask |= GCFillStyle | GCStipple;
4917a84e134Smrg	}
4927a84e134Smrg	else {
4937a84e134Smrg	    gcValues.fill_style = FillTiled;
4947a84e134Smrg	    gcValues.tile = sbw->scrollbar.thumb;
4957a84e134Smrg	    mask |= GCFillStyle | GCTile;
4967a84e134Smrg	}
4977a84e134Smrg    }
4987a84e134Smrg    sbw->scrollbar.gc = XtGetGC(w, mask, &gcValues);
4997a84e134Smrg}
5007a84e134Smrg
5017a84e134Smrg/* ARGSUSED */
5027a84e134Smrgstatic void
5037a84e134SmrgXawScrollbarInitialize(Widget request, Widget cnew,
5047a84e134Smrg		       ArgList args, Cardinal *num_args)
5057a84e134Smrg{
5067a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)cnew;
5077a84e134Smrg
5087a84e134Smrg    CreateGC(cnew);
5097a84e134Smrg
5107a84e134Smrg    if (XtWidth(w) == 0)
5117a84e134Smrg	XtWidth(w) = w->scrollbar.orientation == XtorientVertical ?
5127a84e134Smrg			w->scrollbar.thickness : w->scrollbar.length;
5137a84e134Smrg
5147a84e134Smrg    if (XtHeight(w) == 0)
5157a84e134Smrg	XtHeight(w) = w->scrollbar.orientation == XtorientHorizontal ?
5167a84e134Smrg			w->scrollbar.thickness : w->scrollbar.length;
5177a84e134Smrg
5187a84e134Smrg    SetDimensions(w);
5197a84e134Smrg    w->scrollbar.direction = 0;
5207a84e134Smrg    w->scrollbar.topLoc = 0;
5217a84e134Smrg    w->scrollbar.shownLength = w->scrollbar.min_thumb;
5227a84e134Smrg}
5237a84e134Smrg
5247a84e134Smrgstatic void
5257a84e134SmrgXawScrollbarRealize(Widget gw, Mask *valueMask,
5267a84e134Smrg		    XSetWindowAttributes *attributes)
5277a84e134Smrg{
5287a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
5297a84e134Smrg
5307a84e134Smrg    w->scrollbar.inactiveCursor = w->scrollbar.orientation == XtorientVertical ?
5317a84e134Smrg		w->scrollbar.verCursor : w->scrollbar.horCursor;
5327a84e134Smrg
5337a84e134Smrg    XtVaSetValues(gw, XtNcursor, w->scrollbar.inactiveCursor, NULL);
5347a84e134Smrg
5357a84e134Smrg    /*
5367a84e134Smrg     * The Simple widget actually stuffs the value in the valuemask
5377a84e134Smrg     */
5387a84e134Smrg    (*scrollbarWidgetClass->core_class.superclass->core_class.realize)
5397a84e134Smrg	(gw, valueMask, attributes);
5407a84e134Smrg}
5417a84e134Smrg
5427a84e134Smrg/*ARGSUSED*/
5437a84e134Smrgstatic Boolean
5447a84e134SmrgXawScrollbarSetValues(Widget current, Widget request, Widget desired,
5457a84e134Smrg		      ArgList args, Cardinal *num_args)
5467a84e134Smrg{
5477a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)current;
5487a84e134Smrg    ScrollbarWidget dw = (ScrollbarWidget)desired;
5497a84e134Smrg    Boolean redraw = False;
5507a84e134Smrg
5517a84e134Smrg    /*
5527a84e134Smrg     * If these values are outside the acceptable range ignore them...
5537a84e134Smrg     */
5547a84e134Smrg    if (dw->scrollbar.top < 0.0 || dw->scrollbar.top > 1.0)
5557a84e134Smrg	dw->scrollbar.top = w->scrollbar.top;
5567a84e134Smrg
5577a84e134Smrg    if (dw->scrollbar.shown < 0.0 || dw->scrollbar.shown > 1.0)
5587a84e134Smrg	dw->scrollbar.shown = w->scrollbar.shown;
5597a84e134Smrg
5607a84e134Smrg    if (XtIsRealized (desired)) {
5617a84e134Smrg	if (w->scrollbar.foreground != dw->scrollbar.foreground ||
5627a84e134Smrg	    w->core.background_pixel != dw->core.background_pixel ||
5637a84e134Smrg	    w->scrollbar.thumb != dw->scrollbar.thumb) {
5647a84e134Smrg	    XtReleaseGC((Widget)dw, w->scrollbar.gc);
5657a84e134Smrg	    CreateGC((Widget)dw);
5667a84e134Smrg	    redraw = True;
5677a84e134Smrg	}
5687a84e134Smrg	if (w->scrollbar.top != dw->scrollbar.top ||
5697a84e134Smrg	    w->scrollbar.shown != dw->scrollbar.shown)
5707a84e134Smrg	    redraw = True;
5717a84e134Smrg    }
5727a84e134Smrg
5737a84e134Smrg    return (redraw);
5747a84e134Smrg}
5757a84e134Smrg
5767a84e134Smrgstatic void
5777a84e134SmrgXawScrollbarResize(Widget gw)
5787a84e134Smrg{
5797a84e134Smrg    /* ForgetGravity has taken care of background, but thumb may
5807a84e134Smrg     * have to move as a result of the new size. */
5817a84e134Smrg    SetDimensions((ScrollbarWidget)gw);
5827a84e134Smrg    XawScrollbarRedisplay(gw, NULL, NULL);
5837a84e134Smrg}
5847a84e134Smrg
5857a84e134Smrg/*ARGSUSED*/
5867a84e134Smrgstatic void
5877a84e134SmrgXawScrollbarRedisplay(Widget gw, XEvent *event, Region region)
5887a84e134Smrg{
5897a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
5907a84e134Smrg    int x, y;
5917a84e134Smrg    unsigned int width, height;
5927a84e134Smrg
5937a84e134Smrg    if (Superclass->core_class.expose)
5947a84e134Smrg	(*Superclass->core_class.expose)(gw, event, region);
5957a84e134Smrg
5967a84e134Smrg    if (w->scrollbar.orientation == XtorientHorizontal) {
5977a84e134Smrg	x = w->scrollbar.topLoc;
5987a84e134Smrg	y = 1;
5997a84e134Smrg	width = w->scrollbar.shownLength;
6007a84e134Smrg	height = XtHeight(w) - 2;
6017a84e134Smrg    }
6027a84e134Smrg    else {
6037a84e134Smrg	x = 1;
6047a84e134Smrg	y = w->scrollbar.topLoc;
6057a84e134Smrg	width = XtWidth(w) - 2;
6067a84e134Smrg	height = w->scrollbar.shownLength;
6077a84e134Smrg    }
6087a84e134Smrg
6097a84e134Smrg    if (region == NULL ||
6107a84e134Smrg	XRectInRegion(region, x, y, width, height) != RectangleOut) {
6117a84e134Smrg	/* Forces entire thumb to be painted */
6127a84e134Smrg	w->scrollbar.topLoc = -(w->scrollbar.length + 1);
6137a84e134Smrg	PaintThumb(w);
6147a84e134Smrg    }
6157a84e134Smrg}
6167a84e134Smrg
6177a84e134Smrg/*ARGSUSED*/
6187a84e134Smrgstatic void
6197a84e134SmrgStartScroll(Widget gw, XEvent *event, String *params, Cardinal *num_params)
6207a84e134Smrg{
6217a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
6227a84e134Smrg    Cursor cursor;
6237a84e134Smrg    char direction;
6247a84e134Smrg
6257a84e134Smrg    if (w->scrollbar.direction != 0)	/* if we're already scrolling */
6267a84e134Smrg	return;
6277a84e134Smrg    if (*num_params > 0)
6287a84e134Smrg	direction = *params[0];
6297a84e134Smrg    else
6307a84e134Smrg	direction = 'C';
6317a84e134Smrg
6327a84e134Smrg    w->scrollbar.direction = direction;
6337a84e134Smrg
6347a84e134Smrg    switch(direction) {
6357a84e134Smrg	case 'B':
6367a84e134Smrg	case 'b':
6377a84e134Smrg	    cursor = w->scrollbar.orientation == XtorientVertical ?
6387a84e134Smrg		w->scrollbar.downCursor : w->scrollbar.rightCursor;
6397a84e134Smrg	    break;
6407a84e134Smrg	case 'F':
6417a84e134Smrg	case 'f':
6427a84e134Smrg	    cursor = w->scrollbar.orientation == XtorientVertical ?
6437a84e134Smrg		w->scrollbar.upCursor : w->scrollbar.leftCursor;
6447a84e134Smrg	    break;
6457a84e134Smrg	case 'C':
6467a84e134Smrg	case 'c':
6477a84e134Smrg	     cursor = w->scrollbar.orientation == XtorientVertical ?
6487a84e134Smrg		w->scrollbar.rightCursor : w->scrollbar.upCursor;
6497a84e134Smrg	     break;
6507a84e134Smrg	default:
6517a84e134Smrg	     return;	/* invalid invocation */
6527a84e134Smrg    }
6537a84e134Smrg
6547a84e134Smrg    XtVaSetValues(gw, XtNcursor, cursor, NULL);
6557a84e134Smrg
6567a84e134Smrg    XFlush(XtDisplay(w));
6577a84e134Smrg}
6587a84e134Smrg
6597a84e134Smrgstatic Boolean
6607a84e134SmrgCompareEvents(XEvent *oldEvent, XEvent *newEvent)
6617a84e134Smrg{
6627a84e134Smrg#define Check(field) if (newEvent->field != oldEvent->field) return (False)
6637a84e134Smrg
6647a84e134Smrg    Check(xany.display);
6657a84e134Smrg    Check(xany.type);
6667a84e134Smrg    Check(xany.window);
6677a84e134Smrg
6687a84e134Smrg    switch(newEvent->type) {
6697a84e134Smrg	case MotionNotify:
6707a84e134Smrg	    Check(xmotion.state);
6717a84e134Smrg	    break;
6727a84e134Smrg	case ButtonPress:
6737a84e134Smrg	case ButtonRelease:
6747a84e134Smrg	    Check(xbutton.state);
6757a84e134Smrg	    Check(xbutton.button);
6767a84e134Smrg	    break;
6777a84e134Smrg	case KeyPress:
6787a84e134Smrg	case KeyRelease:
6797a84e134Smrg	    Check(xkey.state);
6807a84e134Smrg	    Check(xkey.keycode);
6817a84e134Smrg	    break;
6827a84e134Smrg	case EnterNotify:
6837a84e134Smrg	case LeaveNotify:
6847a84e134Smrg	    Check(xcrossing.mode);
6857a84e134Smrg	    Check(xcrossing.detail);
6867a84e134Smrg	    Check(xcrossing.state);
6877a84e134Smrg	    break;
6887a84e134Smrg    }
6897a84e134Smrg#undef Check
6907a84e134Smrg
6917a84e134Smrg    return (True);
6927a84e134Smrg}
6937a84e134Smrg
6947a84e134Smrgstruct EventData {
6957a84e134Smrg    XEvent *oldEvent;
6967a84e134Smrg    int count;
6977a84e134Smrg};
6987a84e134Smrg
6997a84e134Smrgstatic Bool
7007a84e134SmrgPeekNotifyEvent(Display *dpy, XEvent *event, char *args)
7017a84e134Smrg{
7027a84e134Smrg    struct EventData *eventData = (struct EventData*)args;
7037a84e134Smrg
7047a84e134Smrg    return (++eventData->count == QLength(dpy)	/* since PeekIf blocks */
7057a84e134Smrg	    || CompareEvents(event, eventData->oldEvent));
7067a84e134Smrg}
7077a84e134Smrg
7087a84e134Smrgstatic Bool
7097a84e134SmrgLookAhead(Widget w, XEvent *event)
7107a84e134Smrg{
7117a84e134Smrg    XEvent newEvent;
7127a84e134Smrg    struct EventData eventData;
7137a84e134Smrg
7147a84e134Smrg    if (QLength(XtDisplay(w)) == 0)
7157a84e134Smrg	return (False);
7167a84e134Smrg
7177a84e134Smrg    eventData.count = 0;
7187a84e134Smrg    eventData.oldEvent = event;
7197a84e134Smrg
7207a84e134Smrg    XPeekIfEvent(XtDisplay(w), &newEvent, PeekNotifyEvent, (char*)&eventData);
7217a84e134Smrg
7227a84e134Smrg    if (CompareEvents(event, &newEvent))
7237a84e134Smrg	return (True);
7247a84e134Smrg
7257a84e134Smrg    return (False);
7267a84e134Smrg}
7277a84e134Smrg
7287a84e134Smrgstatic void
7297a84e134SmrgExtractPosition(XEvent *event, Position *x, Position *y)
7307a84e134Smrg{
7317a84e134Smrg    switch(event->type) {
7327a84e134Smrg	case MotionNotify:
7337a84e134Smrg	    *x = event->xmotion.x;
7347a84e134Smrg	    *y = event->xmotion.y;
7357a84e134Smrg	    break;
7367a84e134Smrg	case ButtonPress:
7377a84e134Smrg	case ButtonRelease:
7387a84e134Smrg	    *x = event->xbutton.x;
7397a84e134Smrg	    *y = event->xbutton.y;
7407a84e134Smrg	    break;
7417a84e134Smrg	case KeyPress:
7427a84e134Smrg	case KeyRelease:
7437a84e134Smrg	    *x = event->xkey.x;
7447a84e134Smrg	    *y = event->xkey.y;
7457a84e134Smrg	    break;
7467a84e134Smrg	case EnterNotify:
7477a84e134Smrg	case LeaveNotify:
7487a84e134Smrg	    *x = event->xcrossing.x;
7497a84e134Smrg	    *y = event->xcrossing.y;
7507a84e134Smrg	    break;
7517a84e134Smrg	default:
7527a84e134Smrg	    *x = 0;
7537a84e134Smrg	    *y = 0;
7547a84e134Smrg	    break;
7557a84e134Smrg    }
7567a84e134Smrg}
7577a84e134Smrg
7587a84e134Smrgstatic void
7597a84e134SmrgNotifyScroll(Widget gw, XEvent *event, String *params, Cardinal *num_params)
7607a84e134Smrg{
7617a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
7627a84e134Smrg    long call_data = 0;
7637a84e134Smrg    char style;
7647a84e134Smrg    Position x, y;
7657a84e134Smrg
7667a84e134Smrg    if (w->scrollbar.direction == 0)	/* if no StartScroll */
7677a84e134Smrg	return;
7687a84e134Smrg
7697a84e134Smrg    if (LookAhead(gw, event))
7707a84e134Smrg	return;
7717a84e134Smrg
7727a84e134Smrg    if (*num_params > 0)
7737a84e134Smrg	style = *params[0];
7747a84e134Smrg    else
7757a84e134Smrg	style = 'P';
7767a84e134Smrg
7777a84e134Smrg    switch(style) {
7787a84e134Smrg	case 'P':    /* Proportional */
7797a84e134Smrg	case 'p':
7807a84e134Smrg	    ExtractPosition(event, &x, &y);
7817a84e134Smrg	    call_data = InRange(PICKLENGTH(w, x, y), 0, (int)w->scrollbar.length);
7827a84e134Smrg	    break;
7837a84e134Smrg	case 'F':    /* FullLength */
7847a84e134Smrg	case 'f':
7857a84e134Smrg	    call_data = w->scrollbar.length;
7867a84e134Smrg	    break;
7877a84e134Smrg    }
7887a84e134Smrg
7897a84e134Smrg    switch(w->scrollbar.direction) {
7907a84e134Smrg	case 'B':
7917a84e134Smrg	case 'b':
7927a84e134Smrg	    call_data = -call_data;
7937a84e134Smrg	    /*FALLTHROUGH*/
7947a84e134Smrg	case 'F':
7957a84e134Smrg	case 'f':
7967a84e134Smrg	    XtCallCallbacks(gw, XtNscrollProc, (XtPointer)call_data);
7977a84e134Smrg	    break;
7987a84e134Smrg	case 'C':
7997a84e134Smrg	case 'c':    /* NotifyThumb has already called the thumbProc(s) */
8007a84e134Smrg	    break;
8017a84e134Smrg    }
8027a84e134Smrg}
8037a84e134Smrg
8047a84e134Smrg/*ARGSUSED*/
8057a84e134Smrgstatic void
8067a84e134SmrgEndScroll(Widget gw, XEvent *event, String *params, Cardinal *num_params)
8077a84e134Smrg{
8087a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
8097a84e134Smrg
8107a84e134Smrg    XtVaSetValues(gw, XtNcursor, w->scrollbar.inactiveCursor, NULL);
8117a84e134Smrg    XFlush(XtDisplay(w));		/* make sure it get propogated */
8127a84e134Smrg
8137a84e134Smrg    w->scrollbar.direction = 0;
8147a84e134Smrg}
8157a84e134Smrg
8167a84e134Smrg/*ARGSUSED*/
8177a84e134Smrgstatic void
8187a84e134SmrgMoveThumb(Widget gw, XEvent *event, String *params, Cardinal *num_params)
8197a84e134Smrg{
8207a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
8217a84e134Smrg    Position x, y;
8227a84e134Smrg
8237a84e134Smrg    if (w->scrollbar.direction == 0)	/* if no StartScroll */
8247a84e134Smrg	return;
8257a84e134Smrg
8267a84e134Smrg    if (LookAhead(gw, event))
8277a84e134Smrg	return;
8287a84e134Smrg
8297a84e134Smrg    if (!event->xmotion.same_screen)
8307a84e134Smrg	return;
8317a84e134Smrg
8327a84e134Smrg    ExtractPosition(event, &x, &y);
8337a84e134Smrg    w->scrollbar.top = FractionLoc(w, x, y);
8347a84e134Smrg}
8357a84e134Smrg
8367a84e134Smrg/*ARGSUSED*/
8377a84e134Smrgstatic void
8387a84e134SmrgNotifyThumb(Widget gw, XEvent *event, String *params, Cardinal *num_params)
8397a84e134Smrg{
8407a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
8417a84e134Smrg    union {
8427a84e134Smrg	XtPointer xtp;
8437a84e134Smrg	float xtf;
8447a84e134Smrg    } xtpf;
8457a84e134Smrg
8467a84e134Smrg    if (w->scrollbar.direction == 0)	/* if no StartScroll */
8477a84e134Smrg	return;
8487a84e134Smrg
8497a84e134Smrg    if (LookAhead(gw, event))
8507a84e134Smrg	return;
8517a84e134Smrg
8527a84e134Smrg    /* thumbProc is not pretty, but is necessary for backwards
8537a84e134Smrg       compatibility on those architectures for which it work{s,ed};
8547a84e134Smrg       the intent is to pass a (truncated) float by value. */
8557a84e134Smrg    xtpf.xtf = w->scrollbar.top;
8567a84e134Smrg    XtCallCallbacks(gw, XtNthumbProc, xtpf.xtp);
8577a84e134Smrg    XtCallCallbacks(gw, XtNjumpProc, (XtPointer)&w->scrollbar.top);
8587a84e134Smrg
8597a84e134Smrg    PaintThumb(w);
8607a84e134Smrg}
8617a84e134Smrg
8627a84e134Smrg/*
8637a84e134Smrg *  Public routines
8647a84e134Smrg */
8657a84e134Smrg/* Set the scroll bar to the given location. */
8667a84e134Smrgvoid
8677a84e134SmrgXawScrollbarSetThumb(Widget gw,
8687a84e134Smrg#if NeedWidePrototypes
8697a84e134Smrg		     double top, double shown
8707a84e134Smrg#else
8717a84e134Smrg		     float top, float shown
8727a84e134Smrg#endif
8737a84e134Smrg		     )
8747a84e134Smrg{
8757a84e134Smrg    ScrollbarWidget w = (ScrollbarWidget)gw;
8767a84e134Smrg
8777a84e134Smrg    if (w->scrollbar.direction == 'c')	/* if still thumbing */
8787a84e134Smrg	return;
8797a84e134Smrg
8807a84e134Smrg    w->scrollbar.top = top > 1.0 ? 1.0 : top >= 0.0 ? top : w->scrollbar.top;
8817a84e134Smrg
8827a84e134Smrg    w->scrollbar.shown = shown > 1.0 ? 1.0 : shown >= 0.0 ?
8837a84e134Smrg				shown : w->scrollbar.shown;
8847a84e134Smrg    PaintThumb(w);
8857a84e134Smrg}
886