+++ /dev/null
-
-/*
- * bltGraph.c --
- *
- * This module implements a graph widget for the BLT toolkit.
- *
- * Copyright 1991-1998 Lucent Technologies, Inc.
- *
- * Permission to use, copy, modify, and distribute this software and
- * its documentation for any purpose and without fee is hereby
- * granted, provided that the above copyright notice appear in all
- * copies and that both that the copyright notice and warranty
- * disclaimer appear in supporting documentation, and that the names
- * of Lucent Technologies any of their entities not be used in
- * advertising or publicity pertaining to distribution of the software
- * without specific, written prior permission.
- *
- * Lucent Technologies disclaims all warranties with regard to this
- * software, including all implied warranties of merchantability and
- * fitness. In no event shall Lucent Technologies be liable for any
- * special, indirect or consequential damages or any damages
- * whatsoever resulting from loss of use, data or profits, whether in
- * an action of contract, negligence or other tortuous action, arising
- * out of or in connection with the use or performance of this
- * software.
- *
- * The graph widget was created by Sani Nassif and George Howlett.
- */
-
-/*
- * To do:
- *
- * 2) Update manual pages.
- *
- * 3) Update comments.
- *
- * 5) Surface, contour, and flow graphs
- *
- * 7) Arrows for line markers
- *
- */
-
-#include "bltGraph.h"
-#include "bltBind.h"
-#include "bltGrElem.h"
-#include "bltSwitch.h"
-#include <X11/Xutil.h>
-
-Blt_Uid bltXAxisUid;
-Blt_Uid bltYAxisUid;
-Blt_Uid bltBarElementUid;
-Blt_Uid bltLineElementUid;
-Blt_Uid bltStripElementUid;
-Blt_Uid bltContourElementUid;
-Blt_Uid bltLineMarkerUid;
-Blt_Uid bltBitmapMarkerUid;
-Blt_Uid bltImageMarkerUid;
-Blt_Uid bltTextMarkerUid;
-Blt_Uid bltPolygonMarkerUid;
-Blt_Uid bltWindowMarkerUid;
-
-extern Tk_CustomOption bltLinePenOption;
-extern Tk_CustomOption bltBarPenOption;
-extern Tk_CustomOption bltDistanceOption;
-extern Tk_CustomOption bltBarModeOption;
-extern Tk_CustomOption bltPadOption;
-extern Tk_CustomOption bltTileOption;
-extern Tk_CustomOption bltShadowOption;
-
-#define DEF_GRAPH_ASPECT_RATIO "0.0"
-#define DEF_GRAPH_BAR_BASELINE "0.0"
-#define DEF_GRAPH_BAR_MODE "normal"
-#define DEF_GRAPH_BAR_WIDTH "0.8"
-#define DEF_GRAPH_BACKGROUND STD_NORMAL_BACKGROUND
-#define DEF_GRAPH_BG_MONO STD_NORMAL_BG_MONO
-#define DEF_GRAPH_BORDERWIDTH STD_BORDERWIDTH
-#define DEF_GRAPH_BUFFER_ELEMENTS "1"
-#define DEF_GRAPH_BUFFER_GRAPH "1"
-#define DEF_GRAPH_CURSOR "crosshair"
-#define DEF_GRAPH_FONT STD_FONT_LARGE
-#define DEF_GRAPH_HALO "2m"
-#define DEF_GRAPH_HALO_BAR "0.1i"
-#define DEF_GRAPH_HEIGHT "4i"
-#define DEF_GRAPH_HIGHLIGHT_BACKGROUND STD_NORMAL_BACKGROUND
-#define DEF_GRAPH_HIGHLIGHT_BG_MONO STD_NORMAL_BG_MONO
-#define DEF_GRAPH_HIGHLIGHT_COLOR RGB_BLACK
-#define DEF_GRAPH_HIGHLIGHT_WIDTH "0"
-#define DEF_GRAPH_INVERT_XY "0"
-#define DEF_GRAPH_JUSTIFY "center"
-#define DEF_GRAPH_MARGIN "0"
-#define DEF_GRAPH_MARGIN_VAR (char *)NULL
-#define DEF_GRAPH_PLOT_BACKGROUND RGB_WHITE
-#define DEF_GRAPH_PLOT_BG_MONO RGB_WHITE
-#define DEF_GRAPH_PLOT_BW_COLOR STD_BORDERWIDTH
-#define DEF_GRAPH_PLOT_BW_MONO "0"
-#define DEF_GRAPH_PLOT_PADX "8"
-#define DEF_GRAPH_PLOT_PADY "8"
-#define DEF_GRAPH_PLOT_RELIEF "sunken"
-#define DEF_GRAPH_RELIEF "flat"
-#define DEF_GRAPH_SHADOW_COLOR (char *)NULL
-#define DEF_GRAPH_SHADOW_MONO (char *)NULL
-#define DEF_GRAPH_SHOW_VALUES "no"
-#define DEF_GRAPH_TAKE_FOCUS ""
-#define DEF_GRAPH_TITLE (char *)NULL
-#define DEF_GRAPH_TITLE_COLOR STD_NORMAL_FOREGROUND
-#define DEF_GRAPH_TITLE_MONO STD_NORMAL_FG_MONO
-#define DEF_GRAPH_WIDTH "5i"
-#define DEF_GRAPH_DATA (char *)NULL
-#define DEF_GRAPH_DATA_COMMAND (char *)NULL
-
-static Tk_ConfigSpec configSpecs[] =
-{
- {TK_CONFIG_DOUBLE, "-aspect", "aspect", "Aspect",
- DEF_GRAPH_ASPECT_RATIO, Tk_Offset(Graph, aspect),
- TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_BORDER, "-background", "background", "Background",
- DEF_GRAPH_BACKGROUND, Tk_Offset(Graph, border),
- TK_CONFIG_COLOR_ONLY},
- {TK_CONFIG_BORDER, "-background", "background", "Background",
- DEF_GRAPH_BG_MONO, Tk_Offset(Graph, border),
- TK_CONFIG_MONO_ONLY},
- {TK_CONFIG_CUSTOM, "-barmode", "barMode", "BarMode",
- DEF_GRAPH_BAR_MODE, Tk_Offset(Graph, mode),
- TK_CONFIG_DONT_SET_DEFAULT, &bltBarModeOption},
- {TK_CONFIG_DOUBLE, "-barwidth", "barWidth", "BarWidth",
- DEF_GRAPH_BAR_WIDTH, Tk_Offset(Graph, barWidth), 0},
- {TK_CONFIG_DOUBLE, "-baseline", "baseline", "Baseline",
- DEF_GRAPH_BAR_BASELINE, Tk_Offset(Graph, baseline), 0},
- {TK_CONFIG_SYNONYM, "-bd", "borderWidth", (char *)NULL, (char *)NULL, 0, 0},
- {TK_CONFIG_SYNONYM, "-bg", "background", (char *)NULL, (char *)NULL, 0, 0},
- {TK_CONFIG_SYNONYM, "-bm", "bottomMargin",
- (char *)NULL, (char *)NULL, 0, 0},
- {TK_CONFIG_CUSTOM, "-borderwidth", "borderWidth", "BorderWidth",
- DEF_GRAPH_BORDERWIDTH, Tk_Offset(Graph, borderWidth),
- TK_CONFIG_DONT_SET_DEFAULT, &bltDistanceOption},
- {TK_CONFIG_CUSTOM, "-bottommargin", "bottomMargin", "Margin",
- DEF_GRAPH_MARGIN, Tk_Offset(Graph, bottomMargin.reqSize), 0,
- &bltDistanceOption},
- {TK_CONFIG_STRING, "-bottomvariable", "bottomVariable", "BottomVariable",
- DEF_GRAPH_MARGIN_VAR, Tk_Offset(Graph, bottomMargin.varName),
- TK_CONFIG_NULL_OK},
- {TK_CONFIG_BOOLEAN, "-bufferelements", "bufferElements", "BufferElements",
- DEF_GRAPH_BUFFER_ELEMENTS, Tk_Offset(Graph, backingStore),
- TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_BOOLEAN, "-buffergraph", "bufferGraph", "BufferGraph",
- DEF_GRAPH_BUFFER_GRAPH, Tk_Offset(Graph, doubleBuffer),
- TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_ACTIVE_CURSOR, "-cursor", "cursor", "Cursor",
- DEF_GRAPH_CURSOR, Tk_Offset(Graph, cursor), TK_CONFIG_NULL_OK},
- {TK_CONFIG_STRING, "-data", "data", "Data",
- (char *)NULL, Tk_Offset(Graph, data), TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_STRING, "-redrawcmd", "redrawCmd", "RedrawCmd",
- (char *)NULL, Tk_Offset(Graph, redrawCmd), TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_SYNONYM, "-fg", "foreground", (char *)NULL, (char *)NULL, 0, 0},
- {TK_CONFIG_FONT, "-font", "font", "Font",
- DEF_GRAPH_FONT, Tk_Offset(Graph, titleTextStyle.font), 0},
- {TK_CONFIG_COLOR, "-foreground", "foreground", "Foreground",
- DEF_GRAPH_TITLE_COLOR, Tk_Offset(Graph, titleTextStyle.color),
- TK_CONFIG_COLOR_ONLY},
- {TK_CONFIG_COLOR, "-foreground", "foreground", "Foreground",
- DEF_GRAPH_TITLE_MONO, Tk_Offset(Graph, titleTextStyle.color),
- TK_CONFIG_MONO_ONLY},
- {TK_CONFIG_CUSTOM, "-halo", "halo", "Halo",
- DEF_GRAPH_HALO, Tk_Offset(Graph, halo), 0, &bltDistanceOption},
- {TK_CONFIG_CUSTOM, "-height", "height", "Height",
- DEF_GRAPH_HEIGHT, Tk_Offset(Graph, reqHeight), 0, &bltDistanceOption},
- {TK_CONFIG_COLOR, "-highlightbackground", "highlightBackground",
- "HighlightBackground",
- DEF_GRAPH_HIGHLIGHT_BACKGROUND, Tk_Offset(Graph, highlightBgColor),
- TK_CONFIG_COLOR_ONLY},
- {TK_CONFIG_COLOR, "-highlightbackground", "highlightBackground",
- "HighlightBackground",
- DEF_GRAPH_HIGHLIGHT_BG_MONO, Tk_Offset(Graph, highlightBgColor),
- TK_CONFIG_MONO_ONLY},
- {TK_CONFIG_COLOR, "-highlightcolor", "highlightColor", "HighlightColor",
- DEF_GRAPH_HIGHLIGHT_COLOR, Tk_Offset(Graph, highlightColor), 0},
- {TK_CONFIG_PIXELS, "-highlightthickness", "highlightThickness",
- "HighlightThickness",
- DEF_GRAPH_HIGHLIGHT_WIDTH, Tk_Offset(Graph, highlightWidth),
- TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_BOOLEAN, "-invertxy", "invertXY", "InvertXY",
- DEF_GRAPH_INVERT_XY, Tk_Offset(Graph, inverted),
- TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_JUSTIFY, "-justify", "justify", "Justify",
- DEF_GRAPH_JUSTIFY, Tk_Offset(Graph, titleTextStyle.justify),
- TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_CUSTOM, "-leftmargin", "leftMargin", "Margin",
- DEF_GRAPH_MARGIN, Tk_Offset(Graph, leftMargin.reqSize),
- TK_CONFIG_DONT_SET_DEFAULT, &bltDistanceOption},
- {TK_CONFIG_STRING, "-leftvariable", "leftVariable", "LeftVariable",
- DEF_GRAPH_MARGIN_VAR, Tk_Offset(Graph, leftMargin.varName),
- TK_CONFIG_NULL_OK},
- {TK_CONFIG_SYNONYM, "-lm", "leftMargin", (char *)NULL, (char *)NULL, 0, 0},
- {TK_CONFIG_COLOR, "-plotbackground", "plotBackground", "Background",
- DEF_GRAPH_PLOT_BG_MONO, Tk_Offset(Graph, plotBg),
- TK_CONFIG_MONO_ONLY},
- {TK_CONFIG_COLOR, "-plotbackground", "plotBackground", "Background",
- DEF_GRAPH_PLOT_BACKGROUND, Tk_Offset(Graph, plotBg),
- TK_CONFIG_COLOR_ONLY},
- {TK_CONFIG_CUSTOM, "-plotborderwidth", "plotBorderWidth", "BorderWidth",
- DEF_GRAPH_PLOT_BW_COLOR, Tk_Offset(Graph, plotBorderWidth),
- TK_CONFIG_COLOR_ONLY, &bltDistanceOption},
- {TK_CONFIG_CUSTOM, "-plotborderwidth", "plotBorderWidth", "BorderWidth",
- DEF_GRAPH_PLOT_BW_MONO, Tk_Offset(Graph, plotBorderWidth),
- TK_CONFIG_MONO_ONLY, &bltDistanceOption},
- {TK_CONFIG_CUSTOM, "-plotpadx", "plotPadX", "PlotPad",
- DEF_GRAPH_PLOT_PADX, Tk_Offset(Graph, padX),
- TK_CONFIG_DONT_SET_DEFAULT, &bltPadOption},
- {TK_CONFIG_CUSTOM, "-plotpady", "plotPadY", "PlotPad",
- DEF_GRAPH_PLOT_PADY, Tk_Offset(Graph, padY),
- TK_CONFIG_DONT_SET_DEFAULT, &bltPadOption},
- {TK_CONFIG_RELIEF, "-plotrelief", "plotRelief", "Relief",
- DEF_GRAPH_PLOT_RELIEF, Tk_Offset(Graph, plotRelief),
- TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_RELIEF, "-relief", "relief", "Relief",
- DEF_GRAPH_RELIEF, Tk_Offset(Graph, relief), TK_CONFIG_DONT_SET_DEFAULT},
- {TK_CONFIG_CUSTOM, "-rightmargin", "rightMargin", "Margin",
- DEF_GRAPH_MARGIN, Tk_Offset(Graph, rightMargin.reqSize),
- TK_CONFIG_DONT_SET_DEFAULT, &bltDistanceOption},
- {TK_CONFIG_STRING, "-rightvariable", "rightVariable", "RightVariable",
- DEF_GRAPH_MARGIN_VAR, Tk_Offset(Graph, rightMargin.varName),
- TK_CONFIG_NULL_OK},
- {TK_CONFIG_SYNONYM, "-rm", "rightMargin", (char *)NULL, (char *)NULL, 0, 0},
- {TK_CONFIG_CUSTOM, "-shadow", "shadow", "Shadow",
- DEF_GRAPH_SHADOW_COLOR, Tk_Offset(Graph, titleTextStyle.shadow),
- TK_CONFIG_COLOR_ONLY, &bltShadowOption},
- {TK_CONFIG_CUSTOM, "-shadow", "shadow", "Shadow",
- DEF_GRAPH_SHADOW_MONO, Tk_Offset(Graph, titleTextStyle.shadow),
- TK_CONFIG_MONO_ONLY, &bltShadowOption},
- {TK_CONFIG_COLOR, "-foreground", "foreground", "Foreground",
- DEF_GRAPH_TITLE_MONO, Tk_Offset(Graph, titleTextStyle.color),
- TK_CONFIG_MONO_ONLY},
- {TK_CONFIG_STRING, "-takefocus", "takeFocus", "TakeFocus",
- DEF_GRAPH_TAKE_FOCUS, Tk_Offset(Graph, takeFocus), TK_CONFIG_NULL_OK},
- {TK_CONFIG_CUSTOM, "-tile", "tile", "Tile",
- (char *)NULL, Tk_Offset(Graph, tile),
- TK_CONFIG_NULL_OK, &bltTileOption},
- {TK_CONFIG_STRING, "-title", "title", "Title",
- DEF_GRAPH_TITLE, Tk_Offset(Graph, title), TK_CONFIG_NULL_OK},
- {TK_CONFIG_SYNONYM, "-tm", "topMargin", (char *)NULL, (char *)NULL, 0, 0},
- {TK_CONFIG_CUSTOM, "-topmargin", "topMargin", "Margin",
- DEF_GRAPH_MARGIN, Tk_Offset(Graph, topMargin.reqSize),
- TK_CONFIG_DONT_SET_DEFAULT, &bltDistanceOption},
- {TK_CONFIG_STRING, "-topvariable", "topVariable", "TopVariable",
- DEF_GRAPH_MARGIN_VAR, Tk_Offset(Graph, topMargin.varName),
- TK_CONFIG_NULL_OK},
- {TK_CONFIG_CUSTOM, "-width", "width", "Width",
- DEF_GRAPH_WIDTH, Tk_Offset(Graph, reqWidth),
- 0, &bltDistanceOption},
- {TK_CONFIG_END, NULL, NULL, NULL, NULL, 0, 0}
-};
-
-static Blt_SwitchParseProc StringToFormat;
-static Blt_SwitchCustom formatSwitch =
-{
- StringToFormat, (Blt_SwitchFreeProc *)NULL, (ClientData)0,
-};
-
-typedef struct {
- char *name;
- int width, height;
- int format;
-} SnapData;
-
-enum SnapFormats { FORMAT_PHOTO, FORMAT_EMF, FORMAT_WMF };
-
-static Blt_SwitchSpec snapSwitches[] =
-{
- {BLT_SWITCH_INT_POSITIVE, "-width", Blt_Offset(SnapData, width), 0},
- {BLT_SWITCH_INT_POSITIVE, "-height", Blt_Offset(SnapData, height), 0},
- {BLT_SWITCH_CUSTOM, "-format", Blt_Offset(SnapData, format), 0,
- &formatSwitch},
- {BLT_SWITCH_END, NULL, 0, 0}
-};
-
-static Tcl_IdleProc DisplayGraph;
-static Tcl_FreeProc DestroyGraph;
-static Tk_EventProc GraphEventProc;
-Tcl_CmdProc Blt_GraphInstCmdProc;
-
-static Blt_BindPickProc PickEntry;
-static Tcl_CmdProc StripchartCmd;
-static Tcl_CmdProc BarchartCmd;
-static Tcl_CmdProc GraphCmd;
-static Tcl_CmdDeleteProc GraphInstCmdDeleteProc;
-static Blt_TileChangedProc TileChangedProc;
-
-static void widgetWorldChanged(ClientData clientData);
-
-static Tk_ClassProcs graphClass = {
- sizeof(Tk_ClassProcs), /* size */
- widgetWorldChanged, /* worldChangedProc */
-};
-
-
-/*
- *--------------------------------------------------------------
- *
- * Blt_EventuallyRedrawGraph --
- *
- * Tells the Tk dispatcher to call the graph display routine at
- * the next idle point. This request is made only if the window
- * is displayed and no other redraw request is pending.
- *
- * Results: None.
- *
- * Side effects:
- * The window is eventually redisplayed.
- *
- *--------------------------------------------------------------
- */
-void
-Blt_EventuallyRedrawGraph(graphPtr)
- Graph *graphPtr; /* Graph widget record */
-{
- if ((graphPtr->tkwin != NULL) && !(graphPtr->flags & REDRAW_PENDING)) {
- Tcl_DoWhenIdle(DisplayGraph, graphPtr);
- graphPtr->flags |= REDRAW_PENDING;
- }
-}
-
-static void widgetWorldChanged(ClientData clientData) {
- Graph *graphPtr = (Graph*)clientData;
-
- graphPtr->flags |= (REDRAW_WORLD | RESET_WORLD | REDRAW_BACKING_STORE | RESET_AXES | MAP_WORLD );
- Blt_ConfigureAxes(graphPtr);
- Blt_EventuallyRedrawGraph(graphPtr);
-}
-
-/*
- *--------------------------------------------------------------
- *
- * GraphEventProc --
- *
- * This procedure is invoked by the Tk dispatcher for various
- * events on graphs.
- *
- * Results:
- * None.
- *
- * Side effects:
- * When the window gets deleted, internal structures get
- * cleaned up. When it gets exposed, the graph is eventually
- * redisplayed.
- *
- *--------------------------------------------------------------
- */
-static void
-GraphEventProc(clientData, eventPtr)
- ClientData clientData; /* Graph widget record */
- register XEvent *eventPtr; /* Event which triggered call to routine */
-{
- Graph *graphPtr = (Graph *)clientData;
-
- if (eventPtr->type == Expose) {
- if (eventPtr->xexpose.count == 0) {
- graphPtr->flags |= REDRAW_WORLD;
- Blt_EventuallyRedrawGraph(graphPtr);
- }
- } else if ((eventPtr->type == FocusIn) || (eventPtr->type == FocusOut)) {
- if (eventPtr->xfocus.detail != NotifyInferior) {
- if (eventPtr->type == FocusIn) {
- graphPtr->flags |= GRAPH_FOCUS;
- } else {
- graphPtr->flags &= ~GRAPH_FOCUS;
- }
- graphPtr->flags |= REDRAW_WORLD;
- Blt_EventuallyRedrawGraph(graphPtr);
- }
- } else if (eventPtr->type == DestroyNotify) {
- if (graphPtr->tkwin != NULL) {
- Blt_DeleteAxisLabelsGC(graphPtr->tkwin);
- Blt_DeleteWindowInstanceData(graphPtr->tkwin);
- graphPtr->tkwin = NULL;
- Tcl_DeleteCommandFromToken(graphPtr->interp, graphPtr->cmdToken);
- }
- if (graphPtr->flags & REDRAW_PENDING) {
- Tcl_CancelIdleCall(DisplayGraph, graphPtr);
- }
- Tcl_EventuallyFree(graphPtr, DestroyGraph);
- } else if (eventPtr->type == ConfigureNotify) {
- graphPtr->flags |= (MAP_WORLD | REDRAW_WORLD);
- Blt_EventuallyRedrawGraph(graphPtr);
- }
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * GraphInstCmdDeleteProc --
- *
- * This procedure is invoked when a widget command is deleted. If
- * the widget isn't already in the process of being destroyed,
- * this command destroys it.
- *
- * Results:
- * None.
- *
- * Side effects:
- * The widget is destroyed.
- *
- *---------------------------------------------------------------------- */
-static void
-GraphInstCmdDeleteProc(clientData)
- ClientData clientData; /* Pointer to widget record. */
-{
- Graph *graphPtr = (Graph *)clientData;
-
- if (graphPtr->tkwin != NULL) { /* NULL indicates window has
- * already been destroyed. */
- Tk_Window tkwin;
-
- tkwin = graphPtr->tkwin;
- graphPtr->tkwin = NULL;
-#ifdef ITCL_NAMESPACES
- Itk_SetWidgetCommand(tkwin, (Tcl_Command) NULL);
-#endif /* ITCL_NAMESPACES */
- Blt_DeleteWindowInstanceData(tkwin);
- Tk_DestroyWindow(tkwin);
- }
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * TileChangedProc
- *
- * Rebuilds the designated GC with the new tile pixmap.
- *
- * Results:
- * None.
- *
- *----------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static void
-TileChangedProc(clientData, tile)
- ClientData clientData;
- Blt_Tile tile; /* Not used. */
-{
- Graph *graphPtr = (Graph *)clientData;
-
- if (graphPtr->tkwin != NULL) {
- graphPtr->flags |= REDRAW_WORLD;
- Blt_EventuallyRedrawGraph(graphPtr);
- }
-}
-
-/*
- *--------------------------------------------------------------
- *
- * AdjustAxisPointers --
- *
- * Sets the axis pointers according to whether the axis is
- * inverted on not. The axis sites are also reset.
- *
- * Results:
- * None.
- *
- *--------------------------------------------------------------
- */
-static void
-AdjustAxisPointers(graphPtr)
- Graph *graphPtr; /* Graph widget record */
-{
- if (graphPtr->inverted) {
- graphPtr->leftMargin.axes = graphPtr->axisChain[0];
- graphPtr->bottomMargin.axes = graphPtr->axisChain[1];
- graphPtr->rightMargin.axes = graphPtr->axisChain[2];
- graphPtr->topMargin.axes = graphPtr->axisChain[3];
- } else {
- graphPtr->leftMargin.axes = graphPtr->axisChain[1];
- graphPtr->bottomMargin.axes = graphPtr->axisChain[0];
- graphPtr->rightMargin.axes = graphPtr->axisChain[3];
- graphPtr->topMargin.axes = graphPtr->axisChain[2];
- }
-}
-
-static int
-InitPens(graphPtr)
- Graph *graphPtr;
-{
- Blt_InitHashTable(&(graphPtr->penTable), BLT_STRING_KEYS);
- if (Blt_CreatePen(graphPtr, "activeLine", bltLineElementUid, 0,
- (char **)NULL) == NULL) {
- return TCL_ERROR;
- }
- if (Blt_CreatePen(graphPtr, "activeBar", bltBarElementUid, 0,
- (char **)NULL) == NULL) {
- return TCL_ERROR;
- }
- return TCL_OK;
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * Blt_GraphTags --
- *
- * Sets the binding tags for a graph object. This routine is
- * called by Tk when an event occurs in the graph. It fills
- * an array of pointers with bind tag addresses.
- *
- * The object addresses are strings hashed in one of two tag
- * tables: one for elements and the another for markers. Note
- * that there's only one binding table for elements and markers.
- * [We don't want to trigger both a marker and element bind
- * command for the same event.] But we don't want a marker and
- * element with the same tag name to activate the others
- * bindings. A tag "all" for markers should mean all markers, not
- * all markers and elements. As a result, element and marker
- * tags are stored in separate hash tables, which means we can't
- * generate the same tag address for both an elements and marker,
- * even if they have the same name.
- *
- * Results:
- * None.
- *
- * Side effects:
- * This information will be used by the binding code in bltUtil.c
- * to determine what graph objects match the current event. The
- * tags are placed in tagArr and *nTagsPtr is set with the
- * number of tags found.
- *
- *----------------------------------------------------------------------
- */
-/*ARGSUSED*/
-void
-Blt_GraphTags(table, object, context, list)
- Blt_BindTable table;
- ClientData object;
- ClientData context; /* Not used. */
- Blt_List list;
-{
- Element *elemPtr;
- MakeTagProc *tagProc;
- Graph *graphPtr;
-
- graphPtr = (Graph *)Blt_GetBindingData(table);
- /*
- * Trick: Markers, elements, and axes have the same first few
- * fields in their structures, such as "type", "name", or
- * "tags". This is so we can look at graph objects
- * interchangably. It doesn't matter what we cast the
- * object to.
- */
- elemPtr = (Element *)object;
-
- if ((elemPtr->classUid == bltLineElementUid) ||
- (elemPtr->classUid == bltStripElementUid) ||
- (elemPtr->classUid == bltBarElementUid)) {
- tagProc = Blt_MakeElementTag;
- } else if ((elemPtr->classUid == bltXAxisUid) ||
- (elemPtr->classUid == bltYAxisUid)) {
- tagProc = Blt_MakeAxisTag;
- } else {
- tagProc = Blt_MakeMarkerTag;
- }
- /*
- * Always add the name of the object to the tag array.
- */
- Blt_ListAppend(list, (*tagProc) (graphPtr, elemPtr->name), 0);
- Blt_ListAppend(list, (*tagProc) (graphPtr, elemPtr->classUid), 0);
- if (elemPtr->tags != NULL) {
- register char **p;
-
- for (p = elemPtr->tags; *p != NULL; p++) {
- Blt_ListAppend(list, (*tagProc) (graphPtr, *p), 0);
- }
- }
-}
-
-/*
- * Find the closest point from the set of displayed elements,
- * searching the display list from back to front. That way, if
- * the points from two different elements overlay each other exactly,
- * the one that's on top (visible) is picked.
- */
-/*ARGSUSED*/
-static ClientData
-PickEntry(clientData, x, y, contextPtr)
- ClientData clientData;
- int x, y;
- ClientData *contextPtr; /* Not used. */
-{
- Graph *graphPtr = (Graph *)clientData;
- Blt_ChainLink *linkPtr;
- Element *elemPtr;
- Marker *markerPtr;
- Extents2D exts;
-
- if (graphPtr->flags & MAP_ALL) {
- return NULL; /* Can't pick anything until the next
- * redraw occurs. */
- }
- Blt_GraphExtents(graphPtr, &exts);
-
- if ((x > exts.right) || (x < exts.left) || (y > exts.bottom) ||
- (y < exts.top)) {
- /*
- * Sample coordinate is in one of the graph margins. Can only
- * pick an axis.
- */
- return (int *)Blt_NearestAxis(graphPtr, x, y);
- }
-
- /*
- * From top-to-bottom check:
- * 1. markers drawn on top (-under false).
- * 2. elements using its display list back to front.
- * 3. markers drawn under element (-under true).
- */
- markerPtr = (Marker *)Blt_NearestMarker(graphPtr, x, y, FALSE);
- if (markerPtr != NULL) {
- return (int *)markerPtr; /* Found a marker (-under false). */
- }
- {
- ClosestSearch search;
-
- search.along = SEARCH_BOTH;
- search.halo = graphPtr->halo + 1;
- search.index = -1;
- search.x = x;
- search.y = y;
- search.dist = (double)(search.halo + 1);
- search.mode = SEARCH_AUTO;
-
- for (linkPtr = Blt_ChainLastLink(graphPtr->elements.displayList);
- linkPtr != NULL; linkPtr = Blt_ChainPrevLink(linkPtr)) {
- elemPtr = (Element *)Blt_ChainGetValue(linkPtr);
- if ((elemPtr->flags & MAP_ITEM) ||
- (Blt_VectorNotifyPending(elemPtr->x.clientId)) ||
- (Blt_VectorNotifyPending(elemPtr->y.clientId))) {
- continue;
- }
- if ((!elemPtr->hidden) && (elemPtr->state == STATE_NORMAL)) {
- (*elemPtr->procsPtr->closestProc) (graphPtr, elemPtr, &search);
- }
- }
- if (search.dist <= (double)search.halo) {
- return (int *)search.elemPtr; /* Found an element within the
- * minimum halo distance. */
- }
- }
- markerPtr = (Marker *)Blt_NearestMarker(graphPtr, x, y, TRUE);
- if (markerPtr != NULL) {
- return (int *)markerPtr; /* Found a marker (-under true) */
- }
- return NULL; /* Nothing found. */
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * ConfigureGraph --
- *
- * Allocates resources for the graph.
- *
- * Results:
- * None.
- *
- * Side effects:
- * Configuration information, such as text string, colors, font,
- * etc. get set for graphPtr; old resources get freed, if there
- * were any. The graph is redisplayed.
- *
- *----------------------------------------------------------------------
- */
-static void
-ConfigureGraph(graphPtr)
- Graph *graphPtr; /* Graph widget record */
-{
- XColor *colorPtr;
- GC newGC;
- XGCValues gcValues;
- unsigned long gcMask;
- Tcl_Interp *interp = graphPtr->interp;
-
- /* Don't allow negative bar widths. Reset to an arbitrary value (0.1) */
- if (graphPtr->barWidth <= 0.0) {
- graphPtr->barWidth = 0.1;
- }
- graphPtr->inset = graphPtr->borderWidth + graphPtr->highlightWidth + 1;
- if ((graphPtr->reqHeight != Tk_ReqHeight(graphPtr->tkwin)) ||
- (graphPtr->reqWidth != Tk_ReqWidth(graphPtr->tkwin))) {
- Tk_GeometryRequest(graphPtr->tkwin, graphPtr->reqWidth,
- graphPtr->reqHeight);
- }
- Tk_SetInternalBorder(graphPtr->tkwin, graphPtr->borderWidth);
- colorPtr = Tk_3DBorderColor(graphPtr->border);
-
- if (graphPtr->title != NULL) {
- int w, h;
-
- Blt_GetTextExtents(&graphPtr->titleTextStyle, graphPtr->title, &w, &h);
- graphPtr->titleTextStyle.height = h + 10;
- } else {
- graphPtr->titleTextStyle.width = graphPtr->titleTextStyle.height = 0;
- }
-
- /*
- * Create GCs for interior and exterior regions, and a background
- * GC for clearing the margins with XFillRectangle
- */
-
- /* Margin GC */
-
- gcValues.foreground = graphPtr->titleTextStyle.color->pixel;
- gcValues.background = colorPtr->pixel;
- gcMask = (GCForeground | GCBackground);
- newGC = Tk_GetGC(graphPtr->tkwin, gcMask, &gcValues);
- if (graphPtr->drawGC != NULL) {
- Tk_FreeGC(graphPtr->display, graphPtr->drawGC);
- }
- graphPtr->drawGC = newGC;
-
- /* Plot fill GC (Background = Foreground) */
-
- gcValues.foreground = graphPtr->plotBg->pixel;
- newGC = Tk_GetGC(graphPtr->tkwin, gcMask, &gcValues);
- if (graphPtr->plotFillGC != NULL) {
- Tk_FreeGC(graphPtr->display, graphPtr->plotFillGC);
- }
- graphPtr->plotFillGC = newGC;
-
- /* Margin fill GC (Background = Foreground) */
-
- gcValues.foreground = colorPtr->pixel;
- gcValues.background = graphPtr->titleTextStyle.color->pixel;
- newGC = Tk_GetGC(graphPtr->tkwin, gcMask, &gcValues);
- if (graphPtr->fillGC != NULL) {
- Tk_FreeGC(graphPtr->display, graphPtr->fillGC);
- }
- graphPtr->fillGC = newGC;
- if (graphPtr->tile != NULL) {
- Blt_SetTileChangedProc(graphPtr->tile, TileChangedProc, graphPtr);
- }
-
- Blt_ResetTextStyle(graphPtr->tkwin, &graphPtr->titleTextStyle);
-
- if (Blt_ConfigModified(configSpecs, interp, "-invertxy", (char *)NULL)) {
-
- /*
- * If the -inverted option changed, we need to readjust the pointers
- * to the axes and recompute the their scales.
- */
-
- AdjustAxisPointers(graphPtr);
- graphPtr->flags |= RESET_AXES;
- }
- if ((!graphPtr->backingStore) && (graphPtr->backPixmap != None)) {
-
- /*
- * Free the pixmap if we're not buffering the display of elements
- * anymore.
- */
-
- Tk_FreePixmap(graphPtr->display, graphPtr->backPixmap);
- graphPtr->backPixmap = None;
- }
- /*
- * Reconfigure the crosshairs, just in case the background color of
- * the plotarea has been changed.
- */
- Blt_ConfigureCrosshairs(graphPtr);
-
- /*
- * Update the layout of the graph (and redraw the elements) if
- * any of the following graph options which affect the size of
- * the plotting area has changed.
- *
- * -aspect
- * -borderwidth, -plotborderwidth
- * -font, -title
- * -width, -height
- * -invertxy
- * -bottommargin, -leftmargin, -rightmargin, -topmargin,
- * -barmode, -barwidth
- */
- if (Blt_ConfigModified(configSpecs, interp, "-invertxy", "-title", "-font",
- "-*margin", "-*width", "-height", "-barmode", "-*pad*", "-aspect",
- (char *)NULL)) {
- graphPtr->flags |= RESET_WORLD;
- }
- if (Blt_ConfigModified(configSpecs, interp, "-plotbackground", (char *)NULL)) {
- graphPtr->flags |= REDRAW_BACKING_STORE;
- }
- graphPtr->flags |= REDRAW_WORLD;
- Blt_EventuallyRedrawGraph(graphPtr);
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * DestroyGraph --
- *
- * This procedure is invoked by Tcl_EventuallyFree or Tcl_Release
- * to clean up the internal structure of a graph at a safe time
- * (when no-one is using it anymore).
- *
- * Results:
- * None.
- *
- * Side effects:
- * Everything associated with the widget is freed up.
- *
- *----------------------------------------------------------------------
- */
-static void
-DestroyGraph(dataPtr)
- DestroyData dataPtr;
-{
- Graph *graphPtr = (Graph *)dataPtr;
-
- Tk_FreeOptions(configSpecs, (char *)graphPtr, graphPtr->display, 0);
- /*
- * Destroy the individual components of the graph: elements, markers,
- * X and Y axes, legend, display lists etc.
- */
- Blt_DestroyMarkers(graphPtr);
- Blt_DestroyElements(graphPtr);
-
- Blt_DestroyAxes(graphPtr);
- Blt_DestroyPens(graphPtr);
-
- if (graphPtr->legend != NULL) {
- Blt_DestroyLegend(graphPtr);
- }
- if (graphPtr->postscript != NULL) {
- Blt_DestroyPostScript(graphPtr);
- }
- if (graphPtr->crosshairs != NULL) {
- Blt_DestroyCrosshairs(graphPtr);
- }
- if (graphPtr->gridPtr != NULL) {
- Blt_DestroyGrid(graphPtr);
- }
- if (graphPtr->bindTable != NULL) {
- Blt_DestroyBindingTable(graphPtr->bindTable);
- }
-
- /* Release allocated X resources and memory. */
- if (graphPtr->drawGC != NULL) {
- Tk_FreeGC(graphPtr->display, graphPtr->drawGC);
- }
- if (graphPtr->fillGC != NULL) {
- Tk_FreeGC(graphPtr->display, graphPtr->fillGC);
- }
- if (graphPtr->plotFillGC != NULL) {
- Tk_FreeGC(graphPtr->display, graphPtr->plotFillGC);
- }
- Blt_FreeTextStyle(graphPtr->display, &graphPtr->titleTextStyle);
- if (graphPtr->backPixmap != None) {
- Tk_FreePixmap(graphPtr->display, graphPtr->backPixmap);
- }
- if (graphPtr->freqArr != NULL) {
- Blt_Free(graphPtr->freqArr);
- }
- if (graphPtr->nStacks > 0) {
- Blt_DeleteHashTable(&graphPtr->freqTable);
- }
- if (graphPtr->tile != NULL) {
- Blt_FreeTile(graphPtr->tile);
- }
- Blt_Free(graphPtr);
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * CreateGraph --
- *
- * This procedure creates and initializes a new widget.
- *
- * Results:
- * The return value is a pointer to a structure describing
- * the new widget. If an error occurred, then the return
- * value is NULL and an error message is left in interp->result.
- *
- * Side effects:
- * Memory is allocated, a Tk_Window is created, etc.
- *
- *----------------------------------------------------------------------
- */
-
-static Graph *
-CreateGraph(interp, argc, argv, classUid)
- Tcl_Interp *interp;
- int argc;
- char **argv;
- Blt_Uid classUid;
-{
- Graph *graphPtr;
- Tk_Window tkwin;
-
- tkwin = Tk_CreateWindowFromPath(interp, Tk_MainWindow(interp), argv[1],
- (char *)NULL);
- if (tkwin == NULL) {
- return NULL;
- }
- graphPtr = Blt_Calloc(1, sizeof(Graph));
- assert(graphPtr);
-
- /* Initialize the graph data structure. */
-
- graphPtr->tkwin = tkwin;
- graphPtr->display = Tk_Display(tkwin);
- graphPtr->interp = interp;
- graphPtr->classUid = classUid;
- graphPtr->backingStore = TRUE;
- graphPtr->doubleBuffer = TRUE;
- graphPtr->highlightWidth = 0;
- graphPtr->plotRelief = TK_RELIEF_SUNKEN;
- graphPtr->relief = TK_RELIEF_FLAT;
- graphPtr->flags = (RESET_WORLD);
- graphPtr->nextMarkerId = 1;
- graphPtr->padLeft = graphPtr->padRight = 8;
- graphPtr->padTop = graphPtr->padBottom = 8;
- graphPtr->bottomMargin.site = MARGIN_BOTTOM;
- graphPtr->leftMargin.site = MARGIN_LEFT;
- graphPtr->topMargin.site = MARGIN_TOP;
- graphPtr->rightMargin.site = MARGIN_RIGHT;
- Blt_InitTextStyle(&graphPtr->titleTextStyle);
-
- Blt_InitHashTable(&graphPtr->axes.table, BLT_STRING_KEYS);
- Blt_InitHashTable(&graphPtr->axes.tagTable, BLT_STRING_KEYS);
- Blt_InitHashTable(&graphPtr->elements.table, BLT_STRING_KEYS);
- Blt_InitHashTable(&graphPtr->elements.tagTable, BLT_STRING_KEYS);
- Blt_InitHashTable(&graphPtr->markers.table, BLT_STRING_KEYS);
- Blt_InitHashTable(&graphPtr->markers.tagTable, BLT_STRING_KEYS);
- graphPtr->elements.displayList = Blt_ChainCreate();
- graphPtr->markers.displayList = Blt_ChainCreate();
- graphPtr->axes.displayList = Blt_ChainCreate();
-
- if (classUid == bltLineElementUid) {
- Tk_SetClass(tkwin, "Graph");
- } else if (classUid == bltBarElementUid) {
- Tk_SetClass(tkwin, "Barchart");
- } else if (classUid == bltStripElementUid) {
- Tk_SetClass(tkwin, "Stripchart");
- }
- Blt_SetWindowInstanceData(tkwin, graphPtr);
-
- if (InitPens(graphPtr) != TCL_OK) {
- goto error;
- }
- if (Tk_ConfigureWidget(interp, tkwin, configSpecs, argc - 2, argv + 2,
- (char *)graphPtr, 0) != TCL_OK) {
- goto error;
- }
- if (Blt_DefaultAxes(graphPtr) != TCL_OK) {
- goto error;
- }
- AdjustAxisPointers(graphPtr);
-
- if (Blt_CreatePostScript(graphPtr) != TCL_OK) {
- goto error;
- }
- if (Blt_CreateCrosshairs(graphPtr) != TCL_OK) {
- goto error;
- }
- if (Blt_CreateLegend(graphPtr) != TCL_OK) {
- goto error;
- }
- if (Blt_CreateGrid(graphPtr) != TCL_OK) {
- goto error;
- }
- Tk_CreateEventHandler(graphPtr->tkwin,
- ExposureMask | StructureNotifyMask | FocusChangeMask, GraphEventProc,
- graphPtr);
-
- graphPtr->cmdToken = Tcl_CreateCommand(interp, argv[1],
- Blt_GraphInstCmdProc, graphPtr, GraphInstCmdDeleteProc);
-#ifdef ITCL_NAMESPACES
- Itk_SetWidgetCommand(graphPtr->tkwin, graphPtr->cmdToken);
-#endif
- ConfigureGraph(graphPtr);
- graphPtr->bindTable = Blt_CreateBindingTable(interp, tkwin, graphPtr,
- PickEntry, Blt_GraphTags);
- Tk_SetClassProcs(tkwin, &graphClass, (ClientData)graphPtr);
-
- return graphPtr;
-
- error:
- DestroyGraph((DestroyData)graphPtr);
- return NULL;
-}
-\f
-/* Widget sub-commands */
-
-/*ARGSUSED*/
-static int
-XAxisOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp; /* Not used. */
- int argc;
- char **argv;
-{
- int margin;
-
- margin = (graphPtr->inverted) ? MARGIN_LEFT : MARGIN_BOTTOM;
- return Blt_AxisOp(graphPtr, margin, argc, argv);
-}
-
-/*ARGSUSED*/
-static int
-X2AxisOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp; /* Not used. */
- int argc;
- char **argv;
-{
- int margin;
-
- margin = (graphPtr->inverted) ? MARGIN_RIGHT : MARGIN_TOP;
- return Blt_AxisOp(graphPtr, margin, argc, argv);
-}
-
-/*ARGSUSED*/
-static int
-YAxisOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp; /* Not used. */
- int argc;
- char **argv;
-{
- int margin;
-
- margin = (graphPtr->inverted) ? MARGIN_BOTTOM : MARGIN_LEFT;
- return Blt_AxisOp(graphPtr, margin, argc, argv);
-}
-
-/*ARGSUSED*/
-static int
-Y2AxisOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp; /* Not used. */
- int argc;
- char **argv;
-{
- int margin;
-
- margin = (graphPtr->inverted) ? MARGIN_TOP : MARGIN_RIGHT;
- return Blt_AxisOp(graphPtr, margin, argc, argv);
-}
-
-/*ARGSUSED*/
-static int
-BarOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp; /* Not used. */
- int argc;
- char **argv;
-{
- return Blt_ElementOp(graphPtr, interp, argc, argv, bltBarElementUid);
-}
-
-/*ARGSUSED*/
-static int
-LineOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp; /* Not used. */
- int argc;
- char **argv;
-{
- return Blt_ElementOp(graphPtr, interp, argc, argv, bltLineElementUid);
-}
-
-/*ARGSUSED*/
-static int
-ElementOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp; /* Not used. */
- int argc;
- char **argv;
-{
- return Blt_ElementOp(graphPtr, interp, argc, argv, graphPtr->classUid);
-}
-
-static int
-ConfigureOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp;
- int argc;
- char **argv;
-{
- int flags;
-
- flags = TK_CONFIG_ARGV_ONLY;
- if (argc == 2) {
- return Tk_ConfigureInfo(interp, graphPtr->tkwin, configSpecs,
- (char *)graphPtr, (char *)NULL, flags);
- } else if (argc == 3) {
- return Tk_ConfigureInfo(interp, graphPtr->tkwin, configSpecs,
- (char *)graphPtr, argv[2], flags);
- } else {
- if (Tk_ConfigureWidget(interp, graphPtr->tkwin, configSpecs, argc - 2,
- argv + 2, (char *)graphPtr, flags) != TCL_OK) {
- return TCL_ERROR;
- }
- ConfigureGraph(graphPtr);
- return TCL_OK;
- }
-}
-
-/* ARGSUSED*/
-static int
-CgetOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- return Tk_ConfigureValue(interp, graphPtr->tkwin, configSpecs,
- (char *)graphPtr, argv[2], 0);
-}
-
-/*
- *--------------------------------------------------------------
- *
- * ExtentsOp --
- *
- * Reports the size of one of several items within the graph.
- * The following are valid items:
- *
- * "bottommargin" Height of the bottom margin
- * "leftmargin" Width of the left margin
- * "legend" x y w h of the legend
- * "plotarea" x y w h of the plotarea
- * "plotheight" Height of the plot area
- * "rightmargin" Width of the right margin
- * "topmargin" Height of the top margin
- * "plotwidth" Width of the plot area
- *
- * Results:
- * Always returns TCL_OK.
- *
- *--------------------------------------------------------------
- */
-/* ARGSUSED*/
-static int
-ExtentsOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- char c;
- unsigned int length;
- char string[200];
-
- c = argv[2][0];
- length = strlen(argv[2]);
- if ((c == 'p') && (length > 4) &&
- (strncmp("plotheight", argv[2], length) == 0)) {
- Tcl_SetResult(interp, Blt_Itoa(graphPtr->bottom - graphPtr->top + 1),
- TCL_VOLATILE);
- } else if ((c == 'p') && (length > 4) &&
- (strncmp("plotwidth", argv[2], length) == 0)) {
- Tcl_SetResult(interp, Blt_Itoa(graphPtr->right - graphPtr->left + 1),
- TCL_VOLATILE);
- } else if ((c == 'p') && (length > 4) &&
- (strncmp("plotarea", argv[2], length) == 0)) {
- sprintf(string, "%d %d %d %d", graphPtr->left, graphPtr->top,
- graphPtr->right - graphPtr->left + 1,
- graphPtr->bottom - graphPtr->top + 1);
- Tcl_SetResult(interp, string, TCL_VOLATILE);
- } else if ((c == 'l') && (length > 2) &&
- (strncmp("legend", argv[2], length) == 0)) {
- sprintf(string, "%d %d %d %d", Blt_LegendX(graphPtr->legend),
- Blt_LegendY(graphPtr->legend),
- Blt_LegendWidth(graphPtr->legend),
- Blt_LegendHeight(graphPtr->legend));
- Tcl_SetResult(interp, string, TCL_VOLATILE);
- } else if ((c == 'l') && (length > 2) &&
- (strncmp("leftmargin", argv[2], length) == 0)) {
- Tcl_SetResult(interp, Blt_Itoa(graphPtr->leftMargin.width),
- TCL_VOLATILE);
- } else if ((c == 'r') && (length > 1) &&
- (strncmp("rightmargin", argv[2], length) == 0)) {
- Tcl_SetResult(interp, Blt_Itoa(graphPtr->rightMargin.width),
- TCL_VOLATILE);
- } else if ((c == 't') && (length > 1) &&
- (strncmp("topmargin", argv[2], length) == 0)) {
- Tcl_SetResult(interp, Blt_Itoa(graphPtr->topMargin.height), TCL_VOLATILE);
- } else if ((c == 'b') && (length > 1) &&
- (strncmp("bottommargin", argv[2], length) == 0)) {
- Tcl_SetResult(interp, Blt_Itoa(graphPtr->bottomMargin.height),
- TCL_VOLATILE);
- } else {
- Tcl_AppendResult(interp, "bad extent item \"", argv[2],
- "\": should be plotheight, plotwidth, leftmargin, rightmargin, \
-topmargin, bottommargin, plotarea, or legend", (char *)NULL);
- return TCL_ERROR;
- }
- return TCL_OK;
-}
-
-/*
- *--------------------------------------------------------------
- *
- * InsideOp --
- *
- * Returns true of false whether the given point is inside
- * the plotting area (defined by left,bottom right, top).
- *
- * Results:
- * Always returns TCL_OK. interp->result will contain
- * the boolean string representation.
- *
- *--------------------------------------------------------------
- */
-/* ARGSUSED*/
-static int
-InsideOp(graphPtr, interp, argc, argv)
- Graph *graphPtr;
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- int x, y;
- Extents2D exts;
- int result;
-
- if (Tk_GetPixels(interp, graphPtr->tkwin, argv[2], &x) != TCL_OK) {
- return TCL_ERROR;
- }
- if (Tk_GetPixels(interp, graphPtr->tkwin, argv[3], &y) != TCL_OK) {
- return TCL_ERROR;
- }
- Blt_GraphExtents(graphPtr, &exts);
- result = PointInRegion(&exts, x, y);
- Blt_SetBooleanResult(interp, result);
- return TCL_OK;
-}
-
-/*
- * -------------------------------------------------------------------------
- *
- * InvtransformOp --
- *
- * This procedure returns a list of the graph coordinate
- * values corresponding with the given window X and Y
- * coordinate positions.
- *
- * Results:
- * Returns a standard Tcl result. If an error occurred while
- * parsing the window positions, TCL_ERROR is returned, and
- * interp->result will contain the error message. Otherwise
- * interp->result will contain a Tcl list of the x and y
- * coordinates.
- *
- * ------------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-InvtransformOp(graphPtr, interp, argc, argv)
- Graph *graphPtr; /* Graph widget record */
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- double x, y;
- Point2D point;
- Axis2D axes;
-
- if (Tcl_ExprDouble(interp, argv[2], &x) != TCL_OK ||
- Tcl_ExprDouble(interp, argv[3], &y) != TCL_OK) {
- return TCL_ERROR;
- }
- if (graphPtr->flags & RESET_AXES) {
- Blt_ResetAxes(graphPtr);
- }
- /* Perform the reverse transformation, converting from window
- * coordinates to graph data coordinates. Note that the point is
- * always mapped to the bottom and left axes (which may not be
- * what the user wants). */
-
- /* Pick the first pair of axes */
- axes.x = Blt_GetFirstAxis(graphPtr->axisChain[0]);
- axes.y = Blt_GetFirstAxis(graphPtr->axisChain[1]);
- point = Blt_InvMap2D(graphPtr, x, y, &axes);
-
- Tcl_AppendElement(interp, Blt_Dtoa(interp, point.x));
- Tcl_AppendElement(interp, Blt_Dtoa(interp, point.y));
- return TCL_OK;
-}
-
-/*
- * --------------------------------------------------------------------------
- *
- * TransformOp --
- *
- * This procedure returns a list of the window coordinates
- * corresponding with the given graph x and y coordinates.
- *
- * Results:
- * Returns a standard Tcl result. interp->result contains
- * the list of the graph coordinates. If an error occurred
- * while parsing the window positions, TCL_ERROR is returned,
- * then interp->result will contain an error message.
- *
- * -------------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-TransformOp(graphPtr, interp, argc, argv)
- Graph *graphPtr; /* Graph widget record */
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- double x, y;
- Point2D point;
- Axis2D axes;
-
- if ((Tcl_ExprDouble(interp, argv[2], &x) != TCL_OK) ||
- (Tcl_ExprDouble(interp, argv[3], &y) != TCL_OK)) {
- return TCL_ERROR;
- }
- if (graphPtr->flags & RESET_AXES) {
- Blt_ResetAxes(graphPtr);
- }
- /*
- * Perform the transformation from window to graph coordinates.
- * Note that the points are always mapped onto the bottom and left
- * axes (which may not be the what the user wants).
- */
- axes.x = Blt_GetFirstAxis(graphPtr->axisChain[0]);
- axes.y = Blt_GetFirstAxis(graphPtr->axisChain[1]);
-
- point = Blt_Map2D(graphPtr, x, y, &axes);
- Tcl_AppendElement(interp, Blt_Itoa(ROUND(point.x)));
- Tcl_AppendElement(interp, Blt_Itoa(ROUND(point.y)));
- return TCL_OK;
-}
-
-#ifndef NO_PRINTER
-
-/*
- * --------------------------------------------------------------------------
- *
- * Print1Op --
- *
- * Prints the equivalent of a screen snapshot of the graph
- * to the designated printer.
- *
- * Results:
- * Returns a standard Tcl result. If an error occurred
- * TCL_ERROR is returned and interp->result will contain an
- * error message.
- *
- * -------------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-Print1Op(graphPtr, interp, argc, argv)
- Graph *graphPtr; /* Graph widget record */
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- int noBackingStore = 0;
- BITMAPINFO info;
- void *data;
- TkWinDCState state;
- TkWinBitmap bd;
- DIBSECTION ds;
- Drawable drawable;
- HBITMAP hBitmap;
- HDC hDC;
- DOCINFO di;
- double pageWidth, pageHeight;
- int result;
- double scale, sx, sy;
- int jobId;
-
- graphPtr->width = Tk_Width(graphPtr->tkwin);
- graphPtr->height = Tk_Height(graphPtr->tkwin);
- if ((graphPtr->width < 2) && (graphPtr->reqWidth > 0)) {
- graphPtr->width = graphPtr->reqWidth;
- }
- if ((graphPtr->height < 2) && (graphPtr->reqHeight > 0)) {
- graphPtr->height = graphPtr->reqHeight;
- }
- if (argc == 2) {
- result = Blt_PrintDialog(interp, &drawable);
- if (result == TCL_ERROR) {
- return TCL_ERROR;
- }
- if (result == TCL_RETURN) {
- return TCL_OK;
- }
- } else {
- if (Blt_GetOpenPrinter(interp, argv[2], &drawable) != TCL_OK) {
- return TCL_ERROR;
- }
- }
- /*
- * This is a taken from Blt_SnapPhoto. The difference is that
- * here we're using the DIBSection directly, without converting
- * the section into a ColorImage.
- */
- ZeroMemory(&info, sizeof(info));
- info.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
- info.bmiHeader.biWidth = graphPtr->width;
- info.bmiHeader.biHeight = graphPtr->height;
- info.bmiHeader.biPlanes = 1;
- info.bmiHeader.biBitCount = 32;
- info.bmiHeader.biCompression = BI_RGB;
- hDC = TkWinGetDrawableDC(graphPtr->display, Tk_WindowId(graphPtr->tkwin),
- &state);
- hBitmap = CreateDIBSection(hDC, &info, DIB_RGB_COLORS, &data, NULL, 0);
- TkWinReleaseDrawableDC(Tk_WindowId(graphPtr->tkwin), hDC, &state);
-
- /*
- * Create our own drawable by hand using the DIB we just created.
- * We'll then draw into it using the standard drawing functions.
- */
- bd.type = TWD_BITMAP;
- bd.handle = hBitmap;
- bd.colormap = DefaultColormap(graphPtr->display,
- DefaultScreen(graphPtr->display));
- bd.depth = Tk_Depth(graphPtr->tkwin);
-
- graphPtr->flags |= RESET_WORLD;
- Blt_DrawGraph(graphPtr, (Drawable)&bd, noBackingStore);
-
- /*
- * Now that the DIB contains the image of the graph, get the the
- * data bits and write them to the printer device, stretching the
- * image to the fit the printer's resolution.
- */
- result = TCL_ERROR;
- if (GetObject(hBitmap, sizeof(DIBSECTION), &ds) == 0) {
- Tcl_AppendResult(interp, "can't get object: ", Blt_LastError(),
- (char *)NULL);
- goto done;
- }
- hDC = ((TkWinDC *) drawable)->hdc;
- /* Get the resolution of the printer device. */
- sx = (double)GetDeviceCaps(hDC, HORZRES) / (double)graphPtr->width;
- sy = (double)GetDeviceCaps(hDC, VERTRES) / (double)graphPtr->height;
- scale = MIN(sx, sy);
- pageWidth = scale * graphPtr->width;
- pageHeight = scale * graphPtr->height;
-
- ZeroMemory(&di, sizeof(di));
- di.cbSize = sizeof(di);
- di.lpszDocName = "Graph Contents";
- jobId = StartDoc(hDC, &di);
- if (jobId <= 0) {
- Tcl_AppendResult(interp, "can't start document: ", Blt_LastError(),
- (char *)NULL);
- goto done;
- }
- if (StartPage(hDC) <= 0) {
- Tcl_AppendResult(interp, "error starting page: ", Blt_LastError(),
- (char *)NULL);
- goto done;
- }
- StretchDIBits(hDC, 0, 0, ROUND(pageWidth), ROUND(pageHeight), 0, 0,
- graphPtr->width, graphPtr->height, ds.dsBm.bmBits,
- (LPBITMAPINFO)&ds.dsBmih, DIB_RGB_COLORS, SRCCOPY);
- EndPage(hDC);
- EndDoc(hDC);
- result = TCL_OK;
- done:
- DeleteBitmap(hBitmap);
- return result;
-}
-
-/*
- * --------------------------------------------------------------------------
- *
- * Print2Op --
- *
- * Prints directly to the designated printer device.
- *
- * Results:
- * Returns a standard Tcl result. If an error occurred,
- * TCL_ERROR is returned and interp->result will contain an
- * error message.
- *
- * -------------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-Print2Op(graphPtr, interp, argc, argv)
- Graph *graphPtr; /* Graph widget record */
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- Drawable drawable;
- int noBackingStore = 0;
- int result;
-
- graphPtr->width = Tk_Width(graphPtr->tkwin);
- graphPtr->height = Tk_Height(graphPtr->tkwin);
- if ((graphPtr->width < 2) && (graphPtr->reqWidth > 0)) {
- graphPtr->width = graphPtr->reqWidth;
- }
- if ((graphPtr->height < 2) && (graphPtr->reqHeight > 0)) {
- graphPtr->height = graphPtr->reqHeight;
- }
- if (argc == 2) {
- result = Blt_PrintDialog(interp, &drawable);
- if (result == TCL_ERROR) {
- return TCL_ERROR;
- }
- if (result == TCL_RETURN) {
- return TCL_OK;
- }
- } else {
- result = Blt_GetOpenPrinter(interp, argv[2], &drawable);
- }
- if (result == TCL_OK) {
- int oldMode;
- HDC hDC;
- double xRatio, yRatio;
- TkWinDC *drawPtr;
- double vportWidth, vportHeight;
-
- drawPtr = (TkWinDC *) drawable;
- hDC = drawPtr->hdc;
- Blt_GetPrinterScale(hDC, &xRatio, &yRatio);
- oldMode = SetMapMode(hDC, MM_ISOTROPIC);
- if (oldMode == 0) {
- Tcl_AppendResult(interp, "can't set mode for printer DC: ",
- Blt_LastError(), (char *)NULL);
- return TCL_ERROR;
- }
- vportWidth = graphPtr->width * xRatio;
- vportHeight = graphPtr->height * yRatio;
- SetViewportExtEx(hDC, ROUND(vportWidth), ROUND(vportHeight), NULL);
- SetWindowExtEx(hDC, graphPtr->width, graphPtr->height, NULL);
-
- Blt_StartPrintJob(interp, drawable);
- graphPtr->flags |= RESET_WORLD;
- Blt_DrawGraph(graphPtr, drawable, noBackingStore);
- Blt_EndPrintJob(interp, drawable);
- }
- return result;
-}
-
-#endif /* NO_PRINTER */
-
-/*
- *----------------------------------------------------------------------
- *
- * StringToFormat --
- *
- * Convert a string represent a node number into its integer
- * value.
- *
- * Results:
- * The return value is a standard Tcl result.
- *
- *----------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-StringToFormat(clientData, interp, switchName, string, record, offset)
- ClientData clientData; /* Contains a pointer to the tabset containing
- * this image. */
- Tcl_Interp *interp; /* Interpreter to send results back to */
- char *switchName; /* Not used. */
- char *string; /* String representation */
- char *record; /* Structure record */
- int offset; /* Offset to field in structure */
-{
- int *formatPtr = (int *)(record + offset);
- char c;
-
- c = string[0];
- if ((c == 'p') && (strcmp(string, "photo") == 0)) {
- *formatPtr = FORMAT_PHOTO;
-#ifdef WIN32
- } else if ((c == 'e') && (strcmp(string, "emf") == 0)) {
- *formatPtr = FORMAT_EMF;
- } else if ((c == 'w') && (strcmp(string, "wmf") == 0)) {
- *formatPtr = FORMAT_WMF;
-#endif /* WIN32 */
- } else {
-#ifdef WIN32
- Tcl_AppendResult(interp, "bad format \"", string,
- "\": should be photo, emf, or wmf.", (char *)NULL);
-#else
- Tcl_AppendResult(interp, "bad format \"", string,
- "\": should be photo.", (char *)NULL);
-#endif /* WIN32 */
- return TCL_ERROR;
- }
- return TCL_OK;
-}
-
-#ifdef WIN32
-static int InitMetaFileHeader(
- Tk_Window tkwin,
- int width, int height,
- APMHEADER *mfhPtr)
-{
- unsigned int *p;
- unsigned int sum;
- Screen *screen;
-#define MM_INCH 25.4
- double dpiX, dpiY;
-
- mfhPtr->key = 0x9ac6cdd7L;
- mfhPtr->hmf = 0;
- mfhPtr->inch = 1440;
-
- screen = Tk_Screen(tkwin);
- dpiX = (WidthOfScreen(screen) * MM_INCH) / WidthMMOfScreen(screen);
- dpiY = (HeightOfScreen(screen) * MM_INCH) / HeightMMOfScreen(screen);
-
- mfhPtr->bbox.Left = mfhPtr->bbox.Top = 0;
- mfhPtr->bbox.Bottom = (SHORT)((width * 1440)/ dpiX);
- mfhPtr->bbox.Right = (SHORT)((height * 1440) / dpiY);
- mfhPtr->reserved = 0;
- sum = 0;
- for (p = (unsigned int *)mfhPtr;
- p < (unsigned int *)&(mfhPtr->checksum); p++) {
- sum ^= *p;
- }
- mfhPtr->checksum = sum;
- return TCL_OK;
-}
-
-static int
-CreateAPMetaFile(
- Tcl_Interp *interp,
- HANDLE hMetaFile,
- HDC hDC,
- APMHEADER *mfhPtr,
- char *fileName)
-{
- HANDLE hFile;
- HANDLE hMem;
- LPVOID buffer;
- int result;
- DWORD count, nBytes;
-
- result = TCL_ERROR;
- hMem = NULL;
- hFile = CreateFile(
- fileName, /* File path */
- GENERIC_WRITE, /* Access mode */
- 0, /* No sharing. */
- NULL, /* Security attributes */
- CREATE_ALWAYS, /* Overwrite any existing file */
- FILE_ATTRIBUTE_NORMAL,
- NULL); /* No template file */
- if (hFile == INVALID_HANDLE_VALUE) {
- Tcl_AppendResult(interp, "can't create metafile \"", fileName,
- "\":", Blt_LastError(), (char *)NULL);
- return TCL_ERROR;
- }
- if ((!WriteFile(hFile, (LPVOID)mfhPtr, sizeof(APMHEADER), &count,
- NULL)) || (count != sizeof(APMHEADER))) {
- Tcl_AppendResult(interp, "can't create metafile header to \"",
- fileName, "\":", Blt_LastError(), (char *)NULL);
- goto error;
- }
- nBytes = GetWinMetaFileBits(hMetaFile, 0, NULL, MM_ANISOTROPIC, hDC);
- hMem = GlobalAlloc(GHND, nBytes);
- if (hMem == NULL) {
- Tcl_AppendResult(interp, "can't create allocate global memory:",
- Blt_LastError(), (char *)NULL);
- goto error;
- }
- buffer = (LPVOID)GlobalLock(hMem);
- if (!GetWinMetaFileBits(hMetaFile, nBytes, buffer, MM_ANISOTROPIC, hDC)) {
- Tcl_AppendResult(interp, "can't get metafile bits:",
- Blt_LastError(), (char *)NULL);
- goto error;
- }
- if ((!WriteFile(hFile, buffer, nBytes, &count, NULL)) ||
- (count != nBytes)) {
- Tcl_AppendResult(interp, "can't write metafile bits:",
- Blt_LastError(), (char *)NULL);
- goto error;
- }
- result = TCL_OK;
- error:
- CloseHandle(hFile);
- if (hMem != NULL) {
- GlobalUnlock(hMem);
- GlobalFree(hMem);
- }
- return result;
-}
-#endif /*WIN32*/
-
-/*
- * --------------------------------------------------------------------------
- *
- * SnapOp --
- *
- * Snaps a picture of the graph and stores it in the specified image
- *
- * Results:
- * Returns a standard Tcl result. interp->result contains
- * the list of the graph coordinates. If an error occurred
- * while parsing the window positions, TCL_ERROR is returned,
- * then interp->result will contain an error message.
- *
- * -------------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-SnapOp(graphPtr, interp, argc, argv)
- Graph *graphPtr; /* Graph widget record */
- Tcl_Interp *interp;
- int argc; /* Not used. */
- char **argv;
-{
- int result;
- Pixmap drawable;
- int noBackingStore = 0;
- register int i;
- SnapData data;
-
- /* .g snap ?switches? name */
- data.height = Tk_Height(graphPtr->tkwin);
- data.width = Tk_Width(graphPtr->tkwin);
- data.format = FORMAT_PHOTO;
- /* Process switches */
- i = Blt_ProcessSwitches(interp, snapSwitches, argc - 2, argv + 2,
- (char *)&data, BLT_SWITCH_OBJV_PARTIAL);
- if (i < 0) {
- return TCL_ERROR;
- }
- i += 2;
- if (i >= argc) {
- Tcl_AppendResult(interp, "missing name argument: should be \"",
- argv[0], "snap ?switches? name\"", (char *)NULL);
- return TCL_ERROR;
- }
- data.name = argv[i];
- if (data.width < 2) {
- data.width = 400;
- }
- if (data.height < 2) {
- data.height = 400;
- }
- /* Always re-compute the layout of the graph before snapping the photo. */
- graphPtr->width = data.width;
- graphPtr->height = data.height;
- Blt_LayoutGraph(graphPtr);
-
- drawable = Tk_WindowId(graphPtr->tkwin);
- if (data.format == FORMAT_PHOTO) {
- drawable = Tk_GetPixmap(graphPtr->display, drawable, graphPtr->width,
- graphPtr->height, Tk_Depth(graphPtr->tkwin));
-#ifdef WIN32
- assert(drawable != None);
-#endif
- graphPtr->flags |= RESET_WORLD;
- Blt_DrawGraph(graphPtr, drawable, noBackingStore);
- result = Blt_SnapPhoto(interp, graphPtr->tkwin, drawable, 0, 0,
- data.width, data.height, data.width, data.height, data.name, 1.0);
- Tk_FreePixmap(graphPtr->display, drawable);
-#ifdef WIN32
- } else if ((data.format == FORMAT_WMF) || (data.format == FORMAT_EMF)) {
- TkWinDC drawableDC;
- TkWinDCState state;
- HDC hRefDC, hDC;
- HENHMETAFILE hMetaFile;
- Tcl_DString dString;
- char *title;
-
- hRefDC = TkWinGetDrawableDC(graphPtr->display, drawable, &state);
-
- Tcl_DStringInit(&dString);
- Tcl_DStringAppend(&dString, "BLT Graph ", -1);
- Tcl_DStringAppend(&dString, BLT_VERSION, -1);
- Tcl_DStringAppend(&dString, "\0", -1);
- Tcl_DStringAppend(&dString, Tk_PathName(graphPtr->tkwin), -1);
- Tcl_DStringAppend(&dString, "\0", -1);
- title = Tcl_DStringValue(&dString);
- hDC = CreateEnhMetaFile(hRefDC, NULL, NULL, title);
- Tcl_DStringFree(&dString);
-
- if (hDC == NULL) {
- Tcl_AppendResult(interp, "can't create metafile: ",
- Blt_LastError(), (char *)NULL);
- return TCL_ERROR;
- }
-
- drawableDC.hdc = hDC;
- drawableDC.type = TWD_WINDC;
-
- Blt_LayoutGraph(graphPtr);
- graphPtr->flags |= RESET_WORLD;
- Blt_DrawGraph(graphPtr, (Drawable)&drawableDC, FALSE);
-
- hMetaFile = CloseEnhMetaFile(hDC);
- if (strcmp(data.name, "CLIPBOARD") == 0) {
- HWND hWnd;
-
- hWnd = Tk_GetHWND(drawable);
- OpenClipboard(hWnd);
- EmptyClipboard();
- SetClipboardData(CF_ENHMETAFILE, hMetaFile);
- CloseClipboard();
- result = TCL_OK;
- } else {
- result = TCL_ERROR;
- if (data.format == FORMAT_WMF) {
- APMHEADER mfh;
-
- assert(sizeof(mfh) == 22);
- InitMetaFileHeader(graphPtr->tkwin, data.width, data.height,
- &mfh);
- result = CreateAPMetaFile(interp, hMetaFile, hRefDC, &mfh,
- data.name);
- } else {
- HENHMETAFILE hMetaFile2;
-
- hMetaFile2 = CopyEnhMetaFile(hMetaFile, data.name);
- if (hMetaFile2 != NULL) {
- result = TCL_OK;
- DeleteEnhMetaFile(hMetaFile2);
- }
- }
- DeleteEnhMetaFile(hMetaFile);
- }
- TkWinReleaseDrawableDC(drawable, hRefDC, &state);
-#endif /*WIN32*/
- } else {
- Tcl_AppendResult(interp, "bad snapshot format", (char *)NULL);
- return TCL_ERROR;
- }
- graphPtr->flags = MAP_WORLD;
- Blt_EventuallyRedrawGraph(graphPtr);
- return result;
-}
-
-/*
- * --------------------------------------------------------------------------
- *
- * GraphWidgetCmd --
- *
- * This procedure is invoked to process the Tcl command that
- * corresponds to a widget managed by this module. See the user
- * documentation for details on what it does.
- *
- * Results:
- * A standard Tcl result.
- *
- * Side effects:
- * See the user documentation.
- *
- * --------------------------------------------------------------------------
- */
-static Blt_OpSpec graphOps[] =
-{
- {"axis", 1, (Blt_Op)Blt_VirtualAxisOp, 2, 0, "oper ?args?",},
- {"bar", 2, (Blt_Op)BarOp, 2, 0, "oper ?args?",},
- {"cget", 2, (Blt_Op)CgetOp, 3, 3, "option",},
- {"configure", 2, (Blt_Op)ConfigureOp, 2, 0, "?option value?...",},
- {"crosshairs", 2, (Blt_Op)Blt_CrosshairsOp, 2, 0, "oper ?args?",},
- {"element", 2, (Blt_Op)ElementOp, 2, 0, "oper ?args?",},
- {"extents", 2, (Blt_Op)ExtentsOp, 3, 3, "item",},
- {"grid", 1, (Blt_Op)Blt_GridOp, 2, 0, "oper ?args?",},
- {"inside", 3, (Blt_Op)InsideOp, 4, 4, "winX winY",},
- {"invtransform", 3, (Blt_Op)InvtransformOp, 4, 4, "winX winY",},
- {"legend", 2, (Blt_Op)Blt_LegendOp, 2, 0, "oper ?args?",},
- {"line", 2, (Blt_Op)LineOp, 2, 0, "oper ?args?",},
- {"marker", 2, (Blt_Op)Blt_MarkerOp, 2, 0, "oper ?args?",},
- {"pen", 2, (Blt_Op)Blt_PenOp, 2, 0, "oper ?args?",},
- {"postscript", 2, (Blt_Op)Blt_PostScriptOp, 2, 0, "oper ?args?",},
-#ifndef NO_PRINTER
- {"print1", 2, (Blt_Op)Print1Op, 2, 3, "?printerName?",},
- {"print2", 2, (Blt_Op)Print2Op, 2, 3, "?printerName?",},
-#endif /*NO_PRINTER*/
- {"snap", 1, (Blt_Op)SnapOp, 3, 0, "?switches? name",},
- {"transform", 1, (Blt_Op)TransformOp, 4, 4, "x y",},
- {"x2axis", 2, (Blt_Op)X2AxisOp, 2, 0, "oper ?args?",},
- {"xaxis", 2, (Blt_Op)XAxisOp, 2, 0, "oper ?args?",},
- {"y2axis", 2, (Blt_Op)Y2AxisOp, 2, 0, "oper ?args?",},
- {"yaxis", 2, (Blt_Op)YAxisOp, 2, 0, "oper ?args?",},
-};
-static int nGraphOps = sizeof(graphOps) / sizeof(Blt_OpSpec);
-
-int
-Blt_GraphInstCmdProc(clientData, interp, argc, argv)
- ClientData clientData;
- Tcl_Interp *interp;
- int argc;
- char **argv;
-{
- Blt_Op proc;
- int result;
- Graph *graphPtr = (Graph *)clientData;
-
- proc = Blt_GetOp(interp, nGraphOps, graphOps, BLT_OP_ARG1, argc, argv, 0);
- if (proc == NULL) {
- return TCL_ERROR;
- }
- Tcl_Preserve(graphPtr);
- result = (*proc) (graphPtr, interp, argc, argv);
- Tcl_Release(graphPtr);
- return result;
-}
-
-/*
- * --------------------------------------------------------------------------
- *
- * NewGraph --
- *
- * Creates a new window and Tcl command representing an
- * instance of a graph widget.
- *
- * Results:
- * A standard Tcl result.
- *
- * Side effects:
- * See the user documentation.
- *
- * --------------------------------------------------------------------------
- */
-static int
-NewGraph(interp, argc, argv, classUid)
- Tcl_Interp *interp;
- int argc;
- char **argv;
- Blt_Uid classUid;
-{
- Graph *graphPtr;
-
- if (argc < 2) {
- Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
- " pathName ?option value?...\"", (char *)NULL);
- return TCL_ERROR;
- }
- graphPtr = CreateGraph(interp, argc, argv, classUid);
- if (graphPtr == NULL) {
- return TCL_ERROR;
- }
- Tcl_SetResult(interp, Tk_PathName(graphPtr->tkwin), TCL_VOLATILE);
- return TCL_OK;
-}
-
-/*
- * --------------------------------------------------------------------------
- *
- * GraphCmd --
- *
- * Creates a new window and Tcl command representing an
- * instance of a graph widget.
- *
- * Results:
- * A standard Tcl result.
- *
- * Side effects:
- * See the user documentation.
- *
- * --------------------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-GraphCmd(clientData, interp, argc, argv)
- ClientData clientData; /* Not used. */
- Tcl_Interp *interp;
- int argc;
- char **argv;
-{
- return NewGraph(interp, argc, argv, bltLineElementUid);
-}
-
-/*
- *--------------------------------------------------------------
- *
- * BarchartCmd --
- *
- * Creates a new window and Tcl command representing an
- * instance of a barchart widget.
- *
- * Results:
- * A standard Tcl result.
- *
- * Side effects:
- * See the user documentation.
- *
- *--------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-BarchartCmd(clientData, interp, argc, argv)
- ClientData clientData; /* Not used. */
- Tcl_Interp *interp;
- int argc;
- char **argv;
-{
- return NewGraph(interp, argc, argv, bltBarElementUid);
-}
-
-/*
- *--------------------------------------------------------------
- *
- * StripchartCmd --
- *
- * Creates a new window and Tcl command representing an
- * instance of a barchart widget.
- *
- * Results:
- * A standard Tcl result.
- *
- * Side effects:
- * See the user documentation.
- *
- *--------------------------------------------------------------
- */
-/*ARGSUSED*/
-static int
-StripchartCmd(clientData, interp, argc, argv)
- ClientData clientData; /* Not used. */
- Tcl_Interp *interp;
- int argc;
- char **argv;
-{
- return NewGraph(interp, argc, argv, bltStripElementUid);
-}
-
-/*
- * -----------------------------------------------------------------------
- *
- * DrawMargins --
- *
- * Draws the exterior region of the graph (axes, ticks, titles, etc)
- * onto a pixmap. The interior region is defined by the given
- * rectangle structure.
- *
- * ---------------------------------
- * | |
- * | rectArr[0] |
- * | |
- * ---------------------------------
- * | |top right| |
- * | | | |
- * | | | |
- * | [1] | | [2] |
- * | | | |
- * | | | |
- * | | | |
- * | | | |
- * | | | |
- * | |left bottom| |
- * ---------------------------------
- * | |
- * | rectArr[3] |
- * | |
- * ---------------------------------
- *
- * X coordinate axis
- * Y coordinate axis
- * legend
- * interior border
- * exterior border
- * titles (X and Y axis, graph)
- *
- * Returns:
- * None.
- *
- * Side Effects:
- * Exterior of graph is displayed in its window.
- *
- * -----------------------------------------------------------------------
- */
-static void
-DrawMargins(graphPtr, drawable)
- Graph *graphPtr;
- Drawable drawable; /* Pixmap or window to draw into */
-{
- XRectangle rects[4];
- /*
- * Draw the four outer rectangles which encompass the plotting
- * surface. This clears the surrounding area and clips the plot.
- */
- rects[0].x = rects[0].y = rects[3].x = rects[1].x = 0;
- rects[0].width = rects[3].width = (short int)graphPtr->width;
- rects[0].height = (short int)graphPtr->top;
- rects[3].y = graphPtr->bottom + 1;
- rects[3].height = graphPtr->height - graphPtr->bottom;
- rects[2].y = rects[1].y = graphPtr->top;
- rects[1].width = graphPtr->left;
- rects[2].height = rects[1].height = graphPtr->bottom - graphPtr->top + 1;
- rects[2].x = graphPtr->right + 1;
- rects[2].width = graphPtr->width - graphPtr->right;
-
- if (Blt_HasTile(graphPtr->tile)) {
- Blt_SetTileOrigin(graphPtr->tkwin, graphPtr->tile, 0, 0);
- Blt_TileRectangles(graphPtr->tkwin, drawable, graphPtr->tile, rects, 4);
- } else {
- XFillRectangles(graphPtr->display, drawable, graphPtr->fillGC, rects,
- 4);
- }
-
- /* Draw 3D border around the plotting area */
-
- if (graphPtr->plotBorderWidth > 0) {
- int x, y, width, height;
-
- x = graphPtr->left - graphPtr->plotBorderWidth;
- y = graphPtr->top - graphPtr->plotBorderWidth;
- width = (graphPtr->right - graphPtr->left + 1) +
- (2 * graphPtr->plotBorderWidth);
- height = (graphPtr->bottom - graphPtr->top + 1) +
- (2 * graphPtr->plotBorderWidth);
- Blt_Draw3DRectangle(graphPtr->tkwin, drawable, graphPtr->border, x, y,
- width, height, graphPtr->plotBorderWidth, graphPtr->plotRelief);
- }
- if (Blt_LegendSite(graphPtr->legend) & LEGEND_IN_MARGIN) {
- /* Legend is drawn on one of the graph margins */
- Blt_DrawLegend(graphPtr->legend, drawable);
- }
- if (graphPtr->title != NULL) {
- Blt_DrawText(graphPtr->tkwin, drawable, graphPtr->title,
- &graphPtr->titleTextStyle, graphPtr->titleX, graphPtr->titleY);
- }
- Blt_DrawAxes(graphPtr, drawable);
-
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * DrawPlotRegion --
- *
- * Draws the contents of the plotting area. This consists of
- * the elements, markers (draw under elements), axis limits,
- * grid lines, and possibly the legend. Typically, the output
- * will be cached into a backing store pixmap, so that redraws
- * can occur quickly.
- *
- * Results:
- * None.
- *
- *----------------------------------------------------------------------
- */
-static void
-DrawPlotRegion(graphPtr, drawable)
- Graph *graphPtr;
- Drawable drawable; /* Pixmap or window to draw into */
-{
- /* Clear the background of the plotting area. */
- XFillRectangle(graphPtr->display, drawable, graphPtr->plotFillGC,
- graphPtr->left, graphPtr->top, graphPtr->right - graphPtr->left + 1,
- graphPtr->bottom - graphPtr->top + 1);
-
- /* Draw the elements, markers, legend, and axis limits. */
-
- if (!graphPtr->gridPtr->hidden) {
- Blt_DrawGrid(graphPtr, drawable);
- }
- Blt_DrawMarkers(graphPtr, drawable, MARKER_UNDER);
- if ((Blt_LegendSite(graphPtr->legend) & LEGEND_IN_PLOT) &&
- (!Blt_LegendIsRaised(graphPtr->legend))) {
- Blt_DrawLegend(graphPtr->legend, drawable);
- }
- Blt_DrawAxisLimits(graphPtr, drawable);
- Blt_DrawElements(graphPtr, drawable);
-}
-
-void
-Blt_LayoutGraph(graphPtr)
- Graph *graphPtr;
-{
- if (graphPtr->flags & RESET_AXES) {
- Blt_ResetAxes(graphPtr);
- }
- if (graphPtr->flags & LAYOUT_NEEDED) {
- Blt_LayoutMargins(graphPtr);
- graphPtr->flags &= ~LAYOUT_NEEDED;
- }
- /* Compute coordinate transformations for graph components */
- if ((graphPtr->vRange > 1) && (graphPtr->hRange > 1)) {
- if (graphPtr->flags & MAP_WORLD) {
- Blt_MapAxes(graphPtr);
- }
- Blt_MapElements(graphPtr);
- Blt_MapMarkers(graphPtr);
- Blt_MapGrid(graphPtr);
- graphPtr->flags &= ~(MAP_ALL);
- }
-}
-
-void
-Blt_DrawGraph(graphPtr, drawable, backingStore)
- Graph *graphPtr;
- Drawable drawable; /* Pixmap or window to draw into */
- int backingStore; /* If non-zero, use backing store for
- * plotting area. */
-{
- if (backingStore) {
- /*
- * Create another pixmap to save elements if one doesn't
- * already exist or the size of the window has changed.
- */
- if ((graphPtr->backPixmap == None) ||
- (graphPtr->backWidth != graphPtr->width) ||
- (graphPtr->backHeight != graphPtr->height)) {
-
- if (graphPtr->backPixmap != None) {
- Tk_FreePixmap(graphPtr->display, graphPtr->backPixmap);
- }
- graphPtr->backPixmap = Tk_GetPixmap(graphPtr->display,
- Tk_WindowId(graphPtr->tkwin), graphPtr->width,
- graphPtr->height, Tk_Depth(graphPtr->tkwin));
- graphPtr->backWidth = graphPtr->width;
- graphPtr->backHeight = graphPtr->height;
- graphPtr->flags |= REDRAW_BACKING_STORE;
- }
- if (graphPtr->flags & REDRAW_BACKING_STORE) {
-
- /* The backing store is new or out-of-date. */
-
- DrawPlotRegion(graphPtr, graphPtr->backPixmap);
- graphPtr->flags &= ~REDRAW_BACKING_STORE;
- }
-
- /* Copy the pixmap to the one used for drawing the entire graph. */
-
- XCopyArea(graphPtr->display, graphPtr->backPixmap, drawable,
- graphPtr->drawGC, graphPtr->left, graphPtr->top,
- (graphPtr->right - graphPtr->left + 1),
- (graphPtr->bottom - graphPtr->top + 1),
- graphPtr->left, graphPtr->top);
- } else {
- DrawPlotRegion(graphPtr, drawable);
- }
-
- /* Draw markers above elements */
- Blt_DrawMarkers(graphPtr, drawable, MARKER_ABOVE);
- Blt_DrawActiveElements(graphPtr, drawable);
-
- if (graphPtr->flags & DRAW_MARGINS) {
- DrawMargins(graphPtr, drawable);
- }
- if (graphPtr->gridPtr->hidden == 0 && graphPtr->gridPtr->raised) {
- Blt_DrawGrid(graphPtr, drawable);
- }
- if ((Blt_LegendSite(graphPtr->legend) & LEGEND_IN_PLOT) &&
- (Blt_LegendIsRaised(graphPtr->legend))) {
- Blt_DrawLegend(graphPtr->legend, drawable);
- }
- /* Draw 3D border just inside of the focus highlight ring. */
- if ((graphPtr->borderWidth > 0) && (graphPtr->relief != TK_RELIEF_FLAT)) {
- Blt_Draw3DRectangle(graphPtr->tkwin, drawable, graphPtr->border,
- graphPtr->highlightWidth, graphPtr->highlightWidth,
- graphPtr->width - 2 * graphPtr->highlightWidth,
- graphPtr->height - 2 * graphPtr->highlightWidth,
- graphPtr->borderWidth, graphPtr->relief);
- }
- /* Draw focus highlight ring. */
- if ((graphPtr->highlightWidth > 0) && (graphPtr->flags & GRAPH_FOCUS)) {
- GC gc;
-
- gc = Tk_GCForColor(graphPtr->highlightColor, drawable);
- Tk_DrawFocusHighlight(graphPtr->tkwin, gc, graphPtr->highlightWidth,
- drawable);
- }
-}
-
-static void
-UpdateMarginTraces(graphPtr)
- Graph *graphPtr;
-{
- Margin *marginPtr;
- int size;
- register int i;
- char *oldVal, *newVal;
-
- for (i = 0; i < 4; i++) {
- marginPtr = graphPtr->margins + i;
- if (marginPtr->varName != NULL) { /* Trigger variable traces */
- if ((marginPtr->site == MARGIN_LEFT) ||
- (marginPtr->site == MARGIN_RIGHT)) {
- size = marginPtr->width;
- } else {
- size = marginPtr->height;
- }
- newVal = Blt_Itoa(size);
- oldVal = Tcl_GetVar(graphPtr->interp, marginPtr->varName,
- TCL_GLOBAL_ONLY);
- if (oldVal && !strcmp(oldVal,newVal)) continue;
- Tcl_SetVar(graphPtr->interp, marginPtr->varName, newVal,
- TCL_GLOBAL_ONLY);
- }
- }
-}
-
-/*
- *----------------------------------------------------------------------
- *
- * DisplayGraph --
- *
- * This procedure is invoked to display a graph widget.
- *
- * Results:
- * None.
- *
- * Side effects:
- * Commands are output to X to display the graph in its
- * current mode.
- *
- *----------------------------------------------------------------------
- */
-static void
-DisplayGraph(clientData)
- ClientData clientData;
-{
- Graph *graphPtr = (Graph *)clientData;
- Pixmap drawable;
-
- graphPtr->flags &= ~REDRAW_PENDING;
- if (graphPtr->tkwin == NULL) {
- return; /* Window destroyed (should not get here) */
- }
-#ifdef notdef
- fprintf(stderr, "Calling DisplayGraph(%s)\n", Tk_PathName(graphPtr->tkwin));
-#endif
- if (Blt_GraphUpdateNeeded(graphPtr)) {
- /*
- * One of the elements of the graph has a vector notification
- * pending. This means that the vector will eventually notify
- * the graph that its data has changed. Since the graph uses
- * the actual vector (not a copy) we need to keep in-sync.
- * Therefore don't draw right now but wait until we've been
- * notified before redrawing.
- */
- return;
- }
-
- /*
- Process the 'redrawcmd' callback as a REDRAW_BACKING_STORE
- event had been triggered.
-
- This callback can be used to generate a stream of postcript
- frames safely.
- */
- if ((graphPtr->flags & REDRAW_BACKING_STORE)
- && !(graphPtr->flags & EXEC_REDRAWCMD)
- && (graphPtr->redrawCmd != NULL) ) {
- Tcl_Interp *interp = graphPtr->interp;
- Tk_Window tkwin = graphPtr->tkwin;
-
- graphPtr->flags |= EXEC_REDRAWCMD;
- if (Tcl_VarEval(interp, graphPtr->redrawCmd, " ",
- Tk_PathName(tkwin), (char *)NULL) != TCL_OK) {
- Tcl_BackgroundError(interp);
- return; /* Error in after data changed proc */
- }
- }
-
-
- graphPtr->width = Tk_Width(graphPtr->tkwin);
- graphPtr->height = Tk_Height(graphPtr->tkwin);
- Blt_LayoutGraph(graphPtr);
- Blt_UpdateCrosshairs(graphPtr);
- if (!Tk_IsMapped(graphPtr->tkwin)) {
- /* The graph's window isn't displayed, so don't bother
- * drawing anything. By getting this far, we've at least
- * computed the coordinates of the graph's new layout. */
- return;
- }
-
- /* Disable crosshairs before redisplaying to the screen */
- Blt_DisableCrosshairs(graphPtr);
- /*
- * Create a pixmap the size of the window for double buffering.
- */
- if (graphPtr->doubleBuffer) {
- drawable = Tk_GetPixmap(graphPtr->display, Tk_WindowId(graphPtr->tkwin),
- graphPtr->width, graphPtr->height, Tk_Depth(graphPtr->tkwin));
- } else {
- drawable = Tk_WindowId(graphPtr->tkwin);
- }
-#ifdef WIN32
- assert(drawable != None);
-#endif
- Blt_DrawGraph(graphPtr, drawable,
- graphPtr->backingStore && graphPtr->doubleBuffer);
- if (graphPtr->flags & DRAW_MARGINS) {
- XCopyArea(graphPtr->display, drawable, Tk_WindowId(graphPtr->tkwin),
- graphPtr->drawGC, 0, 0, graphPtr->width, graphPtr->height, 0, 0);
- } else {
- XCopyArea(graphPtr->display, drawable, Tk_WindowId(graphPtr->tkwin),
- graphPtr->drawGC, graphPtr->left, graphPtr->top,
- (graphPtr->right - graphPtr->left + 1),
- (graphPtr->bottom - graphPtr->top + 1),
- graphPtr->left, graphPtr->top);
- }
- if (graphPtr->doubleBuffer) {
- Tk_FreePixmap(graphPtr->display, drawable);
- }
- graphPtr->flags &= ~EXEC_REDRAWCMD;
- Blt_EnableCrosshairs(graphPtr);
- graphPtr->flags &= ~RESET_WORLD;
- UpdateMarginTraces(graphPtr);
-}
-
-/*LINTLIBRARY*/
-int
-Blt_GraphInit(interp)
- Tcl_Interp *interp;
-{
- static Blt_CmdSpec cmdSpecs[] =
- {
- {"graph", GraphCmd,},
- {"barchart", BarchartCmd,},
- {"stripchart", StripchartCmd,},
- };
- bltBarElementUid = (Blt_Uid)Tk_GetUid("BarElement");
- bltLineElementUid = (Blt_Uid)Tk_GetUid("LineElement");
- bltStripElementUid = (Blt_Uid)Tk_GetUid("StripElement");
- bltContourElementUid = (Blt_Uid)Tk_GetUid("ContourElement");
-
- bltLineMarkerUid = (Blt_Uid)Tk_GetUid("LineMarker");
- bltBitmapMarkerUid = (Blt_Uid)Tk_GetUid("BitmapMarker");
- bltImageMarkerUid = (Blt_Uid)Tk_GetUid("ImageMarker");
- bltTextMarkerUid = (Blt_Uid)Tk_GetUid("TextMarker");
- bltPolygonMarkerUid = (Blt_Uid)Tk_GetUid("PolygonMarker");
- bltWindowMarkerUid = (Blt_Uid)Tk_GetUid("WindowMarker");
-
- bltXAxisUid = (Blt_Uid)Tk_GetUid("X");
- bltYAxisUid = (Blt_Uid)Tk_GetUid("Y");
-
- return Blt_InitCmds(interp, "blt", cmdSpecs, 3);
-}
-
-Graph *
-Blt_GetGraphFromWindowData(tkwin)
- Tk_Window tkwin;
-{
- Graph *graphPtr;
-
- while (tkwin != NULL) {
- graphPtr = (Graph *)Blt_GetWindowInstanceData(tkwin);
- if (graphPtr != NULL) {
- return graphPtr;
- }
- tkwin = Tk_Parent(tkwin);
- }
- return NULL;
-}
-
-int
-Blt_GraphType(graphPtr)
- Graph *graphPtr;
-{
- if (graphPtr->classUid == bltLineElementUid) {
- return GRAPH;
- } else if (graphPtr->classUid == bltBarElementUid) {
- return BARCHART;
- } else if (graphPtr->classUid == bltStripElementUid) {
- return STRIPCHART;
- }
- return 0;
-}