home *** CD-ROM | disk | FTP | other *** search
- /**
- *
- * SAVESCN.C Install a resident program to save the screen image
- * when a "hot key" is pressed.
- *
- * This program installs several interrupt service routines to
- * establish a "hot key" function. When the specified "hot key" is
- * pressed, a window pops up and requests the name of a file.
- * Then the contents of the active video page are saved onto that
- * file. (After each line of text from the screen, a carriage return
- * and a line feed are saved in the file.)
- *
- * Several command line options are supported. To select a hot key,
- * use the -c option for ASCII characters:
- *
- * savescn -cZ (sets capital Z as the hot key)
- *
- * or use the -k option for the extended codes of IBM extended key
- * sequences:
- *
- * savescn -k71 (sets "Home" as the hot key)
- *
- * (The default hot key is Alt-2).
- *
- * To remove a previously-installed copy of SAVESCN, use the -r
- * option:
- *
- * savescn -r
- *
- * (SAVESCN will not permit itself to be loaded as a resident program
- * twice in succession. However, it may be loaded twice if its
- * previous interrupt handlers are concealed by an intervening
- * resident program. SAVESCN 3.02 will not recognize other versions
- * of SAVESCN.)
- *
- * To make SAVESCN pause before it terminates and stays resident, use
- * the -w option:
- *
- * savescn -w
- *
- * (If Ctrl-Break is pressed while SAVESCN is pausing, it will remove
- * its ISRs and abort, leaving no residue.)
- *
- * Any of these command-line options may be combined.
- *
- * General method: ISRs are installed for the keystroke (9), clock tick
- * (8), and DOS idle (0x28) interrupts. The keystroke handler watches
- * for the designated hot key; if the hot key is found, it is removed
- * from the keyboard buffer and the global flag "want_to_save" is set.
- * The clock tick and DOS idle handlers check the flag and perform a
- * screen save if possible.
- *
- * Version 3.00 (C)Copyright Blaise Computing Inc. 1986
- *
- * Version 3.02 April 9, 1987
- * Modified calculation in emptyq() to avoid MS C 4.00
- * H model compiler bug.
- * Corrected location of call to FLCLOSE in write_file().
- * Corrected last row read from screen in write_file().
- * Corrected recognition of hot keys to allow any
- * combination of ASCII code and key code.
- * Simplified some items:
- * moved usage and Ctrl-C messages;
- * made ISR stacks and screen row buffer static;
- * removed get_fname();
- * used APPROMPT for wait message;
- * moved a few comments.
- *
- **/
-
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
-
- #include <bapplic.h>
- #include <bfile.h>
- #include <bgenvid.h>
- #include <bkeybd.h>
- #include <bisr.h>
- #include <bprogctl.h>
- #include <bstring.h>
- #include <bwindow.h>
-
- #if MSC300
- #include <malloc.h>
- #include <memory.h>
- #endif
-
- #define TIMER_VEC 0x08 /* Interrupt types */
- #define KEY_VEC 0x09
- #define CBREAK_VEC 0x23
- #define IDLE_VEC 0x28
-
- #define ENTER_KEY 13
-
- #define STKSIZE 1500 /* Size of one ISR stack */
- #define NUM_STACKS 2 /* Maximum depth of keystroke, */
- /* timer, and INT 28H ISR calls.*/
-
- #define OK 0 /* Error return codes from */
- #define NO_MEMORY 100 /* main program. */
- #define WINDOW_ERROR 101
- #define NO_FILENAME 102
- #define TRY_NEW_FILENAME 103
- #define TARGET_FULL 104
- #define CANT_REMOVE 105
- #define USAGE_ERROR 106
- #define FALL_OUT 107
-
- int want_to_save = 0; /* Internal flag set by */
- /* keystroke ISR: nonzero if */
- /* hot key pressed. This flag */
- /* is cleared by do_save(). */
-
- char hot_ch = '\0'; /* Initial hot key = */
- char hot_code = 121; /* ALT-2 */
-
- int spec_char = 1; /* ASCII character is */
- /* significant in identifying */
- /* hot key. */
- /* */
- int spec_code = 1; /* Key code is significant in */
- /* identifying hot key. */
-
- ADS key_prev = {0,0}, /* Previous contents of */
- tick_prev = {0,0}, /* interrupt vectors. */
- idle_prev = {0,0};
-
- ADS dos_flag_ads = {0,0}; /* Address of DOS critical */
- /* section flag. */
-
- char scn_buffer[PC_COLS]; /* Buffers used by write_file().*/
- ADS scn_buf_ads = {0,0};
- char crlf[] = "\15\12";
- ADS crlf_ads = {0,0};
-
- char filename[70] = {'\0'}; /* Name of file on which to */
- /* save screen image. */
- int name_err = OK; /* Errors encountered by ISRs. */
- int file_err = OK;
-
- char *pspare_mem = NIL; /* Spare memory in local */
- /* dynamic memory pool for ISR */
- /* window routines. */
- int have_spare_mem = 0; /* Nonzero if spare memory has */
- /* been allocated but not freed*/
-
- BWINDOW *pwindow = (BWINDOW *) 0; /* Window used to collect */
- /* name of file. */
- int window_displayed = 0; /* Variables to regulate */
- int cursor_was_off = 0, /* display and removal of that */
- high = 0, /* window. */
- low = 0,
- old_row = 0,
- old_col = 0;
-
- typedef struct /* KEYPAIR: Character and key */
- { /* code as stored in BIOS */
- char ch; /* keyboard buffer. */
- char keycode;
- } KEYPAIR;
-
- void cbreak (ALLREG *,ISRCTRL *,ISRMSG *); /* The ISRs. */
- void keystroke(ALLREG *,ISRCTRL *,ISRMSG *);
- void tick (ALLREG *,ISRCTRL *,ISRMSG *);
- void idle (ALLREG *,ISRCTRL *,ISRMSG *);
-
- int emptyq(KEYPAIR *); /* Internal (see below). */
- int dos_ready(void);
- void do_save(void);
- int write_file(char *);
- int display_window(void);
- void remove_window(void);
- int prompt(char *,char *,int);
- int warning(char *,int);
-
- int main(argc,argv)
- int argc;
- char *argv[];
- {
- ISRCTRL cbrk_ctl,key_ctl,tick_ctl,idle_ctl;
- static char cbrk_stack[STKSIZE],
- key_stack [STKSIZE * NUM_STACKS],
- tick_stack[STKSIZE * NUM_STACKS],
- idle_stack[STKSIZE * NUM_STACKS];
- char dummy[7];
- int ercode,i;
- DOSREG regs;
-
- int bad_arg,have_char,have_code,wait;
- int want_remove,removed;
-
- static char cbrk_ident[] = "SAVESCN3.02CBRK";
- static char key_ident[] = "SAVESCN3.02 KEY";
- static char tick_ident[] = "SAVESCN3.02TICK";
- static char idle_ident[] = "SAVESCN3.02IDLE";
- unsigned keypsp, tickpsp, idlepsp;
-
- /* First analyze command-line arguments. */
-
- want_remove = 0;
- have_char = 0;
- have_code = 0;
- wait = 0;
- for (bad_arg = 0,i = 1; bad_arg == 0 && i < argc; i++)
- {
- if (*argv[i]++ == '-')
- {
- switch (tolower((int) *argv[i]++))
- {
- case 'c': /* Hot key is ASCII character: */
- /* character following "-c" is */
- /* the character. */
- if ( have_char
- || 1 != strlen(argv[i]))
- {
- bad_arg = 1;
- }
- else
- {
- hot_ch = *argv[i];
- if (!have_code)
- spec_code = 0;
- spec_char = 1;
- have_char = 1;
- }
- break;
-
- case 'k': /* Hot key is extended char: */
- /* integer following "-k" is */
- /* its extended code. */
- if ( have_code
- || 0 <= stsverfy(argv[i],"0123456789"))
- {
- bad_arg = 1;
- }
- else
- {
- hot_code = (char) utlobyte(atoi(argv[i]));
- if (!have_char)
- spec_char = '\0';
- spec_code = 1;
- have_code = 1;
- }
- break;
-
- case 'r': /* Remove resident copy of */
- /* SAVESCN. */
- if (*argv[i] == '\0')
- want_remove = 1;
- else
- bad_arg = 1;
- break;
-
- case 'w': /* Await keystroke before */
- wait = 1; /* remaining resident. (This */
- break; /* allows a chance to test */
- /* Ctrl-Break handling.) */
-
- default:
- bad_arg = 1;
- break;
- }
- }
- else
- bad_arg = 1;
- }
-
- if ( bad_arg
- || (spec_char == 0 && spec_code == 0))
- {
- printf("usage: savescn [-r] [-cCHAR] [-kKEYCODE] [-w]\n");
- return USAGE_ERROR;
- }
-
- /* Now check whether this program has been previously installed. */
-
- if ( 0 == isdetect(KEY_VEC, key_ident, &key_prev, &keypsp )
- && 0 == isdetect(TIMER_VEC,tick_ident,&tick_prev,&tickpsp)
- && 0 == isdetect(IDLE_VEC, idle_ident,&idle_prev,&idlepsp))
- {
- printf("Previous copy of SAVESCN 3.02 detected");
-
- if (want_remove)
- {
- removed = 0;
- if (keypsp == tickpsp && keypsp == idlepsp)
- {
- issetvec(KEY_VEC, &key_prev);
- issetvec(TIMER_VEC,&tick_prev);
- issetvec(IDLE_VEC, &idle_prev);
- if (0 == pcremove(keypsp))
- removed = 1;
- }
- if (removed)
- {
- printf(" and successfully removed.\n");
- return OK;
- }
- else
- {
- printf(" but it can't be removed -- aborting.\n");
- return CANT_REMOVE;
- }
- }
- else
- {
- printf(" -- aborting.\n");
- return OK;
- }
- }
- else /* Not detected */
- {
- if (want_remove)
- {
- printf("Previous copy of SAVESCN 3.02 not found -- aborting.\n");
- return OK;
- }
- }
-
- /* Now we know that we're going to terminate and stay resident. */
- /* Proceed to set up what the ISRs need: window, stack space, */
- /* and addresses of various buffers. */
-
- if (NIL == (pwindow = wncreate(6,29,
- (BLUE << 4) + WHITE + INTENSITY)))
- return b_wnerr;
-
- utabsptr(scn_buffer,&scn_buf_ads);
- utabsptr(crlf, &crlf_ads);
-
- /* Make sure local memory pool has some spare memory for ISR */
- /* window routines (so that PCRESEXT can't release it). */
-
- if (NIL == (pspare_mem = malloc(2048)))
- return NO_MEMORY;
- else
- have_spare_mem = 1;
-
- /* Obtain address of DOS critical section flag for use by */
- /* dos_ready(). */
-
- regs.ax = 0x3400;
- dos(®s);
- dos_flag_ads.r = regs.bx;
- dos_flag_ads.s = regs.es;
-
- /* Install Ctrl-Break handler first, so we can be prepared to */
- /* remove the other ISRs in case a Ctrl-Break aborts the program. */
-
- isretvec(KEY_VEC, &key_prev);
- isretvec(TIMER_VEC,&tick_prev);
- isretvec(IDLE_VEC, &idle_prev);
- if (ercode = isinstal(CBREAK_VEC,cbreak,cbrk_ident,
- &cbrk_ctl,cbrk_stack,STKSIZE,1))
- return ercode;
-
- /* Install the three ISRs that will remain effective after the */
- /* program has terminated and stayed resident. */
-
- if (ercode = isinstal(KEY_VEC,keystroke,key_ident,
- &key_ctl,key_stack,STKSIZE,NUM_STACKS))
- return ercode;
- if (ercode = isinstal(TIMER_VEC,tick,tick_ident,
- &tick_ctl,tick_stack,STKSIZE,NUM_STACKS))
- return ercode;
- if (ercode = isinstal(IDLE_VEC,idle,idle_ident,
- &idle_ctl,idle_stack,STKSIZE,NUM_STACKS))
- return ercode;
-
- /* Report successful installation. */
-
- printf("SAVESCN 3.02 installed");
- if (spec_char)
- printf(", hot character value = %d",hot_ch);
- if (spec_code)
- printf(", hot keycode = %d",hot_code);
- printf(".\n");
-
- if (wait)
- apprompt("Press ENTER to continue or Ctrl-C to abort: ",
- dummy,sizeof(dummy));
-
- pcresext(OK);
-
- return FALL_OUT;
- }
-
- /**
- *
- * Name cbreak -- Interrupt service routine for Ctrl-Break
- *
- * Description This function is installed as a handler for interrupt
- * type 0x23. It displays a message using BIOS or direct
- * I/O (using no DOS functions) and then returns to DOS via
- * a "RETF", thus causing the program to be aborted.
- *
- **/
-
- void cbreak(pregs,pisrblk,pmsg)
- ALLREG *pregs;
- ISRCTRL *pisrblk;
- ISRMSG *pmsg;
- {
- int mode,columns,act_page;
- int row,col;
-
- scmode(&mode,&columns,&act_page);
- scpage(act_page);
- sccurpos(&row,&col);
- scwrstr(row,col,
- 0,"\015\012Ctrl-C occurred -- removing SAVESCN ISRs.\015\012",
- -1,-1,CHARS_ONLY+MOVE_CUR+CUR_AFTER);
-
- issetvec(KEY_VEC, &key_prev);
- issetvec(TIMER_VEC,&tick_prev);
- issetvec(IDLE_VEC, &idle_prev);
-
- pmsg->working_flags |= CF_FLAG;
- pmsg->exit_style = IEXIT_RETF;
- }
-
- /**
- *
- * Name keystroke -- Interrupt service routine for keystrokes.
- *
- * Description This function is installed as a handler for interrupt
- * type 0x09, which is invoked whenever a key is pressed or
- * released. It allows the original INT 09H handler to
- * service the keyboard, then checks the type-ahead buffer
- * for the specified hot key. If the hot key is found, the
- * global variable want_to_save is set; all other
- * keystrokes are restored in the buffer.
- *
- * KBPLACE and emptyq() are used in lieu of functions that
- * invoke the BIOS.
- *
- **/
-
- void keystroke(pregs,pisrblk,pmsg)
- ALLREG *pregs;
- ISRCTRL *pisrblk;
- ISRMSG *pmsg;
- {
- int q_wait,stuff_err;
- KEYPAIR queue[16],*pq;
- static char panic_msg[] =
- "SAVESCN 3.02 can't re-stuff whole string!";
- static int busy = 0;
-
- utintoff();
- iscalint(&pisrblk->prev_vec,pregs);
- utintoff();
-
- if (busy)
- return;
-
- busy = 1;
- stuff_err = 0;
-
- pq = queue;
- for (q_wait = emptyq(queue); /* Examine keystroke(s). */
- q_wait > 0;
- q_wait--)
- {
- if ( ((!spec_char) || hot_ch == pq->ch)
- && ((!spec_code) || hot_code == pq->keycode))
-
- want_to_save = 1; /* Do our work. */
- /* Don't re-stuff the hot key. */
-
- else /* Non-hot key: re-stuff it. */
- if (kbplace(KB_TAIL,pq->ch,pq->keycode))
- stuff_err = 1;
- pq++;
- }
-
- if (stuff_err) /* Avoid BIOS call because BIOS */
- /* enables interrupts. */
- viwrrect(23,0,23,((int) strlen(panic_msg)) - 1,
- panic_msg,RED,BLACK,CHARS_ONLY);
-
- busy = 0;
- }
-
- /**
- *
- * Name emptyq -- Retrieve contents of keyboard buffer
- * and empty it.
- *
- * Synopsis num_typed = emptyq(pbuf);
- *
- * int num_typed Number of waiting keystrokes
- * returned.
- * KEYPAIR *pbuf Pointer to buffer in which to
- * return waiting keystrokes.
- *
- * Description This function returns all keystrokes waiting in the BIOS
- * type-ahead buffer and empties the buffer. It leaves
- * interrupts disabled.
- *
- **/
-
- int emptyq(pbuf)
- register KEYPAIR *pbuf;
- {
- static ADS head_ads = {0x001a,0x0040};
- static ADS tail_ads = {0x001c,0x0040};
- #define Q_OFFSET 0x001e
-
- struct /* rawbuf = Raw image of BIOS variables.*/
- {
- unsigned head;
- unsigned tail;
- KEYPAIR queue[16];
- } rawbuf;
-
- ADS rawbuf_ads; /* Physical address of rawbuf. */
- int num_typed;
- register int j;
-
- utabsptr((char *) &rawbuf,&rawbuf_ads);
- utintoff();
- /* Extract raw head, tail, and queue. */
- utslmove(&head_ads, &rawbuf_ads,sizeof(rawbuf));
- /* Forcing head=tail empties queue. */
- utslmove(&rawbuf_ads,&tail_ads, sizeof(rawbuf.head));
-
- rawbuf.head = (rawbuf.head - Q_OFFSET) / sizeof(KEYPAIR);
- rawbuf.tail = (rawbuf.tail - Q_OFFSET) / sizeof(KEYPAIR);
- num_typed = (rawbuf.tail + 16 - rawbuf.head) % 16;
-
- for (j = num_typed; j; j--)
- {
- pbuf->ch = rawbuf.queue[rawbuf.head].ch;
- pbuf->keycode = rawbuf.queue[rawbuf.head].keycode;
- pbuf++;
- rawbuf.head++;
- if (rawbuf.head >= 16)
- rawbuf.head = 0; /* Wrap around to beginning. */
- }
- return num_typed;
-
- #undef Q_OFFSET
- }
-
- /**
- *
- * Name tick -- Interrupt service routine for clock ticks
- *
- * Description This function is installed as a handler for hardware
- * interrupt type 8. It first calls the previous INT 8
- * handler to service the Intel 8259 Programmable Interrupt
- * Controller. Then it checks the DOS critical section
- * flag; if DOS is available, do_save() is called.
- *
- * This function protects itself against secondary
- * invocations by means of the "busy" flag.
- *
- **/
-
- void tick(pregs,pisrblk,pmsg)
- ALLREG *pregs;
- ISRCTRL *pisrblk;
- ISRMSG *pmsg;
- {
- static int busy = 0;
-
- utintoff();
- iscalint(&pisrblk->prev_vec,pregs);
-
- utintoff();
- if (!busy)
- {
- busy = 1;
- if (dos_ready())
- do_save();
- busy = 0;
- }
- }
-
- /**
- *
- * Name idle -- Interrupt service routine for INT 28h
- *
- * Description This function is installed as a handler for software
- * interrupt type 0x28 (DOS idle). It first calls the
- * previous INT 28H handler. Then it calls do_save.
- *
- * This function protects itself against secondary
- * invocations by means of the "busy" flag.
- *
- * (This function need not check dos_ready() because INT
- * 28h handlers are allowed to perform DOS disk I/O.
- * do_save() does not do DOS keyboard I/O.)
- *
- **/
-
- void idle(pregs,pisrblk,pmsg)
- ALLREG *pregs;
- ISRCTRL *pisrblk;
- ISRMSG *pmsg;
- {
- static int busy = 0;
-
- utintoff();
- iscalint(&pisrblk->prev_vec,pregs);
-
- utintoff();
- if (!busy)
- {
- busy = 1;
- do_save();
- busy = 0;
- }
- }
-
- /**
- *
- * Name dos_ready -- Report whether DOS services are available
- *
- * Synopsis ready = dos_ready();
- *
- * int ready 0 if DOS is not ready,
- * nonzero if DOS is ready.
- *
- * Description This function checks the DOS critical section flag and
- * reports whether DOS services are available.
- *
- * This function makes use of dos_flag_ads, the global
- * variable containing the address of the DOS critical
- * section flag. This variable must be loaded before
- * dos_ready() is called.
- *
- * Returns ready 0 if DOS is not ready,
- * nonzero if DOS is ready.
- *
- **/
-
- int dos_ready()
- {
- char our_flag; /* Our copy of the flag. */
- ADS our_ads; /* Address of our_flag. */
-
- utintoff();
- utabsptr(&our_flag,&our_ads);
- utslmove(&dos_flag_ads,&our_ads,1);
-
- return (our_flag == 0);
- }
-
- /**
- *
- * Name do_save -- Obtain filename and save screen image
- * if want_to_save is nonzero.
- *
- * Synopsis do_save();
- *
- * Description This function begins by testing the want_to_save flag.
- * If it is nonzero, it proceeds to obtain the name of a
- * file from the user and to save the current screen image
- * onto the file. This operation is repeated until success
- * occurs or the user gives a null filename.
- *
- * This function protects itself against secondary
- * invocations by means of the "busy" flag.
- *
- * Returns name_err OK or WINDOW_ERROR.
- * file_err OK or DOS file system error.
- * filename File name entered by user.
- *
- **/
-
- void do_save()
- {
- static int busy = 0;
-
- utintoff();
- if ((!busy) && want_to_save)
- {
- busy = 1;
-
- if (have_spare_mem)
- {
- free(pspare_mem);
- have_spare_mem = 0;
- }
-
- do
- {
- name_err = prompt("Enter name of file on which to save screen: ",
- filename,
- sizeof(filename));
- } while (OK != (file_err = write_file(filename)));
- remove_window();
- utintoff();
- want_to_save = 0;
- busy = 0;
- }
-
- utinton();
- }
-
- /**
- *
- * Name write_file -- Copy screen image to file.
- *
- * Synopsis ercode = write_file(pfilename);
- *
- * int ercode OK or DOS file system error.
- * char *pfilename Pointer to character array
- * containing file name.
- *
- * Description This function attempts to copy the screen image to the
- * named file. If the file already exists and has nonzero
- * length, it first obtains permission from the user to
- * delete the existing file.
- *
- * If any file system error occurs, the error number is
- * displayed in a window. The program then pauses until
- * the ENTER key is struck.
- *
- * Returns ercode OK or DOS file system error.
- *
- **/
-
- int write_file(pfilename)
- char *pfilename;
- {
- int handle;
- long size;
- char answer[3];
- int mode,columns,act_page,rows,row;
- int ercode,nwritten;
-
- if (*pfilename == '\0') /* No filename given. */
- return OK;
-
- utinton();
- switch (ercode = flopen(pfilename,&handle,WRTONLY))
- {
- case 0: /* File already exists */
- size = 0L;
- flseek(handle,ENDFILE,&size);
- if (size == 0L)
- break;
-
- flclose(handle);
- prompt("Delete existing file? (Y/N) ",answer,sizeof(answer));
- if (*stpcvt(answer,RLWHITE+TOLOW) != 'y')
- return TRY_NEW_FILENAME;
-
- case 2: /* Truncate existing file */
- /* or create new file. */
- if (ercode = flcreate(pfilename,&handle,AT_GENERAL))
- {
- warning("DOS error creating file: ",ercode);
- return ercode;
- }
- break;
-
- default:
- warning("DOS error opening file: ",ercode);
- return ercode;
- }
-
- remove_window();
- scmode(&mode,&columns,&act_page);
- rows = scrows();
- scpage(act_page);
-
- for (row = 0; row < rows; row++)
- {
- gvrdrect(row,0,row,columns - 1,scn_buffer,CHARS_ONLY);
- if (ercode = flwrite(handle,&scn_buf_ads,columns,&nwritten))
- {
- warning("DOS error writing file: ",ercode);
- return ercode;
- }
- else if (nwritten != columns)
- {
- warning("Target device full",0);
- return TARGET_FULL;
- }
-
- if (ercode = flwrite(handle,&crlf_ads,2,&nwritten))
- {
- warning("DOS error writing file: ",ercode);
- return ercode;
- }
- else if (nwritten != 2)
- {
- warning("Target device full",0);
- return TARGET_FULL;
- }
- }
-
- if (ercode = flclose(handle))
- warning("DOS error closing file: ",ercode);
- return ercode;
- }
-
- /**
- *
- * Name display_window -- Display window for user dialog.
- *
- * Synopsis ercode = display_window();
- *
- * int ercode OK or WINDOW_ERROR.
- *
- * Description This function displays a window (if not already
- * displayed) on the active display page for messages
- * and/or user input. It also records the previous
- * location and size of the cursor so that the cursor may
- * be restored by remove_window().
- *
- * Returns ercode OK or WINDOW_ERROR.
- *
- **/
-
- int display_window()
- {
- static WHERE where_to_put_window =
- {
- 0, /* Video display device */
- 0, /* Video display page */
- {6,5} /* Row & column */
- };
- static BORDER window_border =
- {
- 16, /* Double box */
- RED, /* Color */
- 0 /* Border character (ignored) */
- };
- int mode,columns,act_page;
-
- if (window_displayed) /* Quit if window already */
- return OK; /* displayed. */
-
- where_to_put_window.dev = scmode(&mode,&columns,&act_page);
- where_to_put_window.page = act_page;
-
- scpage(act_page);
- /* Obtain former cursor state. */
- cursor_was_off = sccurst(&old_row,&old_col,&high,&low);
-
- b_wnerr = WN_NO_ERROR; /* Clear residual error code. */
- /* Display the window. */
- if (NIL == wndsplay(pwindow,&where_to_put_window,&window_border))
- return WINDOW_ERROR;
-
- window_displayed = 1;
- return OK;
- }
-
- /**
- *
- * Name remove_window -- Remove window and restore previous
- * screen.
- *
- * Synopsis remove_window();
- *
- * Description This function removes the window (if currently
- * displayed) and restores the previous contents of the
- * screen and the state of the cursor.
- *
- * Returns (None -- function return type is void.)
- *
- **/
-
- void remove_window()
- {
- if (window_displayed)
- {
- wnremove(pwindow);
-
- /* Restore former cursor state. */
- scpgcur(cursor_was_off,high,low,CUR_NO_ADJUST);
- sccurset(old_row,old_col);
- window_displayed = 0;
- }
- }
-
- /**
- *
- * Name prompt -- Display a message and await a response from
- * the user.
- *
- * Synopsis ercode = prompt(pquestion,panswer,ans_size);
- *
- * int ercode OK or WINDOW_ERROR.
- * char *pquestion Message to display.
- * char *panswer Buffer in which to place response.
- * int ans_size Size of answer buffer in bytes
- * (including one byte for trailing
- * NUL ('\0')).
- *
- * Description This function displays a window (if not already
- * displayed) and the message contained in *pquestion.
- * Then it waits for the user to enter a reply terminated
- * by the ENTER key.
- *
- * Returns ercode OK or WINDOW_ERROR.
- *
- **/
-
- int prompt(pquestion,panswer,ans_size)
- char *pquestion,*panswer;
- int ans_size;
- {
- int keycode,ercode;
- int row,col;
- char key;
-
- *panswer = '\0'; /* Null answer in case of error */
-
- b_wnerr = WN_NO_ERROR;
- wnselect(pwindow);
- wnscroll(0,-1,-1,SCR_UP); /* Clear window. */
- wncurmov(0,0);
- /* Display the prompt. */
- wnwrap(0,pquestion,-1,-1,CHARS_ONLY);
- if (b_wnerr != WN_NO_ERROR)
- return WINDOW_ERROR;
-
- if (OK != (ercode = display_window()))
- return ercode;
-
- kbflush();
- utinton();
-
- b_wnerr = WN_NO_ERROR;
- wncurpos(&row,&col);
- do
- {
- wncurmov(row,col);
- key = wnquery(panswer,ans_size,&keycode);
- } while (b_wnerr == WN_NO_ERROR && key != ENTER_KEY);
-
- if (b_wnerr != WN_NO_ERROR)
- {
- *panswer = '\0';
- return WINDOW_ERROR;
- }
-
- stpcvt(panswer,RWHITE+TOLOW);
- return OK;
- }
-
- /**
- *
- * Name warning -- Report a DOS error and wait for user
- * to press the ENTER key.
- *
- * Synopsis ercode = warning(ptext,number);
- *
- * int ercode OK or WINDOW_ERROR.
- * char *pmessage Message to display.
- * int number DOS error number to display.
- *
- * Description This function displays a window (if not already
- * displayed), the message contained in *pquestion, and the
- * error number. Then it waits for the user to enter a
- * reply terminated by the ENTER key.
- *
- * Returns ercode OK or WINDOW_ERROR.
- *
- **/
-
- int warning(ptext,number)
- char *ptext;
- int number;
- {
- char message[81],dummy[2];
-
- message[0] = '\0';
- strcat(message,ptext);
- if (number)
- sprintf(&(message[strlen(message)]),"%d",number);
- strcat(message,"\015\012-- Press ENTER to continue.");
-
- return prompt(message,dummy,sizeof(dummy));
- }