Panner.c revision 5ec34c4c
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
437a84e134Smrg#if defined(ISC) && __STDC__ && !defined(ISC30)
447a84e134Smrgextern double atof(char *);
457a84e134Smrg#else
467a84e134Smrg#include <stdlib.h>			/* for atof() */
477a84e134Smrg#endif
487a84e134Smrg
497a84e134Smrg/*
507a84e134Smrg * Class Methods
517a84e134Smrg */
527a84e134Smrgstatic void XawPannerDestroy(Widget);
537a84e134Smrgstatic void XawPannerInitialize(Widget, Widget, ArgList, Cardinal*);
547a84e134Smrgstatic XtGeometryResult XawPannerQueryGeometry(Widget, XtWidgetGeometry*,
557a84e134Smrg					       XtWidgetGeometry*);
567a84e134Smrgstatic void XawPannerRealize(Widget, XtValueMask*, XSetWindowAttributes*);
577a84e134Smrgstatic void XawPannerRedisplay(Widget, XEvent*, Region);
587a84e134Smrgstatic void XawPannerResize(Widget);
597a84e134Smrgstatic Boolean XawPannerSetValues(Widget, Widget, Widget, ArgList, Cardinal*);
607a84e134Smrgstatic void XawPannerSetValuesAlmost(Widget, Widget, XtWidgetGeometry*,
617a84e134Smrg				     XtWidgetGeometry*);
627a84e134Smrg
637a84e134Smrg/*
647a84e134Smrg * Prototypes
657a84e134Smrg */
667a84e134Smrgstatic void check_knob(PannerWidget, Bool);
677a84e134Smrgstatic void get_default_size(PannerWidget, Dimension*, Dimension*);
687a84e134Smrgstatic Bool get_event_xy(PannerWidget, XEvent*, int*, int*);
697a84e134Smrgstatic void move_shadow(PannerWidget);
705ec34c4cSmrgstatic int parse_page_string(String, int, int, Bool*);
717a84e134Smrgstatic void rescale(PannerWidget);
727a84e134Smrgstatic void reset_shadow_gc(PannerWidget);
737a84e134Smrgstatic void reset_slider_gc(PannerWidget);
747a84e134Smrgstatic void reset_xor_gc(PannerWidget);
757a84e134Smrgstatic void scale_knob(PannerWidget, Bool, Bool);
767a84e134Smrg
777a84e134Smrg/*
787a84e134Smrg * Actions
797a84e134Smrg */
807a84e134Smrgstatic void ActionAbort(Widget, XEvent*, String*, Cardinal*);
817a84e134Smrgstatic void ActionMove(Widget, XEvent*, String*, Cardinal*);
827a84e134Smrgstatic void ActionNotify(Widget, XEvent*, String*, Cardinal*);
837a84e134Smrgstatic void ActionPage(Widget, XEvent*, String*, Cardinal*);
847a84e134Smrgstatic void ActionSet(Widget, XEvent*, String*, Cardinal*);
857a84e134Smrgstatic void ActionStart(Widget, XEvent*, String*, Cardinal*);
867a84e134Smrgstatic void ActionStop(Widget, XEvent*, String*, Cardinal*);
877a84e134Smrg
887a84e134Smrg/*
897a84e134Smrg * From Xmu/Distinct.c
907a84e134Smrg */
917a84e134SmrgBool XmuDistinguishablePixels(Display*, Colormap, unsigned long*, int);
927a84e134Smrg
937a84e134Smrg/*
947a84e134Smrg * Initialization
957a84e134Smrg */
967a84e134Smrgstatic char defaultTranslations[] =
977a84e134Smrg"<Btn1Down>:"		"start()\n"
987a84e134Smrg"<Btn1Motion>:"		"move()\n"
997a84e134Smrg"<Btn1Up>:"		"notify() stop()\n"
1007a84e134Smrg"<Btn2Down>:"		"abort()\n"
1017a84e134Smrg":<Key>KP_Enter:"	"set(rubberband,toggle)\n"
1027a84e134Smrg"<Key>space:"		"page(+1p,+1p)\n"
1037a84e134Smrg"<Key>Delete:"		"page(-1p,-1p)\n"
1047a84e134Smrg":<Key>KP_Delete:"	"page(-1p,-1p)\n"
1057a84e134Smrg"<Key>BackSpace:"	"page(-1p,-1p)\n"
1067a84e134Smrg"<Key>Left:"		"page(-.5p,+0)\n"
1077a84e134Smrg":<Key>KP_Left:"	"page(-.5p,+0)\n"
1087a84e134Smrg"<Key>Right:"		"page(+.5p,+0)\n"
1097a84e134Smrg":<Key>KP_Right:"	"page(+.5p,+0)\n"
1107a84e134Smrg"<Key>Up:"		"page(+0,-.5p)\n"
1117a84e134Smrg":<Key>KP_Up:"		"page(+0,-.5p)\n"
1127a84e134Smrg"<Key>Down:"		"page(+0,+.5p)\n"
1137a84e134Smrg":<Key>KP_Down:"	"page(+0,+.5p)\n"
1147a84e134Smrg"<Key>Home:"		"page(0,0)\n"
1157a84e134Smrg":<Key>KP_Home:"	"page(0,0)\n"
1167a84e134Smrg;
1177a84e134Smrg
1187a84e134Smrgstatic XtActionsRec actions[] = {
1197a84e134Smrg  {"start",	ActionStart},		/* start tmp graphics */
1207a84e134Smrg  {"stop",	ActionStop},		/* stop tmp graphics */
1217a84e134Smrg  {"abort",	ActionAbort},		/* punt */
1227a84e134Smrg  {"move",	ActionMove},		/* move tmp graphics on Motion event */
1237a84e134Smrg  {"page",	ActionPage},		/* page around usually from keyboard */
1247a84e134Smrg  {"notify",	ActionNotify},		/* callback new position */
1257a84e134Smrg  {"set",	ActionSet},		/* set various parameters */
1267a84e134Smrg};
1277a84e134Smrg
1287a84e134Smrg#define offset(field)	XtOffsetOf(PannerRec, panner.field)
1297a84e134Smrgstatic XtResource resources[] = {
1307a84e134Smrg    {
1317a84e134Smrg      XtNallowOff,
1327a84e134Smrg      XtCAllowOff,
1337a84e134Smrg      XtRBoolean,
1347a84e134Smrg      sizeof(Boolean),
1357a84e134Smrg      offset(allow_off),
1367a84e134Smrg      XtRImmediate,
1377a84e134Smrg      (XtPointer)False
1387a84e134Smrg    },
1397a84e134Smrg    {
1407a84e134Smrg      XtNresize,
1417a84e134Smrg      XtCResize,
1427a84e134Smrg      XtRBoolean,
1437a84e134Smrg      sizeof(Boolean),
1447a84e134Smrg      offset(resize_to_pref),
1457a84e134Smrg      XtRImmediate,
1467a84e134Smrg      (XtPointer)True
1477a84e134Smrg    },
1487a84e134Smrg    {
1497a84e134Smrg      XtNreportCallback,
1507a84e134Smrg      XtCReportCallback,
1517a84e134Smrg      XtRCallback,
1527a84e134Smrg      sizeof(XtPointer),
1537a84e134Smrg      offset(report_callbacks),
1547a84e134Smrg      XtRCallback,
1557a84e134Smrg      NULL
1567a84e134Smrg    },
1577a84e134Smrg    {
1587a84e134Smrg      XtNdefaultScale,
1597a84e134Smrg      XtCDefaultScale,
1607a84e134Smrg      XtRDimension,
1617a84e134Smrg      sizeof(Dimension),
1627a84e134Smrg      offset(default_scale),
1637a84e134Smrg      XtRImmediate,
1647a84e134Smrg      (XtPointer)PANNER_DEFAULT_SCALE
1657a84e134Smrg    },
1667a84e134Smrg    {
1677a84e134Smrg      XtNrubberBand,
1687a84e134Smrg      XtCRubberBand,
1697a84e134Smrg      XtRBoolean,
1707a84e134Smrg      sizeof(Boolean),
1717a84e134Smrg      offset(rubber_band),
1727a84e134Smrg      XtRImmediate,
1737a84e134Smrg      (XtPointer)False
1747a84e134Smrg    },
1757a84e134Smrg    {
1767a84e134Smrg      XtNforeground,
1777a84e134Smrg      XtCForeground,
1787a84e134Smrg      XtRPixel,
1797a84e134Smrg      sizeof(Pixel),
1807a84e134Smrg      offset(foreground),
1817a84e134Smrg      XtRString,
1827a84e134Smrg      (XtPointer)XtDefaultBackground
1837a84e134Smrg    },
1847a84e134Smrg    {
1857a84e134Smrg      XtNinternalSpace,
1867a84e134Smrg      XtCInternalSpace,
1877a84e134Smrg      XtRDimension,
1887a84e134Smrg      sizeof(Dimension),
1897a84e134Smrg      offset(internal_border),
1907a84e134Smrg      XtRImmediate,
1917a84e134Smrg      (XtPointer)4
1927a84e134Smrg    },
1937a84e134Smrg    {
1947a84e134Smrg      XtNlineWidth,
1957a84e134Smrg      XtCLineWidth,
1967a84e134Smrg      XtRDimension,
1977a84e134Smrg      sizeof(Dimension),
1987a84e134Smrg      offset(line_width),
1997a84e134Smrg      XtRImmediate,
2007a84e134Smrg      (XtPointer)0
2017a84e134Smrg    },
2027a84e134Smrg    {
2037a84e134Smrg      XtNcanvasWidth,
2047a84e134Smrg      XtCCanvasWidth,
2057a84e134Smrg      XtRDimension,
2067a84e134Smrg      sizeof(Dimension),
2077a84e134Smrg      offset(canvas_width),
2087a84e134Smrg      XtRImmediate,
2097a84e134Smrg      (XtPointer)0
2107a84e134Smrg    },
2117a84e134Smrg    {
2127a84e134Smrg      XtNcanvasHeight,
2137a84e134Smrg      XtCCanvasHeight,
2147a84e134Smrg      XtRDimension,
2157a84e134Smrg      sizeof(Dimension),
2167a84e134Smrg      offset(canvas_height),
2177a84e134Smrg      XtRImmediate,
2187a84e134Smrg      (XtPointer)0
2197a84e134Smrg    },
2207a84e134Smrg    {
2217a84e134Smrg      XtNsliderX,
2227a84e134Smrg      XtCSliderX,
2237a84e134Smrg      XtRPosition,
2247a84e134Smrg      sizeof(Position),
2257a84e134Smrg      offset(slider_x),
2267a84e134Smrg      XtRImmediate,
2277a84e134Smrg      (XtPointer)0
2287a84e134Smrg    },
2297a84e134Smrg    {
2307a84e134Smrg      XtNsliderY,
2317a84e134Smrg      XtCSliderY,
2327a84e134Smrg      XtRPosition,
2337a84e134Smrg      sizeof(Position),
2347a84e134Smrg      offset(slider_y),
2357a84e134Smrg      XtRImmediate,
2367a84e134Smrg      (XtPointer)0
2377a84e134Smrg    },
2387a84e134Smrg    {
2397a84e134Smrg      XtNsliderWidth,
2407a84e134Smrg      XtCSliderWidth,
2417a84e134Smrg      XtRDimension,
2427a84e134Smrg      sizeof(Dimension),
2437a84e134Smrg      offset(slider_width),
2447a84e134Smrg      XtRImmediate,
2457a84e134Smrg      (XtPointer)0
2467a84e134Smrg    },
2477a84e134Smrg    {
2487a84e134Smrg      XtNsliderHeight,
2497a84e134Smrg      XtCSliderHeight,
2507a84e134Smrg      XtRDimension,
2517a84e134Smrg      sizeof(Dimension),
2527a84e134Smrg      offset(slider_height),
2537a84e134Smrg      XtRImmediate,
2547a84e134Smrg      (XtPointer)0
2557a84e134Smrg    },
2567a84e134Smrg    {
2577a84e134Smrg      XtNshadowColor,
2587a84e134Smrg      XtCShadowColor,
2597a84e134Smrg      XtRPixel,
2607a84e134Smrg      sizeof(Pixel),
2617a84e134Smrg      offset(shadow_color),
2627a84e134Smrg      XtRString,
2637a84e134Smrg      (XtPointer)XtDefaultForeground
2647a84e134Smrg    },
2657a84e134Smrg    {
2667a84e134Smrg      XtNshadowThickness,
2677a84e134Smrg      XtCShadowThickness,
2687a84e134Smrg      XtRDimension,
2697a84e134Smrg      sizeof(Dimension),
2707a84e134Smrg      offset(shadow_thickness),
2717a84e134Smrg      XtRImmediate,
2727a84e134Smrg      (XtPointer)2
2737a84e134Smrg    },
2747a84e134Smrg    {
2757a84e134Smrg      XtNbackgroundStipple,
2767a84e134Smrg      XtCBackgroundStipple,
2777a84e134Smrg      XtRString,
2787a84e134Smrg      sizeof(String),
2797a84e134Smrg      offset(stipple_name),
2807a84e134Smrg      XtRImmediate,
2817a84e134Smrg      NULL
2827a84e134Smrg    },
2837a84e134Smrg};
2847a84e134Smrg#undef offset
2857a84e134Smrg
2867a84e134Smrg#define Superclass	(&simpleClassRec)
2877a84e134SmrgPannerClassRec pannerClassRec = {
2887a84e134Smrg  /* core */
2897a84e134Smrg  {
2907a84e134Smrg    (WidgetClass)Superclass,		/* superclass */
2917a84e134Smrg    "Panner",				/* class_name */
2927a84e134Smrg    sizeof(PannerRec),			/* widget_size */
2937a84e134Smrg    XawInitializeWidgetSet,		/* class_initialize */
2947a84e134Smrg    NULL,				/* class_part_initialize */
2957a84e134Smrg    False,				/* class_inited */
2967a84e134Smrg    XawPannerInitialize,		/* initialize */
2977a84e134Smrg    NULL,				/* initialize_hook */
2987a84e134Smrg    XawPannerRealize,			/* realize */
2997a84e134Smrg    actions,				/* actions */
3007a84e134Smrg    XtNumber(actions),			/* num_actions */
3017a84e134Smrg    resources,				/* resources */
3027a84e134Smrg    XtNumber(resources),		/* num_resources */
3037a84e134Smrg    NULLQUARK,				/* xrm_class */
3047a84e134Smrg    True,				/* compress_motion */
3057a84e134Smrg    True,				/* compress_exposure */
3067a84e134Smrg    True,				/* compress_enterleave */
3077a84e134Smrg    False,				/* visible_interest */
3087a84e134Smrg    XawPannerDestroy,			/* destroy */
3097a84e134Smrg    XawPannerResize,			/* resize */
3107a84e134Smrg    XawPannerRedisplay,			/* expose */
3117a84e134Smrg    XawPannerSetValues,			/* set_values */
3127a84e134Smrg    NULL,				/* set_values_hook */
3137a84e134Smrg    XawPannerSetValuesAlmost,		/* set_values_almost */
3147a84e134Smrg    NULL,				/* get_values_hook */
3157a84e134Smrg    NULL,				/* accept_focus */
3167a84e134Smrg    XtVersion,				/* version */
3177a84e134Smrg    NULL,				/* callback_private */
3187a84e134Smrg    defaultTranslations,		/* tm_table */
3197a84e134Smrg    XawPannerQueryGeometry,		/* query_geometry */
3207a84e134Smrg    XtInheritDisplayAccelerator,	/* display_accelerator */
3217a84e134Smrg    NULL,				/* extension */
3227a84e134Smrg  },
3237a84e134Smrg  /* simple */
3247a84e134Smrg  {
3257a84e134Smrg    XtInheritChangeSensitive,		/* change_sensitive */
3265ec34c4cSmrg#ifndef OLDXAW
3275ec34c4cSmrg    NULL,
3285ec34c4cSmrg#endif
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{
4265ec34c4cSmrg    Position pad = (Position)(pw->panner.internal_border << 1);
4275ec34c4cSmrg    Position maxx = (Position)(XtWidth(pw) - pad - pw->panner.knob_width);
4285ec34c4cSmrg    Position maxy = (Position)(XtHeight(pw) - pad - pw->panner.knob_height);
4297a84e134Smrg    Position *x = knob ? &pw->panner.knob_x : &pw->panner.tmp.x;
4307a84e134Smrg    Position *y = knob ? &pw->panner.knob_y : &pw->panner.tmp.y;
4317a84e134Smrg
4327a84e134Smrg    /*
4337a84e134Smrg     * note that positions are already normalized (i.e. internal_border
4347a84e134Smrg     * has been subtracted out)
4357a84e134Smrg     */
4367a84e134Smrg    if (*x < 0)
4377a84e134Smrg	*x = 0;
4387a84e134Smrg    if (*x > maxx)
4397a84e134Smrg	*x = maxx;
4407a84e134Smrg
4417a84e134Smrg    if (*y < 0)
4427a84e134Smrg	*y = 0;
4437a84e134Smrg    if (*y > maxy)
4447a84e134Smrg	*y = maxy;
4457a84e134Smrg
4467a84e134Smrg    if (knob) {
4477a84e134Smrg	pw->panner.slider_x = (Position)((double)pw->panner.knob_x
4487a84e134Smrg					/ pw->panner.haspect + 0.5);
4497a84e134Smrg	pw->panner.slider_y = (Position)((double)pw->panner.knob_y
4507a84e134Smrg					/ pw->panner.vaspect + 0.5);
4517a84e134Smrg	pw->panner.last_x = pw->panner.last_y = PANNER_OUTOFRANGE;
4527a84e134Smrg    }
4537a84e134Smrg}
4547a84e134Smrg
4557a84e134Smrgstatic void
4567a84e134Smrgmove_shadow(PannerWidget pw)
4577a84e134Smrg{
4587a84e134Smrg    if (pw->panner.shadow_thickness > 0) {
4597a84e134Smrg	int lw = pw->panner.shadow_thickness + (pw->panner.line_width << 1);
4607a84e134Smrg	int pad = pw->panner.internal_border;
4617a84e134Smrg
4627a84e134Smrg	if (pw->panner.knob_height > lw && pw->panner.knob_width > lw) {
4637a84e134Smrg	    XRectangle *r = pw->panner.shadow_rects;
4647a84e134Smrg
4655ec34c4cSmrg	    r->x = (short)(pw->panner.knob_x + pad + pw->panner.knob_width);
4665ec34c4cSmrg	    r->y = (short)(pw->panner.knob_y + pad + lw);
4675ec34c4cSmrg	    r->width = (unsigned short)(pw->panner.shadow_thickness);
4685ec34c4cSmrg	    r->height = (unsigned short)(pw->panner.knob_height - lw);
4697a84e134Smrg	    r++;
4705ec34c4cSmrg	    r->x = (short)(pw->panner.knob_x + pad + lw);
4715ec34c4cSmrg	    r->y = (short)(pw->panner.knob_y + pad + pw->panner.knob_height);
4725ec34c4cSmrg	    r->width = (unsigned short)(pw->panner.knob_width - lw + pw->panner.shadow_thickness);
4735ec34c4cSmrg	    r->height = (unsigned short)(pw->panner.shadow_thickness);
4747a84e134Smrg	    pw->panner.shadow_valid = True;
4757a84e134Smrg	    return;
4767a84e134Smrg	}
4777a84e134Smrg    }
4787a84e134Smrg    pw->panner.shadow_valid = False;
4797a84e134Smrg}
4807a84e134Smrg
4817a84e134Smrgstatic void
4827a84e134Smrgscale_knob(PannerWidget pw, Bool location, Bool size)
4837a84e134Smrg{
4847a84e134Smrg    if (location) {
4857a84e134Smrg	pw->panner.knob_x = (Position)PANNER_HSCALE(pw, pw->panner.slider_x);
4867a84e134Smrg	pw->panner.knob_y = (Position)PANNER_VSCALE(pw, pw->panner.slider_y);
4877a84e134Smrg    }
4887a84e134Smrg    if (size) {
4897a84e134Smrg	Dimension width, height;
4907a84e134Smrg
4917a84e134Smrg	if (pw->panner.slider_width < 1)
4927a84e134Smrg	    pw->panner.slider_width = pw->panner.canvas_width;
4937a84e134Smrg	if (pw->panner.slider_height < 1)
4947a84e134Smrg	    pw->panner.slider_height = pw->panner.canvas_height;
4957a84e134Smrg	width = Min(pw->panner.slider_width, pw->panner.canvas_width);
4967a84e134Smrg	height = Min(pw->panner.slider_height, pw->panner.canvas_height);
4977a84e134Smrg
4987a84e134Smrg	pw->panner.knob_width = (Dimension)PANNER_HSCALE(pw, width);
4997a84e134Smrg	pw->panner.knob_height = (Dimension)PANNER_VSCALE(pw, height);
5007a84e134Smrg    }
5017a84e134Smrg    if (!pw->panner.allow_off)
5027a84e134Smrg	check_knob(pw, True);
5037a84e134Smrg    move_shadow(pw);
5047a84e134Smrg}
5057a84e134Smrg
5067a84e134Smrgstatic void
5077a84e134Smrgrescale(PannerWidget pw)
5087a84e134Smrg{
5097a84e134Smrg    int hpad = pw->panner.internal_border << 1;
5107a84e134Smrg    int vpad = hpad;
5117a84e134Smrg
5127a84e134Smrg    if (pw->panner.canvas_width < 1)
5137a84e134Smrg	pw->panner.canvas_width = XtWidth(pw);
5147a84e134Smrg    if (pw->panner.canvas_height < 1)
5157a84e134Smrg	pw->panner.canvas_height = XtHeight(pw);
5167a84e134Smrg
5177a84e134Smrg    if (XtWidth(pw) <= hpad)
5187a84e134Smrg	hpad = 0;
5197a84e134Smrg    if (XtHeight(pw) <= vpad)
5207a84e134Smrg	vpad = 0;
5217a84e134Smrg
5227a84e134Smrg    pw->panner.haspect = ((double)XtWidth(pw) - hpad + .5)
5237a84e134Smrg			 / (double)pw->panner.canvas_width;
5247a84e134Smrg    pw->panner.vaspect = ((double)XtHeight(pw) - vpad + .5)
5257a84e134Smrg			 / (double)pw->panner.canvas_height;
5267a84e134Smrg    scale_knob(pw, True, True);
5277a84e134Smrg}
5287a84e134Smrg
5297a84e134Smrgstatic void
5307a84e134Smrgget_default_size(PannerWidget pw, Dimension *wp, Dimension *hp)
5317a84e134Smrg{
5325ec34c4cSmrg    Dimension pad = (Dimension)(pw->panner.internal_border << 1);
5337a84e134Smrg
5345ec34c4cSmrg    *wp = (Dimension)(PANNER_DSCALE(pw, pw->panner.canvas_width) + pad);
5355ec34c4cSmrg    *hp = (Dimension)(PANNER_DSCALE(pw, pw->panner.canvas_height) + pad);
5367a84e134Smrg}
5377a84e134Smrg
5387a84e134Smrgstatic Bool
5397a84e134Smrgget_event_xy(PannerWidget pw, XEvent *event, int *x, int *y)
5407a84e134Smrg{
5417a84e134Smrg    int pad = pw->panner.internal_border;
5427a84e134Smrg
5437a84e134Smrg    switch (event->type) {
5447a84e134Smrg	case ButtonPress:
5457a84e134Smrg	case ButtonRelease:
5467a84e134Smrg	    *x = event->xbutton.x - pad;
5477a84e134Smrg	    *y = event->xbutton.y - pad;
5487a84e134Smrg	    return (True);
5497a84e134Smrg	case KeyPress:
5507a84e134Smrg	case KeyRelease:
5517a84e134Smrg	    *x = event->xkey.x - pad;
5527a84e134Smrg	    *y = event->xkey.y - pad;
5537a84e134Smrg	    return (True);
5547a84e134Smrg	case EnterNotify:
5557a84e134Smrg	case LeaveNotify:
5567a84e134Smrg	    *x = event->xcrossing.x - pad;
5577a84e134Smrg	    *y = event->xcrossing.y - pad;
5587a84e134Smrg	    return (True);
5597a84e134Smrg	case MotionNotify:
5607a84e134Smrg	    *x = event->xmotion.x - pad;
5617a84e134Smrg	    *y = event->xmotion.y - pad;
5627a84e134Smrg	    return (True);
5637a84e134Smrg    }
5647a84e134Smrg
5657a84e134Smrg    return (False);
5667a84e134Smrg}
5677a84e134Smrg
5687a84e134Smrgstatic int
5695ec34c4cSmrgparse_page_string(String s, int pagesize, int canvassize, Bool *relative)
5707a84e134Smrg{
5715ec34c4cSmrg    String cp;
5727a84e134Smrg    double val = 1.0;
5737a84e134Smrg    Bool rel = False;
5747a84e134Smrg
5757a84e134Smrg    /*
5767a84e134Smrg     * syntax:    spaces [+-] number spaces [pc\0] spaces
5777a84e134Smrg     */
5787a84e134Smrg    for (; isascii(*s) && isspace(*s); s++)	/* skip white space */
5797a84e134Smrg	;
5807a84e134Smrg
5817a84e134Smrg    if (*s == '+' || *s == '-')	{		/* deal with signs */
5827a84e134Smrg	rel = True;
5837a84e134Smrg	if (*s == '-')
5847a84e134Smrg	    val = -1.0;
5857a84e134Smrg	s++;
5867a84e134Smrg    }
5877a84e134Smrg    if (!*s) {				/* if null then return nothing */
5887a84e134Smrg	*relative = True;
5897a84e134Smrg	return (0);
5907a84e134Smrg    }
5917a84e134Smrg
5927a84e134Smrg					/* skip over numbers */
5937a84e134Smrg    for (cp = s; isascii(*s) && (isdigit(*s) || *s == '.'); s++)
5947a84e134Smrg	;
5957a84e134Smrg    val *= atof(cp);
5967a84e134Smrg
5977a84e134Smrg					/* skip blanks */
5987a84e134Smrg    for (; isascii(*s) && isspace(*s); s++)
5997a84e134Smrg	;
6007a84e134Smrg
6017a84e134Smrg    if (*s) {				/* if units */
6027a84e134Smrg	switch (s[0]) {
6037a84e134Smrg	    case 'p':
6047a84e134Smrg	    case 'P':
6057a84e134Smrg		val *= (double)pagesize;
6067a84e134Smrg		break;
6077a84e134Smrg	    case 'c':
6087a84e134Smrg	    case 'C':
6097a84e134Smrg		val *= (double)canvassize;
6107a84e134Smrg		break;
6117a84e134Smrg	}
6127a84e134Smrg    }
6137a84e134Smrg    *relative = rel;
6147a84e134Smrg
6157a84e134Smrg    return ((int)val);
6167a84e134Smrg}
6177a84e134Smrg
6187a84e134Smrg#define DRAW_TMP(pw) \
6197a84e134Smrg{ \
6207a84e134Smrg    XDrawRectangle(XtDisplay(pw), XtWindow(pw),				\
6217a84e134Smrg		   pw->panner.xor_gc,					\
6225ec34c4cSmrg		   (pw->panner.tmp.x + pw->panner.internal_border),	\
6235ec34c4cSmrg		   (pw->panner.tmp.y + pw->panner.internal_border),	\
6245ec34c4cSmrg		   (unsigned)(pw->panner.knob_width - 1),		\
6255ec34c4cSmrg		   (unsigned)(pw->panner.knob_height - 1));		\
6267a84e134Smrg    pw->panner.tmp.showing = !pw->panner.tmp.showing;			\
6277a84e134Smrg}
6287a84e134Smrg
6297a84e134Smrg#define UNDRAW_TMP(pw) \
6307a84e134Smrg{ \
6317a84e134Smrg    if (pw->panner.tmp.showing)			\
6327a84e134Smrg      DRAW_TMP(pw);				\
6337a84e134Smrg}
6347a84e134Smrg
6357a84e134Smrg#define BACKGROUND_STIPPLE(pw) \
6367a84e134SmrgXmuLocatePixmapFile(pw->core.screen, pw->panner.stipple_name,		\
6377a84e134Smrg		    pw->panner.shadow_color, pw->core.background_pixel,	\
6387a84e134Smrg		    pw->core.depth, NULL, 0, NULL, NULL, NULL, NULL)
639421c997bSmrg
6407a84e134Smrg#define PIXMAP_OKAY(pm) ((pm) != None && (pm) != XtUnspecifiedPixmap)
6417a84e134Smrg
6427a84e134Smrg/*ARGSUSED*/
6437a84e134Smrgstatic void
6445ec34c4cSmrgXawPannerInitialize(Widget greq, Widget gnew, ArgList args _X_UNUSED, Cardinal *num_args _X_UNUSED)
6457a84e134Smrg{
6467a84e134Smrg    PannerWidget req = (PannerWidget)greq, cnew = (PannerWidget)gnew;
6477a84e134Smrg    Dimension defwidth, defheight;
6487a84e134Smrg
6497a84e134Smrg    if (req->panner.canvas_width < 1)
6507a84e134Smrg	cnew->panner.canvas_width = 1;
6517a84e134Smrg    if (req->panner.canvas_height < 1)
6527a84e134Smrg	cnew->panner.canvas_height = 1;
6537a84e134Smrg    if (req->panner.default_scale < 1)
6547a84e134Smrg	cnew->panner.default_scale = PANNER_DEFAULT_SCALE;
6557a84e134Smrg
6567a84e134Smrg    get_default_size(req, &defwidth, &defheight);
6577a84e134Smrg    if (XtWidth(req) < 1)
6587a84e134Smrg	XtWidth(cnew) = defwidth;
6597a84e134Smrg    if (XtHeight(req) < 1)
6607a84e134Smrg	XtHeight(cnew) = defheight;
6617a84e134Smrg
6627a84e134Smrg    cnew->panner.shadow_gc = NULL;
6637a84e134Smrg    reset_shadow_gc(cnew);		/* shadowColor */
6647a84e134Smrg    cnew->panner.slider_gc = NULL;
6657a84e134Smrg    reset_slider_gc(cnew);		/* foreground */
6667a84e134Smrg    cnew->panner.xor_gc = NULL;
667421c997bSmrg    reset_xor_gc(cnew);			/* foreground ^ background */
6687a84e134Smrg
6697a84e134Smrg    rescale(cnew);			/* does a position check */
6707a84e134Smrg    cnew->panner.shadow_valid = False;
6717a84e134Smrg    cnew->panner.tmp.doing = False;
6727a84e134Smrg    cnew->panner.tmp.showing = False;
6737a84e134Smrg  }
6747a84e134Smrg
6757a84e134Smrgstatic void
6767a84e134SmrgXawPannerRealize(Widget gw, XtValueMask *valuemaskp,
6777a84e134Smrg		 XSetWindowAttributes *attr)
6787a84e134Smrg{
6797a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
6807a84e134Smrg    Pixmap pm = XtUnspecifiedPixmap;
6817a84e134Smrg    Bool gotpm = False;
6827a84e134Smrg
6837a84e134Smrg    if (pw->core.background_pixmap == XtUnspecifiedPixmap) {
6847a84e134Smrg	if (pw->panner.stipple_name)
6857a84e134Smrg	    pm = BACKGROUND_STIPPLE(pw);
6867a84e134Smrg
6877a84e134Smrg	if (PIXMAP_OKAY(pm)) {
6887a84e134Smrg	    attr->background_pixmap = pm;
6897a84e134Smrg	    *valuemaskp |= CWBackPixmap;
6905ec34c4cSmrg	    *valuemaskp &= (XtValueMask)(~CWBackPixel);
6917a84e134Smrg	    gotpm = True;
6927a84e134Smrg	}
6937a84e134Smrg    }
6947a84e134Smrg    (*pannerWidgetClass->core_class.superclass->core_class.realize)
6957a84e134Smrg	(gw, valuemaskp, attr);
6967a84e134Smrg
6977a84e134Smrg    if (gotpm)
6987a84e134Smrg	XFreePixmap(XtDisplay(gw), pm);
6997a84e134Smrg}
7007a84e134Smrg
7017a84e134Smrgstatic void
7027a84e134SmrgXawPannerDestroy(Widget gw)
7037a84e134Smrg{
7047a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
7057a84e134Smrg
7067a84e134Smrg    XtReleaseGC(gw, pw->panner.shadow_gc);
7077a84e134Smrg    XtReleaseGC(gw, pw->panner.slider_gc);
7087a84e134Smrg    XtReleaseGC(gw, pw->panner.xor_gc);
7097a84e134Smrg}
7107a84e134Smrg
7117a84e134Smrgstatic void
7127a84e134SmrgXawPannerResize(Widget gw)
7137a84e134Smrg{
7147a84e134Smrg    rescale((PannerWidget)gw);
7157a84e134Smrg}
7167a84e134Smrg
7177a84e134Smrgstatic void
7187a84e134SmrgXawPannerRedisplay(Widget gw, XEvent *event, Region region)
7197a84e134Smrg{
7207a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
7217a84e134Smrg    Display *dpy = XtDisplay(gw);
7227a84e134Smrg    Window w = XtWindow(gw);
7237a84e134Smrg    int pad = pw->panner.internal_border;
7247a84e134Smrg    Dimension lw = pw->panner.line_width;
7255ec34c4cSmrg    Dimension extra = (Dimension)(pw->panner.shadow_thickness + (lw << 1));
7267a84e134Smrg    int kx = pw->panner.knob_x + pad, ky = pw->panner.knob_y + pad;
7277a84e134Smrg
7287a84e134Smrg    if (Superclass->core_class.expose)
7297a84e134Smrg	(Superclass->core_class.expose)(gw, event, region);
7307a84e134Smrg
7317a84e134Smrg    pw->panner.tmp.showing = False;
7327a84e134Smrg    XClearArea(XtDisplay(pw), XtWindow(pw),
7337a84e134Smrg	       (int)pw->panner.last_x - ((int)lw) + pad,
7347a84e134Smrg	       (int)pw->panner.last_y - ((int)lw) + pad,
7355ec34c4cSmrg	       (unsigned)(pw->panner.knob_width + extra),
7365ec34c4cSmrg	       (unsigned)(pw->panner.knob_height + extra),
7377a84e134Smrg	       False);
7387a84e134Smrg    pw->panner.last_x = pw->panner.knob_x;
7397a84e134Smrg    pw->panner.last_y = pw->panner.knob_y;
7407a84e134Smrg
7417a84e134Smrg    XFillRectangle(dpy, w, pw->panner.slider_gc, kx, ky,
7425ec34c4cSmrg		   (unsigned)(pw->panner.knob_width - 1),
7435ec34c4cSmrg		   (unsigned)(pw->panner.knob_height - 1));
7447a84e134Smrg
7457a84e134Smrg    if (lw)
7467a84e134Smrg	XDrawRectangle(dpy, w, pw->panner.shadow_gc, kx, ky,
7475ec34c4cSmrg		       (unsigned)(pw->panner.knob_width - 1),
7485ec34c4cSmrg		       (unsigned)(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
7595ec34c4cSmrgXawPannerSetValues(Widget gcur, Widget greq _X_UNUSED, Widget gnew,
7605ec34c4cSmrg		   ArgList args _X_UNUSED, Cardinal *num_args _X_UNUSED)
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 ||
828421c997bSmrg		   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
8365ec34c4cSmrg    return (Boolean)(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
8715ec34c4cSmrgActionStart(Widget gw, XEvent *event, String *params _X_UNUSED, Cardinal *num_params _X_UNUSED)
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;
8845ec34c4cSmrg    pw->panner.tmp.dx = (Position)(x - pw->panner.knob_x);
8855ec34c4cSmrg    pw->panner.tmp.dy = (Position)(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
8945ec34c4cSmrgActionStop(Widget gw, XEvent *event, String *params _X_UNUSED, Cardinal *num_params _X_UNUSED)
8957a84e134Smrg{
8967a84e134Smrg    PannerWidget pw = (PannerWidget)gw;
8977a84e134Smrg    int x, y;
8987a84e134Smrg
8997a84e134Smrg    if (get_event_xy(pw, event, &x, &y)) {
9005ec34c4cSmrg	pw->panner.tmp.x = (Position)(x - pw->panner.tmp.dx);
9015ec34c4cSmrg	pw->panner.tmp.y = (Position)(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);
9455ec34c4cSmrg    pw->panner.tmp.x = (Position)(x - pw->panner.tmp.dx);
9465ec34c4cSmrg    pw->panner.tmp.y = (Position)(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;
9935ec34c4cSmrg	pw->panner.tmp.x = (Position)x;
9945ec34c4cSmrg	pw->panner.tmp.y = (Position)y;
9957a84e134Smrg	ActionNotify(gw, event, NULL, &zero);
9967a84e134Smrg	pw->panner.tmp.doing = False;
9977a84e134Smrg    }
9987a84e134Smrg}
9997a84e134Smrg
10007a84e134Smrg/*ARGSUSED*/
10017a84e134Smrgstatic void
10025ec34c4cSmrgActionNotify(Widget gw, XEvent *event _X_UNUSED, String *params _X_UNUSED, Cardinal *num_params _X_UNUSED)
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
10235ec34c4cSmrg	    > (tmp = (Position)(pw->panner.canvas_width -
10245ec34c4cSmrg				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
10295ec34c4cSmrg	    > (tmp = (Position)(pw->panner.canvas_height -
10305ec34c4cSmrg				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
10545ec34c4cSmrgActionSet(Widget gw, XEvent *event _X_UNUSED, 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