[comp.sources.x] v08i054: wscrawl, Part02/05

brianw@hpcvlx.cv.hp.com (Brian Wilson) (07/16/90)

Submitted-by: Brian Wilson <brianw@hpcvlx.cv.hp.com>
Posting-number: Volume 8, Issue 54
Archive-name: wscrawl/part02

#! /bin/sh
# This is a shell archive.  Remove anything before this line, then feed it
# into a shell via "sh file" or similar.  To overwrite existing files,
# type "sh file -c".
# The tool that generated this appeared in the comp.sources.unix newsgroup;
# send mail to comp-sources-unix@uunet.uu.net if you want that tool.
# If this archive is complete, you will see the following message at the end:
#		"End of archive 2 (of 5)."
# Contents:  wscrawl/README wscrawl/xac
# Wrapped by argv@turnpike on Sun Jul 15 11:47:10 1990
PATH=/bin:/usr/bin:/usr/ucb ; export PATH
if test -f 'wscrawl/README' -a "${1}" != "-c" ; then 
  echo shar: Will not clobber existing file \"'wscrawl/README'\"
else
echo shar: Extracting \"'wscrawl/README'\" \(3336 characters\)
sed "s/^X//" >'wscrawl/README' <<'END_OF_FILE'
X
X                               WSCRAWL 
X         (multi-display, multi-user, communications program)
X	          by Brian Wilson, brianw@cv.hp.com
X
X
XWhat "wscrawl" is:
X------------------
X
X    The program "wscrawl" is a demo of the networking capabilities of
XX-Windows.  The word "wscrawl" stands for "window-scrawl".  The user may 
Xthink of wscrawl as a paint program shared by any number of people at 
Xthe same time.  When wscrawl is run, it opens up a separate window on each 
Xparticipant's display.  From that point onward, each participant sees 
Xthe actions and ideas of every other participant as they occur. Each individual
Xmay simply watch, or participate at any moment.  Any individual may exit
Xout of the session at any time without affecting the other participants.  
X
X
X
XTo quickly get the concept of wscrawl:
X--------------------------------------
X
X     1) set your DISPLAY environment variable appropriately.  
X        (example for me at my workstation: "export DISPLAY=hpcvxbw:0")
X
X     2) have a friend (with X11 running) xhost your workstation
X	(example for my friend: "xhost +"  <-- xhost everyone in the world)
X
X     3) start a wscrawl session between the two of you
X	(example for me: "wscrawl -d hpcvxca:0" <-- my friend is hpcvxca:0)
X
X     4) push the left mouse button down and move the mouse.  For some menu
X	options, click on the menus at the top of the wscrawl window.
X
X     5) Concepts to try:
X
X	  a) Play tic-tac-toe between two workstations.  Change colors of pen 
X	     and draw the board in the center of the wscrawl window.  Each 
X	     player should use a different color.
X          
X          b) type a message to your friend.  This is done by selecting the
X	     "Type" item from the "Control" menu, then clicking the left
X	     button somewhere.  Type on the keyboard.  Try changine fonts.
X
X          c) Wscrawl to someone out of sight earshot.  Wscrawl to them,
X	     and then call them on the telephone.  By talking and scrawling
X	     diagrams, you can simulate being there with an pad and pencil.
X
X          d) resize your wscrawl window.  Notice that ALL wscrawl windows
X	     on ALL displays will resize accordingly.  This guarantees you
X	     are all seeing the same picture at all times.  Resize your window
X	     to the size of the display you are working at.
X
X          e) any person out of the total group may select "Close Window" from
X	     the "Control" menu at any time.  This change is duely noted in
X	     the "STATUS" window at the bottom of each wscrawl window.  The
X	     other people may continue to scrawl with no problems.
X
X	  f) scrawl a picture, then pull down the "Control" menu and select
X	     "Add a Display".  Type the name of your friend into the dialog
X	     box and hit return.  After a few seconds, your friend will see
X	     this new picture.
X
X          g) Fire up several wscrawl windows on your one display.  For me,
X	     I would type "wscrawl -d hpcvxbw:0 -d hpcvxbw:0 -d hpcvxbw:0".  
X	     See how it behaves.
X
X          h) Try some "File I/O".  Wscrawl saves and reads both standard
X	     X bitmap files and xwd standard files.  Import or export some
X	     of your favorite bitmaps, or dump a window using "xwd" and 
X	     load it into wscrawl.
X
X
XQuestions, comments, hacked source code: 
X----------------------------------------
XBrian Wilson
Xbrianw@cv.hp.com
X
END_OF_FILE
if test 3336 -ne `wc -c <'wscrawl/README'`; then
    echo shar: \"'wscrawl/README'\" unpacked with wrong size!
fi
# end of 'wscrawl/README'
fi
if test -f 'wscrawl/xac' -a "${1}" != "-c" ; then 
  echo shar: Will not clobber existing file \"'wscrawl/xac'\"
else
echo shar: Extracting \"'wscrawl/xac'\" \(37771 characters\)
sed "s/^X//" >'wscrawl/xac' <<'END_OF_FILE'
X	       "-adobe-courier-medium-r-normal--12-120-75-75-m-70-iso8859-1");
X    else if (strcmp(font_file_name, "courR14") == 0)
X	strcpy(xlfd_string, 
X	       "-adobe-courier-medium-r-normal--14-140-75-75-m-90-iso8859-1");
X    else if (strcmp(font_file_name, "timBI18") == 0)
X	strcpy(xlfd_string, 
X	       "-adobe-times-bold-i-normal--18-180-75-75-p-98-iso8859-1");
X    else if (strcmp(font_file_name, "courR24") == 0)
X	strcpy(xlfd_string, 
X	       "-adobe-courier-medium-r-normal--24-240-75-75-m-150-iso8859-1");
X    else if (strcmp(font_file_name, "timR24") == 0)
X	strcpy(xlfd_string, 
X	       "-adobe-times-medium-r-normal--24-240-75-75-p-124-iso8859-1");
X    else if (strcmp(font_file_name, "timBI24") == 0)
X	strcpy(xlfd_string, 
X	       "-adobe-times-bold-i-normal--24-240-75-75-p-128-iso8859-1");
X    else if (strcmp(font_file_name, "ncenR24") == 0)
X	strcpy(xlfd_string, 
X "-adobe-new century schoolbook-medium-r-normal--24-240-75-75-p-137-iso8859-1");
X    else if (strcmp(font_file_name, "vbee-36") == 0)
X	strcpy(xlfd_string, "vbee-36");
X    else if (strcmp(font_file_name, "vr-40") == 0)
X	strcpy(xlfd_string, "vr-40");
X    else if (strcmp(font_file_name, "vgl-40") == 0)
X	strcpy(xlfd_string, "vgl-40");
X    else if (strcmp(font_file_name, "vgl-40") == 0)
X	strcpy(xlfd_string, "vsgn-57");
X    else
X	return(FALSE);  /*not one of these fonts*/
X
X    if ((the_font_struct = XLoadQueryFont(disp_info[i].disp,
X          xlfd_string)) != NULL)
X    {
X        gcvalues.font = the_font_struct->fid;
X        disp_info[disp_num].the_font_struct = the_font_struct;
X        gcvalues.font = the_font_struct->fid;
X        XChangeGC(disp_info[i].disp, disp_info[disp_num].win_gc[i], GCFont, 
X                &gcvalues);
X        return(TRUE);
X     }
X     else 
X        return(FALSE);
X}
X
X
X/*
X * menu_selection - this function does the appropriate actions as specified
X *              by the menu referenced by the menu_num and item_num passed as
X *              parameters.
X */
Xmenu_selection(disp_num, menu_num, num_people_drawing, item_num)
Xint disp_num, menu_num, *num_people_drawing, item_num;
X{
X     int i, temp_num;
X     XColor scrn_def_ret, exact_def_ret;
X     XFontStruct *the_font_struct;
X     XWindowAttributes attr_ret;
X
X     switch (menu_num)
X     {
X	 case 0:                       /*Control menu*/
X	     switch (item_num)
X	     {
X		 case 1:	      /*scrawl*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         disp_info[disp_num].win_id,
X		         disp_info[disp_num].cursor_gc, 
X		         disp_info[disp_num].cur_pos.x, 
X		         disp_info[disp_num].cur_pos.y, 
X		         disp_info[disp_num].cur_pos.x + 
X		         disp_info[disp_num].prompt_width, 
X		         disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X 		     disp_info[disp_num].scrawl_mode = SCRAWLING;
X		     disp_info[disp_num].menu[menu_num].checked_item = 1;
X		     disp_info[disp_num].pen_width = 
X					 disp_info[disp_num].pen_widths.scrawl;
X		     for (i=1; i<disp_info[disp_num].menu[2].num_items; i++)
X		     {   /*determine which pen width we have*/
X	                 sscanf(menu_text[2][i], "%d", &temp_num);
X                         if (temp_num >= disp_info[disp_num].pen_width)
X			 {
X		             disp_info[disp_num].menu[2].checked_item=i;
X			     break;
X                         }
X		     }
X	             for (i=0; i<num_of_disps; i++)
X	             {
X			 if (disp_info[i].in_session_bool)
X			 {
X                             gcvalues.line_width =disp_info[disp_num].pen_width;
X	                     XChangeGC(disp_info[i].disp, 
X			           disp_info[disp_num].win_gc[i], 
X			           GCLineWidth, &gcvalues);
X			 }
X	             }
X		     XDefineCursor(disp_info[disp_num].disp,
X		        disp_info[disp_num].win_id,
X			XCreateFontCursor(disp_info[disp_num].disp, XC_dot));
X		     break;
X		 case 2:	      /*airbrush*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         disp_info[disp_num].win_id,
X		         disp_info[disp_num].cursor_gc, 
X		         disp_info[disp_num].cur_pos.x, 
X		         disp_info[disp_num].cur_pos.y, 
X		         disp_info[disp_num].cur_pos.x + 
X		         disp_info[disp_num].prompt_width, 
X		         disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     disp_info[disp_num].scrawl_mode = AIRBRUSHING;
X		     disp_info[disp_num].menu[menu_num].checked_item = 2;
X		     disp_info[disp_num].pen_width = 
X				       disp_info[disp_num].pen_widths.airbrush;
X		     for (i=1; i<disp_info[disp_num].menu[2].num_items; i++)
X		     {   /*determine which pen width we have*/
X	                 sscanf(menu_text[2][i], "%d", &temp_num);
X                         if (temp_num >= disp_info[disp_num].pen_width)
X			 {
X		             disp_info[disp_num].menu[2].checked_item=i;
X			     break;
X                         }
X		     }
X		     XDefineCursor(disp_info[disp_num].disp,
X		        disp_info[disp_num].win_id,
X			XCreateFontCursor(disp_info[disp_num].disp, 
X			XC_spraycan));
X		     break;
X		 case 3:	      /*type*/
X		     if (disp_info[disp_num].scrawl_mode != TYPING)
X                         (*num_people_drawing)++;  /*started input for now*/
X		     disp_info[disp_num].scrawl_mode = TYPING;
X		     disp_info[disp_num].menu[menu_num].checked_item = 3;
X		     XDefineCursor(disp_info[disp_num].disp,
X		        disp_info[disp_num].win_id,
X			XCreateFontCursor(disp_info[disp_num].disp, XC_xterm));
X		     break;
X		 case 4:	       /*eraser*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         disp_info[disp_num].win_id,
X		         disp_info[disp_num].cursor_gc, 
X		         disp_info[disp_num].cur_pos.x, 
X		         disp_info[disp_num].cur_pos.y, 
X		         disp_info[disp_num].cur_pos.x + 
X		         disp_info[disp_num].prompt_width, 
X		         disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     disp_info[disp_num].scrawl_mode = ERASING;
X		     disp_info[disp_num].menu[menu_num].checked_item = 4;
X		     disp_info[disp_num].pen_width = 
X				       disp_info[disp_num].pen_widths.eraser;
X		     for (i=1; i<disp_info[disp_num].menu[2].num_items; i++)
X		     {   /*determine which pen width we have*/
X	                 sscanf(menu_text[2][i], "%d", &temp_num);
X                         if (temp_num >= disp_info[disp_num].pen_width)
X			 {
X		             disp_info[disp_num].menu[2].checked_item=i;
X			     break;
X                         }
X		     }
X		     XDefineCursor(disp_info[disp_num].disp,
X		        disp_info[disp_num].win_id,
X			XCreateFontCursor(disp_info[disp_num].disp, 
X			XC_draped_box));
X		     break;
X		 case 5:	       /*shapes*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         disp_info[disp_num].win_id,
X		         disp_info[disp_num].cursor_gc, 
X		         disp_info[disp_num].cur_pos.x, 
X		         disp_info[disp_num].cur_pos.y, 
X		         disp_info[disp_num].cur_pos.x + 
X		         disp_info[disp_num].prompt_width, 
X		         disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     disp_info[disp_num].scrawl_mode = SELECTING_AN_AREA;
X		     disp_info[disp_num].menu[menu_num].checked_item = 5;
X		     disp_info[disp_num].pen_width = 
X					 disp_info[disp_num].pen_widths.shape;
X		     for (i=1; i<disp_info[disp_num].menu[2].num_items; i++)
X		     {   /*determine which pen width we have*/
X	                 sscanf(menu_text[2][i], "%d", &temp_num);
X                         if (temp_num >= disp_info[disp_num].pen_width)
X			 {
X		             disp_info[disp_num].menu[2].checked_item = i;
X			     break;
X                         }
X		     }
X		     disp_info[disp_num].dialog_what = DRAW_SHAPE;
X	             for (i=0; i<num_of_disps; i++)
X	             {
X			 if (disp_info[i].in_session_bool)
X			 {
X                             gcvalues.line_width =disp_info[disp_num].pen_width;
X	                     XChangeGC(disp_info[i].disp, 
X			           disp_info[disp_num].win_gc[i], 
X			           GCLineWidth, &gcvalues);
X			 }
X	             }
X		     XDefineCursor(disp_info[disp_num].disp,
X		        disp_info[disp_num].win_id,
X			XCreateFontCursor(disp_info[disp_num].disp, 
X			XC_crosshair));
X		     break;
X		 case 6:	       /*Rubber Pointer*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         disp_info[disp_num].win_id,
X		         disp_info[disp_num].cursor_gc, 
X		         disp_info[disp_num].cur_pos.x, 
X		         disp_info[disp_num].cur_pos.y, 
X		         disp_info[disp_num].cur_pos.x + 
X		         disp_info[disp_num].prompt_width, 
X		         disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X 		     disp_info[disp_num].scrawl_mode = RUBBER_POINTING;
X		     disp_info[disp_num].menu[menu_num].checked_item = 6;
X		     XDefineCursor(disp_info[disp_num].disp,
X		        disp_info[disp_num].win_id,
X			XCreateFontCursor(disp_info[disp_num].disp, XC_target));
X		     break;
X		 case 7:	       /*clear all windows*/
X	             for (i=0; i<num_of_disps; i++)
X	             {
X			 if (disp_info[i].in_session_bool)
X			 {
X			     XGetWindowAttributes(disp_info[i].disp, 
X				              disp_info[i].win_id, &attr_ret);
X			     XClearArea(disp_info[i].disp, 
X				    disp_info[i].win_id, 0, 0, attr_ret.width,
X				    attr_ret.height, False);
X                             XFlush(disp_info[i].disp);
X
X	                     if (disp_info[i].cursor_on == TRUE)
X	                     {/*we must turn it BACK ON, because XClear offed*/
X         	                 XDrawLine(disp_info[i].disp, 
X		                 disp_info[i].win_id,
X		                 disp_info[i].cursor_gc, 
X		                 disp_info[i].cur_pos.x, 
X		                 disp_info[i].cur_pos.y, 
X		                 disp_info[i].cur_pos.x + 
X		                 disp_info[i].prompt_width, 
X		                 disp_info[i].cur_pos.y);
X                             }
X			 }
X	             }
X		     NOTHING_DRAWN_YET = TRUE;
X		     break;
X		 case 8:	      /*dotted line (not a valid choice)*/
X		     break;
X		 case 9:	      /*Add Display*/
X		     disp_info[disp_num].previous_scrawl_mode = 
X				             disp_info[disp_num].scrawl_mode;
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         	disp_info[disp_num].win_id,
X		         	disp_info[disp_num].cursor_gc, 
X		         	disp_info[disp_num].cur_pos.x, 
X		         	disp_info[disp_num].cur_pos.y, 
X		         	disp_info[disp_num].cur_pos.x + 
X		         	disp_info[disp_num].prompt_width, 
X		         	disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     XDefineCursor(disp_info[disp_num].disp,
X		                   disp_info[disp_num].win_id,
X			           XCreateFontCursor(disp_info[disp_num].disp, 
X					  XC_cross));
X		     disp_info[disp_num].dialog_reply_index = 0;
X		     disp_info[disp_num].dialog_text_prompt = 
X					           "Enter Display To Add:";
X		     put_up_dialog(disp_num);
X		     disp_info[disp_num].scrawl_mode = RESPONDING_TO_DIALOG;
X		     disp_info[disp_num].dialog_what = ADD_A_DISPLAY;
X		     break;
X		 case 10:	      /*close window. Succeed from the session*/
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X                     XDestroyWindow(disp_info[disp_num].disp, 
X				    disp_info[disp_num].win_id);
X                     disp_info[disp_num].in_session_bool = FALSE; /*no window*/
X                     XCloseDisplay(disp_info[disp_num].disp);
X                     for (i=temp_num=0; i< num_of_disps; i++)
X                     {
X                        if (disp_info[i].in_session_bool)
X	                {
X	                    temp_num = TRUE;
X	                    break;
X	                }
X	             }
X	             if (temp_num)
X		     {
X                         for (i=0; i<num_of_disps; i++)
X	                 {
X	                     if (disp_info[i].in_session_bool)
X	                         place_and_draw_status_win(i);
X	                 }
X		     }
X		     else
X	                exit(0);/*if no one is still in session, stop running*/
X		     break;
X	     }
X	     break;
X	 case 1:                       /*PenColor menu*/
X	     strncpy(disp_info[disp_num].pen_color_str, menu_text[1][item_num],
X		     40);
X	     disp_info[disp_num].menu[menu_num].checked_item = item_num;
X	     for (i=0; i<num_of_disps; i++)
X	     {
X                 if (disp_info[i].in_session_bool)
X	         {
X                     if (!XAllocNamedColor(disp_info[i].disp, DefaultColormap(
X		         disp_info[i].disp, DefaultScreen(disp_info[i].disp)), 
X		         menu_text[1][item_num], &scrn_def_ret, &exact_def_ret))
X	             {
X		         continue;        /*sorry, no good. No colors left.*/
X	             }
X                     gcvalues.foreground = scrn_def_ret.pixel;
X		     XChangeGC(disp_info[i].disp, 
X		           disp_info[disp_num].win_gc[i], GCForeground, 
X		           &gcvalues);
X		 }
X	     }
X	     break;
X	 case 2:                       /*PenWidth menu*/
X	     sscanf(menu_text[2][item_num], "%d", &temp_num);
X	     disp_info[disp_num].pen_width = temp_num;
X	     disp_info[disp_num].menu[menu_num].checked_item = item_num;
X	     switch(disp_info[disp_num].scrawl_mode)
X	     {
X		 case SCRAWLING: 
X	             disp_info[disp_num].pen_widths.scrawl = temp_num;
X		     break;
X		 case AIRBRUSHING: 
X	             disp_info[disp_num].pen_widths.airbrush = temp_num;
X		     break;
X		 case ERASING: 
X	             disp_info[disp_num].pen_widths.eraser = temp_num;
X		     break;
X		 case SELECTING_AN_AREA: 
X	             disp_info[disp_num].pen_widths.shape = temp_num;
X		     break;
X		 default: 
X		     break;
X	     }
X		     
X	     for (i=0; i<num_of_disps; i++)
X	     {
X                 if (disp_info[i].in_session_bool)
X	         {
X                     gcvalues.line_width = temp_num;
X	             XChangeGC(disp_info[i].disp, 
X			   disp_info[disp_num].win_gc[i], 
X			   GCLineWidth, &gcvalues);
X		 }
X	     }
X	     break;
X	 case 3:                       /*PenCapStyle menu*/
X             switch(item_num)
X	     {
X		 case 1: temp_num = CapRound; break;
X		 case 2: temp_num = CapButt; break;
X		 case 3: temp_num = CapNotLast; break;
X		 case 4: temp_num = CapProjecting; break;
X	     }
X	     disp_info[disp_num].capstyle = temp_num;
X	     disp_info[disp_num].menu[menu_num].checked_item = item_num;
X	     for (i=0; i<num_of_disps; i++)
X	     {
X                 if (disp_info[i].in_session_bool)
X	         {
X                     gcvalues.cap_style = temp_num;
X	             XChangeGC(disp_info[i].disp, 
X			   disp_info[disp_num].win_gc[i], 
X			   GCCapStyle, &gcvalues);
X		 }
X	     }
X	     break;
X	 case 4:                       /*Font menu*/
X	     strncpy(disp_info[disp_num].font_str, menu_text[4][item_num],
X		     250);
X
X	     for (i=0; i<num_of_disps; i++)
X	     {
X                 if (disp_info[i].in_session_bool)
X	         {
X	             if ((the_font_struct=XLoadQueryFont(disp_info[i].disp,
X		              menu_text[4][item_num])) != NULL)
X	             {
X		             gcvalues.font = the_font_struct->fid;
X		             disp_info[disp_num].the_font_struct = 
X				                  the_font_struct;
X                             gcvalues.font = the_font_struct->fid;
X		             XChangeGC(disp_info[i].disp, 
X		                   disp_info[disp_num].win_gc[i], GCFont, 
X		                   &gcvalues);
X	             }
X		     else if (load_moron_fonts(disp_num, i, 
X				  menu_text[4][item_num]) == FALSE)
X		     {
X		         printf("Couldn't allocate font.\n");
X		     }
X		 }
X	     }
X
X	     if (disp_info[disp_num].cursor_on == TRUE)     /*turn it off   */
X	     {						    /*and don't tell*/
X                 XDrawLine(disp_info[disp_num].disp, 
X		     disp_info[disp_num].win_id,
X	             disp_info[disp_num].cursor_gc, 
X	             disp_info[disp_num].cur_pos.x, 
X	             disp_info[disp_num].cur_pos.y, 
X	             disp_info[disp_num].cur_pos.x + 
X	             disp_info[disp_num].prompt_width, 
X	             disp_info[disp_num].cur_pos.y);
X             }
X
X             disp_info[disp_num].prompt_width = 
X	               disp_info[disp_num].the_font_struct->max_bounds.width;
X	     disp_info[disp_num].char_height = 
X	           (disp_info[disp_num].the_font_struct)->max_bounds.ascent +
X	           (disp_info[disp_num].the_font_struct)->max_bounds.descent;
X	     disp_info[disp_num].type_history = NULL;
X	     disp_info[disp_num].menu[menu_num].checked_item = item_num;
X         
X	     if (disp_info[disp_num].cursor_on == TRUE)   /*it is really off*/
X	     {                                            /*'cause look above*/
X	         disp_info[disp_num].cursor_on = TRUE;
X                 XDrawLine(disp_info[disp_num].disp, 
X		     disp_info[disp_num].win_id,
X	             disp_info[disp_num].cursor_gc, 
X	             disp_info[disp_num].cur_pos.x, 
X	             disp_info[disp_num].cur_pos.y, 
X	             disp_info[disp_num].cur_pos.x + 
X	             disp_info[disp_num].prompt_width, 
X	             disp_info[disp_num].cur_pos.y);
X             }
X	     break;
X	 case 5:                       /*Shape menu*/
X	     menu_selection(disp_num, 0, num_people_drawing, 5);/*choose shape*/
X	     disp_info[disp_num].menu[menu_num].checked_item = item_num;
X             switch(item_num)
X	     {
X		 case 1: 
X		     disp_info[disp_num].current_shape = STRAIGHT_LINE;
X		     break;
X		 case 2: 
X		     disp_info[disp_num].current_shape = OUTLINE_RECT;
X		     break;
X		 case 3: 
X		     disp_info[disp_num].current_shape = FILLED_RECT;
X		     break;
X		 case 4: 
X		     disp_info[disp_num].current_shape = OUTLINE_OVAL;
X		     break;
X		 case 5: 
X		     disp_info[disp_num].current_shape = FILLED_OVAL;
X		     break;
X		 default: break;
X	     }
X	     break;
X	 case 6:                       /*File I/O menu*/
X             switch(item_num)
X	     {
X		 case 1:          /*Save Bitmap*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         	disp_info[disp_num].win_id,
X		         	disp_info[disp_num].cursor_gc, 
X		         	disp_info[disp_num].cur_pos.x, 
X		         	disp_info[disp_num].cur_pos.y, 
X		         	disp_info[disp_num].cur_pos.x + 
X		         	disp_info[disp_num].prompt_width, 
X		         	disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     disp_info[disp_num].previous_scrawl_mode = 
X		                            disp_info[disp_num].scrawl_mode;
X		     disp_info[disp_num].scrawl_mode = SELECTING_AN_AREA;
X		     disp_info[disp_num].dialog_what = SAVE_BITMAP;
X		     XDefineCursor(disp_info[disp_num].disp,
X		                   disp_info[disp_num].win_id,
X			           XCreateFontCursor(disp_info[disp_num].disp, 
X					  XC_crosshair));
X		     break;
X		 case 2:         /*Read In Bitmap*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         	disp_info[disp_num].win_id,
X		         	disp_info[disp_num].cursor_gc, 
X		         	disp_info[disp_num].cur_pos.x, 
X		         	disp_info[disp_num].cur_pos.y, 
X		         	disp_info[disp_num].cur_pos.x + 
X		         	disp_info[disp_num].prompt_width, 
X		         	disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     XDefineCursor(disp_info[disp_num].disp,
X		                   disp_info[disp_num].win_id,
X			           XCreateFontCursor(disp_info[disp_num].disp, 
X					  XC_cross));
X		     disp_info[disp_num].dialog_reply_index = 0;
X		     disp_info[disp_num].dialog_text_prompt = 
X					           "File To Read Bitmap From:";
X		     put_up_dialog(disp_num);
X		     disp_info[disp_num].previous_scrawl_mode = 
X		                            disp_info[disp_num].scrawl_mode;
X		     disp_info[disp_num].scrawl_mode = RESPONDING_TO_DIALOG;
X		     disp_info[disp_num].dialog_what = READ_IN_BITMAP;
X		     break;
X		 case 3:             /*Read Text File*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         	disp_info[disp_num].win_id,
X		         	disp_info[disp_num].cursor_gc, 
X		         	disp_info[disp_num].cur_pos.x, 
X		         	disp_info[disp_num].cur_pos.y, 
X		         	disp_info[disp_num].cur_pos.x + 
X		         	disp_info[disp_num].prompt_width, 
X		         	disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     XDefineCursor(disp_info[disp_num].disp,
X		                   disp_info[disp_num].win_id,
X			           XCreateFontCursor(disp_info[disp_num].disp, 
X					  XC_cross));
X		     disp_info[disp_num].dialog_reply_index = 0;
X		     disp_info[disp_num].dialog_text_prompt = 
X					           "File To Read Text From:";
X		     put_up_dialog(disp_num);
X		     disp_info[disp_num].previous_scrawl_mode = 
X		                            disp_info[disp_num].scrawl_mode;
X		     disp_info[disp_num].scrawl_mode = RESPONDING_TO_DIALOG;
X		     disp_info[disp_num].dialog_what = READ_TEXTFILE;
X		     break;
X		 case 4:          /*Save Image*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         	disp_info[disp_num].win_id,
X		         	disp_info[disp_num].cursor_gc, 
X		         	disp_info[disp_num].cur_pos.x, 
X		         	disp_info[disp_num].cur_pos.y, 
X		         	disp_info[disp_num].cur_pos.x + 
X		         	disp_info[disp_num].prompt_width, 
X		         	disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     disp_info[disp_num].previous_scrawl_mode = 
X		                            disp_info[disp_num].scrawl_mode;
X		     disp_info[disp_num].scrawl_mode = SELECTING_AN_AREA;
X		     disp_info[disp_num].dialog_what = SAVE_IMAGE;
X		     XDefineCursor(disp_info[disp_num].disp,
X		                   disp_info[disp_num].win_id,
X			           XCreateFontCursor(disp_info[disp_num].disp, 
X					  XC_crosshair));
X		     break;
X		 case 5:         /*Read In Image*/
X	             if (disp_info[disp_num].cursor_on == TRUE)
X	             {
X			 disp_info[disp_num].cursor_on = FALSE;
X         	         XDrawLine(disp_info[disp_num].disp, 
X		         	disp_info[disp_num].win_id,
X		         	disp_info[disp_num].cursor_gc, 
X		         	disp_info[disp_num].cur_pos.x, 
X		         	disp_info[disp_num].cur_pos.y, 
X		         	disp_info[disp_num].cur_pos.x + 
X		         	disp_info[disp_num].prompt_width, 
X		         	disp_info[disp_num].cur_pos.y);
X                     }
X		     if (disp_info[disp_num].scrawl_mode == TYPING)
X                         (*num_people_drawing)--;  /*stopped input for now*/
X		     XDefineCursor(disp_info[disp_num].disp,
X		                   disp_info[disp_num].win_id,
X			           XCreateFontCursor(disp_info[disp_num].disp, 
X					  XC_cross));
X		     disp_info[disp_num].dialog_reply_index = 0;
X		     disp_info[disp_num].dialog_text_prompt = 
X					           "File To Read Image From:";
X		     put_up_dialog(disp_num);
X		     disp_info[disp_num].previous_scrawl_mode = 
X		                            disp_info[disp_num].scrawl_mode;
X		     disp_info[disp_num].scrawl_mode = RESPONDING_TO_DIALOG;
X		     disp_info[disp_num].dialog_what = READ_IN_IMAGE;
X		     break;
X	     }
X	     break;
X	 default:
X	     printf("ERROR: Unknown menu number in menu_selection().\n");
X	     exit(0);
X	     break;
X     }
X}
X
X
X/*
X * draw_menu_text - this function draws the menu contents such as text 
X *              and checkmarks as specified by the display and the menu_num. 
X *              This is usually called in response to an expose event, etc.
X */
Xdraw_menu_text(disp_num, menu_num)
Xint disp_num, menu_num;
X{
X    int i;
X
X    /*
X     * draw the bar under the menu heading
X     */
X    XDrawLine(disp_info[disp_num].disp,
X	      disp_info[disp_num].menu[menu_num].win_id,
X	      disp_info[disp_num].fg_menu_gc, 0, MENU_ITEM_HEIGHT+1, 
X	      MENU_ITEM_WIDTH, MENU_ITEM_HEIGHT+1);
X
X    /*
X     * draw all the text
X     */
X    for (i=0; i<disp_info[disp_num].menu[menu_num].num_items; i++)
X    {
X        XDrawString(disp_info[disp_num].disp,
X	       disp_info[disp_num].menu[menu_num].win_id, 
X	       disp_info[disp_num].fg_menu_gc, 15, i*MENU_ITEM_HEIGHT + 18, 
X	       menu_text[menu_num][i], strlen(menu_text[menu_num][i]));
X    }
X
X    /*
X     * draw the check mark of the current menu as long as the menu has a check
X     */
X    if (menu_num != 6)
X    {
X        XDrawLine(disp_info[disp_num].disp,
X	      disp_info[disp_num].menu[menu_num].win_id,
X	      disp_info[disp_num].fg_menu_gc, 4, 
X	      disp_info[disp_num].menu[menu_num].checked_item * 
X	      MENU_ITEM_HEIGHT + 15, 
X	      7, disp_info[disp_num].menu[menu_num].checked_item * 
X	      MENU_ITEM_HEIGHT + 18);
X        XDrawLine(disp_info[disp_num].disp,
X	      disp_info[disp_num].menu[menu_num].win_id,
X	      disp_info[disp_num].fg_menu_gc, 7, 
X	      disp_info[disp_num].menu[menu_num].checked_item * 
X	      MENU_ITEM_HEIGHT + 18, 
X	      12, disp_info[disp_num].menu[menu_num].checked_item * 
X	      MENU_ITEM_HEIGHT + 9);
X    }
X    XFlush(disp_info[disp_num].disp);
X}
X
X
X/*
X * draw_menu - this function draws the menus on the display whose index is
X *              passed as the argument.  This function is invoked usually
X *              when someone has just boinked the menu, causing it to unroll
X *              out with all the choices.  It can also be invoked when someone
X *              has let up the mouse button, probably making a choice. 
X *              The "current_menu" field of the disp_info struct should be
X *              set to the menu which changed.
X */
Xdraw_menu(disp_num)
Xint disp_num;
X{
X    int menu_num;
X    XWindowChanges new_dim;
X
X    menu_num = disp_info[disp_num].current_menu;
X
X    if (disp_info[disp_num].pointer_state != IN_MENU)    /*retract menu*/
X    {
X        new_dim.height = MENU_ITEM_HEIGHT;
X    }
X    else                                                 /*extend a menu*/
X    {
X        new_dim.height = disp_info[disp_num].menu[menu_num].num_items * 
X		     MENU_ITEM_HEIGHT;
X    }
X    XConfigureWindow(disp_info[disp_num].disp, 
X	       disp_info[disp_num].menu[menu_num].win_id, CWHeight, &new_dim);
X
X    /*
X     * The following is commented out because an expose event will occur,
X     * causing the menu text to be displayed WITHOUT this line.  Why keep
X     * it here you ask me?  Well, I can't answer that; it would reveal the
X     * location of insidious bugs in certain HP systems.  :-)
X     */
X    /*
X    draw_menu_text(disp_num, menu_num);
X     */
X}
X
X
X/*
X * slide_rubberband_box - this function waves the fixed size rubberband
X *                  box all over the window, waiting for the user to
X *                  click the button, thus placing the bitmap or image on all
X *                  of the displays.
X */
Xslide_rubberband_box(disp_num, num_people_drawing)
Xint disp_num, *num_people_drawing;
X{
X    Window rr, cr;          /* <-- These here are good variables, buckwheat.*/
X    unsigned int mskr;
X    int rxr, ryr, win_x, win_y;
X    int top, left, width, height, old_top, old_left;
X
X    width = disp_info[disp_num].rubber_band_width;
X    height = disp_info[disp_num].rubber_band_height;
X
X    if (disp_info[disp_num].first_point_bool)
X    {  
X        XQueryPointer(disp_info[disp_num].disp, 
X			  disp_info[disp_num].win_id, &rr, &cr, &rxr, 
X			  &ryr, &win_x, &win_y,&mskr);
X        disp_info[disp_num].cur_pos.x = win_x;
X        disp_info[disp_num].cur_pos.y = win_y;
X        disp_info[disp_num].first_point_bool = FALSE;
X        (*num_people_drawing)++;     /*slide this around till you click*/
X
X	top = disp_info[disp_num].cur_pos.y - 
X	                disp_info[disp_num].rubber_band_height/2;
X	left = disp_info[disp_num].cur_pos.x - 
X	                disp_info[disp_num].rubber_band_width/2;
X
X	XDrawRectangle(disp_info[disp_num].disp, disp_info[disp_num].win_id,
X	       disp_info[disp_num].rubber_band_gc,
X	       left, top, width, height);
X        XFlush(disp_info[disp_num].disp);
X    }
X
X    XQueryPointer(disp_info[disp_num].disp, 
X			  disp_info[disp_num].win_id, &rr, &cr, &rxr, 
X			  &ryr, &win_x, &win_y,&mskr);
X    if ((disp_info[disp_num].cur_pos.x != win_x) ||
X        (disp_info[disp_num].cur_pos.y != win_y))
X    {
X	/*
X	 * erase the old rubber band box
X	 */
X	old_top = disp_info[disp_num].cur_pos.y - 
X	                disp_info[disp_num].rubber_band_height/2;
X	old_left = disp_info[disp_num].cur_pos.x - 
X	                disp_info[disp_num].rubber_band_width/2;
X	XDrawRectangle(disp_info[disp_num].disp, disp_info[disp_num].win_id,
X	       disp_info[disp_num].rubber_band_gc,
X	       old_left, old_top, width, height);
X	/*
X	 * draw the new rubber band box
X	 */
X        disp_info[disp_num].cur_pos.x = win_x;
X        disp_info[disp_num].cur_pos.y = win_y;
X	top = disp_info[disp_num].cur_pos.y - height/2;
X	left = disp_info[disp_num].cur_pos.x - width/2;
X	XDrawRectangle(disp_info[disp_num].disp, disp_info[disp_num].win_id,
X	       disp_info[disp_num].rubber_band_gc, left, top, width, height);
X    }
X}
X
X
X/*
X * type_on_dialog - this routine is called once a cycle if the user is 
X *                  currently supposed to be typing in the dialog box.
X *                  This function handles the text, etc.
X *                  Global index: disp_info[disp_num].dialog_reply_index
X */
Xtype_on_dialog(disp_num)
Xint disp_num;
X{
X    XEvent the_event;
X    KeySym ksr;
X    char the_char[256];
X    int index;
X
X    if (XPending(disp_info[disp_num].disp))   /*check if there is a KeyPress*/
X    {
X	XNextEvent(disp_info[disp_num].disp, &the_event);
X        if (the_event.type != KeyPress)
X	{
X            XPutBackEvent(disp_info[disp_num].disp, &the_event);
X	    return(0);
X	}
X	else    /*it IS a KeyPressedEvent*/
X	{
X	    index = disp_info[disp_num].dialog_reply_index;
X            the_char[0] = 0;
X            XLookupString((XKeyEvent *) (&the_event),the_char,10, &ksr, NULL);
X            if (the_char[0] == 0)
X                return(0);                /*it was a shift or some such*/
X            else if (the_char[0] == 8)  /*8 == Backspace*/
X            {
X		if (index != 0)
X		{
X		    index--;
X                    (disp_info[disp_num].dialog_reply_index)--;
X		     XClearArea(disp_info[disp_num].disp,
X	                    disp_info[disp_num].dialog_win_id, 
X			    14 + index*FIXED_CHAR_WIDTH, DIALOG_WIN_HEIGHT-27,
X			    FIXED_CHAR_WIDTH, 19, False);
X		}
X	    }
X            else if (the_char[0] == 13)   /*13 == Carriage Return*/
X            {
X	        /*unmap dialog and take action*/
X                XUnmapWindow(disp_info[disp_num].disp, 
X			   disp_info[disp_num].dialog_win_id);
X		XSync(disp_info[disp_num].disp, False);
X		disp_info[disp_num].dialog_text_return[index] = '\0';
X		parse_and_fix_dialog_text(disp_num);
X		do_dialog_action(disp_num);
X	    }
X            else
X	    {
X		XDrawString(disp_info[disp_num].disp, 
X			disp_info[disp_num].dialog_win_id,
X			disp_info[disp_num].fg_menu_gc, 
X			14 + index*FIXED_CHAR_WIDTH, DIALOG_WIN_HEIGHT - 12,
X			the_char, 1);
X		disp_info[disp_num].dialog_text_return[index] = the_char[0];
X		(disp_info[disp_num].dialog_reply_index)++;
X	    }
X        }
X    }
X}
X
X
X/*
X * parse_and_fix_dialog_text - this function parses the dialog text
X *                  and replaces anything like a tilda or etc with the
X *                  appropriate environment variable or string.
X */
Xparse_and_fix_dialog_text(disp_num)
Xint disp_num;
X{
X    char temp_str[512], temp_str2[512], current_line[512], login_name[512];
X    char temp_str3[512];
X    char *tmp_ptr;
X    int index, i, c, length_name;
X    FILE *fp;
X
X    strncpy(temp_str, disp_info[disp_num].dialog_text_return, 510);
X
X    for (index=0; temp_str[index] != '\0'; index++)
X	if (temp_str[index] != ' ')
X	    break;
X
X    if ((temp_str[index] == '~') && (temp_str[index+1] == '/'))
X    {
X	strncpy(temp_str2, getenv("HOME"), 250);
X	strncat(temp_str2, (char *)&(temp_str[index+1]), 250);
X	strncpy(disp_info[disp_num].dialog_text_return, temp_str2, 510);
X	disp_info[disp_num].dialog_reply_index = strlen(temp_str2);
X	return(1);
X    }
X    else if (temp_str[index] == '~')
X    {
X	for (i=index+1; (temp_str[i] != '/') && (temp_str[i] != '\0'); i++)
X            login_name[i-index-1] = temp_str[i];
X	login_name[i-index-1] = '\0';
X        length_name = strlen(login_name);
X
X        if ((fp = fopen("/etc/passwd", "r")) == NULL)
X        {
X            printf("ERROR: Could not open /etc/passwd.\n");
X	    return(0);
X	}
X	while ((c = getc(fp)) != EOF)
X	{
X	    ungetc(c, fp);
X	    fgets(current_line, 510, fp);       /*get the line*/
X	    strncpy(temp_str2, current_line, length_name);
X	    temp_str2[length_name] = '\0';
X            if (strcmp(temp_str2, login_name) == 0)    /*this is the line*/
X	    {
X                tmp_ptr = &strpbrk(&strpbrk(&strpbrk(&strpbrk(&strpbrk(
X		                   current_line, ":")[1], 
X				   ":")[1], ":")[1], ":")[1], ":")[1];
X
X		for (i=0; tmp_ptr[i] != ':'; i++)
X		     temp_str2[i] = tmp_ptr[i];
X		temp_str2[i] = '\0';
X
X	        sprintf(temp_str3, "%s%s", temp_str2, 
X			(char *)&(temp_str[index+1+length_name]));
X                fclose(fp);
X
X	        strncpy(disp_info[disp_num].dialog_text_return, temp_str3, 510);
X	        disp_info[disp_num].dialog_reply_index = strlen(temp_str3);
X                return(1);
X	    }
X	}
X        fclose(fp);
X	printf("ERROR: user specified was not found in /etc/passwd.\n");
X	return(0);
X    }
X}
X
X
X/*
X * type_on_screens - this function types on the screens of all the people
X *                   by reading if any keys have been depressed, and if so,
X *                   it puts that character up on all the displays currently
X *                   open.  <-- Darn smart, eh?  You probably couldn't have
X *                   figured out that on your own.  ;-)
X */
Xtype_on_screens(input_disp)
Xint input_disp;
X{
X    XEvent the_event;
X    Window rr, cr;                 /* <-- These here are variables, son.*/
X    KeySym ksr;
X    unsigned int mskr;
X    int i, rxr, ryr, win_x, win_y;
X    char the_char[256];
X
X    if (disp_info[input_disp].first_point_bool)
X    {
X        XQueryPointer(disp_info[input_disp].disp, 
X			  disp_info[input_disp].win_id, &rr, &cr, &rxr, 
X			  &ryr, &win_x, &win_y,&mskr);
X	disp_info[input_disp].orig_x = win_x;
X        disp_info[input_disp].cur_pos.x = win_x;
X        disp_info[input_disp].cur_pos.y = win_y;
X        disp_info[input_disp].first_point_bool = FALSE;
X        disp_info[input_disp].prompt_width = 
X	         disp_info[input_disp].the_font_struct->max_bounds.width;
X	disp_info[input_disp].char_height = 
X	   (disp_info[input_disp].the_font_struct)->max_bounds.ascent +
X	   (disp_info[input_disp].the_font_struct)->max_bounds.descent;
X	disp_info[input_disp].type_history = NULL;
X
X	/*draw the cursor*/
X	disp_info[input_disp].cursor_on = TRUE;
X	XDrawLine(disp_info[input_disp].disp, disp_info[input_disp].win_id,
X		  disp_info[input_disp].cursor_gc, 
X		  disp_info[input_disp].cur_pos.x, 
X		  disp_info[input_disp].cur_pos.y, 
X		  disp_info[input_disp].cur_pos.x + 
X		  disp_info[input_disp].prompt_width, 
X		  disp_info[input_disp].cur_pos.y);
X    }
X
X    if (XPending(disp_info[input_disp].disp))  /*check if there is a KeyPress*/
X    {
X	XNextEvent(disp_info[input_disp].disp, &the_event);
X        if (the_event.type != KeyPress)
X	{
X            XPutBackEvent(disp_info[input_disp].disp, &the_event);
X	    return(0);
X	}
X	else    /*it IS a KeyPressedEvent*/
X	{
X            the_char[0] = 0;
X            XLookupString((XKeyEvent *) (&the_event),the_char,10, &ksr, NULL);
X            if (the_char[0] == 0)
X                return(0);                /*it was a shift or some such*/
X            if (the_char[0] == 13)        /*13 == Carriage Return*/
X            {
X	        /*erase the cursor*/
X	        disp_info[input_disp].cursor_on = FALSE;
X	        XDrawLine(disp_info[input_disp].disp, 
X		  disp_info[input_disp].win_id,
X		  disp_info[input_disp].cursor_gc, 
X		  disp_info[input_disp].cur_pos.x, 
X		  disp_info[input_disp].cur_pos.y, 
X		  disp_info[input_disp].cur_pos.x + 
END_OF_FILE
if test 37771 -ne `wc -c <'wscrawl/xac'`; then
    echo shar: \"'wscrawl/xac'\" unpacked with wrong size!
fi
# end of 'wscrawl/xac'
fi
echo shar: End of archive 2 \(of 5\).
cp /dev/null ark2isdone
MISSING=""
for I in 1 2 3 4 5 ; do
    if test ! -f ark${I}isdone ; then
	MISSING="${MISSING} ${I}"
    fi
done
if test "${MISSING}" = "" ; then
    echo You have unpacked all 5 archives.
    rm -f ark[1-9]isdone
else
    echo You still need to unpack the following archives:
    echo "        " ${MISSING}
fi
##  End of shell archive.
exit 0
dan
----------------------------------------------------
O'Reilly && Associates   argv@sun.com / argv@ora.com
Opinions expressed reflect those of the author only.