home *** CD-ROM | disk | FTP | other *** search
/ OS/2 Shareware BBS: 10 Tools / 10-Tools.zip / tkisrc04.zip / tk / os2 / tkScale.c < prev    next >
C/C++ Source or Header  |  1998-08-07  |  65KB  |  2,055 lines

  1. /* 
  2.  * tkScale.c --
  3.  *
  4.  *    This module implements a scale widgets for the Tk toolkit.
  5.  *    A scale displays a slider that can be adjusted to change a
  6.  *    value;  it also displays numeric labels and a textual label,
  7.  *    if desired.
  8.  *    
  9.  *    The modifications to use floating-point values are based on
  10.  *    an implementation by Paul Mackerras.  The -variable option
  11.  *    is due to Henning Schulzrinne.  All of these are used with
  12.  *    permission.
  13.  *
  14.  * Copyright (c) 1990-1994 The Regents of the University of California.
  15.  * Copyright (c) 1994-1995 Sun Microsystems, Inc.
  16.  *
  17.  * See the file "license.terms" for information on usage and redistribution
  18.  * of this file, and for a DISCLAIMER OF ALL WARRANTIES.
  19.  *
  20.  * SCCS: @(#) tkScale.c 1.80 96/03/21 13:11:55
  21.  */
  22.  
  23. #include "tkPort.h"
  24. #include "default.h"
  25. #include "tkInt.h"
  26. #include <math.h>
  27.  
  28. /*
  29.  * A data structure of the following type is kept for each scale
  30.  * widget managed by this file:
  31.  */
  32.  
  33. typedef struct {
  34.     Tk_Window tkwin;        /* Window that embodies the scale.  NULL
  35.                  * means that the window has been destroyed
  36.                  * but the data structures haven't yet been
  37.                  * cleaned up.*/
  38.     Display *display;        /* Display containing widget.  Used, among
  39.                  * other things, so that resources can be
  40.                  * freed even after tkwin has gone away. */
  41.     Tcl_Interp *interp;        /* Interpreter associated with scale. */
  42.     Tcl_Command widgetCmd;    /* Token for scale's widget command. */
  43.     Tk_Uid orientUid;        /* Orientation for window ("vertical" or
  44.                  * "horizontal"). */
  45.     int vertical;        /* Non-zero means vertical orientation,
  46.                  * zero means horizontal. */
  47.     int width;            /* Desired narrow dimension of scale,
  48.                  * in pixels. */
  49.     int length;            /* Desired long dimension of scale,
  50.                  * in pixels. */
  51.     double value;        /* Current value of scale. */
  52.     char *varName;        /* Name of variable (malloc'ed) or NULL.
  53.                  * If non-NULL, scale's value tracks
  54.                  * the contents of this variable and
  55.                  * vice versa. */
  56.     double fromValue;        /* Value corresponding to left or top of
  57.                  * scale. */
  58.     double toValue;        /* Value corresponding to right or bottom
  59.                  * of scale. */
  60.     double tickInterval;    /* Distance between tick marks;  0 means
  61.                  * don't display any tick marks. */
  62.     double resolution;        /* If > 0, all values are rounded to an
  63.                  * even multiple of this value. */
  64.     int digits;            /* Number of significant digits to print
  65.                  * in values.  0 means we get to choose the
  66.                  * number based on resolution and/or the
  67.                  * range of the scale. */
  68.     char format[10];        /* Sprintf conversion specifier computed from
  69.                  * digits and other information. */
  70.     double bigIncrement;    /* Amount to use for large increments to
  71.                  * scale value.  (0 means we pick a value). */
  72.     char *command;        /* Command prefix to use when invoking Tcl
  73.                  * commands because the scale value changed.
  74.                  * NULL means don't invoke commands.
  75.                  * Malloc'ed. */
  76.     int repeatDelay;        /* How long to wait before auto-repeating
  77.                  * on scrolling actions (in ms). */
  78.     int repeatInterval;        /* Interval between autorepeats (in ms). */
  79.     char *label;        /* Label to display above or to right of
  80.                  * scale;  NULL means don't display a
  81.                  * label.  Malloc'ed. */
  82.     int labelLength;        /* Number of non-NULL chars. in label. */
  83.     Tk_Uid state;        /* Normal or disabled.  Value cannot be
  84.                  * changed when scale is disabled. */
  85.  
  86.     /*
  87.      * Information used when displaying widget:
  88.      */
  89.  
  90.     int borderWidth;        /* Width of 3-D border around window. */
  91.     Tk_3DBorder bgBorder;    /* Used for drawing slider and other
  92.                  * background areas. */
  93.     Tk_3DBorder activeBorder;    /* For drawing the slider when active. */
  94.     int sliderRelief;        /* Is slider to be drawn raised, sunken, etc. */
  95.     XColor *troughColorPtr;    /* Color for drawing trough. */
  96.     GC troughGC;        /* For drawing trough. */
  97.     GC copyGC;            /* Used for copying from pixmap onto screen. */
  98.     XFontStruct *fontPtr;    /* Information about text font, or NULL. */
  99.     XColor *textColorPtr;    /* Color for drawing text. */
  100.     GC textGC;            /* GC for drawing text in normal mode. */
  101.     int relief;            /* Indicates whether window as a whole is
  102.                  * raised, sunken, or flat. */
  103.     int highlightWidth;        /* Width in pixels of highlight to draw
  104.                  * around widget when it has the focus.
  105.                  * <= 0 means don't draw a highlight. */
  106.     XColor *highlightBgColorPtr;
  107.                 /* Color for drawing traversal highlight
  108.                  * area when highlight is off. */
  109.     XColor *highlightColorPtr;    /* Color for drawing traversal highlight. */
  110.     int inset;            /* Total width of all borders, including
  111.                  * traversal highlight and 3-D border.
  112.                  * Indicates how much interior stuff must
  113.                  * be offset from outside edges to leave
  114.                  * room for borders. */
  115.     int sliderLength;        /* Length of slider, measured in pixels along
  116.                  * long dimension of scale. */
  117.     int showValue;        /* Non-zero means to display the scale value
  118.                  * below or to the left of the slider;  zero
  119.                  * means don't display the value. */
  120.  
  121.     /*
  122.      * Layout information for horizontal scales, assuming that window
  123.      * gets the size it requested:
  124.      */
  125.  
  126.     int horizLabelY;        /* Y-coord at which to draw label. */
  127.     int horizValueY;        /* Y-coord at which to draw value text. */
  128.     int horizTroughY;        /* Y-coord of top of slider trough. */
  129.     int horizTickY;        /* Y-coord at which to draw tick text. */
  130.     /*
  131.      * Layout information for vertical scales, assuming that window
  132.      * gets the size it requested:
  133.      */
  134.  
  135.     int vertTickRightX;        /* X-location of right side of tick-marks. */
  136.     int vertValueRightX;    /* X-location of right side of value string. */
  137.     int vertTroughX;        /* X-location of scale's slider trough. */
  138.     int vertLabelX;        /* X-location of origin of label. */
  139.  
  140.     /*
  141.      * Miscellaneous information:
  142.      */
  143.  
  144.     Tk_Cursor cursor;        /* Current cursor for window, or None. */
  145.     char *takeFocus;        /* Value of -takefocus option;  not used in
  146.                  * the C code, but used by keyboard traversal
  147.                  * scripts.  Malloc'ed, but may be NULL. */
  148.     int flags;            /* Various flags;  see below for
  149.                  * definitions. */
  150. } Scale;
  151.  
  152. /*
  153.  * Flag bits for scales:
  154.  *
  155.  * REDRAW_SLIDER -        1 means slider (and numerical readout) need
  156.  *                to be redrawn.
  157.  * REDRAW_OTHER -        1 means other stuff besides slider and value
  158.  *                need to be redrawn.
  159.  * REDRAW_ALL -            1 means the entire widget needs to be redrawn.
  160.  * ACTIVE -            1 means the widget is active (the mouse is
  161.  *                in its window).
  162.  * INVOKE_COMMAND -        1 means the scale's command needs to be
  163.  *                invoked during the next redisplay (the
  164.  *                value of the scale has changed since the
  165.  *                last time the command was invoked).
  166.  * SETTING_VAR -        1 means that the associated variable is
  167.  *                being set by us, so there's no need for
  168.  *                ScaleVarProc to do anything.
  169.  * NEVER_SET -            1 means that the scale's value has never
  170.  *                been set before (so must invoke -command and
  171.  *                set associated variable even if the value
  172.  *                doesn't appear to have changed).
  173.  * GOT_FOCUS -            1 means that the focus is currently in
  174.  *                this widget.
  175.  */
  176.  
  177. #define REDRAW_SLIDER        1
  178. #define REDRAW_OTHER        2
  179. #define REDRAW_ALL        3
  180. #define ACTIVE            4
  181. #define INVOKE_COMMAND        0x10
  182. #define SETTING_VAR        0x20
  183. #define NEVER_SET        0x40
  184. #define GOT_FOCUS        0x80
  185.  
  186. /*
  187.  * Symbolic values for the active parts of a slider.  These are
  188.  * the values that may be returned by the ScaleElement procedure.
  189.  */
  190.  
  191. #define OTHER        0
  192. #define TROUGH1        1
  193. #define SLIDER        2
  194. #define TROUGH2        3
  195.  
  196. /*
  197.  * Space to leave between scale area and text, and between text and
  198.  * edge of window.
  199.  */
  200.  
  201. #define SPACING 2
  202.  
  203. /*
  204.  * How many characters of space to provide when formatting the
  205.  * scale's value:
  206.  */
  207.  
  208. #define PRINT_CHARS 150
  209.  
  210. /*
  211.  * Information used for argv parsing.
  212.  */
  213.  
  214. static Tk_ConfigSpec configSpecs[] = {
  215.     {TK_CONFIG_BORDER, "-activebackground", "activeBackground", "Foreground",
  216.     DEF_SCALE_ACTIVE_BG_COLOR, Tk_Offset(Scale, activeBorder),
  217.     TK_CONFIG_COLOR_ONLY},
  218.     {TK_CONFIG_BORDER, "-activebackground", "activeBackground", "Foreground",
  219.     DEF_SCALE_ACTIVE_BG_MONO, Tk_Offset(Scale, activeBorder),
  220.     TK_CONFIG_MONO_ONLY},
  221.     {TK_CONFIG_BORDER, "-background", "background", "Background",
  222.     DEF_SCALE_BG_COLOR, Tk_Offset(Scale, bgBorder),
  223.     TK_CONFIG_COLOR_ONLY},
  224.     {TK_CONFIG_BORDER, "-background", "background", "Background",
  225.     DEF_SCALE_BG_MONO, Tk_Offset(Scale, bgBorder),
  226.     TK_CONFIG_MONO_ONLY},
  227.     {TK_CONFIG_DOUBLE, "-bigincrement", "bigIncrement", "BigIncrement",
  228.     DEF_SCALE_BIG_INCREMENT, Tk_Offset(Scale, bigIncrement), 0},
  229.     {TK_CONFIG_SYNONYM, "-bd", "borderWidth", (char *) NULL,
  230.     (char *) NULL, 0, 0},
  231.     {TK_CONFIG_SYNONYM, "-bg", "background", (char *) NULL,
  232.     (char *) NULL, 0, 0},
  233.     {TK_CONFIG_PIXELS, "-borderwidth", "borderWidth", "BorderWidth",
  234.     DEF_SCALE_BORDER_WIDTH, Tk_Offset(Scale, borderWidth), 0},
  235.     {TK_CONFIG_STRING, "-command", "command", "Command",
  236.     DEF_SCALE_COMMAND, Tk_Offset(Scale, command), TK_CONFIG_NULL_OK},
  237.     {TK_CONFIG_ACTIVE_CURSOR, "-cursor", "cursor", "Cursor",
  238.     DEF_SCALE_CURSOR, Tk_Offset(Scale, cursor), TK_CONFIG_NULL_OK},
  239.     {TK_CONFIG_INT, "-digits", "digits", "Digits",
  240.     DEF_SCALE_DIGITS, Tk_Offset(Scale, digits), 0},
  241.     {TK_CONFIG_SYNONYM, "-fg", "foreground", (char *) NULL,
  242.     (char *) NULL, 0, 0},
  243.     {TK_CONFIG_FONT, "-font", "font", "Font",
  244.     DEF_SCALE_FONT, Tk_Offset(Scale, fontPtr),
  245.     0},
  246.     {TK_CONFIG_COLOR, "-foreground", "foreground", "Foreground",
  247.     DEF_SCALE_FG_COLOR, Tk_Offset(Scale, textColorPtr),
  248.     TK_CONFIG_COLOR_ONLY},
  249.     {TK_CONFIG_COLOR, "-foreground", "foreground", "Foreground",
  250.     DEF_SCALE_FG_MONO, Tk_Offset(Scale, textColorPtr),
  251.     TK_CONFIG_MONO_ONLY},
  252.     {TK_CONFIG_DOUBLE, "-from", "from", "From",
  253.     DEF_SCALE_FROM, Tk_Offset(Scale, fromValue), 0},
  254.     {TK_CONFIG_COLOR, "-highlightbackground", "highlightBackground",
  255.     "HighlightBackground", DEF_SCALE_HIGHLIGHT_BG,
  256.     Tk_Offset(Scale, highlightBgColorPtr), 0},
  257.     {TK_CONFIG_COLOR, "-highlightcolor", "highlightColor", "HighlightColor",
  258.     DEF_SCALE_HIGHLIGHT, Tk_Offset(Scale, highlightColorPtr), 0},
  259.     {TK_CONFIG_PIXELS, "-highlightthickness", "highlightThickness",
  260.     "HighlightThickness",
  261.     DEF_SCALE_HIGHLIGHT_WIDTH, Tk_Offset(Scale, highlightWidth), 0},
  262.     {TK_CONFIG_STRING, "-label", "label", "Label",
  263.     DEF_SCALE_LABEL, Tk_Offset(Scale, label), TK_CONFIG_NULL_OK},
  264.     {TK_CONFIG_PIXELS, "-length", "length", "Length",
  265.     DEF_SCALE_LENGTH, Tk_Offset(Scale, length), 0},
  266.     {TK_CONFIG_UID, "-orient", "orient", "Orient",
  267.     DEF_SCALE_ORIENT, Tk_Offset(Scale, orientUid), 0},
  268.     {TK_CONFIG_RELIEF, "-relief", "relief", "Relief",
  269.     DEF_SCALE_RELIEF, Tk_Offset(Scale, relief), 0},
  270.     {TK_CONFIG_INT, "-repeatdelay", "repeatDelay", "RepeatDelay",
  271.     DEF_SCALE_REPEAT_DELAY, Tk_Offset(Scale, repeatDelay), 0},
  272.     {TK_CONFIG_INT, "-repeatinterval", "repeatInterval", "RepeatInterval",
  273.     DEF_SCALE_REPEAT_INTERVAL, Tk_Offset(Scale, repeatInterval), 0},
  274.     {TK_CONFIG_DOUBLE, "-resolution", "resolution", "Resolution",
  275.     DEF_SCALE_RESOLUTION, Tk_Offset(Scale, resolution), 0},
  276.     {TK_CONFIG_BOOLEAN, "-showvalue", "showValue", "ShowValue",
  277.     DEF_SCALE_SHOW_VALUE, Tk_Offset(Scale, showValue), 0},
  278.     {TK_CONFIG_PIXELS, "-sliderlength", "sliderLength", "SliderLength",
  279.     DEF_SCALE_SLIDER_LENGTH, Tk_Offset(Scale, sliderLength), 0},
  280.     {TK_CONFIG_RELIEF, "-sliderrelief", "sliderRelief", "SliderRelief",
  281.     DEF_SCALE_SLIDER_RELIEF, Tk_Offset(Scale, sliderRelief),
  282.     TK_CONFIG_DONT_SET_DEFAULT},
  283.     {TK_CONFIG_UID, "-state", "state", "State",
  284.     DEF_SCALE_STATE, Tk_Offset(Scale, state), 0},
  285.     {TK_CONFIG_STRING, "-takefocus", "takeFocus", "TakeFocus",
  286.     DEF_SCALE_TAKE_FOCUS, Tk_Offset(Scale, takeFocus),
  287.     TK_CONFIG_NULL_OK},
  288.     {TK_CONFIG_DOUBLE, "-tickinterval", "tickInterval", "TickInterval",
  289.     DEF_SCALE_TICK_INTERVAL, Tk_Offset(Scale, tickInterval), 0},
  290.     {TK_CONFIG_DOUBLE, "-to", "to", "To",
  291.     DEF_SCALE_TO, Tk_Offset(Scale, toValue), 0},
  292.     {TK_CONFIG_COLOR, "-troughcolor", "troughColor", "Background",
  293.     DEF_SCALE_TROUGH_COLOR, Tk_Offset(Scale, troughColorPtr),
  294.     TK_CONFIG_COLOR_ONLY},
  295.     {TK_CONFIG_COLOR, "-troughcolor", "troughColor", "Background",
  296.     DEF_SCALE_TROUGH_MONO, Tk_Offset(Scale, troughColorPtr),
  297.     TK_CONFIG_MONO_ONLY},
  298.     {TK_CONFIG_STRING, "-variable", "variable", "Variable",
  299.     DEF_SCALE_VARIABLE, Tk_Offset(Scale, varName), TK_CONFIG_NULL_OK},
  300.     {TK_CONFIG_PIXELS, "-width", "width", "Width",
  301.     DEF_SCALE_WIDTH, Tk_Offset(Scale, width), 0},
  302.     {TK_CONFIG_END, (char *) NULL, (char *) NULL, (char *) NULL,
  303.     (char *) NULL, 0, 0}
  304. };
  305.  
  306. /*
  307.  * Forward declarations for procedures defined later in this file:
  308.  */
  309.  
  310. static void        ComputeFormat _ANSI_ARGS_((Scale *scalePtr));
  311. static void        ComputeScaleGeometry _ANSI_ARGS_((Scale *scalePtr));
  312. static int        ConfigureScale _ANSI_ARGS_((Tcl_Interp *interp,
  313.                 Scale *scalePtr, int argc, char **argv,
  314.                 int flags));
  315. static void        DestroyScale _ANSI_ARGS_((char *memPtr));
  316. static void        DisplayScale _ANSI_ARGS_((ClientData clientData));
  317. static void        DisplayHorizontalScale _ANSI_ARGS_((Scale *scalePtr,
  318.                 Drawable drawable, XRectangle *drawnAreaPtr));
  319. static void        DisplayHorizontalValue _ANSI_ARGS_((Scale *scalePtr,
  320.                 Drawable drawable, double value, int top));
  321. static void        DisplayVerticalScale _ANSI_ARGS_((Scale *scalePtr,
  322.                 Drawable drawable, XRectangle *drawnAreaPtr));
  323. static void        DisplayVerticalValue _ANSI_ARGS_((Scale *scalePtr,
  324.                 Drawable drawable, double value, int rightEdge));
  325. static void        EventuallyRedrawScale _ANSI_ARGS_((Scale *scalePtr,
  326.                 int what));
  327. static double        PixelToValue _ANSI_ARGS_((Scale *scalePtr, int x,
  328.                 int y));
  329. static double        RoundToResolution _ANSI_ARGS_((Scale *scalePtr,
  330.                 double value));
  331. static void        ScaleCmdDeletedProc _ANSI_ARGS_((
  332.                 ClientData clientData));
  333. static int        ScaleElement _ANSI_ARGS_((Scale *scalePtr, int x,
  334.                 int y));
  335. static void        ScaleEventProc _ANSI_ARGS_((ClientData clientData,
  336.                 XEvent *eventPtr));
  337. static char *        ScaleVarProc _ANSI_ARGS_((ClientData clientData,
  338.                 Tcl_Interp *interp, char *name1, char *name2,
  339.                 int flags));
  340. static int        ScaleWidgetCmd _ANSI_ARGS_((ClientData clientData,
  341.                 Tcl_Interp *interp, int argc, char **argv));
  342. static void        SetScaleValue _ANSI_ARGS_((Scale *scalePtr,
  343.                 double value, int setVar, int invokeCommand));
  344. static int        ValueToPixel _ANSI_ARGS_((Scale *scalePtr, double value));
  345.  
  346. /*
  347.  *--------------------------------------------------------------
  348.  *
  349.  * Tk_ScaleCmd --
  350.  *
  351.  *    This procedure is invoked to process the "scale" Tcl
  352.  *    command.  See the user documentation for details on what
  353.  *    it does.
  354.  *
  355.  * Results:
  356.  *    A standard Tcl result.
  357.  *
  358.  * Side effects:
  359.  *    See the user documentation.
  360.  *
  361.  *--------------------------------------------------------------
  362.  */
  363.  
  364. int
  365. Tk_ScaleCmd(clientData, interp, argc, argv)
  366.     ClientData clientData;        /* Main window associated with
  367.                  * interpreter. */
  368.     Tcl_Interp *interp;        /* Current interpreter. */
  369.     int argc;            /* Number of arguments. */
  370.     char **argv;        /* Argument strings. */
  371. {
  372.     Tk_Window tkwin = (Tk_Window) clientData;
  373.     register Scale *scalePtr;
  374.     Tk_Window new;
  375.  
  376.     if (argc < 2) {
  377.     Tcl_AppendResult(interp, "wrong # args: should be \"",
  378.         argv[0], " pathName ?options?\"", (char *) NULL);
  379.     return TCL_ERROR;
  380.     }
  381.  
  382.     new = Tk_CreateWindowFromPath(interp, tkwin, argv[1], (char *) NULL);
  383.     if (new == NULL) {
  384.     return TCL_ERROR;
  385.     }
  386.  
  387.     /*
  388.      * Initialize fields that won't be initialized by ConfigureScale,
  389.      * or which ConfigureScale expects to have reasonable values
  390.      * (e.g. resource pointers).
  391.      */
  392.  
  393.     scalePtr = (Scale *) ckalloc(sizeof(Scale));
  394.     scalePtr->tkwin = new;
  395.     scalePtr->display = Tk_Display(new);
  396.     scalePtr->interp = interp;
  397.     scalePtr->widgetCmd = Tcl_CreateCommand(interp,
  398.         Tk_PathName(scalePtr->tkwin), ScaleWidgetCmd,
  399.         (ClientData) scalePtr, ScaleCmdDeletedProc);
  400.     scalePtr->orientUid = NULL;
  401.     scalePtr->vertical = 0;
  402.     scalePtr->width = 0;
  403.     scalePtr->length = 0;
  404.     scalePtr->value = 0;
  405.     scalePtr->varName = NULL;
  406.     scalePtr->fromValue = 0;
  407.     scalePtr->toValue = 0;
  408.     scalePtr->tickInterval = 0;
  409.     scalePtr->resolution = 1;
  410.     scalePtr->bigIncrement = 0.0;
  411.     scalePtr->command = NULL;
  412.     scalePtr->repeatDelay = 0;
  413.     scalePtr->repeatInterval = 0;
  414.     scalePtr->label = NULL;
  415.     scalePtr->labelLength = 0;
  416.     scalePtr->state = tkNormalUid;
  417.     scalePtr->borderWidth = 0;
  418.     scalePtr->bgBorder = NULL;
  419.     scalePtr->activeBorder = NULL;
  420.     scalePtr->sliderRelief = TK_RELIEF_RAISED;
  421.     scalePtr->troughColorPtr = NULL;
  422.     scalePtr->troughGC = None;
  423.     scalePtr->copyGC = None;
  424.     scalePtr->fontPtr = NULL;
  425.     scalePtr->textColorPtr = NULL;
  426.     scalePtr->textGC = None;
  427.     scalePtr->relief = TK_RELIEF_FLAT;
  428.     scalePtr->highlightWidth = 0;
  429.     scalePtr->highlightBgColorPtr = NULL;
  430.     scalePtr->highlightColorPtr = NULL;
  431.     scalePtr->inset = 0;
  432.     scalePtr->sliderLength = 0;
  433.     scalePtr->showValue = 0;
  434.     scalePtr->horizLabelY = 0;
  435.     scalePtr->horizValueY = 0;
  436.     scalePtr->horizTroughY = 0;
  437.     scalePtr->horizTickY = 0;
  438.     scalePtr->vertTickRightX = 0;
  439.     scalePtr->vertValueRightX = 0;
  440.     scalePtr->vertTroughX = 0;
  441.     scalePtr->vertLabelX = 0;
  442.     scalePtr->cursor = None;
  443.     scalePtr->takeFocus = NULL;
  444.     scalePtr->flags = NEVER_SET;
  445.  
  446.     Tk_SetClass(scalePtr->tkwin, "Scale");
  447.     Tk_CreateEventHandler(scalePtr->tkwin,
  448.         ExposureMask|StructureNotifyMask|FocusChangeMask,
  449.         ScaleEventProc, (ClientData) scalePtr);
  450.     if (ConfigureScale(interp, scalePtr, argc-2, argv+2, 0) != TCL_OK) {
  451.     goto error;
  452.     }
  453.  
  454.     interp->result = Tk_PathName(scalePtr->tkwin);
  455.     return TCL_OK;
  456.  
  457.     error:
  458.     Tk_DestroyWindow(scalePtr->tkwin);
  459.     return TCL_ERROR;
  460. }
  461.  
  462. /*
  463.  *--------------------------------------------------------------
  464.  *
  465.  * ScaleWidgetCmd --
  466.  *
  467.  *    This procedure is invoked to process the Tcl command
  468.  *    that corresponds to a widget managed by this module.
  469.  *    See the user documentation for details on what it does.
  470.  *
  471.  * Results:
  472.  *    A standard Tcl result.
  473.  *
  474.  * Side effects:
  475.  *    See the user documentation.
  476.  *
  477.  *--------------------------------------------------------------
  478.  */
  479.  
  480. static int
  481. ScaleWidgetCmd(clientData, interp, argc, argv)
  482.     ClientData clientData;        /* Information about scale
  483.                      * widget. */
  484.     Tcl_Interp *interp;            /* Current interpreter. */
  485.     int argc;                /* Number of arguments. */
  486.     char **argv;            /* Argument strings. */
  487. {
  488.     register Scale *scalePtr = (Scale *) clientData;
  489.     int result = TCL_OK;
  490.     size_t length;
  491.     int c;
  492.  
  493.     if (argc < 2) {
  494.     Tcl_AppendResult(interp, "wrong # args: should be \"",
  495.         argv[0], " option ?arg arg ...?\"", (char *) NULL);
  496.     return TCL_ERROR;
  497.     }
  498.     Tcl_Preserve((ClientData) scalePtr);
  499.     c = argv[1][0];
  500.     length = strlen(argv[1]);
  501.     if ((c == 'c') && (strncmp(argv[1], "cget", length) == 0)
  502.         && (length >= 2)) {
  503.     if (argc != 3) {
  504.         Tcl_AppendResult(interp, "wrong # args: should be \"",
  505.             argv[0], " cget option\"",
  506.             (char *) NULL);
  507.         goto error;
  508.     }
  509.     result = Tk_ConfigureValue(interp, scalePtr->tkwin, configSpecs,
  510.         (char *) scalePtr, argv[2], 0);
  511.     } else if ((c == 'c') && (strncmp(argv[1], "configure", length) == 0)
  512.         && (length >= 3)) {
  513.     if (argc == 2) {
  514.         result = Tk_ConfigureInfo(interp, scalePtr->tkwin, configSpecs,
  515.             (char *) scalePtr, (char *) NULL, 0);
  516.     } else if (argc == 3) {
  517.         result = Tk_ConfigureInfo(interp, scalePtr->tkwin, configSpecs,
  518.             (char *) scalePtr, argv[2], 0);
  519.     } else {
  520.         result = ConfigureScale(interp, scalePtr, argc-2, argv+2,
  521.             TK_CONFIG_ARGV_ONLY);
  522.     }
  523.     } else if ((c == 'c') && (strncmp(argv[1], "coords", length) == 0)
  524.         && (length >= 3)) {
  525.     int x, y ;
  526.     double value;
  527.  
  528.     if ((argc != 2) && (argc != 3)) {
  529.         Tcl_AppendResult(interp, "wrong # args: should be \"",
  530.             argv[0], " coords ?value?\"", (char *) NULL);
  531.         goto error;
  532.     }
  533.     if (argc == 3) {
  534.         if (Tcl_GetDouble(interp, argv[2], &value) != TCL_OK) {
  535.         goto error;
  536.         }
  537.     } else {
  538.         value = scalePtr->value;
  539.     }
  540.     if (scalePtr->vertical) {
  541.         x = scalePtr->vertTroughX + scalePtr->width/2
  542.             + scalePtr->borderWidth;
  543.         y = ValueToPixel(scalePtr, value);
  544.     } else {
  545.         x = ValueToPixel(scalePtr, value);
  546.         y = scalePtr->horizTroughY + scalePtr->width/2
  547.             + scalePtr->borderWidth;
  548.     }
  549.     sprintf(interp->result, "%d %d", x, y);
  550.     } else if ((c == 'g') && (strncmp(argv[1], "get", length) == 0)) {
  551.     double value;
  552.     int x, y;
  553.  
  554.     if ((argc != 2) && (argc != 4)) {
  555.         Tcl_AppendResult(interp, "wrong # args: should be \"",
  556.             argv[0], " get ?x y?\"", (char *) NULL);
  557.         goto error;
  558.     }
  559.     if (argc == 2) {
  560.         value = scalePtr->value;
  561.     } else {
  562.         if ((Tcl_GetInt(interp, argv[2], &x) != TCL_OK)
  563.             || (Tcl_GetInt(interp, argv[3], &y) != TCL_OK)) {
  564.         goto error;
  565.         }
  566.         value = PixelToValue(scalePtr, x, y);
  567.     }
  568.     sprintf(interp->result, scalePtr->format, value);
  569.     } else if ((c == 'i') && (strncmp(argv[1], "identify", length) == 0)) {
  570.     int x, y, thing;
  571.  
  572.     if (argc != 4) {
  573.         Tcl_AppendResult(interp, "wrong # args: should be \"",
  574.             argv[0], " identify x y\"", (char *) NULL);
  575.         goto error;
  576.     }
  577.     if ((Tcl_GetInt(interp, argv[2], &x) != TCL_OK)
  578.         || (Tcl_GetInt(interp, argv[3], &y) != TCL_OK)) {
  579.         goto error;
  580.     }
  581.     thing = ScaleElement(scalePtr, x,y);
  582.     switch (thing) {
  583.         case TROUGH1:    interp->result = "trough1";    break;
  584.         case SLIDER:    interp->result = "slider";    break;
  585.         case TROUGH2:    interp->result = "trough2";    break;
  586.     }
  587.     } else if ((c == 's') && (strncmp(argv[1], "set", length) == 0)) {
  588.     double value;
  589.  
  590.     if (argc != 3) {
  591.         Tcl_AppendResult(interp, "wrong # args: should be \"",
  592.             argv[0], " set value\"", (char *) NULL);
  593.         goto error;
  594.     }
  595.     if (Tcl_GetDouble(interp, argv[2], &value) != TCL_OK) {
  596.         goto error;
  597.     }
  598.     if (scalePtr->state != tkDisabledUid) {
  599.         SetScaleValue(scalePtr, value, 1, 1);
  600.     }
  601.     } else {
  602.     Tcl_AppendResult(interp, "bad option \"", argv[1],
  603.         "\": must be cget, configure, coords, get, identify, or set",
  604.         (char *) NULL);
  605.     goto error;
  606.     }
  607.     Tcl_Release((ClientData) scalePtr);
  608.     return result;
  609.  
  610.     error:
  611.     Tcl_Release((ClientData) scalePtr);
  612.     return TCL_ERROR;
  613. }
  614.  
  615. /*
  616.  *----------------------------------------------------------------------
  617.  *
  618.  * DestroyScale --
  619.  *
  620.  *    This procedure is invoked by Tcl_EventuallyFree or Tcl_Release
  621.  *    to clean up the internal structure of a button at a safe time
  622.  *    (when no-one is using it anymore).
  623.  *
  624.  * Results:
  625.  *    None.
  626.  *
  627.  * Side effects:
  628.  *    Everything associated with the scale is freed up.
  629.  *
  630.  *----------------------------------------------------------------------
  631.  */
  632.  
  633. static void
  634. DestroyScale(memPtr)
  635.     char *memPtr;    /* Info about scale widget. */
  636. {
  637.     register Scale *scalePtr = (Scale *) memPtr;
  638.  
  639.     /*
  640.      * Free up all the stuff that requires special handling, then
  641.      * let Tk_FreeOptions handle all the standard option-related
  642.      * stuff.
  643.      */
  644.  
  645.     if (scalePtr->varName != NULL) {
  646.     Tcl_UntraceVar(scalePtr->interp, scalePtr->varName,
  647.         TCL_GLOBAL_ONLY|TCL_TRACE_WRITES|TCL_TRACE_UNSETS,
  648.         ScaleVarProc, (ClientData) scalePtr);
  649.     }
  650.     if (scalePtr->troughGC != None) {
  651.     Tk_FreeGC(scalePtr->display, scalePtr->troughGC);
  652.     }
  653.     if (scalePtr->copyGC != None) {
  654.     Tk_FreeGC(scalePtr->display, scalePtr->copyGC);
  655.     }
  656.     if (scalePtr->textGC != None) {
  657.     Tk_FreeGC(scalePtr->display, scalePtr->textGC);
  658.     }
  659.     Tk_FreeOptions(configSpecs, (char *) scalePtr, scalePtr->display, 0);
  660.     ckfree((char *) scalePtr);
  661. }
  662.  
  663. /*
  664.  *----------------------------------------------------------------------
  665.  *
  666.  * ConfigureScale --
  667.  *
  668.  *    This procedure is called to process an argv/argc list, plus
  669.  *    the Tk option database, in order to configure (or
  670.  *    reconfigure) a scale widget.
  671.  *
  672.  * Results:
  673.  *    The return value is a standard Tcl result.  If TCL_ERROR is
  674.  *    returned, then interp->result contains an error message.
  675.  *
  676.  * Side effects:
  677.  *    Configuration information, such as colors, border width,
  678.  *    etc. get set for scalePtr;  old resources get freed,
  679.  *    if there were any.
  680.  *
  681.  *----------------------------------------------------------------------
  682.  */
  683.  
  684. static int
  685. ConfigureScale(interp, scalePtr, argc, argv, flags)
  686.     Tcl_Interp *interp;        /* Used for error reporting. */
  687.     register Scale *scalePtr;    /* Information about widget;  may or may
  688.                  * not already have values for some fields. */
  689.     int argc;            /* Number of valid entries in argv. */
  690.     char **argv;        /* Arguments. */
  691.     int flags;            /* Flags to pass to Tk_ConfigureWidget. */
  692. {
  693.     XGCValues gcValues;
  694.     GC newGC;
  695.     size_t length;
  696.  
  697.     /*
  698.      * Eliminate any existing trace on a variable monitored by the scale.
  699.      */
  700.  
  701.     if (scalePtr->varName != NULL) {
  702.     Tcl_UntraceVar(interp, scalePtr->varName, 
  703.         TCL_GLOBAL_ONLY|TCL_TRACE_WRITES|TCL_TRACE_UNSETS,
  704.         ScaleVarProc, (ClientData) scalePtr);
  705.     }
  706.  
  707.     if (Tk_ConfigureWidget(interp, scalePtr->tkwin, configSpecs,
  708.         argc, argv, (char *) scalePtr, flags) != TCL_OK) {
  709.     return TCL_ERROR;
  710.     }
  711.  
  712.     /*
  713.      * If the scale is tied to the value of a variable, then set up
  714.      * a trace on the variable's value and set the scale's value from
  715.      * the value of the variable, if it exists.
  716.      */
  717.  
  718.     if (scalePtr->varName != NULL) {
  719.     char *stringValue, *end;
  720.     double value;
  721.  
  722.     stringValue = Tcl_GetVar(interp, scalePtr->varName, TCL_GLOBAL_ONLY);
  723.     if (stringValue != NULL) {
  724.         value = strtod(stringValue, &end);
  725.         if ((end != stringValue) && (*end == 0)) {
  726.         scalePtr->value = RoundToResolution(scalePtr, value);
  727.         }
  728.     }
  729.     Tcl_TraceVar(interp, scalePtr->varName,
  730.         TCL_GLOBAL_ONLY|TCL_TRACE_WRITES|TCL_TRACE_UNSETS,
  731.         ScaleVarProc, (ClientData) scalePtr);
  732.     }
  733.  
  734.     /*
  735.      * Several options need special processing, such as parsing the
  736.      * orientation and creating GCs.
  737.      */
  738.  
  739.     length = strlen(scalePtr->orientUid);
  740.     if (strncmp(scalePtr->orientUid, "vertical", length) == 0) {
  741.     scalePtr->vertical = 1;
  742.     } else if (strncmp(scalePtr->orientUid, "horizontal", length) == 0) {
  743.     scalePtr->vertical = 0;
  744.     } else {
  745.     Tcl_AppendResult(interp, "bad orientation \"", scalePtr->orientUid,
  746.         "\": must be vertical or horizontal", (char *) NULL);
  747.     return TCL_ERROR;
  748.     }
  749.  
  750.     scalePtr->fromValue = RoundToResolution(scalePtr, scalePtr->fromValue);
  751.     scalePtr->toValue = RoundToResolution(scalePtr, scalePtr->toValue);
  752.     scalePtr->tickInterval = RoundToResolution(scalePtr,
  753.         scalePtr->tickInterval);
  754.  
  755.     /*
  756.      * Make sure that the tick interval has the right sign so that
  757.      * addition moves from fromValue to toValue.
  758.      */
  759.  
  760.     if ((scalePtr->tickInterval < 0)
  761.         ^ ((scalePtr->toValue - scalePtr->fromValue) <  0)) {
  762.     scalePtr->tickInterval = -scalePtr->tickInterval;
  763.     }
  764.  
  765.     /*
  766.      * Set the scale value to itself;  all this does is to make sure
  767.      * that the scale's value is within the new acceptable range for
  768.      * the scale and reflect the value in the associated variable,
  769.      * if any.
  770.      */
  771.  
  772.     ComputeFormat(scalePtr);
  773.     SetScaleValue(scalePtr, scalePtr->value, 1, 1);
  774.  
  775.     if (scalePtr->label != NULL) {
  776.     scalePtr->labelLength = strlen(scalePtr->label);
  777.     } else {
  778.     scalePtr->labelLength = 0;
  779.     }
  780.  
  781.     if ((scalePtr->state != tkNormalUid)
  782.         && (scalePtr->state != tkDisabledUid)
  783.         && (scalePtr->state != tkActiveUid)) {
  784.     Tcl_AppendResult(interp, "bad state value \"", scalePtr->state,
  785.         "\": must be normal, active, or disabled", (char *) NULL);
  786.     scalePtr->state = tkNormalUid;
  787.     return TCL_ERROR;
  788.     }
  789.  
  790.     Tk_SetBackgroundFromBorder(scalePtr->tkwin, scalePtr->bgBorder);
  791.  
  792.     gcValues.foreground = scalePtr->troughColorPtr->pixel;
  793.     newGC = Tk_GetGC(scalePtr->tkwin, GCForeground, &gcValues);
  794.     if (scalePtr->troughGC != None) {
  795.     Tk_FreeGC(scalePtr->display, scalePtr->troughGC);
  796.     }
  797.     scalePtr->troughGC = newGC;
  798.     if (scalePtr->copyGC == None) {
  799.     gcValues.graphics_exposures = False;
  800.     scalePtr->copyGC = Tk_GetGC(scalePtr->tkwin, GCGraphicsExposures,
  801.         &gcValues);
  802.     }
  803.     if (scalePtr->highlightWidth < 0) {
  804.     scalePtr->highlightWidth = 0;
  805.     }
  806.     gcValues.font = scalePtr->fontPtr->fid;
  807.     gcValues.foreground = scalePtr->textColorPtr->pixel;
  808.     newGC = Tk_GetGC(scalePtr->tkwin, GCForeground|GCFont, &gcValues);
  809.     if (scalePtr->textGC != None) {
  810.     Tk_FreeGC(scalePtr->display, scalePtr->textGC);
  811.     }
  812.     scalePtr->textGC = newGC;
  813.  
  814.     scalePtr->inset = scalePtr->highlightWidth + scalePtr->borderWidth;
  815.  
  816.     /*
  817.      * Recompute display-related information, and let the geometry
  818.      * manager know how much space is needed now.
  819.      */
  820.  
  821.     ComputeScaleGeometry(scalePtr);
  822.  
  823.     EventuallyRedrawScale(scalePtr, REDRAW_ALL);
  824.     return TCL_OK;
  825. }
  826.  
  827. /*
  828.  *----------------------------------------------------------------------
  829.  *
  830.  * ComputeFormat --
  831.  *
  832.  *    This procedure is invoked to recompute the "format" field
  833.  *    of a scale's widget record, which determines how the value
  834.  *    of the scale is converted to a string.
  835.  *
  836.  * Results:
  837.  *    None.
  838.  *
  839.  * Side effects:
  840.  *    The format field of scalePtr is modified.
  841.  *
  842.  *----------------------------------------------------------------------
  843.  */
  844.  
  845. static void
  846. ComputeFormat(scalePtr)
  847.     Scale *scalePtr;            /* Information about scale widget. */
  848. {
  849.     double maxValue, x;
  850.     int mostSigDigit, numDigits, leastSigDigit, afterDecimal;
  851.     int eDigits, fDigits;
  852.  
  853.     /*
  854.      * Compute the displacement from the decimal of the most significant
  855.      * digit required for any number in the scale's range.
  856.      */
  857.  
  858.     maxValue = fabs(scalePtr->fromValue);
  859.     x = fabs(scalePtr->toValue);
  860.     if (x > maxValue) {
  861.     maxValue = x;
  862.     }
  863.     if (maxValue == 0) {
  864.     maxValue = 1;
  865.     }
  866.     mostSigDigit = floor(log10(maxValue));
  867.  
  868.     /*
  869.      * If the number of significant digits wasn't specified explicitly,
  870.      * compute it. It's the difference between the most significant
  871.      * digit needed to represent any number on the scale and the
  872.      * most significant digit of the smallest difference between
  873.      * numbers on the scale.  In other words, display enough digits so
  874.      * that at least one digit will be different between any two adjacent
  875.      * positions of the scale.
  876.      */
  877.  
  878.     numDigits = scalePtr->digits;
  879.     if (numDigits <= 0) {
  880.     if  (scalePtr->resolution > 0) {
  881.         /*
  882.          * A resolution was specified for the scale, so just use it.
  883.          */
  884.  
  885.         leastSigDigit = floor(log10(scalePtr->resolution));
  886.     } else {
  887.         /*
  888.          * No resolution was specified, so compute the difference
  889.          * in value between adjacent pixels and use it for the least
  890.          * significant digit.
  891.          */
  892.  
  893.         x = fabs(scalePtr->fromValue - scalePtr->toValue);
  894.         if (scalePtr->length > 0) {
  895.         x /= scalePtr->length;
  896.         }
  897.         if (x > 0){
  898.         leastSigDigit = floor(log10(x));
  899.         } else {
  900.         leastSigDigit = 0;
  901.         }
  902.     }
  903.     numDigits = mostSigDigit - leastSigDigit + 1;
  904.     if (numDigits < 1) {
  905.         numDigits = 1;
  906.     }
  907.     }
  908.  
  909.     /*
  910.      * Compute the number of characters required using "e" format and
  911.      * "f" format, and then choose whichever one takes fewer characters.
  912.      */
  913.  
  914.     eDigits = numDigits + 4;
  915.     if (numDigits > 1) {
  916.     eDigits++;            /* Decimal point. */
  917.     }
  918.     afterDecimal = numDigits - mostSigDigit - 1;
  919.     if (afterDecimal < 0) {
  920.     afterDecimal = 0;
  921.     }
  922.     fDigits = (mostSigDigit >= 0) ? mostSigDigit + afterDecimal : afterDecimal;
  923.     if (afterDecimal > 0) {
  924.     fDigits++;            /* Decimal point. */
  925.     }
  926.     if (mostSigDigit < 0) {
  927.     fDigits++;            /* Zero to left of decimal point. */
  928.     }
  929.     if (fDigits <= eDigits) {
  930.     sprintf(scalePtr->format, "%%.%df", afterDecimal);
  931.     } else {
  932.     sprintf(scalePtr->format, "%%.%de", numDigits-1);
  933.     }
  934. }
  935.  
  936. /*
  937.  *----------------------------------------------------------------------
  938.  *
  939.  * ComputeScaleGeometry --
  940.  *
  941.  *    This procedure is called to compute various geometrical
  942.  *    information for a scale, such as where various things get
  943.  *    displayed.  It's called when the window is reconfigured.
  944.  *
  945.  * Results:
  946.  *    None.
  947.  *
  948.  * Side effects:
  949.  *    Display-related numbers get changed in *scalePtr.  The
  950.  *    geometry manager gets told about the window's preferred size.
  951.  *
  952.  *----------------------------------------------------------------------
  953.  */
  954.  
  955. static void
  956. ComputeScaleGeometry(scalePtr)
  957.     register Scale *scalePtr;        /* Information about widget. */
  958. {
  959.     XCharStruct bbox;
  960.     char valueString[PRINT_CHARS];
  961.     int dummy, lineHeight, valuePixels, x, y, extraSpace;
  962.  
  963.     /*
  964.      * Horizontal scales are simpler than vertical ones because
  965.      * all sizes are the same (the height of a line of text);
  966.      * handle them first and then quit.
  967.      */
  968.  
  969.     if (!scalePtr->vertical) {
  970.     lineHeight = scalePtr->fontPtr->ascent + scalePtr->fontPtr->descent;
  971.     y = scalePtr->inset;
  972.     extraSpace = 0;
  973.     if (scalePtr->labelLength != 0) {
  974.         scalePtr->horizLabelY = y + SPACING;
  975.         y += lineHeight + SPACING;
  976.         extraSpace = SPACING;
  977.     }
  978.     if (scalePtr->showValue) {
  979.         scalePtr->horizValueY = y + SPACING;
  980.         y += lineHeight + SPACING;
  981.         extraSpace = SPACING;
  982.     } else {
  983.         scalePtr->horizValueY = y;
  984.     }
  985.     y += extraSpace;
  986.     scalePtr->horizTroughY = y;
  987.     y += scalePtr->width + 2*scalePtr->borderWidth;
  988.     if (scalePtr->tickInterval != 0) {
  989.         scalePtr->horizTickY = y + SPACING;
  990.         y += lineHeight + 2*SPACING;
  991.     }
  992.     Tk_GeometryRequest(scalePtr->tkwin,
  993.         scalePtr->length + 2*scalePtr->inset, y + scalePtr->inset);
  994.     Tk_SetInternalBorder(scalePtr->tkwin, scalePtr->inset);
  995.     return;
  996.     }
  997.  
  998.     /*
  999.      * Vertical scale:  compute the amount of space needed to display
  1000.      * the scales value by formatting strings for the two end points;
  1001.      * use whichever length is longer.
  1002.      */
  1003.  
  1004.     sprintf(valueString, scalePtr->format, scalePtr->fromValue);
  1005.     XTextExtents(scalePtr->fontPtr, valueString, (int) strlen(valueString),
  1006.         &dummy, &dummy, &dummy, &bbox);
  1007.     valuePixels = bbox.rbearing - bbox.lbearing;
  1008.     sprintf(valueString, scalePtr->format, scalePtr->toValue);
  1009.     XTextExtents(scalePtr->fontPtr, valueString, (int) strlen(valueString),
  1010.         &dummy, &dummy, &dummy, &bbox);
  1011.     if (valuePixels < bbox.rbearing - bbox.lbearing) {
  1012.     valuePixels = bbox.rbearing - bbox.lbearing;
  1013.     }
  1014.  
  1015.     /*
  1016.      * Assign x-locations to the elements of the scale, working from
  1017.      * left to right.
  1018.      */
  1019.  
  1020.     x = scalePtr->inset;
  1021.     if ((scalePtr->tickInterval != 0) && (scalePtr->showValue)) {
  1022.     scalePtr->vertTickRightX = x + SPACING + valuePixels;
  1023.     scalePtr->vertValueRightX = scalePtr->vertTickRightX + valuePixels
  1024.         + scalePtr->fontPtr->ascent/2;
  1025.     x = scalePtr->vertValueRightX + SPACING;
  1026.     } else if (scalePtr->tickInterval != 0) {
  1027.     scalePtr->vertTickRightX = x + SPACING + valuePixels;
  1028.     scalePtr->vertValueRightX = scalePtr->vertTickRightX;
  1029.     x = scalePtr->vertTickRightX + SPACING;
  1030.     } else if (scalePtr->showValue) {
  1031.     scalePtr->vertTickRightX = x;
  1032.     scalePtr->vertValueRightX = x + SPACING + valuePixels;
  1033.     x = scalePtr->vertValueRightX + SPACING;
  1034.     } else {
  1035.     scalePtr->vertTickRightX = x;
  1036.     scalePtr->vertValueRightX = x;
  1037.     }
  1038.     scalePtr->vertTroughX = x;
  1039.     x += 2*scalePtr->borderWidth + scalePtr->width;
  1040.     if (scalePtr->labelLength == 0) {
  1041.     scalePtr->vertLabelX = 0;
  1042.     } else {
  1043.     XTextExtents(scalePtr->fontPtr, scalePtr->label,
  1044.         scalePtr->labelLength, &dummy, &dummy, &dummy, &bbox);
  1045.     scalePtr->vertLabelX = x + scalePtr->fontPtr->ascent/2 - bbox.lbearing;
  1046.     x = scalePtr->vertLabelX + bbox.rbearing
  1047.         + scalePtr->fontPtr->ascent/2;
  1048.     }
  1049.     Tk_GeometryRequest(scalePtr->tkwin, x + scalePtr->inset,
  1050.         scalePtr->length + 2*scalePtr->inset);
  1051.     Tk_SetInternalBorder(scalePtr->tkwin, scalePtr->inset);
  1052. }
  1053.  
  1054. /*
  1055.  *--------------------------------------------------------------
  1056.  *
  1057.  * DisplayVerticalScale --
  1058.  *
  1059.  *    This procedure redraws the contents of a vertical scale
  1060.  *    window.  It is invoked as a do-when-idle handler, so it only
  1061.  *    runs when there's nothing else for the application to do.
  1062.  *
  1063.  * Results:
  1064.  *    There is no return value.  If only a part of the scale needs
  1065.  *    to be redrawn, then drawnAreaPtr is modified to reflect the
  1066.  *    area that was actually modified.
  1067.  *
  1068.  * Side effects:
  1069.  *    Information appears on the screen.
  1070.  *
  1071.  *--------------------------------------------------------------
  1072.  */
  1073.  
  1074. static void
  1075. DisplayVerticalScale(scalePtr, drawable, drawnAreaPtr)
  1076.     Scale *scalePtr;            /* Widget record for scale. */
  1077.     Drawable drawable;            /* Where to display scale (window
  1078.                      * or pixmap). */
  1079.     XRectangle *drawnAreaPtr;        /* Initally contains area of window;
  1080.                      * if only a part of the scale is
  1081.                      * redrawn, gets modified to reflect
  1082.                      * the part of the window that was
  1083.                      * redrawn. */
  1084. {
  1085.     Tk_Window tkwin = scalePtr->tkwin;
  1086.     int x, y, width, height, shadowWidth;
  1087.     double tickValue;
  1088.     Tk_3DBorder sliderBorder;
  1089.  
  1090.     /*
  1091.      * Display the information from left to right across the window.
  1092.      */
  1093.  
  1094.     if (!(scalePtr->flags & REDRAW_OTHER)) {
  1095.     drawnAreaPtr->x = scalePtr->vertTickRightX;
  1096.     drawnAreaPtr->y = scalePtr->inset;
  1097.     drawnAreaPtr->width = scalePtr->vertTroughX + scalePtr->width
  1098.         + 2*scalePtr->borderWidth - scalePtr->vertTickRightX;
  1099.     drawnAreaPtr->height -= 2*scalePtr->inset;
  1100.     }
  1101.     Tk_Fill3DRectangle(tkwin, drawable, scalePtr->bgBorder,
  1102.         drawnAreaPtr->x, drawnAreaPtr->y, drawnAreaPtr->width,
  1103.         drawnAreaPtr->height, 0, TK_RELIEF_FLAT);
  1104.     if (scalePtr->flags & REDRAW_OTHER) {
  1105.     /*
  1106.      * Display the tick marks.
  1107.      */
  1108.  
  1109.     if (scalePtr->tickInterval != 0) {
  1110.         for (tickValue = scalePtr->fromValue; ;
  1111.             tickValue += scalePtr->tickInterval) {
  1112.         /*
  1113.          * The RoundToResolution call gets rid of accumulated
  1114.          * round-off errors, if any.
  1115.          */
  1116.  
  1117.         tickValue = RoundToResolution(scalePtr, tickValue);
  1118.         if (scalePtr->toValue >= scalePtr->fromValue) {
  1119.             if (tickValue > scalePtr->toValue) {
  1120.             break;
  1121.             }
  1122.         } else {
  1123.             if (tickValue < scalePtr->toValue) {
  1124.             break;
  1125.             }
  1126.         }
  1127.         DisplayVerticalValue(scalePtr, drawable, tickValue,
  1128.             scalePtr->vertTickRightX);
  1129.         }
  1130.     }
  1131.     }
  1132.  
  1133.     /*
  1134.      * Display the value, if it is desired.
  1135.      */
  1136.  
  1137.     if (scalePtr->showValue) {
  1138.     DisplayVerticalValue(scalePtr, drawable, scalePtr->value,
  1139.         scalePtr->vertValueRightX);
  1140.     }
  1141.  
  1142.     /*
  1143.      * Display the trough and the slider.
  1144.      */
  1145.  
  1146.     Tk_Draw3DRectangle(tkwin, drawable,
  1147.         scalePtr->bgBorder, scalePtr->vertTroughX, scalePtr->inset,
  1148.         scalePtr->width + 2*scalePtr->borderWidth,
  1149.         Tk_Height(tkwin) - 2*scalePtr->inset, scalePtr->borderWidth,
  1150.         TK_RELIEF_SUNKEN);
  1151.     XFillRectangle(scalePtr->display, drawable, scalePtr->troughGC,
  1152.         scalePtr->vertTroughX + scalePtr->borderWidth,
  1153.         scalePtr->inset + scalePtr->borderWidth,
  1154.         (unsigned) scalePtr->width,
  1155.         (unsigned) (Tk_Height(tkwin) - 2*scalePtr->inset
  1156.         - 2*scalePtr->borderWidth));
  1157.     if (scalePtr->state == tkActiveUid) {
  1158.     sliderBorder = scalePtr->activeBorder;
  1159.     } else {
  1160.     sliderBorder = scalePtr->bgBorder;
  1161.     }
  1162.     width = scalePtr->width;
  1163.     height = scalePtr->sliderLength/2;
  1164.     x = scalePtr->vertTroughX + scalePtr->borderWidth;
  1165.     y = ValueToPixel(scalePtr, scalePtr->value) - height;
  1166.     shadowWidth = scalePtr->borderWidth/2;
  1167.     if (shadowWidth == 0) {
  1168.     shadowWidth = 1;
  1169.     }
  1170.     Tk_Draw3DRectangle(tkwin, drawable, sliderBorder, x, y, width,
  1171.         2*height, shadowWidth, scalePtr->sliderRelief);
  1172.     x += shadowWidth;
  1173.     y += shadowWidth;
  1174.     width -= 2*shadowWidth;
  1175.     height -= shadowWidth;
  1176.     Tk_Fill3DRectangle(tkwin, drawable, sliderBorder, x, y, width,
  1177.         height, shadowWidth, scalePtr->sliderRelief);
  1178.     Tk_Fill3DRectangle(tkwin, drawable, sliderBorder, x, y+height,
  1179.         width, height, shadowWidth, scalePtr->sliderRelief);
  1180.  
  1181.     /*
  1182.      * Draw the label to the right of the scale.
  1183.      */
  1184.  
  1185.     if ((scalePtr->flags & REDRAW_OTHER) && (scalePtr->labelLength != 0)) {
  1186.     XDrawString(scalePtr->display, drawable,
  1187.         scalePtr->textGC, scalePtr->vertLabelX,
  1188.         scalePtr->inset + (3*scalePtr->fontPtr->ascent)/2,
  1189.         scalePtr->label, scalePtr->labelLength);
  1190.     }
  1191. }
  1192.  
  1193. /*
  1194.  *----------------------------------------------------------------------
  1195.  *
  1196.  * DisplayVerticalValue --
  1197.  *
  1198.  *    This procedure is called to display values (scale readings)
  1199.  *    for vertically-oriented scales.
  1200.  *
  1201.  * Results:
  1202.  *    None.
  1203.  *
  1204.  * Side effects:
  1205.  *    The numerical value corresponding to value is displayed with
  1206.  *    its right edge at "rightEdge", and at a vertical position in
  1207.  *    the scale that corresponds to "value".
  1208.  *
  1209.  *----------------------------------------------------------------------
  1210.  */
  1211.  
  1212. static void
  1213. DisplayVerticalValue(scalePtr, drawable, value, rightEdge)
  1214.     register Scale *scalePtr;    /* Information about widget in which to
  1215.                  * display value. */
  1216.     Drawable drawable;        /* Pixmap or window in which to draw
  1217.                  * the value. */
  1218.     double value;        /* Y-coordinate of number to display,
  1219.                  * specified in application coords, not
  1220.                  * in pixels (we'll compute pixels). */
  1221.     int rightEdge;        /* X-coordinate of right edge of text,
  1222.                  * specified in pixels. */
  1223. {
  1224.     register Tk_Window tkwin = scalePtr->tkwin;
  1225.     int y, dummy, length;
  1226.     char valueString[PRINT_CHARS];
  1227.     XCharStruct bbox;
  1228.  
  1229.     y = ValueToPixel(scalePtr, value) + scalePtr->fontPtr->ascent/2;
  1230.     sprintf(valueString, scalePtr->format, value);
  1231.     length = strlen(valueString);
  1232.     XTextExtents(scalePtr->fontPtr, valueString, length,
  1233.         &dummy, &dummy, &dummy, &bbox);
  1234.  
  1235.     /*
  1236.      * Adjust the y-coordinate if necessary to keep the text entirely
  1237.      * inside the window.
  1238.      */
  1239.  
  1240.     if ((y - bbox.ascent) < (scalePtr->inset + SPACING)) {
  1241.     y = scalePtr->inset + SPACING + bbox.ascent;
  1242.     }
  1243.     if ((y + bbox.descent) > (Tk_Height(tkwin) - scalePtr->inset - SPACING)) {
  1244.     y = Tk_Height(tkwin) - scalePtr->inset - SPACING - bbox.descent;
  1245.     }
  1246.     XDrawString(scalePtr->display, drawable, scalePtr->textGC,
  1247.         rightEdge - bbox.rbearing, y, valueString, length);
  1248. }
  1249.  
  1250. /*
  1251.  *--------------------------------------------------------------
  1252.  *
  1253.  * DisplayHorizontalScale --
  1254.  *
  1255.  *    This procedure redraws the contents of a horizontal scale
  1256.  *    window.  It is invoked as a do-when-idle handler, so it only
  1257.  *    runs when there's nothing else for the application to do.
  1258.  *
  1259.  * Results:
  1260.  *    There is no return value.  If only a part of the scale needs
  1261.  *    to be redrawn, then drawnAreaPtr is modified to reflect the
  1262.  *    area that was actually modified.
  1263.  *
  1264.  * Side effects:
  1265.  *    Information appears on the screen.
  1266.  *
  1267.  *--------------------------------------------------------------
  1268.  */
  1269.  
  1270. static void
  1271. DisplayHorizontalScale(scalePtr, drawable, drawnAreaPtr)
  1272.     Scale *scalePtr;            /* Widget record for scale. */
  1273.     Drawable drawable;            /* Where to display scale (window
  1274.                      * or pixmap). */
  1275.     XRectangle *drawnAreaPtr;        /* Initally contains area of window;
  1276.                      * if only a part of the scale is
  1277.                      * redrawn, gets modified to reflect
  1278.                      * the part of the window that was
  1279.                      * redrawn. */
  1280. {
  1281.     register Tk_Window tkwin = scalePtr->tkwin;
  1282.     int x, y, width, height, shadowWidth;
  1283.     double tickValue;
  1284.     Tk_3DBorder sliderBorder;
  1285.  
  1286.     /*
  1287.      * Display the information from bottom to top across the window.
  1288.      */
  1289.  
  1290.     if (!(scalePtr->flags & REDRAW_OTHER)) {
  1291.     drawnAreaPtr->x = scalePtr->inset;
  1292.     drawnAreaPtr->y = scalePtr->horizValueY;
  1293.     drawnAreaPtr->width -= 2*scalePtr->inset;
  1294.     drawnAreaPtr->height = scalePtr->horizTroughY + scalePtr->width
  1295.         + 2*scalePtr->borderWidth - scalePtr->horizValueY;
  1296.     }
  1297.     Tk_Fill3DRectangle(tkwin, drawable, scalePtr->bgBorder,
  1298.         drawnAreaPtr->x, drawnAreaPtr->y, drawnAreaPtr->width,
  1299.         drawnAreaPtr->height, 0, TK_RELIEF_FLAT);
  1300.     if (scalePtr->flags & REDRAW_OTHER) {
  1301.     /*
  1302.      * Display the tick marks.
  1303.      */
  1304.  
  1305.     if (scalePtr->tickInterval != 0) {
  1306.         for (tickValue = scalePtr->fromValue; ;
  1307.             tickValue += scalePtr->tickInterval) {
  1308.         /*
  1309.          * The RoundToResolution call gets rid of accumulated
  1310.          * round-off errors, if any.
  1311.          */
  1312.  
  1313.         tickValue = RoundToResolution(scalePtr, tickValue);
  1314.         if (scalePtr->toValue >= scalePtr->fromValue) {
  1315.             if (tickValue > scalePtr->toValue) {
  1316.             break;
  1317.             }
  1318.         } else {
  1319.             if (tickValue < scalePtr->toValue) {
  1320.             break;
  1321.             }
  1322.         }
  1323.         DisplayHorizontalValue(scalePtr, drawable, tickValue,
  1324.             scalePtr->horizTickY);
  1325.         }
  1326.     }
  1327.     }
  1328.  
  1329.     /*
  1330.      * Display the value, if it is desired.
  1331.      */
  1332.  
  1333.     if (scalePtr->showValue) {
  1334.     DisplayHorizontalValue(scalePtr, drawable, scalePtr->value,
  1335.         scalePtr->horizValueY);
  1336.     }
  1337.  
  1338.     /*
  1339.      * Display the trough and the slider.
  1340.      */
  1341.  
  1342.     y = scalePtr->horizTroughY;
  1343.     Tk_Draw3DRectangle(tkwin, drawable,
  1344.         scalePtr->bgBorder, scalePtr->inset, y,
  1345.         Tk_Width(tkwin) - 2*scalePtr->inset,
  1346.         scalePtr->width + 2*scalePtr->borderWidth,
  1347.         scalePtr->borderWidth, TK_RELIEF_SUNKEN);
  1348.     XFillRectangle(scalePtr->display, drawable, scalePtr->troughGC,
  1349.         scalePtr->inset + scalePtr->borderWidth,
  1350.         y + scalePtr->borderWidth,
  1351.         (unsigned) (Tk_Width(tkwin) - 2*scalePtr->inset
  1352.         - 2*scalePtr->borderWidth),
  1353.         (unsigned) scalePtr->width);
  1354.     if (scalePtr->state == tkActiveUid) {
  1355.     sliderBorder = scalePtr->activeBorder;
  1356.     } else {
  1357.     sliderBorder = scalePtr->bgBorder;
  1358.     }
  1359.     width = scalePtr->sliderLength/2;
  1360.     height = scalePtr->width;
  1361.     x = ValueToPixel(scalePtr, scalePtr->value) - width;
  1362.     y += scalePtr->borderWidth;
  1363.     shadowWidth = scalePtr->borderWidth/2;
  1364.     if (shadowWidth == 0) {
  1365.     shadowWidth = 1;
  1366.     }
  1367.     Tk_Draw3DRectangle(tkwin, drawable, sliderBorder,
  1368.         x, y, 2*width, height, shadowWidth, scalePtr->sliderRelief);
  1369.     x += shadowWidth;
  1370.     y += shadowWidth;
  1371.     width -= shadowWidth;
  1372.     height -= 2*shadowWidth;
  1373.     Tk_Fill3DRectangle(tkwin, drawable, sliderBorder, x, y, width, height,
  1374.         shadowWidth, scalePtr->sliderRelief);
  1375.     Tk_Fill3DRectangle(tkwin, drawable, sliderBorder, x+width, y,
  1376.         width, height, shadowWidth, scalePtr->sliderRelief);
  1377.  
  1378.     /*
  1379.      * Draw the label at the top of the scale.
  1380.      */
  1381.  
  1382.     if ((scalePtr->flags & REDRAW_OTHER) && (scalePtr->labelLength != 0)) {
  1383.     XDrawString(scalePtr->display, drawable,
  1384.         scalePtr->textGC, scalePtr->inset + scalePtr->fontPtr->ascent/2,
  1385.         scalePtr->horizLabelY + scalePtr->fontPtr->ascent,
  1386.         scalePtr->label, scalePtr->labelLength);
  1387.     }
  1388. }
  1389.  
  1390. /*
  1391.  *----------------------------------------------------------------------
  1392.  *
  1393.  * DisplayHorizontalValue --
  1394.  *
  1395.  *    This procedure is called to display values (scale readings)
  1396.  *    for horizontally-oriented scales.
  1397.  *
  1398.  * Results:
  1399.  *    None.
  1400.  *
  1401.  * Side effects:
  1402.  *    The numerical value corresponding to value is displayed with
  1403.  *    its bottom edge at "bottom", and at a horizontal position in
  1404.  *    the scale that corresponds to "value".
  1405.  *
  1406.  *----------------------------------------------------------------------
  1407.  */
  1408.  
  1409. static void
  1410. DisplayHorizontalValue(scalePtr, drawable, value, top)
  1411.     register Scale *scalePtr;    /* Information about widget in which to
  1412.                  * display value. */
  1413.     Drawable drawable;        /* Pixmap or window in which to draw
  1414.                  * the value. */
  1415.     double value;        /* X-coordinate of number to display,
  1416.                  * specified in application coords, not
  1417.                  * in pixels (we'll compute pixels). */
  1418.     int top;            /* Y-coordinate of top edge of text,
  1419.                  * specified in pixels. */
  1420. {
  1421.     register Tk_Window tkwin = scalePtr->tkwin;
  1422.     int x, y, dummy, length;
  1423.     char valueString[PRINT_CHARS];
  1424.     XCharStruct bbox;
  1425.  
  1426.     x = ValueToPixel(scalePtr, value);
  1427.     y = top + scalePtr->fontPtr->ascent;
  1428.     sprintf(valueString, scalePtr->format, value);
  1429.     length = strlen(valueString);
  1430.     XTextExtents(scalePtr->fontPtr, valueString, length,
  1431.         &dummy, &dummy, &dummy, &bbox);
  1432.  
  1433.     /*
  1434.      * Adjust the x-coordinate if necessary to keep the text entirely
  1435.      * inside the window.
  1436.      */
  1437.  
  1438.     x -= (bbox.rbearing - bbox.lbearing)/2;
  1439.     if ((x + bbox.lbearing) < (scalePtr->inset + SPACING)) {
  1440.     x = scalePtr->inset + SPACING - bbox.lbearing;
  1441.     }
  1442.     if ((x + bbox.rbearing) > (Tk_Width(tkwin) - scalePtr->inset)) {
  1443.     x = Tk_Width(tkwin) - scalePtr->inset - SPACING - bbox.rbearing;
  1444.     }
  1445.     XDrawString(scalePtr->display, drawable, scalePtr->textGC, x, y,
  1446.         valueString, length);
  1447. }
  1448.  
  1449. /*
  1450.  *----------------------------------------------------------------------
  1451.  *
  1452.  * DisplayScale --
  1453.  *
  1454.  *    This procedure is invoked as an idle handler to redisplay
  1455.  *    the contents of a scale widget.
  1456.  *
  1457.  * Results:
  1458.  *    None.
  1459.  *
  1460.  * Side effects:
  1461.  *    The scale gets redisplayed.
  1462.  *
  1463.  *----------------------------------------------------------------------
  1464.  */
  1465.  
  1466. static void
  1467. DisplayScale(clientData)
  1468.     ClientData clientData;    /* Widget record for scale. */
  1469. {
  1470.     Scale *scalePtr = (Scale *) clientData;
  1471.     Tk_Window tkwin = scalePtr->tkwin;
  1472.     Tcl_Interp *interp = scalePtr->interp;
  1473.     Pixmap pixmap;
  1474.     int result;
  1475.     char string[PRINT_CHARS];
  1476.     XRectangle drawnArea;
  1477.  
  1478.     if ((scalePtr->tkwin == NULL) || !Tk_IsMapped(scalePtr->tkwin)) {
  1479.     goto done;
  1480.     }
  1481.  
  1482.     /*
  1483.      * Invoke the scale's command if needed.
  1484.      */
  1485.  
  1486.     Tcl_Preserve((ClientData) scalePtr);
  1487.     Tcl_Preserve((ClientData) interp);
  1488.     if ((scalePtr->flags & INVOKE_COMMAND) && (scalePtr->command != NULL)) {
  1489.     sprintf(string, scalePtr->format, scalePtr->value);
  1490.     result = Tcl_VarEval(interp, scalePtr->command,    " ", string,
  1491.                              (char *) NULL);
  1492.     if (result != TCL_OK) {
  1493.         Tcl_AddErrorInfo(interp, "\n    (command executed by scale)");
  1494.         Tcl_BackgroundError(interp);
  1495.     }
  1496.     }
  1497.     Tcl_Release((ClientData) interp);
  1498.     scalePtr->flags &= ~INVOKE_COMMAND;
  1499.     if (scalePtr->tkwin == NULL) {
  1500.     Tcl_Release((ClientData) scalePtr);
  1501.     return;
  1502.     }
  1503.     Tcl_Release((ClientData) scalePtr);
  1504.  
  1505.     /*
  1506.      * In order to avoid screen flashes, this procedure redraws
  1507.      * the scale in a pixmap, then copies the pixmap to the
  1508.      * screen in a single operation.  This means that there's no
  1509.      * point in time where the on-sreen image has been cleared.
  1510.      */
  1511.  
  1512.     pixmap = Tk_GetPixmap(scalePtr->display, Tk_WindowId(tkwin),
  1513.         Tk_Width(tkwin), Tk_Height(tkwin), Tk_Depth(tkwin));
  1514.     drawnArea.x = 0;
  1515.     drawnArea.y = 0;
  1516.     drawnArea.width = Tk_Width(tkwin);
  1517.     drawnArea.height = Tk_Height(tkwin);
  1518.  
  1519.     /*
  1520.      * Much of the redisplay is done totally differently for
  1521.      * horizontal and vertical scales.  Handle the part that's
  1522.      * different.
  1523.      */
  1524.  
  1525.     if (scalePtr->vertical) {
  1526.     DisplayVerticalScale(scalePtr, pixmap, &drawnArea);
  1527.     } else {
  1528.     DisplayHorizontalScale(scalePtr, pixmap, &drawnArea);
  1529.     }
  1530.  
  1531.     /*
  1532.      * Now handle the part of redisplay that is the same for
  1533.      * horizontal and vertical scales:  border and traversal
  1534.      * highlight.
  1535.      */
  1536.  
  1537.     if (scalePtr->flags & REDRAW_OTHER) {
  1538.     if (scalePtr->relief != TK_RELIEF_FLAT) {
  1539.         Tk_Draw3DRectangle(tkwin, pixmap, scalePtr->bgBorder,
  1540.             scalePtr->highlightWidth, scalePtr->highlightWidth,
  1541.             Tk_Width(tkwin) - 2*scalePtr->highlightWidth,
  1542.             Tk_Height(tkwin) - 2*scalePtr->highlightWidth,
  1543.             scalePtr->borderWidth, scalePtr->relief);
  1544.     }
  1545.     if (scalePtr->highlightWidth != 0) {
  1546.         GC gc;
  1547.     
  1548.         if (scalePtr->flags & GOT_FOCUS) {
  1549.         gc = Tk_GCForColor(scalePtr->highlightColorPtr, pixmap);
  1550.         } else {
  1551.         gc = Tk_GCForColor(scalePtr->highlightBgColorPtr, pixmap);
  1552.         }
  1553.         Tk_DrawFocusHighlight(tkwin, gc, scalePtr->highlightWidth, pixmap);
  1554.     }
  1555.     }
  1556.  
  1557.     /*
  1558.      * Copy the information from the off-screen pixmap onto the screen,
  1559.      * then delete the pixmap.
  1560.      */
  1561.  
  1562.     XCopyArea(scalePtr->display, pixmap, Tk_WindowId(tkwin),
  1563.         scalePtr->copyGC, drawnArea.x, drawnArea.y, drawnArea.width,
  1564.         drawnArea.height, drawnArea.x, drawnArea.y);
  1565.     Tk_FreePixmap(scalePtr->display, pixmap);
  1566.  
  1567.     done:
  1568.     scalePtr->flags &= ~REDRAW_ALL;
  1569. }
  1570.  
  1571. /*
  1572.  *----------------------------------------------------------------------
  1573.  *
  1574.  * ScaleElement --
  1575.  *
  1576.  *    Determine which part of a scale widget lies under a given
  1577.  *    point.
  1578.  *
  1579.  * Results:
  1580.  *    The return value is either TROUGH1, SLIDER, TROUGH2, or
  1581.  *    OTHER, depending on which of the scale's active elements
  1582.  *    (if any) is under the point at (x,y).
  1583.  *
  1584.  * Side effects:
  1585.  *    None.
  1586.  *
  1587.  *----------------------------------------------------------------------
  1588.  */
  1589.  
  1590. static int
  1591. ScaleElement(scalePtr, x, y)
  1592.     Scale *scalePtr;        /* Widget record for scale. */
  1593.     int x, y;            /* Coordinates within scalePtr's window. */
  1594. {
  1595.     int sliderFirst;
  1596.  
  1597.     if (scalePtr->vertical) {
  1598.     if ((x < scalePtr->vertTroughX)
  1599.         || (x >= (scalePtr->vertTroughX + 2*scalePtr->borderWidth +
  1600.         scalePtr->width))) {
  1601.         return OTHER;
  1602.     }
  1603.     if ((y < scalePtr->inset)
  1604.         || (y >= (Tk_Height(scalePtr->tkwin) - scalePtr->inset))) {
  1605.         return OTHER;
  1606.     }
  1607.     sliderFirst = ValueToPixel(scalePtr, scalePtr->value)
  1608.         - scalePtr->sliderLength/2;
  1609.     if (y < sliderFirst) {
  1610.         return TROUGH1;
  1611.     }
  1612.     if (y < (sliderFirst+scalePtr->sliderLength)) {
  1613.         return SLIDER;
  1614.     }
  1615.     return TROUGH2;
  1616.     }
  1617.  
  1618.     if ((y < scalePtr->horizTroughY)
  1619.         || (y >= (scalePtr->horizTroughY + 2*scalePtr->borderWidth +
  1620.         scalePtr->width))) {
  1621.     return OTHER;
  1622.     }
  1623.     if ((x < scalePtr->inset)
  1624.         || (x >= (Tk_Width(scalePtr->tkwin) - scalePtr->inset))) {
  1625.     return OTHER;
  1626.     }
  1627.     sliderFirst = ValueToPixel(scalePtr, scalePtr->value)
  1628.         - scalePtr->sliderLength/2;
  1629.     if (x < sliderFirst) {
  1630.     return TROUGH1;
  1631.     }
  1632.     if (x < (sliderFirst+scalePtr->sliderLength)) {
  1633.     return SLIDER;
  1634.     }
  1635.     return TROUGH2;
  1636. }
  1637.  
  1638. /*
  1639.  *----------------------------------------------------------------------
  1640.  *
  1641.  * PixelToValue --
  1642.  *
  1643.  *    Given a pixel within a scale window, return the scale
  1644.  *    reading corresponding to that pixel.
  1645.  *
  1646.  * Results:
  1647.  *    A double-precision scale reading.  If the value is outside
  1648.  *    the legal range for the scale then it's rounded to the nearest
  1649.  *    end of the scale.
  1650.  *
  1651.  * Side effects:
  1652.  *    None.
  1653.  *
  1654.  *----------------------------------------------------------------------
  1655.  */
  1656.  
  1657. static double
  1658. PixelToValue(scalePtr, x, y)
  1659.     register Scale *scalePtr;        /* Information about widget. */
  1660.     int x, y;                /* Coordinates of point within
  1661.                      * window. */
  1662. {
  1663.     double value, pixelRange;
  1664.  
  1665.     if (scalePtr->vertical) {
  1666.     pixelRange = Tk_Height(scalePtr->tkwin) - scalePtr->sliderLength
  1667.         - 2*scalePtr->inset - 2*scalePtr->borderWidth;
  1668.     value = y;
  1669.     } else {
  1670.     pixelRange = Tk_Width(scalePtr->tkwin) - scalePtr->sliderLength
  1671.         - 2*scalePtr->inset - 2*scalePtr->borderWidth;
  1672.     value = x;
  1673.     }
  1674.  
  1675.     if (pixelRange <= 0) {
  1676.     /*
  1677.      * Not enough room for the slider to actually slide:  just return
  1678.      * the scale's current value.
  1679.      */
  1680.  
  1681.     return scalePtr->value;
  1682.     }
  1683.     value -= scalePtr->sliderLength/2 + scalePtr->inset
  1684.         + scalePtr->borderWidth;
  1685.     value /= pixelRange;
  1686.     if (value < 0) {
  1687.     value = 0;
  1688.     }
  1689.     if (value > 1) {
  1690.     value = 1;
  1691.     }
  1692.     value = scalePtr->fromValue +
  1693.         value * (scalePtr->toValue - scalePtr->fromValue);
  1694.     return RoundToResolution(scalePtr, value);
  1695. }
  1696.  
  1697. /*
  1698.  *----------------------------------------------------------------------
  1699.  *
  1700.  * ValueToPixel --
  1701.  *
  1702.  *    Given a reading of the scale, return the x-coordinate or
  1703.  *    y-coordinate corresponding to that reading, depending on
  1704.  *    whether the scale is vertical or horizontal, respectively.
  1705.  *
  1706.  * Results:
  1707.  *    An integer value giving the pixel location corresponding
  1708.  *    to reading.  The value is restricted to lie within the
  1709.  *    defined range for the scale.
  1710.  *
  1711.  * Side effects:
  1712.  *    None.
  1713.  *
  1714.  *----------------------------------------------------------------------
  1715.  */
  1716.  
  1717. static int
  1718. ValueToPixel(scalePtr, value)
  1719.     register Scale *scalePtr;        /* Information about widget. */
  1720.     double value;            /* Reading of the widget. */
  1721. {
  1722.     int y, pixelRange;
  1723.     double valueRange;
  1724.  
  1725.     valueRange = scalePtr->toValue - scalePtr->fromValue;
  1726.     pixelRange = (scalePtr->vertical ? Tk_Height(scalePtr->tkwin)
  1727.         : Tk_Width(scalePtr->tkwin)) - scalePtr->sliderLength
  1728.         - 2*scalePtr->inset - 2*scalePtr->borderWidth;
  1729.     if (valueRange == 0) {
  1730.     y = 0;
  1731.     } else {
  1732.     y = (int) ((value - scalePtr->fromValue) * pixelRange
  1733.           / valueRange + 0.5);
  1734.     if (y < 0) {
  1735.         y = 0;
  1736.     } else if (y > pixelRange) {
  1737.         y = pixelRange;
  1738.     }
  1739.     }
  1740.     y += scalePtr->sliderLength/2 + scalePtr->inset + scalePtr->borderWidth;
  1741.     return y;
  1742. }
  1743.  
  1744. /*
  1745.  *--------------------------------------------------------------
  1746.  *
  1747.  * ScaleEventProc --
  1748.  *
  1749.  *    This procedure is invoked by the Tk dispatcher for various
  1750.  *    events on scales.
  1751.  *
  1752.  * Results:
  1753.  *    None.
  1754.  *
  1755.  * Side effects:
  1756.  *    When the window gets deleted, internal structures get
  1757.  *    cleaned up.  When it gets exposed, it is redisplayed.
  1758.  *
  1759.  *--------------------------------------------------------------
  1760.  */
  1761.  
  1762. static void
  1763. ScaleEventProc(clientData, eventPtr)
  1764.     ClientData clientData;    /* Information about window. */
  1765.     XEvent *eventPtr;        /* Information about event. */
  1766. {
  1767.     Scale *scalePtr = (Scale *) clientData;
  1768.  
  1769.     if ((eventPtr->type == Expose) && (eventPtr->xexpose.count == 0)) {
  1770.     EventuallyRedrawScale(scalePtr, REDRAW_ALL);
  1771.     } else if (eventPtr->type == DestroyNotify) {
  1772.     if (scalePtr->tkwin != NULL) {
  1773.         scalePtr->tkwin = NULL;
  1774.         Tcl_DeleteCommand(scalePtr->interp,
  1775.             Tcl_GetCommandName(scalePtr->interp, scalePtr->widgetCmd));
  1776.     }
  1777.     if (scalePtr->flags & REDRAW_ALL) {
  1778.         Tcl_CancelIdleCall(DisplayScale, (ClientData) scalePtr);
  1779.     }
  1780.     Tcl_EventuallyFree((ClientData) scalePtr, DestroyScale);
  1781.     } else if (eventPtr->type == ConfigureNotify) {
  1782.     ComputeScaleGeometry(scalePtr);
  1783.     EventuallyRedrawScale(scalePtr, REDRAW_ALL);
  1784.     } else if (eventPtr->type == FocusIn) {
  1785.     if (eventPtr->xfocus.detail != NotifyInferior) {
  1786.         scalePtr->flags |= GOT_FOCUS;
  1787.         if (scalePtr->highlightWidth > 0) {
  1788.         EventuallyRedrawScale(scalePtr, REDRAW_ALL);
  1789.         }
  1790.     }
  1791.     } else if (eventPtr->type == FocusOut) {
  1792.     if (eventPtr->xfocus.detail != NotifyInferior) {
  1793.         scalePtr->flags &= ~GOT_FOCUS;
  1794.         if (scalePtr->highlightWidth > 0) {
  1795.         EventuallyRedrawScale(scalePtr, REDRAW_ALL);
  1796.         }
  1797.     }
  1798.     }
  1799. }
  1800.  
  1801. /*
  1802.  *----------------------------------------------------------------------
  1803.  *
  1804.  * ScaleCmdDeletedProc --
  1805.  *
  1806.  *    This procedure is invoked when a widget command is deleted.  If
  1807.  *    the widget isn't already in the process of being destroyed,
  1808.  *    this command destroys it.
  1809.  *
  1810.  * Results:
  1811.  *    None.
  1812.  *
  1813.  * Side effects:
  1814.  *    The widget is destroyed.
  1815.  *
  1816.  *----------------------------------------------------------------------
  1817.  */
  1818.  
  1819. static void
  1820. ScaleCmdDeletedProc(clientData)
  1821.     ClientData clientData;    /* Pointer to widget record for widget. */
  1822. {
  1823.     Scale *scalePtr = (Scale *) clientData;
  1824.     Tk_Window tkwin = scalePtr->tkwin;
  1825.  
  1826.     /*
  1827.      * This procedure could be invoked either because the window was
  1828.      * destroyed and the command was then deleted (in which case tkwin
  1829.      * is NULL) or because the command was deleted, and then this procedure
  1830.      * destroys the widget.
  1831.      */
  1832.  
  1833.     if (tkwin != NULL) {
  1834.     scalePtr->tkwin = NULL;
  1835.     Tk_DestroyWindow(tkwin);
  1836.     }
  1837. }
  1838.  
  1839. /*
  1840.  *--------------------------------------------------------------
  1841.  *
  1842.  * SetScaleValue --
  1843.  *
  1844.  *    This procedure changes the value of a scale and invokes
  1845.  *    a Tcl command to reflect the current position of a scale
  1846.  *
  1847.  * Results:
  1848.  *    None.
  1849.  *
  1850.  * Side effects:
  1851.  *    A Tcl command is invoked, and an additional error-processing
  1852.  *    command may also be invoked.  The scale's slider is redrawn.
  1853.  *
  1854.  *--------------------------------------------------------------
  1855.  */
  1856.  
  1857. static void
  1858. SetScaleValue(scalePtr, value, setVar, invokeCommand)
  1859.     register Scale *scalePtr;    /* Info about widget. */
  1860.     double value;        /* New value for scale.  Gets adjusted
  1861.                  * if it's off the scale. */
  1862.     int setVar;            /* Non-zero means reflect new value through
  1863.                  * to associated variable, if any. */
  1864.     int invokeCommand;        /* Non-zero means invoked -command option
  1865.                  * to notify of new value, 0 means don't. */
  1866. {
  1867.     char string[PRINT_CHARS];
  1868.  
  1869.     value = RoundToResolution(scalePtr, value);
  1870.     if ((value < scalePtr->fromValue)
  1871.         ^ (scalePtr->toValue < scalePtr->fromValue)) {
  1872.     value = scalePtr->fromValue;
  1873.     }
  1874.     if ((value > scalePtr->toValue)
  1875.         ^ (scalePtr->toValue < scalePtr->fromValue)) {
  1876.     value = scalePtr->toValue;
  1877.     }
  1878.     if (scalePtr->flags & NEVER_SET) {
  1879.     scalePtr->flags &= ~NEVER_SET;
  1880.     } else if (scalePtr->value == value) {
  1881.     return;
  1882.     }
  1883.     scalePtr->value = value;
  1884.     if (invokeCommand) {
  1885.     scalePtr->flags |= INVOKE_COMMAND;
  1886.     }
  1887.     EventuallyRedrawScale(scalePtr, REDRAW_SLIDER);
  1888.  
  1889.     if (setVar && (scalePtr->varName != NULL)) {
  1890.     sprintf(string, scalePtr->format, scalePtr->value);
  1891.     scalePtr->flags |= SETTING_VAR;
  1892.     Tcl_SetVar(scalePtr->interp, scalePtr->varName, string,
  1893.            TCL_GLOBAL_ONLY);
  1894.     scalePtr->flags &= ~SETTING_VAR;
  1895.     }
  1896. }
  1897.  
  1898. /*
  1899.  *--------------------------------------------------------------
  1900.  *
  1901.  * EventuallyRedrawScale --
  1902.  *
  1903.  *    Arrange for part or all of a scale widget to redrawn at
  1904.  *    the next convenient time in the future.
  1905.  *
  1906.  * Results:
  1907.  *    None.
  1908.  *
  1909.  * Side effects:
  1910.  *    If "what" is REDRAW_SLIDER then just the slider and the
  1911.  *    value readout will be redrawn;  if "what" is REDRAW_ALL
  1912.  *    then the entire widget will be redrawn.
  1913.  *
  1914.  *--------------------------------------------------------------
  1915.  */
  1916.  
  1917. static void
  1918. EventuallyRedrawScale(scalePtr, what)
  1919.     register Scale *scalePtr;    /* Information about widget. */
  1920.     int what;            /* What to redraw:  REDRAW_SLIDER
  1921.                  * or REDRAW_ALL. */
  1922. {
  1923.     if ((what == 0) || (scalePtr->tkwin == NULL)
  1924.         || !Tk_IsMapped(scalePtr->tkwin)) {
  1925.     return;
  1926.     }
  1927.     if ((scalePtr->flags & REDRAW_ALL) == 0) {
  1928.     Tcl_DoWhenIdle(DisplayScale, (ClientData) scalePtr);
  1929.     }
  1930.     scalePtr->flags |= what;
  1931. }
  1932.  
  1933. /*
  1934.  *--------------------------------------------------------------
  1935.  *
  1936.  * RoundToResolution --
  1937.  *
  1938.  *    Round a given floating-point value to the nearest multiple
  1939.  *    of the scale's resolution.
  1940.  *
  1941.  * Results:
  1942.  *    The return value is the rounded result.
  1943.  *
  1944.  * Side effects:
  1945.  *    None.
  1946.  *
  1947.  *--------------------------------------------------------------
  1948.  */
  1949.  
  1950. static double
  1951. RoundToResolution(scalePtr, value)
  1952.     Scale *scalePtr;        /* Information about scale widget. */
  1953.     double value;        /* Value to round. */
  1954. {
  1955.     double rem, new;
  1956.  
  1957.     if (scalePtr->resolution <= 0) {
  1958.     return value;
  1959.     }
  1960.     new = scalePtr->resolution * floor(value/scalePtr->resolution);
  1961.     rem = value - new;
  1962.     if (rem < 0) {
  1963.     if (rem <= -scalePtr->resolution/2) {
  1964.         new -= scalePtr->resolution;
  1965.     }
  1966.     } else {
  1967.     if (rem >= scalePtr->resolution/2) {
  1968.         new += scalePtr->resolution;
  1969.     }
  1970.     }
  1971.     return new;
  1972. }
  1973.  
  1974. /*
  1975.  *----------------------------------------------------------------------
  1976.  *
  1977.  * ScaleVarProc --
  1978.  *
  1979.  *    This procedure is invoked by Tcl whenever someone modifies a
  1980.  *    variable associated with a scale widget.
  1981.  *
  1982.  * Results:
  1983.  *    NULL is always returned.
  1984.  *
  1985.  * Side effects:
  1986.  *    The value displayed in the scale will change to match the
  1987.  *    variable's new value.  If the variable has a bogus value then
  1988.  *    it is reset to the value of the scale.
  1989.  *
  1990.  *----------------------------------------------------------------------
  1991.  */
  1992.  
  1993.     /* ARGSUSED */
  1994. static char *
  1995. ScaleVarProc(clientData, interp, name1, name2, flags)
  1996.     ClientData clientData;    /* Information about button. */
  1997.     Tcl_Interp *interp;        /* Interpreter containing variable. */
  1998.     char *name1;        /* Name of variable. */
  1999.     char *name2;        /* Second part of variable name. */
  2000.     int flags;            /* Information about what happened. */
  2001. {
  2002.     register Scale *scalePtr = (Scale *) clientData;
  2003.     char *stringValue, *end, *result;
  2004.     double value;
  2005.  
  2006.     /*
  2007.      * If the variable is unset, then immediately recreate it unless
  2008.      * the whole interpreter is going away.
  2009.      */
  2010.  
  2011.     if (flags & TCL_TRACE_UNSETS) {
  2012.     if ((flags & TCL_TRACE_DESTROYED) && !(flags & TCL_INTERP_DESTROYED)) {
  2013.         Tcl_TraceVar(interp, scalePtr->varName,
  2014.             TCL_GLOBAL_ONLY|TCL_TRACE_WRITES|TCL_TRACE_UNSETS,
  2015.             ScaleVarProc, clientData);
  2016.         scalePtr->flags |= NEVER_SET;
  2017.         SetScaleValue(scalePtr, scalePtr->value, 1, 0);
  2018.     }
  2019.     return (char *) NULL;
  2020.     }
  2021.  
  2022.     /*
  2023.      * If we came here because we updated the variable (in SetScaleValue),
  2024.      * then ignore the trace.  Otherwise update the scale with the value
  2025.      * of the variable.
  2026.      */
  2027.  
  2028.     if (scalePtr->flags & SETTING_VAR) {
  2029.     return (char *) NULL;
  2030.     }
  2031.     result = NULL;
  2032.     stringValue = Tcl_GetVar(interp, scalePtr->varName, TCL_GLOBAL_ONLY);
  2033.     if (stringValue != NULL) {
  2034.     value = strtod(stringValue, &end);
  2035.     if ((end == stringValue) || (*end != 0)) {
  2036.         result = "can't assign non-numeric value to scale variable";
  2037.     } else {
  2038.         scalePtr->value = RoundToResolution(scalePtr, value);
  2039.     }
  2040.  
  2041.     /*
  2042.      * This code is a bit tricky because it sets the scale's value before
  2043.      * calling SetScaleValue.  This way, SetScaleValue won't bother to
  2044.      * set the variable again or to invoke the -command.  However, it
  2045.      * also won't redisplay the scale, so we have to ask for that
  2046.      * explicitly.
  2047.      */
  2048.  
  2049.     SetScaleValue(scalePtr, scalePtr->value, 1, 0);
  2050.     EventuallyRedrawScale(scalePtr, REDRAW_SLIDER);
  2051.     }
  2052.  
  2053.     return result;
  2054. }
  2055.