Panner.c revision 5b16253f
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 */ 5785b16253fSmrg for (; isascii((unsigned char)*s) && isspace((unsigned char)*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 */ 5935b16253fSmrg for (cp = s; isascii((unsigned char)*s) && (isdigit((unsigned char)*s) || *s == '.'); s++) 5947a84e134Smrg ; 5957a84e134Smrg val *= atof(cp); 5967a84e134Smrg 5977a84e134Smrg /* skip blanks */ 5985b16253fSmrg for (; isascii((unsigned char)*s) && isspace((unsigned char)*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