Panner.c revision 7a84e134
17a84e134Smrg/*
27a84e134Smrg * $Xorg: Panner.c,v 1.4 2001/02/09 02:03:45 xorgcvs Exp $
37a84e134Smrg *
47a84e134SmrgCopyright 1989, 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
297a84e134Smrg/* $XFree86: xc/lib/Xaw/Panner.c,v 3.8 2001/07/25 15:04:49 dawes Exp $ */
307a84e134Smrg
317a84e134Smrg#ifdef HAVE_CONFIG_H
327a84e134Smrg#include <config.h>
337a84e134Smrg#endif
347a84e134Smrg#include <ctype.h>
357a84e134Smrg#include <math.h>
367a84e134Smrg#include <X11/IntrinsicP.h>
377a84e134Smrg#include <X11/StringDefs.h>
387a84e134Smrg#include <X11/Xos.h>
397a84e134Smrg#include <X11/Xmu/CharSet.h>
407a84e134Smrg#include <X11/Xmu/Drawing.h>
417a84e134Smrg#include <X11/Xmu/Misc.h>
427a84e134Smrg#include <X11/Xaw/PannerP.h>
437a84e134Smrg#include <X11/Xaw/XawInit.h>
447a84e134Smrg#include "Private.h"
457a84e134Smrg
467a84e134Smrg#if defined(ISC) && __STDC__ && !defined(ISC30)
477a84e134Smrgextern double atof(char *);
487a84e134Smrg#else
497a84e134Smrg#include <stdlib.h>			/* for atof() */
507a84e134Smrg#endif
517a84e134Smrg
527a84e134Smrg/*
537a84e134Smrg * Class Methods
547a84e134Smrg */
557a84e134Smrgstatic void XawPannerDestroy(Widget);
567a84e134Smrgstatic void XawPannerInitialize(Widget, Widget, ArgList, Cardinal*);
577a84e134Smrgstatic XtGeometryResult XawPannerQueryGeometry(Widget, XtWidgetGeometry*,
587a84e134Smrg					       XtWidgetGeometry*);
597a84e134Smrgstatic void XawPannerRealize(Widget, XtValueMask*, XSetWindowAttributes*);
607a84e134Smrgstatic void XawPannerRedisplay(Widget, XEvent*, Region);
617a84e134Smrgstatic void XawPannerResize(Widget);
627a84e134Smrgstatic Boolean XawPannerSetValues(Widget, Widget, Widget, ArgList, Cardinal*);
637a84e134Smrgstatic void XawPannerSetValuesAlmost(Widget, Widget, XtWidgetGeometry*,
647a84e134Smrg				     XtWidgetGeometry*);
657a84e134Smrg
667a84e134Smrg/*
677a84e134Smrg * Prototypes
687a84e134Smrg */
697a84e134Smrgstatic void check_knob(PannerWidget, Bool);
707a84e134Smrgstatic void get_default_size(PannerWidget, Dimension*, Dimension*);
717a84e134Smrgstatic Bool get_event_xy(PannerWidget, XEvent*, int*, int*);
727a84e134Smrgstatic void move_shadow(PannerWidget);
737a84e134Smrgstatic int parse_page_string(char*, int, int, Bool*);
747a84e134Smrgstatic void rescale(PannerWidget);
757a84e134Smrgstatic void reset_shadow_gc(PannerWidget);
767a84e134Smrgstatic void reset_slider_gc(PannerWidget);
777a84e134Smrgstatic void reset_xor_gc(PannerWidget);
787a84e134Smrgstatic void scale_knob(PannerWidget, Bool, Bool);
797a84e134Smrg
807a84e134Smrg/*
817a84e134Smrg * Actions
827a84e134Smrg */
837a84e134Smrgstatic void ActionAbort(Widget, XEvent*, String*, Cardinal*);
847a84e134Smrgstatic void ActionMove(Widget, XEvent*, String*, Cardinal*);
857a84e134Smrgstatic void ActionNotify(Widget, XEvent*, String*, Cardinal*);
867a84e134Smrgstatic void ActionPage(Widget, XEvent*, String*, Cardinal*);
877a84e134Smrgstatic void ActionSet(Widget, XEvent*, String*, Cardinal*);
887a84e134Smrgstatic void ActionStart(Widget, XEvent*, String*, Cardinal*);
897a84e134Smrgstatic void ActionStop(Widget, XEvent*, String*, Cardinal*);
907a84e134Smrg
917a84e134Smrg/*
927a84e134Smrg * From Xmu/Distinct.c
937a84e134Smrg */
947a84e134SmrgBool XmuDistinguishablePixels(Display*, Colormap, unsigned long*, int);
957a84e134Smrg
967a84e134Smrg/*
977a84e134Smrg * Initialization
987a84e134Smrg */
997a84e134Smrgstatic char defaultTranslations[] =
1007a84e134Smrg"<Btn1Down>:"		"start()\n"
1017a84e134Smrg"<Btn1Motion>:"		"move()\n"
1027a84e134Smrg"<Btn1Up>:"		"notify() stop()\n"
1037a84e134Smrg"<Btn2Down>:"		"abort()\n"
1047a84e134Smrg":<Key>KP_Enter:"	"set(rubberband,toggle)\n"
1057a84e134Smrg"<Key>space:"		"page(+1p,+1p)\n"
1067a84e134Smrg"<Key>Delete:"		"page(-1p,-1p)\n"
1077a84e134Smrg":<Key>KP_Delete:"	"page(-1p,-1p)\n"
1087a84e134Smrg"<Key>BackSpace:"	"page(-1p,-1p)\n"
1097a84e134Smrg"<Key>Left:"		"page(-.5p,+0)\n"
1107a84e134Smrg":<Key>KP_Left:"	"page(-.5p,+0)\n"
1117a84e134Smrg"<Key>Right:"		"page(+.5p,+0)\n"
1127a84e134Smrg":<Key>KP_Right:"	"page(+.5p,+0)\n"
1137a84e134Smrg"<Key>Up:"		"page(+0,-.5p)\n"
1147a84e134Smrg":<Key>KP_Up:"		"page(+0,-.5p)\n"
1157a84e134Smrg"<Key>Down:"		"page(+0,+.5p)\n"
1167a84e134Smrg":<Key>KP_Down:"	"page(+0,+.5p)\n"
1177a84e134Smrg"<Key>Home:"		"page(0,0)\n"
1187a84e134Smrg":<Key>KP_Home:"	"page(0,0)\n"
1197a84e134Smrg;
1207a84e134Smrg
1217a84e134Smrgstatic XtActionsRec actions[] = {
1227a84e134Smrg  {"start",	ActionStart},		/* start tmp graphics */
1237a84e134Smrg  {"stop",	ActionStop},		/* stop tmp graphics */
1247a84e134Smrg  {"abort",	ActionAbort},		/* punt */
1257a84e134Smrg  {"move",	ActionMove},		/* move tmp graphics on Motion event */
1267a84e134Smrg  {"page",	ActionPage},		/* page around usually from keyboard */
1277a84e134Smrg  {"notify",	ActionNotify},		/* callback new position */
1287a84e134Smrg  {"set",	ActionSet},		/* set various parameters */
1297a84e134Smrg};
1307a84e134Smrg
1317a84e134Smrg#define offset(field)	XtOffsetOf(PannerRec, panner.field)
1327a84e134Smrgstatic XtResource resources[] = {
1337a84e134Smrg    {
1347a84e134Smrg      XtNallowOff,
1357a84e134Smrg      XtCAllowOff,
1367a84e134Smrg      XtRBoolean,
1377a84e134Smrg      sizeof(Boolean),
1387a84e134Smrg      offset(allow_off),
1397a84e134Smrg      XtRImmediate,
1407a84e134Smrg      (XtPointer)False
1417a84e134Smrg    },
1427a84e134Smrg    {
1437a84e134Smrg      XtNresize,
1447a84e134Smrg      XtCResize,
1457a84e134Smrg      XtRBoolean,
1467a84e134Smrg      sizeof(Boolean),
1477a84e134Smrg      offset(resize_to_pref),
1487a84e134Smrg      XtRImmediate,
1497a84e134Smrg      (XtPointer)True
1507a84e134Smrg    },
1517a84e134Smrg    {
1527a84e134Smrg      XtNreportCallback,
1537a84e134Smrg      XtCReportCallback,
1547a84e134Smrg      XtRCallback,
1557a84e134Smrg      sizeof(XtPointer),
1567a84e134Smrg      offset(report_callbacks),
1577a84e134Smrg      XtRCallback,
1587a84e134Smrg      NULL
1597a84e134Smrg    },
1607a84e134Smrg    {
1617a84e134Smrg      XtNdefaultScale,
1627a84e134Smrg      XtCDefaultScale,
1637a84e134Smrg      XtRDimension,
1647a84e134Smrg      sizeof(Dimension),
1657a84e134Smrg      offset(default_scale),
1667a84e134Smrg      XtRImmediate,
1677a84e134Smrg      (XtPointer)PANNER_DEFAULT_SCALE
1687a84e134Smrg    },
1697a84e134Smrg    {
1707a84e134Smrg      XtNrubberBand,
1717a84e134Smrg      XtCRubberBand,
1727a84e134Smrg      XtRBoolean,
1737a84e134Smrg      sizeof(Boolean),
1747a84e134Smrg      offset(rubber_band),
1757a84e134Smrg      XtRImmediate,
1767a84e134Smrg      (XtPointer)False
1777a84e134Smrg    },
1787a84e134Smrg    {
1797a84e134Smrg      XtNforeground,
1807a84e134Smrg      XtCForeground,
1817a84e134Smrg      XtRPixel,
1827a84e134Smrg      sizeof(Pixel),
1837a84e134Smrg      offset(foreground),
1847a84e134Smrg      XtRString,
1857a84e134Smrg      (XtPointer)XtDefaultBackground
1867a84e134Smrg    },
1877a84e134Smrg    {
1887a84e134Smrg      XtNinternalSpace,
1897a84e134Smrg      XtCInternalSpace,
1907a84e134Smrg      XtRDimension,
1917a84e134Smrg      sizeof(Dimension),
1927a84e134Smrg      offset(internal_border),
1937a84e134Smrg      XtRImmediate,
1947a84e134Smrg      (XtPointer)4
1957a84e134Smrg    },
1967a84e134Smrg    {
1977a84e134Smrg      XtNlineWidth,
1987a84e134Smrg      XtCLineWidth,
1997a84e134Smrg      XtRDimension,
2007a84e134Smrg      sizeof(Dimension),
2017a84e134Smrg      offset(line_width),
2027a84e134Smrg      XtRImmediate,
2037a84e134Smrg      (XtPointer)0
2047a84e134Smrg    },
2057a84e134Smrg    {
2067a84e134Smrg      XtNcanvasWidth,
2077a84e134Smrg      XtCCanvasWidth,
2087a84e134Smrg      XtRDimension,
2097a84e134Smrg      sizeof(Dimension),
2107a84e134Smrg      offset(canvas_width),
2117a84e134Smrg      XtRImmediate,
2127a84e134Smrg      (XtPointer)0
2137a84e134Smrg    },
2147a84e134Smrg    {
2157a84e134Smrg      XtNcanvasHeight,
2167a84e134Smrg      XtCCanvasHeight,
2177a84e134Smrg      XtRDimension,
2187a84e134Smrg      sizeof(Dimension),
2197a84e134Smrg      offset(canvas_height),
2207a84e134Smrg      XtRImmediate,
2217a84e134Smrg      (XtPointer)0
2227a84e134Smrg    },
2237a84e134Smrg    {
2247a84e134Smrg      XtNsliderX,
2257a84e134Smrg      XtCSliderX,
2267a84e134Smrg      XtRPosition,
2277a84e134Smrg      sizeof(Position),
2287a84e134Smrg      offset(slider_x),
2297a84e134Smrg      XtRImmediate,
2307a84e134Smrg      (XtPointer)0
2317a84e134Smrg    },
2327a84e134Smrg    {
2337a84e134Smrg      XtNsliderY,
2347a84e134Smrg      XtCSliderY,
2357a84e134Smrg      XtRPosition,
2367a84e134Smrg      sizeof(Position),
2377a84e134Smrg      offset(slider_y),
2387a84e134Smrg      XtRImmediate,
2397a84e134Smrg      (XtPointer)0
2407a84e134Smrg    },
2417a84e134Smrg    {
2427a84e134Smrg      XtNsliderWidth,
2437a84e134Smrg      XtCSliderWidth,
2447a84e134Smrg      XtRDimension,
2457a84e134Smrg      sizeof(Dimension),
2467a84e134Smrg      offset(slider_width),
2477a84e134Smrg      XtRImmediate,
2487a84e134Smrg      (XtPointer)0
2497a84e134Smrg    },
2507a84e134Smrg    {
2517a84e134Smrg      XtNsliderHeight,
2527a84e134Smrg      XtCSliderHeight,
2537a84e134Smrg      XtRDimension,
2547a84e134Smrg      sizeof(Dimension),
2557a84e134Smrg      offset(slider_height),
2567a84e134Smrg      XtRImmediate,
2577a84e134Smrg      (XtPointer)0
2587a84e134Smrg    },
2597a84e134Smrg    {
2607a84e134Smrg      XtNshadowColor,
2617a84e134Smrg      XtCShadowColor,
2627a84e134Smrg      XtRPixel,
2637a84e134Smrg      sizeof(Pixel),
2647a84e134Smrg      offset(shadow_color),
2657a84e134Smrg      XtRString,
2667a84e134Smrg      (XtPointer)XtDefaultForeground
2677a84e134Smrg    },
2687a84e134Smrg    {
2697a84e134Smrg      XtNshadowThickness,
2707a84e134Smrg      XtCShadowThickness,
2717a84e134Smrg      XtRDimension,
2727a84e134Smrg      sizeof(Dimension),
2737a84e134Smrg      offset(shadow_thickness),
2747a84e134Smrg      XtRImmediate,
2757a84e134Smrg      (XtPointer)2
2767a84e134Smrg    },
2777a84e134Smrg    {
2787a84e134Smrg      XtNbackgroundStipple,
2797a84e134Smrg      XtCBackgroundStipple,
2807a84e134Smrg      XtRString,
2817a84e134Smrg      sizeof(String),
2827a84e134Smrg      offset(stipple_name),
2837a84e134Smrg      XtRImmediate,
2847a84e134Smrg      NULL
2857a84e134Smrg    },
2867a84e134Smrg};
2877a84e134Smrg#undef offset
2887a84e134Smrg
2897a84e134Smrg#define Superclass	(&simpleClassRec)
2907a84e134SmrgPannerClassRec pannerClassRec = {
2917a84e134Smrg  /* core */
2927a84e134Smrg  {
2937a84e134Smrg    (WidgetClass)Superclass,		/* superclass */
2947a84e134Smrg    "Panner",				/* class_name */
2957a84e134Smrg    sizeof(PannerRec),			/* widget_size */
2967a84e134Smrg    XawInitializeWidgetSet,		/* class_initialize */
2977a84e134Smrg    NULL,				/* class_part_initialize */
2987a84e134Smrg    False,				/* class_inited */
2997a84e134Smrg    XawPannerInitialize,		/* initialize */
3007a84e134Smrg    NULL,				/* initialize_hook */
3017a84e134Smrg    XawPannerRealize,			/* realize */
3027a84e134Smrg    actions,				/* actions */
3037a84e134Smrg    XtNumber(actions),			/* num_actions */
3047a84e134Smrg    resources,				/* resources */
3057a84e134Smrg    XtNumber(resources),		/* num_resources */
3067a84e134Smrg    NULLQUARK,				/* xrm_class */
3077a84e134Smrg    True,				/* compress_motion */
3087a84e134Smrg    True,				/* compress_exposure */
3097a84e134Smrg    True,				/* compress_enterleave */
3107a84e134Smrg    False,				/* visible_interest */
3117a84e134Smrg    XawPannerDestroy,			/* destroy */
3127a84e134Smrg    XawPannerResize,			/* resize */
3137a84e134Smrg    XawPannerRedisplay,			/* expose */
3147a84e134Smrg    XawPannerSetValues,			/* set_values */
3157a84e134Smrg    NULL,				/* set_values_hook */
3167a84e134Smrg    XawPannerSetValuesAlmost,		/* set_values_almost */
3177a84e134Smrg    NULL,				/* get_values_hook */
3187a84e134Smrg    NULL,				/* accept_focus */
3197a84e134Smrg    XtVersion,				/* version */
3207a84e134Smrg    NULL,				/* callback_private */
3217a84e134Smrg    defaultTranslations,		/* tm_table */
3227a84e134Smrg    XawPannerQueryGeometry,		/* query_geometry */
3237a84e134Smrg    XtInheritDisplayAccelerator,	/* display_accelerator */
3247a84e134Smrg    NULL,				/* extension */
3257a84e134Smrg  },
3267a84e134Smrg  /* simple */
3277a84e134Smrg  {
3287a84e134Smrg    XtInheritChangeSensitive,		/* change_sensitive */
3297a84e134Smrg  },
3307a84e134Smrg  /* panner */
3317a84e134Smrg  {
3327a84e134Smrg    NULL,				/* extension */
3337a84e134Smrg  }
3347a84e134Smrg};
3357a84e134Smrg
3367a84e134SmrgWidgetClass pannerWidgetClass = (WidgetClass) &pannerClassRec;
3377a84e134Smrg
3387a84e134Smrg
3397a84e134Smrg/*
3407a84e134Smrg * Implementation
3417a84e134Smrg */
3427a84e134Smrgstatic void
3437a84e134Smrgreset_shadow_gc(PannerWidget pw)
3447a84e134Smrg{
3457a84e134Smrg    XtGCMask valuemask = GCForeground;
3467a84e134Smrg    XGCValues values;
3477a84e134Smrg    unsigned long   pixels[3];
3487a84e134Smrg
3497a84e134Smrg    if (pw->panner.shadow_gc)
3507a84e134Smrg	XtReleaseGC((Widget)pw, pw->panner.shadow_gc);
3517a84e134Smrg
3527a84e134Smrg    pixels[0] = pw->panner.foreground;
3537a84e134Smrg    pixels[1] = pw->core.background_pixel;
3547a84e134Smrg    pixels[2] = pw->panner.shadow_color;
3557a84e134Smrg
3567a84e134Smrg    if (!pw->panner.stipple_name &&
3577a84e134Smrg	!XmuDistinguishablePixels(XtDisplay(pw), pw->core.colormap,
3587a84e134Smrg				  pixels, 3) &&
3597a84e134Smrg	XmuDistinguishablePixels(XtDisplay(pw), pw->core.colormap,
3607a84e134Smrg				 pixels, 2)) {
3617a84e134Smrg	valuemask = GCTile | GCFillStyle;
3627a84e134Smrg	values.fill_style = FillTiled;
3637a84e134Smrg	values.tile = XmuCreateStippledPixmap(XtScreen((Widget)pw),
3647a84e134Smrg					      pw->panner.foreground,
3657a84e134Smrg					      pw->core.background_pixel,
3667a84e134Smrg					      pw->core.depth);
3677a84e134Smrg    }
3687a84e134Smrg    else {
3697a84e134Smrg	if (!pw->panner.line_width &&
3707a84e134Smrg	    !XmuDistinguishablePixels(XtDisplay(pw), pw->core.colormap,
3717a84e134Smrg				      pixels, 2))
3727a84e134Smrg	    pw->panner.line_width = 1;
3737a84e134Smrg	valuemask = GCForeground;
3747a84e134Smrg	values.foreground = pw->panner.shadow_color;
3757a84e134Smrg    }
3767a84e134Smrg    if (pw->panner.line_width > 0) {
3777a84e134Smrg	values.line_width = pw->panner.line_width;
3787a84e134Smrg	valuemask |= GCLineWidth;
3797a84e134Smrg    }
3807a84e134Smrg
3817a84e134Smrg    pw->panner.shadow_gc = XtGetGC((Widget)pw, valuemask, &values);
3827a84e134Smrg}
3837a84e134Smrg
3847a84e134Smrgstatic void
3857a84e134Smrgreset_slider_gc(PannerWidget pw)
3867a84e134Smrg{
3877a84e134Smrg    XtGCMask valuemask = GCForeground;
3887a84e134Smrg    XGCValues values;
3897a84e134Smrg
3907a84e134Smrg    if (pw->panner.slider_gc)
3917a84e134Smrg	XtReleaseGC((Widget)pw, pw->panner.slider_gc);
3927a84e134Smrg
3937a84e134Smrg    values.foreground = pw->panner.foreground;
3947a84e134Smrg
3957a84e134Smrg    pw->panner.slider_gc = XtGetGC((Widget)pw, valuemask, &values);
3967a84e134Smrg}
3977a84e134Smrg
3987a84e134Smrgstatic void
3997a84e134Smrgreset_xor_gc(PannerWidget pw)
4007a84e134Smrg{
4017a84e134Smrg    if (pw->panner.xor_gc)
4027a84e134Smrg	XtReleaseGC((Widget)pw, pw->panner.xor_gc);
4037a84e134Smrg
4047a84e134Smrg    if (pw->panner.rubber_band) {
4057a84e134Smrg	XtGCMask valuemask = (GCForeground | GCFunction);
4067a84e134Smrg	XGCValues values;
4077a84e134Smrg	Pixel tmp;
4087a84e134Smrg
4097a84e134Smrg	tmp = (pw->panner.foreground == pw->core.background_pixel ?
4107a84e134Smrg	       pw->panner.shadow_color : pw->panner.foreground);
4117a84e134Smrg	values.foreground = tmp ^ pw->core.background_pixel;
4127a84e134Smrg	values.function = GXxor;
4137a84e134Smrg	if (pw->panner.line_width > 0) {
4147a84e134Smrg	    valuemask |= GCLineWidth;
4157a84e134Smrg	    values.line_width = pw->panner.line_width;
4167a84e134Smrg	}
4177a84e134Smrg	pw->panner.xor_gc = XtGetGC((Widget)pw, valuemask, &values);
4187a84e134Smrg    }
4197a84e134Smrg    else
4207a84e134Smrg	pw->panner.xor_gc = NULL;
4217a84e134Smrg}
4227a84e134Smrg
4237a84e134Smrgstatic void
4247a84e134Smrgcheck_knob(PannerWidget pw, Bool knob)
4257a84e134Smrg{
4267a84e134Smrg    Position pad = pw->panner.internal_border << 1;
4277a84e134Smrg    Position maxx = (Position)XtWidth(pw) - pad -
4287a84e134Smrg		    (Position)pw->panner.knob_width;
4297a84e134Smrg    Position maxy = (Position)XtHeight(pw) - pad -
4307a84e134Smrg		    (Position)pw->panner.knob_height;
4317a84e134Smrg    Position *x = knob ? &pw->panner.knob_x : &pw->panner.tmp.x;
4327a84e134Smrg    Position *y = knob ? &pw->panner.knob_y : &pw->panner.tmp.y;
4337a84e134Smrg
4347a84e134Smrg    /*
4357a84e134Smrg     * note that positions are already normalized (i.e. internal_border
4367a84e134Smrg     * has been subtracted out)
4377a84e134Smrg     */
4387a84e134Smrg    if (*x < 0)
4397a84e134Smrg	*x = 0;
4407a84e134Smrg    if (*x > maxx)
4417a84e134Smrg	*x = maxx;
4427a84e134Smrg
4437a84e134Smrg    if (*y < 0)
4447a84e134Smrg	*y = 0;
4457a84e134Smrg    if (*y > maxy)
4467a84e134Smrg	*y = maxy;
4477a84e134Smrg
4487a84e134Smrg    if (knob) {
4497a84e134Smrg	pw->panner.slider_x = (Position)((double)pw->panner.knob_x
4507a84e134Smrg					/ pw->panner.haspect + 0.5);
4517a84e134Smrg	pw->panner.slider_y = (Position)((double)pw->panner.knob_y
4527a84e134Smrg					/ pw->panner.vaspect + 0.5);
4537a84e134Smrg	pw->panner.last_x = pw->panner.last_y = PANNER_OUTOFRANGE;
4547a84e134Smrg    }
4557a84e134Smrg}
4567a84e134Smrg
4577a84e134Smrgstatic void
4587a84e134Smrgmove_shadow(PannerWidget pw)
4597a84e134Smrg{
4607a84e134Smrg    if (pw->panner.shadow_thickness > 0) {
4617a84e134Smrg	int lw = pw->panner.shadow_thickness + (pw->panner.line_width << 1);
4627a84e134Smrg	int pad = pw->panner.internal_border;
4637a84e134Smrg
4647a84e134Smrg	if (pw->panner.knob_height > lw && pw->panner.knob_width > lw) {
4657a84e134Smrg	    XRectangle *r = pw->panner.shadow_rects;
4667a84e134Smrg
4677a84e134Smrg	    r->x = pw->panner.knob_x + pad + pw->panner.knob_width;
4687a84e134Smrg	    r->y = pw->panner.knob_y + pad + lw;
4697a84e134Smrg	    r->width = pw->panner.shadow_thickness;
4707a84e134Smrg	    r->height = pw->panner.knob_height - lw;
4717a84e134Smrg	    r++;
4727a84e134Smrg	    r->x = pw->panner.knob_x + pad + lw;
4737a84e134Smrg	    r->y = pw->panner.knob_y + pad + pw->panner.knob_height;
4747a84e134Smrg	    r->width = pw->panner.knob_width - lw + pw->panner.shadow_thickness;
4757a84e134Smrg	    r->height = pw->panner.shadow_thickness;
4767a84e134Smrg	    pw->panner.shadow_valid = True;
4777a84e134Smrg	    return;
4787a84e134Smrg	}
4797a84e134Smrg    }
4807a84e134Smrg    pw->panner.shadow_valid = False;
4817a84e134Smrg}
4827a84e134Smrg
4837a84e134Smrgstatic void
4847a84e134Smrgscale_knob(PannerWidget pw, Bool location, Bool size)
4857a84e134Smrg{
4867a84e134Smrg    if (location) {
4877a84e134Smrg	pw->panner.knob_x = (Position)PANNER_HSCALE(pw, pw->panner.slider_x);
4887a84e134Smrg	pw->panner.knob_y = (Position)PANNER_VSCALE(pw, pw->panner.slider_y);
4897a84e134Smrg    }
4907a84e134Smrg    if (size) {
4917a84e134Smrg	Dimension width, height;
4927a84e134Smrg
4937a84e134Smrg	if (pw->panner.slider_width < 1)
4947a84e134Smrg	    pw->panner.slider_width = pw->panner.canvas_width;
4957a84e134Smrg	if (pw->panner.slider_height < 1)
4967a84e134Smrg	    pw->panner.slider_height = pw->panner.canvas_height;
4977a84e134Smrg	width = Min(pw->panner.slider_width, pw->panner.canvas_width);
4987a84e134Smrg	height = Min(pw->panner.slider_height, pw->panner.canvas_height);
4997a84e134Smrg
5007a84e134Smrg	pw->panner.knob_width = (Dimension)PANNER_HSCALE(pw, width);
5017a84e134Smrg	pw->panner.knob_height = (Dimension)PANNER_VSCALE(pw, height);
5027a84e134Smrg    }
5037a84e134Smrg    if (!pw->panner.allow_off)
5047a84e134Smrg	check_knob(pw, True);
5057a84e134Smrg    move_shadow(pw);
5067a84e134Smrg}
5077a84e134Smrg
5087a84e134Smrgstatic void
5097a84e134Smrgrescale(PannerWidget pw)
5107a84e134Smrg{
5117a84e134Smrg    int hpad = pw->panner.internal_border << 1;
5127a84e134Smrg    int vpad = hpad;
5137a84e134Smrg
5147a84e134Smrg    if (pw->panner.canvas_width < 1)
5157a84e134Smrg	pw->panner.canvas_width = XtWidth(pw);
5167a84e134Smrg    if (pw->panner.canvas_height < 1)
5177a84e134Smrg	pw->panner.canvas_height = XtHeight(pw);
5187a84e134Smrg
5197a84e134Smrg    if (XtWidth(pw) <= hpad)
5207a84e134Smrg	hpad = 0;
5217a84e134Smrg    if (XtHeight(pw) <= vpad)
5227a84e134Smrg	vpad = 0;
5237a84e134Smrg
5247a84e134Smrg    pw->panner.haspect = ((double)XtWidth(pw) - hpad + .5)
5257a84e134Smrg			 / (double)pw->panner.canvas_width;
5267a84e134Smrg    pw->panner.vaspect = ((double)XtHeight(pw) - vpad + .5)
5277a84e134Smrg			 / (double)pw->panner.canvas_height;
5287a84e134Smrg    scale_knob(pw, True, True);
5297a84e134Smrg}
5307a84e134Smrg
5317a84e134Smrgstatic void
5327a84e134Smrgget_default_size(PannerWidget pw, Dimension *wp, Dimension *hp)
5337a84e134Smrg{
5347a84e134Smrg    Dimension pad = pw->panner.internal_border << 1;
5357a84e134Smrg
5367a84e134Smrg    *wp = PANNER_DSCALE(pw, pw->panner.canvas_width) + pad;
5377a84e134Smrg    *hp = PANNER_DSCALE(pw, pw->panner.canvas_height) + pad;
5387a84e134Smrg}
5397a84e134Smrg
5407a84e134Smrgstatic Bool
5417a84e134Smrgget_event_xy(PannerWidget pw, XEvent *event, int *x, int *y)
5427a84e134Smrg{
5437a84e134Smrg    int pad = pw->panner.internal_border;
5447a84e134Smrg
5457a84e134Smrg    switch (event->type) {
5467a84e134Smrg	case ButtonPress:
5477a84e134Smrg	case ButtonRelease:
5487a84e134Smrg	    *x = event->xbutton.x - pad;
5497a84e134Smrg	    *y = event->xbutton.y - pad;
5507a84e134Smrg	    return (True);
5517a84e134Smrg	case KeyPress:
5527a84e134Smrg	case KeyRelease:
5537a84e134Smrg	    *x = event->xkey.x - pad;
5547a84e134Smrg	    *y = event->xkey.y - pad;
5557a84e134Smrg	    return (True);
5567a84e134Smrg	case EnterNotify:
5577a84e134Smrg	case LeaveNotify:
5587a84e134Smrg	    *x = event->xcrossing.x - pad;
5597a84e134Smrg	    *y = event->xcrossing.y - pad;
5607a84e134Smrg	    return (True);
5617a84e134Smrg	case MotionNotify:
5627a84e134Smrg	    *x = event->xmotion.x - pad;
5637a84e134Smrg	    *y = event->xmotion.y - pad;
5647a84e134Smrg	    return (True);
5657a84e134Smrg    }
5667a84e134Smrg
5677a84e134Smrg    return (False);
5687a84e134Smrg}
5697a84e134Smrg
5707a84e134Smrgstatic int
5717a84e134Smrgparse_page_string(char *s, int pagesize, int canvassize, Bool *relative)
5727a84e134Smrg{
5737a84e134Smrg    char *cp;
5747a84e134Smrg    double val = 1.0;
5757a84e134Smrg    Bool rel = False;
5767a84e134Smrg
5777a84e134Smrg    /*
5787a84e134Smrg     * syntax:    spaces [+-] number spaces [pc\0] spaces
5797a84e134Smrg     */
5807a84e134Smrg    for (; isascii(*s) && isspace(*s); s++)	/* skip white space */
5817a84e134Smrg	;
5827a84e134Smrg
5837a84e134Smrg    if (*s == '+' || *s == '-')	{		/* deal with signs */
5847a84e134Smrg	rel = True;
5857a84e134Smrg	if (*s == '-')
5867a84e134Smrg	    val = -1.0;
5877a84e134Smrg	s++;
5887a84e134Smrg    }
5897a84e134Smrg    if (!*s) {				/* if null then return nothing */
5907a84e134Smrg	*relative = True;
5917a84e134Smrg	return (0);
5927a84e134Smrg    }
5937a84e134Smrg
5947a84e134Smrg					/* skip over numbers */
5957a84e134Smrg    for (cp = s; isascii(*s) && (isdigit(*s) || *s == '.'); s++)
5967a84e134Smrg	;
5977a84e134Smrg    val *= atof(cp);
5987a84e134Smrg
5997a84e134Smrg					/* skip blanks */
6007a84e134Smrg    for (; isascii(*s) && isspace(*s); s++)
6017a84e134Smrg	;
6027a84e134Smrg
6037a84e134Smrg    if (*s) {				/* if units */
6047a84e134Smrg	switch (s[0]) {
6057a84e134Smrg	    case 'p':
6067a84e134Smrg	    case 'P':
6077a84e134Smrg		val *= (double)pagesize;
6087a84e134Smrg		break;
6097a84e134Smrg	    case 'c':
6107a84e134Smrg	    case 'C':
6117a84e134Smrg		val *= (double)canvassize;
6127a84e134Smrg		break;
6137a84e134Smrg	}
6147a84e134Smrg    }
6157a84e134Smrg    *relative = rel;
6167a84e134Smrg
6177a84e134Smrg    return ((int)val);
6187a84e134Smrg}
6197a84e134Smrg
6207a84e134Smrg#define DRAW_TMP(pw) \
6217a84e134Smrg{ \
6227a84e134Smrg    XDrawRectangle(XtDisplay(pw), XtWindow(pw),				\
6237a84e134Smrg		   pw->panner.xor_gc,					\
6247a84e134Smrg		   pw->panner.tmp.x + pw->panner.internal_border,	\
6257a84e134Smrg		   pw->panner.tmp.y + pw->panner.internal_border,	\
6267a84e134Smrg		   pw->panner.knob_width - 1,				\
6277a84e134Smrg		   pw->panner.knob_height - 1);				\
6287a84e134Smrg    pw->panner.tmp.showing = !pw->panner.tmp.showing;			\
6297a84e134Smrg}
6307a84e134Smrg
6317a84e134Smrg#define UNDRAW_TMP(pw) \
6327a84e134Smrg{ \
6337a84e134Smrg    if (pw->panner.tmp.showing)			\
6347a84e134Smrg      DRAW_TMP(pw);				\
6357a84e134Smrg}
6367a84e134Smrg
6377a84e134Smrg#define BACKGROUND_STIPPLE(pw) \
6387a84e134SmrgXmuLocatePixmapFile(pw->core.screen, pw->panner.stipple_name,		\
6397a84e134Smrg		    pw->panner.shadow_color, pw->core.background_pixel,	\
6407a84e134Smrg		    pw->core.depth, NULL, 0, NULL, NULL, NULL, NULL)
6417a84e134Smrg
6427a84e134Smrg#define PIXMAP_OKAY(pm) ((pm) != None && (pm) != XtUnspecifiedPixmap)
6437a84e134Smrg
6447a84e134Smrg/*ARGSUSED*/
6457a84e134Smrgstatic void
6467a84e134SmrgXawPannerInitialize(Widget greq, Widget gnew, ArgList args, Cardinal *num_args)
6477a84e134Smrg{
6487a84e134Smrg    PannerWidget req = (PannerWidget)greq, cnew = (PannerWidget)gnew;
6497a84e134Smrg    Dimension defwidth, defheight;
6507a84e134Smrg
6517a84e134Smrg    if (req->panner.canvas_width < 1)
6527a84e134Smrg	cnew->panner.canvas_width = 1;
6537a84e134Smrg    if (req->panner.canvas_height < 1)
6547a84e134Smrg	cnew->panner.canvas_height = 1;
6557a84e134Smrg    if (req->panner.default_scale < 1)
6567a84e134Smrg	cnew->panner.default_scale = PANNER_DEFAULT_SCALE;
6577a84e134Smrg
6587a84e134Smrg    get_default_size(req, &defwidth, &defheight);
6597a84e134Smrg    if (XtWidth(req) < 1)
6607a84e134Smrg	XtWidth(cnew) = defwidth;
6617a84e134Smrg    if (XtHeight(req) < 1)
6627a84e134Smrg	XtHeight(cnew) = defheight;
6637a84e134Smrg
6647a84e134Smrg    cnew->panner.shadow_gc = NULL;
6657a84e134Smrg    reset_shadow_gc(cnew);		/* shadowColor */
6667a84e134Smrg    cnew->panner.slider_gc = NULL;
6677a84e134Smrg    reset_slider_gc(cnew);		/* foreground */
6687a84e134Smrg    cnew->panner.xor_gc = NULL;
6697a84e134Smrg    reset_xor_gc(cnew); 		/* foreground ^ background */
6707a84e134Smrg
6717a84e134Smrg    rescale(cnew);			/* does a position check */
6727a84e134Smrg    cnew->panner.shadow_valid = False;
6737a84e134Smrg    cnew->panner.tmp.doing = False;
6747a84e134Smrg    cnew->panner.tmp.showing = False;
6757a84e134Smrg  }
6767a84e134Smrg
6777a84e134Smrgstatic void
6787a84e134SmrgXawPannerRealize(Widget gw, XtValueMask *valuemaskp,
6797a84e134Smrg		 XSetWindowAttributes *attr)
6807a84e134Smrg{
6817a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
6827a84e134Smrg    Pixmap pm = XtUnspecifiedPixmap;
6837a84e134Smrg    Bool gotpm = False;
6847a84e134Smrg
6857a84e134Smrg    if (pw->core.background_pixmap == XtUnspecifiedPixmap) {
6867a84e134Smrg	if (pw->panner.stipple_name)
6877a84e134Smrg	    pm = BACKGROUND_STIPPLE(pw);
6887a84e134Smrg
6897a84e134Smrg	if (PIXMAP_OKAY(pm)) {
6907a84e134Smrg	    attr->background_pixmap = pm;
6917a84e134Smrg	    *valuemaskp |= CWBackPixmap;
6927a84e134Smrg	    *valuemaskp &= ~CWBackPixel;
6937a84e134Smrg	    gotpm = True;
6947a84e134Smrg	}
6957a84e134Smrg    }
6967a84e134Smrg    (*pannerWidgetClass->core_class.superclass->core_class.realize)
6977a84e134Smrg	(gw, valuemaskp, attr);
6987a84e134Smrg
6997a84e134Smrg    if (gotpm)
7007a84e134Smrg	XFreePixmap(XtDisplay(gw), pm);
7017a84e134Smrg}
7027a84e134Smrg
7037a84e134Smrgstatic void
7047a84e134SmrgXawPannerDestroy(Widget gw)
7057a84e134Smrg{
7067a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
7077a84e134Smrg
7087a84e134Smrg    XtReleaseGC(gw, pw->panner.shadow_gc);
7097a84e134Smrg    XtReleaseGC(gw, pw->panner.slider_gc);
7107a84e134Smrg    XtReleaseGC(gw, pw->panner.xor_gc);
7117a84e134Smrg}
7127a84e134Smrg
7137a84e134Smrgstatic void
7147a84e134SmrgXawPannerResize(Widget gw)
7157a84e134Smrg{
7167a84e134Smrg    rescale((PannerWidget)gw);
7177a84e134Smrg}
7187a84e134Smrg
7197a84e134Smrgstatic void
7207a84e134SmrgXawPannerRedisplay(Widget gw, XEvent *event, Region region)
7217a84e134Smrg{
7227a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
7237a84e134Smrg    Display *dpy = XtDisplay(gw);
7247a84e134Smrg    Window w = XtWindow(gw);
7257a84e134Smrg    int pad = pw->panner.internal_border;
7267a84e134Smrg    Dimension lw = pw->panner.line_width;
7277a84e134Smrg    Dimension extra = pw->panner.shadow_thickness + (lw << 1);
7287a84e134Smrg    int kx = pw->panner.knob_x + pad, ky = pw->panner.knob_y + pad;
7297a84e134Smrg
7307a84e134Smrg    if (Superclass->core_class.expose)
7317a84e134Smrg	(Superclass->core_class.expose)(gw, event, region);
7327a84e134Smrg
7337a84e134Smrg    pw->panner.tmp.showing = False;
7347a84e134Smrg    XClearArea(XtDisplay(pw), XtWindow(pw),
7357a84e134Smrg	       (int)pw->panner.last_x - ((int)lw) + pad,
7367a84e134Smrg	       (int)pw->panner.last_y - ((int)lw) + pad,
7377a84e134Smrg	       pw->panner.knob_width + extra,
7387a84e134Smrg	       pw->panner.knob_height + extra,
7397a84e134Smrg	       False);
7407a84e134Smrg    pw->panner.last_x = pw->panner.knob_x;
7417a84e134Smrg    pw->panner.last_y = pw->panner.knob_y;
7427a84e134Smrg
7437a84e134Smrg    XFillRectangle(dpy, w, pw->panner.slider_gc, kx, ky,
7447a84e134Smrg		   pw->panner.knob_width - 1, pw->panner.knob_height - 1);
7457a84e134Smrg
7467a84e134Smrg    if (lw)
7477a84e134Smrg	XDrawRectangle(dpy, w, pw->panner.shadow_gc, kx, ky,
7487a84e134Smrg		       pw->panner.knob_width - 1,  pw->panner.knob_height - 1);
7497a84e134Smrg
7507a84e134Smrg    if (pw->panner.shadow_valid)
7517a84e134Smrg	XFillRectangles(dpy, w, pw->panner.shadow_gc, pw->panner.shadow_rects, 2);
7527a84e134Smrg
7537a84e134Smrg    if (pw->panner.tmp.doing && pw->panner.rubber_band)
7547a84e134Smrg	DRAW_TMP(pw);
7557a84e134Smrg}
7567a84e134Smrg
7577a84e134Smrg/*ARGSUSED*/
7587a84e134Smrgstatic Boolean
7597a84e134SmrgXawPannerSetValues(Widget gcur, Widget greq, Widget gnew,
7607a84e134Smrg		   ArgList args, Cardinal *num_args)
7617a84e134Smrg{
7627a84e134Smrg    PannerWidget cur = (PannerWidget)gcur;
7637a84e134Smrg    PannerWidget cnew = (PannerWidget)gnew;
7647a84e134Smrg    Bool redisplay = False;
7657a84e134Smrg
7667a84e134Smrg    if (cur->panner.foreground != cnew->panner.foreground) {
7677a84e134Smrg	reset_slider_gc(cnew);
7687a84e134Smrg	if (cur->panner.foreground != cur->core.background_pixel)
7697a84e134Smrg	    reset_xor_gc(cnew);
7707a84e134Smrg	redisplay = True;
7717a84e134Smrg    }
7727a84e134Smrg    else if (cur->panner.line_width != cnew->panner.line_width ||
7737a84e134Smrg	     cur->core.background_pixel != cnew->core.background_pixel) {
7747a84e134Smrg	reset_xor_gc(cnew);
7757a84e134Smrg	redisplay = True;
7767a84e134Smrg    }
7777a84e134Smrg    if (cur->panner.shadow_color != cnew->panner.shadow_color) {
7787a84e134Smrg	reset_shadow_gc(cnew);
7797a84e134Smrg	if (cur->panner.foreground == cur->core.background_pixel)
7807a84e134Smrg	    reset_xor_gc(cnew);
7817a84e134Smrg	redisplay = True;
7827a84e134Smrg    }
7837a84e134Smrg    if (cur->panner.shadow_thickness != cnew->panner.shadow_thickness) {
7847a84e134Smrg	move_shadow(cnew);
7857a84e134Smrg	redisplay = True;
7867a84e134Smrg    }
7877a84e134Smrg    if (cur->panner.rubber_band != cnew->panner.rubber_band) {
7887a84e134Smrg	reset_xor_gc(cnew);
7897a84e134Smrg	if (cnew->panner.tmp.doing)
7907a84e134Smrg	    redisplay = True;
7917a84e134Smrg    }
7927a84e134Smrg
7937a84e134Smrg    if ((cur->panner.stipple_name != cnew->panner.stipple_name
7947a84e134Smrg	 || cur->panner.shadow_color != cnew->panner.shadow_color
7957a84e134Smrg	 || cur->core.background_pixel != cnew->core.background_pixel)
7967a84e134Smrg	&& XtIsRealized(gnew)) {
7977a84e134Smrg	Pixmap pm = cnew->panner.stipple_name ?
7987a84e134Smrg			BACKGROUND_STIPPLE(cnew) : XtUnspecifiedPixmap;
7997a84e134Smrg
8007a84e134Smrg	if (PIXMAP_OKAY(pm)) {
8017a84e134Smrg	    XSetWindowBackgroundPixmap(XtDisplay(cnew), XtWindow(cnew), pm);
8027a84e134Smrg	    XFreePixmap(XtDisplay(cnew), pm);
8037a84e134Smrg	}
8047a84e134Smrg	else
8057a84e134Smrg	    XSetWindowBackground(XtDisplay(cnew), XtWindow(cnew),
8067a84e134Smrg				 cnew->core.background_pixel);
8077a84e134Smrg
8087a84e134Smrg	redisplay = True;
8097a84e134Smrg    }
8107a84e134Smrg
8117a84e134Smrg    if (cnew->panner.resize_to_pref &&
8127a84e134Smrg	(cur->panner.canvas_width != cnew->panner.canvas_width
8137a84e134Smrg	 || cur->panner.canvas_height != cnew->panner.canvas_height
8147a84e134Smrg	 || cur->panner.resize_to_pref != cnew->panner.resize_to_pref)) {
8157a84e134Smrg	get_default_size(cnew, &cnew->core.width, &cnew->core.height);
8167a84e134Smrg	redisplay = True;
8177a84e134Smrg    }
8187a84e134Smrg    else if (cur->panner.canvas_width != cnew->panner.canvas_width
8197a84e134Smrg	     || cur->panner.canvas_height != cnew->panner.canvas_height
8207a84e134Smrg	     || cur->panner.internal_border != cnew->panner.internal_border) {
8217a84e134Smrg	rescale(cnew);			/* does a scale_knob as well */
8227a84e134Smrg	redisplay = True;
8237a84e134Smrg    }
8247a84e134Smrg    else {
8257a84e134Smrg	Bool loc = cur->panner.slider_x != cnew->panner.slider_x ||
8267a84e134Smrg		   cur->panner.slider_y != cnew->panner.slider_y;
8277a84e134Smrg	Bool siz = cur->panner.slider_width != cnew->panner.slider_width ||
8287a84e134Smrg	 	   cur->panner.slider_height != cnew->panner.slider_height;
8297a84e134Smrg	if (loc || siz || (cur->panner.allow_off != cnew->panner.allow_off
8307a84e134Smrg			   && cnew->panner.allow_off)) {
8317a84e134Smrg	    scale_knob(cnew, loc, siz);
8327a84e134Smrg	    redisplay = True;
8337a84e134Smrg	}
8347a84e134Smrg    }
8357a84e134Smrg
8367a84e134Smrg    return (redisplay);
8377a84e134Smrg}
8387a84e134Smrg
8397a84e134Smrgstatic void
8407a84e134SmrgXawPannerSetValuesAlmost(Widget gold, Widget gnew, XtWidgetGeometry *req,
8417a84e134Smrg			 XtWidgetGeometry *reply)
8427a84e134Smrg{
8437a84e134Smrg    if (reply->request_mode == 0)	/* got turned down, so cope */
8447a84e134Smrg	XawPannerResize(gnew);
8457a84e134Smrg
8467a84e134Smrg    (*pannerWidgetClass->core_class.superclass->core_class.set_values_almost)
8477a84e134Smrg	(gold, gnew, req, reply);
8487a84e134Smrg}
8497a84e134Smrg
8507a84e134Smrgstatic XtGeometryResult
8517a84e134SmrgXawPannerQueryGeometry(Widget gw, XtWidgetGeometry *intended,
8527a84e134Smrg		       XtWidgetGeometry *pref)
8537a84e134Smrg{
8547a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
8557a84e134Smrg
8567a84e134Smrg    pref->request_mode = (CWWidth | CWHeight);
8577a84e134Smrg    get_default_size(pw, &pref->width, &pref->height);
8587a84e134Smrg
8597a84e134Smrg    if (((intended->request_mode & (CWWidth | CWHeight)) == (CWWidth | CWHeight))
8607a84e134Smrg	&& intended->width == pref->width && intended->height == pref->height)
8617a84e134Smrg	return (XtGeometryYes);
8627a84e134Smrg    else if (pref->width == XtWidth(pw) && pref->height == XtHeight(pw))
8637a84e134Smrg	return (XtGeometryNo);
8647a84e134Smrg
8657a84e134Smrg    return (XtGeometryAlmost);
8667a84e134Smrg}
8677a84e134Smrg
8687a84e134Smrg
8697a84e134Smrg/*ARGSUSED*/
8707a84e134Smrgstatic void
8717a84e134SmrgActionStart(Widget gw, XEvent *event, String *params, Cardinal *num_params)
8727a84e134Smrg{
8737a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
8747a84e134Smrg    int x, y;
8757a84e134Smrg
8767a84e134Smrg    if (!get_event_xy(pw, event, &x, &y)) {
8777a84e134Smrg	XBell(XtDisplay(gw), 0);
8787a84e134Smrg	return;
8797a84e134Smrg    }
8807a84e134Smrg
8817a84e134Smrg    pw->panner.tmp.doing = True;
8827a84e134Smrg    pw->panner.tmp.startx = pw->panner.knob_x;
8837a84e134Smrg    pw->panner.tmp.starty = pw->panner.knob_y;
8847a84e134Smrg    pw->panner.tmp.dx = x - pw->panner.knob_x;
8857a84e134Smrg    pw->panner.tmp.dy = y - pw->panner.knob_y;
8867a84e134Smrg    pw->panner.tmp.x = pw->panner.knob_x;
8877a84e134Smrg    pw->panner.tmp.y = pw->panner.knob_y;
8887a84e134Smrg    if (pw->panner.rubber_band)
8897a84e134Smrg	DRAW_TMP(pw);
8907a84e134Smrg}
8917a84e134Smrg
8927a84e134Smrg/*ARGSUSED*/
8937a84e134Smrgstatic void
8947a84e134SmrgActionStop(Widget gw, XEvent *event, String *params, Cardinal *num_params)
8957a84e134Smrg{
8967a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
8977a84e134Smrg    int x, y;
8987a84e134Smrg
8997a84e134Smrg    if (get_event_xy(pw, event, &x, &y)) {
9007a84e134Smrg	pw->panner.tmp.x = x - pw->panner.tmp.dx;
9017a84e134Smrg	pw->panner.tmp.y = y - pw->panner.tmp.dy;
9027a84e134Smrg	if (!pw->panner.allow_off)
9037a84e134Smrg	    check_knob(pw, False);
9047a84e134Smrg    }
9057a84e134Smrg    if (pw->panner.rubber_band)
9067a84e134Smrg	DRAW_TMP(pw);
9077a84e134Smrg    pw->panner.tmp.doing = False;
9087a84e134Smrg}
9097a84e134Smrg
9107a84e134Smrgstatic void
9117a84e134SmrgActionAbort(Widget gw, XEvent *event, String *params, Cardinal *num_params)
9127a84e134Smrg{
9137a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
9147a84e134Smrg
9157a84e134Smrg    if (!pw->panner.tmp.doing)
9167a84e134Smrg	return;
9177a84e134Smrg
9187a84e134Smrg    if (pw->panner.rubber_band)
9197a84e134Smrg	UNDRAW_TMP(pw);
9207a84e134Smrg
9217a84e134Smrg    if (!pw->panner.rubber_band) {		/* restore old position */
9227a84e134Smrg	pw->panner.tmp.x = pw->panner.tmp.startx;
9237a84e134Smrg	pw->panner.tmp.y = pw->panner.tmp.starty;
9247a84e134Smrg	ActionNotify(gw, event, params, num_params);
9257a84e134Smrg    }
9267a84e134Smrg    pw->panner.tmp.doing = False;
9277a84e134Smrg}
9287a84e134Smrg
9297a84e134Smrgstatic void
9307a84e134SmrgActionMove(Widget gw, XEvent *event, String *params, Cardinal *num_params)
9317a84e134Smrg{
9327a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
9337a84e134Smrg    int x, y;
9347a84e134Smrg
9357a84e134Smrg    if (!pw->panner.tmp.doing)
9367a84e134Smrg      return;
9377a84e134Smrg
9387a84e134Smrg    if (!get_event_xy(pw, event, &x, &y)) {
9397a84e134Smrg	XBell(XtDisplay(gw), 0);	/* should do error message */
9407a84e134Smrg	return;
9417a84e134Smrg    }
9427a84e134Smrg
9437a84e134Smrg    if (pw->panner.rubber_band)
9447a84e134Smrg	UNDRAW_TMP(pw);
9457a84e134Smrg    pw->panner.tmp.x = x - pw->panner.tmp.dx;
9467a84e134Smrg    pw->panner.tmp.y = y - pw->panner.tmp.dy;
9477a84e134Smrg
9487a84e134Smrg    if (!pw->panner.rubber_band)
9497a84e134Smrg	ActionNotify(gw, event, params, num_params);
9507a84e134Smrg    else {
9517a84e134Smrg	if (!pw->panner.allow_off)
9527a84e134Smrg	    check_knob(pw, False);
9537a84e134Smrg	DRAW_TMP(pw);
9547a84e134Smrg    }
9557a84e134Smrg}
9567a84e134Smrg
9577a84e134Smrg
9587a84e134Smrgstatic void
9597a84e134SmrgActionPage(Widget gw, XEvent *event, String *params, Cardinal *num_params)
9607a84e134Smrg{
9617a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
9627a84e134Smrg    Cardinal zero = 0;
9637a84e134Smrg    Bool isin = pw->panner.tmp.doing;
9647a84e134Smrg    int x, y;
9657a84e134Smrg    Bool relx, rely;
9667a84e134Smrg    int pad = pw->panner.internal_border << 1;
9677a84e134Smrg
9687a84e134Smrg    if (*num_params != 2) {
9697a84e134Smrg      XBell(XtDisplay(gw), 0);
9707a84e134Smrg	return;
9717a84e134Smrg    }
9727a84e134Smrg
9737a84e134Smrg    x = parse_page_string(params[0], pw->panner.knob_width,
9747a84e134Smrg			  (int)XtWidth(pw) - pad, &relx);
9757a84e134Smrg    y = parse_page_string(params[1], pw->panner.knob_height,
9767a84e134Smrg			  (int)XtHeight(pw) - pad, &rely);
9777a84e134Smrg
9787a84e134Smrg    if (relx)
9797a84e134Smrg	x += pw->panner.knob_x;
9807a84e134Smrg    if (rely)
9817a84e134Smrg	y += pw->panner.knob_y;
9827a84e134Smrg
9837a84e134Smrg    if (isin) {				/* if in, then use move */
9847a84e134Smrg	XEvent ev;
9857a84e134Smrg
9867a84e134Smrg	ev.xbutton.type = ButtonPress;
9877a84e134Smrg	ev.xbutton.x = x;
9887a84e134Smrg	ev.xbutton.y = y;
9897a84e134Smrg	ActionMove(gw, &ev, NULL, &zero);
9907a84e134Smrg    }
9917a84e134Smrg    else {
9927a84e134Smrg	pw->panner.tmp.doing = True;
9937a84e134Smrg	pw->panner.tmp.x = x;
9947a84e134Smrg	pw->panner.tmp.y = y;
9957a84e134Smrg	ActionNotify(gw, event, NULL, &zero);
9967a84e134Smrg	pw->panner.tmp.doing = False;
9977a84e134Smrg    }
9987a84e134Smrg}
9997a84e134Smrg
10007a84e134Smrg/*ARGSUSED*/
10017a84e134Smrgstatic void
10027a84e134SmrgActionNotify(Widget gw, XEvent *event, String *params, Cardinal *num_params)
10037a84e134Smrg{
10047a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
10057a84e134Smrg
10067a84e134Smrg    if (!pw->panner.tmp.doing)
10077a84e134Smrg	return;
10087a84e134Smrg
10097a84e134Smrg    if (!pw->panner.allow_off)
10107a84e134Smrg	check_knob(pw, False);
10117a84e134Smrg    pw->panner.knob_x = pw->panner.tmp.x;
10127a84e134Smrg    pw->panner.knob_y = pw->panner.tmp.y;
10137a84e134Smrg    move_shadow(pw);
10147a84e134Smrg
10157a84e134Smrg    pw->panner.slider_x = (Position)((double)pw->panner.knob_x
10167a84e134Smrg				   / pw->panner.haspect + 0.5);
10177a84e134Smrg    pw->panner.slider_y = (Position)((double) pw->panner.knob_y
10187a84e134Smrg				   / pw->panner.vaspect + 0.5);
10197a84e134Smrg    if (!pw->panner.allow_off) {
10207a84e134Smrg	Position tmp;
10217a84e134Smrg
10227a84e134Smrg	if (pw->panner.slider_x
10237a84e134Smrg	    > (tmp = (Position)pw->panner.canvas_width -
10247a84e134Smrg		     (Position)pw->panner.slider_width))
10257a84e134Smrg	    pw->panner.slider_x = tmp;
10267a84e134Smrg	if (pw->panner.slider_x < 0)
10277a84e134Smrg	    pw->panner.slider_x = 0;
10287a84e134Smrg	if (pw->panner.slider_y
10297a84e134Smrg	    > (tmp = (Position)pw->panner.canvas_height -
10307a84e134Smrg		     (Position)pw->panner.slider_height))
10317a84e134Smrg	    pw->panner.slider_y = tmp;
10327a84e134Smrg	if (pw->panner.slider_y < 0)
10337a84e134Smrg	    pw->panner.slider_y = 0;
10347a84e134Smrg    }
10357a84e134Smrg
10367a84e134Smrg    if (pw->panner.last_x != pw->panner.knob_x ||
10377a84e134Smrg	pw->panner.last_y != pw->panner.knob_y) {
10387a84e134Smrg	XawPannerReport rep;
10397a84e134Smrg
10407a84e134Smrg	XawPannerRedisplay(gw, NULL, NULL);
10417a84e134Smrg	rep.changed = XawPRSliderX | XawPRSliderY;
10427a84e134Smrg	rep.slider_x = pw->panner.slider_x;
10437a84e134Smrg	rep.slider_y = pw->panner.slider_y;
10447a84e134Smrg	rep.slider_width = pw->panner.slider_width;
10457a84e134Smrg	rep.slider_height = pw->panner.slider_height;
10467a84e134Smrg	rep.canvas_width = pw->panner.canvas_width;
10477a84e134Smrg	rep.canvas_height = pw->panner.canvas_height;
10487a84e134Smrg	XtCallCallbackList(gw, pw->panner.report_callbacks, (XtPointer)&rep);
10497a84e134Smrg    }
10507a84e134Smrg}
10517a84e134Smrg
10527a84e134Smrg/*ARGSUSED*/
10537a84e134Smrgstatic void
10547a84e134SmrgActionSet(Widget gw, XEvent *event, String *params, Cardinal *num_params)
10557a84e134Smrg{
10567a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
10577a84e134Smrg    Bool rb;
10587a84e134Smrg
10597a84e134Smrg    if (*num_params < 2 ||
10607a84e134Smrg	XmuCompareISOLatin1(params[0], "rubberband") != 0) {
10617a84e134Smrg	XBell(XtDisplay(gw), 0);
10627a84e134Smrg	return;
10637a84e134Smrg    }
10647a84e134Smrg
10657a84e134Smrg    if (XmuCompareISOLatin1(params[1], "on") == 0)
10667a84e134Smrg	rb = True;
10677a84e134Smrg    else if (XmuCompareISOLatin1(params[1], "off") == 0)
10687a84e134Smrg	rb = False;
10697a84e134Smrg    else if (XmuCompareISOLatin1(params[1], "toggle") == 0)
10707a84e134Smrg	rb = !pw->panner.rubber_band;
10717a84e134Smrg    else {
10727a84e134Smrg      XBell(XtDisplay(gw), 0);
10737a84e134Smrg	return;
10747a84e134Smrg    }
10757a84e134Smrg
10767a84e134Smrg    if (rb != pw->panner.rubber_band) {
10777a84e134Smrg	Arg args[1];
10787a84e134Smrg
10797a84e134Smrg	XtSetArg(args[0], XtNrubberBand, rb);
10807a84e134Smrg	XtSetValues(gw, args, 1);
10817a84e134Smrg    }
10827a84e134Smrg}
1083