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