Logo Search packages:      
Sourcecode: r-base version File versions

dataentry.c

/*
 *  R : A Computer Langage for Statistical Data Analysis
 *  Copyright (C) 1995, 1996  Robert Gentleman and Ross Ihaka
 *  Copyright (C) 1998--2005  Robert Gentleman, Ross Ihaka and the
 *                            R Development Core Team
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  (at your option) any later version.
 *
 *  This program 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 General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

/* <UTF8> 
   Used XTextWidth and XDrawText, so need to use fontsets

   Also needed input context.
*/

/* The version for R 2.1.0 is partly based on patches by 
   Eiji Nakama <nakama@ki.rim.or.jp> for use with Japanese fonts. */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <stdlib.h>

/* don't use X11 function prototypes (which tend to ...): */
#define NeedFunctionPrototypes 0
/* XFree 4.4.0 forgets to define this if NeedFunctionPrototypes=0 */
#ifndef _Xconst
#define _Xconst const
#endif
#include <X11/X.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/keysym.h>
#include <X11/cursorfont.h>

#include "Defn.h"
#include "Print.h"

#ifdef SUPPORT_MBCS
/* This only uses a FontSet in a MBCS */
# define USE_FONTSET 1
#endif

#ifndef HAVE_KEYSYM
#define KeySym int
#endif

#define DEEvent XEvent

typedef enum { UP, DOWN, LEFT, RIGHT } DE_DIRECTION;

typedef enum {UNKNOWNN, NUMERIC, CHARACTER} CellType;

/* EXPORTS : */
SEXP RX11_dataentry(SEXP call, SEXP op, SEXP args, SEXP rho);

/* Local Function Definitions */

static void advancerect(DE_DIRECTION);
static int  CheckControl(DEEvent*);
static int  CheckShift(DEEvent*);
static int  checkquit(int);
static void clearrect(void);
static void clearwindow(void);
static void closerect(void);
static void closewin(void);
static void copycell(void);
static void doControl(DEEvent*);
static int  doMouseDown(DEEvent*);
static void eventloop(void);
static void doSpreadKey(int, DEEvent*);
static void downlightrect(void);
static void drawwindow(void);
static void drawcol(int);
static void drawrow(int);
static void find_coords(int, int, int*, int*);
static int  findcell(void);
static char *GetCharP(DEEvent*);
static KeySym GetKey(DEEvent*);
static void handlechar(char*);
static void highlightrect(void);
static Rboolean initwin(void);
static void jumppage(DE_DIRECTION);
static void jumpwin(int, int);
static void pastecell(int, int);
static void popdownmenu(void);
static void popupmenu(int, int, int, int);
static void printlabs(void);
static void printrect(int, int);
static void printstring(char*, int, int, int, int);
static void printelt(SEXP, int, int, int);
static void RefreshKeyboardMapping(DEEvent*);

/* Functions to hide Xlib calls */
static void bell(void);
static void cleararea(int, int, int, int);
static void copyH(int, int, int);
static void copyarea(int, int, int, int);
static void doConfigure(DEEvent *ioevent);
#if 0
static void drawline(int, int, int, int);
#endif
static void drawrectangle(int, int, int, int, int, int);
static void drawtext(int, int, char*, int);
static int  NextEvent(DEEvent *ioevent);
static void RefreshKeyboardMapping(DEEvent *ioevent);
static void Rsync(void);
static int textwidth(char*, int);
static int WhichEvent(DEEvent ioevent);

static SEXP work, names, lens;
static PROTECT_INDEX wpi, npi, lpi;
static SEXP ssNA_STRING;
static double ssNA_REAL;

/* Global variables needed for the graphics */

static int box_w;                       /* width of a box */
static int boxw[100];                   /* widths of cells */
static int box_h;                       /* height of a box */
static int windowWidth;                 /* current width of the window */
static int fullwindowWidth;
static int windowHeight;                /* current height of the window */
static int fullwindowHeight;
static int currentexp;                  /* boolean: whether an cell is active */
static int crow;                        /* current row */
static int ccol;                        /* current column */
static int nwide, nhigh;
static int colmax, colmin, rowmax, rowmin;
static int ndecimal;                    /* count decimal points */
static int ne;                          /* count exponents */
static int nneg;              /* indicate whether its a negative */
static int clength;                     /* number of characters currently entered */
static char buf[201];               /* boosted to allow for MBCS */
static char *bufp;
static int bwidth;                  /* width of the border */
static int hwidth;                  /* width of header  */
static int text_offset;

static SEXP ssNewVector(SEXPTYPE, int);
static SEXP ssNA_STRING;
static double ssNA_REAL;

static Atom _XA_WM_PROTOCOLS, protocol;

static Rboolean newcol, CellModified;
static int nboxchars;
static int xmaxused, ymaxused;
static int box_coords[6];
static char copycontents[sizeof(buf)+1] ;
static int labdigs=4;
static char labform[6];


/* Xwindows Globals */

static Display          *iodisplay;
static Window           iowindow, menuwindow, menupanes[4];
static GC               iogc;
static XSizeHints       iohint;
static XFontStruct      *font_info;
static char             *font_name="9x15";
#ifdef USE_FONTSET
static XFontSet         font_set;
static XFontStruct      **fs_list;
static int        font_set_cnt;
static char             *fontset_name="-*-fixed-medium-r-normal--13-";
static XIM        ioim;
static XIC        ioic;
#endif

#define mouseDown       ButtonPress
#define keyDown         KeyPress
#define activateEvt     MapNotify
#define updateEvt Expose

#ifndef max
#define max(a, b) (((a)>(b))?(a):(b))
#endif
#ifndef min
#define min(a, b) (((a)<(b))?(a):(b))
#endif
#define BOXW(x) (min(((x<100 && nboxchars==0)?boxw[x]:box_w), fullwindowWidth-boxw[0]-2*bwidth-2))

/*
  Underlying assumptions (for this version R >= 1.8.0)

  The data are stored in a list `work', with unused columns having
  NULL entries.  The names for the list are in `names', which should
  have a name for all displayable columns (up to xmaxused).
  The *used* lengths of the columns are in `lens': this needs only be
  set for non-NULL columns.

  If the list was originally length(0), that should work with
  0 pre-defined cols.  (It used to have 1 pre-defined numeric column.)

  All row and col numbers are 1-based.

  BDR May 2003
 */

/*
   The spreadsheet function returns a list of vectors. The types of
   these vectors can be specified by the user as can their names. It
   the names are specified they are set during initialization. The
   user can change these via a menu interface, they can also change
   the type.

   The vectors are created too long and if they need to be increased
   this is done by using the next higher power of 2. They start 100
   long.  Vectors are initialized to NA when they are created so that
   NA is returned for any cell that was not set by the user.  So that
   coercion back and forth maintains values of ssNA_REAL and
   ssNA_STRING I have set ssNA_STRING to be coerceVector(ssNA_REAL),
   very weird but easy.

   In Macintosh we needed to call the main event loop to get
   events. This ensures that the spreadsheet interacts well with the
   other windows. Under X windows we let the window manager handle
   those sorts of details.

 */

static char *menu_label[] =
{
    "Real",
    "Character",
    "Change Name",
};

/*
   ssNewVector is just an interface to allocVector but it lets us
   set the fields to NA. We need to have a special NA for reals and
   strings so that we can differentiate between uninitialized elements
   in the vectors and user supplied NA's; hence ssNA_REAL and ssNA_STRING
 */

static SEXP ssNewVector(SEXPTYPE type, int vlen)
{
    SEXP tvec;
    int j;

    tvec = allocVector(type, vlen);
    for (j = 0; j < vlen; j++)
      if (type == REALSXP)
          REAL(tvec)[j] = ssNA_REAL;
      else if (type == STRSXP)
          SET_STRING_ELT(tvec, j, STRING_ELT(ssNA_STRING, 0));
    return (tvec);
}

static void closewin_cend(void *data)
{
    closewin();
}

SEXP RX11_dataentry(SEXP call, SEXP op, SEXP args, SEXP rho)
{
    SEXP colmodes, tnames, tvec, tvec2, work2;
    SEXPTYPE type;
    int i, j, cnt, len, nprotect;
    RCNTXT cntxt;
    char clab[25];

    nprotect = 0;/* count the PROTECT()s */
    PROTECT_WITH_INDEX(work = duplicate(CAR(args)), &wpi); nprotect++;
    colmodes = CADR(args);
    tnames = getAttrib(work, R_NamesSymbol);

    if (TYPEOF(work) != VECSXP || TYPEOF(colmodes) != VECSXP)
      errorcall(call, "invalid argument");

    /* initialize the constants */

    bufp = buf;
    ne = 0;
    currentexp = 0;
    nneg = 0;
    ndecimal = 0;
    clength = 0;
    ccol = 1;
    crow = 1;
    colmin = 1;
    rowmin = 1;
    ssNA_REAL = -NA_REAL;
    tvec = allocVector(REALSXP, 1);
    REAL(tvec)[0] = ssNA_REAL;
    PROTECT(ssNA_STRING = coerceVector(tvec, STRSXP)); nprotect++;
    bwidth = 5;
    hwidth = 30;

    /* setup work, names, lens  */
    xmaxused = length(work); ymaxused = 0;
    PROTECT_WITH_INDEX(lens = allocVector(INTSXP, xmaxused), &lpi);
    nprotect++;

    if (isNull(tnames)) {
      PROTECT_WITH_INDEX(names = allocVector(STRSXP, xmaxused), &npi);
      for(i = 0; i < xmaxused; i++) {
          sprintf(clab, "var%d", i);
          SET_STRING_ELT(names, i, mkChar(clab));
      }
    } else
      PROTECT_WITH_INDEX(names = duplicate(tnames), &npi);
    nprotect++;
    for (i = 0; i < xmaxused; i++) {
      int len = LENGTH(VECTOR_ELT(work, i));
      INTEGER(lens)[i] = len;
      ymaxused = max(len, ymaxused);
      type = TYPEOF(VECTOR_ELT(work, i));
      if (LENGTH(colmodes) > 0 && !isNull(VECTOR_ELT(colmodes, i)))
          type = str2type(CHAR(STRING_ELT(VECTOR_ELT(colmodes, i), 0)));
      if (type != STRSXP) type = REALSXP;
      if (isNull(VECTOR_ELT(work, i))) {
          if (type == NILSXP) type = REALSXP;
          SET_VECTOR_ELT(work, i, ssNewVector(type, 100));
      } else if (!isVector(VECTOR_ELT(work, i)))
          errorcall(call, "invalid type for value");
      else {
          if (TYPEOF(VECTOR_ELT(work, i)) != type)
            SET_VECTOR_ELT(work, i,
                         coerceVector(VECTOR_ELT(work, i), type));
      }
    }


    /* start up the window, more initializing in here */
    if (initwin())
      errorcall(call, "invalid device");

    /* set up a context which will close the window if there is an error */
    begincontext(&cntxt, CTXT_CCODE, R_NilValue, R_NilValue, R_NilValue,
             R_NilValue, R_NilValue);
    cntxt.cend = &closewin_cend;
    cntxt.cenddata = NULL;

    highlightrect();

    eventloop();

    endcontext(&cntxt);
    closewin();

    /* drop out unused columns */
    for(i = 0, cnt = 0; i < xmaxused; i++)
      if(!isNull(VECTOR_ELT(work, i))) cnt++;
    if (cnt < xmaxused) {
      PROTECT(work2 = allocVector(VECSXP, cnt)); nprotect++;
      for(i = 0, j = 0; i < xmaxused; i++) {
          if(!isNull(VECTOR_ELT(work, i))) {
            SET_VECTOR_ELT(work2, j, VECTOR_ELT(work, i));
            INTEGER(lens)[j] = INTEGER(lens)[i];
            SET_STRING_ELT(names, j, STRING_ELT(names, i));
            j++;
          }
      }
      REPROTECT(names = lengthgets(names, cnt), npi);
    } else work2 = work;

    for (i = 0; i < LENGTH(work2); i++) {
      len = INTEGER(lens)[i];
      tvec = VECTOR_ELT(work2, i);
      if (LENGTH(tvec) != len) {
          tvec2 = ssNewVector(TYPEOF(tvec), len);
          for (j = 0; j < len; j++) {
            if (TYPEOF(tvec) == REALSXP) {
                if (REAL(tvec)[j] != ssNA_REAL)
                  REAL(tvec2)[j] = REAL(tvec)[j];
                else
                  REAL(tvec2)[j] = NA_REAL;
            } else if (TYPEOF(tvec) == STRSXP) {
                if (!streql(CHAR(STRING_ELT(tvec, j)),
                        CHAR(STRING_ELT(ssNA_STRING, 0))))
                  SET_STRING_ELT(tvec2, j, STRING_ELT(tvec, j));
                else
                  SET_STRING_ELT(tvec2, j, NA_STRING);
            } else
                error("dataentry: internal memory problem");
          }
          SET_VECTOR_ELT(work2, i, tvec2);
      }
    }

    setAttrib(work2, R_NamesSymbol, names);
    UNPROTECT(nprotect);
    return work2;
}


/* Window Drawing Routines */

static void setcellwidths(void)
{
    int i, w, dw;

    windowWidth = w = 2*bwidth + boxw[0] + BOXW(colmin);
    nwide = 2;
    for (i = 2; i < 100; i++) { /* 100 on-screen columns cannot occur */
      dw = BOXW(i + colmin - 1);
      if((w += dw) > fullwindowWidth) {
          nwide = i;
          windowWidth = w - dw;
          break;
      }
    }
}

static void drawwindow(void)
{
    int i, st;
    XWindowAttributes attribs;

    /* if there is an active cell enter the data in it */
    closerect();

    /* now set up the window with the new dimensions */
    XGetWindowAttributes(iodisplay, iowindow, &attribs);
    bwidth = attribs.border_width;
    fullwindowWidth = attribs.width;
    fullwindowHeight = attribs.height;
    setcellwidths();
    nhigh = (fullwindowHeight - 2 * bwidth - hwidth) / box_h;
    windowHeight = nhigh * box_h + 2 * bwidth;

    clearwindow();


    for (i = 1; i < nhigh; i++)
      drawrectangle(0, hwidth + i * box_h, boxw[0], box_h, 1, 1);
     /* so row 0 and col 0 are reserved for labels */
    colmax = colmin + (nwide - 2);
    rowmax = rowmin + (nhigh - 2);
    printlabs();
    for (i = colmin; i <= colmax; i++) drawcol(i);

    /* draw the quit etc boxes */

    i = textwidth("Quit", 4);
    box_coords[0] = st = fullwindowWidth - 6 - bwidth;
    box_coords[1] = st - i;
    drawrectangle(st - i, 3, i + 4, hwidth - 6, 1, 1);
    drawtext(st + 2 - i, hwidth - 7, "Quit", 4);

    box_coords[4] = st = st - 5*i;
    i = textwidth("Paste", 5);
    box_coords[5] = st - i;
    drawrectangle(st - i, 3, i + 4, hwidth - 6, 1, 1);
    drawtext(st + 2 - i, hwidth - 7, "Paste", 5);

    box_coords[2] = st = st - 2*i;
    i = textwidth("Copy", 4);
    box_coords[3] = st - i;
    drawrectangle(st - i, 3, i + 4, hwidth - 6, 1, 1);
    drawtext(st + 2 - i, hwidth - 7, "Copy", 4);

    highlightrect();

    Rsync();

}

static void doHscroll(int oldcol)
{
    int i, dw;
    int oldnwide = nwide, oldwindowWidth = windowWidth;

    /* horizontal re-position */
    setcellwidths();
    colmax = colmin + (nwide - 2);
    if (oldcol < colmin) { /* drop oldcol...colmin - 1 */
      dw = boxw[0];
      for (i = oldcol; i < colmin; i++) dw += BOXW(i);
      copyH(dw, boxw[0], oldwindowWidth - dw + 1);
      dw = oldwindowWidth - BOXW(oldcol) + 1;
      cleararea(dw, hwidth, fullwindowWidth-dw, fullwindowHeight);
      /* oldnwide includes the row labels */
      for (i = oldcol+oldnwide-1; i <= colmax; i++) drawcol(i);
    } else {
      /* move one or more cols left */
      dw = BOXW(colmin);
      copyH(boxw[0], boxw[0] + dw, windowWidth - dw + 1);
      dw = windowWidth + 1;
      cleararea(dw, hwidth, fullwindowWidth-dw, fullwindowHeight);
      drawcol(colmin);
    }

    highlightrect();

    Rsync();
}

/* find_coords finds the coordinates of the upper left corner of the
   given cell on the screen: row and col are on-screen coords */

static void find_coords(int row, int col, int *xcoord, int *ycoord)
{
    int i, w;
    w = bwidth;
    if (col > 0) w += boxw[0];
    for(i = 1; i < col; i ++) w += BOXW(i + colmin - 1);
    *xcoord = w;
    *ycoord = bwidth + hwidth + box_h * row;
}

/* draw the window with the top left box at column wcol and row wrow */

static void jumpwin(int wcol, int wrow)
{
    if (wcol < 0 || wrow < 0) {
      bell();
      return;
    }
    closerect();
    if (colmin != wcol || rowmin != wrow) {
      colmin = wcol;
      rowmin = wrow;
      drawwindow();
    } else highlightrect();
}

static void advancerect(DE_DIRECTION which)
{

    /* if we are in the header, changing a name then only down is
       allowed */
    if (crow < 1 && which != DOWN) {
      bell();
      return;
    }

    closerect();

    switch (which) {
    case UP:
      if (crow == 1) {
          if (rowmin == 1)
            bell();
          else
            jumppage(UP);
      } else
          crow--;
      break;
    case DOWN:
      if (crow == (nhigh - 1))
          jumppage(DOWN);
      else
          crow++;
      break;
    case RIGHT:
      if (ccol == (nwide - 1))
          jumppage(RIGHT);
      else
          ccol++;
      break;
    case LEFT:
      if (ccol == 1) {
          if (colmin == 1)
            bell();
          else
            jumppage(LEFT);
      } else
          ccol--;
      break;
    default:
      UNIMPLEMENTED("advancerect");
    }

    highlightrect();
}

static char *get_col_name(int col)
{
    static char clab[25];
    if (col <= xmaxused) {
      /* don't use NA labels */
      SEXP tmp = STRING_ELT(names, col - 1);
      if(tmp != NA_STRING) return(CHAR(tmp));
    }
    sprintf(clab, "var%d", col);
    return clab;
}

static int get_col_width(int col)
{
    int i, w = 0, w1;
    char *strp;
    SEXP tmp, lab;

    if (nboxchars > 0) return box_w;
    if (col <= xmaxused) {
      tmp = VECTOR_ELT(work, col - 1);
      if (isNull(tmp)) return box_w;
      /* don't use NA labels */
      lab = STRING_ELT(names, col - 1);
      if(lab != NA_STRING) strp = CHAR(lab); else strp = "var12";
      PrintDefaults(R_NilValue);
      w = textwidth(strp, strlen(strp));
      for (i = 0; i < INTEGER(lens)[col - 1]; i++) {
          strp = EncodeElement(tmp, i, 0);
          w1 = textwidth(strp, strlen(strp));
          if (w1 > w) w = w1;
      }
      if(w < 0.5*box_w) w = 0.5*box_w;
      if(w < 0.8*box_w) w+= 0.1*box_w;
      if(w > 600) w = 600;
      return w+8;
    }
    return box_w;
}

static CellType get_col_type(int col)
{
    SEXP tmp;
    CellType res = UNKNOWNN;

    if (col <= xmaxused) {
      tmp = VECTOR_ELT(work, col - 1);
      if(TYPEOF(tmp) == REALSXP) res = NUMERIC;
      if(TYPEOF(tmp) == STRSXP) res = CHARACTER;
    }
    return res;
}


/* whichcol is absolute col no, col is position on screen */
static void drawcol(int whichcol)
{
    int i, src_x, src_y, len, col = whichcol - colmin + 1, bw = BOXW(whichcol);
    char *clab;
    SEXP tmp;

    find_coords(0, col, &src_x, &src_y);
    cleararea(src_x, src_y, bw, windowHeight);
    for (i = 0; i < nhigh; i++)
      drawrectangle(src_x, hwidth + i * box_h, bw, box_h, 1, 1);

    /* now fill it in if it is active */
    clab = get_col_name(whichcol);
    printstring(clab, strlen(clab), 0, col, 0);

   if (xmaxused >= whichcol) {
      tmp = VECTOR_ELT(work, whichcol - 1);
      if (!isNull(tmp)) {
          len = min(rowmax, INTEGER(lens)[whichcol - 1]);
          for (i = (rowmin - 1); i < len; i++)
            printelt(tmp, i, i - rowmin + 2, col);
      }
    }
    Rsync();
}


/* whichrow is absolute row no */
static void drawrow(int whichrow)
{
    int i, src_x, src_y, row = whichrow - rowmin + 1, w;
    char rlab[15];
    SEXP tvec;

    find_coords(row, 0, &src_x, &src_y);
    cleararea(src_x, src_y, windowWidth, box_h);
    drawrectangle(src_x, src_y, boxw[0], box_h, 1, 1);

    sprintf(rlab, labform, whichrow);
    printstring(rlab, strlen(rlab), row, 0, 0);

    w = bwidth + boxw[0];
    for (i = colmin; i <= colmax; i++) {
      drawrectangle(w, src_y, BOXW(i), box_h, 1, 1);
      w += BOXW(i);
    }

    for (i = colmin; i <= colmax; i++) {
      if (i > xmaxused) break;
      if (!isNull(tvec = VECTOR_ELT(work, i - 1)))
          if (whichrow <= INTEGER(lens)[i - 1])
            printelt(tvec, whichrow - 1, row, i - colmin + 1);
    }

    Rsync();
}

/* printelt: print the correct value from vector[vrow] into the
   spreadsheet in row ssrow and col sscol */

/* WARNING: This has no check that you're not beyond the end of the
   vector. Caller must check. */

static void printelt(SEXP invec, int vrow, int ssrow, int sscol)
{
    char *strp;
    PrintDefaults(R_NilValue);
    if (TYPEOF(invec) == REALSXP) {
      if (REAL(invec)[vrow] != ssNA_REAL) {
          strp = EncodeElement(invec, vrow, 0);
          printstring(strp, strlen(strp), ssrow, sscol, 0);
      }
    }
    else if (TYPEOF(invec) == STRSXP) {
      if (!streql(CHAR(STRING_ELT(invec, vrow)),
                CHAR(STRING_ELT(ssNA_STRING, 0)))) {
          strp = EncodeElement(invec, vrow, 0);
          printstring(strp, strlen(strp), ssrow, sscol, 0);
      }
    }
    else
      error("dataentry: internal memory error");
}


static void drawelt(int whichrow, int whichcol)
{
    int i;
    char *clab;
    SEXP tmp;

    if (whichrow == 0) {
      clab = get_col_name(whichcol + colmin - 1);
      printstring(clab, strlen(clab), 0, whichcol, 0);
    } else {
      if (xmaxused >= whichcol + colmin - 1) {
          tmp = VECTOR_ELT(work, whichcol + colmin - 2);
          if (!isNull(tmp) && (i = rowmin + whichrow - 2) <
            INTEGER(lens)[whichcol + colmin - 2] )
            printelt(tmp, i, whichrow, whichcol);
      } else
          printstring("", 0, whichrow,  whichcol, 0);
    }

    Rsync();
}

static void jumppage(DE_DIRECTION dir)
{
    int i, w, oldcol, wcol;

    switch (dir) {
    case UP:
      rowmin--;
      rowmax--;
      copyarea(0, hwidth + box_h, 0, hwidth + 2 * box_h);
      drawrow(rowmin);
      break;
    case DOWN:
      if (rowmax >= 65535) return;
      rowmin++;
      rowmax++;
      copyarea(0, hwidth + 2 * box_h, 0, hwidth + box_h);
      drawrow(rowmax);
      break;
    case LEFT:
      colmin--;
      doHscroll(colmin+1);
      break;
    case RIGHT:
      oldcol = colmin;
      wcol = colmin + ccol + 1; /* column to be selected */
        /* There may not be room to fit the next column in */
      w = fullwindowWidth - boxw[0] - BOXW(colmax + 1);
      for (i = colmax; i >= oldcol; i--) {
          w -= BOXW(i);
          if(w < 0) {
            colmin = i + 1;
            break;
          }
      }
      ccol = wcol - colmin;
      doHscroll(oldcol);
      break;
    }
}
/* draw a rectangle, used to highlight/downlight the current box */

static void printrect(int lwd, int fore)
{
    int x, y;
    find_coords(crow, ccol, &x, &y);
    drawrectangle(x + lwd - 1, y + lwd - 1,
              BOXW(ccol+colmin-1) - lwd + 1,
              box_h - lwd + 1, lwd, fore);
    Rsync();
}

static void downlightrect(void)
{
    printrect(2, 0);
    printrect(1, 1);
}

static void highlightrect(void)
{
    printrect(2, 1);
}


static void getccol()
{
    SEXP tmp, tmp2;
    int i, len, newlen, wcol, wrow;
    SEXPTYPE type;
    char clab[25];

    wcol = ccol + colmin - 1;
    wrow = crow + rowmin - 1;
    if (wcol > xmaxused) {
      /* extend work, names and lens */
      REPROTECT(work = lengthgets(work, wcol), wpi);
      REPROTECT(names = lengthgets(names, wcol), npi);
      for (i = xmaxused; i < wcol; i++) {
          sprintf(clab, "var%d", i + 1);
          SET_STRING_ELT(names, i, mkChar(clab));
      }
      REPROTECT(lens = lengthgets(lens, wcol), lpi);
      xmaxused = wcol;
    }
    newcol = FALSE;
    if (isNull(VECTOR_ELT(work, wcol - 1))) {
      newcol = TRUE;
      SET_VECTOR_ELT(work, wcol - 1, ssNewVector(REALSXP, max(100, wrow)));
      INTEGER(lens)[wcol - 1] = 0;
    }
    if (!isVector(tmp = VECTOR_ELT(work, wcol - 1)))
      error("internal type error in dataentry");
    len = INTEGER(lens)[wcol - 1];
    type = TYPEOF(tmp);
    if (len < wrow) {
      for (newlen = max(len * 2, 10) ; newlen < wrow ; newlen *= 2)
          ;
      tmp2 = ssNewVector(type, newlen);
      for (i = 0; i < len; i++)
          if (type == REALSXP)
            REAL(tmp2)[i] = REAL(tmp)[i];
          else if (type == STRSXP)
            SET_STRING_ELT(tmp2, i, STRING_ELT(tmp, i));
          else
            error("internal type error in dataentry");
      SET_VECTOR_ELT(work, wcol - 1, tmp2);
    }
}

/* close up the entry to a cell, put the value that has been entered
   into the correct place and as the correct type */

static void closerect(void)
{
    SEXP cvec;
    int i, wcol = ccol + colmin - 1, wrow = rowmin + crow - 1, wrow0;
    char clab[25];

    *bufp = '\0';

    /* first check to see if anything has been entered */
    if (CellModified) {
      if (crow == 0) {
          if (clength != 0) {
            /* then we are entering a new column name */
            if (xmaxused < wcol) {
                /* extend work, names and lens */
                REPROTECT(work = lengthgets(work, wcol), wpi);
                REPROTECT(names = lengthgets(names, wcol), npi);
                for (i = xmaxused; i < wcol - 1; i++) {
                  sprintf(clab, "var%d", i + 1);
                  SET_STRING_ELT(names, i, mkChar(clab));
                }
                REPROTECT(lens = lengthgets(lens, wcol), lpi);
                xmaxused = wcol;
            }
            SET_STRING_ELT(names, wcol - 1, mkChar(buf));
            printstring(buf, strlen(buf), 0, wcol, 0);
          } else {
            sprintf(buf, "var%d", ccol);
            printstring(buf, strlen(buf), 0, wcol, 0);
          }
      } else {
          getccol();
          cvec = VECTOR_ELT(work, wcol - 1);
          wrow0 = INTEGER(lens)[wcol - 1];
          if (wrow > wrow0) INTEGER(lens)[wcol - 1] = wrow;
          ymaxused = max(ymaxused, wrow);
          if (clength != 0) {
            /* do it this way to ensure NA, Inf, ...  can get set */
            char *endp;
            double new = R_strtod(buf, &endp);
            Rboolean warn = !isBlankString(endp);
            if (TYPEOF(cvec) == STRSXP)
                SET_STRING_ELT(cvec, wrow - 1, mkChar(buf));
            else
                REAL(cvec)[wrow - 1] = new;
            if (newcol & warn) {
                /* change mode to character */
                SET_VECTOR_ELT(work, wcol - 1, coerceVector(cvec, STRSXP));
                SET_STRING_ELT(VECTOR_ELT(work, wcol - 1), wrow - 1,
                           mkChar(buf));
            }
          } else {
            if (TYPEOF(cvec) == STRSXP)
                SET_STRING_ELT(cvec, wrow - 1, NA_STRING);
            else
                REAL(cvec)[wrow - 1] = NA_REAL;
          }
          drawelt(crow, ccol);  /* to get the cell scrolling right */
          if(wrow > wrow0) drawcol(wcol); /* to fill in NAs */
      }
    }
    CellModified = FALSE;

    downlightrect();

    ndecimal = 0;
    nneg = 0;
    ne = 0;
    currentexp = 0;
    clength = 0;
    bufp = buf;
}

/* print a null terminated string, check to see if it is longer than
   the print area and print it, left adjusted if necessary; clear the
   area of previous text; */

/* This version will only display 200 chars, but the maximum col width
   will not allow that many */
static void printstring(char *ibuf, int buflen, int row, int col, int left)
{
    int i, x_pos, y_pos, bw, bufw;
    char pbuf[201], *pc = pbuf;

    find_coords(row, col, &x_pos, &y_pos);
    if (col == 0) bw = boxw[0]; else bw = BOXW(col+colmin-1);
    cleararea(x_pos + 2, y_pos + 2, bw - 3, box_h - 3);
    bufw = (buflen > 200) ? 200 : buflen;
    strncpy(pbuf, ibuf, bufw);
    if(left) {
      for (i = bufw; i > 1; i--) {
          if (textwidth(pc, i) < (bw - text_offset)) break;
          *(++pc) = '<';
      }
    } else {
      for (i = bufw; i > 1; i--) {
          if (textwidth(pbuf, i) < (bw - text_offset)) break;
          *(pbuf + i - 2) = '>';
      }
    }
    drawtext(x_pos + text_offset, y_pos + box_h - text_offset, pc, i);
    Rsync();
}

static void clearrect(void)
{
    int x_pos, y_pos;

    find_coords(crow, ccol, &x_pos, &y_pos);
    cleararea(x_pos, y_pos, BOXW(ccol+colmin-1), box_h);
    Rsync();
}

/* handlechar has to be able to parse decimal numbers and strings,
   depending on the current column type, only printing characters
   should get this far */

/* --- Not true! E.g. ESC ends up in here... */
#ifdef USE_FONTSET
#include <wchar.h>
#include <wctype.h>
#endif

/* <FIXME> This is not correct for stateful MBCSs, but that's hard to 
   do as we get a char at a time */
static void handlechar(char *text)
{
    int i, c = text[0];
#ifdef USE_FONTSET
    wchar_t wc;
#endif

    if ( c == '\033' ) { /* ESC */
      CellModified = FALSE;
      clength = 0;
        bufp = buf;
      drawelt(crow, ccol);
      return;
    } else
      CellModified = TRUE;

    if (clength == 0) {

      if (crow == 0)                            /* variable name */
          currentexp = 3;
      else
          switch(get_col_type(ccol + colmin - 1)) {
          case NUMERIC:
            currentexp = 1;
            break;
          default:
            currentexp = 2;
          }
      clearrect();
      highlightrect();
    }

    if (currentexp == 1) {    /* we are parsing a number */
#ifdef USE_FONTSET
      mbrtowc(&wc, text, MB_CUR_MAX, NULL);
      switch (wc) {
      case L'-':
          if (nneg == 0) nneg++; else goto donehc;
          break;
      case '.':
          if (ndecimal == 0) ndecimal++; else goto donehc;
          break;
      case L'e':
      case L'E':
          if (ne == 0) {
            nneg = ndecimal = 0;    /* might have decimal in exponent */
            ne++;
          } else goto donehc;
          break;
      default:
          if (!iswdigit(wc)) goto donehc;
          break;
      }
#else
      switch (c) {
      case '-':
          if (nneg == 0) nneg++; else goto donehc;
          break;
      case '.':
          if (ndecimal == 0) ndecimal++; else goto donehc;
          break;
      case 'e':
      case 'E':
          if (ne == 0) {
            nneg = ndecimal = 0;    /* might have decimal in exponent */
            ne++;
          } else goto donehc;
          break;
      default:
          if (!isdigit(c)) goto donehc;
          break;
      }
#endif
    }
    if (currentexp == 3) {
#ifdef USE_FONTSET
      mbrtowc(&wc, text, MB_CUR_MAX , NULL);
      if (iswspace(wc)) goto donehc;
      if (clength == 0 && wc != '.' && !iswalpha(wc)) goto donehc;
      else if (wc != '.' && !iswalnum(wc)) goto donehc;
#else
      if (isspace(c)) goto donehc;
      if (clength == 0  && c != '.' && !isalpha(c)) goto donehc;
      else if (c != '.' && !isalnum(c)) goto donehc;
#endif
    }

    if (clength++ > 29) {
      warning("dataentry: expression too long");
      clength--;
      goto donehc;
    }

    for(i = 0; i < strlen(text); i++) *bufp++ = text[i];
    printstring(buf, clength, crow, ccol, 1);
    return;

 donehc:
    bell();
}

static void printlabs(void)
{
    char clab[15], *p;
    int i;

    for (i = colmin; i <= colmax; i++) {
      p = get_col_name(i);
      printstring(p, strlen(p), 0, i - colmin + 1, 0);
    }
    for (i = rowmin; i <= rowmax; i++) {
      sprintf(clab, labform, i);
      printstring(clab, strlen(clab), i - rowmin + 1, 0, 0);
    }
}

               /* ================ X11-specific ================ */

/* find out whether the button click was in the quit box */
static int checkquit(int xw)
{
    if (xw > box_coords[1] && xw < box_coords[0]) return 1;
    if (xw > box_coords[3] && xw < box_coords[2]) return 2;
    if (xw > box_coords[5] && xw < box_coords[4]) return 3;
    return 0;
}

/* when a buttonpress event happens find the square that is being
   pointed to if the pointer is in the header we need to see if the
   quit button was pressed and if so quit. This is done by having
   findcell return an int which is one if we should quit and zero
   otherwise */

static int findcell(void)
{

    int xw, yw, xr, yr, wcol=0, wrow, i, w;
    unsigned int keys;
    Window root, child;

    closerect();
    XQueryPointer(iodisplay, iowindow, &root, &child,
              &xr, &yr, &xw, &yw, &keys);

    if (keys & Button1Mask) { /* left click */

      /* check to see if the click was in the header */

      if (yw < hwidth + bwidth) {
          i =  checkquit(xw);
          if (i == 1) return 1;
          else if (i == 2) copycell();
          else if (i == 3) pastecell(crow, ccol);
          return 0;
      }


      /* see if it is in the row labels */
      if (xw < bwidth + boxw[0]) {
          bell();
          highlightrect();
          return 0;
      }
      /* translate to box coordinates */
      wrow = (yw - bwidth - hwidth) / box_h;
      w = bwidth + boxw[0];
      for (i = 1; i <= nwide; i++)
          if((w += BOXW(i+colmin-1)) > xw) {
            wcol = i;
            break;
          }

      /* next check to see if it is in the column labels */

      if (yw < hwidth + bwidth + box_h) {
          if (xw > bwidth + boxw[0])
            popupmenu(xr, yr, wcol, wrow);
          else {
            highlightrect();
            bell();
          }
      } else if (wrow > nhigh - 1 || wcol > nwide -1) {
            /* off the grid */
            highlightrect();
            bell();
      } else if (wcol != ccol || wrow != crow) {
          ccol = wcol;
          crow = wrow;
      }
    }
    if (keys & Button2Mask) { /* Paste */
      int row, col = 0;

      if (yw < hwidth + bwidth || xw < bwidth + boxw[0]) return 0;

      /* translate to box coordinates */
      row = (yw - bwidth - hwidth) / box_h;
      w = bwidth + boxw[0];
      for (i = 1; i <= nwide; i++)
          if ((w += BOXW(i+colmin-1)) > xw) {
            col = i;
            break;
          }
      pastecell(row, col);
    }
    highlightrect();
    return 0;
}


/* Event Loop Functions */

static void eventloop(void)
{
    int done;
    DEEvent ioevent;


    done = 0;
    while (done == 0) {
      if (NextEvent(&ioevent)) {
          switch (WhichEvent(ioevent)) {
          case activateEvt:
            drawwindow();
            break;
          case mouseDown:
            done  = doMouseDown(&ioevent);
            break;
          case keyDown:
            doSpreadKey(0, &ioevent);
            break;
          case MappingNotify:
            RefreshKeyboardMapping(&ioevent);
            break;
          case ConfigureNotify:
            doConfigure(&ioevent);
            break;
          case ClientMessage:
            if(ioevent.xclient.message_type == _XA_WM_PROTOCOLS
               && ioevent.xclient.data.l[0] == protocol) {
                /* user clicked on `close' aka `destroy' */
                   done = 1;
            }
            break;
          }
      }
    }
}

static int doMouseDown(DEEvent * event)
{
    return findcell();
}

static void doSpreadKey(int key, DEEvent * event)
{
    KeySym iokey;
    char *text;

    iokey = GetKey(event);
    text = GetCharP(event);

    if (CheckControl(event))
      doControl(event);
    else if ((iokey == XK_Return)  || (iokey == XK_KP_Enter) ||
           (iokey == XK_Linefeed)|| (iokey == XK_Down))
      advancerect(DOWN);
    else if (iokey == XK_Left)
      advancerect(LEFT);
    else if (iokey == XK_Right)
      advancerect(RIGHT);
    else if (iokey == XK_Up)
      advancerect(UP);
#ifdef XK_Page_Up
    else if (iokey == XK_Page_Up) {
      int i = rowmin - nhigh + 2;
      jumpwin(colmin, max(1, i));
    }
#elif defined(XK_Prior)
    else if (iokey == XK_Prior) {
      int i = rowmin - nhigh + 2;
      jumpwin(colmin, max(1, i));
    }
#endif
#ifdef XK_Page_Down
    else if (iokey == XK_Page_Down)
      jumpwin(colmin, rowmax);
#elif defined(XK_Next)
    else if (iokey == XK_Next)
      jumpwin(colmin, rowmax);
#endif
    else if ((iokey == XK_BackSpace) || (iokey == XK_Delete)) {
      if (clength > 0) {
          clength--;
          bufp--;
          printstring(buf, clength, crow, ccol, 1);
      } else bell();
    }
    else if (iokey == XK_Tab) {
      if(CheckShift(event)) advancerect(LEFT);
      else advancerect(RIGHT);
    }
    else if (iokey == XK_Home) {
      jumpwin(1, 1);
      downlightrect();
      crow = ccol = 1;
      highlightrect();
    }
    else if (iokey == XK_End) {
      int i = ymaxused - nhigh + 2;
      jumpwin(xmaxused, max(i, 1));
      downlightrect();
      crow = ymaxused - rowmin + 1;
      ccol = 1;
      highlightrect();
    }
    else if (IsModifierKey(iokey)) {
    }
    else
      handlechar(text);
}

static int NextEvent(DEEvent * ioevent)
{
    XNextEvent(iodisplay, ioevent);
    return 1;
}

static int WhichEvent(DEEvent ioevent)
{
    return ioevent.type;
}

static KeySym GetKey(DEEvent * event)
{
    char text[1];
    KeySym iokey;

    XLookupString((XKeyEvent *)event, text, 1, &iokey, NULL);
    return iokey;
}

static char *GetCharP(DEEvent * event)
{
    static char text[8];
    KeySym iokey;

#ifdef USE_FONTSET
    if(mbcslocale) {
#ifdef HAVE_XUTF8LOOKUPSTRING
      Xutf8LookupString(ioic, (XKeyEvent *)event, text, 8, &iokey, NULL);
#else
      XmbLookupString(ioic, (XKeyEvent *)event, text, 8, &iokey, NULL);
#endif
      /* FIXME check the return code */
    } else
#endif
      XLookupString((XKeyEvent *)event, text, 8, &iokey, NULL);
    return text;
}

static int CheckControl(DEEvent * event)
{
    return (*event).xkey.state & ControlMask;
}

static int CheckShift(DEEvent * event)
{
    return (*event).xkey.state & ShiftMask;
}

static void doControl(DEEvent * event)
{
    int i;
    char text[1];
    KeySym iokey;

    (*event).xkey.state = 0;
    XLookupString((XKeyEvent *)event, text, 1, &iokey, NULL);
    /* one row overlap when scrolling: top line <--> bottom line */
    switch (text[0]) {
      case 'b':
          i = rowmin - nhigh + 2;
          jumpwin(colmin, max(1, i));
          break;
      case 'f':
          jumpwin(colmin, rowmax);
          break;
      case 'l':
          closerect();
          for (i = 1 ; i <= min(100, xmaxused); i++)
            boxw[i] = get_col_width(i);
          drawwindow();
          break;
    }
}


static void doConfigure(DEEvent * event)
{
    if ((fullwindowWidth != (*event).xconfigure.width) ||
      (fullwindowHeight != (*event).xconfigure.height))
      drawwindow();
}

static void RefreshKeyboardMapping(DEEvent * event)
{
    XRefreshKeyboardMapping((XMappingEvent *)event);
}

/* Initialize/Close Windows */

void closewin(void)
{
    XFreeGC(iodisplay, iogc);
#ifdef USE_FONTSET
    if(mbcslocale) {
      XDestroyIC(ioic);
      XCloseIM(ioim);
    }
#endif
    XDestroyWindow(iodisplay, iowindow);
    XCloseDisplay(iodisplay);
}

static int R_X11Err(Display *dsp, XErrorEvent *event)
{
    char buff[1000];
    XGetErrorText(dsp, event->error_code, buff, 1000);
    warning("X11 protocol error: %s", buff);
    return 0;
}

static int R_X11IOErr(Display *dsp)
{
    error("X11 fatal IO error: please save work and shut down R");
    return 0; /* but should never get here */
}

/* set up the window, print the grid and column/row labels */

static Rboolean initwin(void) /* TRUE = Error */
{
    int i, twidth, w, minwidth;
    int ioscreen;
    unsigned long iowhite, ioblack;
    char ioname[] = "R DataEntryWindow";
    char digits[] = "123456789.0";
    Window root;
    XEvent ioevent;
    XSetWindowAttributes winattr;
    XWindowAttributes attribs;
    XWMHints hints;

    strcpy(copycontents, "");

#ifdef USE_FONTSET
    if (!XSupportsLocale ()) 
      warning("locale not supported by Xlib: some X ops will operate in C locale");
    if (!XSetLocaleModifiers ("")) warning("X cannot set locale modifiers");
#endif

    if ((iodisplay = XOpenDisplay(NULL)) == NULL) {
      warning("unable to open display");
      return TRUE;
    }
    XSetErrorHandler(R_X11Err);
    XSetIOErrorHandler(R_X11IOErr);

    /* Get Font Loaded if we can */

#ifdef USE_FONTSET
    if(mbcslocale) {
      int  missing_charset_count;
      char **missing_charset_list;
      char *def_string;
      font_set = XCreateFontSet(iodisplay, fontset_name, 
                          &missing_charset_list, 
                          &missing_charset_count, &def_string);
      if (missing_charset_count) XFreeStringList(missing_charset_list);
      if (font_set == NULL) {
          warning("unable to create fontset %s", fontset_name);
          return TRUE; /* ERROR */
      }
    } else
#endif
    {
      font_info = XLoadQueryFont(iodisplay, font_name);
      if (font_info == NULL) {
          warning("unable to losd font %s", font_name);
          return TRUE; /* ERROR */
      }
    }

    /* find out how wide the input boxes should be and set up the
       window size defaults */

    nboxchars = asInteger(GetOption(install("de.cellwidth"), R_GlobalEnv));
    if (nboxchars == NA_INTEGER || nboxchars < 0) nboxchars = 0;

    twidth = textwidth(digits, strlen(digits));
    if (nboxchars > 0) twidth = (twidth * nboxchars)/10;
    box_w = twidth + 4;
#ifdef USE_FONTSET
    if(mbcslocale) {
      XFontSetExtents *extent = XExtentsOfFontSet(font_set);
      char **ml;
      box_h = (extent->max_logical_extent.height)
          + (extent->max_logical_extent.height / 5) + 4;
      font_set_cnt = XFontsOfFontSet(font_set, &fs_list, &ml);
      text_offset = 2 + fs_list[0]->max_bounds.descent;
    } else
#endif
    {
      box_h = font_info->max_bounds.ascent
          + font_info->max_bounds.descent + 4;
      text_offset = 2 + font_info->max_bounds.descent;
    }
    windowHeight = 26 * box_h + hwidth + 2;
    /* this used to presume 4 chars sufficed for row numbering */
    labdigs = max(3, 1+floor(log10((double)ymaxused)));
    sprintf(labform, "%%%dd", labdigs);
    boxw[0] = 0.1*labdigs*textwidth("0123456789", 10) + textwidth(" ", 1) + 8;
    for(i = 1; i < 100; i++) boxw[i] = get_col_width(i);
    /* try for a window width that covers all the columns, or is around
       800 pixels */
    w = windowWidth = 0;
    for(i = 0; i <= xmaxused; i++) {
      w += boxw[i];
      if(w > 800) {
          windowWidth = w - boxw[i];
          break;
      }
    }
    if(windowWidth == 0) windowWidth = w;
    windowWidth += 2;
    /* allow enough width for buttons */
    minwidth = 7.5 * textwidth("Paste", 5);
    if(windowWidth < minwidth) windowWidth = minwidth;

    ioscreen = DefaultScreen(iodisplay);
    iowhite = WhitePixel(iodisplay, ioscreen);
    ioblack = BlackPixel(iodisplay, ioscreen);


    iohint.x = 0;
    iohint.y = 0;
    iohint.width = windowWidth;
    iohint.height = windowHeight;
    iohint.flags = PPosition | PSize;
    hints.flags = InputHint;
    hints.input = True;
    root = DefaultRootWindow(iodisplay);

    if ((iowindow = XCreateSimpleWindow(
           iodisplay,
           root,
           iohint.x,
           iohint.y,
           iohint.width,
           iohint.height,
           bwidth,
           ioblack,
           iowhite)) == 0) {
      warning("unable to open window for data editor");
      return TRUE;
    }

    XSetStandardProperties(iodisplay, iowindow, ioname, ioname, None,
                     (char **)NULL, 0, &iohint);

    winattr.backing_store = Always;
    XChangeWindowAttributes(iodisplay, iowindow, CWBackingStore, &winattr);

    /* set up protocols so that window manager sends */
    /* me an event when user "destroys" window */
    _XA_WM_PROTOCOLS = XInternAtom(iodisplay, "WM_PROTOCOLS", 0);
    protocol = XInternAtom(iodisplay, "WM_DELETE_WINDOW", 0);
    XSetWMProtocols(iodisplay, iowindow, &protocol, 1);
    XSetWMHints(iodisplay, iowindow, &hints);
#ifdef USE_FONTSET
    if(mbcslocale) {
      ioim = XOpenIM(iodisplay, NULL, NULL, NULL);
      if(!ioim) {
          XDestroyWindow(iodisplay, iowindow);
          XCloseDisplay(iodisplay);
          error("unable to open X Input Method");
      }

      ioic = XCreateIC(ioim, 
                   XNInputStyle, XIMPreeditNothing | XIMStatusNothing,
                   XNClientWindow, iowindow,
                   NULL);
      if(!ioic) {
          XCloseIM(ioim);
          XDestroyWindow(iodisplay, iowindow);
          XCloseDisplay(iodisplay);
          error("unable to open X Input Context");
      }
    }
#endif

    iogc = XCreateGC(iodisplay, iowindow, 0, 0);
#ifdef USE_FONTSET
    if(!mbcslocale)
#endif
      XSetFont(iodisplay, iogc, font_info->fid);
    XSetBackground(iodisplay, iogc, iowhite);
    XSetForeground(iodisplay, iogc, BlackPixel(iodisplay,
                                     DefaultScreen(iodisplay)));
    XSetLineAttributes(iodisplay, iogc, 1, LineSolid, CapRound, JoinRound);

    XSelectInput(iodisplay, iowindow,
             ButtonPressMask | KeyPressMask
             | ExposureMask | StructureNotifyMask);
    XMapRaised(iodisplay, iowindow);

    /* now set up the menu-window, for now use the same text
       dimensions as above */

    menuwindow = XCreateSimpleWindow(iodisplay, root, 0, 0, twidth,
                             4 * box_h, 2, ioblack, iowhite);
    for (i = 0; i < 4; i++) {
      menupanes[i] = XCreateSimpleWindow(iodisplay, menuwindow, 0,
                                 box_h * i, twidth, box_h,
                                 1, ioblack, iowhite);
      XSelectInput(iodisplay, menupanes[i],
                 ButtonPressMask | ButtonReleaseMask | ExposureMask);
    }

    /* XMapSubwindows(iodisplay, menuwindow); */


    winattr.override_redirect = True;
    XChangeWindowAttributes(iodisplay, menuwindow,
                      CWBackingStore | CWOverrideRedirect, &winattr);
    Rsync();

    /* this next sequence makes sure the window is up and ready before
       you start drawing in it */

    XNextEvent(iodisplay, &ioevent);
    if (ioevent.xany.type == Expose) {
      while (ioevent.xexpose.count)
          XNextEvent(iodisplay, &ioevent);
    }
    XGetWindowAttributes(iodisplay, iowindow, &attribs);
    bwidth = attribs.border_width;
    fullwindowWidth = attribs.width;
    fullwindowHeight = attribs.height;


    /* set the active rectangle to be the upper left one */
    crow = 1;
    ccol = 1;
    CellModified = FALSE;
    return FALSE;/* success */
}

/* MAC/X11 BASICS */

static void bell(void)
{
    XBell(iodisplay, 20);
}

static void cleararea(int xpos, int ypos, int width, int height)
{
    XClearArea(iodisplay, iowindow, xpos, ypos, width, height, 0);
}

static void clearwindow(void)
{
    XClearWindow(iodisplay, iowindow);
}

static void copyarea(int src_x, int src_y, int dest_x, int dest_y)
{
    int mx = max(src_x, dest_x), my = max(src_y, dest_y);
    XCopyArea(iodisplay, iowindow, iowindow, iogc, src_x, src_y,
            fullwindowWidth - mx, fullwindowHeight - my,
            dest_x, dest_y);
    Rsync();
}

static void copyH(int src_x, int dest_x, int width)
{
    XCopyArea(iodisplay, iowindow, iowindow, iogc, src_x+bwidth, hwidth,
            width, windowHeight+1, dest_x+bwidth, hwidth);
}

#if 0
static void drawline(int fromx, int fromy, int tox, int toy)
{
    XDrawLine(iodisplay, iowindow, iogc, fromx, fromy, tox, toy);
}
#endif

static void drawrectangle(int xpos, int ypos, int width, int height,
                    int lwd, int fore)
{
    if (fore == 0)
      XSetForeground(iodisplay, iogc, WhitePixel(iodisplay,
                                       DefaultScreen(iodisplay)));
    else
      XSetForeground(iodisplay, iogc, BlackPixel(iodisplay,
                                       DefaultScreen(iodisplay)));
    XSetLineAttributes(iodisplay, iogc, lwd, LineSolid, CapRound, JoinRound);
    XDrawRectangle(iodisplay, iowindow, iogc, xpos, ypos, width, height);
}

static void drawtext(int xpos, int ypos, char *text, int len)
{
#ifdef USE_FONTSET
    if(mbcslocale)
#ifdef HAVE_XUTF8DRAWIMAGESTRING
      Xutf8DrawImageString(iodisplay, iowindow, font_set, iogc, xpos, ypos,
                       text, len);
#else
      XmbDrawImageString(iodisplay, iowindow, font_set, iogc, xpos, ypos,
                     text, len);
#endif
    else
#endif
      XDrawImageString(iodisplay, iowindow, iogc, xpos, ypos, text, len);
    Rsync();
}

static void Rsync()
{
    XSync(iodisplay, 0);
}

static int textwidth(char *text, int nchar)
{
#ifdef USE_FONTSET
    if(mbcslocale)
#ifdef HAVE_XUTF8TEXTESCAPEMENT
      return Xutf8TextEscapement(font_set, text, nchar);
#else
      return XmbTextEscapement(font_set, text, nchar);
#endif
#endif
    return XTextWidth(font_info, text, nchar);
}

/* Menus */

void popupmenu(int x_pos, int y_pos, int col, int row)
{
    int i, button, popupcol = col + colmin - 1;
    char *name, clab[20];
    XEvent event;
    Window selected_pane;
    SEXP tvec;

    XMapSubwindows(iodisplay, menuwindow);
    XMapRaised(iodisplay, menuwindow);
    XMoveWindow(iodisplay, menuwindow, x_pos, y_pos);

    /* now fill in the menu panes with the correct information */

    if (popupcol > xmaxused) {
      /* extend work, names and lens */
      REPROTECT(work = lengthgets(work, popupcol), wpi);
      REPROTECT(names = lengthgets(names, popupcol), npi);
      for (i = xmaxused+1; i < popupcol; i++) {
          sprintf(clab, "var%d", i + 1);
          SET_STRING_ELT(names, i, mkChar(clab));
      }
      REPROTECT(lens = lengthgets(lens, popupcol), lpi);
      xmaxused = popupcol;
    }
    tvec = VECTOR_ELT(work, popupcol - 1);
    name = CHAR(STRING_ELT(names, popupcol - 1));
    XDrawString(iodisplay,
            menupanes[0], iogc, 3, box_h - 3, name, strlen(name));
    for (i = 1; i < 4; i++)
      XDrawString(iodisplay,
                menupanes[i], iogc, 3, box_h - 3,
                menu_label[i - 1], strlen(menu_label[i - 1]));
    if (isNull(tvec) || TYPEOF(tvec) == REALSXP)
      XDrawString(iodisplay, menupanes[1], iogc, box_w - 20, box_h - 3,
                "X", 1);
    else
      XDrawString(iodisplay, menupanes[2], iogc, box_w - 20, box_h - 3,
                "X", 1);

/*
  start an event loop; we're looking for a button press and a button
  release in the same window
*/

    while (1) {
      XNextEvent(iodisplay, &event);
      if (event.type == ButtonPress) {
          button = event.xbutton.button;
          selected_pane = event.xbutton.window;
          for (i = 0; selected_pane != menupanes[i]; i++)
            if (i >= 4) goto done;
          while (1) {
            while (XCheckTypedEvent(iodisplay, ButtonPress, &event));
            XMaskEvent(iodisplay, ButtonReleaseMask, &event);
            if (event.xbutton.button == button)
                break;
          }
          if (selected_pane == event.xbutton.window) {
            for (i = 0; selected_pane != menupanes[i]; i++);
            switch (i) {
            case 0:
                bell();
                break;
            case 1:
                if (isNull(tvec))
                  SET_VECTOR_ELT(work, popupcol - 1,
                               ssNewVector(REALSXP, 100));
                else
                  SET_VECTOR_ELT(work, popupcol - 1,
                               coerceVector(tvec, REALSXP));
                goto done;
            case 2:
                if (isNull(tvec))
                  SET_VECTOR_ELT(work, popupcol - 1,
                               ssNewVector(STRSXP, 100));
                else
                  SET_VECTOR_ELT(work, popupcol - 1,
                               coerceVector(tvec, STRSXP));
                goto done;
            case 3:
                closerect();
                ccol = col;
                crow = 0;
                clearrect();
                goto done;
            }
          }
      }
        /* this doesn't work and perhaps I should move it up to the
           main control loop */
      else if (event.type == Expose) {
          if (event.xexpose.window == menuwindow) {
            XDrawString(iodisplay, menupanes[0], iogc, 3, box_h - 3,
                      name, strlen(name));
            for (i = 1; i < 4; i++)
                XDrawString(iodisplay, menupanes[i], iogc, 3, box_h - 3,
                        menu_label[i - 1], strlen(menu_label[i - 1]));
          }
      }
    }
 done:
    popdownmenu();
    highlightrect();
}

void popdownmenu(void)
{
    XUnmapWindow(iodisplay, menuwindow);
    XUnmapSubwindows(iodisplay, menuwindow);
}

static void copycell(void)
{
    int i, whichrow = crow + colmin - 1, whichcol = ccol + colmin -1;
    SEXP tmp;

    if (whichrow == 0) {
      /* won't have  cell here */
    } else {
      strcpy(copycontents, "");
      if (length(work) >= whichcol) {
          tmp = VECTOR_ELT(work, whichcol - 1);
          if (tmp != R_NilValue &&
            (i = whichrow - 1) < LENGTH(tmp) ) {
            PrintDefaults(R_NilValue);
            if (TYPEOF(tmp) == REALSXP) {
                if (REAL(tmp)[i] != ssNA_REAL)
                  strcpy(copycontents, EncodeElement(tmp, i, 0));
            } else if (TYPEOF(tmp) == STRSXP) {
                if (!streql(CHAR(STRING_ELT(tmp, i)),
                        CHAR(STRING_ELT(ssNA_STRING, 0))))
                  strcpy(copycontents, EncodeElement(tmp, i, 0));
            }
          }
      }
    }
    highlightrect();
}

static void pastecell(int row, int col)
{
    downlightrect();
    crow = row; ccol = col;
    if (strlen(copycontents)) {
      strcpy(buf, copycontents);
      clength = strlen(copycontents);
      bufp = buf + clength;
      CellModified = TRUE;
    }
    closerect();
    highlightrect();
}

Generated by  Doxygen 1.6.0   Back to index