home *** CD-ROM | disk | FTP | other *** search
- /*Prg7_5 - Open/Close Screen Windows
- by Stephen R. Davis, 1987
-
- Example of using our screen writing skills to open and close
- windows on the screen
- */
-
- #include <stdio.h>
- #include <dos.h>
- #include <stdlib.h>
- #include <process.h>
-
- /*define screen parameters*/
- #define height 25
- #define width 80
- #define cga (unsigned far *)0xb8000000 /*same for ega*/
- #define mono (unsigned far *)0xb0000000
-
- /*define screen BIOS functions*/
- #define setcursor 0x02
- #define getmode 0x0f
-
- /*define the colors - 'b'ackground and 'f'oreground*/
- #define bred 0x4000
- #define bgreen 0x2000
- #define bblue 0x1000
- #define bwhite bred+bgreen+bblue
- #define fred 0x0400
- #define fgreen 0x0200
- #define fblue 0x0100
- #define fwhite fblue+fgreen+fred
-
- /*type definitions*/
- struct WINDOW {
- unsigned x0, y0; /*upper, left hand corner*/
- unsigned x1, y1; /*lower, right hand corner*/
- };
-
- /*our prototype definitions*/
- void main (void);
- char gtcr (struct WINDOW *, char *);
- void openwindow (struct WINDOW *, unsigned *, unsigned);
- void closewindow (struct WINDOW *, unsigned *);
- void wscroll (struct WINDOW *, unsigned);
- void wprintf (struct WINDOW *, char *);
- void wpcursor (struct WINDOW *, unsigned, unsigned);
- void init (void);
-
- /*data definitions*/
- union REGS regs;
- unsigned far *screen;
- typedef unsigned SCREEN [height * width + 5];
- unsigned windowwidth, windowheight, color;
- unsigned v_pos, h_pos;
-
- /*Main - exercise the window routines*/
-
- struct WINDOW full = { 0, 0, 79, 24};
- struct WINDOW win1 = {10, 5, 60, 20};
- struct WINDOW win2 = {20, 0, 40, 24};
- SCREEN buff1, buff2;
-
- void main (void)
- {
- unsigned i;
-
- /*initialize the works*/
- init ();
-
- /*put pattern up before opening first window*/
- for (i = 1; i < height; i++)
- wprintf (&full, "***************************************"
- "***************************************\n");
- wpcursor (&full, 0, 0);
-
- /*open first window with pattern and then scroll it*/
- gtcr (&full, "This is the main screen\n"
- "Press Enter key to open first window");
- openwindow (&win1, buff1, bred+fwhite);
- for (i = 1; i < 50; i++)
- wprintf (&win1, "Window #1 --- scrolling\n\n");
-
- /*now open window #2 and scroll it also*/
- gtcr (&win1, "Press Enter key for #2");
- openwindow (&win2, buff2, bblue+fgreen);
- for (i = 1; i < 50; i++)
- wprintf (&win2, "Window #2 --\n"
- " scrolling\n\n");
-
- /*now, prepare to close*/
- gtcr (&win2, "Press Enter key");
- closewindow (&win2, buff2);
- gtcr (&win1, "Press Enter key");
- closewindow (&win1, buff1);
- }
-
- /*Gtcr - output a prompt and then await a character response*/
- char gtcr (window, prompt)
- struct WINDOW *window;
- char *prompt;
- {
- char buffer [80];
-
- /*build the message string and write it into current window*/
- sprintf (buffer, "%s :\n", prompt);
- wprintf (window, buffer);
-
- /*now await a response*/
- return (char)getchar ();
- }
-
- /*Openwindow - save off a specified box from the screen and set
- to a box of the given box*/
- void openwindow (window, buffer, fill)
- struct WINDOW *window;
- unsigned *buffer, fill;
- {
- unsigned line, column, far *screenptr;
-
- /*first save the cursor and window info*/
- *buffer++ = h_pos;
- *buffer++ = v_pos;
- *buffer++ = windowwidth;
- *buffer++ = windowheight;
- *buffer++ = color;
-
- /*now save off the window and cover it with 'fill'*/
- for (line = window -> y0; line < window -> y1; line++) {
- screenptr = screen + ((line * width) + window -> x0);
- for (column = window -> x0; column < window ->x1;
- column++, screenptr++, buffer++) {
- *buffer = *screenptr;
- *screenptr = fill;
- }
- }
-
- /*calculate new width and height*/
- h_pos = 0;
- v_pos = 0;
- windowwidth = window -> x1 - window -> x0;
- windowheight = window -> y1 - window -> y0;
- color = fill & 0xff00;
- }
-
- /*Closewindow - restore the previously saved window to the screen*/
- void closewindow (window, buffer)
- struct WINDOW *window;
- unsigned *buffer;
- {
- unsigned line, column, far *screenptr;
-
- /*first restore the cursor position and screen dimensions*/
- h_pos = *buffer++;
- v_pos = *buffer++;
- windowwidth = *buffer++;
- windowheight = *buffer++;
- color = *buffer++;
-
- /*now restore the window area of the screen*/
- for (line = window -> y0; line < window -> y1; line++) {
- screenptr = screen + ((line * width) + window -> x0);
- for (column = window -> x0; column < window ->x1; column++)
- *screenptr++ = *buffer++;
- }
- }
-
- /*WScroll - scroll current window up N lines*/
-
- #define SCREENLOC(y,x) screen + ((width * (y + window -> y0)) \
- + (window -> x0 + x))
-
- void wscroll (window, nlines)
- struct WINDOW *window;
- unsigned nlines;
- {
- unsigned far *source, far *dest, number, i, j;
-
- if (nlines >= windowheight)
- nlines = windowheight;
-
- h_pos = 0;
- if ((v_pos += nlines) >= windowheight) {
- nlines = (v_pos - windowheight) + 1;
-
- /*scroll the screen up 'nlines' amount*/
- number = windowheight - nlines;
- for (i = 0; i < number; i++) {
- source = SCREENLOC (i + nlines, 0);
- dest = SCREENLOC (i, 0);
- for (j = 0; j < windowwidth; j++)
- *dest++ = *source++;
- }
-
- /*now blank the lines abandoned*/
- for (i = number; i < windowheight; i++)
- dest = SCREENLOC (i, 0);
- for (j = 0; j < windowwidth; j++)
- *dest++ = color + 0x20;
-
- v_pos = windowheight - 1;
- }
- }
-
- /*Wprintf - output a string using the BIOS screen handler. If
- an attribute is not provided, use the default.*/
- void wprintf (window, c)
- struct WINDOW *window;
- char *c;
- {
- unsigned far *sp;
-
- sp = SCREENLOC (v_pos, h_pos);
- for (; *c; c++)
- if (*c == '\n') {
- wscroll (window, 1);
- sp = SCREENLOC (v_pos, 0);
- }
- else
- if (h_pos < windowwidth) {
- h_pos++;
- *sp++ = color + *c;
- }
- wpcursor (window, v_pos, h_pos);
- }
-
- /*WPCursor - place the cursor at the current x and y location.*/
- void wpcursor (window, y, x)
- struct WINDOW *window;
- unsigned x, y;
- {
- v_pos = y;
- h_pos = x;
-
- regs.h.ah = setcursor;
- regs.h.bh = 0;
- regs.h.dh = v_pos + window -> y0;
- regs.h.dl = h_pos + window -> x0;
- int86 (0x10, ®s, ®s);
- }
-
- /*Init - set the screen address and clear the screen*/
- void init ()
- {
- short mode;
-
- regs.h.ah = getmode;
- int86 (0x10, ®s, ®s);
- mode = regs.h.al;
-
- /*we are only set up for 80 column widths...*/
- if (regs.h.ah != width)
- abort ();
-
- /*...and one of the character modes*/
- if (mode == 7)
- screen = mono;
- else
- if (mode == 3 || mode == 2)
- screen = cga;
- else
- abort ();
-
- /*now initialize the cursor and screen dimensions*/
- windowheight = height;
- windowwidth = width;
- color = fwhite;
- h_pos = v_pos = 0;
-
- /*and clear the screen*/
- wscroll (&full, height);
- wpcursor (&full, 0, 0);
- }
-