Google

i/style

yorick banner

Home

Manual

Packages

Global Index

Keywords

Quick Reference


/*
   STYLE.I
   Get and set graphics styles from within the Yorick interpreter.

   $Id$
 */
/*    Copyright (c) 1996.  The Regents of the University of California.
                    All rights reserved.  */


func get_style (&landscape, &systems, &legends, &clegends)

/* DOCUMENT get_style, landscape, systems, legends, clegends

     get the detailed style of the current drawing.  The arguments
     are all outputs:

     landscape: 1 if drawing is landscape orientation, 0 if portrait

     system:    an array of GfakeSystem struct instances, one per

                coordinate system in this drawing (ordinarily just one).

     legends:   a GeLegendBox structure instance describing the layout
                of the plot legends

     clegends:  a GeLegendBox structure instance describing the layout

                of the contour legends


     See the help for the GeLegendBox and GpTextAttribs structs for
     the details of the legends and clegends arguments.  Basically,
     you can adjust the location of the legends on the page, the

     font and height of the characters used to render legends, and
     whether the legends are split into two columns.

     The coordinate systems are the systems accessible via the

     plsys command.  The index of the system in the system array is

     the index you use to switch to it in the plsys command.  Simple

     styles have only one coordinate system, and you should carefully
     consider whether you should design a graphic style with multiple
     coordinate systems -- most likely, you can do a better job by

     combining several separate Yorick pictures with some sort of
     page layout program, rather than trying to do this work within
     Yorick itself.


     See the help for the GfakeSystem struct for complete details of
     what you can adjust.  The most interesting features you can

     control are the location and aspect ratio of the viewport, and

     the details of the axis ticks and labels.  The gridxy function
     provides a simpler interface for fiddling with ticks and labels
     if that is all you need.  The system.viewport member is the

     [xmin,xmax,ymin,ymax] of the rectangle on the page where your
     plots will appear, expressed in NDC coordinates (0.0013 NDC units
     equals one point, and there are 72.27 points per inch, and 2.54
     cm per inch; the NDC origin is always at the lower left hand
     corner of the paper, with x increasing leftward and y increasing

     upward).  If you change the size of the viewport, you will also
     need to change the parameters of the tick-generating model; like
     other problems in typography and page layout, this is harder
     than you might think.


   SEE ALSO: set_style, read_style, write_style
 */
{
  landscape= [0n];

  n= raw_style(0, landscape, &[], &[]);
  if (!n) error, "no current drawing";

  systems= array(GfakeSystem, n);

  legends= array(GeLegendBox, 2);

  raw_style, 0, landscape, &systems, &legends;
  landscape= landscape(1);
  clegends= legends(2);
  legends= legends(1);
}


func set_style (landscape, systems, legends, clegends)

/* DOCUMENT set_style, landscape, systems, legends, clegends

     set the detailed style of the current drawing.  The arguments

     are all inputs, having the same meanings as for get_style (which

     see).  All arguments are required, so you may need to call

     get_style as a starting point, if you only want to make a few
     changes.  See the Gist/work.gs and the other .gs files for
     examples of reasonable values to choose.


     Calling set_style destroys anything that was plotted in the

     window, like the style= keyword of the window command.


   SEE ALSO: get_style, read_style, write_style
 */
{

  if (structof(systems)!=GfakeSystem || structof(legends)!=GeLegendBox ||

      structof(clegends)!=GeLegendBox || numberof(legends)!=1 ||

      numberof(clegends)!=1 || numberof(landscape)!=1 ||
      structof(landscape+0)!=long)

    error, "illegal input data types or sizes";
  landscape= [int(landscape(1))];

  raw_style, numberof(systems),  landscape, &systems, &[legends,clegends];
}

/* ------------------------------------------------------------------------ */


func write_style (file, landscape, systems, legends, clegends)

/* DOCUMENT write_style, file, landscape, systems, legends, clegends


     write a Gist style sheet (.gs file), using the data structures

     as described in the get_style function.  The FILE can be a
     filename or a text file stream.


   SEE ALSO: get_style, set_style, read_style
 */
{

  if (structof(file)==string) file= create(file);

  write,file,format="# %s\n",

    "Gist style sheet made by Yorick write_style function";

  write,file,format="# Created: %s\n", timestamp();

  write,file,format="# (%ld coordinate systems)\n\n", numberof(systems);


  write,file,format="landscape= %d\n\n", (landscape!=0);


  for (i=1 ; i<=numberof(systems) ; ++i) {
    sys= systems(i);
    if (i==1) {
      default= sys;
      which= "default";
    } else {
      which= "system";
    }
    legend= sys.legend;

    if (!strlen(legend)) legend= "0";

    else legend= "\""+legend+"\"";

    write,file,format="%s= { legend= %s", which, (i>1? legend : "0");
    final= " }";
    vp= sys.viewport;

    if (i==1 || anyof(vp!=default.viewport)) {
      final= "}";

      write,file,format=",\n  viewport= { %f, %f, %f, %f }",
        vp(1),vp(2),vp(3),vp(4);
    }
    ticks= sys.ticks;
    if (i==1 || ticks!=default.ticks) {
      final= "}";

      write,file,format=",\n%s\n", "  ticks= {"
      axes= [ticks.horiz, ticks.vert];
      daxes= [default.ticks.horiz, default.ticks.vert];
      prefix= "\n    ";
      for (j=1 ; j<=2 ; ++j) {
	axis= axes(j);
	daxis= daxes(j);
	if (i==1 || axis!=daxis) {

	  write,file,format="%s%s= {\n", prefix, (j==1? "horiz" : "vert");
	  nitems= 0;
	  prefix= "      ";  suffix= "";
	  if (i==1 || axis.nMajor!=daxis.nMajor) {

	    write,file,format="%snMajor= %f", prefix, axis.nMajor;
	    prefix= ",  ";  suffix= " ";
	    ++nitems;
	  }
	  if (i==1 || axis.nMinor!=daxis.nMinor) {

	    write,file,format="%snMinor= %f", prefix, axis.nMinor;
	    prefix= ",  ";  suffix= " ";
	    ++nitems;
	  }
	  if (i==1 || axis.logAdjMajor!=daxis.logAdjMajor) {

	    write,file,format="%slogAdjMajor= %f", prefix, axis.logAdjMajor;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) {
	      nitems= 0;
	      prefix= ",\n      ";
	    }
	  }
	  if (i==1 || axis.logAdjMinor!=daxis.logAdjMinor) {

	    write,file,format="%slogAdjMinor= %f", prefix, axis.logAdjMinor;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) {
	      nitems= 0;
	      prefix= ",\n      ";
	    }
	  }
	  if (i==1 || axis.nDigits!=daxis.nDigits) {

	    write,file,format="%snDigits= %d", prefix, axis.nDigits;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) {
	      nitems= 0;
	      prefix= ",\n      ";
	    }
	  }
	  if (i==1 || axis.gridLevel!=daxis.gridLevel) {

	    write,file,format="%sgridLevel= %d", prefix, axis.gridLevel;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) {
	      nitems= 0;
	      prefix= ",\n      ";
	    }
	  }
	  if (i==1 || axis.flags!=daxis.flags) {

	    write,file,format="%sflags= 0x%03x", prefix, axis.flags;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) {
	      nitems= 0;
	      prefix= ",\n      ";
	    }
	  }
	  if (i==1 || axis.tickOff!=daxis.tickOff) {

	    write,file,format="%stickOff= %f", prefix, axis.tickOff;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) {
	      nitems= 0;
	      prefix= ",\n      ";
	    }
	  }
	  if (i==1 || axis.labelOff!=daxis.labelOff) {

	    write,file,format="%slabelOff= %f", prefix, axis.labelOff;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) {
	      nitems= 0;
	      prefix= ",\n      ";
	    }
	  }

	  if (i==1 || anyof(axis.tickLen!=daxis.tickLen)) {
	    if (prefix!="      ") prefix= ",\n      ";

	    _style_wvect, file,  prefix, "tickLen", axis.tickLen;
	    prefix= ",\n      ";  suffix= "";  nitems= 0;
	  }
	  if (i==1 || axis.tickStyle!=daxis.tickStyle) {
	    if (prefix!="      ") prefix= ",\n      ";

	    _style_wline, file,  prefix, "tickStyle", axis.tickStyle;
	    prefix= ",\n      ";  suffix= "";  nitems= 0;
	  }
	  if (i==1 || axis.gridStyle!=daxis.gridStyle) {
	    if (prefix!="      ") prefix= ",\n      ";

	    _style_wline, file,  prefix, "gridStyle", axis.gridStyle;
	    prefix= ",\n      ";  suffix= "";  nitems= 0;
	  }
	  if (i==1 || axis.textStyle!=daxis.textStyle) {
	    if (prefix!="      ") prefix= ",\n      ";

	    _style_wtext, file,  prefix, "textStyle", axis.textStyle;
	    prefix= ",\n      ";  suffix= "";  nitems= 0;
	  }
	  if (i==1 || axis.xOver!=daxis.xOver) {

	    write,file,format="%sxOver= %f", prefix, axis.xOver;
	    prefix= ",  ";  suffix= " ";
	    if ((++nitems)==3) prefix= ",\n      ";
	  }
	  if (i==1 || axis.yOver!=daxis.yOver) {

	    write,file,format="%syOver= %f", prefix, axis.yOver;
	    suffix= " ";
	  }
	  write,file,format="%s}", suffix;
	  prefix= ",\n\n    ";  suffix= "";
	}
      }
      if (i==1 || ticks.frame!=default.ticks.frame) {
	write,file,format="%sframe= %d", prefix, ticks.frame;
	prefix= ",\n    ";  suffix= " ";
      }
      if (i==1 || ticks.frameStyle!=default.ticks.frameStyle) {
	_style_wline, file,  prefix, "frameStyle", ticks.frameStyle;
	suffix= "";
      }
      write,file,format="%s}", suffix;
    }
    write,file,format="%s\n", final;

    if (i==1) {

      write,file,format="\nsystem= { legend= %s }\n", legend;
    }
  }

  legs= [legends,clegends];
  for (i=1 ; i<=2 ; ++i) {
    leg= legs(i);
    if (leg.nlines) {

      write,file,format="\n%slegends= {\n", (i==1? "" : "c");

      write,file,format="  x= %f, y= %f, dx= %f, dy= %f",
        leg.x, leg.y, leg.dx, leg.dy;

      _style_wtext, file, ",\n  ", "textStyle", leg.textStyle;

      write,file,format=",\n  nchars= %d, nlines= %d, nwrap= %d }\n",
        leg.nchars, leg.nlines, leg.nwrap;
    } else {

      write,file,format="\n%slegends= { nlines= 0 }\n", (i==1? "" : "c");
    }
  }

  return file;
}


func _style_wvect (file, prefix, member, value)
{

  write,file,format="%s%s= {", prefix, member;
  prefix= " ";

  for (i=1 ; i<=numberof(value) ; ++i) {

    write,file,format="%s%f", prefix, value(i);
    prefix= ", ";
  }
  write,file,format="%s}", " ";
}


func _style_wline (file, prefix, member, style)
{

  write,file,format="%s%s= { color= %d, type= %d, width= %f }",
    prefix, member, style.color, style.type, style.width;
}


func _style_wtext (file, prefix, member, style)
{

  write,file,format="%s%s= { color= %d, font= 0x%02x, height= %f",
    prefix, member, style.color, style.font, style.height;
  if (strpart(prefix,1:1)!=",") {

    if (strpart(prefix,1:1)!="\n") prefix= ",\n"+prefix;
    else prefix= ","+prefix;
  }

  write,file,format="%s  orient= %d, alignH= %d, alignV= %d, opaque= %d }",
    prefix, style.orient, style.alignH, style.alignV, style.opaque;
}


func read_style (file, &landscape, &systems, &legends, &clegends)

/* DOCUMENT read_style, file, landscape, systems, legends, clegends


     read a Gist style sheet (.gs file), and return the data

     structures as described in the get_style function.  The FILE
     can be a filename or a text file stream.


   SEE ALSO: get_style, set_style, write_style
 */
{
  if (structof(file)==string) {
    f= open(file, "r", 1);
    if (!f) {
      /* maybe the file is in one of the standard locations */
      f= open("~/Gist"+file, "r", 1);
      if (!f) {

	f= open(Y_SITE+"gist/"+file, "r", 1);

	if (!f) error, "missing style file: "+file;
      }
    }
  } else {
    f= file;
  }

  /* set up default values (same as work.gs) */
  landscape= 0n;

  default_line= GpLineAttribs(color= -2,  type= 1,  width= 1.0);

  default_text= GpTextAttribs(

    color= -2,  font= 0x08,  height= 0.0182,

    orient= 0,  alignH= 0,  alignV= 0,  opaque= 0);

  default_ltxt= GpTextAttribs(

    color= -2,  font= 0x00,  height= 0.0156,

    orient= 0,  alignH= 1,  alignV= 1,  opaque= 0);

  default= GfakeSystem(viewport=[0.19, 0.60, 0.44, 0.85],
    ticks= GaTickStyle(
      horiz= GaAxisStyle(
	nMajor=7.5, nMinor=50.0, logAdjMajor=1.2, logAdjMinor=1.2,
	nDigits=3, gridLevel=1, flags=0x033, tickOff=0.0007, labelOff=0.0182,
        tickLen= [0.0143, 0.0091, 0.0052, 0.0026, 0.0013],
        tickStyle= default_line,

        gridStyle= GpLineAttribs(color= -2,  type= 3,  width= 1.0),
        textStyle= default_text,
	xOver= 0.395,  yOver= 0.370),
      vert= GaAxisStyle(
	nMajor=7.5, nMinor=50.0, logAdjMajor=1.2, logAdjMinor=1.2,
	nDigits=3, gridLevel=1, flags=0x033, tickOff=0.0007, labelOff=0.0182,
        tickLen= [0.0143, 0.0091, 0.0052, 0.0026, 0.0013],
        tickStyle= default_line,

        gridStyle= GpLineAttribs(color= -2,  type= 3,  width= 1.0),
        textStyle= default_text,
	xOver= 0.150,  yOver= 0.370),
      frame= 0,

      frameStyle= GpLineAttribs(color= -2,  type= 1,  width= 1.0)));
  default_legb= GeLegendBox(
    x= 0.04698,  y= 0.360,  dx= 0.3758,  dy= 0.0,
    textStyle= default_ltxt, nchars= 36,  nlines= 20,  nwrap= 2);
  default_clegb= GeLegendBox(
    x= 0.6182,  y= 0.8643,  dx= 0.0,  dy= 0.0,
    textStyle= default_ltxt, nchars= 14,  nlines= 28,  nwrap= 1);
  legends= default_legb;
  clegends= default_clegb;

  /* parse the file */
  systems= [];
  type= [];
  line= "";
  for (;;) {

    s= _style_token(f, line, type);
    if (!line) break;

    if (type!=3) _style_goof, f, line, s;
    if (s=="landscape") {

      landscape= _style_token(f, line, type);

      if (type!=2 || structof(landscape)!=long)
	_style_goof, f, line, "landscape= ????";
      landscape= (landscape!=0);
    } else if (s=="default") {

      default= _style_system(f, line, default);
    } else if (s=="system") {

      grow, systems, [_style_system(f, line, default)];
    } else if (s=="legends") {

      legends= _style_legends(f, line, default_legb);
    } else if (s=="clegends") {

      clegends= _style_legends(f, line, default_clegb);
    } else {
      _style_goof, f, line, s;
    }
  }


  if (is_void(systems)) systems= [default];
}


func _style_system (f, &line, default)
{
  system= default;
  type= [];

  s= _style_token(f, line, type);

  if (s!="{") _style_goof, f, line, s;
  for (;;) {

    s= _style_token(f, line, type);

    if (type!=3) _style_goof, f, line, s;
    if (s=="legend") {

      s= _style_token(f, line, type);
      if (type!=1) {
	if (type==2 && s==0) s= string(0);
	else _style_goof, f, line, s;
      }
      system.legend= s;
    } else if (s=="viewport") {

      system.viewport= _style_vector(f,line);
    } else if (s=="ticks") {

      system.ticks= _style_ticks(f,line,system.ticks);
    } else {
      _style_goof, f, line, s;
    }

    s= _style_token(f, line, type);
    if (s=="}") break;

    if (s!=",") _style_goof, f, line, s;
  }
  return system;
}


func _style_legends (f, &line, default)
{
  legend= default;
  type= [];

  s= _style_token(f, line, type);

  if (s!="{") _style_goof, f, line, s;
  for (;;) {

    s= _style_token(f, line, type);

    if (type!=3) _style_goof, f, line, s;
    if (s=="textStyle")

      get_member(legend,s)= _style_text(f,line,default.textStyle);
    else

      get_member(legend,s)= _style_token(f,line,type);

    s= _style_token(f, line, type);
    if (s=="}") break;

    if (s!=",") _style_goof, f, line, s;
  }
  return legend;
}


func _style_ticks (f, &line, default)
{
  ticks= default;
  type= [];

  s= _style_token(f, line, type);

  if (s!="{") _style_goof, f, line, s;
  for (;;) {

    s= _style_token(f, line, type);

    if (type!=3) _style_goof, f, line, s;
    if (s=="horiz" || s=="vert") {

      get_member(ticks,s)= _style_axis(f,line,get_member(ticks,s));
    } else if (s=="frame") {

      ticks.frame= _style_token(f,line,type);
    } else if (s=="frameStyle") {

      ticks.frameStyle= _style_line(f,line,ticks.frameStyle);
    } else {
      _style_goof, f, line, s;
    }

    s= _style_token(f, line, type);
    if (s=="}") break;

    if (s!=",") _style_goof, f, line, s;
  }
  return ticks;
}


func _style_axis (f, &line, default)
{
  axis= default;
  type= [];

  s= _style_token(f, line, type);

  if (s!="{") _style_goof, f, line, s;
  for (;;) {

    s= _style_token(f, line, type);

    if (type!=3) _style_goof, f, line, s;
    if (s=="tickLen") {

      value= _style_vector(f, line);

      axis.tickLen(1:numberof(value))= value;
    } else if (s=="tickStyle" || s=="gridStyle") {

      get_member(axis,s)= _style_line(f,line,get_member(axis,s));
    } else if (s=="textStyle") {

      axis.textStyle= _style_text(f,line,axis.textStyle);
    } else {

      get_member(axis,s)= _style_token(f,line,type);
    }

    s= _style_token(f, line, type);
    if (s=="}") break;

    if (s!=",") _style_goof, f, line, s;
  }
  return axis;
}


func _style_vector (f, &line)
{
  vector= type= [];

  s= _style_token(f, line, type);

  if (s!="{") _style_goof, f, line, s;
  for (;;) {

    s= _style_token(f, line, type);

    if (type!=2) _style_goof, f, line, s;
    grow, vector, [double(s)];

    s= _style_token(f, line, type);
    if (s=="}") break;

    if (s!=",") _style_goof, f, line, s;
  }
  return vector;
}


func _style_text (f, &line, default)
{
  junk= [0.0];
  text= default;
  type= [];

  s= _style_token(f, line, type);

  if (s!="{") _style_goof, f, line, s;
  for (;;) {

    s= _style_token(f, line, type);

    if (type!=3) _style_goof, f, line, s;
    if (text=="path") text= "orient";

    if (noneof(text==["expand","spacing","upX","upY"]))

      get_member(text,s)= _style_token(f,line,type);
    else

      junk(1)= _style_token(f,line,type);

    s= _style_token(f, line, type);
    if (s=="}") break;

    if (s!=",") _style_goof, f, line, s;
  }
  return text;
}


func _style_line (f, &line, default)
{
  style= default;
  type= [];

  s= _style_token(f, line, type);

  if (s!="{") _style_goof, f, line, s;
  for (;;) {

    s= _style_token(f, line, type);

    if (type!=3) _style_goof, f, line, s;

    get_member(style,s)= _style_token(f,line,type);

    s= _style_token(f, line, type);
    if (s=="}") break;

    if (s!=",") _style_goof, f, line, s;
  }
  return style;
}

/* retrieve next token from file, updating current line
   -- set line to "" initially

   -- on output, type= 0 is delimiter, 1 is quoted, 2 is number, 3 keyword
 */

func _style_token (f, &line, &type, norecurse)
{
  s= "";
  for (;;) {
    /* give up if no more lines in file */
    if (!line) return line;
    /* remove leading whitespace */
    sread, line, s, format="%[ \t]";

    line= strpart(line, 1+strlen(s):0);
    /* stop if line has more chars, and is not a comment */
    if (strlen(line)) {
      s= strpart(line,1:1);
      if (s!="#") break;
    }
    /* otherwise get the next line and try again */
    line= rdline(f);
  }

  /* look at first character to see if this is a delimiter */
  cs= (*pointer(s))(1);
  if (anyof(['{','}',',','=']==cs)) {
    /* token is one of the four valid delimiters */
    line= strpart(line,2:0);
    type= 0;
    return string(&[cs,'\0']);
  } else if (cs=='"') {
    /* token is a quoted string */
    s= strtok(line, "\"");
    type= 1;
    line= s(2);
    return s(1);
  }

  /* token must be either a number or a keyword */
  s= strtok(line, " \t{},=")(1);

  line= strpart(line, strlen(s)+1:0);
  if ((cs>='0' && cs<='9') || cs=='.' || cs=='-') {
    /* token is a number */

    if (strmatch(s,".") || strmatch(s,"e",1)) {
      value= 0.0;

      if (!sread(s,value)) _style_goof, f, line, s;
    } else {
      value= 0;

      if (!sread(s,value,format="%i")) _style_goof, f, line, s;
    }
    type= 2;
    return value;
  } else {
    /* if next token is =, this token is a keyword */

    if (!norecurse) next= _style_token(f, line, type, 1);
    if (type==0 && next=="=") {
      type= 3;
      return s;
    } else {
      _style_goof, f, line, s;
    }
  }
}


func _style_goof (f, line, s)
{

  write, "graphics style file format error at or just before:";

  write, print(f)(2);

  error, "unrecognized token: "+pr1(s);
}

/* ------------------------------------------------------------------------ */
/* The following structure definitions must match those in
   Gist/gist.h and Gist/draw.h */

/* Note: NDC units 0.0013 equals one point equals 1/72.27 inch */
one_point= 0.0013;
one_inch= 72.27*one_point;

struct GpLineAttribs {

  long color;     /* -1=bg -2=fg -3=b -4=w -567=rgb -8910=cmy */

  int type;       /* line types: 0=none 1=solid 2=- 3=. 4=-. 5=-..  */

  double width;   /* 1.0 is normal width of a line (1/2 point) */
}

struct GpTextAttribs {

  long color;       /* -1=bg -2=fg -3=b -4=w -567=rgb -8910=cmy */

  int font;         /* text font
		       fonts: 0=courier 4=times 8=helvetica 12=symbol
		             16=newcentury
			      or with 1 for bold, 2 for italic */

  double height;    /* character height in NDC, default 0.0156 (12pt)

		       UNLIKE GKS, GIST font sizes are always specified
		       in NDC.  This drastically simplifies coding for

		       devices like X windows, which must load a font
		       at each size.  It also conforms better with

		       a Mac-like user interface in which font size
		       in points is selected by the user.  */

  int orient;          /* text orientation: 0=right 1=left 2=up 3=down  */
  int alignH, alignV;  /* text alignments:
			  alignH: 0=normal 1=left 2=center 3=right
			  alignV: 0=normal 1=top 2=cap 3=half 4=base 5=bot */

  int opaque;
}

struct GaAxisStyle {
  double nMajor, nMinor, logAdjMajor, logAdjMinor;
  int nDigits, gridLevel;
  int flags;    /* 0x001 ticks on lower edge
		   0x002 ticks on upper edge
		   0x004 ticks in center
		   0x008 inward ticks
		   0x010 outward ticks
		   0x020 labels on lower edge
		   0x040 labels on upper edge
		   0x080 full grid lines
		   0x100 origin grid line   */

  double tickOff, labelOff;  /* offsets in NDC from the edge of the
				viewport to the ticks or labels */
  double tickLen(5);         /* tick lengths in NDC */


  GpLineAttribs tickStyle, gridStyle;

  GpTextAttribs textStyle;   /* alignment ignored, set correctly */
  double xOver, yOver;       /* position for overflow label */
}

struct GaTickStyle {
  GaAxisStyle horiz, vert;
  int frame;
  GpLineAttribs frameStyle;
}

struct GeLegendBox {

  double x, y;              /* NDC location of this legend box */
  double dx, dy;            /* if non-zero, offset to 2nd column */

  GpTextAttribs textStyle;  /* font, size, etc. of these legends */

  int nchars, nlines;       /* max number of characters per line, lines */

  int nwrap;                /* max number of lines to wrap long legends */
}

struct GfakeSystem {

  double viewport(4);    /* [xmin,xmax,ymin,ymax] in NDC coordinates */

  GaTickStyle ticks;     /* tick style for this coordinate system */

  string legend;         /* e.g.- "System 0" or "System 1" */
}

/* ------------------------------------------------------------------------ */