Annotation of 43BSDTahoe/new/X/bitmap/bitmap.c, revision 1.1

1.1     ! root        1: #include <X/mit-copyright.h>
        !             2: 
        !             3: /* Copyright 1985, Massachusetts Institute of Technology */
        !             4: 
        !             5: #ifndef lint
        !             6: static char *rcsid_bitmap_c = "$Header: bitmap.c,v 10.9 86/11/19 19:13:23 jg Rel $";
        !             7: #endif
        !             8: 
        !             9: #include <errno.h>
        !            10: #include <stdio.h>
        !            11: #include <X/Xlib.h>
        !            12: #include <sys/types.h>
        !            13: 
        !            14: #include "../cursors/cross.cursor"
        !            15: #include "../cursors/cross_mask.cursor"
        !            16: #include "../cursors/ul_angle.cursor"
        !            17: #include "../cursors/ul_angle_mask.cursor"
        !            18: #include "../cursors/lr_angle.cursor"
        !            19: #include "../cursors/lr_angle_mask.cursor"
        !            20: #include "../cursors/dot.cursor"
        !            21: #include "../cursors/dot_mask.cursor"
        !            22: 
        !            23: #define TOP_MARGIN 10
        !            24: #define LEFT_MARGIN 10
        !            25: #define BOTTOM_MARGIN 10
        !            26: #define AROUND_RASTER_MARGIN 20
        !            27: #define GRID_TO_COMMAND_MARGIN 5
        !            28: #define RIGHT_MARGIN 5
        !            29: 
        !            30: #define MIN_SQUARE_SIZE 3
        !            31: #define DEFAULT_SQUARE_SIZE 13
        !            32: 
        !            33: #define bit int
        !            34: #define boolean int
        !            35: #define TRUE 1
        !            36: #define FALSE 0
        !            37: #define OUT_OF_RANGE 10000
        !            38: 
        !            39: #define COPY           0
        !            40: #define MOVE           1
        !            41: #define OVERLAY                2
        !            42: 
        !            43: #define min(x,y) ((x < y) ? x : y)
        !            44: #define max(x,y) ((x < y) ? y : x)
        !            45: 
        !            46: /* error handling stuff */
        !            47: extern int errno;
        !            48: extern char *sys_errlist[];
        !            49: 
        !            50: /* global "constants" -- set once at startup time */
        !            51: /* the first few variables are not static because they are shared
        !            52:    with dialog.c */
        !            53: int foreground = BlackPixel;
        !            54: int background = WhitePixel;
        !            55: Pixmap backmap;
        !            56: Pixmap border;
        !            57: int borderwidth = 3;
        !            58: int invertplane = 1;
        !            59: int highlightplane = 1;
        !            60: int mousepix = BlackPixel;
        !            61: static int squares_wide = OUT_OF_RANGE;
        !            62: static int squares_high = OUT_OF_RANGE;
        !            63: static short *raster;
        !            64: static int raster_length; /* how many shorts in the raster[] array */
        !            65: static Window outer_window, grid_window;
        !            66: static Window raster_window, raster_invert_window;
        !            67: static Font font;
        !            68: static FontInfo fontInfo;
        !            69: static Cursor cross, upper_left, lower_right, dot;
        !            70: static char *filename = NULL; /* name of input file */
        !            71: static char *backup_filename;
        !            72: static char *stripped_name; 
        !            73:   /* file name without directory path or extension */
        !            74: static char *progname; /* name this program was invoked by */
        !            75: static Pattern DottedPattern = XMakePattern (
        !            76:     1 /* pattern */,
        !            77:     2 /* length */,
        !            78:     1); /* multiplier */
        !            79: static Pattern InverseDottedPattern = XMakePattern (
        !            80:     2 /* pattern */,
        !            81:     2 /* length */,
        !            82:     1); /* multiplier */
        !            83: 
        !            84: /* command-button data */
        !            85: #define N_COMMANDS     16
        !            86: static struct command_data {
        !            87:   Window window;
        !            88:   char *name;
        !            89:   int name_length;
        !            90:   int x_offset;  /* so text is centered within command box */
        !            91:   boolean inverted;
        !            92:   int (*proc)(); 
        !            93:      /* function to invoke when command button is "pressed" */
        !            94:      /* actually no return value, but compiler doesn't like "void" here */
        !            95:   int data;  /* arbitrary instance data to call procedure back with */
        !            96:   } commands [N_COMMANDS];
        !            97: 
        !            98: /* global variables */
        !            99: /* layout-related variables */
        !           100: static int square_size;  /* length of square's side, in pixels */
        !           101: static OpaqueFrame frames[N_COMMANDS+3], outer_frame;
        !           102:      /* frames[0] throgh frames[N_COMMANDS-1] are the command windows;
        !           103:         frames[N_COMMANDS] is the raster;
        !           104:        frames[N_COMMANDS+1] is the inverted raster;
        !           105:        frames[N_COMMANDS+2] is the grid */
        !           106: 
        !           107:   /* location of x'd-through squares, if any */
        !           108: static int x1_square_exed_through = OUT_OF_RANGE;
        !           109: static int y1_square_exed_through = OUT_OF_RANGE;
        !           110: static int x2_square_exed_through = OUT_OF_RANGE;
        !           111: static int y2_square_exed_through = OUT_OF_RANGE;
        !           112: 
        !           113:   /* location of "plus'd through" squares, if any */
        !           114: static int x1_square_plus_through = OUT_OF_RANGE;
        !           115: static int y1_square_plus_through = OUT_OF_RANGE;
        !           116: static int x2_square_plus_through = OUT_OF_RANGE;
        !           117: static int y2_square_plus_through = OUT_OF_RANGE;
        !           118: 
        !           119:   /* location of hot spot, if any */
        !           120: static int x_hot_spot = OUT_OF_RANGE;
        !           121: static int y_hot_spot = OUT_OF_RANGE;
        !           122: 
        !           123: static boolean changed = FALSE;
        !           124:    /* has user changed bitmap since starting program or last write? */
        !           125: 
        !           126: static enum RepaintGridType {e_AgainstBackground, e_AgainstForeground, e_Invert};
        !           127: 
        !           128: extern char *malloc();
        !           129: 
        !           130: main (argc, argv)
        !           131:   int argc;
        !           132:   char **argv;
        !           133:   {
        !           134:   SetUp (argc, argv);
        !           135:   while (TRUE) {
        !           136:     XEvent event;
        !           137:     XNextEvent(&event);
        !           138:     ProcessEvent(&event);
        !           139:     }
        !           140:   }   /* end of main procedure */
        !           141: 
        !           142: 
        !           143: SetUp (argc, argv)
        !           144:   int argc;
        !           145:   char **argv;
        !           146:   {
        !           147:   char *StripName(), *BackupName(), *index();
        !           148:   char *option;
        !           149:   FILE *file;
        !           150:   char *geometry = NULL, *host = NULL, *dimensions = NULL;
        !           151:   int i;
        !           152: 
        !           153:   progname = argv[0];
        !           154:   setlinebuf (stderr);
        !           155: 
        !           156:   /* Parse command line */
        !           157:   for (i = 1; i < argc; i++) {
        !           158:     if (argv[i][0] == '=')
        !           159:        geometry = argv[i];
        !           160: 
        !           161:     else if (index (argv[i], ':') != NULL)
        !           162:        host = argv[i];
        !           163:        
        !           164:     else if (filename == NULL)
        !           165:        filename = argv[i];
        !           166: 
        !           167:     else 
        !           168:        dimensions = argv[i];
        !           169:     }
        !           170: 
        !           171:   if (filename == NULL) {
        !           172:     fprintf (stderr, "%s: no file name specified\n", progname);
        !           173:     exit (1);
        !           174:     }
        !           175:   
        !           176:   stripped_name = StripName (filename);
        !           177:   backup_filename = BackupName (filename);
        !           178: 
        !           179:   file = fopen (filename, "r");
        !           180:   if (!file && (errno != ENOENT)) {
        !           181:     fprintf (stderr, "%s: could not open file '%s' for reading -- %s\n",
        !           182:       progname, filename, sys_errlist[errno]);
        !           183:     exit (1);
        !           184:     }
        !           185:       
        !           186:   if (file)
        !           187:     DimensionsFromFile(file);
        !           188:   else {
        !           189:     if (dimensions != NULL) {
        !           190:       if (sscanf (dimensions, "%dx%d", &squares_wide, &squares_high) != 2) {
        !           191:        fprintf (stderr, "%s: invalid dimensions '%s'\n",
        !           192:          progname, dimensions);
        !           193:        exit (1);
        !           194:        }
        !           195:       if ((squares_wide <=0) || (squares_high <=0)) {
        !           196:         fprintf (stderr, "%s: dimensions must be positive\n", progname);
        !           197:         exit (1);
        !           198:         }
        !           199:       }
        !           200:     else   /* dimensions not supplied on command line */
        !           201:       squares_wide = squares_high = 16;
        !           202:     }
        !           203: 
        !           204:   raster_length = ((squares_wide+15)/16)*squares_high;
        !           205:   raster = (short *) malloc (raster_length*sizeof(short));
        !           206:   
        !           207:   if (file) {
        !           208:     InitialValuesFromFile(file);
        !           209:     fclose (file);
        !           210:     }
        !           211:   else {
        !           212:     /* set raster to all 0's (background color) */
        !           213:     register int i;
        !           214:     for (i=0;i<raster_length;i++)
        !           215:       raster[i] = 0;
        !           216:     }
        !           217: 
        !           218:   if (!XOpenDisplay(host)) {
        !           219:        fprintf(stderr, "%s: Can't open display '%s'\n",
        !           220:                argv[0], XDisplayName(host));
        !           221:        exit (1);
        !           222:     }
        !           223: 
        !           224:   backmap = WhitePixmap;
        !           225:   border = BlackPixmap;
        !           226:   if ((option = XGetDefault(progname, "BorderWidth")) != NULL)
        !           227:     borderwidth = atoi(option);
        !           228:   if ((option = XGetDefault(progname, "BodyFont")) == NULL)
        !           229:     option = "vtsingle";
        !           230:   font = XGetFont (option);
        !           231:   XQueryFont (font, &fontInfo);
        !           232:   if (DisplayCells() > 2) {
        !           233:     char *fore_color = XGetDefault(progname, "Foreground");
        !           234:     char *back_color = XGetDefault(progname, "Background");
        !           235:     char *high_color = XGetDefault(progname, "Highlight");
        !           236:     char *brdr_color = XGetDefault(progname, "Border");
        !           237:     char *mous_color = XGetDefault(progname, "Mouse");
        !           238:     Color fdef, bdef, hdef;
        !           239:     if (fore_color && XParseColor(fore_color, &fdef) &&
        !           240:       back_color && XParseColor(back_color, &bdef) &&
        !           241:       (high_color == NULL || XParseColor(high_color, &hdef)) &&
        !           242:       XGetColorCells(0, 1, high_color ? 2 : 1, &invertplane, &bdef.pixel)) {
        !           243:       background = bdef.pixel;
        !           244:       backmap = XMakeTile(background);
        !           245:       if (high_color) {
        !           246:        hdef.pixel = bdef.pixel | invertplane;
        !           247:        XStoreColor(&hdef);
        !           248:        highlightplane = 1 << (ffs(invertplane) - 1);
        !           249:        hdef.pixel = bdef.pixel | highlightplane;
        !           250:        XStoreColor(&hdef);
        !           251:        invertplane ^= highlightplane;
        !           252:       } else
        !           253:        highlightplane = invertplane;
        !           254:       XStoreColor(&bdef);
        !           255:       foreground = background | invertplane;
        !           256:       fdef.pixel = foreground;
        !           257:       XStoreColor(&fdef);
        !           258:     }
        !           259:     if (brdr_color && XParseColor(brdr_color, &bdef) &&
        !           260:        XGetHardwareColor(&bdef))
        !           261:       border = XMakeTile(bdef.pixel);
        !           262:     if (mous_color && XParseColor(mous_color, &fdef) &&
        !           263:        XGetHardwareColor(&fdef))
        !           264:       mousepix = fdef.pixel;
        !           265:   }
        !           266: 
        !           267:   {
        !           268:   int right_side_bottom, right_side_width;
        !           269:   int minwidth, minheight;
        !           270:   int default_width, default_height, default_x, default_y;
        !           271:   int display_width = DisplayWidth();
        !           272:   int display_height = DisplayHeight();
        !           273:   char default_geometry[20];
        !           274: 
        !           275:   LayoutStage1();
        !           276:   right_side_bottom = frames[N_COMMANDS+1].y + frames[N_COMMANDS+1].height
        !           277:     + 2 /* borders */ + AROUND_RASTER_MARGIN;
        !           278:   right_side_width = frames[0].width + 2 /* borders */
        !           279:     + GRID_TO_COMMAND_MARGIN + RIGHT_MARGIN;
        !           280:   OuterWindowDims (MIN_SQUARE_SIZE, right_side_width, right_side_bottom,
        !           281:     &minwidth, &minheight);
        !           282:   OuterWindowDims (DEFAULT_SQUARE_SIZE, right_side_width, right_side_bottom,
        !           283:     &default_width, &default_height);
        !           284:   default_x = min (200, display_width - default_width - 2*borderwidth);
        !           285:   default_y = min (200, display_height - default_height - 2*borderwidth);
        !           286:   sprintf (default_geometry, "=%dx%d+%d+%d", default_width,
        !           287:     default_height, default_x, default_y);
        !           288:   outer_frame.bdrwidth = borderwidth;
        !           289:   outer_frame.border = border;
        !           290:   outer_frame.background = backmap;
        !           291:   outer_window = XCreate ("Bitmap Editor", progname, geometry,
        !           292:      default_geometry, &outer_frame, minwidth, minheight);
        !           293:   LayoutStage2 ();
        !           294:   }
        !           295:   
        !           296:   upper_left = XCreateCursor (ul_angle_width, ul_angle_height, 
        !           297:     ul_angle_bits, ul_angle_mask_bits, ul_angle_x_hot, ul_angle_y_hot,
        !           298:     mousepix, background, GXcopy);
        !           299:   lower_right = XCreateCursor (lr_angle_width, lr_angle_height,
        !           300:     lr_angle_bits, lr_angle_mask_bits,
        !           301:     lr_angle_x_hot, lr_angle_y_hot, mousepix, background, GXcopy);
        !           302:   cross = XCreateCursor (cross_width, cross_height, cross_bits, 
        !           303:     cross_mask_bits, cross_x_hot, cross_y_hot, mousepix, background,
        !           304:     GXcopy);
        !           305:   dot = XCreateCursor (dot_width, dot_height, dot_bits,
        !           306:     dot_mask_bits, dot_x_hot, dot_y_hot, mousepix, background,
        !           307:     GXcopy);
        !           308:   XDefineCursor (outer_window, cross);
        !           309: 
        !           310:   XExpandEvents();  /* do NOT collapse adjacent MouseMotion events */
        !           311:   XSelectInput (outer_window, ExposeWindow);  /* to detect size changes */
        !           312:   XMapWindow (outer_window);
        !           313:   XMapSubwindows (outer_window);
        !           314:   }    /* end of Set_Up procedure */
        !           315: 
        !           316: 
        !           317: /* Unfortunately, the current implementation of X (version 10) does not
        !           318:    handle resize event notification very well.  When the outer window
        !           319:    is resized, X sends ExposeWindow events for each subwindow BEFORE sending
        !           320:    an ExposeWindow for the outer window.
        !           321: 
        !           322:    If I handled the events in the order sent, I would repaint each
        !           323:    subwindow, only then to discover that the outer window had changed
        !           324:    size...at which time I would unmap and rearrange the subwindows,
        !           325:    causing ANOTHER set of exposure events on the subwindows.  This would
        !           326:    not just be inefficient, it would also look ugly on the screen.
        !           327: 
        !           328:    To work around this misfeature, I do not process ExposeWindow events
        !           329:    immediately upon receipt, but instead go into a recursion (HandleExposure)
        !           330:    until I either run out of events or get a non-ExposeWindow event.  If
        !           331:    the list of ExposeWindow events ends with a resize, I discard all of the
        !           332:    earlier ExposeWindow events.  Otherwise, I unwind out of the recursion,
        !           333:    processing each exposure event in the opposite of the order received,
        !           334:    eventually returning to the normal event notification loop.
        !           335: 
        !           336:    This code is admittedly convoluted, and will hopefully go away in a
        !           337:    future version of X.  */
        !           338: 
        !           339: #define PROCESS_IT 0
        !           340: #define DISCARD_IT 1
        !           341: 
        !           342: ProcessEvent (event)
        !           343:   register XEvent *event;
        !           344:   {
        !           345:   register Window w = event->window;
        !           346:   register int i;
        !           347:   if (event->type == ExposeWindow) {
        !           348:     int status;
        !           349:     /* make sure that I get all the exposure events that have been sent */
        !           350:     XSync(0)        !           351:     status = HandleExposure (event);
        !           352:     if (status == DISCARD_IT)
        !           353:        return;
        !           354:     }
        !           355:   ProcessEventReally (event);
        !           356:   }
        !           357: 
        !           358: ProcessEventReally (event)
        !           359:   register XEvent *event;
        !           360:   {
        !           361:   register Window w = event->window;
        !           362:   register int i;
        !           363:   if (w == grid_window)
        !           364:     ProcessGridWindowEvent (event);
        !           365:   else if (w == outer_window)
        !           366:     ProcessOuterWindowEvent (event);
        !           367:   else if (w == raster_window)
        !           368:     RepaintRaster();
        !           369:   else if (w == raster_invert_window)
        !           370:     RepaintRasterInverted();
        !           371:   else for (i=0;i<N_COMMANDS;i++)
        !           372:     if (w == commands[i].window)
        !           373:       ProcessCommandButtonEvent (&commands[i], event);
        !           374:   }  /* end of ProcessInput procedure */
        !           375: 
        !           376: 
        !           377: int HandleExposure (event)
        !           378:   XExposeWindowEvent *event;
        !           379:   {
        !           380:   int status;
        !           381:   XEvent next_event;
        !           382:   
        !           383:   if (QLength() == 0)
        !           384:     /* there are no more events, so this can't be a resize */
        !           385:     return (PROCESS_IT);
        !           386:     
        !           387:   XNextEvent (&next_event);
        !           388:   if (next_event.type != ExposeWindow) {
        !           389:     /* the list of exposures ended, so this can't be a resize */
        !           390:     XPutBackEvent (&next_event);
        !           391:     return (PROCESS_IT);
        !           392:     }
        !           393:     
        !           394:   if (next_event.window == outer_window) {
        !           395:     if ((outer_frame.height == ((XExposeWindowEvent *)&next_event)->height)
        !           396:       && (outer_frame.width == ((XExposeWindowEvent *)&next_event)->width))
        !           397:         /* the list of exposures ended with a non-resize */
        !           398:         status = PROCESS_IT;
        !           399:     else
        !           400:        /* this IS a resize */
        !           401:         status = DISCARD_IT;
        !           402:     /* Handle the outer window exposure, whether or not it's a resize */
        !           403:     ProcessEventReally (&next_event);
        !           404:     return (status);
        !           405:     }
        !           406: 
        !           407:   if ((status = HandleExposure (&next_event)) == PROCESS_IT)
        !           408:     ProcessEventReally (&next_event);
        !           409:   return (status);
        !           410:   }
        !           411:   
        !           412: ProcessGridWindowEvent (event)
        !           413:   XEvent *event;
        !           414:   {
        !           415:   int x_square, y_square;
        !           416:   static int x_square_prev, y_square_prev;
        !           417:   static boolean raster_outdated;
        !           418:   switch (event->type) {
        !           419: 
        !           420:     case ExposeWindow:
        !           421:       RepaintGridLines(e_AgainstBackground);
        !           422:       RefillGrid(FALSE);
        !           423:       if (x1_square_exed_through != OUT_OF_RANGE)
        !           424:        ExThroughRectangle (x1_square_exed_through, y1_square_exed_through,
        !           425:          x2_square_exed_through, y2_square_exed_through);
        !           426:       if (x1_square_plus_through != OUT_OF_RANGE)
        !           427:         PlusThroughRectangle (x1_square_plus_through, y1_square_plus_through,
        !           428:          x2_square_plus_through, y2_square_plus_through);
        !           429:       if (x_hot_spot != OUT_OF_RANGE)
        !           430:        HighlightHotSpot();
        !           431:       break;
        !           432: 
        !           433:     case ExposeRegion: {
        !           434: #define this_event ((XExposeRegionEvent *)event)
        !           435:       int x1 = this_event->x;
        !           436:       int y1 = this_event->y;
        !           437:       int x2 = x1 + this_event->width;
        !           438:       int y2 = y1 + this_event->height;
        !           439: #undef this_event
        !           440:       x1 /= square_size;
        !           441:       x2 /= square_size;
        !           442:       y1 /= square_size;
        !           443:       y2 /= square_size;
        !           444:       if (x2 >= squares_wide)
        !           445:         x2 = squares_wide - 1;  /* sanity check */
        !           446:       if (y2 >= squares_high)
        !           447:         y2 = squares_high - 1;  /* sanity check */
        !           448:       RepaintGridLinesPartially(x1,y1,x2+1,y2+1,e_AgainstBackground,TRUE);
        !           449:       RefillGridPartially (x1,y1,x2,y2,FALSE);
        !           450:       if (x1_square_exed_through != OUT_OF_RANGE)
        !           451:         ExThroughRectangle (
        !           452:          max (x1, x1_square_exed_through),
        !           453:          max (y1, y1_square_exed_through),
        !           454:          min (x2, x2_square_exed_through),
        !           455:          min (y2, y2_square_exed_through));
        !           456:       if (x1_square_plus_through != OUT_OF_RANGE)
        !           457:         PlusThroughRectangle (
        !           458:          max (x1, x1_square_plus_through),
        !           459:          max (y1, y1_square_plus_through),
        !           460:          min (x2, x2_square_plus_through),
        !           461:          min (y2, y2_square_plus_through));
        !           462:       if (x_hot_spot >= x1 && x_hot_spot <= x2
        !           463:        && y_hot_spot >= y1 && y_hot_spot <= y2)
        !           464:            HighlightHotSpot();
        !           465:       break;
        !           466:       }
        !           467: 
        !           468:     case ButtonPressed:
        !           469:       if (WhatSquare (event, &x_square, &y_square))
        !           470:         return;  /* mouse outside grid; really shouldn't happen, but... */
        !           471:       switch (((XButtonPressedEvent *)event)->detail & ValueMask) {
        !           472:        case LeftButton:
        !           473:          PaintSquare (x_square, y_square, foreground);
        !           474:          if (x_square == x_hot_spot && y_square == y_hot_spot)
        !           475:            HighlightHotSpot();
        !           476:          SetRasterBit (raster, x_square, y_square, 1);
        !           477:          break;
        !           478:        case MiddleButton:
        !           479:          InvertSquare (x_square, y_square);
        !           480:          InvertRasterBit (raster, x_square, y_square);
        !           481:          break;
        !           482:        case RightButton:
        !           483:          PaintSquare (x_square, y_square, background);
        !           484:          if (x_square == x_hot_spot && y_square == y_hot_spot)
        !           485:            HighlightHotSpot();
        !           486:          SetRasterBit (raster, x_square, y_square, 0);
        !           487:          break;
        !           488:        }
        !           489:       RepaintRaster();
        !           490:       RepaintRasterInverted();
        !           491:       x_square_prev = x_square;
        !           492:       y_square_prev = y_square;
        !           493:       raster_outdated = FALSE;
        !           494:       changed = TRUE;
        !           495:       break;
        !           496:     
        !           497:     case MouseMoved:
        !           498:       if (WhatSquare (event, &x_square, &y_square))
        !           499:         return;  /* mouse outside grid; really shouldn't happen, but... */
        !           500:       if ((x_square != x_square_prev) || (y_square != y_square_prev))
        !           501:                  switch (((XMouseMovedEvent *)event)->detail) {
        !           502:            case LeftMask:
        !           503:              PaintSquare (x_square, y_square, foreground);
        !           504:              if (x_square == x_hot_spot && y_square == y_hot_spot)
        !           505:                HighlightHotSpot();
        !           506:              SetRasterBit (raster, x_square, y_square, 1);
        !           507:              changed = raster_outdated = TRUE;
        !           508:              break;
        !           509:            case MiddleMask:
        !           510:              InvertSquare (x_square, y_square);
        !           511:              InvertRasterBit (raster, x_square, y_square);
        !           512:              changed = raster_outdated = TRUE;
        !           513:              break;
        !           514:            case RightMask:
        !           515:              PaintSquare (x_square, y_square, background);
        !           516:              if (x_square == x_hot_spot && y_square == y_hot_spot)
        !           517:                HighlightHotSpot();
        !           518:              SetRasterBit (raster, x_square, y_square, 0);
        !           519:              changed = raster_outdated = TRUE;
        !           520:              break;
        !           521:            default:
        !           522:              break;  /* ignore events with multiple buttons down */
        !           523:            }
        !           524:       if (raster_outdated && !MouseMovedEventQueued()) {
        !           525:        RepaintRaster();
        !           526:        RepaintRasterInverted();
        !           527:        raster_outdated = FALSE;
        !           528:        }
        !           529:       x_square_prev = x_square;
        !           530:       y_square_prev = y_square;
        !           531:       break;
        !           532:   
        !           533:     }
        !           534:   }  /* end of ProcessGridWindowEvent procedure */
        !           535: 
        !           536: boolean MouseMovedEventQueued () {
        !           537:   XEvent event;
        !           538:   if (XPending() == 0) return (FALSE);
        !           539:   XPeekEvent (&event);
        !           540:   return (event.type == MouseMoved);
        !           541:   }
        !           542: 
        !           543: 
        !           544: ProcessOuterWindowEvent (event)
        !           545:   XEvent *event;
        !           546:   {
        !           547:   if (event->type != ExposeWindow)
        !           548:     return;
        !           549:   if ((outer_frame.height == ((XExposeWindowEvent *)event)->height)
        !           550:      && (outer_frame.width == ((XExposeWindowEvent *)event)->width))
        !           551:      /* if this isn't a resize, there's nothing to do here. */
        !           552:      return;
        !           553: 
        !           554:   /* the outer window's size has changed.  Must rearrange subwindows. */
        !           555:   outer_frame.height = ((XExposeWindowEvent *)event)->height;
        !           556:   outer_frame.width = ((XExposeWindowEvent *)event)->width;
        !           557:   XDestroySubwindows (outer_window);
        !           558:   LayoutStage2 ();
        !           559:   XMapSubwindows (outer_window);
        !           560:   }
        !           561:   
        !           562: ProcessCommandButtonEvent (command, event)
        !           563:   struct command_data *command;
        !           564:   XEvent *event;
        !           565:   {
        !           566:   static struct command_data *button_down_command;
        !           567:   
        !           568:   switch (event->type) {
        !           569: 
        !           570:     case ExposeWindow:
        !           571:       if (command->inverted)
        !           572:         XClear (command->window);
        !           573:       XTextMask (
        !           574:         command->window,    /* w */
        !           575:        command->x_offset,  /* x */
        !           576:        0,                  /* y */
        !           577:        command->name,      /* string */
        !           578:        command->name_length,    /* length */
        !           579:        font,               /* font */
        !           580:        foreground);        /* source pixel */
        !           581:       if (command->inverted)
        !           582:        InvertCommandWindow (command);
        !           583:       break;
        !           584: 
        !           585:     case ButtonPressed:
        !           586:       if (button_down_command != NULL)
        !           587:         break;  /* must be a second button push--ignore */
        !           588:       button_down_command = command;
        !           589:       InvertCommandWindow (command);
        !           590:       command->inverted = TRUE;
        !           591:       break;
        !           592: 
        !           593:     case LeaveWindow:
        !           594:       if (command == button_down_command) {
        !           595:        InvertCommandWindow (command);
        !           596:        command->inverted = FALSE;
        !           597:        button_down_command = NULL;
        !           598:        }
        !           599:       break;
        !           600: 
        !           601:     case ButtonReleased:
        !           602:       if (command == button_down_command) {
        !           603:        (*command->proc)(command->data);
        !           604:        button_down_command = NULL;
        !           605:        InvertCommandWindow (command);
        !           606:        command->inverted = FALSE;
        !           607:        }
        !           608:       break;
        !           609:       
        !           610:     }
        !           611:   }
        !           612: 
        !           613: 
        !           614: InvertCommandWindow (command)
        !           615:   struct command_data *command;
        !           616:   {
        !           617:   XPixFill (
        !           618:     command->window,
        !           619:     0,     /* x */
        !           620:     0,             /* y */
        !           621:     400,    /* width = "infinity " */
        !           622:     fontInfo.height,   /* height */
        !           623:     1,     /* pixel */
        !           624:     NULL,   /* clipmask bitmap */
        !           625:     GXinvert,  /* function */
        !           626:     invertplane);      /* planes */
        !           627:   }  /* end of InvertCommandWindow procedure */
        !           628: 
        !           629:          
        !           630: /* WhatSquare returns TRUE if mouse is outside grid, FALSE if inside.
        !           631:    If it returns FALSE, it assigns to *x_square and *y_square. */
        !           632: 
        !           633: boolean WhatSquare (event, x_square, y_square)
        !           634:   register XEvent *event;
        !           635:   register int *x_square, *y_square; /*RETURN*/
        !           636:   {
        !           637:   int x = ((XKeyPressedEvent *)event)->x;
        !           638:   int y = ((XKeyPressedEvent *)event)->y;
        !           639:   if ((x < 0) || (y < 0))
        !           640:     return (TRUE);
        !           641:   *x_square = x/square_size;
        !           642:   *y_square = y/square_size;
        !           643:   return ((*x_square >= squares_wide) || (*y_square >= squares_high));
        !           644:   }  /* end of WhatSquare procedure */
        !           645: 
        !           646: 
        !           647: RepaintGridLines(how)
        !           648:   enum RepaintGridType how;
        !           649:   {
        !           650:   RepaintGridLinesPartially (0, 0, squares_wide, squares_high, how, TRUE);
        !           651:   }
        !           652: 
        !           653: RepaintGridLinesPartially (x1, y1, x2, y2, how, include_boundaries)
        !           654:   int x1, y1, x2, y2;
        !           655:   enum RepaintGridType how;
        !           656:   boolean include_boundaries;
        !           657:   {
        !           658:   register int i;
        !           659:   Vertex v[2];
        !           660:   int pixel, func, planes;
        !           661:   Pattern pattern;
        !           662:   switch (how) {
        !           663:     case e_AgainstBackground:
        !           664:        pixel = foreground;
        !           665:        pattern = DottedPattern;
        !           666:        func = GXcopy;
        !           667:        planes = AllPlanes;
        !           668:        break;
        !           669:     case e_AgainstForeground:
        !           670:        pixel = background;
        !           671:        pattern = InverseDottedPattern;
        !           672:        func = GXcopy;
        !           673:        planes = AllPlanes;
        !           674:        break;
        !           675:     case e_Invert:
        !           676:        pixel = 1;
        !           677:        pattern = SolidLine;
        !           678:        func = GXinvert;
        !           679:        planes = invertplane;
        !           680:        break;
        !           681:     }
        !           682:     
        !           683:   /* draw vertical grid lines */
        !           684:   v[0].flags = v[1].flags = 0;
        !           685:   v[0].y = y1*square_size;
        !           686:   v[0].y += (v[0].y & 1);  /* make sure pattern is aligned on even bit boundary */
        !           687:   v[1].y = y2*square_size;
        !           688:   if (!include_boundaries) {x1++;x2--;}
        !           689:   v[0].x = v[1].x = x1*square_size;
        !           690:   for (i=x1;i<=x2; i++) {
        !           691:      XDrawDashed (
        !           692:        grid_window,
        !           693:        v,  /* vertices */
        !           694:        2,  /* vertex count */
        !           695:        1,  /* width */
        !           696:        1,  /* height */
        !           697:        pixel,
        !           698:        pattern,
        !           699:        func,
        !           700:        planes);        /* planes */
        !           701:      v[0].x = (v[1].x += square_size);
        !           702:      }
        !           703:   if (!include_boundaries) {x1--;x2++;}
        !           704: 
        !           705:   /* draw horizontal grid lines */
        !           706:   v[0].flags = v[1].flags = 0;
        !           707:   v[0].x = x1*square_size;
        !           708:   v[0].x += (v[0].x & 1);  /* make sure pattern is aligned on even bit boundary */
        !           709:   v[1].x = x2*square_size;
        !           710:   if (!include_boundaries) {y1++;y2--;}
        !           711:   v[0].y = v[1].y = y1*square_size;
        !           712:   for (i=y1;i<=y2;i++) {
        !           713:     XDrawDashed (
        !           714:        grid_window,
        !           715:        v,  /* vertices */
        !           716:        2,  /* vertex count */
        !           717:        1,  /* width */
        !           718:        1,  /* height */
        !           719:        pixel,
        !           720:        pattern,
        !           721:        func,
        !           722:        planes);        /* planes */
        !           723:      v[0].y = (v[1].y += square_size);
        !           724:      }
        !           725:   }  /* end of RepaintGridLinesPartially procedure */
        !           726: 
        !           727: 
        !           728: RefillGrid (paint_background)
        !           729:   boolean paint_background;
        !           730:   {
        !           731:   RefillGridPartially (0, 0, squares_wide-1, squares_high-1, paint_background);
        !           732:   }
        !           733: 
        !           734: 
        !           735: RefillGridPartially(x1, y1, x2, y2, paint_background)
        !           736:   register int x1, y1, x2, y2;
        !           737:   boolean paint_background;
        !           738:   {
        !           739:   register i, j;
        !           740:   for (i=x1; i<=x2; i++) {
        !           741:     for (j=y1; j<=y2; j++) {
        !           742:       bit b = GetRasterBit (raster, i, j);
        !           743:       if (b || paint_background)
        !           744:        PaintSquare (i, j, (b ? foreground : background));
        !           745:       }
        !           746:     }
        !           747:   }  /* end of RefillGridPartially procedure */
        !           748: 
        !           749: 
        !           750: PaintSquare(x, y, pixel)
        !           751:   int x, y;
        !           752:   int pixel;  /* display function */  
        !           753:   {
        !           754:   XPixSet (
        !           755:     grid_window,
        !           756:     x*square_size + 1, /* x */
        !           757:     y*square_size + 1,         /* y */
        !           758:     square_size - 1,   /* width */
        !           759:     square_size - 1,   /* height */
        !           760:     pixel);                            /* pixel */
        !           761:   }  /* end of PaintSquare procedure */
        !           762: 
        !           763: InvertSquare(x, y)
        !           764:   int x, y;
        !           765:   {
        !           766:   XPixFill (
        !           767:     grid_window,
        !           768:     x*square_size + 1, /* x */
        !           769:     y*square_size + 1,         /* y */
        !           770:     square_size - 1,   /* width */
        !           771:     square_size - 1,   /* height */
        !           772:     1,                         /* pixel */
        !           773:     NULL,                      /* clipmask */
        !           774:     GXinvert,                          /* function */
        !           775:     invertplane);              /* planes */
        !           776:   }
        !           777: 
        !           778: bit GetRasterBit (raster, x, y)
        !           779:   short *raster;
        !           780:   register int x;
        !           781:   int y;
        !           782:   {
        !           783:   register short *word = raster + x/16 + y*((squares_wide+15)/16);
        !           784:   return ( (*word & (1 << (x % 16))) ? 1 : 0);
        !           785:   }  /* end of GetRasterBit procedure */
        !           786: 
        !           787: 
        !           788: SetRasterBit (raster, x, y, new)
        !           789:   short *raster;
        !           790:   register int x;
        !           791:   int y;
        !           792:   bit new;
        !           793:   {
        !           794:   register short *word = raster + x/16 + y*((squares_wide+15)/16);
        !           795:   x %= 16;
        !           796:   *word = (new << x) | (*word & ~(1 << x));
        !           797:   }  /* end of SetRasterBit procedure */
        !           798: 
        !           799: 
        !           800: InvertRasterBit (raster, x, y)
        !           801:   short *raster;
        !           802:   register int x;
        !           803:   int y;
        !           804:   {
        !           805:   register short *word = raster + x/16 + y*((squares_wide+15)/16);
        !           806:   *word ^= (1 << (x % 16));
        !           807:   }  /* end of InvertRasterBit procedure */
        !           808: 
        !           809: 
        !           810: RepaintRaster() {
        !           811:   XBitmapBitsPut (
        !           812:     raster_window,
        !           813:     3, /* x */
        !           814:     3, /* y */
        !           815:     squares_wide,   /* width */
        !           816:     squares_high,   /* height */
        !           817:     raster,        /* data */
        !           818:     foreground,            /* foreground */
        !           819:     background,            /* background */
        !           820:     0,             /* mask */
        !           821:     GXcopy,        /* func */
        !           822:     AllPlanes);            /* planes */
        !           823:   }  /* end of RepaintRaster procedure */
        !           824: 
        !           825: 
        !           826: RepaintRasterInverted () {
        !           827:   XBitmapBitsPut (
        !           828:     raster_invert_window,
        !           829:     3, /* x */
        !           830:     3, /* y */
        !           831:     squares_wide,   /* width */
        !           832:     squares_high,   /* height */
        !           833:     raster,        /* data */
        !           834:     background,            /* foreground */
        !           835:     foreground,            /* background */
        !           836:     0,             /* mask */
        !           837:     GXcopy,        /* func */
        !           838:     AllPlanes);            /* planes */
        !           839:   } /* end of RepaintRasterInverted procedure */
        !           840: 
        !           841: 
        !           842: WriteOutputToFile (file)
        !           843:   FILE *file;
        !           844:   {
        !           845:   register int i;
        !           846:   fprintf (file, "#define %s_width %d\n", stripped_name, squares_wide);
        !           847:   fprintf (file, "#define %s_height %d\n", stripped_name, squares_high);
        !           848:   if (x_hot_spot != OUT_OF_RANGE)
        !           849:     fprintf (file, "#define %s_x_hot %d\n", stripped_name, x_hot_spot);
        !           850:   if (y_hot_spot != OUT_OF_RANGE)
        !           851:     fprintf (file, "#define %s_y_hot %d\n", stripped_name, y_hot_spot);
        !           852:   fprintf (file, "static short %s_bits[] = {\n   0x%04x",
        !           853:     stripped_name, (u_short) raster[0]);
        !           854: 
        !           855:   for (i=1;i<raster_length;i++) {
        !           856:     fprintf (file, ",");
        !           857:     fprintf (file, (i % 4) ? " " : "\n   ");
        !           858:     fprintf (file, "0x%04x", (u_short) raster[i]);
        !           859:     }
        !           860:   fprintf (file, "};\n");
        !           861:   }  /* end of WriteOutputToFile procedure */
        !           862: 
        !           863: 
        !           864: DimensionsFromFile(file) 
        !           865:   FILE *file;
        !           866:   {
        !           867:   char variable[81];
        !           868:   int value;
        !           869:   while (fscanf (file, "#define %80s %d\n", variable, &value) == 2) {
        !           870:     if (StringEndsWith (variable, "width"))
        !           871:        squares_wide = value;
        !           872:     else if (StringEndsWith (variable, "height"))
        !           873:        squares_high = value;
        !           874:     else if (StringEndsWith (variable, "x_hot"))
        !           875:        x_hot_spot = value;
        !           876:     else if (StringEndsWith (variable, "y_hot"))
        !           877:        y_hot_spot = value;
        !           878:     else fprintf (stderr, "Unrecognized variable '%s' in file '%s'\n",
        !           879:        variable, filename);
        !           880:     }
        !           881: 
        !           882:   if (squares_wide <= 0 || squares_wide == OUT_OF_RANGE) {
        !           883:     fprintf (stderr,
        !           884:       "%s: file '%s' does not have a valid width dimension\n",
        !           885:       progname, filename);
        !           886:     exit(1);
        !           887:     }
        !           888:     
        !           889:   if (squares_high <= 0 || squares_high == OUT_OF_RANGE) {
        !           890:     fprintf (stderr,
        !           891:       "%s: file '%s' does not have a valid height dimension\n",
        !           892:       progname, filename);
        !           893:     exit(1);
        !           894:     }  
        !           895:   }  /* end of DimensionsFromFile procedure */
        !           896: 
        !           897: 
        !           898: InitialValuesFromFile(file)
        !           899:   FILE *file;
        !           900:   {
        !           901:   register int i, status;
        !           902:   char s[81];
        !           903: 
        !           904:   status = fscanf (file, "static short %80s = { 0x%4hx", s, &raster[0]);
        !           905:   if ((status != 2) || !StringEndsWith (s, "bits[]")) {
        !           906:     fprintf (stderr,
        !           907:       "%s: file '%s' has an invalid 1st array element\n",
        !           908:       progname, filename);
        !           909:     exit (1);
        !           910:     }
        !           911:   for (i=1;i<raster_length;i++) {
        !           912:     status = fscanf (file, ", 0x%4hx", &raster[i]);
        !           913:     if (status != 1) {
        !           914:       fprintf (stderr,
        !           915:         "%s: file '%s' has an invalid %dth array element\n",
        !           916:         progname, filename, i+1);
        !           917:       exit (1);
        !           918:       }
        !           919:     }
        !           920: 
        !           921:   }  /* end of InitialValuesFromFile procedure */
        !           922: 
        !           923: 
        !           924: char *StripName(name)
        !           925:   char *name;
        !           926:   {
        !           927:   char *rindex(), *index();
        !           928:   char *begin = rindex (name, '/');
        !           929:   char *end, *result;
        !           930:   int length;
        !           931:   begin = (begin ? begin+1 : name);
        !           932:   end = index (begin, '.');
        !           933:   length = (end ? (end - begin) : strlen (begin));
        !           934:   result = (char *) malloc (length + 1);
        !           935:   strncpy (result, begin, length);
        !           936:   result [length] = '\0';
        !           937:   return (result);
        !           938:   }
        !           939: 
        !           940: 
        !           941: char *BackupName(name)
        !           942:   char *name;
        !           943:   {
        !           944:   int name_length = strlen (name);
        !           945:   char *result = (char *) malloc (name_length+2);
        !           946:   strncpy (result, name, name_length);
        !           947:   result [name_length] = '~';
        !           948:   result [name_length+1] = '\0';
        !           949:   return (result);
        !           950:   }
        !           951: 
        !           952: char *TmpFileName(name)
        !           953:   char *name;
        !           954:   {
        !           955:   {
        !           956:   char *rindex();
        !           957:   char *begin = rindex (name, '/');
        !           958:   if (begin)
        !           959:     name = begin+1;
        !           960:   }
        !           961:   {
        !           962:   char *tmp = "/tmp/";
        !           963:   int name_length = strlen (name);
        !           964:   int tmp_length = strlen (tmp);
        !           965:   int result_length = name_length + tmp_length;
        !           966:   char *result = (char *) malloc (result_length + 1);
        !           967:   strncpy (result, tmp, tmp_length);
        !           968:   strncpy (result+tmp_length, name, name_length);
        !           969:   result [result_length] = '\0';
        !           970:   return (result);
        !           971:   }
        !           972:   }
        !           973: 
        !           974: /* StringEndsWith returns TRUE if "s" ends with "suffix", else returns FALSE */
        !           975: boolean StringEndsWith (s, suffix)
        !           976:   char *s, *suffix;
        !           977:   {
        !           978:   int s_len = strlen (s);
        !           979:   int suffix_len = strlen (suffix);
        !           980:   return (strcmp (s + s_len - suffix_len, suffix) == 0);
        !           981:   }
        !           982: 
        !           983: /* LayoutStage1 determines the size and y-position of all commmand
        !           984:    windows and both raster windows.  It fills in everything in the
        !           985:    commands[] array except the "window" field, and fills in everything in
        !           986:    the frames[] array except for the "self" (window) and "x" fields. It
        !           987:    returns in *width and *bottom the dimensions of the command area to be
        !           988:    created.  
        !           989:    This routine is called only once, at startup time.
        !           990:    Everything done at this stage stays the same even if the user later
        !           991:    reshapes the window. */
        !           992: 
        !           993: LayoutStage1 ()
        !           994:   {
        !           995:   int widths [N_COMMANDS];
        !           996:   int maxwidth = 0;
        !           997:   int ypos = TOP_MARGIN;
        !           998:   register int i;
        !           999:   register OpaqueFrame *frame;
        !          1000:   int ClearOrSetAll(), InvertAll(),
        !          1001:       ClearOrSetArea(), InvertArea(), CopyOrMoveArea(),
        !          1002:       Line(), Circle(), FilledCircle(),
        !          1003:       SetHotSpot(), ClearHotSpot(),
        !          1004:       WriteOutput(), Quit();
        !          1005: 
        !          1006:   commands[0].name = "Clear All";
        !          1007:   commands[0].proc = ClearOrSetAll;
        !          1008:   commands[0].data = 0;
        !          1009:   
        !          1010:   commands[1].name = "Set All";
        !          1011:   commands[1].proc = ClearOrSetAll;
        !          1012:   commands[1].data = 1;
        !          1013: 
        !          1014:   commands[2].name = "Invert All";
        !          1015:   commands[2].proc = InvertAll;
        !          1016: 
        !          1017:   commands[3].name = "Clear Area";
        !          1018:   commands[3].proc = ClearOrSetArea;
        !          1019:   commands[3].data = 0;
        !          1020: 
        !          1021:   commands[4].name = "Set Area";
        !          1022:   commands[4].proc = ClearOrSetArea;
        !          1023:   commands[4].data = 1;
        !          1024: 
        !          1025:   commands[5].name = "Invert Area";
        !          1026:   commands[5].proc = InvertArea;
        !          1027: 
        !          1028:   commands[6].name = "Copy Area";
        !          1029:   commands[6].proc = CopyOrMoveArea;
        !          1030:   commands[6].data = COPY;
        !          1031: 
        !          1032:   commands[7].name = "Move Area";
        !          1033:   commands[7].proc = CopyOrMoveArea;
        !          1034:   commands[7].data = MOVE;
        !          1035: 
        !          1036:   commands[8].name = "Overlay Area";
        !          1037:   commands[8].proc = CopyOrMoveArea;
        !          1038:   commands[8].data = OVERLAY;
        !          1039: 
        !          1040:   commands[9].name = "Line";
        !          1041:   commands[9].proc = Line;
        !          1042: 
        !          1043:   commands[10].name = "Circle";
        !          1044:   commands[10].proc = Circle;
        !          1045:   commands[10].data = 0;
        !          1046: 
        !          1047:   commands[11].name = "Filled Circle";
        !          1048:   commands[11].proc = Circle;
        !          1049:   commands[11].data = 1;
        !          1050: 
        !          1051:   commands[12].name = "Set HotSpot";
        !          1052:   commands[12].proc = SetHotSpot;
        !          1053: 
        !          1054:   commands[13].name = "Clear HotSpot";
        !          1055:   commands[13].proc = ClearHotSpot;
        !          1056: 
        !          1057:   commands[14].name = "Write Output";
        !          1058:   commands[14].proc = WriteOutput;
        !          1059: 
        !          1060:   commands[15].name = "Quit";
        !          1061:   commands[15].proc = Quit;
        !          1062: 
        !          1063: 
        !          1064:   for (i=0;i<N_COMMANDS;i++) {
        !          1065:     widths[i] = XQueryWidth (commands[i].name, font);
        !          1066:     if (maxwidth < widths[i])
        !          1067:       maxwidth = widths[i];
        !          1068:     }
        !          1069: 
        !          1070:   maxwidth += 4; /* so even widest command has a little space around it */
        !          1071: 
        !          1072:   for (i=0;i<N_COMMANDS;i++) {
        !          1073:     register struct command_data *command = &commands[i];
        !          1074:     frame = &frames[i];
        !          1075:     command->name_length = strlen (command->name);
        !          1076:     command->x_offset = (maxwidth - widths[i])/2;
        !          1077:     frame->y = ypos;
        !          1078:     frame->width = maxwidth;
        !          1079:     frame->height = fontInfo.height;
        !          1080:     frame->bdrwidth = 1;
        !          1081:     frame->border = border;
        !          1082:     frame->background = backmap;
        !          1083:     ypos += fontInfo.height + 5;
        !          1084:     if (i==2 || i == 5 || i == 8 || i == 11 || i == 13)
        !          1085:       ypos += fontInfo.height + 5;
        !          1086:       /* for gaps between groups;  pretty random! */
        !          1087: 
        !          1088:     }
        !          1089:   
        !          1090:   /* set up raster window */
        !          1091:   frame = &frame[N_COMMANDS];
        !          1092:   frame = &frames[i];
        !          1093:   frame->y = (ypos += AROUND_RASTER_MARGIN);
        !          1094:   frame->width = squares_wide + 6;
        !          1095:   frame->height = squares_high + 6;
        !          1096:   frame->bdrwidth = 1;
        !          1097:   frame->border = border;
        !          1098:   frame->background = backmap;
        !          1099:   
        !          1100:   /* raster invert window is the same, except for y position */
        !          1101:   *(frame+1) = *frame;
        !          1102:   (++frame)->y += squares_high + 8 + AROUND_RASTER_MARGIN;
        !          1103: 
        !          1104:   }
        !          1105: 
        !          1106: /* LayoutStage2 is called both at startup time and whenever the user
        !          1107:    resizes the outer window.  It figures out what the new grid square size
        !          1108:    should be, determines the size and position of all subwindows, then
        !          1109:    creates (but does not map) the subwindows. */
        !          1110: 
        !          1111: LayoutStage2 ()
        !          1112:   {
        !          1113:   int x_room = outer_frame.width - 1
        !          1114:      - LEFT_MARGIN - frames[0].width - GRID_TO_COMMAND_MARGIN - RIGHT_MARGIN;
        !          1115:   int y_room = outer_frame.height - 1
        !          1116:      - TOP_MARGIN - BOTTOM_MARGIN;
        !          1117:   int i;
        !          1118:   int command_x_offset;
        !          1119:   OpaqueFrame *grid_frame = &frames[N_COMMANDS+2];
        !          1120:   
        !          1121:   x_room /= squares_wide;
        !          1122:   y_room /= squares_high;
        !          1123:   square_size = min (x_room, y_room);
        !          1124:   
        !          1125:   /* fill in the grid window's frame */
        !          1126:   grid_frame->x = LEFT_MARGIN;
        !          1127:   grid_frame->y = TOP_MARGIN;
        !          1128:   grid_frame->width = (squares_wide * square_size) + 1;
        !          1129:   grid_frame->height = (squares_high * square_size) + 1;
        !          1130:   grid_frame->bdrwidth = 0;
        !          1131:   grid_frame->border = NULL;
        !          1132:   grid_frame->background = backmap;
        !          1133: 
        !          1134:   /* fill in x offsets for command window frames */
        !          1135:   command_x_offset = grid_frame->x + grid_frame->width
        !          1136:     + GRID_TO_COMMAND_MARGIN;
        !          1137:   for (i=0;i<N_COMMANDS;i++)
        !          1138:     frames[i].x = command_x_offset;
        !          1139: 
        !          1140:   /* fill in x offsets for raster and raster-inverted window frames */
        !          1141:   frames[N_COMMANDS].x = frames[N_COMMANDS+1].x =
        !          1142:     grid_frame->x + grid_frame->width + AROUND_RASTER_MARGIN;
        !          1143: 
        !          1144:   /* create all the subwindows */
        !          1145:   XCreateWindows (outer_window, frames, N_COMMANDS+3);
        !          1146: 
        !          1147:   /* stow away all the resulting window id's, and select input */
        !          1148:   for (i=0;i<N_COMMANDS;i++)
        !          1149:     XSelectInput (commands[i].window = frames[i].self, 
        !          1150:        ButtonPressed | ButtonReleased | LeaveWindow | ExposeWindow);
        !          1151: 
        !          1152:   XSelectInput (raster_window = frames[N_COMMANDS].self, ExposeWindow);
        !          1153:   XSelectInput (raster_invert_window = frames[N_COMMANDS+1].self,
        !          1154:     ExposeWindow);
        !          1155:   XSelectInput (grid_window = grid_frame->self,
        !          1156:     RightDownMotion | MiddleDownMotion | LeftDownMotion
        !          1157:     | ExposeRegion | ButtonPressed | ButtonReleased);
        !          1158:     /* ButtonReleased is selected for AskUserForArea's benefit */
        !          1159: 
        !          1160:   }
        !          1161: 
        !          1162: /* OuterWindowDims determines the minimum size for the outer window,
        !          1163:    based on three constraints:  the grid square size, the width of
        !          1164:    the command/raster area, and the minimum height of the
        !          1165:    command/raster area ("right side" of the window).  It is called
        !          1166:    at startup time. */
        !          1167: 
        !          1168: OuterWindowDims (square_size, right_side_width,
        !          1169:   right_side_bottom, width, height)
        !          1170:   int square_size, right_side_width, right_side_bottom;
        !          1171:   int *width, *height; /* RETURN */
        !          1172:   {
        !          1173:   *width = LEFT_MARGIN + squares_wide*square_size + 1 + right_side_width;
        !          1174:   *height = TOP_MARGIN + squares_high*square_size + 1 + BOTTOM_MARGIN;
        !          1175:   if (*height < right_side_bottom)
        !          1176:     *height = right_side_bottom;
        !          1177:   }
        !          1178: 
        !          1179: 
        !          1180: ClearOrSetAll(b)
        !          1181:   bit b;  /* 0 for clear, 1 for set */
        !          1182:   {
        !          1183:   register int i;
        !          1184:   register int new = (b ? ~0: 0);
        !          1185:   for (i=0;i<raster_length;i++)
        !          1186:     raster[i] = new;
        !          1187:   changed = TRUE;
        !          1188:   XPixSet (
        !          1189:     grid_window,    /* window */
        !          1190:     0,     /* x */
        !          1191:     0,     /* y */
        !          1192:     squares_wide*square_size+1,        /* width */
        !          1193:     squares_high*square_size+1,        /* height */
        !          1194:     b ? foreground : background);
        !          1195:   RepaintGridLines (b ? e_AgainstForeground : e_AgainstBackground);
        !          1196:   RepaintRaster();
        !          1197:   RepaintRasterInverted();
        !          1198:   if (x_hot_spot != OUT_OF_RANGE)
        !          1199:     HighlightHotSpot();
        !          1200:   }  /* end of ClearAll procedure */
        !          1201: 
        !          1202: 
        !          1203: ClearOrSetArea(b)
        !          1204:   bit b;  /* 0 for clear, 1 for set */
        !          1205:   {
        !          1206:   int x1, y1, x2, y2;
        !          1207:   register int x, y;
        !          1208:   if (AskUserForArea (&x1, &y1, &x2, &y2))
        !          1209:     return;
        !          1210:   for (x=x1;x<=x2;x++)
        !          1211:     for (y=y1;y<=y2;y++)
        !          1212:       SetRasterBit (raster, x, y, b);
        !          1213:   XPixSet (
        !          1214:     grid_window,       /* window */
        !          1215:     x1*square_size+1,  /* x */
        !          1216:     y1*square_size+1,  /* y */
        !          1217:     (x2-x1+1)*square_size-1,  /* width */
        !          1218:     (y2-y1+1)*square_size-1,  /* height */
        !          1219:     b ? foreground : background);
        !          1220:   RepaintGridLinesPartially (x1, y1, x2+1, y2+1, b ? e_AgainstForeground : e_AgainstBackground, FALSE);
        !          1221:   if (x_hot_spot >= x1 && x_hot_spot <= x2
        !          1222:     && y_hot_spot >= y1 && y_hot_spot <= y2)
        !          1223:        HighlightHotSpot();
        !          1224:   changed = TRUE;
        !          1225:   RepaintRaster();
        !          1226:   RepaintRasterInverted();
        !          1227:   x1_square_exed_through = y1_square_exed_through = OUT_OF_RANGE;
        !          1228:   x2_square_exed_through = y2_square_exed_through = OUT_OF_RANGE;
        !          1229:   }  /* end of ClearArea procedure */
        !          1230: 
        !          1231: 
        !          1232: InvertAll() {
        !          1233:   register int i;
        !          1234:   for (i=0;i<raster_length;i++)
        !          1235:     raster[i] ^= ~0;  /* invert = exclusive or with all 1's */
        !          1236:   changed = TRUE;
        !          1237:   XPixFill (
        !          1238:     grid_window,    /* window */
        !          1239:     0,     /* x */
        !          1240:     0,     /* y */
        !          1241:     squares_wide*square_size+1,        /* width */
        !          1242:     squares_high*square_size+1,        /* height */
        !          1243:     1,     /* pixel */
        !          1244:     NULL,   /* clipmask */
        !          1245:     GXinvert,  /* function */
        !          1246:     invertplane);  /* plane mask */
        !          1247:   RepaintGridLines (e_Invert);
        !          1248:   RepaintRaster();
        !          1249:   RepaintRasterInverted();
        !          1250:   }  /* end of InvertAll procedure */
        !          1251: 
        !          1252: 
        !          1253: InvertArea() {
        !          1254:   int x1, y1, x2, y2;
        !          1255:   register int x, y;
        !          1256:   if (AskUserForArea (&x1, &y1, &x2, &y2))
        !          1257:     return;
        !          1258:   for (x=x1;x<=x2;x++)
        !          1259:     for (y=y1;y<=y2;y++)
        !          1260:       InvertRasterBit (raster, x, y);
        !          1261:   ExThroughRectangle (x1, y1, x2, y2);  /* wipe out X-outs */
        !          1262:   XPixFill (
        !          1263:     grid_window,       /* window */
        !          1264:     x1*square_size+1,  /* x */
        !          1265:     y1*square_size+1,  /* y */
        !          1266:     (x2-x1+1)*square_size-1,  /* width */
        !          1267:     (y2-y1+1)*square_size-1,  /* height */
        !          1268:     1,     /* pixel */
        !          1269:     NULL,   /* clipmask */
        !          1270:     GXinvert,  /* function */
        !          1271:     invertplane); /* plane mask */
        !          1272:   RepaintGridLinesPartially (x1, y1, x2+1, y2+1, e_Invert, FALSE);
        !          1273:   changed = TRUE;
        !          1274:   RepaintRaster();
        !          1275:   RepaintRasterInverted();
        !          1276:   x1_square_exed_through = y1_square_exed_through = OUT_OF_RANGE;
        !          1277:   x2_square_exed_through = y2_square_exed_through = OUT_OF_RANGE;
        !          1278:   }  /* end of InvertArea procedure */
        !          1279: 
        !          1280: 
        !          1281: CopyOrMoveArea (what)
        !          1282:   {
        !          1283:   int x1, y1, x2, y2;
        !          1284:   int x1dest, y1dest;
        !          1285:   if (AskUserForArea (&x1, &y1, &x2, &y2))
        !          1286:     return;
        !          1287:   if (AskUserForDest (&x1dest, &y1dest, x2-x1+1, y2-y1+1))
        !          1288:     /* button released outside grid */
        !          1289:     ExThroughRectangle (x1_square_exed_through, y1_square_exed_through,
        !          1290:       x2_square_exed_through, y2_square_exed_through);
        !          1291:   else {
        !          1292:     register int xsrc, ysrc, xdest, ydest;
        !          1293:     register short *new_raster =
        !          1294:       (short *) malloc (raster_length*sizeof(short));
        !          1295: 
        !          1296:     if (x_hot_spot != OUT_OF_RANGE)
        !          1297:        HighlightHotSpot();  /* actually UNhighlight it */
        !          1298: 
        !          1299:     /* copy whole raster to new raster */
        !          1300:     bcopy (raster, new_raster, raster_length*sizeof(short));
        !          1301:     
        !          1302:     if (what == MOVE)
        !          1303:       /* clear source bits in new raster.  this is VERY inefficient.
        !          1304:          sure wish we had BitBlt available in user memory! */
        !          1305:       for (xsrc = x1; xsrc <= x2; xsrc++)
        !          1306:         for (ysrc = y1; ysrc <= y2; ysrc++)
        !          1307:          SetRasterBit (new_raster, xsrc, ysrc, 0);
        !          1308: 
        !          1309:     /* copy old source bits to new destination. this is VERY inefficient.
        !          1310:        sure wish we had BitBlt available in user memory! */
        !          1311: 
        !          1312:     for (xsrc = x1, xdest = x1dest;
        !          1313:       xsrc<=x2 && xdest < squares_wide; xsrc++, xdest++) 
        !          1314:         for (ysrc = y1, ydest = y1dest;
        !          1315:           ysrc<=y2 && ydest < squares_high; ysrc++, ydest++)
        !          1316:            if (what == OVERLAY) {
        !          1317:                if (GetRasterBit (raster, xsrc, ysrc))
        !          1318:                    SetRasterBit (new_raster, xdest, ydest, 1);
        !          1319:            } else
        !          1320:                SetRasterBit (new_raster, xdest, ydest, 
        !          1321:                  GetRasterBit (raster, xsrc, ysrc));
        !          1322: 
        !          1323:     free (raster);
        !          1324:     raster = new_raster;
        !          1325:     if (what == MOVE)
        !          1326:        RepaintRectangles (x1, y1, x2, y2, x1dest, y1dest);
        !          1327:     else {
        !          1328:        ExThroughRectangle (x1_square_exed_through, y1_square_exed_through,
        !          1329:            x2_square_exed_through, y2_square_exed_through);
        !          1330:        RefillGridPartially (x1dest, y1dest, xdest-1, ydest-1, TRUE);
        !          1331:        }
        !          1332: 
        !          1333:     if (x_hot_spot != OUT_OF_RANGE)
        !          1334:        HighlightHotSpot();  /* put the hot spot back on the screen */
        !          1335: 
        !          1336:     RepaintRaster();
        !          1337:     RepaintRasterInverted();
        !          1338:     changed = TRUE;
        !          1339:     x1_square_plus_through = y1_square_plus_through = OUT_OF_RANGE;
        !          1340:     x2_square_plus_through = y2_square_plus_through = OUT_OF_RANGE;
        !          1341:     }
        !          1342:     
        !          1343:   x1_square_exed_through = y1_square_exed_through = OUT_OF_RANGE;
        !          1344:   x2_square_exed_through = y2_square_exed_through = OUT_OF_RANGE;
        !          1345:   }  /* end of CopyArea procedure */
        !          1346: 
        !          1347: #define MAX(a,b)       (((a) > (b)) ? (a) : (b))
        !          1348: #define MIN(a,b)       (((a) < (b)) ? (a) : (b))
        !          1349: #define ABS(a)         (((a) >= 0) ? (a) : -(a))
        !          1350: #define CheckSetRasterBit(r,x,y,c)     \
        !          1351:        if ((x) >= 0 && (x) < squares_wide && (y) >= 0 && (y) < squares_high) \
        !          1352:                SetRasterBit(r, x, y, c)
        !          1353: 
        !          1354: Line ()
        !          1355: {
        !          1356:        int     i, x1, y1, x2, y2;
        !          1357:        double  dx, dy, x, y, diff;
        !          1358: 
        !          1359:        if (AskUserForPoint(&x1, &y1, 0))
        !          1360:                return;
        !          1361:        if (AskUserForPoint(&x2, &y2, 1))
        !          1362:                return;
        !          1363:        ExThroughRectangle (x1_square_exed_through, y1_square_exed_through,
        !          1364:                x2_square_exed_through, y2_square_exed_through);
        !          1365:        PlusThroughRectangle (x1_square_plus_through, y1_square_plus_through,
        !          1366:                x2_square_plus_through, y2_square_plus_through);
        !          1367: 
        !          1368:        dx = x2 - x1;
        !          1369:        dy = y2 - y1;
        !          1370:        x = x1 + 0.5;
        !          1371:        y = y1 + 0.5;
        !          1372:        diff = MAX(ABS(dx), ABS(dy));
        !          1373:        if (diff == 0)
        !          1374:                diff = 0.9;
        !          1375:        dx /= diff;
        !          1376:        dy /= diff;
        !          1377:        for (i = 0; i <= (int)diff; i++) {
        !          1378:                SetRasterBit(raster, (int)x, (int)y, 1);
        !          1379:                x += dx;
        !          1380:                y += dy;
        !          1381:        }
        !          1382:        RefillGridPartially(MIN(x1, x2), MIN(y1, y2), MAX(x1, x2), MAX(y1, y2),
        !          1383:                FALSE);
        !          1384:        changed = TRUE;
        !          1385:        x1_square_exed_through = y1_square_exed_through = OUT_OF_RANGE;
        !          1386:        x2_square_exed_through = y2_square_exed_through = OUT_OF_RANGE;
        !          1387:        x1_square_plus_through = y1_square_plus_through = OUT_OF_RANGE;
        !          1388:        x2_square_plus_through = y2_square_plus_through = OUT_OF_RANGE;
        !          1389:        RepaintRaster();
        !          1390:        RepaintRasterInverted();
        !          1391: }
        !          1392: 
        !          1393: #include <math.h>
        !          1394: 
        !          1395: Circle(filled)
        !          1396: {
        !          1397:        int     i, j, x, x1, y1, x2, y2, dx, dy;
        !          1398:        double  rad, half;
        !          1399: 
        !          1400:        if (AskUserForPoint(&x1, &y1, 0))
        !          1401:                return;
        !          1402:        if (AskUserForPoint(&x2, &y2, 1))
        !          1403:                return;
        !          1404:        ExThroughRectangle (x1_square_exed_through, y1_square_exed_through,
        !          1405:                x2_square_exed_through, y2_square_exed_through);
        !          1406:        PlusThroughRectangle (x1_square_plus_through, y1_square_plus_through,
        !          1407:                x2_square_plus_through, y2_square_plus_through);
        !          1408: 
        !          1409:        dx = x2 - x1;
        !          1410:        dy = y2 - y1;
        !          1411:        rad = sqrt((double)(dx * dx + dy * dy)) + 0.5;
        !          1412:        if (filled)
        !          1413:                for (i = 0; i <= (int)rad; i++) {
        !          1414:                        x = sqrt(rad * rad - i * i);
        !          1415:                        for (j = x1 - x; j <= x1 + x; j++) {
        !          1416:                                CheckSetRasterBit(raster, j, y1 - i, 1);
        !          1417:                                CheckSetRasterBit(raster, j, y1 + i, 1);
        !          1418:                        }
        !          1419:                }
        !          1420:        else {
        !          1421:                half = rad * sqrt(2.0)/2;
        !          1422:                for (i = 0; i <= (int)half; i++) {
        !          1423:                        x = sqrt(rad * rad - i * i);
        !          1424:                        CheckSetRasterBit(raster, x1 - x, y1 - i, 1);
        !          1425:                        CheckSetRasterBit(raster, x1 - x, y1 + i, 1);
        !          1426:                        CheckSetRasterBit(raster, x1 + x, y1 - i, 1);
        !          1427:                        CheckSetRasterBit(raster, x1 + x, y1 + i, 1);
        !          1428:                        CheckSetRasterBit(raster, x1 - i, y1 - x, 1);
        !          1429:                        CheckSetRasterBit(raster, x1 - i, y1 + x, 1);
        !          1430:                        CheckSetRasterBit(raster, x1 + i, y1 - x, 1);
        !          1431:                        CheckSetRasterBit(raster, x1 + i, y1 + x, 1);
        !          1432:                }
        !          1433:        }
        !          1434:        RefillGridPartially(x1-(int)rad, y1-(int)rad, 
        !          1435:                x1+(int)rad, y1+(int)rad, FALSE);
        !          1436:        changed = TRUE;
        !          1437:        x1_square_exed_through = y1_square_exed_through = OUT_OF_RANGE;
        !          1438:        x2_square_exed_through = y2_square_exed_through = OUT_OF_RANGE;
        !          1439:        x1_square_plus_through = y1_square_plus_through = OUT_OF_RANGE;
        !          1440:        x2_square_plus_through = y2_square_plus_through = OUT_OF_RANGE;
        !          1441:        RepaintRaster();
        !          1442:        RepaintRasterInverted();
        !          1443: }
        !          1444: 
        !          1445: ClearHotSpot() {
        !          1446:     if (x_hot_spot == OUT_OF_RANGE)
        !          1447:        return;
        !          1448:     HighlightHotSpot();  /* UNhighlight existing hot spot */
        !          1449:     x_hot_spot = y_hot_spot = OUT_OF_RANGE;
        !          1450:     changed = TRUE;
        !          1451:     }
        !          1452: 
        !          1453: SetHotSpot() {
        !          1454:     XCompressEvents();
        !          1455:     XDefineCursor (outer_window, dot);
        !          1456:     XSelectInput (outer_window, ButtonPressed | ButtonReleased | ExposeWindow);
        !          1457:        /* so that we can detect button pressed outside grid */
        !          1458: 
        !          1459:     while (TRUE) {
        !          1460:        XEvent event;
        !          1461:        int x1, y1;
        !          1462:        XNextEvent (&event);
        !          1463:        switch (event.type) {
        !          1464: 
        !          1465:            case ButtonPressed:
        !          1466:            case MouseMoved:
        !          1467:                if ((event.window == grid_window)
        !          1468:                && !WhatSquare (&event, &x1, &y1)
        !          1469:                && (x_hot_spot != x1 || y_hot_spot != y1)) {
        !          1470: 
        !          1471:                    /* UNhighlight old hot spot */
        !          1472:                    if (x_hot_spot != OUT_OF_RANGE)
        !          1473:                        HighlightHotSpot();  
        !          1474: 
        !          1475:                    x_hot_spot = x1;
        !          1476:                    y_hot_spot = y1;
        !          1477: 
        !          1478:                    /* highlight new hot spot */
        !          1479:                    HighlightHotSpot();
        !          1480: 
        !          1481:                    changed = TRUE;
        !          1482:                    }
        !          1483:                break;  /* keep looping until button is released */
        !          1484: 
        !          1485:            case ButtonReleased:
        !          1486:                XExpandEvents();
        !          1487:                XDefineCursor (outer_window, cross);
        !          1488:                XSelectInput (outer_window, ExposeWindow);
        !          1489:                return;
        !          1490: 
        !          1491:            case ExposeWindow:
        !          1492:            case ExposeRegion:
        !          1493:                ProcessEvent (&event);
        !          1494:                break;
        !          1495:                
        !          1496:            default:
        !          1497:                break;  /* just throw it away */
        !          1498:                
        !          1499:            }
        !          1500:        }
        !          1501:     }
        !          1502: 
        !          1503: RepaintRectangles (x1, y1, x2, y2, x3, y3)
        !          1504:     int x1, y1; /* first rectangle's top & left */
        !          1505:     int x2, y2; /* first rectangle's bottom & right */
        !          1506:     int x3, y3; /* second rectangle's top & left */
        !          1507:     {
        !          1508:     int x4 = x3 + (x2 - x1);  /* second rectangle's right edge */
        !          1509:     int y4 = y3 + (y2 - y1);  /* second rectangle's bottom edge */
        !          1510: 
        !          1511:     if (x4 >= squares_wide) x4 = squares_wide-1;
        !          1512:     if (y4 >= squares_wide) y4 = squares_high-1;
        !          1513: 
        !          1514:     /* if first rectangle is right of second, swap "first" and "second" variables */
        !          1515:     if (x1 > x3)
        !          1516:        {int temp;
        !          1517: #define swap(a,b) {temp = a; a = b; b = temp;}
        !          1518:        swap (x1, x3); swap (y1, y3); swap (x2, x4); swap (y2, y4);
        !          1519: #undef swap
        !          1520:        }
        !          1521:     
        !          1522:     RefillGridPartially (x1, y1, x2, y2, TRUE);
        !          1523: 
        !          1524:     if ((x3 > x2) || (max (y1, y3) > min (y2, y4)))
        !          1525:        /* rectangles don't overlap */
        !          1526:        RefillGridPartially (x3, y3, x4, y4, TRUE);
        !          1527: 
        !          1528:     else if (y1 < y3) {
        !          1529:        /* second rectangle is below & right of first */
        !          1530:        RefillGridPartially (x2+1, y3, x4, y2, TRUE);
        !          1531:        RefillGridPartially (x3, y2+1, x4, y4, TRUE);
        !          1532:        }
        !          1533: 
        !          1534:     else {
        !          1535:        /* second rectangle is above & right of first */
        !          1536:        RefillGridPartially (x3, y3, x4, y1-1, TRUE);
        !          1537:        RefillGridPartially (x2+1, y1, x4, y4, TRUE);
        !          1538:        }
        !          1539:     }
        !          1540: 
        !          1541: 
        !          1542: /* AskUserForArea returns FALSE if the user has defined a valid area,
        !          1543:    TRUE if the user hasn't (e.g. by clicking outside grid) */
        !          1544: 
        !          1545: boolean AskUserForArea(px1, py1, px2, py2) 
        !          1546:   int *px1, *py1, *px2, *py2;
        !          1547:   {
        !          1548:   XEvent event;
        !          1549:   int x1, y1, x2, y2;
        !          1550:   boolean result;
        !          1551: 
        !          1552:   XSelectInput (outer_window, ButtonPressed | ExposeWindow);
        !          1553:     /* so that we can detect button pressed outside grid */
        !          1554: 
        !          1555:   XDefineCursor (outer_window, upper_left);
        !          1556:   
        !          1557:   while (TRUE) {
        !          1558:     XNextEvent (&event);
        !          1559:     switch (event.type) {
        !          1560:       case ButtonPressed:
        !          1561:         if ((event.window != grid_window)
        !          1562:        || WhatSquare (&event, &x1, &y1)) {
        !          1563:           XDefineCursor (outer_window, cross);
        !          1564:          XSelectInput (outer_window, ExposeWindow);
        !          1565:           return (TRUE);
        !          1566:          }
        !          1567:        goto out1;  /* get out of the loop */
        !          1568:       case ExposeWindow:
        !          1569:       case ExposeRegion:
        !          1570:        ProcessEvent (&event);
        !          1571:        break;
        !          1572:       default:
        !          1573:        break;  /* just throw it away */
        !          1574:       }
        !          1575:     }
        !          1576: 
        !          1577:   out1:
        !          1578:   XCompressEvents();  /* DO collapse consecutive MouseMoved events */
        !          1579:   ExThroughSquare (x1, y1);
        !          1580:   FlushLineBuffer();
        !          1581:   x1_square_exed_through = x2_square_exed_through = x2 = x1;
        !          1582:   y1_square_exed_through = y2_square_exed_through = y2 = y1;
        !          1583:   XDefineCursor (outer_window, lower_right);
        !          1584:   
        !          1585:   while (TRUE) {
        !          1586:     XNextEvent (&event);
        !          1587:     switch (event.type) {
        !          1588:       case ButtonPressed:
        !          1589:        result = TRUE;
        !          1590:        goto out2;
        !          1591:       case ExposeWindow:
        !          1592:       case ExposeRegion:
        !          1593:        ProcessEvent (&event);
        !          1594:        break;
        !          1595:       case MouseMoved:
        !          1596:       case ButtonReleased: {
        !          1597:         int x, y;
        !          1598:        result = (event.window != grid_window)
        !          1599:             || WhatSquare (&event, &x, &y)  /* mouse outside grid? */
        !          1600:            || (x < x1) || (y < y1);
        !          1601:        if (result) {
        !          1602:          ExThroughRectangle (x1+1, y1, x2, y2);
        !          1603:          ExThroughRectangle (x1, y1+1, x1, y2);
        !          1604:          x2 = x2_square_exed_through = x1;
        !          1605:          y2 = y2_square_exed_through = y1;
        !          1606:          }
        !          1607:        else if ((x == x2) && (y == y2))
        !          1608:          ; /* both dimensions the same; do nothing */
        !          1609:        else if ((x > x2) == (y > y2)) {
        !          1610:          /* both dimensions bigger or smaller */
        !          1611:          ExThroughRectangle (min(x2,x)+1, y1, max(x2,x), max(y2,y));
        !          1612:          ExThroughRectangle (x1, min(y2,y)+1, min(x2,x), max(y2,y));
        !          1613:          x2 = x2_square_exed_through = x;
        !          1614:          y2 = y2_square_exed_through = y;
        !          1615:          }
        !          1616:         else {
        !          1617:          /* one dimension bigger, the other smaller */
        !          1618:          ExThroughRectangle (min(x2,x)+1, y1, max(x2,x), min(y2,y));
        !          1619:          ExThroughRectangle (x1, min(y2,y)+1, min(x2,x), max(y2,y));
        !          1620:          x2 = x2_square_exed_through = x;
        !          1621:          y2 = y2_square_exed_through = y;
        !          1622:          }
        !          1623:        if (event.type == ButtonReleased)
        !          1624:          goto out2;
        !          1625:        break;
        !          1626:        }
        !          1627:       default:
        !          1628:        break;  /* just throw it away */
        !          1629:       }
        !          1630:     }
        !          1631: 
        !          1632:   out2:
        !          1633:   XSelectInput (outer_window, ExposeWindow);
        !          1634:   XDefineCursor (outer_window, cross);
        !          1635:   if (result) {
        !          1636:     /* no area properly selected; remove X-outs from display */
        !          1637:     ExThroughRectangle (x1, y1, x2, y2);
        !          1638:     x1_square_exed_through = y1_square_exed_through = OUT_OF_RANGE;
        !          1639:     x2_square_exed_through = y2_square_exed_through = OUT_OF_RANGE;
        !          1640:     }
        !          1641:   else {
        !          1642:     *px1 = x1;
        !          1643:     *px2 = x2;
        !          1644:     *py1 = y1;
        !          1645:     *py2 = y2;
        !          1646:     }
        !          1647:   XExpandEvents();
        !          1648:   return (result);
        !          1649:   }  /* end of AskUserForArea procedure */
        !          1650: 
        !          1651: boolean AskUserForDest (px1, py1, width, height)
        !          1652:   int *px1, *py1;
        !          1653:   int width, height;
        !          1654:   {
        !          1655:   XEvent event;
        !          1656:   boolean result;
        !          1657:   XCompressEvents();  /* DO collapse consecutive MouseMoved events */
        !          1658:   XSelectInput (outer_window, ButtonPressed | ButtonReleased | ExposeWindow);
        !          1659:     /* so we can detect button action outside grid */
        !          1660:   XDefineCursor (outer_window, upper_left);
        !          1661: 
        !          1662:   while (TRUE) {
        !          1663:     XNextEvent (&event);
        !          1664:     switch (event.type) {
        !          1665: 
        !          1666:       case ExposeWindow:
        !          1667:       case ExposeRegion:
        !          1668:         ProcessEvent (&event);
        !          1669:        break;
        !          1670: 
        !          1671:       case ButtonPressed:
        !          1672:       case MouseMoved: {
        !          1673:        int x1_new, y1_new;
        !          1674:        boolean this_window = (event.window == grid_window)
        !          1675:          && !WhatSquare (&event, &x1_new, &y1_new);
        !          1676: 
        !          1677:        if (this_window && (x1_new == *px1) && (y1_new == *py1))
        !          1678:          break;  /* mouse is still in same square as before; do nothing */
        !          1679: 
        !          1680:         if (x1_square_plus_through != OUT_OF_RANGE)
        !          1681:           PlusThroughRectangle (x1_square_plus_through, y1_square_plus_through,
        !          1682:               x2_square_plus_through, y2_square_plus_through);
        !          1683: 
        !          1684:        if (this_window) {
        !          1685:          *px1 = x1_square_plus_through = x1_new;
        !          1686:          *py1 = y1_square_plus_through = y1_new;
        !          1687:          x2_square_plus_through = min (x1_new + width, squares_wide) - 1;
        !          1688:          y2_square_plus_through = min (y1_new + height, squares_high) - 1;
        !          1689:           PlusThroughRectangle (x1_square_plus_through, y1_square_plus_through,
        !          1690:               x2_square_plus_through, y2_square_plus_through);
        !          1691:          }
        !          1692:        else {
        !          1693:           x1_square_plus_through = y1_square_plus_through = OUT_OF_RANGE;
        !          1694:           x2_square_plus_through = y2_square_plus_through = OUT_OF_RANGE;
        !          1695:          *px1 = *py1 = OUT_OF_RANGE;
        !          1696:          }
        !          1697:         break;
        !          1698:        }
        !          1699: 
        !          1700:       case ButtonReleased: {
        !          1701:         result = (event.window != grid_window)
        !          1702:           || WhatSquare (&event, px1, py1);
        !          1703:        goto out;
        !          1704:        }
        !          1705: 
        !          1706:       default:
        !          1707:         break;  /* throw it away */
        !          1708:       }
        !          1709:     }
        !          1710: 
        !          1711:     out:
        !          1712:     if (result) {
        !          1713:       /* button released outside grid */
        !          1714:       if (x1_square_plus_through != OUT_OF_RANGE)
        !          1715:         PlusThroughRectangle (x1_square_plus_through, y1_square_plus_through,
        !          1716:           x2_square_plus_through, y2_square_plus_through);
        !          1717:       x1_square_plus_through = y1_square_plus_through = OUT_OF_RANGE;
        !          1718:       x2_square_plus_through = y2_square_plus_through = OUT_OF_RANGE;
        !          1719:       }
        !          1720: 
        !          1721:     XExpandEvents();
        !          1722:     XSelectInput (outer_window, ExposeWindow);
        !          1723:     XDefineCursor (outer_window, cross);
        !          1724:     return (result);
        !          1725:     }  /* end of AskUserForDest procedure */
        !          1726: 
        !          1727: boolean AskUserForPoint (xp, yp, plus)
        !          1728:   int *xp, *yp;
        !          1729:   {
        !          1730:   XEvent event;
        !          1731:   boolean this_window;
        !          1732: 
        !          1733:   XCompressEvents();  /* DO collapse consecutive MouseMoved events */
        !          1734:   XSelectInput (outer_window, ButtonPressed | ExposeWindow);
        !          1735:     /* so we can detect button action outside grid */
        !          1736:   XDefineCursor (outer_window, dot);
        !          1737: 
        !          1738:   while (TRUE) {
        !          1739:     XNextEvent (&event);
        !          1740:     switch (event.type) {
        !          1741: 
        !          1742:       case ExposeWindow:
        !          1743:       case ExposeRegion:
        !          1744:         ProcessEvent (&event);
        !          1745:        break;
        !          1746: 
        !          1747:       case ButtonReleased:
        !          1748:        this_window = (event.window == grid_window)
        !          1749:                              && !WhatSquare (&event, xp, yp);
        !          1750:        if (this_window) {
        !          1751:                if (plus) {
        !          1752:                        PlusThroughRectangle (*xp, *yp, *xp, *yp);
        !          1753:                        x1_square_plus_through = x2_square_plus_through = *xp;
        !          1754:                        y1_square_plus_through = y2_square_plus_through = *yp;
        !          1755:                } else {
        !          1756:                        ExThroughRectangle (*xp, *yp, *xp, *yp);
        !          1757:                        x1_square_exed_through = x2_square_exed_through = *xp;
        !          1758:                        y1_square_exed_through = y2_square_exed_through = *yp;
        !          1759:                }
        !          1760:        }
        !          1761:        goto out;
        !          1762:         break;
        !          1763: 
        !          1764:       default:
        !          1765:         break;  /* throw it away */
        !          1766:       }
        !          1767:     }
        !          1768: 
        !          1769:     out:
        !          1770: 
        !          1771:     XExpandEvents();
        !          1772:     XSelectInput (outer_window, ExposeWindow);
        !          1773:     XDefineCursor (outer_window, cross);
        !          1774:     return (!this_window);
        !          1775:     }
        !          1776: 
        !          1777: DialogInputHandler (event)
        !          1778:   XEvent *event;
        !          1779:   {
        !          1780:   switch (event->type) {
        !          1781:     case ExposeWindow:
        !          1782:     case ExposeRegion:
        !          1783:       ProcessEvent (event);
        !          1784:     }
        !          1785:   }
        !          1786: 
        !          1787: enum output_error {e_rename, e_write};
        !          1788: 
        !          1789: /* WriteOutput returns TRUE if output successfully written, FALSE if not */
        !          1790: 
        !          1791: WriteOutput() {
        !          1792:   FILE *file;
        !          1793:   if (!changed)
        !          1794:     return (TRUE);
        !          1795:   if (rename (filename, backup_filename) && errno != ENOENT)
        !          1796:     return (HandleOutputError(e_rename));
        !          1797:   file = fopen (filename, "w+");
        !          1798:   if (!file)
        !          1799:     return (HandleOutputError(e_write));
        !          1800:   WriteOutputToFile (file);
        !          1801:   fclose (file);
        !          1802:   changed = FALSE;
        !          1803:   return (TRUE);
        !          1804:   }
        !          1805: 
        !          1806: 
        !          1807: /* HandleOutputError returns TRUE if alternate file written, FALSE if not */
        !          1808: 
        !          1809: int HandleOutputError(e)
        !          1810:   enum output_error e;
        !          1811:   {
        !          1812:   int result;
        !          1813:   char *strings[2];
        !          1814:   char msg1[120], msg2[120];
        !          1815:   char *tmp_filename;
        !          1816:   if (e == e_rename)
        !          1817:     sprintf (msg1, "Can't rename %s to %s -- %s",
        !          1818:       filename, backup_filename, sys_errlist[errno]);
        !          1819:   else
        !          1820:     sprintf (msg1, "Can't write on file %s -- %s",
        !          1821:       filename, sys_errlist[errno]);
        !          1822:   tmp_filename = TmpFileName (filename);
        !          1823:   sprintf (msg2, "Should I write output to file %s?", tmp_filename);
        !          1824:   strings[0] = "Yes";
        !          1825:   strings[1] = "No";
        !          1826:   result = dialog (outer_window, font, fontInfo.height,
        !          1827:     msg1, msg2, strings, 2, DialogInputHandler);
        !          1828: 
        !          1829:   if (result == 0)  /* "yes" */ {
        !          1830:     filename = tmp_filename;
        !          1831:     free (backup_filename);
        !          1832:     backup_filename = BackupName (filename);
        !          1833:     return (WriteOutput());
        !          1834:     }
        !          1835:   else {  /* "no" */
        !          1836:     free (tmp_filename);
        !          1837:     return (FALSE);
        !          1838:     }
        !          1839:   }
        !          1840: 
        !          1841:     
        !          1842: Quit() {
        !          1843:   if (changed) {
        !          1844:     int result;
        !          1845:     char *strings[3];
        !          1846:     strings[0] = "Yes";
        !          1847:     strings[1] = "No";
        !          1848:     strings[2] = "Cancel";
        !          1849:     result = dialog (outer_window, font, fontInfo.height,
        !          1850:       "Save changes before quitting?", "", strings, 3, DialogInputHandler);
        !          1851:       
        !          1852:     switch (result) {
        !          1853:       case 0:     /* "yes" */
        !          1854:        if (WriteOutput())
        !          1855:          exit(0);
        !          1856:        else return;
        !          1857:       case 1:    /* "no" */
        !          1858:         exit(0);
        !          1859:       default:  /* "cancel" */
        !          1860:        return;
        !          1861:       }
        !          1862:     }
        !          1863: 
        !          1864:   exit(0);
        !          1865:   }
        !          1866: 
        !          1867: HighlightHotSpot() {
        !          1868:   /* Draw a diamond in the hot spot square */
        !          1869:   /* x1 and y1 are the center of the hot spot square */
        !          1870:   register int x1 = x_hot_spot*square_size + square_size/2;
        !          1871:   register int y1 = y_hot_spot*square_size + square_size/2;
        !          1872:   register int radius = square_size/6;
        !          1873:   register int i;
        !          1874:   Vertex v[5];
        !          1875:   v[0].x = v[2].x = v[4].x = x1;
        !          1876:   v[1].x = x1 + radius;
        !          1877:   v[3].x = x1 - radius;
        !          1878:   v[0].y = v[4].y = y1 + radius;
        !          1879:   v[1].y = v[3].y = y1;
        !          1880:   v[2].y = y1 - radius;
        !          1881:   for (i=0;i<5;i++)
        !          1882:     v[i].flags = 0;
        !          1883:   XDraw (grid_window, v, 5, 1, 1, 1, GXinvert, highlightplane);
        !          1884:   }
        !          1885: 
        !          1886: ExThroughRectangle (x1, y1, x2, y2)
        !          1887:   register int x1, y1, x2, y2;
        !          1888:   {
        !          1889:   register int x, y;
        !          1890:   for (x=x1;x<=x2;x++)
        !          1891:     for (y=y1;y<=y2;y++)
        !          1892:       ExThroughSquare (x, y);
        !          1893:   FlushLineBuffer();
        !          1894:   }
        !          1895: 
        !          1896: 
        !          1897: ExThroughSquare (x, y)
        !          1898:   register int x, y;
        !          1899:   {
        !          1900:   register int x1 = x*square_size;
        !          1901:   register int y1 = y*square_size;
        !          1902:   LineIntoBuffer (x1+1, y1+1,
        !          1903:     x1+square_size, y1+square_size);
        !          1904:   LineIntoBuffer (x1+square_size-1, y1+1,
        !          1905:     x1, y1+square_size);
        !          1906:   }
        !          1907: 
        !          1908: 
        !          1909: PlusThroughRectangle (x1, y1, x2, y2)
        !          1910:   register int x1, y1, x2, y2;
        !          1911:   {
        !          1912:   register int x, y;
        !          1913:   for (x=x1;x<=x2;x++)
        !          1914:     for (y=y1;y<=y2;y++)
        !          1915:       PlusThroughSquare (x, y);
        !          1916:   FlushLineBuffer();
        !          1917:   }
        !          1918: 
        !          1919: PlusThroughSquare (x, y)
        !          1920:   register int x, y;
        !          1921:   {
        !          1922:   register int x1 = x*square_size;
        !          1923:   register int y1 = y*square_size;
        !          1924:   LineIntoBuffer (x1+square_size/2, y1+1,
        !          1925:     x1+square_size/2, y1+square_size);
        !          1926:   LineIntoBuffer (x1+1, y1+square_size/2,
        !          1927:     x1+square_size, y1+square_size/2);
        !          1928:   }
        !          1929: 
        !          1930: 
        !          1931: #define BUFFER_MAXLENGTH 200  /* must be even */
        !          1932: static Vertex buffer [BUFFER_MAXLENGTH];
        !          1933: static int buffer_length = 0;
        !          1934: 
        !          1935: LineIntoBuffer (x1, y1, x2, y2) {
        !          1936:   buffer [buffer_length].x = x1;
        !          1937:   buffer [buffer_length].y = y1;
        !          1938:   buffer [buffer_length++].flags = VertexDontDraw;
        !          1939:   buffer [buffer_length].x = x2;
        !          1940:   buffer [buffer_length].y = y2;
        !          1941:   buffer [buffer_length++].flags = 0;
        !          1942:   if (buffer_length == BUFFER_MAXLENGTH)
        !          1943:     FlushLineBuffer();
        !          1944:   }
        !          1945:   
        !          1946: FlushLineBuffer () {
        !          1947:   XDraw (grid_window, buffer, buffer_length, 1, 1, 1, GXinvert,highlightplane);
        !          1948:   buffer_length = 0;
        !          1949:   }
        !          1950: 
        !          1951: #ifdef romp
        !          1952: /* 
        !          1953:  * prerelease IBM RT/PC software does not have ffs in its C library.
        !          1954:  * This code should be thrown away by summer, 1986.
        !          1955:  */
        !          1956: int ffs(i)
        !          1957:    int i;
        !          1958:    {
        !          1959:    int j = 1;
        !          1960:    if (i == 0) return (0);
        !          1961:    while (1) {
        !          1962:         if (i & 1) return (j);
        !          1963:         j++;
        !          1964:         i >>= 1;
        !          1965:         }
        !          1966:    }
        !          1967: 
        !          1968: #endif

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.