aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--generic/Item.c4363
1 files changed, 4363 insertions, 0 deletions
diff --git a/generic/Item.c b/generic/Item.c
new file mode 100644
index 0000000..a81b16c
--- /dev/null
+++ b/generic/Item.c
@@ -0,0 +1,4363 @@
+/*
+ * Item.c -- Implementation of items.
+ *
+ * Authors : Patrick Lecoanet.
+ * Creation date :
+ *
+ * $Id$
+ */
+
+/*
+ * Copyright (c) 1993 - 1999 CENA, Patrick Lecoanet --
+ *
+ * This code is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This code is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this code; if not, write to the Free
+ * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ *
+ */
+
+
+#include <limits.h> /* For INT_MAX */
+#include <sys/param.h>
+#include <malloc.h>
+#include <stdarg.h>
+#include <stdio.h>
+#include <string.h>
+
+#include "Item.h"
+#include "Types.h"
+#include "WidgetInfo.h"
+#include "Geo.h"
+#include "Draw.h"
+#include "MapInfo.h"
+#include "Image.h"
+#include "Color.h"
+#include "tkRadar.h"
+
+
+static const char rcsid[] = "$Id$";
+static const char compile_id[]="$Compile: " __FILE__ " " __DATE__ " " __TIME__ " $";
+
+
+#define FIELD_SENSITIVE_BIT 1
+#define FIELD_VISIBLE_BIT 2
+#define FILLED_BIT 4
+#define TEXT_ON_TOP_BIT 8
+#define CACHE_OK 16
+
+
+static RadarList item_classes = NULL;
+static RadarList item_stack = NULL;
+
+
+/*
+ * Field record.
+ */
+typedef struct _FieldStruct {
+ /* Public data */
+ RadarColor color;
+ RadarColor back_color;
+ RadarColor border_color;
+ Pixmap fill_pattern;
+ char *text;
+ char *image_name;
+ char *tile_name;
+ RadarFont font;
+ unsigned char flags;
+ Border border_edges;
+ RadarJustify alignment;
+ ReliefStyle relief;
+ int relief_thickness;
+ AutoAlign auto_alignment;
+
+ /* Private data */
+ RadarImage image;
+ RadarImage tile;
+ RadarColorGradient gradient;
+ short orig_x;
+ short orig_y;
+ short corner_x;
+ short corner_y;
+ FieldSet field_set;
+} FieldStruct, *Field;
+
+
+/*
+ * The -text, -image, -border, -relief, -visible and
+ * -filled attributes set the RADAR_COORDS_FLAG to update
+ * the leader that might protude if not clipped by the text.
+ */
+RadarAttrConfig field_attrs[] = {
+ { RADAR_CONFIG_JUSTIFY, "-alignment", NULL,
+ Tk_Offset(FieldStruct, alignment), 0, RADAR_DRAW_FLAG, False },
+ { RADAR_CONFIG_AUTO_JUSTIFY, "-autoalignment", NULL,
+ Tk_Offset(FieldStruct, auto_alignment), 0, RADAR_DRAW_FLAG, False },
+ { RADAR_CONFIG_COLOR, "-backcolor", NULL,
+ Tk_Offset(FieldStruct, back_color), 0,
+ RADAR_DRAW_FLAG|RADAR_BORDER_FLAG, False },
+ { RADAR_CONFIG_BORDER, "-border", NULL,
+ Tk_Offset(FieldStruct, border_edges), 0, RADAR_COORDS_FLAG, False },
+ { RADAR_CONFIG_COLOR, "-bordercolor", NULL,
+ Tk_Offset(FieldStruct, border_color), 0, RADAR_DRAW_FLAG, False },
+ { RADAR_CONFIG_COLOR, "-color", NULL,
+ Tk_Offset(FieldStruct, color), 0, RADAR_DRAW_FLAG, False },
+ { RADAR_CONFIG_BOOL, "-filled", NULL,
+ Tk_Offset(FieldStruct, flags), FILLED_BIT, RADAR_COORDS_FLAG, False },
+ { RADAR_CONFIG_PATTERN, "-fillpattern", NULL,
+ Tk_Offset(FieldStruct, fill_pattern), 0, RADAR_DRAW_FLAG, False },
+ { RADAR_CONFIG_FONT|RADAR_CLFC_FLAG, "-font", NULL,
+ Tk_Offset(FieldStruct, font), 0, RADAR_DRAW_FLAG, False },
+ { RADAR_CONFIG_IMAGE, "-image", NULL,
+ Tk_Offset(FieldStruct, image_name), 0,
+ RADAR_COORDS_FLAG|RADAR_IMAGE_FLAG|RADAR_CLFC_FLAG, False },
+ { RADAR_CONFIG_RELIEF, "-relief", NULL,
+ Tk_Offset(FieldStruct, relief), 0, RADAR_COORDS_FLAG, False },
+ { RADAR_CONFIG_DIM, "-reliefthickness", NULL,
+ Tk_Offset(FieldStruct, relief_thickness), 0, RADAR_DRAW_FLAG, False },
+ { RADAR_CONFIG_BOOL, "-sensitive", NULL,
+ Tk_Offset(FieldStruct, flags), FIELD_SENSITIVE_BIT, RADAR_REPICK_FLAG, False },
+ { RADAR_CONFIG_TEXT, "-text", NULL,
+ Tk_Offset(FieldStruct, text), 0, RADAR_COORDS_FLAG, False },
+ { RADAR_CONFIG_IMAGE, "-tile", NULL,
+ Tk_Offset(FieldStruct, tile_name), 0,
+ RADAR_COORDS_FLAG|RADAR_TILE_FLAG|RADAR_CLFC_FLAG, False },
+ { RADAR_CONFIG_BOOL, "-visible", NULL,
+ Tk_Offset(FieldStruct, flags), FIELD_VISIBLE_BIT,
+ RADAR_COORDS_FLAG|RADAR_CLFC_FLAG, False }, /* Keep RADAR_COORDS_FLAG here */
+
+ { RADAR_CONFIG_END, NULL, NULL, 0, 0, 0 }
+};
+
+/*
+ * This array must be kept in sync with the
+ * corresponding defines in Types.h.
+ */
+static char *attribute_type_strings[] = {
+ "",
+ "color",
+ "boolean",
+ "pattern",
+ "text",
+ "font",
+ "border",
+ "relief",
+ "dimension",
+ "priority",
+ "justify",
+ "autojustify",
+ "lineend",
+ "labelformat",
+ "linestyle",
+ "lineshape",
+ "item",
+ "angle",
+ "integer",
+ "unsignedint",
+ "point",
+ "rectangle",
+ "patterns",
+ "anchor",
+ "tags",
+ "points",
+ "mapinfo",
+ "image",
+ "leaderanchors",
+ "JoinStyle",
+ "CapStyle"
+};
+
+
+/*
+ **********************************************************************************
+ *
+ * Forward functions
+ *
+ **********************************************************************************
+ */
+static void Damage(WidgetInfo *wi, RadarBBox *damage);
+static void Invalidate(Item item, int reason);
+static int ConfigureField(FieldSet field_set, unsigned int field,
+ int argc, RadarAttrList argv, int *flags);
+static int QueryField(FieldSet field_set, unsigned int field,
+ int argc, RadarAttrList argv);
+static Arg AttributeToString(WidgetInfo *wi, char *record, RadarAttrConfig *desc,
+ char *buffer, Tcl_FreeProc **free_proc);
+static void FieldImageChange(ClientData client_data, int x, int y, int width,
+ int height, int image_width, int image_height);
+static void FieldTileChange(ClientData client_data, int x, int y, int width,
+ int height, int image_width, int image_height);
+
+
+/*
+ **********************************************************************************
+ *
+ * InitAttrDesc --
+ *
+ **********************************************************************************
+ */
+static void
+InitAttrDesc(RadarAttrConfig *attr_desc)
+{
+ if (!attr_desc) {
+ return;
+ }
+
+ while (attr_desc->type != RADAR_CONFIG_END) {
+ attr_desc->uid = Tk_GetUid(attr_desc->name);
+ attr_desc++;
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * AttributesInfo --
+ *
+ **********************************************************************************
+ */
+static int
+AttributesInfo(Item item,
+ int field, /* 0< means the item itself. */
+ int argc,
+ Arg *args)
+{
+ WidgetInfo *wi = item->wi;
+ char *record;
+ RadarAttrConfig *desc;
+ Tk_Uid attr_uid = NULL;
+#ifndef PTK
+ Arg entries[5];
+#else
+ Arg *entries;
+#endif
+ Arg result;
+ char buffer[256];
+ Tcl_FreeProc *free_proc;
+
+ if (field < 0) {
+ record = (char *) item;
+ desc = item->class->attr_desc;
+ }
+ else {
+ FieldSet field_set;
+ if (!item->class->has_fields) {
+ Tcl_AppendResult(wi->interp, "item class \"", item->class->name,
+ "\" doesn't support fields", NULL);
+ return RADAR_ERROR;
+ }
+ field_set = item->class->GetFieldSet(item);
+ if (field >= field_set->num_fields) {
+ Tcl_AppendResult(wi->interp, "invalid field index \"", NULL);
+ return RADAR_ERROR;
+ }
+ record = (char *) &field_set->fields[field];
+ desc = field_attrs;
+ }
+
+ if (argc == 1) {
+ attr_uid = Tk_GetUid(LangString(args[0]));
+
+ while (True) {
+ if (desc->type == RADAR_CONFIG_END) {
+ Tcl_AppendResult(wi->interp, "unknown attribute \"",
+ LangString(args[0]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ else if (attr_uid == desc->uid) {
+ break;
+ }
+ else {
+ desc++;
+ }
+ }
+#ifdef PTK
+ entries = LangAllocVec(5);
+ LangSetInt(&entries[2], desc->read_only ? 1 : 0);
+#else
+ entries[2] = desc->read_only ? "1" : "0";
+#endif
+ LangSetString(&entries[0], desc->name);
+ LangSetString(&entries[1], attribute_type_strings[desc->type]);
+ LangSetString(&entries[3], "");
+ LangSetArg(&entries[4], AttributeToString(wi, record, desc, buffer, &free_proc));
+ result = Tcl_Merge(5, entries);
+#ifndef PTK
+ if (free_proc == TCL_DYNAMIC) {
+ RadarFree(entries[4]);
+ }
+ Tcl_SetResult(wi->interp, result, TCL_DYNAMIC);
+#else
+ LangFreeVec(5, entries);
+ Tcl_ArgResult(wi->interp, result);
+#endif
+ }
+ else {
+ while (desc->type != RADAR_CONFIG_END) {
+#ifdef PTK
+ entries = LangAllocVec(5);
+ LangSetInt(&entries[2], desc->read_only ? 1 : 0);
+#else
+ entries[2] = desc->read_only ? "1" : "0";
+#endif
+ LangSetString(&entries[0], desc->name);
+ LangSetString(&entries[1], attribute_type_strings[desc->type]);
+ LangSetString(&entries[3], "");
+ LangSetArg(&entries[4], AttributeToString(wi, record, desc, buffer, &free_proc));
+ result = Tcl_Merge(5, entries);
+#ifndef PTK
+ if (free_proc == TCL_DYNAMIC) {
+ RadarFree(entries[4]);
+ }
+ Tcl_AppendElement(wi->interp, result);
+ RadarFree(result);
+#else
+ LangFreeVec(5, entries);
+ Tcl_AppendArg(wi->interp, result);
+#endif
+ desc++;
+ }
+ }
+
+ return TCL_OK;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ConfigureAttributes --
+ *
+ **********************************************************************************
+ */
+static int
+ConfigureAttributes(char *record,
+ int field, /* 0< means item itself. */
+ int argc,
+ Arg *args,
+ int *flags)
+{
+ WidgetInfo *wi;
+ Item item = NULL;
+ int i;
+ Tk_Uid attr_uid;
+ RadarAttrConfig *desc;
+ RadarPtr valp;
+ RadarAttrConfig *attr_desc;
+ FieldSet field_set = NULL;
+
+
+ if (field < 0) {
+ item = (Item) record;
+ wi = item->wi;
+ attr_desc = item->class->attr_desc;
+ }
+ else {
+ field_set = (FieldSet) record;
+ wi = field_set->wi;
+ if (field >= field_set->num_fields) {
+ Tcl_AppendResult(wi->interp, "invalid field index \"", NULL);
+ return RADAR_ERROR;
+ }
+ record = (char *) &field_set->fields[field];
+ /*printf("record <0x%X>, field %d\n", record, field);*/
+ attr_desc = field_attrs;
+ }
+
+ for (i = 0; i < argc; i += 2) {
+ attr_uid = Tk_GetUid(LangString(args[i]));
+
+ desc = attr_desc;
+ while (True) {
+ if (desc->type == RADAR_CONFIG_END) {
+ /*printf("ERROR: record <0x%X>\n", record);*/
+ Tcl_AppendResult(wi->interp, "unknown attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ else if (attr_uid == desc->uid) {
+ if (desc->read_only) {
+ Tcl_AppendResult(wi->interp, "attribute \"",
+ LangString(args[i]), "\" can only be read", NULL);
+ return RADAR_ERROR;
+ }
+ valp = record + desc->offset;
+ /*printf("record <0x%X>, valp <0x%X>, offset %d\n", record, valp, desc->offset);*/
+ switch (desc->type) {
+ case RADAR_CONFIG_COLOR:
+ {
+ XColor *color;
+ Tk_Uid new_name = Tk_GetUid(LangString(args[i+1]));
+ char *name = NULL;
+ if (*((XColor **) valp)) {
+ name = RadarNameOfColor(*((XColor **) valp));
+ }
+ if (name != new_name) {
+ color = RadarGetColor(wi->interp, wi->win, new_name);
+ if (!color) {
+ Tcl_AppendResult(wi->interp, " color expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (*((XColor **) valp)) {
+ RadarFreeColor(*((XColor **) valp));
+ }
+ *((XColor **) valp) = color;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_BOOL:
+ {
+ int b;
+ if (Tcl_GetBoolean(wi->interp, args[i+1], &b) != RADAR_OK) {
+ Tcl_AppendResult(wi->interp, " boolean expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (b ^ (ISSET(*((char *) valp), desc->bool_bit) != 0)) {
+ ASSIGN(*((char *) valp), desc->bool_bit, b);
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_PATTERN:
+ {
+ Pixmap pattern = RadarUnspecifiedPattern;
+ char *name = "";
+ if (*((Pixmap *) valp) != RadarUnspecifiedPattern) {
+ name = Tk_NameOfBitmap(wi->dpy, *((Pixmap *) valp));
+ }
+ if (strcmp(name, LangString(args[i+1])) != 0) {
+ if (strlen(LangString(args[i+1])) != 0) {
+ pattern = Tk_GetBitmap(wi->interp, wi->win, Tk_GetUid(LangString(args[i+1])));
+ if (pattern == None) {
+ Tcl_AppendResult(wi->interp, " pattern expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ }
+ if (*((Pixmap *) valp) != RadarUnspecifiedPattern) {
+ Tk_FreeBitmap(wi->dpy, *((Pixmap *) valp));
+ }
+ *((Pixmap *) valp) = pattern;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_PATTERNS:
+ {
+ RadarList new_pat_list = NULL;
+ Pixmap *pats;
+ int num_pats, result, j;
+ Arg *elems;
+#ifdef PTK
+ LangFreeProc *freeProc = NULL;
+#endif
+
+ if (strlen(LangString(args[i+1])) != 0) {
+ result = Lang_SplitList(wi->interp, args[i+1], &num_pats, &elems, &freeProc);
+ if (result == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp,
+ " pattern list expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+#ifdef PTK
+ if (elems != NULL && freeProc) {
+ (*freeProc)(num_pats, elems);
+ }
+#endif
+ return RADAR_ERROR;
+ }
+ if (num_pats) {
+ new_pat_list = RadarListNew(num_pats, sizeof(Pixmap));
+ RadarListAssertSize(new_pat_list, num_pats);
+ pats = (Pixmap *) RadarListArray(new_pat_list);
+ for (j = 0; j < num_pats; j++) {
+ if (strlen(LangString(elems[j])) != 0) {
+ pats[j] = Tk_GetBitmap(wi->interp, wi->win,
+ Tk_GetUid(LangString(elems[j])));
+ if (pats[j] == None) {
+ Tcl_AppendResult(wi->interp,
+ " unknown pattern \"", LangString(elems[j]),
+ "\" in pattern list", NULL);
+ RadarListFree(new_pat_list);
+#ifndef PTK
+ RadarFree(elems);
+#else
+ if (freeProc) {
+ (*freeProc)(num_pats, elems);
+ }
+#endif
+ return RADAR_ERROR;
+ }
+ }
+ else {
+ pats[j] = RadarUnspecifiedPattern;
+ }
+ }
+ }
+ RadarFree(elems);
+ }
+ if (*((RadarList *) valp)) {
+ num_pats = RadarListSize(*((RadarList *) valp));
+ pats = (Pixmap *) RadarListArray(*((RadarList *) valp));
+ for (j = 0; j < num_pats; j++) {
+ if (pats[j] != RadarUnspecifiedPattern) {
+ Tk_FreeBitmap(wi->dpy, pats[j]);
+ }
+ }
+ RadarListFree(*((RadarList *) valp));
+ *((RadarList *) valp) = new_pat_list;
+ *flags |= desc->flags;
+ }
+ else {
+ if (new_pat_list) {
+ *((RadarList *) valp) = new_pat_list;
+ *flags |= desc->flags;
+ }
+ }
+ break;
+ }
+ case RADAR_CONFIG_TAGS:
+ {
+ int num_tags, result, j;
+ Arg *elems;
+#ifdef PTK
+ LangFreeProc *freeProc = NULL;
+#endif
+
+ if (strlen(LangString(args[i+1])) != 0) {
+ result = Lang_SplitList(wi->interp, args[i+1], &num_tags, &elems, &freeProc);
+ if (result == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp,
+ " tag list expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+#ifdef PTK
+ if (elems != NULL && freeProc) {
+ (*freeProc)(num_tags, elems);
+ }
+#endif
+ return RADAR_ERROR;
+ }
+ if (*((RadarList *) valp)) {
+ ITEM.FreeTags(item);
+ *flags |= desc->flags;
+ }
+ if (num_tags) {
+ for (j = 0; j < num_tags; j++) {
+ ITEM.AddTag(item, Tk_GetUid(LangString(elems[j])));
+ }
+ *flags |= desc->flags;
+ }
+#ifndef PTK
+ RadarFree(elems);
+#else
+ if (freeProc) {
+ (*freeProc)(num_tags, elems);
+ }
+#endif
+ }
+ break;
+ }
+ case RADAR_CONFIG_TEXT:
+ case RADAR_CONFIG_MAP_INFO:
+ case RADAR_CONFIG_IMAGE:
+ {
+ char *text = "";
+ if (strcmp(LangString(args[i+1]), *((char **) valp)) != 0) {
+ if (strlen(LangString(args[i+1])) != 0) {
+ text = (char *) RadarMalloc(strlen(LangString(args[i+1]))+1);
+ strcpy(text, LangString(args[i+1]));
+ }
+ if (strlen(*((char **) valp)) != 0) {
+ RadarFree(*((char **) valp));
+ }
+ *((char **) valp) = text;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_FONT:
+ {
+ Tk_Font font;
+ char *name = "";
+ if (*((Tk_Font *) valp)) {
+ name = Tk_NameOfFont(*((Tk_Font *) valp));
+ }
+ if (strcmp(name, LangString(args[i+1])) != 0) {
+ font = Tk_GetFont(wi->interp, wi->win, LangString(args[i+1]));
+ if (!font) {
+ Tcl_AppendResult(wi->interp, " font expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (*((Tk_Font *) valp)) {
+ Tk_FreeFont(*((Tk_Font *) valp));
+ }
+ *((Tk_Font *) valp) = font;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_BORDER:
+ {
+ Border border = NO_BORDER;
+ int j, len, largc, result;
+ Arg *largv;
+#ifdef PTK
+ LangFreeProc *freeProc = NULL;
+#endif
+
+ result = Lang_SplitList(wi->interp, args[i+1], &largc, &largv, &freeProc);
+ if (result == RADAR_ERROR) {
+ border_error:
+ Tcl_AppendResult(wi->interp, " border expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+#ifdef PTK
+ if (largv != NULL && freeProc) {
+ (*freeProc)(largc, largv);
+ }
+#endif
+ return RADAR_ERROR;
+ }
+ len = strlen(LangString(args[i+1]));
+ for (j = 0; j < largc; j++) {
+ if (strncasecmp(LangString(largv[j]), LEFT_SPEC, len) == 0) {
+ border |= LEFT_BORDER;
+ }
+ else if (strncasecmp(LangString(largv[j]), RIGHT_SPEC, len) == 0) {
+ border |= RIGHT_BORDER;
+ }
+ else if (strncasecmp(LangString(largv[j]), TOP_SPEC, len) == 0) {
+ border |= TOP_BORDER;
+ }
+ else if (strncasecmp(LangString(largv[j]), BOTTOM_SPEC, len) == 0) {
+ border |= BOTTOM_BORDER;
+ }
+ else if (strncasecmp(LangString(largv[j]), CONTOUR_SPEC, len) == 0) {
+ border |= CONTOUR_BORDER;
+ }
+ else if (strncasecmp(LangString(largv[j]), OBLIQUE_SPEC, len) == 0) {
+ border |= OBLIQUE;
+ }
+ else if (strncasecmp(LangString(largv[j]), COUNTER_OBLIQUE_SPEC, len) == 0) {
+ border |= COUNTER_OBLIQUE;
+ }
+ else if (strncasecmp(LangString(largv[j]), NO_BORDER_SPEC, len) == 0) {
+ border |= NO_BORDER;
+ }
+ else {
+#ifndef PTK
+ RadarFree(largv);
+#endif
+ goto border_error;
+ }
+ }
+#ifndef PTK
+ RadarFree(largv);
+#else
+ if (largv != NULL && freeProc) {
+ (*freeProc)(largc, largv);
+ }
+#endif
+ if (border != *((Border *) valp)) {
+ *((Border *) valp) = border;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_LINE_SHAPE:
+ {
+ LineShape line_shape;
+ int len;
+ len = strlen(LangString(args[i+1]));
+ if (strncasecmp(LangString(args[i+1]), STRAIGHT_SPEC, len) == 0) {
+ line_shape = LINE_STRAIGHT;
+ }
+ else if (strncasecmp(LangString(args[i+1]), RIGHT_LIGHTNING_SPEC, len) == 0) {
+ line_shape = LINE_RIGHT_LIGHTNING;
+ }
+ else if (strncasecmp(LangString(args[i+1]), LEFT_LIGHTNING_SPEC, len) == 0) {
+ line_shape = LINE_LEFT_LIGHTNING;
+ }
+ else if (strncasecmp(LangString(args[i+1]), RIGHT_CORNER_SPEC, len) == 0) {
+ line_shape = LINE_LEFT_CORNER;
+ }
+ else if (strncasecmp(LangString(args[i+1]), LEFT_CORNER_SPEC, len) == 0) {
+ line_shape = LINE_LEFT_CORNER;
+ }
+ else if (strncasecmp(LangString(args[i+1]), DOUBLE_RIGHT_CORNER_SPEC, len) == 0) {
+ line_shape = LINE_DOUBLE_LEFT_CORNER;
+ }
+ else if (strncasecmp(LangString(args[i+1]), DOUBLE_LEFT_CORNER_SPEC, len) == 0) {
+ line_shape = LINE_DOUBLE_LEFT_CORNER;
+ }
+ else {
+ Tcl_AppendResult(wi->interp, " line shape expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (line_shape != *((LineShape *) valp)) {
+ *((LineShape *) valp) = line_shape;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_LINE_STYLE:
+ {
+ LineStyle line_style;
+ int len;
+ len = strlen(LangString(args[i+1]));
+ if (strncasecmp(LangString(args[i+1]), SIMPLE_SPEC, len) == 0)
+ line_style = LINE_SIMPLE;
+ else if (strncasecmp(LangString(args[i+1]), DASHED_SPEC, len) == 0)
+ line_style = LINE_DASHED;
+ else if (strncasecmp(LangString(args[i+1]), MIXED_SPEC, len) == 0)
+ line_style = LINE_MIXED;
+ else if (strncasecmp(LangString(args[i+1]), DOTTED_SPEC, len) == 0)
+ line_style = LINE_DOTTED;
+ else {
+ Tcl_AppendResult(wi->interp, " line style expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (line_style != *((LineStyle *) valp)) {
+ *((LineStyle *) valp) = line_style;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_LINE_END:
+ {
+ LineEnd line_end = NULL;
+ char *ptr = LangString(args[i+1]);
+ if (strlen(ptr) != 0) {
+ line_end = LineEndCreate(wi->interp, LangString(args[i+1]));
+ if (line_end == NULL) {
+ return RADAR_ERROR;
+ }
+ }
+ if (*((LineEnd *) valp) != NULL) {
+ LineEndDelete(*((LineEnd *) valp));
+ *((LineEnd *) valp) = line_end;
+ *flags |= desc->flags;
+ }
+ else {
+ if (line_end != NULL) {
+ *((LineEnd *) valp) = line_end;
+ *flags |= desc->flags;
+ }
+ }
+ break;
+ }
+ case RADAR_CONFIG_RELIEF:
+ {
+ int relief;
+ if (Tk_GetRelief(wi->interp, LangString(args[i+1]), &relief) == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp, " relief expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (relief != *((ReliefStyle *) valp)) {
+ /*printf("valp <0x%X>, flags <0x%X>, relief %d\n", valp, flags, relief);*/
+ *((ReliefStyle *) valp) = relief;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_JOIN_STYLE:
+ {
+ int join;
+ if (Tk_GetJoinStyle(wi->interp, LangString(args[i+1]), &join) == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp, " join expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (join != *((int *) valp)) {
+ *((int *) valp) = join;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_CAP_STYLE:
+ {
+ int cap;
+ if (Tk_GetCapStyle(wi->interp, LangString(args[i+1]), &cap) == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp, " cap expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (cap != *((int *) valp)) {
+ *((int *) valp) = cap;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_POINT:
+ {
+ RadarPoint point;
+ int largc, result;
+ Arg *largv;
+#ifdef PTK
+ LangFreeProc *freeProc = NULL;
+#endif
+
+ result = Lang_SplitList(wi->interp, args[i+1], &largc, &largv, &freeProc);
+ if (result == RADAR_ERROR || largc != 2) {
+ point_error:
+#ifdef PTK
+ if (largv != NULL && freeProc) {
+ (*freeProc)(largc, largv);
+ }
+#endif
+ Tcl_AppendResult(wi->interp, " position expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if ((Tcl_GetDouble(wi->interp, largv[0], &point.x) == RADAR_ERROR) ||
+ (Tcl_GetDouble(wi->interp, largv[1], &point.y) == RADAR_ERROR)) {
+#ifndef PTK
+ RadarFree(largv);
+#endif
+ goto point_error;
+ }
+#ifndef PTK
+ RadarFree(largv);
+#else
+ if (largv != NULL && freeProc) {
+ (*freeProc)(largc, largv);
+ }
+#endif
+ if ((point.x != ((RadarPoint *) valp)->x) ||
+ (point.y != ((RadarPoint *) valp)->y)) {
+ *((RadarPoint *) valp) = point;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_RECT:
+ {
+ RadarRect rect;
+ int largc, result;
+ Arg *largv;
+#ifdef PTK
+ LangFreeProc *freeProc = NULL;
+#endif
+
+ result = Lang_SplitList(wi->interp, args[i+1], &largc, &largv, &freeProc);
+ if (result == RADAR_ERROR || largc != 4) {
+ rect_error:
+#ifdef PTK
+ if (largv != NULL && freeProc) {
+ (*freeProc)(largc, largv);
+ }
+#endif
+ Tcl_AppendResult(wi->interp, " rectangle expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if ((Tcl_GetDouble(wi->interp, largv[0], &rect.x) == RADAR_ERROR) ||
+ (Tcl_GetDouble(wi->interp, largv[1], &rect.y) == RADAR_ERROR) ||
+ (Tcl_GetDouble(wi->interp, largv[2], &rect.w) == RADAR_ERROR) ||
+ (Tcl_GetDouble(wi->interp, largv[3], &rect.h) == RADAR_ERROR)) {
+#ifndef PTK
+ RadarFree(largv);
+#endif
+ goto rect_error;
+ }
+#ifndef PTK
+ RadarFree(largv);
+#else
+ if (largv != NULL && freeProc) {
+ (*freeProc)(largc, largv);
+ }
+#endif
+ if ((rect.x != ((RadarRect *) valp)->x) &&
+ (rect.y != ((RadarRect *) valp)->y) &&
+ (rect.w != ((RadarRect *) valp)->w) &&
+ (rect.h != ((RadarRect *) valp)->h)) {
+ *((RadarRect *) valp) = rect;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_DIM:
+ {
+ int size;
+ if (Tk_GetPixels(wi->interp, wi->win, LangString(args[i+1]), &size) == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp, " dimension expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (size != *((int *) valp)) {
+ *((int *) valp) = size;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_PRI:
+ {
+ int pri;
+ if (Tcl_GetInt(wi->interp, args[i+1], &pri) == RADAR_ERROR) {
+ return RADAR_ERROR;
+ }
+ if (pri < 0) {
+ Tcl_AppendResult(wi->interp, " priority must be a positive integer \"",
+ LangString(args[i+1]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (pri != *((int *) valp)) {
+ *((int *) valp) = pri;
+ ITEM.UpdateItemPriority((Item) record, RADAR_NO_ITEM, True);
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_ITEM:
+ /*
+ * Can be an item id or a tag. In this last case
+ * consider only the first item (unspecified order)
+ * associated with the tag.
+ */
+ {
+ Item item2;
+ int num;
+ if (strlen(LangString(args[i+1])) == 0) {
+ item2 = RADAR_NO_ITEM;
+ }
+ else {
+ num = RadarItemsWithTagOrId(wi, LangString(args[i+1]), &item2, NULL);
+ if (num == 0) {
+ return RADAR_ERROR;
+ }
+ }
+ if (item2 != *((Item *) valp)) {
+ *((Item *) valp) = item2;
+ *flags |= desc->flags;
+ }
+ }
+ break;
+ case RADAR_CONFIG_INT:
+ case RADAR_CONFIG_UINT:
+ case RADAR_CONFIG_ANGLE:
+ {
+ int integer;
+ if (Tcl_GetInt(wi->interp, args[i+1], &integer) == RADAR_ERROR) {
+ return RADAR_ERROR;
+ }
+ if (desc->type == RADAR_CONFIG_ANGLE) {
+ if ((integer < 0) || (integer > 360)) {
+ Tcl_AppendResult(wi->interp, " angle must be between 0 and 360 \"",
+ LangString(args[i+1]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ }
+ else if (desc->type == RADAR_CONFIG_UINT) {
+ if (integer < 0) {
+ Tcl_AppendResult(wi->interp, " positive integer expected for \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ }
+ if (integer != *((int *) valp)) {
+ *((int *) valp) = integer;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_JUSTIFY:
+ {
+ Tk_Justify justify;
+ if (Tk_GetJustify(wi->interp, LangString(args[i+1]), &justify) == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp, " justify expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (justify != *((RadarJustify *) valp)) {
+ *((RadarJustify *) valp) = justify;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_ANCHOR:
+ {
+ Tk_Anchor anchor;
+ if (Tk_GetAnchor(wi->interp, LangString(args[i+1]), &anchor) == RADAR_ERROR) {
+ Tcl_AppendResult(wi->interp, " anchor expected for attribute \"",
+ LangString(args[i]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ if (anchor != *((RadarAnchor *) valp)) {
+ *((RadarAnchor *) valp) = anchor;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+ case RADAR_CONFIG_LABEL_FORMAT:
+ {
+ LabelFormat frmt = NULL;
+ char *ptr = LangString(args[i+1]);
+
+ while (*ptr && (*ptr == ' ')) {
+ ptr++;
+ }
+ if (strlen(ptr) != 0) {
+ frmt = LabelFormatCreate(wi->interp, ptr,
+ item->class->GetFieldSet(item)->num_fields);
+ if (frmt == NULL) {
+ return RADAR_ERROR;
+ }
+ }
+
+ if (*((LabelFormat *) valp) != NULL) {
+ LabelFormatDelete(*((LabelFormat *) valp));
+ *((LabelFormat *) valp) = frmt;
+ *flags |= desc->flags;
+ }
+ else {
+ if (frmt != NULL) {
+ *((LabelFormat *) valp) = frmt;
+ *flags |= desc->flags;
+ }
+ }
+ break;
+ }
+
+ case RADAR_CONFIG_AUTO_JUSTIFY:
+ {
+ AutoAlign aa;
+ int j;
+ if (strcmp(LangString(args[i+1]), "-") == 0) {
+ aa.automatic = False;
+ }
+ else if (strlen(LangString(args[i+1])) == 3) {
+ aa.automatic = True;
+ for (j = 0; j < 3; j++) {
+ switch(LangString(args[i+1])[j]) {
+ case 'l':
+ case 'L':
+ aa.align[j] = RadarJustifyLeft;
+ break;
+ case 'c':
+ case 'C':
+ aa.align[j] = RadarJustifyCenter;
+ break;
+ case 'r':
+ case 'R':
+ aa.align[j] = RadarJustifyRight;
+ break;
+ default:
+ Tcl_AppendResult(wi->interp, "invalid auto justify specifcation \"",
+ LangString(args[i+1]), "\" should be - or a triple of lcr", NULL);
+ return RADAR_ERROR;
+ }
+ }
+ }
+ else {
+ Tcl_AppendResult(wi->interp, "invalid auto alignment specification \"",
+ LangString(args[i+1]), "\" should be - or a triple of lcr", NULL);
+ return RADAR_ERROR;
+ }
+ if ((aa.automatic != ((AutoAlign *) valp)->automatic) ||
+ (aa.align[0] != ((AutoAlign *) valp)->align[0]) ||
+ (aa.align[1] != ((AutoAlign *) valp)->align[1]) ||
+ (aa.align[2] != ((AutoAlign *) valp)->align[2])) {
+ *((AutoAlign *) valp) = aa;
+ *flags |= desc->flags;
+ }
+ break;
+ }
+
+ case RADAR_CONFIG_LEADER_ANCHORS:
+ /*
+ * Format is: lChar leftLeaderAnchor [ lChar rightLeaderAnchor]
+ *
+ * If lChar is a '|', leftLeaderAnchor and rightLeaderAnchor are the indices
+ * of the fields that serve to anchor the label's leader. More specifically
+ * the bottom left corner of the left field and the bottom right corner of
+ * the right field are used as the anchors.
+ * If lChar is '%', leftLeaderAnchor and rightLeaderAnchor should be specified
+ * as 'valxval', 'val' being a percentage (max 100) of the width/height of
+ * the label bounding box. If rightLeaderAnchor is not specified it defaults to
+ * field 0. If rightLeaderField is not specified it defaults to
+ * leftLeaderAnchor. If neither of them are specified, the center of the label
+ * is used as an anchor.
+ *
+ */
+ {
+ LeaderAnchors lanch = NULL;
+ int anchors[4];
+ int index, num_tok, anchor_index=0;
+ char *ptr = LangString(args[i+1]);
+
+ while (*ptr && (*ptr == ' ')) {
+ ptr++;
+ }
+ while (!*ptr && (anchor_index < 4)) {
+ switch (*ptr) {
+ case '|':
+ num_tok = sscanf(ptr, "|%d%n", &anchors[anchor_index], &index);
+ if (num_tok != 1) {
+ la_error:
+ Tcl_AppendResult(wi->interp, " incorrect leader anchors \"",
+ LangString(args[i+1]), "\"", NULL);
+ return RADAR_ERROR;
+ }
+ anchors[anchor_index+1] = -1;
+ break;
+ case '%':
+ num_tok = sscanf(ptr, "%%%dx%d%n", &anchors[anchor_index],
+ &anchors[anchor_index+1], &index);
+ if (num_tok != 2) {
+ goto la_error;
+ }
+ if (anchors[anchor_index] < 0) {
+ anchors[anchor_index] = 0;
+ }
+ if (anchors[anchor_index] > 100) {
+ anchors[anchor_index] = 100;
+ }
+ if (anchors[anchor_index+1] < 0) {
+ anchors[anchor_index+1] = 0;
+ }
+ if (anchors[anchor_index+1] > 100) {
+ anchors[anchor_index+1] = 100;
+ }
+ break;
+ }
+ anchor_index += 2;
+ ptr += index;
+ }
+ /*
+ * If empty, pick the default (center of the bounding box).
+ */
+ if (anchor_index != 0) {
+ lanch = (LeaderAnchors ) RadarMalloc(sizeof(LeaderAnchorsStruct));
+ lanch->left_x = anchors[0];
+ lanch->left_y = anchors[1];
+ if (anchor_index == 2) {
+ lanch->right_x = lanch->left_x;
+ lanch->right_y = lanch->left_y;
+ }
+ else {
+ lanch->right_x = anchors[2];
+ lanch->right_y = anchors[3];
+ }
+ }
+ if (*((LeaderAnchors *) valp) != NULL) {
+ RadarFree(*((LeaderAnchors *) valp));
+ *((LeaderAnchors *) valp) = lanch;
+ *flags |= desc->flags;
+ }
+ else {
+ if (lanch != NULL) {
+ *((LeaderAnchors *) valp) = lanch;
+ *flags |= desc->flags;
+ }
+ }
+ break;
+ }
+ }
+ break;
+ }
+ else {
+ desc++;
+ }
+ }
+ }
+ return RADAR_OK;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * AttributeToString --
+ *
+ * Returns the string representation of the attribute pointed
+ * by 'valp'. The attribute type is given by 'type'. The function
+ * never fail. The buffer parameter should be able to
+ * contain 256 characters at least.
+ *
+ **********************************************************************************
+ */
+static Arg
+AttributeToString(WidgetInfo *wi,
+ char *record,
+ RadarAttrConfig *desc,
+ char *buffer,
+ Tcl_FreeProc **free_proc)
+{
+#ifndef PTK
+#define NUM_ELEMS 10
+ Arg elems[NUM_ELEMS];
+#endif
+ Arg *el_ptr;
+ Arg result = NULL;
+ char *valp = record + desc->offset;
+
+ *free_proc = TCL_STATIC;
+
+ switch (desc->type) {
+ case RADAR_CONFIG_COLOR:
+ LangSetString(&result, "");
+ if (*((XColor **) valp)) {
+ LangSetString(&result, RadarNameOfColor(*((XColor **) valp)));
+ }
+ break;
+ case RADAR_CONFIG_BOOL:
+#ifndef PTK
+ result = ISSET(*((char *) valp), desc->bool_bit) ? "1" : "0";
+#else
+ LangSetInt(&result, ISSET(*((char *) valp), desc->bool_bit) ? 1 : 0);
+#endif
+ break;
+ case RADAR_CONFIG_PATTERN:
+ LangSetString(&result, "");
+ if (*((Pixmap *) valp)) {
+ LangSetString(&result, Tk_NameOfBitmap(wi->dpy, *((Pixmap *) valp)));
+ }
+ break;
+ case RADAR_CONFIG_PATTERNS:
+ {
+ int num_pats, i;
+ Pixmap *pats;
+
+ if (*((RadarList *) valp)) {
+ pats = (Pixmap *) RadarListArray(*((RadarList *) valp));
+ num_pats = RadarListSize(*((RadarList *) valp));
+#ifndef PTK
+ if (num_pats > NUM_ELEMS) {
+ el_ptr = (char **) RadarMalloc(num_pats*sizeof(char *));
+ }
+ else {
+ el_ptr = elems;
+ }
+#else
+ el_ptr = LangAllocVec(num_pats);
+#endif
+ for (i = 0; i < num_pats; i++) {
+ if (pats[i] != RadarUnspecifiedPattern) {
+ LangSetString(&el_ptr[i], Tk_NameOfBitmap(wi->dpy, pats[i]));
+ }
+ else {
+ LangSetString(&el_ptr[i], "");
+ }
+ }
+ result = Tcl_Merge(num_pats, el_ptr);
+ *free_proc = TCL_DYNAMIC;
+#ifndef PTK
+ if (el_ptr != elems) {
+ RadarFree(el_ptr);
+ }
+#else
+ LangFreeVec(num_pats, el_ptr);
+#endif
+ }
+ else {
+ LangSetString(&result, "");
+ }
+ break;
+ }
+ case RADAR_CONFIG_TAGS:
+ {
+ int num_tags, i;
+ Tk_Uid *tags;
+
+ if (*((RadarList *) valp)) {
+ tags = (Tk_Uid *) RadarListArray(*((RadarList *) valp));
+ num_tags = RadarListSize(*((RadarList *) valp));
+#ifndef PTK
+ if (num_tags > NUM_ELEMS) {
+ el_ptr = (char **) RadarMalloc(num_tags*sizeof(char *));
+ }
+ else {
+ el_ptr = elems;
+ }
+#else
+ el_ptr = LangAllocVec(num_tags);
+#endif
+ for (i = 0; i < num_tags; i++) {
+ LangSetString(&el_ptr[i], tags[i]);
+ }
+ result = Tcl_Merge(num_tags, el_ptr);
+ *free_proc = TCL_DYNAMIC;
+#ifndef PTK
+ if (el_ptr != elems) {
+ RadarFree(el_ptr);
+ }
+#else
+ LangFreeVec(num_tags, el_ptr);
+#endif
+ }
+ else {
+ LangSetString(&result, "");
+ }
+ break;
+ }
+ case RADAR_CONFIG_TEXT:
+ case RADAR_CONFIG_MAP_INFO:
+ case RADAR_CONFIG_IMAGE:
+ LangSetString(&result, *((char **) valp));
+ break;
+ case RADAR_CONFIG_FONT:
+ LangSetString(&result, "");
+ if (*((Tk_Font *) valp)) {
+ LangSetString(&result, Tk_NameOfFont(*((Tk_Font *) valp)));
+ }
+ break;
+ case RADAR_CONFIG_BORDER:
+ {
+ Border border = *((Border *) valp);
+ if (border == NO_BORDER) {
+ LangSetString(&result, NO_BORDER_SPEC);
+ break;
+ }
+ *free_proc = TCL_VOLATILE;
+ buffer[0] = 0;
+ if ((border & CONTOUR_BORDER) == CONTOUR_BORDER) {
+ strcat(buffer, CONTOUR_SPEC);
+ }
+ else {
+ if (border & LEFT_BORDER) {
+ strcat(buffer, LEFT_SPEC);
+ }
+ if (border & RIGHT_BORDER) {
+ if (buffer[0] != 0) {
+ strcat(buffer, " ");
+ }
+ strcat(buffer, RIGHT_SPEC);
+ }
+ if (border & TOP_BORDER) {
+ if (buffer[0] != 0) {
+ strcat(buffer, " ");
+ }
+ strcat(buffer, TOP_SPEC);
+ }
+ if (border & BOTTOM_BORDER) {
+ if (buffer[0] != 0) {
+ strcat(buffer, " ");
+ }
+ strcat(buffer, BOTTOM_SPEC);
+ }
+ }
+ if (border & OBLIQUE) {
+ if (buffer[0] != 0) {
+ strcat(buffer, " ");
+ }
+ strcat(buffer, OBLIQUE_SPEC);
+ }
+ if (border & COUNTER_OBLIQUE) {
+ if (buffer[0] != 0) {
+ strcat(buffer, " ");
+ }
+ strcat(buffer, COUNTER_OBLIQUE_SPEC);
+ }
+ LangSetString(&result, buffer);
+ }
+ break;
+ case RADAR_CONFIG_LINE_SHAPE:
+ {
+ LineShape line_shape = *((LineShape *) valp);
+ switch (line_shape) {
+ case LINE_STRAIGHT:
+ LangSetString(&result, STRAIGHT_SPEC);
+ break;
+ case LINE_RIGHT_LIGHTNING:
+ LangSetString(&result, RIGHT_LIGHTNING_SPEC);
+ break;
+ case LINE_LEFT_LIGHTNING:
+ LangSetString(&result, LEFT_LIGHTNING_SPEC);
+ break;
+ case LINE_RIGHT_CORNER:
+ LangSetString(&result, RIGHT_CORNER_SPEC);
+ break;
+ case LINE_LEFT_CORNER:
+ LangSetString(&result, LEFT_CORNER_SPEC);
+ break;
+ case LINE_DOUBLE_RIGHT_CORNER:
+ LangSetString(&result, DOUBLE_RIGHT_CORNER_SPEC);
+ break;
+ case LINE_DOUBLE_LEFT_CORNER:
+ LangSetString(&result, DOUBLE_LEFT_CORNER_SPEC);
+ break;
+ }
+ break;
+ }
+ case RADAR_CONFIG_LINE_STYLE:
+ {
+ LineStyle line_style = *((LineStyle *) valp);
+ switch (line_style) {
+ case LINE_SIMPLE:
+ LangSetString(&result, SIMPLE_SPEC);
+ break;
+ case LINE_DASHED:
+ LangSetString(&result, DASHED_SPEC);
+ break;
+ case LINE_MIXED:
+ LangSetString(&result, MIXED_SPEC);
+ break;
+ case LINE_DOTTED:
+ LangSetString(&result, DOTTED_SPEC);
+ break;
+ }
+ break;
+ }
+ case RADAR_CONFIG_LINE_END:
+ {
+ LineEnd line_end = *((LineEnd *) valp);
+
+ if (!line_end) {
+ LangSetString(&result, "");
+ }
+ else {
+ LangSetString(&result, LineEndGetString(line_end));
+ }
+ break;
+ }
+ case RADAR_CONFIG_RELIEF:
+ LangSetString(&result , Tk_NameOfRelief(*((ReliefStyle *) valp)));
+ break;
+ case RADAR_CONFIG_JOIN_STYLE:
+ LangSetString(&result , Tk_NameOfJoinStyle(*((int *) valp)));
+ break;
+ case RADAR_CONFIG_CAP_STYLE:
+ LangSetString(&result , Tk_NameOfCapStyle(*((int *) valp)));
+ break;
+ case RADAR_CONFIG_POINT:
+ {
+#ifndef PTK
+ *free_proc = TCL_VOLATILE;
+ sprintf(buffer, "%g %g", ((RadarPoint *) valp)->x, ((RadarPoint *) valp)->y);
+ result = buffer;
+#else
+ Arg *list = LangAllocVec(2);
+ LangSetInt(&list[0], ((RadarPoint *) valp)->x);
+ LangSetInt(&list[1], ((RadarPoint *) valp)->y);
+ result = Tcl_Merge(2, list);
+ LangFreeVec(2, list);
+#endif
+ break;
+ }
+ case RADAR_CONFIG_RECT:
+ {
+#ifndef PTK
+ *free_proc = TCL_VOLATILE;
+ sprintf(buffer, "%g %g %g %g",
+ ((RadarRect *) valp)->x, ((RadarRect *) valp)->y,
+ ((RadarRect *) valp)->w, ((RadarRect *) valp)->h);
+ result = buffer;
+#else
+ Arg *list = LangAllocVec(4);
+ LangSetInt(&list[0], ((RadarRect *) valp)->x);
+ LangSetInt(&list[1], ((RadarRect *) valp)->x);
+ LangSetInt(&list[2], ((RadarRect *) valp)->w);
+ LangSetInt(&list[3], ((RadarRect *) valp)->h);
+ result = Tcl_Merge(4, list);
+ LangFreeVec(4, list);
+#endif
+ break;
+ }
+ case RADAR_CONFIG_ITEM:
+ if (*((Item *) valp) == RADAR_NO_ITEM) {
+ LangSetString(&result, "");
+ }
+ else {
+ *free_proc = TCL_VOLATILE;
+ sprintf(buffer, "%d", (*((Item *) valp))->id);
+ LangSetString(&result, buffer);
+ }
+ break;
+ case RADAR_CONFIG_INT:
+ case RADAR_CONFIG_UINT:
+ case RADAR_CONFIG_DIM:
+ case RADAR_CONFIG_PRI:
+ case RADAR_CONFIG_ANGLE:
+ *free_proc = TCL_VOLATILE;
+ sprintf(buffer, "%d", *((int *) valp));
+ LangSetString(&result, buffer);
+ break;
+ case RADAR_CONFIG_JUSTIFY:
+ {
+ Tk_Justify justify = *((RadarJustify *) valp);
+ LangSetString(&result, Tk_NameOfJustify(justify));
+ break;
+ }
+ case RADAR_CONFIG_ANCHOR:
+ LangSetString(&result, Tk_NameOfAnchor(*((Tk_Anchor *) valp)));
+ break;
+ case RADAR_CONFIG_LABEL_FORMAT:
+ {
+ LabelFormat frmt = *((LabelFormat *) valp);
+
+ if (!frmt) {
+ LangSetString(&result, "");
+ }
+ else {
+ LangSetString(&result, LabelFormatGetString(frmt));
+ }
+ break;
+ }
+ case RADAR_CONFIG_AUTO_JUSTIFY:
+ {
+ AutoAlign *aa = (AutoAlign *) valp;
+ int i;
+ if (aa->automatic == False) {
+ LangSetString(&result, "-");
+ }
+ else {
+ *free_proc = TCL_VOLATILE;
+ buffer[0] = 0;
+ for (i = 0; i < 3; i++) {
+ switch (aa->align[i]) {
+ case RadarJustifyLeft:
+ strcat(buffer, "l");
+ break;
+ case RadarJustifyCenter:
+ strcat(buffer, "c");
+ break;
+ case RadarJustifyRight:
+ strcat(buffer, "r");
+ break;
+ }
+ }
+ LangSetString(&result, buffer);
+ }
+ }
+ break;
+
+ case RADAR_CONFIG_LEADER_ANCHORS:
+ {
+ LeaderAnchors lanch = (LeaderAnchors) valp;
+ char *ptr = buffer;
+ int count;
+
+ if (lanch->left_y < 0) {
+ count = sprintf(ptr, "|%d", lanch->left_x);
+ }
+ else {
+ count = sprintf(ptr, "%%%dx%d", lanch->left_x, lanch->left_y);
+ }
+ ptr += count;
+ if (lanch->right_y < 0) {
+ count = sprintf(ptr, "|%d", lanch->right_x);
+ }
+ else {
+ count = sprintf(ptr, "%%%dx%d", lanch->right_x, lanch->right_y);
+ }
+ LangSetString(&result, buffer);
+ }
+ }
+
+ return result;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * QueryAttribute --
+ *
+ **********************************************************************************
+ */
+static int
+QueryAttribute(char *record,
+ int field, /* 0< means item itself. */
+ Arg attr_name)
+{
+ WidgetInfo *wi;
+ Item item;
+ Tk_Uid attr_uid = Tk_GetUid(LangString(attr_name));
+ Arg result = NULL;
+ Tcl_FreeProc *free_proc;
+ char buffer[256];
+ RadarAttrConfig *desc;
+
+ if (field < 0) {
+ item = (Item) record;
+ wi = item->wi;
+ desc = item->class->attr_desc;
+ }
+ else {
+ FieldSet field_set = (FieldSet) record;
+ wi = field_set->wi;
+ if (field >= field_set->num_fields) {
+ Tcl_AppendResult(wi->interp, "invalid field index \"", NULL);
+ return RADAR_ERROR;
+ }
+ record = (char *) &field_set->fields[field];
+ desc = field_attrs;
+ }
+
+ while (True) {
+ if (desc->type == RADAR_CONFIG_END) {
+ Tcl_AppendResult(wi->interp, "unknown attribute \"",
+ attr_uid, "\"", NULL);
+ return RADAR_ERROR;
+ }
+ else if (attr_uid == desc->uid) {
+ result = AttributeToString(wi, record, desc, buffer, &free_proc);
+#ifndef PTK
+ Tcl_SetResult(wi->interp, result, free_proc);
+#else
+ Tcl_ArgResult(wi->interp, result);
+#endif
+ break;
+ }
+ else {
+ desc++;
+ }
+ }
+ return RADAR_OK;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ItemClassList --
+ *
+ **********************************************************************************
+ */
+static RadarList
+ItemClassList()
+{
+ return item_classes;
+}
+
+/*
+ **********************************************************************************
+ *
+ * LookupItemClass --
+ *
+ **********************************************************************************
+ */
+static ItemClass
+LookupItemClass(char *class_name)
+{
+ ItemClass *class;
+ int i, num_classes;
+
+ class = (ItemClass *) RadarListArray(item_classes);
+ num_classes = RadarListSize(item_classes);
+ for (i = 0; i < num_classes; i++) {
+ if (strcasecmp((class[i])->name, class_name) == 0) {
+ return class[i];
+ }
+ }
+
+ return NULL;
+}
+
+/*
+ **********************************************************************************
+ *
+ * AddItemClass --
+ *
+ **********************************************************************************
+ */
+static void
+AddItemClass(ItemClass class)
+{
+ if (!LookupItemClass(class->name)) {
+ RadarListAdd(item_classes, &class, RadarListTail);
+ InitAttrDesc(class->attr_desc);
+ }
+}
+
+/*
+ **********************************************************************************
+ *
+ * GlobalModuleInit --
+ * Initialize classes static state.
+ *
+ **********************************************************************************
+ */
+static void
+GlobalModuleInit()
+{
+ /* First check if static part already inited */
+ if (item_classes == NULL) {
+ item_classes = RadarListNew(16, sizeof(ItemClass));
+ AddItemClass(RadarTrack);
+ AddItemClass(RadarWayPoint);
+ AddItemClass(RadarMap);
+ AddItemClass(RadarReticle);
+ AddItemClass(RadarTabular);
+ AddItemClass(RadarRectangle);
+ AddItemClass(RadarArc);
+ AddItemClass(RadarCurve);
+ AddItemClass(RadarGroup);
+ AddItemClass(RadarIcon);
+ AddItemClass(RadarText);
+ /*AddItemClass(RadarMosaic);*/
+ InitAttrDesc(field_attrs);
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ComputeFieldAttachment --
+ * Compute the location/size of the field, computing attachments if any.
+ *
+ **********************************************************************************
+ */
+static void
+ComputeFieldAttachment(FieldSet field_set,
+ int field,
+ RadarBBox *field_bbox)
+{
+
+ RadarBBox ref_bbox;
+ RadarDim real_width, real_height;
+ int ref_field, num_fields;
+ char x_attach, y_attach, x_dim, y_dim;
+ short width_spec, height_spec;
+ int x_spec, y_spec, icon_width=0, icon_height=0;
+ Field field_ptr;
+ Tk_FontMetrics fm;
+
+ field_ptr = &field_set->fields[field];
+ if (ISSET(field_ptr->flags, CACHE_OK)) {
+ field_bbox->orig.x = (RadarPos) field_ptr->orig_x;
+ field_bbox->orig.y = (RadarPos) field_ptr->orig_y;
+ field_bbox->corner.x = field_ptr->corner_x;
+ field_bbox->corner.y = field_ptr->corner_y;
+ return;
+ }
+
+ num_fields = LabelFormatNumFields(field_set->label_format);
+ LabelFormatGetField(field_set->label_format, field,
+ &x_attach, &y_attach, &x_dim, &y_dim,
+ &x_spec, &y_spec, &width_spec, &height_spec);
+
+ /*
+ * First try to compute the field size which may be a factor
+ * of the field content (but not a factor of other fields).
+ */
+ if ((field_ptr->image != RadarUnspecifiedImage) &&
+ ((x_dim == LF_DIM_ICON) || (y_dim == LF_DIM_ICON) ||
+ (x_dim == LF_DIM_AUTO) || (y_dim == LF_DIM_AUTO))) {
+ Tk_SizeOfImage(field_ptr->image, &icon_width, &icon_height);
+ }
+
+ switch (x_dim) {
+ case LF_DIM_FONT:
+ real_width = (RadarDim) (width_spec*RadarTextWidth(field_ptr->font, "N", 1)/100);
+ break;
+ case LF_DIM_ICON:
+ real_width = (RadarDim) (width_spec*icon_width/100);
+ break;
+ case LF_DIM_AUTO:
+ {
+ int len = strlen(field_ptr->text);
+ RadarDim text_width;
+
+ real_width = 0.0;
+ if (field_ptr->image != RadarUnspecifiedImage) {
+ real_width = (RadarDim) icon_width;
+ }
+ if (len) {
+ text_width = (RadarDim) RadarTextWidth(field_ptr->font, field_ptr->text, len);
+ real_width = text_width < real_width ? real_width : text_width;
+ }
+ real_width += (RadarDim) width_spec;
+ break;
+ }
+ case LF_DIM_PIXEL:
+ default:
+ real_width = (RadarDim) width_spec;
+ break;
+ }
+ /*printf("field %d, width = %g\n", field, real_width);*/
+
+ switch (y_dim) {
+ case LF_DIM_FONT:
+ {
+ Tk_GetFontMetrics(field_ptr->font, &fm);
+ real_height = (RadarDim) (height_spec*(fm.ascent + fm.descent)/100);
+ break;
+ }
+ case LF_DIM_ICON:
+ real_height = (RadarDim) (height_spec*icon_height/100);
+ break;
+ case LF_DIM_AUTO:
+ {
+ RadarDim text_height;
+
+ real_height = 0.0;
+ if (field_ptr->image != RadarUnspecifiedImage) {
+ real_height = (RadarDim) icon_height;
+ }
+ if (strlen(field_ptr->text)) {
+ Tk_GetFontMetrics(field_ptr->font, &fm);
+ text_height = (RadarDim) (fm.ascent + fm.descent);
+ real_height = text_height < real_height ? real_height : text_height;
+ }
+ real_height += (RadarDim) height_spec;
+ break;
+ }
+ case LF_DIM_PIXEL:
+ default:
+ real_height = (RadarDim) height_spec;
+ break;
+ }
+ /*printf("field %d, height = %g\n", field, real_height);*/
+
+ /*
+ * Then try to deduce the position, resolving any attachments
+ * if needed.
+ */
+ /*
+ * Preset this field to a default position/size and pretend
+ * its cache is ok to break any deadlocks.
+ */
+ field_ptr->orig_x = field_ptr->orig_y = 0;
+ field_ptr->corner_x = (short) real_width;
+ field_ptr->corner_y = (short) real_height;
+ SET(field_ptr->flags, CACHE_OK);
+
+ /*
+ * Do the x axis.
+ */
+ if (x_attach == LF_ATTACH_PIXEL) {
+ field_bbox->orig.x = (RadarPos) x_spec;
+ field_bbox->corner.x = field_bbox->orig.x + real_width;
+ }
+ else {
+ ref_field = x_spec;
+ field_bbox->orig.x = field_bbox->corner.x = 0;
+ if ((ref_field < 0) || (ref_field >= num_fields)) {
+ RadarWarning ("Attached (x) to an inexistant field geometry");
+ }
+ else {
+ ComputeFieldAttachment(field_set, ref_field, &ref_bbox);
+ switch (x_attach) {
+ case LF_ATTACH_FWD:
+ if (ISSET(field_set->fields[ref_field].flags, FIELD_VISIBLE_BIT)) {
+ field_bbox->orig.x = ref_bbox.corner.x;
+ }
+ else {
+ field_bbox->orig.x = ref_bbox.orig.x;
+ }
+ field_bbox->corner.x = field_bbox->orig.x + real_width;
+ break;
+ case LF_ATTACH_BWD:
+ if (ISSET(field_set->fields[ref_field].flags, FIELD_VISIBLE_BIT)) {
+ field_bbox->corner.x = ref_bbox.orig.x;
+ }
+ else {
+ field_bbox->corner.x = ref_bbox.corner.x;
+ }
+ field_bbox->orig.x = field_bbox->corner.x - real_width;
+ break;
+ case LF_ATTACH_LEFT:
+ field_bbox->orig.x = ref_bbox.orig.x;
+ field_bbox->corner.x = field_bbox->orig.x + real_width;
+ break;
+ case LF_ATTACH_RIGHT:
+ if (ISSET(field_set->fields[ref_field].flags, FIELD_VISIBLE_BIT)) {
+ field_bbox->corner.x = ref_bbox.corner.x;
+ }
+ else {
+ field_bbox->corner.x = ref_bbox.orig.x;
+ }
+ field_bbox->orig.x = field_bbox->corner.x - real_width;
+ break;
+ }
+ }
+ }
+ /*printf("field %d, x = %g\n", field, field_bbox->orig.x);*/
+
+ /*
+ * Then the y axis.
+ */
+ if (y_attach == LF_ATTACH_PIXEL) {
+ field_bbox->orig.y = (RadarPos) y_spec;
+ field_bbox->corner.y = field_bbox->orig.y + real_height;
+ }
+ else {
+ ref_field = y_spec;
+ field_bbox->orig.y = field_bbox->corner.y = 0;
+ if ((ref_field < 0) || (ref_field >= num_fields)) {
+ RadarWarning ("Attached (y) to an inexistant field geometry");
+ }
+ else {
+ ComputeFieldAttachment(field_set, ref_field, &ref_bbox);
+ switch (y_attach) {
+ case LF_ATTACH_FWD:
+ if (ISSET(field_set->fields[ref_field].flags, FIELD_VISIBLE_BIT)) {
+ field_bbox->orig.y = ref_bbox.corner.y;
+ }
+ else {
+ field_bbox->orig.y = ref_bbox.orig.y;
+ }
+ field_bbox->corner.y = field_bbox->orig.y + real_height;
+ break;
+ case LF_ATTACH_BWD:
+ if (ISSET(field_set->fields[ref_field].flags, FIELD_VISIBLE_BIT)) {
+ field_bbox->corner.y = ref_bbox.orig.y;
+ }
+ else {
+ field_bbox->corner.y = ref_bbox.corner.y;
+ }
+ field_bbox->orig.y = field_bbox->corner.y - real_height;
+ break;
+ case LF_ATTACH_LEFT:
+ field_bbox->orig.y = ref_bbox.orig.y;
+ field_bbox->corner.y = field_bbox->orig.y + real_height;
+ break;
+ case LF_ATTACH_RIGHT:
+ if (ISSET(field_set->fields[ref_field].flags, FIELD_VISIBLE_BIT)) {
+ field_bbox->corner.y = ref_bbox.corner.y;
+ }
+ else {
+ field_bbox->corner.y = ref_bbox.orig.y;
+ }
+ field_bbox->orig.y = field_bbox->corner.y - real_height;
+ break;
+ }
+ }
+ }
+ /*printf("field %d, y = %g\n", field, field_bbox->orig.y);*/
+
+ field_ptr->orig_x = (short) field_bbox->orig.x;
+ field_ptr->orig_y = (short) field_bbox->orig.y;
+ field_ptr->corner_x = (short) field_bbox->corner.x;
+ field_ptr->corner_y = (short) field_bbox->corner.y;
+ SET(field_ptr->flags, CACHE_OK);
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ClearFieldCache --
+ * Reset the geometric cache of all fields depending on a given field (or
+ * of all fields if the field is < 0). Clear also the label bounding box
+ * cache if some action has been taken on a field.
+ *
+ **********************************************************************************
+ */
+void
+ClearFieldCache(FieldSet field_set,
+ int field)
+{
+ int i, num_fields;
+ RadarBool clear_bbox;
+ int x_spec, y_spec;
+ char x_attach, y_attach, x_dim, y_dim;
+ short width_spec, height_spec;
+
+ if (field < 0) {
+ for (i = 0; i < field_set->num_fields; i++) {
+ CLEAR(field_set->fields[i].flags, CACHE_OK);
+ }
+ field_set->label_width = field_set->label_height = -1.0;
+ return;
+ }
+
+ clear_bbox = False;
+ if (!field_set->label_format) {
+ return;
+ }
+ num_fields = LabelFormatNumFields(field_set->label_format);
+ if (field >= num_fields) {
+ return;
+ }
+ LabelFormatGetField(field_set->label_format, field,
+ &x_attach, &y_attach, &x_dim, &y_dim,
+ &x_spec, &y_spec, &width_spec, &height_spec);
+ if ((x_dim != LF_DIM_PIXEL) || (y_dim != LF_DIM_PIXEL)) {
+ CLEAR(field_set->fields[field].flags, CACHE_OK);
+ }
+ for (i = 0; i < num_fields; i++) {
+ LabelFormatGetField(field_set->label_format, i,
+ &x_attach, &y_attach, &x_dim, &y_dim,
+ &x_spec, &y_spec, &width_spec, &height_spec);
+ if ((x_attach == LF_ATTACH_PIXEL) && (y_attach == LF_ATTACH_PIXEL)) {
+ continue;
+ }
+ if (x_attach != LF_ATTACH_PIXEL) {
+ if ((x_spec == field) && ISSET(field_set->fields[i].flags, CACHE_OK)) {
+ CLEAR(field_set->fields[i].flags, CACHE_OK);
+ ClearFieldCache(field_set, i);
+ clear_bbox = True;
+ }
+ }
+ if (y_attach != LF_ATTACH_PIXEL) {
+ if ((y_spec == field) && ISSET(field_set->fields[i].flags, CACHE_OK)) {
+ CLEAR(field_set->fields[i].flags, CACHE_OK);
+ ClearFieldCache(field_set, i);
+ clear_bbox = True;
+ }
+ }
+ }
+
+ if (clear_bbox) {
+ field_set->label_width = field_set->label_height = -1.0;
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * GetLabelBBox --
+ *
+ **********************************************************************************
+ */
+void
+GetLabelBBox(FieldSet field_set,
+ RadarDim *w,
+ RadarDim *h)
+{
+ RadarBBox bbox, tmp_bbox;
+ LabelFormat lf;
+ int i, num_fields;
+ RadarDim clip_w, clip_h;
+
+ if ((field_set->label_width >= 0.0) && (field_set->label_height >= 0.0)) {
+ *w = field_set->label_width;
+ *h = field_set->label_height;
+ return;
+ }
+
+ lf = field_set->label_format;
+ if (lf == NULL) {
+ *w = *h = field_set->label_width = field_set->label_height = 0.0;
+ return;
+ }
+
+ ResetBBox(&bbox);
+ num_fields = LabelFormatNumFields(lf);
+ for (i = 0; i < num_fields; i++) {
+ ComputeFieldAttachment(field_set, i, &tmp_bbox);
+ AddBBoxToBBox(&bbox, &tmp_bbox);
+ }
+ field_set->label_width = (RadarDim) (bbox.corner.x - bbox.orig.x);
+ field_set->label_height = (RadarDim) (bbox.corner.y - bbox.orig.y);
+ if (bbox.orig.x < 0) {
+ field_set->label_width += (RadarDim) bbox.orig.x;
+ }
+ if (bbox.orig.y < 0) {
+ field_set->label_height += (RadarDim) bbox.orig.y;
+ }
+
+ if (LabelFormatGetClipBox(lf, &clip_w, &clip_h)) {
+ if (clip_w < field_set->label_width) {
+ field_set->label_width = clip_w;
+ }
+ if (clip_h < field_set->label_height) {
+ field_set->label_height = clip_h;
+ }
+ }
+
+ *w = field_set->label_width;
+ *h = field_set->label_height;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * GetFieldBBox --
+ * Compute the location of the field described
+ * by the field entry index in the item current LabelFormat.
+ *
+ **********************************************************************************
+ */
+static void
+GetFieldBBox(FieldSet field_set,
+ unsigned int index,
+ RadarBBox *field_bbox)
+{
+ if (field_set->label_format) {
+ ComputeFieldAttachment(field_set, index, field_bbox);
+ field_bbox->orig.x += field_set->label_pos.x;
+ field_bbox->orig.y += field_set->label_pos.y;
+ field_bbox->corner.x += field_set->label_pos.x;
+ field_bbox->corner.y += field_set->label_pos.y;
+ }
+ else {
+ ResetBBox(field_bbox);
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ComputeFieldTextLocation --
+ * Compute the position of the text in a field. This is a position
+ * that we can give to XDrawText. The position is deduced from the
+ * field bounding box passed in bbox.
+ * Return also the text bounding box.
+ *
+ **********************************************************************************
+ */
+static void
+ComputeFieldTextLocation(Field field_ptr,
+ RadarBBox *bbox,
+ RadarPoint *pos,
+ RadarBBox *text_bbox)
+{
+ RadarBool relief_thickness;
+ RadarDim w, h;
+ Tk_FontMetrics fm;
+
+ relief_thickness = (field_ptr->relief != RELIEF_FLAT &&
+ field_ptr->relief_thickness > 1) ? field_ptr->relief_thickness : 0;
+ Tk_GetFontMetrics(field_ptr->font, &fm);
+ w = RadarTextWidth(field_ptr->font, field_ptr->text, strlen(field_ptr->text));
+ h = fm.ascent + fm.descent;
+ text_bbox->orig.y = (bbox->orig.y + bbox->corner.y - h) / 2;
+ text_bbox->corner.y = text_bbox->orig.y + h;
+ pos->y = text_bbox->orig.y + fm.ascent;
+
+ switch (field_ptr->alignment) {
+ case RadarJustifyLeft:
+ text_bbox->orig.x = bbox->orig.x + 1 + relief_thickness;
+ break;
+ case RadarJustifyRight:
+ text_bbox->orig.x = bbox->corner.x - w - 1 - relief_thickness;
+ break;
+ default:
+ text_bbox->orig.x = (bbox->orig.x + bbox->corner.x - w) / 2;
+ break;
+ }
+ text_bbox->corner.x = text_bbox->orig.x + w;
+ pos->x = text_bbox->orig.x;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * LeaderToLabel --
+ * Compute the segment part of segment <start, end> that lies
+ * outside the fields of item.
+ *
+ **********************************************************************************
+ */
+static void
+LeaderToLabel(FieldSet field_set,
+ RadarPoint *start,
+ RadarPoint *end)
+{
+ int b_num;
+ RadarPoint delta, inf, sup;
+ RadarPos xt=0, yu=0, yw=0, xv=0;
+ Field field_ptr;
+ int i;
+ RadarBBox field_bbox;
+
+ /* Intersection points : */
+ /* T |xt / delta_y U |x1 V |y1 W |yw / delta_x */
+ /* |y2 |yu / delta_x |xv / delta_y |x2 */
+ /* */
+ /* y = ax + b; */
+ /* a = delta_y / delta_x */
+ /* b = (y * delta_x - x * delta_y) / delta_x */
+
+ delta.x = start->x - end->x;
+ delta.y = start->y - end->y;
+ b_num = start->y*delta.x - start->x*delta.y;
+
+ for (i = 0; i < LabelFormatNumFields(field_set->label_format); i++) {
+ field_ptr = &field_set->fields[i];
+ /*
+ * If the field is made invisible or has no graphics of
+ * its own, don't clip.
+ */
+ if (ISCLEAR(field_ptr->flags, FIELD_VISIBLE_BIT) ||
+ ((*field_ptr->text == 0) &&
+ ISCLEAR(field_ptr->flags, FILLED_BIT) &&
+ (field_ptr->border_edges == NO_BORDER) &&
+ (field_ptr->relief == RELIEF_FLAT) &&
+ (field_ptr->image == RadarUnspecifiedImage))) {
+ continue;
+ }
+
+ /*
+ * field_bbox is in absolute device coordinates.
+ */
+ GetFieldBBox(field_set, i, &field_bbox);
+
+ /*
+ * Adjust leader on real text, not on field boundaries. This is
+ * important when there are leading and trailing spaces.
+ * The correct test here is really *field_ptr->text, an empty
+ * text is represented by an empty string NOT by a NULL pointer.
+ */
+ if (*field_ptr->text && ISCLEAR(field_ptr->flags, FILLED_BIT) &&
+ (field_ptr->border_edges == NO_BORDER) &&
+ (field_ptr->relief == RELIEF_FLAT) &&
+ (field_ptr->image == RadarUnspecifiedImage)) {
+ RadarBBox text_bbox;
+ RadarPoint text_pos; /* dummy */
+ int space_width;
+ int scan_forw, scan_back;
+ space_width = RadarTextWidth(field_ptr->font, " ", 1);
+
+ ComputeFieldTextLocation(field_ptr, &field_bbox, &text_pos, &text_bbox);
+ /*
+ * Correct adjusments made by ComputeFieldTextLocation (Vincent Pomey).
+ *
+ * PLC: IMHO, this is to compensate for exotic fonts like 'symbolesATC'.
+ * I am not planning to port this to Tk for two reasons:
+ * 1/ Current positions are no longer implemented as characters
+ * and 2/ Tk does not give access (easily) to lbearings and rbearings.
+ * This patch has been contributed by Phidias team. I don't know the
+ * problem it was meant to solve.
+ * text_bbox.x -= field_ptr->font->per_char[field_ptr->text[0]].lbearing + 3;
+ * text_bbox.width += field_ptr->font->per_char[field_ptr->text[0]].lbearing + 3;
+ */
+ /*
+ * Change bbox according to leading and trailing spaces.
+ */
+ scan_forw = 0;
+ while (field_ptr->text[scan_forw] == ' ') {
+ /* leading spaces */
+ text_bbox.orig.x += space_width;
+ scan_forw++;
+ }
+
+ /*
+ * Empty text.
+ */
+ if (field_ptr->text[scan_forw] == 0) {
+ continue;
+ }
+
+ scan_back = strlen(field_ptr->text)-1;
+ while ((field_ptr->text[scan_back] == ' ') && (scan_back > scan_forw)) {
+ /* trailing spaces */
+ text_bbox.corner.x -= space_width;
+ scan_back--;
+ }
+
+ field_bbox = text_bbox;
+ }
+
+ if (field_bbox.corner.x <= field_bbox.orig.x) {
+ continue;
+ }
+
+ if ((start->x >= field_bbox.orig.x) && (start->x < field_bbox.corner.x) &&
+ (start->y >= field_bbox.orig.y) && (start->y < field_bbox.corner.y)) {
+ end->x = start->x;
+ end->y = start->y;
+ }
+ if (delta.x) {
+ yu = (field_bbox.orig.x*delta.y + b_num) / delta.x;
+ yw = (field_bbox.corner.x*delta.y + b_num) / delta.x;
+ }
+ if (delta.y) {
+ xt = (field_bbox.corner.y*delta.x - b_num) / delta.y;
+ xv = (field_bbox.orig.y*delta.x - b_num) / delta.y;
+ }
+
+ inf.x = MIN(start->x, end->x);
+ sup.x = MAX(start->x, end->x);
+ inf.y = MIN(start->y, end->y);
+ sup.y = MAX(start->y, end->y);
+
+ if (delta.x) {
+ if ((yu >= field_bbox.orig.y) && (yu <= field_bbox.corner.y) &&
+ (field_bbox.orig.x >= inf.x) && (field_bbox.orig.x <= sup.x) &&
+ (yu >= inf.y) && (yu <= sup.y)) {
+ end->x = field_bbox.orig.x;
+ end->y = yu;
+ inf.x = MIN(start->x, end->x);
+ sup.x = MAX(start->x, end->x);
+ inf.y = MIN(start->y, end->y);
+ sup.y = MAX(start->y, end->y);
+ }
+ if ((yw >= field_bbox.orig.y) && (yw <= field_bbox.corner.y) &&
+ (field_bbox.corner.x >= inf.x) && (field_bbox.corner.x <= sup.x) &&
+ (yw >= inf.y) && (yw <= sup.y)) {
+ end->x = field_bbox.corner.x;
+ end->y = yw;
+ inf.x = MIN(start->x, end->x);
+ sup.x = MAX(start->x, end->x);
+ inf.y = MIN(start->y, end->y);
+ sup.y = MAX(start->y, end->y);
+ }
+ }
+ if (delta.y) {
+ if ((xt >= field_bbox.orig.x) && (xt <= field_bbox.corner.x) &&
+ (xt >= inf.x) && (xt <= sup.x) &&
+ (field_bbox.corner.y >= inf.y) && (field_bbox.corner.y <= sup.y)) {
+ end->x = xt;
+ end->y = field_bbox.corner.y;
+ inf.x = MIN(start->x, end->x);
+ sup.x = MAX(start->x, end->x);
+ inf.y = MIN(start->y, end->y);
+ sup.y = MAX(start->y, end->y);
+ }
+ if ((xv >= field_bbox.orig.x) && (xv <= field_bbox.corner.x) &&
+ (xv >= inf.x) && (xv <= sup.x) &&
+ (field_bbox.orig.y >= inf.y) && (field_bbox.orig.y <= sup.y)) {
+ end->x = xv;
+ end->y = field_bbox.orig.y;
+ inf.x = MIN(start->x, end->x);
+ sup.x = MAX(start->x, end->x);
+ inf.y = MIN(start->y, end->y);
+ sup.y = MAX(start->y, end->y);
+ }
+ }
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * InsertDependentItem --
+ *
+ **********************************************************************************
+ */
+static void
+InsertDependentItem(Item item)
+{
+ GroupItem group = (GroupItem) item->parent;
+
+ if (!group) {
+ return;
+ }
+ if (!group->dependents) {
+ group->dependents = RadarListNew(2, sizeof(Item));
+ }
+ RadarListAdd(group->dependents, &item, RadarListTail);
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ExtractDependentItem --
+ *
+ **********************************************************************************
+ */
+static int CmpItem(void *item1,
+ void *item2)
+{
+ return (item1 == item2);
+}
+
+static void
+ExtractDependentItem(Item item)
+{
+ GroupItem group = (GroupItem) item->parent;
+ int index;
+
+ if (!group || !group->dependents) {
+ return;
+ }
+ index = RadarListDo(group->dependents, item, CmpItem);
+ if (index >= 0) {
+ RadarListDelete(group->dependents, index);
+ if (RadarListSize(group->dependents) == 0) {
+ RadarListFree(group->dependents);
+ group->dependents = NULL;
+ }
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * UpdateItemDependency --
+ * Update the group dependency list following a change in the
+ * connection of an item.
+ *
+ **********************************************************************************
+ */
+static void
+UpdateItemDependency(Item item,
+ Item old_connection)
+{
+ if (old_connection == RADAR_NO_ITEM) {
+ /* Add a connection */
+ InsertDependentItem(item);
+ }
+ else if (item->connected_item == RADAR_NO_ITEM) {
+ /* Remove a connection */
+ ExtractDependentItem(item);
+ }
+ else {
+ /* Move at end to ensure that it will be updated after
+ * the (new) item it depends upon.
+ */
+ ExtractDependentItem(item);
+ InsertDependentItem(item);
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * DisconnectDependentItems --
+ *
+ *
+ **********************************************************************************
+ */
+static void
+DisconnectDependentItems(Item item)
+{
+ Item current_item;
+ GroupItem group = (GroupItem) item->parent;
+ Item *deps;
+ int num_deps, i;
+
+ if (!group || !group->dependents) {
+ return;
+ }
+ deps = (Item *) RadarListArray(group->dependents);
+ num_deps = RadarListSize(group->dependents);
+
+ for (i = num_deps-1; i >= 0; i--) {
+ current_item = deps[i];
+ if (current_item->connected_item == item) {
+ current_item->connected_item = RADAR_NO_ITEM;
+ RadarListDelete(group->dependents, i);
+ Invalidate(current_item, RADAR_COORDS_FLAG);
+ }
+ }
+ if (RadarListSize(group->dependents) == 0) {
+ RadarListFree(group->dependents);
+ group->dependents = NULL;
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * InsertItem --
+ *
+ * Insert an item in the display list according to its priority.
+ * It is inserted in front of items of lower or same priority. If
+ * mark_item is not RADAR_NO_ITEM the insertion is done relative
+ * to this item, before it if 'before' is True, after it otherwise.
+ * mark_item must be in the group 'group'.
+ *
+ **********************************************************************************
+ */
+static void
+InsertItem(Item item,
+ Item grp,
+ Item mark_item,
+ RadarBool before)
+{
+ WidgetInfo *wi = item->wi;
+ GroupItem group = (GroupItem) grp;
+
+ if (!group) {
+ group = (GroupItem) wi->top_group;
+ }
+ item->parent = (Item) group;
+
+ if (mark_item && (mark_item->parent != (Item) group)) {
+ mark_item = RADAR_NO_ITEM;
+ }
+ /*
+ * Empty list, add the first item.
+ */
+ if (group->head == RADAR_NO_ITEM) {
+ group->head = item;
+ group->tail = item;
+ item->previous = item->next = RADAR_NO_ITEM;
+ return;
+ }
+
+ if (mark_item != RADAR_NO_ITEM) {
+ /*
+ * Better leave here, mark_item will not
+ * have the links set right.
+ */
+ if (mark_item == item) {
+ return;
+ }
+ /*
+ * Force the priority to be the same as the reference
+ * item;
+ */
+ item->priority = mark_item->priority;
+ }
+ else {
+ mark_item = group->head;
+ while ((mark_item != RADAR_NO_ITEM) &&
+ (mark_item->priority > item->priority)) {
+ mark_item = mark_item->next;
+ }
+ before = True;
+ }
+
+ if (before && (mark_item != RADAR_NO_ITEM)) {
+ /*
+ * Insert before mark.
+ */
+ item->next = mark_item;
+ item->previous = mark_item->previous;
+ if (mark_item->previous == RADAR_NO_ITEM) {
+ group->head = item;
+ }
+ else {
+ mark_item->previous->next = item;
+ }
+ mark_item->previous = item;
+ }
+ else {
+ /*
+ * Insert after mark either because 'before' is False
+ * and mark_item valid or because the right place is at
+ * the end of the list and mark_item is RADAR_NO_ITEM.
+ */
+ if (mark_item == RADAR_NO_ITEM) {
+ group->tail->next = item;
+ item->previous = group->tail;
+ group->tail = item;
+ }
+ else {
+ item->previous = mark_item;
+ item->next = mark_item->next;
+ if (item->next == RADAR_NO_ITEM) {
+ group->tail = item;
+ }
+ else {
+ item->next->previous = item;
+ }
+ mark_item->next = item;
+ }
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ExtractItem --
+ *
+ **********************************************************************************
+ */
+static void
+ExtractItem(Item item)
+{
+ GroupItem group;
+
+ if (!item->parent) {
+ return;
+ }
+ group = (GroupItem) item->parent;
+
+ if (item->previous != RADAR_NO_ITEM) {
+ item->previous->next = item->next;
+ }
+ else {
+ group->head = item->next;
+ }
+
+ if (item->next != RADAR_NO_ITEM) {
+ item->next->previous = item->previous;
+ }
+ else {
+ group->tail = item->previous;
+ }
+
+ item->previous = RADAR_NO_ITEM;
+ item->next = RADAR_NO_ITEM;
+ item->parent = NULL;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * UpdateItemPriority --
+ * Reorder a group's item list following a change in an
+ * item priority or a call to lower/raise.
+ *
+ **********************************************************************************
+ */
+static void
+UpdateItemPriority(Item item,
+ Item mark_item,
+ RadarBool before)
+{
+ Item parent = item->parent;
+
+ ExtractItem(item);
+ InsertItem(item, parent, mark_item, before);
+ Invalidate(item, RADAR_DRAW_FLAG);
+ SET(item->wi->events_flags, INTERNAL_NEED_REPICK);
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * SetId,
+ * FreeId --
+ * Get a fresh object id from the widget and enter the new
+ * object with this id in the object hash table. The id is
+ * incremented. FreeId on the other hand suppress the item
+ * from the hash table and set its object id to zero.
+ *
+ **********************************************************************************
+ */
+static void
+SetId(Item item)
+{
+ WidgetInfo *wi = item->wi;
+ Tcl_HashEntry *entry;
+ int dummy;
+
+ item->id = wi->obj_id;
+ wi->obj_id++;
+ entry = Tcl_CreateHashEntry(wi->id_table, (char *) item->id, &dummy);
+ Tcl_SetHashValue(entry, item);
+}
+
+static void
+FreeId(Item item)
+{
+ Tcl_HashEntry *entry;
+
+ if (item->id) {
+ entry = Tcl_FindHashEntry(item->wi->id_table, (char *) item->id);
+ if (entry) {
+ Tcl_DeleteHashEntry(entry);
+ item->id = 0;
+ }
+ }
+}
+
+/*
+ **********************************************************************************
+ *
+ * AddTag --
+ * Add a tag to the item. If the tag is already on the list it
+ * is not added. As a side effect the tag/item pair is added to
+ * the tag table of the widget.
+ * 'tag' must be a Tk_Uid.
+ *
+ **********************************************************************************
+ */
+static void
+AddTag(Item item,
+ Tk_Uid tag)
+{
+ int new_entry, num, i;
+ char **ptr;
+ Tcl_HashEntry *entry;
+ RadarList item_list;
+
+ /*
+ * No tags yet.
+ */
+ if (!item->tags) {
+ item->tags = RadarListNew(1, sizeof(char *));
+ }
+ else {
+ /*
+ * If the tag is already there, that's done.
+ */
+ ptr = (char **) RadarListArray(item->tags);
+ num = RadarListSize(item->tags);
+ for (i = 0; i < num; i++) {
+ if (ptr[i] == tag) {
+ return;
+ }
+ }
+ }
+ /*
+ * Add it.
+ */
+ RadarListAdd(item->tags, &tag, RadarListTail);
+ entry = Tcl_CreateHashEntry(item->wi->tag_table, tag, &new_entry);
+ if (new_entry) {
+ item_list = RadarListNew(1, sizeof(Item));
+ Tcl_SetHashValue(entry, item_list);
+ }
+ else {
+ item_list = (RadarList) Tcl_GetHashValue(entry);
+ }
+ RadarListAdd(item_list, &item, RadarListTail);
+}
+
+/*
+ **********************************************************************************
+ *
+ * RemoveTag --
+ *
+ **********************************************************************************
+ */
+static void
+RemoveTag(Item item,
+ Tk_Uid tag)
+{
+ int indx, num;
+ char **ptr;
+ Item *item_ptr;
+ Tcl_HashEntry *entry;
+ RadarList item_list;
+
+ if (!item->tags) {
+ return;
+ }
+ /*
+ * look up the tag in the list.
+ */
+ ptr = (char **) RadarListArray(item->tags);
+ num = RadarListSize(item->tags);
+ for (indx = 0; indx < num; indx++) {
+ if (ptr[indx] == tag) {
+ /* The tag list is not freed when empty to avoid
+ * overhead when using tags intensively. */
+ RadarListDelete(item->tags, indx);
+ entry = Tcl_FindHashEntry(item->wi->tag_table, tag);
+ if (entry) { /* Should not fail */
+ item_list = (RadarList) Tcl_GetHashValue(entry);
+ num = RadarListSize(item_list);
+ item_ptr = (Item *) RadarListArray(item_list);
+ for (indx = 0; indx < num; indx++) {
+ if (item_ptr[indx] == item) {
+ /* Should not fail to get here */
+ RadarListDelete(item_list, indx);
+ if (RadarListSize(item_list) == 0) {
+ RadarListFree(item_list);
+ Tcl_DeleteHashEntry(entry);
+ }
+ }
+ }
+ }
+ return;
+ }
+ }
+}
+
+/*
+ **********************************************************************************
+ *
+ * FreeTags --
+ *
+ **********************************************************************************
+ */
+static void
+FreeTags(Item item)
+{
+ int num, num_items, i, j;
+ char **ts;
+ Item *item_ptr;
+ Tcl_HashEntry *entry;
+ RadarList item_list;
+
+ if (!item->tags) {
+ return;
+ }
+ num = RadarListSize(item->tags);
+ ts = (char **) RadarListArray(item->tags);
+ for (i = 0; i < num; i++) {
+ entry = Tcl_FindHashEntry(item->wi->tag_table, ts[i]);
+ if (entry) { /* Should not fail */
+ item_list = (RadarList) Tcl_GetHashValue(entry);
+ item_ptr = (Item *) RadarListArray(item_list);
+ num_items = RadarListSize(item_list);
+ for (j = 0; j < num_items; j++) {
+ if (item_ptr[j] == item) {
+ /* Should not fail to get here */
+ RadarListDelete(item_list, j);
+ if (RadarListSize(item_list) == 0) {
+ RadarListFree(item_list);
+ Tcl_DeleteHashEntry(entry);
+ }
+ }
+ }
+ }
+ }
+ RadarListFree(item->tags);
+ item->tags = NULL;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * CreateItem --
+ *
+ * InsertItem and ConfigureItem must be called after CreateItem
+ * to finalize the setup of a new item. This is so even if
+ * there are no attributes to be changed after creation.
+ * ConfigureItem must be called in this case with the 'init'
+ * parameter set to True.
+ *
+ **********************************************************************************
+ */
+static Item
+CreateItem(WidgetInfo *wi,
+ ItemClass item_class,
+ int *argc,
+ Arg **args)
+{
+ Item item;
+
+ item = (Item) RadarMalloc(item_class->item_size);
+
+ /* Initialize common state */
+ item->class = item_class;
+ item->wi = wi;
+ item->parent = NULL;
+ item->previous = RADAR_NO_ITEM;
+ item->next = RADAR_NO_ITEM;
+ CLEAR(item->flags, UPDATE_DEPENDENT_BIT);
+ item->inv_flags = 0;
+ item->transfo = NULL;
+ item->parent = NULL;
+ item->connected_item = RADAR_NO_ITEM;
+
+ ResetBBox(&item->item_bounding_box);
+
+ /* Init item specific attributes */
+ if (item_class->Init(item, argc, args) == RADAR_ERROR) {
+ RadarFree(item);
+ return RADAR_NO_ITEM;
+ }
+ SetId(item);
+ item->tags = NULL;
+
+ SET(wi->events_flags, INTERNAL_NEED_REPICK);
+
+ return (item);
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * CloneItem --
+ * Can't clone the top level group.
+ *
+ **********************************************************************************
+ */
+static Item
+CloneItem(Item model)
+{
+ WidgetInfo *wi = model->wi;
+ Item item;
+ Tk_Uid *tags;
+ int i, num_tags;
+
+ if (!model->parent) {
+ return RADAR_NO_ITEM;
+ }
+
+ item = (Item) RadarMalloc(model->class->item_size);
+ memcpy(item, model, model->class->item_size);
+
+ item->previous = RADAR_NO_ITEM;
+ item->next = RADAR_NO_ITEM;
+ item->connected_item = RADAR_NO_ITEM;
+ CLEAR(item->flags, UPDATE_DEPENDENT_BIT);
+ item->inv_flags = 0;
+ SetId(item);
+
+ if (model->tags) {
+ item->tags = NULL;
+ tags = (Tk_Uid *) RadarListArray(model->tags);
+ num_tags = RadarListSize(model->tags);
+ for (i = num_tags-1; i >= 0; i--, tags++) {
+ AddTag(item, *tags);
+ }
+ }
+
+ if (item->transfo) {
+ item->transfo = RadarTransfoDuplicate(item->transfo);
+ }
+
+ /* Call item's clone duplicate not shared resources */
+ item->class->Clone(item);
+
+ SET(wi->events_flags, INTERNAL_NEED_REPICK);
+ return item;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ConfigureItem --
+ *
+ **********************************************************************************
+ */
+static int
+ConfigureItem(Item item,
+ int field,
+ int argc,
+ RadarAttrList argv,
+ RadarBool init)
+{
+ WidgetInfo *wi = item->wi;
+ int flags;
+ RadarBool previous_visible = init ? False : ISSET(item->flags, VISIBLE_BIT);
+
+ flags = 0;
+ ASSIGN(flags, RADAR_COORDS_FLAG, init);
+ if (argv) {
+ if (field < 0){
+ if (item->class->Configure(item, argc, argv, &flags) == RADAR_ERROR) {
+ return RADAR_ERROR;
+ }
+ if (item->class->has_fields && ISSET(flags, RADAR_CLFC_FLAG)) {
+ ClearFieldCache(item->class->GetFieldSet(item), -1);
+ }
+ }
+ else if (item->class->has_fields) {
+ FieldSet field_set = item->class->GetFieldSet(item);
+ if (ConfigureField(field_set, field, argc, argv, &flags) == RADAR_ERROR) {
+ return RADAR_ERROR;
+ }
+ }
+ else {
+ return RADAR_ERROR;
+ }
+ }
+
+ if (previous_visible && ISCLEAR(item->flags, VISIBLE_BIT)) {
+ /*
+ * Special case when the item has its visibility
+ * just turned out.
+ */
+ Damage(wi, &item->item_bounding_box);
+ RadarNeedRedisplay(wi);
+ }
+
+ Invalidate(item, flags);
+
+ return RADAR_OK;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * QueryItem --
+ *
+ **********************************************************************************
+ */
+static int
+QueryItem(Item item,
+ int field,
+ int argc,
+ RadarAttrList argv)
+{
+ if (field < 0) {
+ return item->class->Query(item, argc, argv);
+ }
+ else if (item->class->has_fields) {
+ FieldSet field_set = item->class->GetFieldSet(item);
+ return QueryField(field_set, field, argc, argv);
+ }
+
+ return RADAR_ERROR;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ComposeItemTransform --
+ * Compose the item transform with current_t transform in new_t.
+ *
+ **********************************************************************************
+ */
+static void
+ComposeItemTransform(Item item,
+ RadarTransfo *current_t,
+ RadarTransfo *new_t)
+{
+ RadarBool full;
+
+ full = (ISSET(item->flags, COMPOSE_SCALE_BIT) &&
+ ISSET(item->flags, COMPOSE_ROTATION_BIT));
+
+ if (!item->transfo && full) {
+ *new_t = *current_t;
+ return;
+ }
+ if (full) {
+ /*
+ * Full concatenation.
+ */
+ /*printf("Item transfo for item: 0x%X;", item); RadarPrintTransfo(item->transfo);*/
+ RadarTransfoCompose(new_t, item->transfo, current_t);
+ }
+ else {
+ RadarPoint scale, trans, local_scale, local_trans;
+ RadarReal local_rot, rot;
+ RadarTransfo t, t2;
+
+ /*
+ * Need to decompose the local transform in translation,
+ * rotation and scale.
+ */
+ RadarTransfoSetIdentity(&t);
+ RadarTransfoSetIdentity(new_t);
+ RadarTransfoDecompose(item->transfo, &local_scale, &local_trans, &local_rot, NULL);
+ RadarTranslate(&t, local_trans.x, local_trans.y);
+ RadarTransfoCompose(&t2, &t, current_t);
+ RadarTransfoDecompose(&t2, &scale, &trans, &rot, NULL);
+ if (ISSET(item->flags, COMPOSE_SCALE_BIT)) {
+ RadarScale(new_t, scale.x, scale.y);
+ }
+ if (ISSET(item->flags, COMPOSE_ROTATION_BIT)) {
+ RadarRotateRad(new_t, rot);
+ }
+ RadarScale(new_t, local_scale.x, local_scale.y);
+ RadarRotateRad(new_t, local_rot);
+ RadarTranslate(new_t, trans.x, trans.y);
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * GetTransform --
+ * Compute the current transform for an item.
+ *
+ **********************************************************************************
+ */
+static void
+GetTransform(Item item,
+ RadarTransfo *t)
+{
+ Item *items;
+ int i;
+ RadarTransfo t_tmp, *t1, *t2, *swap;
+
+ if (item_stack == NULL) {
+ item_stack = RadarListNew(16, sizeof(Item));
+ }
+ else {
+ RadarListEmpty(item_stack);
+ }
+
+ while (item != RADAR_NO_ITEM) {
+ RadarListAdd(item_stack, &item, RadarListTail);
+ item = item->parent;
+ }
+
+ RadarTransfoSetIdentity(t);
+ t1 = t;
+ t2 = &t_tmp;
+ items = (Item *) RadarListArray(item_stack);
+ for (i = RadarListSize(item_stack)-1; i >= 0; i--) {
+ ComposeItemTransform(items[i], t1, t2);
+ swap = t2;
+ t2 = t1;
+ t1 = swap;
+ }
+ if (t1 != t) {
+ *t = *t1;
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ResetTransformStack --
+ * Empty the transform stack and restore the first transform on the stack.
+ *
+ **********************************************************************************
+ */
+static void
+ResetTransformStack(WidgetInfo *wi)
+{
+ RadarListAssertSize(wi->transfo_stack, 1);
+ wi->current_transfo = (RadarTransfo *) RadarListAt(wi->transfo_stack, 0);
+ RadarTransfoSetIdentity(wi->current_transfo);
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ResetClipStack --
+ *
+ **********************************************************************************
+ */
+static void
+ResetClipStack(WidgetInfo *wi)
+{
+ int i;
+ ClipState *clips = (ClipState *) RadarListArray(wi->clip_stack);
+
+ /*
+ * Should not happen, clip stack should be
+ * empty when this function is called.
+ */
+ for (i = RadarListSize(wi->clip_stack)-1; i >= 0; i--) {
+ XDestroyRegion(clips[i].region);
+ }
+ RadarListEmpty(wi->clip_stack);
+ wi->current_clip = NULL;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * Invalidate --
+ *
+ **********************************************************************************
+ */
+static void
+Invalidate(Item item,
+ int reason)
+{
+ if (ISSET(item->inv_flags, RADAR_TRANSFO_FLAG)) {
+ return;
+ }
+
+ if (ISSET(reason, RADAR_COORDS_FLAG) ||
+ ISSET(reason, RADAR_TRANSFO_FLAG)) {
+ Item parent = item->parent;
+ while ((parent != NULL) &&
+ ISCLEAR(parent->inv_flags, RADAR_COORDS_FLAG) &&
+ ISCLEAR(parent->inv_flags, RADAR_TRANSFO_FLAG)) {
+ SET(parent->inv_flags, RADAR_COORDS_FLAG);
+ /*printf("invalidate coords for parent %d\n", parent->id);*/
+ parent = parent->parent;
+ }
+ /*
+ * There is no need to set the DRAW flag to force the invalidation
+ * of the current bounding box. This will be done by ComputeCoordinates
+ * in Group.
+ */
+ item->inv_flags |= reason;
+ /*printf("invalidate %s for item %d, flags %s\n",
+ ISSET(reason, RADAR_TRANSFO_FLAG)?"TRANSFO":"COORDS", item->id,
+ ISSET(item->inv_flags, RADAR_TRANSFO_FLAG)?"TRANSFO":"COORDS");*/
+ RadarNeedRedisplay(item->wi);
+ }
+ else if (ISSET(reason, RADAR_DRAW_FLAG)) {
+ if (ISSET(item->flags, VISIBLE_BIT)) {
+ /*printf("invalidate graphics for item %d\n", item->id);*/
+ Damage(item->wi, &item->item_bounding_box);
+ RadarNeedRedisplay(item->wi);
+ }
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * InvalidateItems --
+ * Invalidate the geometric state of all items of a group
+ * belonging to a given class.
+ *
+ **********************************************************************************
+ */
+static void
+InvalidateItems(Item group,
+ ItemClass item_class)
+{
+ Item item;
+
+ if (group->class != RadarGroup) {
+ return;
+ }
+ item = ((GroupItem) group)->head;
+ while (item != RADAR_NO_ITEM) {
+ if (item->class == item_class) {
+ Invalidate(item, RADAR_COORDS_FLAG);
+ }
+ item = item->next;
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ResetTransfo
+ * SetTransfo
+ * TranslateItem
+ * ScaleItem
+ * RotateItem --
+ * Set of functions that deal with item transform. They take care
+ * of all details including managing NULL transforms and invalidating
+ * the item hierarchy.
+ *
+ **********************************************************************************
+ */
+static void
+ResetTransfo(Item item)
+{
+ if (item->transfo) {
+ RadarFree(item->transfo);
+ item->transfo = NULL;
+ }
+ Invalidate(item, RADAR_TRANSFO_FLAG);
+}
+
+
+static void
+SetTransfo(Item item,
+ RadarTransfo *t)
+{
+ if (item->transfo) {
+ RadarFree(item->transfo);
+ }
+ if (!t || RadarTransfoIsIdentity(t)) {
+ item->transfo = NULL;
+ }
+ else {
+ item->transfo = RadarTransfoDuplicate(t);
+ }
+ Invalidate(item, RADAR_TRANSFO_FLAG);
+}
+
+
+static void
+TranslateItem(Item item,
+ RadarReal dx,
+ RadarReal dy)
+{
+ if (!item->transfo) {
+ item->transfo = RadarTransfoNew();
+ }
+ RadarTranslate(item->transfo, dx, dy);
+ Invalidate(item, RADAR_TRANSFO_FLAG);
+}
+
+
+static void
+ScaleItem(Item item,
+ RadarReal sx,
+ RadarReal sy)
+{
+ if (!item->transfo) {
+ item->transfo = RadarTransfoNew();
+ }
+ RadarScale(item->transfo, sx, sy);
+ Invalidate(item, RADAR_TRANSFO_FLAG);
+}
+
+
+static void
+RotateItem(Item item,
+ RadarReal angle,
+ RadarPoint *p)
+{
+ if (!item->transfo) {
+ item->transfo = RadarTransfoNew();
+ }
+ if (p) {
+ RadarTranslate(item->transfo, -p->x, -p->y);
+ }
+ RadarRotateRad(item->transfo, angle);
+ if (p) {
+ RadarTranslate(item->transfo, p->x, p->y);
+ }
+
+ Invalidate(item, RADAR_TRANSFO_FLAG);
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * RemoveItem --
+ * Extract an item from its context, includes updating graphic
+ * state flags.
+ *
+ **********************************************************************************
+ */
+static void
+RemoveItem(Item item)
+{
+ WidgetInfo *wi = item->wi;
+ GroupItem group;
+
+ group = (GroupItem) item->parent;
+
+ /* damage bounding boxes */
+ if (ISSET(item->flags, VISIBLE_BIT)) {
+ Damage(wi, &item->item_bounding_box);
+ RadarNeedRedisplay(wi);
+ }
+
+ /*
+ * Tell that we need to repick
+ */
+ if (item->class != RadarGroup) {
+ SET(wi->events_flags, INTERNAL_NEED_REPICK);
+ }
+
+ if (group) {
+ if (group->dependents) {
+ /* Remove me from dependency list. */
+ ExtractDependentItem(item);
+
+ /* Disconnect all dependents on me. */
+ DisconnectDependentItems(item);
+ }
+ /*
+ * Remove me from item list.
+ */
+ ExtractItem(item);
+ /*
+ * Remove me as a clip item.
+ */
+ if (group->clip == item) {
+ group->clip = RADAR_NO_ITEM;
+ Invalidate((Item) group, RADAR_COORDS_FLAG);
+ }
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * DestroyItem --
+ *
+ **********************************************************************************
+ */
+static void
+DestroyItem(Item item)
+{
+ WidgetInfo *wi = item->wi;
+
+ /*
+ * Extract it from its group.
+ */
+ RemoveItem(item);
+
+ /*
+ * Update state variables to prevent dangling pointers.
+ */
+ if (wi->current_item == item) {
+ wi->current_item = RADAR_NO_ITEM;
+ wi->current_part = RADAR_NO_PART;
+ }
+ if (wi->new_item == item) {
+ wi->new_item = RADAR_NO_ITEM;
+ wi->new_part = RADAR_NO_PART;
+ }
+
+ /*
+ * Call per class removal code.
+ */
+ (item->class->Destroy)(item);
+ /*
+ * Free the transform if any.
+ */
+ if (item->transfo) {
+ RadarFree(item->transfo);
+ }
+ /*
+ * Remove the item from the item table and free
+ * all its tags.
+ */
+ FreeId(item);
+ FreeTags(item);
+ /*
+ * Free the item own memory
+ */
+ RadarFree(item);
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * FieldImageChange --
+ *
+ **********************************************************************************
+ */
+static void
+FieldImageChange(ClientData client_data,
+ int x,
+ int y,
+ int width,
+ int height,
+ int image_width,
+ int image_height)
+{
+ Field field = (Field) client_data;
+ int i;
+ RadarBBox bbox;
+
+ if (field->image != RadarUnspecifiedImage) {
+ i = (((char *) field) - ((char *) field->field_set->fields)) / sizeof(FieldStruct);
+ GetFieldBBox(field->field_set, i, &bbox);
+ Damage(field->field_set->wi, &bbox);
+ RadarNeedRedisplay(field->field_set->wi);
+ ClearFieldCache(field->field_set, i);
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * FieldTileChange --
+ *
+ **********************************************************************************
+ */
+static void
+FieldTileChange(ClientData client_data,
+ int x,
+ int y,
+ int width,
+ int height,
+ int image_width,
+ int image_height)
+{
+ Field field = (Field) client_data;
+ int i;
+ RadarBBox bbox;
+
+ if (field->tile != RadarUnspecifiedImage) {
+ i = (((char *) field) - ((char *) field->field_set->fields)) / sizeof(FieldStruct);
+ InvalidateImage(field->tile);
+ GetFieldBBox(field->field_set, i, &bbox);
+ Damage(field->field_set->wi, &bbox);
+ RadarNeedRedisplay(field->field_set->wi);
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * InitFields --
+ *
+ * Perform the init of each field in a FieldSet. The number of such
+ * fields must have been inited before calling this fun.
+ *
+ **********************************************************************************
+ */
+static void
+InitFields(FieldSet field_set)
+{
+ WidgetInfo *wi = field_set->wi;
+ Field field;
+ unsigned int i, num_fields;
+
+ /*printf("size of a field = %d\n", sizeof(FieldStruct));*/
+
+ if (field_set->num_fields < 0) {
+ field_set->num_fields = 0;
+ }
+ if (!field_set->num_fields) {
+ return;
+ }
+ num_fields = field_set->num_fields;
+ field_set->fields = (Field) RadarMalloc(num_fields*sizeof(FieldStruct));
+
+ for (i = 0; i < num_fields; i++){
+ field = &field_set->fields[i];
+
+ field->field_set = field_set;
+ field->color = RadarGetColorByValue(wi->win, wi->fore_color);
+ field->back_color = RadarGetColorByValue(wi->win, wi->back_color);
+ field->border_color = RadarGetColorByValue(wi->win, wi->fore_color);
+ SET(field->flags, FIELD_VISIBLE_BIT);
+ SET(field->flags, FIELD_SENSITIVE_BIT);
+ CLEAR(field->flags, FILLED_BIT);
+ CLEAR(field->flags, CACHE_OK);
+ field->fill_pattern = RadarUnspecifiedPattern;
+ field->text = "";
+ field->image = RadarUnspecifiedImage;
+ field->image_name = "";
+ field->tile = RadarUnspecifiedImage;
+ field->tile_name = "";
+ field->font = Tk_GetFont(wi->interp, wi->win, Tk_NameOfFont(wi->font));
+ field->border_edges = NO_BORDER;
+ field->alignment = RadarJustifyLeft;
+ field->auto_alignment.automatic = False;
+
+ field->relief = RELIEF_FLAT;
+ field->relief_thickness = 2;
+ SET(field->flags, TEXT_ON_TOP_BIT);
+
+ field->gradient = NULL;
+ }
+ field_set->label_pos.x = field_set->label_pos.y = 0.0;
+ field_set->label_width = field_set->label_height = -1.0;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * CloneFields --
+ *
+ **********************************************************************************
+ */
+static void
+CloneFields(FieldSet field_set)
+{
+ WidgetInfo *wi = field_set->wi;
+ Field field, fields_ret;
+ unsigned int i, num_fields;
+ char *text;
+
+ if (field_set->label_format) {
+ field_set->label_format = LabelFormatDuplicate(field_set->label_format);
+ }
+ num_fields = field_set->num_fields;
+ if (!num_fields) {
+ return;
+ }
+ fields_ret = (Field) RadarMalloc(num_fields*sizeof(FieldStruct));
+ memcpy(fields_ret, field_set->fields, num_fields*sizeof(FieldStruct));
+ field_set->fields = fields_ret;
+
+ for (i = 0; i < num_fields; i++) {
+ field = &fields_ret[i];
+ if (field->gradient) {
+ field->gradient = RadarGetColorGradientByValue(field->gradient);
+ }
+
+ if (strlen(field->image_name) != 0) {
+ text = RadarMalloc((strlen(field->image_name) + 1) * sizeof(char));
+ strcpy(text, field->image_name);
+ field->image_name = text;
+ field->image = Tk_GetImage(wi->interp, wi->win, field->image_name,
+ FieldImageChange, (ClientData) field);
+ }
+ if (strlen(field->tile_name) != 0) {
+ text = RadarMalloc((strlen(field->tile_name) + 1) * sizeof(char));
+ strcpy(text, field->tile_name);
+ field->tile_name = text;
+ field->tile = Tk_GetImage(wi->interp, wi->win, field->tile_name,
+ FieldTileChange, (ClientData) field);
+ }
+ if (field->fill_pattern != RadarUnspecifiedPattern) {
+ field->fill_pattern = Tk_GetBitmap(wi->interp, wi->win,
+ Tk_NameOfBitmap(wi->dpy, field->fill_pattern));
+ }
+ field->font = Tk_GetFont(wi->interp, wi->win, Tk_NameOfFont(field->font));
+ field->color = RadarGetColorByValue(wi->win, field->color);
+ field->back_color = RadarGetColorByValue(wi->win, field->back_color);
+ field->border_color = RadarGetColorByValue(wi->win, field->border_color);
+
+ if (strlen(field->text) != 0) {
+ text = (char *) RadarMalloc((strlen(field->text) + 1) * sizeof(char));
+ strcpy(text, field->text);
+ field->text = text;
+ }
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ConfigureField --
+ *
+ **********************************************************************************
+ */
+static int
+ConfigureField(FieldSet field_set,
+ unsigned int field,
+ int argc,
+ RadarAttrList argv,
+ int *flags)
+{
+ int i;
+ Field field_ptr;
+ RadarBBox bbox;
+ WidgetInfo *wi = field_set->wi;
+
+ if (field >= field_set->num_fields) {
+ return RADAR_ERROR;
+ }
+
+ field_ptr = &field_set->fields[field];
+
+ if (ConfigureAttributes((char *) field_set, field, argc, argv, flags) == RADAR_ERROR) {
+ return RADAR_ERROR;
+ }
+
+ if (ISSET(*flags, RADAR_REPICK_FLAG)) {
+ SET(wi->events_flags, INTERNAL_NEED_REPICK);
+ }
+ if (ISSET(*flags, RADAR_CLFC_FLAG)) {
+ ClearFieldCache(field_set, field);
+ }
+
+ if (field_ptr->gradient &&
+ (ISSET(*flags, RADAR_BORDER_FLAG) || (field_ptr->relief == RELIEF_FLAT))) {
+ RadarFreeColorGradient(field_ptr->gradient);
+ field_ptr->gradient = NULL;
+ }
+ if ((field_ptr->relief != RELIEF_FLAT) && !field_ptr->gradient) {
+ field_ptr->gradient = RadarGetReliefGradient(wi->interp, wi->win,
+ RadarNameOfColor(field_ptr->back_color));
+ }
+
+ if (ISSET(*flags, RADAR_IMAGE_FLAG)) {
+ Tk_Image image;
+
+ if (strcmp(field_ptr->image_name, "") != 0) {
+ image = Tk_GetImage(wi->interp, wi->win, field_ptr->image_name,
+ FieldImageChange, (ClientData) field_ptr);
+ if (image == NULL) {
+ /*
+ * The name will not be in sync with the image in
+ * this case.
+ */
+ return RADAR_ERROR;
+ }
+ }
+ else {
+ image = RadarUnspecifiedImage;
+ }
+ if (field_ptr->image != RadarUnspecifiedImage) {
+ Tk_FreeImage(field_ptr->image);
+ }
+ field_ptr->image = image;
+ }
+
+ if (ISSET(*flags, RADAR_TILE_FLAG)) {
+ Tk_Image tile;
+
+ if (strcmp(field_ptr->tile_name, "") != 0) {
+ tile = Tk_GetImage(wi->interp, wi->win, field_ptr->tile_name,
+ FieldTileChange, (ClientData) field_ptr);
+ if (tile == NULL) {
+ return RADAR_ERROR;
+ }
+ }
+ else {
+ tile = RadarUnspecifiedImage;
+ }
+ if (field_ptr->tile != RadarUnspecifiedImage) {
+ Tk_FreeImage(field_ptr->tile);
+ }
+ field_ptr->tile = tile;
+ }
+
+ /*
+ * This is done here to limit the redraw to the area of the
+ * modified fields.
+ */
+ if (ISCLEAR(*flags, RADAR_COORDS_FLAG) &&
+ field_set->label_format && ISSET(*flags, RADAR_DRAW_FLAG)) {
+ for (i = 0; i < LabelFormatNumFields(field_set->label_format); i++) {
+ if (i == field) {
+ GetFieldBBox(field_set, i, &bbox);
+ Damage(wi, &bbox);
+ break;
+ }
+ }
+ }
+
+ return RADAR_OK;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * QueryField --
+ *
+ **********************************************************************************
+ */
+static int
+QueryField(FieldSet field_set,
+ unsigned int field,
+ int argc,
+ RadarAttrList argv)
+{
+ Field field_ptr;
+
+ if (field >= field_set->num_fields) {
+ return RADAR_ERROR;
+ }
+
+ field_ptr = &field_set->fields[field];
+ if (QueryAttribute((char *) field_set, field, argv[0]) == RADAR_ERROR) {
+ return RADAR_ERROR;
+ }
+
+ return RADAR_OK;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * FreeFields --
+ *
+ **********************************************************************************
+ */
+static void
+FreeFields(FieldSet field_set)
+{
+ WidgetInfo *wi = field_set->wi;
+ unsigned int i, num_fields;
+ Field field;
+
+ if (field_set->label_format) {
+ LabelFormatDelete(field_set->label_format);
+ }
+
+ num_fields = field_set->num_fields;
+ for (i = 0; i < num_fields; i++) {
+ field = &field_set->fields[i];
+
+ if (strlen(field->text) != 0) {
+ RadarFree(field->text);
+ }
+ if (field->gradient) {
+ RadarFreeColorGradient(field->gradient);
+ }
+ if (strlen(field->image_name) != 0) {
+ RadarFree(field->image_name);
+ }
+ if (field->image != RadarUnspecifiedImage) {
+ Tk_FreeImage(field->image);
+ field->image = RadarUnspecifiedImage;
+ }
+ if (strlen(field->tile_name) != 0) {
+ RadarFree(field->tile_name);
+ }
+ if (field->tile != RadarUnspecifiedImage) {
+ Tk_FreeImage(field->tile);
+ field->tile = RadarUnspecifiedImage;
+ }
+ if (field->fill_pattern != RadarUnspecifiedPattern) {
+ Tk_FreeBitmap(wi->dpy, field->fill_pattern);
+ field->fill_pattern = RadarUnspecifiedPattern;
+ }
+ /*printf("freeing a font\n");*/
+ Tk_FreeFont(field->font);
+ RadarFreeColor(field->color);
+ RadarFreeColor(field->back_color);
+ RadarFreeColor(field->border_color);
+ }
+ if (num_fields) {
+ RadarFree(field_set->fields);
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * ComputeFieldImageLocation --
+ * Compute the bounding box of the pixmap in a field. The position is
+ * deduced from the field bounding box passed in bbox.
+ *
+ **********************************************************************************
+ */
+static void
+ComputeFieldImageLocation(Field field_ptr,
+ RadarBBox *bbox,
+ RadarBBox *pm_bbox)
+{
+ RadarBool relief_thickness;
+ int width, height;
+
+ Tk_SizeOfImage(field_ptr->image, &width, &height);
+ pm_bbox->orig.y = (bbox->orig.y + bbox->corner.y - height) / 2;
+ pm_bbox->corner.y = pm_bbox->orig.y + height;
+ relief_thickness = (field_ptr->relief != RELIEF_FLAT &&
+ field_ptr->relief_thickness > 1) ? field_ptr->relief_thickness : 0;
+
+ switch (field_ptr->alignment) {
+ case RadarJustifyLeft:
+ pm_bbox->orig.x = bbox->orig.x + 1 + relief_thickness;
+ break;
+ case RadarJustifyRight:
+ pm_bbox->orig.x = bbox->corner.x - width - 1 - relief_thickness;
+ break;
+ default:
+ pm_bbox->orig.x = (bbox->orig.x + bbox->corner.x - width) / 2;
+ break;
+ }
+ pm_bbox->corner.x = pm_bbox->orig.x + width;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * DrawFields --
+ *
+ **********************************************************************************
+ */
+static void
+DrawFields(FieldSet field_set)
+{
+ WidgetInfo *wi = field_set->wi;
+ int i; /* This one *NEED* to be an int */
+ int j, num;
+ Field field_ptr;
+ XGCValues values;
+ RadarBBox label_clip_box, clip_bbox, bbox, *global_clip_box;
+ RadarBBox text_bbox, clip_text_bbox;
+ XRectangle r;
+ Region reg, reg2;
+ RadarPoint text_pos;
+ RadarBBox pm_bbox, clip_pm_bbox;
+ RadarBool restore = False;
+ RadarDim lwidth, lheight;
+
+ if (field_set->label_format && LabelFormatNumFields(field_set->label_format)) {
+ bbox.orig = field_set->label_pos;
+ GetLabelBBox(field_set, &lwidth, &lheight);
+ bbox.corner.x = field_set->label_pos.x + lwidth;
+ bbox.corner.y = field_set->label_pos.y + lheight;
+ if (wi->current_clip) {
+ global_clip_box = &wi->current_clip->clip_box;
+ }
+ else {
+ global_clip_box = &wi->damaged_area;
+ }
+ IntersectBBox(global_clip_box, &bbox, &label_clip_box);
+ if (IsEmptyBBox(&label_clip_box)) {
+ return;
+ }
+
+ num = LabelFormatNumFields(field_set->label_format);
+ for (i = 0; i < num; i++) {
+ field_ptr = &field_set->fields[i];
+
+ if (ISCLEAR(field_ptr->flags, FIELD_VISIBLE_BIT)) {
+ continue;
+ }
+
+ GetFieldBBox(field_set, i, &bbox);
+ IntersectBBox(&label_clip_box, &bbox, &clip_bbox);
+ if (IsEmptyBBox(&clip_bbox)) {
+ continue;
+ }
+
+ if (field_ptr->text) {
+ ComputeFieldTextLocation(field_ptr, &bbox, &text_pos, &text_bbox);
+ }
+ else {
+ ResetBBox(&text_bbox);
+ }
+
+ IntersectBBox(&clip_bbox, &text_bbox, &clip_text_bbox);
+
+ if (field_ptr->image != RadarUnspecifiedImage) {
+ ComputeFieldImageLocation(field_ptr, &bbox, &pm_bbox);
+ }
+ else {
+ ResetBBox(&pm_bbox);
+ }
+
+ IntersectBBox(&clip_bbox, &pm_bbox, &clip_pm_bbox);
+
+ /* we must call XSetClipRectangles only if it's required */
+ if (clip_bbox.orig.x != bbox.orig.x || clip_bbox.orig.y != bbox.orig.y ||
+ clip_bbox.corner.x != bbox.corner.x || clip_bbox.corner.y != bbox.corner.y ||
+ clip_text_bbox.orig.x != text_bbox.orig.x ||
+ clip_text_bbox.orig.y != text_bbox.orig.y ||
+ clip_text_bbox.corner.x != text_bbox.corner.x ||
+ clip_text_bbox.corner.y != text_bbox.corner.y ||
+ clip_pm_bbox.orig.x != pm_bbox.orig.x || clip_pm_bbox.orig.y != pm_bbox.orig.y ||
+ clip_pm_bbox.corner.x != pm_bbox.corner.x ||
+ clip_pm_bbox.corner.y != pm_bbox.corner.y) {
+ /* we must clip. */
+ BBox2XRect(&clip_bbox, &r);
+ /*printf("must clip to %g %g %g %g, clip rect %d %d %d %d\n",
+ clip_bbox.orig.x, clip_bbox.orig.y,
+ clip_bbox.corner.x, clip_bbox.corner.y,
+ r.x, r.y, r.x+r.width, r.y+r.height);*/
+ reg = XCreateRegion();
+ XUnionRectWithRegion(&r, reg, reg);
+ if (wi->current_clip) {
+ reg2 = XCreateRegion();
+ XIntersectRegion(reg, wi->current_clip->region, reg2);
+ XDestroyRegion(reg);
+ reg = reg2;
+ }
+ XSetRegion(wi->dpy, wi->gc, reg);
+ XDestroyRegion(reg);
+ restore = True;
+ }
+ else {
+ if (restore) {
+ /* Restore the previous clip. */
+ if (wi->current_clip) {
+ XSetRegion(wi->dpy, wi->gc, wi->current_clip->region);
+ }
+ else {
+ XSetRegion(wi->dpy, wi->gc, wi->damaged_region);
+ }
+ restore = False;
+ }
+ }
+
+ /*
+ * Draw the background.
+ */
+ if (ISSET(field_ptr->flags, FILLED_BIT)) {
+ values.foreground = RadarPixel(field_ptr->back_color);
+
+ if (field_ptr->tile != RadarUnspecifiedImage) { /* Fill tiled */
+ Pixmap pmap = GetImagePixmap(wi->win, field_ptr->tile);
+ values.fill_style = FillTiled;
+ values.tile = pmap;
+ values.ts_x_origin = (int) bbox.orig.x;
+ values.ts_y_origin = (int) bbox.orig.y;
+ XChangeGC(wi->dpy, wi->gc,
+ GCTileStipXOrigin|GCTileStipYOrigin|GCFillStyle|GCTile,
+ &values);
+ }
+ else if (field_ptr->fill_pattern != RadarUnspecifiedPattern) { /* Fill stippled */
+ values.fill_style = FillStippled;
+ values.stipple = field_ptr->fill_pattern;
+ values.ts_x_origin = (int) bbox.orig.x;
+ values.ts_y_origin = (int) bbox.orig.y;
+ XChangeGC(wi->dpy, wi->gc,
+ GCTileStipXOrigin|GCTileStipYOrigin|GCFillStyle|GCStipple|GCForeground,
+ &values);
+ }
+ else { /* Fill solid */
+ values.fill_style = FillSolid;
+ XChangeGC(wi->dpy, wi->gc, GCForeground|GCFillStyle, &values);
+ }
+ BBox2XRect(&clip_bbox, &r);
+ XFillRectangle(wi->dpy, wi->draw_buffer, wi->gc, r.x, r.y, r.width, r.height);
+ }
+
+ /*
+ * Draw the image and the text, whichever is in back depends on
+ * the value of text_on_top.
+ */
+ for (j = 0; j < 2; j++) {
+ if ((j == 0 && ISSET(field_ptr->flags, TEXT_ON_TOP_BIT)) ||
+ (j == 1 && ISCLEAR(field_ptr->flags, TEXT_ON_TOP_BIT))) {
+ /*
+ * Draw the image.
+ */
+ if (field_ptr->image != RadarUnspecifiedImage) {
+ Tk_RedrawImage(field_ptr->image,
+ clip_pm_bbox.orig.x-pm_bbox.orig.x,
+ clip_pm_bbox.orig.y-pm_bbox.orig.y,
+ REAL_TO_INT(clip_pm_bbox.corner.x-clip_pm_bbox.orig.x),
+ REAL_TO_INT(clip_pm_bbox.corner.y-clip_pm_bbox.orig.y),
+ wi->draw_buffer,
+ clip_pm_bbox.orig.x, clip_pm_bbox.orig.y);
+ }
+ }
+ else {
+ /*
+ * Draw the text.
+ */
+ if (field_ptr->text && strlen(field_ptr->text)) {
+ values.foreground = RadarPixel(field_ptr->color);
+ values.fill_style = FillSolid;
+ values.font = RadarFontId(field_ptr->font);
+ XChangeGC(wi->dpy, wi->gc, GCForeground | GCFillStyle | GCFont, &values);
+ Tk_DrawChars(wi->dpy, wi->draw_buffer, wi->gc, field_ptr->font,
+ field_ptr->text, strlen(field_ptr->text),
+ text_pos.x, text_pos.y);
+ }
+ }
+ }
+
+ BBox2XRect(&bbox, &r);
+ /*
+ * Draw the border relief.
+ */
+ if ((field_ptr->relief != RELIEF_FLAT) && (field_ptr->relief_thickness > 1)) {
+ DrawRectangleRelief(wi, field_ptr->relief, field_ptr->gradient,
+ &r, (unsigned int) field_ptr->relief_thickness);
+ }
+
+ /*
+ * Draw the border line.
+ */
+ if (field_ptr->border_edges != NO_BORDER) {
+ values.foreground = RadarPixel(field_ptr->border_color);
+ values.line_width = 0;
+ values.line_style = LineSolid;
+ values.fill_style = FillSolid;
+ XChangeGC(wi->dpy, wi->gc,
+ GCForeground | GCLineWidth | GCLineStyle | GCFillStyle, &values);
+ if (field_ptr->border_edges & LEFT_BORDER) {
+ XDrawLine(wi->dpy, wi->draw_buffer, wi->gc, r.x, r.y,
+ r.x, r.y + r.height - 1);
+ }
+ if (field_ptr->border_edges & RIGHT_BORDER) {
+ XDrawLine(wi->dpy, wi->draw_buffer, wi->gc,
+ r.x + r.width - 1, r.y,
+ r.x + r.width - 1, r.y + r.height - 1);
+ }
+ if (field_ptr->border_edges & TOP_BORDER) {
+ XDrawLine(wi->dpy, wi->draw_buffer, wi->gc,
+ r.x, r.y, r.x + r.width - 1, r.y);
+ }
+ if (field_ptr->border_edges & BOTTOM_BORDER) {
+ XDrawLine(wi->dpy, wi->draw_buffer, wi->gc,
+ r.x, r.y + r.height - 1,
+ r.x + r.width - 1, r.y + r.height - 1);
+ }
+ if (field_ptr->border_edges & OBLIQUE) {
+ XDrawLine(wi->dpy, wi->draw_buffer, wi->gc,
+ r.x, r.y, r.x + r.width - 1, r.y + r.height - 1);
+ }
+ if (field_ptr->border_edges & COUNTER_OBLIQUE) {
+ XDrawLine(wi->dpy, wi->draw_buffer, wi->gc,
+ r.x, r.y + r.height - 1,
+ r.x + r.width - 1, r.y);
+ }
+ }
+ }
+
+ if (restore) {
+ /* Restore the previous clip. */
+ if (wi->current_clip) {
+ XSetRegion(wi->dpy, wi->gc, wi->current_clip->region);
+ }
+ else {
+ XSetRegion(wi->dpy, wi->gc, wi->damaged_region);
+ }
+ }
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * IsFieldsSensitive --
+ *
+ **********************************************************************************
+ */
+static RadarBool
+IsFieldSensitive(FieldSet field_set,
+ int part)
+{
+ if ((part >= 0) && (part < field_set->num_fields)) {
+ return ISSET(field_set->fields[part].flags, FIELD_SENSITIVE_BIT);
+ }
+ else {
+ return False;
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * FieldsPick --
+ * Return the first field that contains <x, y>.
+ * A field is selected if <x, y> is over a non transparent area.
+ * Such areas are : a solid filled background, a text, an icon.
+ * This does *NOT* do with *GLOBAL* visible and sensitive.
+ * But we need to take into account local field visible and
+ * sensitive as they modifiy local transparency. Local means
+ * within a single item.
+ *
+ **********************************************************************************
+ */
+static double
+FieldsPick(FieldSet field_set,
+ RadarPoint *p,
+ int *part)
+{
+ Field field_ptr;
+ RadarBBox bbox;
+ int i, best_field = 0;
+ double new_dist, dist = 1e40;
+
+ if (field_set->label_format) {
+ for (i = LabelFormatNumFields(field_set->label_format)-1; i >= 0; i--) {
+ field_ptr = &field_set->fields[i];
+
+ if (ISCLEAR(field_ptr->flags, FIELD_VISIBLE_BIT) &&
+ ISCLEAR(field_ptr->flags, FIELD_SENSITIVE_BIT)) {
+ continue;
+ }
+
+ GetFieldBBox(field_set, i, &bbox);
+
+ new_dist = RectangleToPointDist(&bbox, p);
+ if (new_dist < dist) {
+ dist = new_dist;
+ best_field = i;
+ }
+ if (dist <= 0.0) {
+ dist = 0.0;
+ break;
+ }
+ }
+ }
+
+ *part = best_field;
+ return dist;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * FieldsToArea --
+ * Return -1 if no field is in the given area, 1 if they are
+ * all in it or 0 if there is some overlap. The function consider
+ * only fields that are either sensible or visible.
+ *
+ **********************************************************************************
+ */
+static int
+FieldsToArea(FieldSet field_set,
+ RadarBBox *area)
+{
+ Field field_ptr;
+ RadarBBox bbox;
+ int inside = -1;
+ int i;
+ RadarBool first_done = False;
+
+ for (i = LabelFormatNumFields(field_set->label_format)-1; i >= 0; i--) {
+ field_ptr = &field_set->fields[i];
+
+ if (ISCLEAR(field_ptr->flags, FIELD_VISIBLE_BIT) &&
+ ISCLEAR(field_ptr->flags, FIELD_SENSITIVE_BIT)) {
+ continue;
+ }
+
+ GetFieldBBox(field_set, i, &bbox);
+ if (!first_done) {
+ first_done = True;
+ inside = BBoxInBBox(&bbox, area);
+ if (inside == 0) {
+ return 0;
+ }
+ }
+ else {
+ if (BBoxInBBox(&bbox, area) != inside) {
+ return 0;
+ }
+ }
+ }
+
+ return inside;
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * SetFieldsAutoAlign --
+ *
+ **********************************************************************************
+ */
+static void
+SetFieldsAutoAlign(Item item,
+ int alignment)
+{
+ int i;
+ FieldSet field_set;
+ Field field;
+
+ if (!item->class->has_fields) {
+ return;
+ }
+ if ((alignment >= AA_LEFT) && (alignment <= AA_RIGHT)) {
+ field_set = item->class->GetFieldSet(item);
+ for (i = 0; i < field_set->num_fields; i++) {
+ field = &field_set->fields[i];
+ if (field->auto_alignment.automatic) {
+ field->alignment = field->auto_alignment.align[alignment];
+ }
+ }
+ }
+}
+
+
+static void
+Damage(WidgetInfo *wi,
+ RadarBBox *damage)
+{
+ if ((damage == NULL) || IsEmptyBBox(damage)) {
+ return;
+ }
+
+ /*printf("damaging area: %g %g %g %g\n", damage->orig.x,
+ damage->orig.y, damage->corner.x, damage->corner.y);*/
+
+ if (IsEmptyBBox(&wi->damaged_area)) {
+ wi->damaged_area.orig.x = damage->orig.x;
+ wi->damaged_area.orig.y = damage->orig.y;
+ wi->damaged_area.corner.x = damage->corner.x;
+ wi->damaged_area.corner.y = damage->corner.y;
+ }
+ else {
+ wi->damaged_area.orig.x = MIN(wi->damaged_area.orig.x, damage->orig.x);
+ wi->damaged_area.orig.y = MIN(wi->damaged_area.orig.y, damage->orig.y);
+ wi->damaged_area.corner.x = MAX(wi->damaged_area.corner.x, damage->corner.x);
+ wi->damaged_area.corner.y = MAX(wi->damaged_area.corner.y, damage->corner.y);
+ }
+ if (wi->damaged_area.orig.x < 0) {
+ wi->damaged_area.orig.x = 0;
+ }
+ if (wi->damaged_area.orig.y < 0) {
+ wi->damaged_area.orig.y = 0;
+ }
+ if (wi->damaged_area.corner.x > wi->width) {
+ wi->damaged_area.corner.x = wi->width;
+ }
+ if (wi->damaged_area.orig.y > wi->height) {
+ wi->damaged_area.orig.y = wi->height;
+ }
+ /*printf("damaged area: %g %g %g %g\n", wi->damaged_area.orig.x,
+ wi->damaged_area.orig.y, wi->damaged_area.corner.x,
+ wi->damaged_area.corner.y);*/
+}
+
+
+static void
+Update(WidgetInfo *wi)
+{
+ if (ISSET(wi->top_group->inv_flags, RADAR_COORDS_FLAG) ||
+ ISSET(wi->top_group->inv_flags, RADAR_TRANSFO_FLAG)) {
+ wi->top_group->class->ComputeCoordinates(wi->top_group, False);
+ }
+}
+
+
+static void
+Repair(WidgetInfo *wi)
+{
+ XGCValues values;
+ XRectangle r;
+
+ /* To be done only if we are realized and
+ there is something to update. */
+ if (wi->realized && !IsEmptyBBox(&wi->damaged_area)) {
+
+ /* Set the whole damaged area as clip rect. */
+ /*printf("damaged area %g %g %g %g\n",
+ wi->damaged_area.orig.x, wi->damaged_area.orig.y,
+ wi->damaged_area.corner.x, wi->damaged_area.corner.y);*/
+
+ BBox2XRect(&wi->damaged_area, &r);
+ wi->damaged_region = XCreateRegion();
+ XUnionRectWithRegion(&r, wi->damaged_region, wi->damaged_region);
+
+#ifdef PERFOS
+ XStartChrono(draw_time, wi->dpy, wi->draw_buffer);
+ /*StartChrono(int_draw_time);*/
+#endif
+ /* Fill the background */
+ if (wi->tile == RadarUnspecifiedImage) {
+ values.foreground = RadarPixel(wi->back_color);
+ values.fill_style = FillSolid;
+ XChangeGC(wi->dpy, wi->gc, GCFillStyle|GCForeground, &values);
+ }
+ else {
+ values.fill_style = FillTiled;
+ values.tile = GetImagePixmap(wi->win, wi->tile);
+ values.ts_x_origin = values.ts_y_origin = 0;
+ XChangeGC(wi->dpy, wi->gc,
+ GCFillStyle|GCTile|GCTileStipXOrigin|GCTileStipYOrigin,
+ &values);
+ }
+ /*printf("filling rectangle: %d %d %d %d\n", r.x, r.y, r.width, r.height);*/
+ XFillRectangle(wi->dpy, wi->draw_buffer, wi->gc, r.x, r.y, r.width, r.height);
+
+ /* Draw the items */
+ /*printf("Drawing\n");*/
+ wi->top_group->class->Draw(wi->top_group);
+
+#ifdef PERFOS
+ /*StopChrono(int_draw_time);*/
+ XStopChrono(draw_time, wi->dpy, wi->draw_buffer);
+#endif
+ XDestroyRegion(wi->damaged_region);
+ wi->damaged_region = None;
+ }
+}
+
+
+/*
+ **********************************************************************************
+ *
+ * Exported functions structs --
+ *
+ **********************************************************************************
+ */
+struct _ITEM_P ITEM_P = {
+ GlobalModuleInit,
+ CreateItem,
+ AddItemClass,
+ LookupItemClass,
+ ItemClassList,
+ Damage,
+ Repair,
+ Update,
+ ConfigureAttributes,
+ QueryAttribute,
+ InitFields,
+ CloneFields,
+ FreeFields,
+ DrawFields,
+ FieldsToArea,
+ IsFieldSensitive,
+ FieldsPick,
+ LeaderToLabel,
+ GetLabelBBox,
+ GetFieldBBox,
+ ResetTransformStack,
+ ResetClipStack
+};
+
+struct _ITEM ITEM = {
+ CloneItem,
+ DestroyItem,
+ ConfigureItem,
+ QueryItem,
+ AttributesInfo,
+ SetFieldsAutoAlign,
+ InsertItem,
+ UpdateItemPriority,
+ InsertDependentItem,
+ UpdateItemDependency,
+ RemoveItem,
+ SetId,
+ FreeId,
+ AddTag,
+ RemoveTag,
+ FreeTags,
+ ResetTransfo,
+ SetTransfo,
+ TranslateItem,
+ ScaleItem,
+ RotateItem,
+ Invalidate,
+ InvalidateItems,
+ ComposeItemTransform,
+ GetTransform
+};