home *** CD-ROM | disk | FTP | other *** search
/ vsiftp.vmssoftware.com / VSIPUBLIC@vsiftp.vmssoftware.com.tar / FREEWARE / FREEWARE40.ZIP / xv310a / xvpopup.c < prev    next >
C/C++ Source or Header  |  1995-06-12  |  35KB  |  1,241 lines

  1. /* 
  2.  * xvpopup.c - popup "Are you sure?  Yes/No/Maybe" sort of dialog box
  3.  *
  4.  * callable functions:
  5.  *
  6.  *   CenterMapWindow(win,x,y)  -  maps and centers a window around the mouse
  7.  *   PopUp(str,...)        -  maps, sets up popW
  8.  *   ErrPopUp(str,str)     -  maps, sets up popW
  9.  *   GetStrPopUp(...)      -  opens a 1-line, editable text popup window
  10.  *   GrabPopUp(*hide,*del) -  opens 'grab' popup dialog
  11.  *   PadPopUp()            -  opens 'grab' popup dialog
  12.  *   ClosePopUp()          -  closes pop-up or alert window, if open
  13.  *   OpenAlert(str)        -  maps a button-less window
  14.  *   CloseAlert()          -  closes a button-less window
  15.  *   PUCheckEvent(event)   -  called by event handler
  16.  *   TextRect()            -  draws semi-complex strings in a rectangle
  17.  */
  18.  
  19. #include "copyright.h"
  20.  
  21. #include "xv.h"
  22.  
  23. #include "bits/icon"
  24.  
  25. #define PUWIDE 400
  26. #define PUHIGH 170
  27.  
  28. #define PAD_PUWIDE 480
  29. #define PAD_PUHIGH 215
  30.  
  31. #define BUTTH   24
  32.  
  33. static int  doPopUp       PARM((char *, char **, int, int, char *));
  34. static void attachPUD     PARM((void));
  35. static void createPUD     PARM((void));
  36. static void drawPUD       PARM((int, int, int, int));
  37. static void drawPadOMStr  PARM((void));
  38. static void clickPUD      PARM((int, int));
  39. static void doGetStrKey   PARM((int));
  40. static int  doGSKey       PARM((int));
  41. static void changedGSBuf  PARM((void));
  42. static void drawGSBuf     PARM((void));
  43. static void buildPadLists PARM((void));
  44. static void build1PadList PARM((char *, char **, char **, int *,
  45.                 char **, char **, int));
  46.  
  47.  
  48. /* values 'popUp' can take */
  49. #define ISPOPUP  1
  50. #define ISALERT  2
  51. #define ISGETSTR 3
  52. #define ISGRAB   4
  53. #define ISPAD    5
  54.  
  55. #define DELAYSTR "Delay:"
  56. #define SECSTR   "seconds"
  57. #define HIDESTR  "Hide XV windows"
  58.  
  59. /* local variables */
  60. Window popW;
  61. int    nbts, selected, popUp=0, firsttime=1;
  62. int    puwide = PUWIDE;
  63. int    puhigh = PUHIGH;
  64. BUTT  *bts;
  65. char  *text;
  66. char   accel[8];
  67.  
  68. char *gsBuf, *gsFilter;       /* stuff needed for GetStrPopUp() handling */
  69. int   gsBufLen, gsAllow, gsCurPos, gsStPos, gsEnPos;
  70. int   gsx, gsy, gsw, gsh;
  71.  
  72. /* stuff for GrabPopUp */
  73. CBUTT ahideCB;
  74.  
  75.  
  76. /*** stuff for PadPopUp ***/
  77. static char   padSbuf[256], padBbuf[256], padLbuf[256], padBuf[256];
  78. static char  *padInst, padSinst[200], padBinst[200], padLinst[200];
  79. static MBUTT  padDfltMB, padMthdMB;
  80. static BUTT   padDButt, padOMButt;
  81. static int    padHaveDooDads = 0;
  82. static int    padMode, padOMode;
  83. static DIAL   padWDial, padHDial, padODial;
  84.  
  85. static int    padMthdLen=3;
  86. static char  *padMthdNames[] = { "Solid Fill", "Run 'bggen'", "Load Image" };
  87.  
  88. static int    padColDefLen = 9;
  89. static char  *padColDefNames[] = { "black", "red",  "yellow", "green", 
  90.                    "cyan",  "blue", "magenta", "white", 
  91.                    "50% gray" };
  92.  
  93. static char  *padColDefVals[]  = { "black", "red", "yellow", "green",
  94.                    "cyan",  "blue", "magenta", "white",
  95.                                    "gray50" };
  96.  
  97. static int    padBgDefLen = 8;
  98. static char  *padBgDefNames[] = {
  99.   "Black->White",
  100.   "Blue Gradient",
  101.   "RGB Rainbow",
  102.   "Full Rainbow",
  103.   "Color Assortment",
  104.   "Green Tiles",
  105.   "Red Balls",
  106.   "Red+Yellow Diamonds" };
  107.  
  108. static char  *padBgDefVals[]  = { 
  109.   "black white", 
  110.   "100 100 255  50 50 150",
  111.   "red green blue",
  112.   "black red yellow green blue purple black",
  113.   "black white red black yellow white green black cyan white blue black magenta white red yellow green cyan blue magenta red",
  114.   "green black -r 30 -G 32x32",
  115.   "red black -r 45 -G 32x32",
  116.   "red yellow -r 45 -G 32x32" };
  117.  
  118.  
  119. /* this should match with PAD_O* defs in xv.h */
  120. static char *padOMStr[] = { "RGB", "Int.", "Hue", "Sat." };
  121.  
  122. #define PAD_MAXDEFLEN 10
  123. static int    padColLen = 0, padBgLen = 0, padLoadLen = 0;
  124. static char  *padColNames [PAD_MAXDEFLEN], *padColVals [PAD_MAXDEFLEN];
  125. static char  *padBgNames  [PAD_MAXDEFLEN], *padBgVals  [PAD_MAXDEFLEN];
  126. static char  *padLoadNames[PAD_MAXDEFLEN], *padLoadVals[PAD_MAXDEFLEN];
  127.  
  128.  
  129. /***************************************************/
  130. void CenterMapWindow(win, dx, dy, w, h)
  131.      Window win;
  132.      int    dx, dy, w, h;
  133. {
  134.   XSizeHints hints;
  135.   Window       rW,cW;
  136.   int          rx,ry,x,y,wx,wy;
  137.   unsigned int mask;
  138.  
  139.  
  140.   if (!XQueryPointer(theDisp,rootW,&rW,&cW,&rx,&ry,&x,&y,&mask)) {
  141.     /* couldn't query mouse.  just center on screen */
  142.     wx = (dispWIDE-w)/2;   wy = (dispHIGH-h)/2;
  143.   }
  144.   else {
  145.     wx = x - dx;
  146.     wy = y - dy;
  147.     if (wx<0) wx = 0;
  148.     if (wy<0) wy = 0;
  149.     if (wx + w > dispWIDE) wx = dispWIDE - w;
  150.     if (wy + h > dispHIGH) wy = dispHIGH - h;
  151.   }
  152.  
  153.   
  154.   if (winCtrPosKludge) {
  155.     wx -= (p_offx + ch_offx);
  156.     wy -= (p_offy + ch_offy);
  157.   } 
  158.   else {
  159.     wx -= (ch_offx);
  160.     wy -= (ch_offy);
  161.   }
  162.  
  163.   if (!XGetNormalHints(theDisp, win, &hints)) hints.flags = 0;
  164.   hints.width  = hints.min_width  = hints.max_width  = w;
  165.   hints.height = hints.min_height = hints.max_height = h;
  166.   hints.x = wx;  hints.y = wy;
  167.   hints.flags  |= (USSize | PMinSize | PMaxSize | USPosition);
  168.   XSetNormalHints(theDisp, win, &hints);
  169.  
  170.   XMoveWindow(theDisp, win, wx, wy);
  171.   XMapRaised(theDisp, win);
  172. }
  173.  
  174.  
  175. /***************************************************/
  176. int PopUp(txt, labels, n)
  177.      char *txt, *labels[];
  178.      int   n;
  179. {
  180.   return doPopUp(txt, labels, n, ISPOPUP, "xv confirm");
  181. }
  182.  
  183. /***************************************************/
  184. static int doPopUp(txt, labels, n, poptyp, wname)
  185.      char *txt, *labels[], *wname;
  186.      int   n, poptyp;
  187. {
  188.   int    i;
  189.   XEvent event;
  190.  
  191.   if (firsttime) createPUD();
  192.  
  193.   if (poptyp != ISPAD) { puwide = PUWIDE;      puhigh = PUHIGH;     }
  194.                   else { puwide = PAD_PUWIDE;  puhigh = PAD_PUHIGH; }
  195.  
  196.  
  197.   /* attach controls to popW, now that it exists */
  198.   if      (poptyp==ISGRAB) ahideCB.win = popW;
  199.   else if (poptyp == ISPAD) {
  200.     
  201.     if (!padHaveDooDads) {
  202.       DCreate(&padWDial, popW, 16,      puhigh-16-100-1,75,100,
  203.           1, 2048, pWIDE, 10,
  204.           infofg, infobg, hicol, locol, "Width", NULL);
  205.       DCreate(&padHDial, popW, 16+1+75, puhigh-16-100-1,75,100,
  206.           1, 2048, pHIGH, 10,
  207.           infofg, infobg, hicol, locol, "Height", NULL);
  208.  
  209.       DCreate(&padODial, popW, 16+1+75+75+9, puhigh-16-100-1,75,100,
  210.           0, 100, 100, 10,
  211.           infofg, infobg, hicol, locol, "Opaque", NULL);
  212.  
  213.       MBCreate(&padMthdMB, popW, 100-2+44, 10, 140, 19, NULL,
  214.            padMthdNames, padMthdLen, infofg, infobg, hicol, locol);
  215.       padMthdMB.hascheck = 1;
  216.       padMthdMB.flags[0] = 1;
  217.  
  218.       MBCreate(&padDfltMB, popW, 250-2+44, 10, 140, 19, "Defaults",
  219.            padColNames, padColLen, infofg, infobg, hicol, locol);
  220.  
  221.       BTCreate(&padDButt, popW, padHDial.x+padHDial.w-12, puhigh-140+6,
  222.            13,13, "", infofg, infobg, hicol, locol);
  223.  
  224.       BTCreate(&padOMButt, popW, padODial.x+padODial.w-12, puhigh-140+6,
  225.            13,13, "", infofg, infobg, hicol, locol);
  226.  
  227.       padHaveDooDads = 1;
  228.     }
  229.  
  230.     XMapWindow(theDisp, padWDial.win);
  231.     XMapWindow(theDisp, padHDial.win);
  232.     XMapWindow(theDisp, padODial.win);
  233.   }      
  234.   
  235.  
  236.   XResizeWindow(theDisp, popW, (u_int) puwide, (u_int) puhigh);
  237.   XStoreName   (theDisp, popW, wname);
  238.   XSetIconName (theDisp, popW, wname);
  239.   attachPUD();
  240.  
  241.   bts = (BUTT *) malloc(n * sizeof(BUTT));
  242.   if (!bts) FatalError("unable to malloc buttons in popup\n");
  243.   nbts = n;
  244.   selected = 0;
  245.   text = txt;
  246.  
  247.   for (i=0; i<n; i++) {
  248.     BTCreate(&bts[i], popW, puwide - (n-i) * (80 + 10), puhigh - 10 - BUTTH,
  249.          80, BUTTH, labels[i]+1, infofg, infobg, hicol, locol);
  250.     accel[i] = labels[i][0];
  251.   }
  252.  
  253.  
  254.   if (poptyp == ISGRAB) {
  255.     BTSetActive(&bts[0], (int) strlen(gsBuf));
  256.     BTSetActive(&bts[1], (strlen(gsBuf)>(size_t)0 && atoi(gsBuf)>(size_t)0));
  257.   }
  258.   else if (poptyp == ISPAD) {
  259.     BTSetActive(&bts[0], (int) strlen(gsBuf));
  260.     i = pWIDE * 3;  RANGE(i,2048,9999);  
  261.     DSetRange(&padWDial, 1, i, padWDial.val, 10);
  262.     i = pHIGH * 3;  RANGE(i,2048,9999);  
  263.     DSetRange(&padHDial, 1, i, padHDial.val, 10);
  264.  
  265.     DSetActive(&padWDial, (padMode!=PAD_LOAD));  /* DSetRange activates dial */
  266.     DSetActive(&padHDial, (padMode!=PAD_LOAD));
  267.     DSetActive(&padODial, 1);
  268.  
  269.     switch (padMode) {
  270.     case PAD_SOLID:
  271.       padDfltMB.list  = padColNames;
  272.       padDfltMB.nlist = padColLen;
  273.       break;
  274.     case PAD_BGGEN:
  275.       padDfltMB.list  = padBgNames;
  276.       padDfltMB.nlist = padBgLen;
  277.       break;
  278.     case PAD_LOAD:
  279.       padDfltMB.list  = padLoadNames;
  280.       padDfltMB.nlist = padLoadLen;
  281.       break;
  282.     default: break;             /* shouldn't happen */
  283.     }
  284.   }
  285.  
  286.   /* center first button in window around mouse position, with constraint that 
  287.      window be fully on the screen */
  288.  
  289.   CenterMapWindow(popW, 40 + bts[0].x, BUTTH/2 + bts[0].y, puwide, puhigh);
  290.   popUp = poptyp;
  291.  
  292.   /* MUST wait for VisibilityNotify event to come in, else we run the risk
  293.      of UnMapping the window *before* the Map request completed.  This 
  294.      appears to be bad, (It leaves an empty window frame up.) though it
  295.      generally only happens on slow servers.  Better safe than screwed... */
  296.  
  297.   XWindowEvent(theDisp, popW, VisibilityChangeMask, &event);
  298.  
  299.   /* block until this window gets closed */
  300.   while (popUp) {
  301.     XNextEvent(theDisp, &event);
  302.     HandleEvent(&event, &i);
  303.   }
  304.  
  305.   /* free stuff */
  306.   XUnmapWindow(theDisp, popW);
  307.   free(bts);
  308.  
  309.   return(selected);
  310. }
  311.  
  312.  
  313. /***************************************************/
  314. void ErrPopUp(txt, label)
  315.      char *txt, *label;
  316. {
  317.   /* simplified interface to PopUp.  Takes a string and the label for the
  318.      (one) button */
  319.  
  320.   PopUp(txt, &label, 1);
  321. }
  322.  
  323.  
  324. /***************************************************/
  325. int GetStrPopUp(txt, labels, n, buf, buflen, filstr, allow)
  326.      char *txt, *labels[], *buf, *filstr;
  327.      int   n, buflen, allow;
  328. {
  329.   /* pops up a window with a prompt string, a 1-line editable
  330.      text thingy, and a row of buttons.  'txt' is the prompt
  331.      string, 'labels' are the labels for the buttons, 'n' is the
  332.      number of buttons, 'buf' is the buffer displayed and edited
  333.      in the window, buflen is its length, filstr is a filter string, of
  334.      characters to block from entry (or, if 'allow' is '1', a list
  335.      of the *only* characters allowed for entry)
  336.  
  337.      It returns the index of the button clicked on.  Note that the
  338.      button labels have 1-character accellerators at the front, same
  339.      as in PopUp().  Note that it would be suboptimal to make any
  340.      of the 1-character accellerators be the same character as one of
  341.      the edit-text command keys 
  342.  
  343.      Also note that the filter string should only contain normal printable
  344.      characters (' ' through '\177'), as ctrl chars are pre-filtered 
  345.      (ie, interpreted as emacs-like commands) */
  346.  
  347.   gsBuf = buf;        gsBufLen = buflen;
  348.   gsFilter = filstr;  gsAllow = allow;
  349.  
  350.   gsCurPos = strlen(gsBuf);
  351.   gsStPos = gsEnPos = 0;
  352.  
  353.   gsh = LINEHIGH+5;
  354.   gsx = 10 + icon_width + 20;
  355.   gsy = 10+(PUHIGH-30-BUTTH-gsh)/2;
  356.  
  357.   if (strlen(txt) > (size_t) 60)
  358.     gsy = PUHIGH - 10 - BUTTH - 10 - gsh - 20;
  359.  
  360.   gsw = PUWIDE - gsx - 10;
  361.   
  362.   changedGSBuf();      /* careful!  popW doesn't exist yet! */
  363.  
  364.   return doPopUp(txt, labels, n, ISGETSTR, "xv prompt");
  365. }
  366.  
  367.  
  368. /***************************************************/
  369. int GrabPopUp(pHide, pDelay)
  370.      int *pHide, *pDelay;
  371. {
  372.   /* pops up Grab options dialog box */
  373.  
  374.   int  rv;
  375.   char delaybuf[32], grabTxt[1024];
  376.   static char *grabLabels[] = { "\nGrab", "aAutoGrab", "\033Cancel" };
  377.  
  378.   sprintf(delaybuf,"%d", *pDelay);
  379.   gsBuf = delaybuf;          gsBufLen = 3;
  380.   gsFilter = "0123456789";   gsAllow = 1;
  381.  
  382.   gsCurPos = strlen(gsBuf);
  383.   gsStPos = gsEnPos = 0;
  384.  
  385.   gsw = 32;
  386.   gsh = LINEHIGH+5;
  387.   gsx = 10 + StringWidth(DELAYSTR) + 5;
  388.   gsy = (PUHIGH-BUTTH-10-5-gsh);
  389.  
  390.   changedGSBuf();      /* careful!  popW doesn't exist yet! */
  391.  
  392.   /* window value gets filled in in doPopUp() */
  393.   CBCreate(&ahideCB, (Window) NULL, 
  394.        PUWIDE-10-18-StringWidth(HIDESTR),
  395.        gsy+2, HIDESTR, infofg, infobg, hicol, locol);
  396.   ahideCB.val = *pHide;
  397.  
  398.   sprintf(grabTxt, "Grab: after delay, Left button grabs a window, ");
  399.   strcat (grabTxt, "Middle button ");
  400.   strcat (grabTxt, "grabs a rectangular area, Right button cancels.\n\n");
  401.   strcat (grabTxt, "AutoGrab: after delay, grabs ");
  402.   strcat (grabTxt, "the window the cursor is positioned in.  ");
  403.   strcat (grabTxt, "Delay must be non-zero.");
  404.  
  405.   rv = doPopUp(grabTxt, grabLabels, 3, ISGRAB, "xv grab");
  406.  
  407.   *pHide  = ahideCB.val;
  408.   *pDelay = atoi(delaybuf);
  409.   return rv;
  410. }
  411.  
  412.  
  413. /***************************************************/
  414. int PadPopUp(pMode, pStr, pWide,pHigh, pOpaque, pOmode)
  415.      int  *pMode, *pWide, *pHigh, *pOpaque, *pOmode;
  416.      char **pStr;
  417. {
  418.   /* pops up 'Pad' options dialog box */
  419.  
  420.   int         rv, oldW, oldH, oldO;
  421.   static int  firsttime=1;
  422.   static char *labels[] = { "\nOk", "\033Cancel" };
  423.  
  424.   if (firsttime) {
  425.     padSbuf[0] = '\0';
  426.     padBbuf[0] = '\0';
  427.     padLbuf[0] = '\0';
  428.  
  429.     sprintf(padSinst, "Enter a color name ('orange'), %s%s",
  430.         "or an RGB color specification.  ",
  431.         "(e.g. 'r,g,b' or '0xrrggbb')");
  432.     sprintf(padBinst, "Enter command line options for 'bggen'.  (%s)",
  433.         "No '-w', '-h', or '-g' options allowed.");
  434.     sprintf(padLinst, "Enter a filename.  The padded image %s",
  435.         "will be the same size as the loaded image.");
  436.  
  437.     /* can't create MBUTT or DIALs here, parent window must exist first... */
  438.  
  439.     padMode = PAD_SOLID;
  440.     padInst = padSinst;
  441.     padOMode = PAD_ORGB;
  442.     firsttime = 0;
  443.   }
  444.  
  445.  
  446.   buildPadLists();
  447.  
  448.   switch (padMode) {
  449.   case PAD_SOLID:  strcpy(padBuf, padSbuf);  break;
  450.   case PAD_BGGEN:  strcpy(padBuf, padBbuf);  break;
  451.   case PAD_LOAD:   strcpy(padBuf, padLbuf);  break;
  452.   }
  453.  
  454.  
  455.   gsBuf    = padBuf;         gsBufLen = 256;
  456.   gsFilter = "";             gsAllow  = 0;
  457.   gsCurPos = strlen(gsBuf);
  458.   gsStPos  = gsEnPos = 0;
  459.  
  460.   gsw = PAD_PUWIDE - 20;
  461.   gsh = LINEHIGH+5;
  462.   gsx = 10;
  463.   gsy = 40;
  464.  
  465.   changedGSBuf();      /* careful!  popW doesn't exist yet! */
  466.  
  467.   if (padHaveDooDads) { 
  468.     oldW = padWDial.val;  
  469.     oldH = padHDial.val;
  470.     oldO = padODial.val;
  471.   }
  472.   else { oldW = pWIDE;  oldH = pHIGH;  oldO = 100; }
  473.  
  474.  
  475.  
  476.   rv = doPopUp("", labels, 2, ISPAD, "xv pad");
  477.  
  478.  
  479.  
  480.   if (rv == 0) {  /* copy padBuf to appropriate mode buffer */
  481.     switch (padMode) {
  482.     case PAD_SOLID:  strcpy(padSbuf, padBuf);  break;
  483.     case PAD_BGGEN:  strcpy(padBbuf, padBuf);  break;
  484.     case PAD_LOAD:   strcpy(padLbuf, padBuf);  break;
  485.     }
  486.   }
  487.  
  488.   if (rv == 1) {   /* cancelled:  restore normal values */
  489.     DSetVal(&padWDial, oldW);
  490.     DSetVal(&padHDial, oldH);
  491.     DSetVal(&padODial, oldO);
  492.   }
  493.  
  494.   XUnmapWindow(theDisp, padWDial.win);
  495.   XUnmapWindow(theDisp, padHDial.win);
  496.   XUnmapWindow(theDisp, padODial.win);
  497.  
  498.   /* load up return values */
  499.   *pMode   = padMode;  
  500.   *pStr    = padBuf;  
  501.   *pWide   = padWDial.val;
  502.   *pHigh   = padHDial.val;
  503.   *pOpaque = padODial.val;
  504.   *pOmode  = padOMode;
  505.  
  506.   return rv;
  507. }
  508.  
  509.  
  510. /***************************************************/
  511. static void buildPadLists()
  512. {
  513.   /* generates padCol* and padBg* lists used in 'Defaults' MBUTT.  Grabs
  514.      all the X resources values it can, and adds appropriate defaults */
  515.  
  516.   int  i;
  517.  
  518.   rd_str_cl("foo", "", 1);                    /* rebuild database */
  519.  
  520.   build1PadList("color", padColVals, padColNames, &padColLen,
  521.         padColDefVals, padColDefNames, padColDefLen);
  522.  
  523.   build1PadList("bggen", padBgVals, padBgNames, &padBgLen,
  524.         padBgDefVals, padBgDefNames, padBgDefLen);
  525.  
  526.   build1PadList("load", padLoadVals, padLoadNames, &padLoadLen,
  527.         (char **) NULL, (char **) NULL, 0);
  528. }
  529.     
  530.       
  531. /***************************************************/
  532. static void build1PadList(typstr, vals, nams, lenp, dvals, dnams, dlen)
  533.      char *typstr, **vals, **nams, **dvals, **dnams;
  534.      int  *lenp, dlen;
  535. {
  536.   int i;
  537.   char resname[128], name[256], value[256];
  538.  
  539.   for (i=0; i<*lenp; i++) {   /* kill old lists */
  540.     free(nams[i]);
  541.     free(vals[i]);
  542.   }
  543.   *lenp = 0;
  544.  
  545.   for (i=0; i<10; i++) {
  546.     sprintf(resname, "pad.%s.val%d", typstr, i);
  547.     if (rd_str_cl(resname, "Dialog.Menu.Slot",0)) {    /* got one! */
  548.       vals[*lenp] = (char *) malloc(strlen(def_str)+1);
  549.       if (!vals[*lenp]) continue;
  550.       strcpy(vals[*lenp], def_str);
  551.       
  552.       sprintf(resname, "pad.%s.name%d", typstr, i);
  553.       if (rd_str_cl(resname, "Dialog.Menu.Slot",0)) {  /* and it has a name! */
  554.     nams[*lenp] = (char *) malloc(strlen(def_str)+1);
  555.     if (!nams[*lenp]) { free(vals[*lenp]); continue; }
  556.     strcpy(nams[*lenp], def_str);
  557.  
  558.       }
  559.       else {  /* it doesn't have a name.  fabricate one */
  560.     nams[*lenp] = (char *) malloc((size_t) 32);
  561.     if (!nams[*lenp]) { free(vals[*lenp]); continue; }
  562.     strncpy(nams[*lenp], vals[*lenp], (size_t) 31);
  563.       }
  564.       
  565.       if (strlen(nams[*lenp]) > (size_t) 20) {   /* fix long names */
  566.     char *sp = nams[*lenp] + 18;
  567.     *sp++ = '.';  *sp++ = '.';  *sp++ = '.';  *sp++ = '\0';
  568.       }
  569.     
  570.       *lenp = (*lenp) + 1;
  571.     }
  572.   }
  573.  
  574.  
  575.   /* add 'built-in' defaults to the lists */
  576.   for (i=0; i<dlen && *lenp<PAD_MAXDEFLEN; i++) {
  577.     vals[*lenp] = (char *) malloc(strlen(dvals[i])+1);
  578.     if (!vals[*lenp]) break;
  579.     strcpy(vals[*lenp], dvals[i]);
  580.  
  581.     nams[*lenp] = (char *) malloc(strlen(dnams[i])+1);
  582.     if (!nams[*lenp]) { free(vals[*lenp]); break; }
  583.     strcpy(nams[*lenp], dnams[i]);
  584.     *lenp = (*lenp) + 1;
  585.   }
  586. }    
  587.  
  588.  
  589.  
  590. /***************************************************/
  591. void ClosePopUp()
  592. {
  593.   /* closes popW:  if it's a pop-up, returns 'cancel'.  If it's an alert,
  594.      simply closes it */
  595.  
  596.   if      (popUp == ISALERT) CloseAlert();
  597.   else if (popUp == ISPOPUP) {
  598.     popUp = 0;
  599.     selected = nbts-1;
  600.   }
  601. }
  602.  
  603.  
  604. /***************************************************/
  605. void OpenAlert(txt)
  606.      char *txt;
  607. {
  608.   /* pops up a window with txt displayed in it (*no buttons*).  
  609.      returns immediately.  window is closed by 'CloseAlert()'.
  610.      No 'PopUp()' calls are allowed while an Alert is displayed. */
  611.  
  612.   XEvent event;
  613.  
  614.   if (firsttime) createPUD();
  615.  
  616.   XStoreName(theDisp, popW, "xv notice");
  617.   XSetIconName(theDisp, popW, "xv notice");
  618.   attachPUD();
  619.  
  620.   nbts = 0;
  621.   selected = 0;
  622.   text = txt;
  623.  
  624.   puwide = PUWIDE;  puhigh = PUHIGH;
  625.   XResizeWindow(theDisp, popW, (u_int) puwide, (u_int) puhigh);
  626.  
  627.   /* center last button in window around mouse position, with constraint that 
  628.      window be fully on the screen */
  629.  
  630.   CenterMapWindow(popW, puwide/2, puhigh/2, puwide, puhigh);
  631.   popUp = ISALERT;
  632.  
  633.   /* MUST wait for VisibilityNotify event to come in, else we run the risk
  634.      of UnMapping the window *before* the Map request completed.  This 
  635.      appears to be bad, (It leaves an empty window frame up.) though it
  636.      generally only happens on slow servers.  Better safe than screwed... */
  637.  
  638.   XWindowEvent(theDisp, popW, VisibilityChangeMask, &event);
  639.   drawPUD(0, 0, puwide, puhigh);
  640.   XFlush(theDisp);
  641. }
  642.  
  643.  
  644. /***************************************************/
  645. void CloseAlert()
  646. {
  647.   popUp = 0;
  648.   XUnmapWindow(theDisp, popW);
  649. }
  650.  
  651.  
  652. /***************************************************/
  653. int PUCheckEvent(xev)
  654.      XEvent *xev;
  655. {
  656.   /* check event to see if it's for us.  If so, return 1, otherwise 0 */
  657.  
  658.   int rv = 0;
  659.  
  660.   if (!popUp) return(0);
  661.  
  662.   if (xev->type == Expose) {
  663.     XExposeEvent *e = (XExposeEvent *) xev;
  664.     if (e->window == popW) {
  665.       drawPUD(e->x, e->y, e->width, e->height);
  666.       rv = 1;
  667.     }
  668.     else if (popUp == ISPAD && padHaveDooDads && e->window == padWDial.win)
  669.       { DRedraw(&padWDial);  rv = 1; }
  670.     else if (popUp == ISPAD && padHaveDooDads && e->window == padHDial.win)
  671.       { DRedraw(&padHDial);  rv = 1; }
  672.     else if (popUp == ISPAD && padHaveDooDads && e->window == padODial.win)
  673.       { DRedraw(&padODial);  rv = 1; }
  674.   }
  675.  
  676.   else if (xev->type == ButtonPress) {
  677.     XButtonEvent *e = (XButtonEvent *) xev;
  678.  
  679.     if (e->button == Button1) {
  680.       if (e->window == popW) {
  681.     clickPUD(e->x,e->y);
  682.     rv = 1;
  683.       }
  684.       else if (popUp == ISPAD && padHaveDooDads && e->window == padWDial.win)
  685.     { DTrack(&padWDial, e->x, e->y);  rv = 1; }
  686.       else if (popUp == ISPAD && padHaveDooDads && e->window == padHDial.win)
  687.     { DTrack(&padHDial, e->x, e->y);  rv = 1; }
  688.       else if (popUp == ISPAD && padHaveDooDads && e->window == padODial.win)
  689.     { DTrack(&padODial, e->x, e->y);  rv = 1; }
  690.     }
  691.   }
  692.  
  693.  
  694.   else if (xev->type == KeyPress) {
  695.     XKeyEvent *e = (XKeyEvent *) xev;
  696.     char buf[128];  KeySym ks;
  697.     int stlen, i, shift, ck;
  698.     
  699.     stlen = XLookupString(e,buf,128,&ks,(XComposeStatus *) NULL);
  700.     shift = e->state & ShiftMask;
  701.     ck = CursorKey(ks, shift, 0);
  702.     buf[stlen] = '\0';
  703.  
  704.     RemapKeyCheck(ks, buf, &stlen);
  705.  
  706.     /* check cursor keys, which may or may not have a str assoc'd with them */
  707.     if (popUp==ISGETSTR || popUp==ISGRAB || popUp==ISPAD) {
  708.       if      (ck==CK_LEFT)  { doGetStrKey('\002'); rv = 1; }
  709.       else if (ck==CK_RIGHT) { doGetStrKey('\006'); rv = 1; }
  710.     }
  711.  
  712.     if (stlen && !rv) {      /* note: we accept kbd accel's in any win */
  713.       if (buf[0] == '\r') buf[0] = '\n';
  714.  
  715.       /* search for character in accel table */
  716.       for (i=0; i<nbts; i++) {
  717.     if (buf[0] == accel[i] && buf[0] != ' ') {
  718.       FakeButtonPress(&bts[i]);
  719.       rv = 1;
  720.     }
  721.       }
  722.  
  723.       if (!rv && buf[0]=='\033' && nbts==1) { /* ESC accepted in 1-but pu's */
  724.     FakeButtonPress(&bts[0]);
  725.     rv = 1;
  726.       }
  727.  
  728.       if (!rv && (popUp==ISGETSTR || popUp==ISGRAB || popUp==ISPAD)) { 
  729.     if (e->window == popW) { doGetStrKey(buf[0]);  rv = 1; }
  730.       }
  731.     }
  732.  
  733.     if (!stlen) rv = 1;  /* quietly eat mute keys */
  734.   }
  735.  
  736.  
  737.   else if (xev->type == ClientMessage) {
  738.     Atom proto, delwin;
  739.     XClientMessageEvent *client_event = (XClientMessageEvent *) xev;
  740.  
  741.     proto  = XInternAtom(theDisp, "WM_PROTOCOLS", FALSE);
  742.     delwin = XInternAtom(theDisp, "WM_DELETE_WINDOW", FALSE);
  743.  
  744.     if (client_event->message_type == proto &&
  745.     client_event->data.l[0]    == delwin) {
  746.       /* it's a WM_DELETE_WINDOW event */
  747.  
  748.       if (client_event->window == popW) {
  749.     FakeButtonPress(&bts[(nbts>1) ? nbts-1 : 0]);
  750.     rv = 1;
  751.       }
  752.     }
  753.   }
  754.  
  755.   if (rv==0 && (xev->type == KeyPress || xev->type == ButtonPress)) {
  756.     XBell(theDisp, 0);
  757.     rv = 1;            /* eat it */
  758.   }
  759.  
  760.   return rv;
  761. }
  762.  
  763.  
  764.  
  765. #define TR_MAXLN 10
  766.  
  767. /***************************************************/
  768. void TextRect(win, txt, x, y, w, h, fg)
  769.      Window  win;
  770.      char   *txt;
  771.      int     x,y,w,h;
  772.      u_long  fg;
  773. {
  774.   char *sp, *ep, *oldep, *start[TR_MAXLN];
  775.   int   i, inbreak, lineno, top, hardcr, maxln, len[TR_MAXLN];
  776.  
  777.   XSetForeground(theDisp, theGC, fg);
  778.   
  779.   sp = txt;  lineno = hardcr = 0;
  780.  
  781.   maxln = h / LINEHIGH;  
  782.   RANGE(maxln,0,TR_MAXLN);
  783.   while (*sp && lineno<maxln) {
  784.  
  785.     /* drop off any leading spaces (except on first line or after \n) */
  786.     if (sp!=txt && !hardcr) {
  787.       while(*sp==' ') sp++;
  788.     }
  789.  
  790.     hardcr = 0;   ep = sp;
  791.  
  792.     /* increment ep until we   A) get too wide, B) hit eos or
  793.        C) hit a '\n' character */
  794.  
  795.     /* NOTE: ep points to the character AFTER the end of the line */
  796.  
  797.     while (XTextWidth(mfinfo, sp, (int)(ep-sp))<= w && *ep && *ep!='\n') ep++;
  798.     if (*ep=='\n') { ep++;  hardcr=1; }   /* eat newline */
  799.  
  800.     /* if we got too wide, back off until we find a break position 
  801.        (last char before a space or a '/') */
  802.  
  803.     if (XTextWidth(mfinfo, sp, (int)(ep-sp)) > w) {
  804.       oldep = ep;  inbreak = 0;
  805.       while (ep!=sp) {
  806.     ep--;
  807.     if ( inbreak && *ep!=' ') { ep++;  break; }
  808.     if (!inbreak && *ep==' ') inbreak = 1;
  809.     if (*ep=='/') { ep++; break; }
  810.       }
  811.       if (ep==sp) ep = oldep-1;  /* can't break this line.  oh well */
  812.     }
  813.  
  814.     start[lineno] = sp;  len[lineno] = ep-sp;
  815.     
  816.     /* make sure we don't print a trailing '\n' character! */
  817.     if (len[lineno] > 0) {
  818.       while (sp[len[lineno]-1] == '\n') len[lineno] = len[lineno] - 1;
  819.     }
  820.  
  821.     sp = ep;
  822.     lineno++;
  823.   }
  824.  
  825.   top = y + h/2 + (ASCENT-DESCENT)/2 - ((lineno-1)*LINEHIGH)/2;
  826.   if (top<y+ASCENT) top = y+ASCENT;
  827.  
  828.   for (i=0, y=top; i<lineno; i++, y+=LINEHIGH) {
  829.     if (start[i][0] != '\n')
  830.       XDrawString(theDisp, win, theGC, x, y, start[i], len[i]);
  831.   }
  832. }
  833.  
  834.  
  835. /***************************************************/
  836. static void createPUD()
  837. {
  838.   popW = CreateWindow("xv confirm", "XVconfirm", "+0+0", 
  839.               PUWIDE, PUHIGH, infofg, infobg, 0);
  840.   if (!popW) FatalError("can't create popup window!");
  841.  
  842.   XSelectInput(theDisp, popW, ExposureMask | ButtonPressMask | KeyPressMask
  843.            | VisibilityChangeMask);
  844.   /* XSetTransientForHint(theDisp, popW, mainW); */
  845.  
  846.   XDefineCursor(theDisp, popW, arrow);
  847.   bts = (BUTT *) NULL;
  848.   nbts = selected = firsttime = 0;
  849. }
  850.   
  851.  
  852. /***************************************************/
  853. static void attachPUD()
  854. {
  855.   /* used to make PUD a transient window of something.  Doesn't
  856.      do anything anymore, as I got tired of having window layering
  857.      shifted around everytime a popup window happened.  Screw the
  858.      business about having the popup iconify when you iconify the
  859.      appropriate XV window.  There generally ISN'T an appropriate
  860.      XV window... */
  861. }
  862.  
  863.  
  864. /***************************************************/
  865. static void drawPUD(x,y,w,h)
  866. int x,y,w,h;
  867. {
  868.   int  i,xt,yt;
  869.   XRectangle xr;
  870.  
  871.   xr.x = x;  xr.y = y;  xr.width = w;  xr.height = h;
  872.   XSetClipRectangles(theDisp, theGC, 0,0, &xr, 1, Unsorted);
  873.  
  874.   XSetForeground(theDisp, theGC, infofg);
  875.   XSetBackground(theDisp, theGC, infobg);
  876.  
  877.   if (popUp == ISGRAB) {
  878.     xt = 10;  yt = 10;
  879.     TextRect(popW, text, xt, yt, puwide-10-xt, gsy-20, infofg);
  880.     drawGSBuf();
  881.  
  882.     XSetForeground(theDisp, theGC, infofg);
  883.     DrawString(popW, 10,        gsy+ASCENT+4, DELAYSTR);
  884.     DrawString(popW, gsx+gsw+5, gsy+ASCENT+4, SECSTR);
  885.  
  886.     CBRedraw(&ahideCB);
  887.   }
  888.  
  889.   else if (popUp == ISPAD) {
  890.     drawGSBuf();
  891.  
  892.     XSetForeground(theDisp, theGC, infofg);
  893.     DrawString(popW, 10+44,10+ASCENT+4,"Pad Method:");
  894.  
  895.     MBRedraw(&padMthdMB);
  896.     MBRedraw(&padDfltMB);
  897.     DRedraw (&padWDial);
  898.     DRedraw (&padHDial);
  899.     BTRedraw(&padDButt);
  900.     BTRedraw(&padOMButt);
  901.  
  902.     XSetForeground(theDisp, theGC, infofg);
  903.     drawPadOMStr();
  904.  
  905.     XDrawRectangle(theDisp, popW, theGC, 10, puhigh-140, 16+2*74+84, 130);
  906.     Draw3dRect(popW, 10+1, puhigh-140+1, 16+2*74+84-2, 130-2,
  907.            R3D_IN,2,hicol,locol,infobg);
  908.     XSetForeground(theDisp, theGC, infofg);
  909.     CenterString(popW, 16+1+75-13, puhigh-16-100-12, "New Image Size");
  910.  
  911.     if (ctrlColor) {
  912.       XSetForeground(theDisp, theGC, locol);
  913.       XDrawLine(theDisp, popW, theGC, 16+1+75+75+5, puhigh-140 + 6+8,
  914.         16+1+75+75+5, puhigh-10-4);
  915.     }
  916.       
  917.  
  918.     XSetForeground(theDisp, theGC, infofg);
  919.     XDrawRectangle(theDisp, popW, theGC, 268, puhigh-140, 
  920.            (u_int) puwide - 10 - 268, 130-BUTTH-10);
  921.     Draw3dRect(popW, 268+1, puhigh-140+1, (u_int) puwide -10-268-2, 
  922.            130-2 - BUTTH-10, R3D_IN,2,hicol,locol,infobg);
  923.     
  924.     TextRect(popW,padInst,268+5, puhigh-140+3, puwide-10-268-10, 
  925.          130-6 - BUTTH-10, infofg);
  926.   }
  927.  
  928.   else {
  929.     XCopyPlane(theDisp, iconPix, popW, theGC, 0,0, icon_width, icon_height,
  930.            10,10+(puhigh-30-BUTTH-icon_height)/2, 1L);
  931.     
  932.     xt = 10+icon_width+20;  yt = 10;
  933.     
  934.     if (popUp == ISGETSTR) {
  935.       TextRect(popW, text, xt, yt, puwide-10-xt, gsy-20, infofg);
  936.       drawGSBuf();
  937.     }
  938.     else TextRect(popW,text,xt,yt,puwide-10-xt,puhigh-10-BUTTH-20,infofg);
  939.   }
  940.  
  941.  
  942.   for (i=0; i<nbts; i++) BTRedraw(&bts[i]);
  943.   XSetClipMask(theDisp, theGC, None);
  944. }
  945.  
  946.  
  947. /***************************************************/
  948. static void drawPadOMStr()
  949. {
  950.   CenterString(popW, padODial.x + (padODial.w - 13)/2, 
  951.            puhigh-16-100-12, padOMStr[padOMode]);
  952. }
  953.   
  954. /***************************************************/
  955. static void clickPUD(x,y)
  956.      int x,y;
  957. {
  958.   int i;
  959.   BUTT *bp;
  960.   
  961.   for (i=0; i<nbts; i++) {
  962.     bp = &bts[i];
  963.     if (PTINRECT(x, y, bp->x, bp->y, bp->w, bp->h)) break;
  964.   }
  965.   
  966.   if (i<nbts && BTTrack(bp)) {
  967.     popUp = 0;  selected = i;  return;
  968.   }
  969.  
  970.   if (popUp==ISGRAB && CBClick(&ahideCB, x,y)) CBTrack(&ahideCB);
  971.  
  972.   else if (popUp == ISPAD) {
  973.     if (PTINRECT(x, y, padDButt.x, padDButt.y, padDButt.w, padDButt.h)) {
  974.       if (BTTrack(&padDButt)) {
  975.     DSetVal(&padWDial, pWIDE);
  976.     DSetVal(&padHDial, pHIGH);
  977.       }
  978.     }
  979.  
  980.     else if (PTINRECT(x,y,padOMButt.x,padOMButt.y,padOMButt.w,padOMButt.h)) {
  981.       if (BTTrack(&padOMButt)) {
  982.     XSetForeground(theDisp, theGC, infobg);
  983.     drawPadOMStr();
  984.     padOMode = (padOMode + 1) % PAD_OMAX;
  985.     XSetForeground(theDisp, theGC, infofg);
  986.     drawPadOMStr();
  987.       }
  988.     }
  989.  
  990.  
  991.     else if (MBClick(&padMthdMB, x,y)) {
  992.       i = MBTrack(&padMthdMB);
  993.       if (i<0 || i==padMode) return;
  994.  
  995.       switch (i) {
  996.       case PAD_SOLID:
  997.     strcpy(padBuf, padSbuf);
  998.     padDfltMB.list  = padColNames;
  999.     padDfltMB.nlist = padColLen;
  1000.     padInst = padSinst;
  1001.     break;
  1002.       case PAD_BGGEN:
  1003.     strcpy(padBuf, padBbuf);
  1004.     padDfltMB.list  = padBgNames;
  1005.     padDfltMB.nlist = padBgLen;
  1006.     padInst = padBinst;
  1007.     break;
  1008.       case PAD_LOAD:
  1009.     strcpy(padBuf,  padLbuf);
  1010.     padDfltMB.list  = padLoadNames;
  1011.     padDfltMB.nlist = padLoadLen;
  1012.     padInst = padLinst;
  1013.     break;
  1014.       default: break;             /* shouldn't happen */
  1015.       }
  1016.  
  1017.       gsCurPos = strlen(gsBuf);
  1018.       gsStPos = gsEnPos = 0;
  1019.       changedGSBuf();
  1020.       if (ctrlColor) 
  1021.     XClearArea(theDisp, popW, gsx+3,gsy+3, 
  1022.            (u_int)gsw-5, (u_int)gsh-5, False);
  1023.       else
  1024.     XClearArea(theDisp, popW, gsx+1,gsy+1, 
  1025.            (u_int)gsw-1, (u_int)gsh-1, False);
  1026.       drawGSBuf();
  1027.  
  1028.       BTSetActive(&bts[0], (int) strlen(gsBuf));
  1029.  
  1030.       MBSelect(&padMthdMB, i);
  1031.       MBSetActive(&padDfltMB, 1);
  1032.       DSetActive (&padWDial,  (i!=PAD_LOAD));
  1033.       DSetActive (&padHDial,  (i!=PAD_LOAD));
  1034.  
  1035.       XClearArea(theDisp, popW, 184+5, puhigh-140+3, 
  1036.          (u_int) puwide-10-184-10, 130-6 - BUTTH-10, True);
  1037.  
  1038.       padMode = i;
  1039.     }
  1040.  
  1041.     else if (MBClick(&padDfltMB, x,y)) {
  1042.       i = MBTrack(&padDfltMB);
  1043.       if (i<0) return;
  1044.  
  1045.       if      (padMode == PAD_SOLID) strcpy(padBuf, padColVals[i]);
  1046.       else if (padMode == PAD_BGGEN) strcpy(padBuf, padBgVals[i]);
  1047.       else if (padMode == PAD_LOAD)  strcpy(padBuf, padLoadVals[i]);
  1048.  
  1049.       gsCurPos = strlen(gsBuf);
  1050.       gsStPos = gsEnPos = 0;
  1051.       changedGSBuf();
  1052.       if (ctrlColor) 
  1053.     XClearArea(theDisp, popW, gsx+3,gsy+3, 
  1054.            (u_int)gsw-5, (u_int)gsh-5, False);
  1055.       else
  1056.     XClearArea(theDisp, popW, gsx+1,gsy+1, 
  1057.            (u_int)gsw-1, (u_int)gsh-1, False);
  1058.       drawGSBuf();
  1059.  
  1060.       BTSetActive(&bts[0], (int) strlen(gsBuf));
  1061.     }
  1062.   }
  1063. }
  1064.  
  1065.  
  1066.  
  1067. /***************************************************/
  1068. static void doGetStrKey(c)
  1069.      int c;
  1070. {
  1071.   if (doGSKey(c)) XBell(theDisp, 0);
  1072. }
  1073.  
  1074.  
  1075. /***************************************************/
  1076. static int doGSKey(c)
  1077.      int c;
  1078. {
  1079.   /* handle characters typed at GetStrPopUp window.  Button accel. keys
  1080.      have already been checked for elsewhere.  Practical upshot is that
  1081.      we don't have to do anything with ESC or Return (as these will normally
  1082.      be Cancel and Ok buttons) 
  1083.  
  1084.      Normally returns '0'.  Returns '1' if character wasn't accepted, for
  1085.      whatever reason. */
  1086.  
  1087.   int i, len, flen;
  1088.  
  1089.   len = strlen(gsBuf);
  1090.   if (gsFilter) flen = strlen(gsFilter);
  1091.            else flen = 0;
  1092.   
  1093.  
  1094.   if (c>=' ' && c<'\177') {              /* 'NORMAL' CHARACTERS */
  1095.     if (flen) {                          /* check filter string */
  1096.       for (i=0; i<flen && c!=gsFilter[i]; i++);
  1097.       if (!gsAllow && i< flen) return 1;    /* found in 'disallow' filter */
  1098.       if ( gsAllow && i==flen) return 1;    /* not found in 'allow' filter */
  1099.     }
  1100.     
  1101.     if (len >= gsBufLen-1) return 1;     /* at max length */
  1102.  
  1103.     xvbcopy(&gsBuf[gsCurPos], &gsBuf[gsCurPos+1], (size_t) len-gsCurPos+1);
  1104.     gsBuf[gsCurPos]=c;  gsCurPos++;
  1105.   }
  1106.  
  1107.  
  1108.   else if (c=='\010' || c=='\177') {    /* BS or DEL */
  1109.     if (gsCurPos==0) return 1;                     /* at beginning of str */
  1110.     xvbcopy(&gsBuf[gsCurPos], &gsBuf[gsCurPos-1], (size_t) len-gsCurPos+1);
  1111.     gsCurPos--;
  1112.   }
  1113.  
  1114.   else if (c=='\025') {                 /* ^U: clear entire line */
  1115.     gsBuf[0] = '\0';
  1116.     gsCurPos = 0;
  1117.   }
  1118.  
  1119.   else if (c=='\013') {                 /* ^K: clear to end of line */
  1120.     gsBuf[gsCurPos] = '\0';
  1121.   }
  1122.  
  1123.   else if (c=='\001') {                 /* ^A: move to beginning */
  1124.     gsCurPos = 0;
  1125.   }
  1126.  
  1127.   else if (c=='\005') {                 /* ^E: move to end */
  1128.     gsCurPos = len;
  1129.   }
  1130.  
  1131.   else if (c=='\004') {                 /* ^D: delete character at gsCurPos */
  1132.     if (gsCurPos==len) return 1;
  1133.     xvbcopy(&gsBuf[gsCurPos+1], &gsBuf[gsCurPos], (size_t) len-gsCurPos);
  1134.   }
  1135.  
  1136.   else if (c=='\002') {                 /* ^B: move backwards char */
  1137.     if (gsCurPos==0) return 1;
  1138.     gsCurPos--;
  1139.   }
  1140.  
  1141.   else if (c=='\006') {                 /* ^F: move forwards char */
  1142.     if (gsCurPos==len) return 1;
  1143.     gsCurPos++;
  1144.   }
  1145.  
  1146.   else return 1;                        /* unhandled character */
  1147.  
  1148.   changedGSBuf();      /* compute gsEnPos, gsStPos */
  1149.  
  1150.   if (ctrlColor) 
  1151.     XClearArea(theDisp, popW, gsx+3,gsy+3, (u_int)gsw-5, (u_int)gsh-5, False);
  1152.   else
  1153.     XClearArea(theDisp, popW, gsx+1,gsy+1, (u_int)gsw-1, (u_int)gsh-1, False);
  1154.  
  1155.   drawGSBuf();
  1156.  
  1157.   if (popUp == ISGETSTR || popUp == ISPAD) {
  1158.     /* if we have a string of any sort, turn on the default '\n' button
  1159.        (if there is one) */
  1160.     for (i=0; i<nbts && accel[i]!='\n'; i++);
  1161.     if (i<nbts) BTSetActive(&bts[i], (strlen(gsBuf) > (size_t) 0));
  1162.   }
  1163.   else if (popUp == ISGRAB) {
  1164.     /* need a string of length 1 to enable Grab (bts[0]), and a string
  1165.        with an atoi() of at least '1' to enable AutoGrab (bts[1]) */
  1166.     BTSetActive(&bts[0], (strlen(gsBuf) > (size_t) 0));
  1167.     BTSetActive(&bts[1], (strlen(gsBuf)>(size_t)0 && atoi(gsBuf)>(size_t)0));
  1168.   }
  1169.  
  1170.   return(0);
  1171. }
  1172.  
  1173.  
  1174.  
  1175. /***************************************************/
  1176. static void changedGSBuf()
  1177. {
  1178.   /* cursor position (or whatever) may have changed.  adjust displayed 
  1179.      portion of gsBuf */
  1180.  
  1181.   int len;
  1182.  
  1183.   len = strlen(gsBuf);
  1184.  
  1185.   if (gsCurPos < gsStPos) gsStPos = gsCurPos;
  1186.   if (gsCurPos > gsEnPos) gsEnPos = gsCurPos;
  1187.  
  1188.   if (gsStPos>len) gsStPos = (len>0) ? len-1 : 0;
  1189.   if (gsEnPos>len) gsEnPos = (len>0) ? len-1 : 0;
  1190.  
  1191.   /* while substring is shorter than window, inc enPos */
  1192.  
  1193.   while (XTextWidth(mfinfo, &gsBuf[gsStPos], gsEnPos-gsStPos) < (gsw-6)
  1194.      && gsEnPos<len) { gsEnPos++; }
  1195.  
  1196.   /* while substring is longer than window, dec enpos, unless enpos==curpos,
  1197.      in which case, inc stpos */
  1198.  
  1199.   while (XTextWidth(mfinfo, &gsBuf[gsStPos], gsEnPos-gsStPos) > (gsw-6)) {
  1200.     if (gsEnPos != gsCurPos) gsEnPos--;
  1201.     else gsStPos++;
  1202.   }
  1203. }
  1204.  
  1205.  
  1206. /***************************************************/
  1207. static void drawGSBuf()
  1208. {
  1209.   /* draw edittext thingy in GetStrPopUp window */
  1210.  
  1211.   int cpos;
  1212.  
  1213.   XSetForeground(theDisp, theGC, infofg);
  1214.   XDrawRectangle(theDisp, popW, theGC, gsx, gsy, (u_int) gsw, (u_int) gsh);
  1215.   Draw3dRect(popW, gsx+1, gsy+1, (u_int) gsw-2, (u_int) gsh-2, 
  1216.          R3D_IN, 2, hicol,locol,infobg);
  1217.  
  1218.   XSetForeground(theDisp, theGC, infofg);
  1219.  
  1220.   if (gsStPos>0) {  /* draw a "there's more over here" doowah */
  1221.     XDrawLine(theDisp, popW, theGC, gsx+1, gsy+1, gsx+1, gsy + gsh-1);
  1222.     XDrawLine(theDisp, popW, theGC, gsx+2, gsy+1, gsx+2, gsy + gsh-1);
  1223.     XDrawLine(theDisp, popW, theGC, gsx+3, gsy+1, gsx+3, gsy + gsh-1);
  1224.   }
  1225.  
  1226.   if ((size_t) gsEnPos < strlen(gsBuf)) {
  1227.     /* draw a "there's more over here" doowah */
  1228.     XDrawLine(theDisp, popW, theGC, gsx+gsw-3, gsy+1, gsx+gsw-3, gsy+gsh-1);
  1229.     XDrawLine(theDisp, popW, theGC, gsx+gsw-2, gsy+1, gsx+gsw-2, gsy+gsh-1);
  1230.     XDrawLine(theDisp, popW, theGC, gsx+gsw-1, gsy+1, gsx+gsw-1, gsy+gsh-1);
  1231.   }
  1232.  
  1233.   XDrawString(theDisp, popW, theGC, gsx+4, gsy+ASCENT+4,
  1234.           gsBuf+gsStPos, gsEnPos-gsStPos);
  1235.  
  1236.   cpos = gsx+XTextWidth(mfinfo, &gsBuf[gsStPos], gsCurPos-gsStPos);
  1237.   XDrawLine(theDisp,popW,theGC, 4+cpos, gsy+3,         4+cpos, gsy+2+CHIGH+1);
  1238.   XDrawLine(theDisp,popW,theGC, 4+cpos, gsy+2+CHIGH+1, 6+cpos, gsy+2+CHIGH+3);
  1239.   XDrawLine(theDisp,popW,theGC, 4+cpos, gsy+2+CHIGH+1, 2+cpos, gsy+2+CHIGH+3);
  1240. }
  1241.