minix/lib/libcurses/update.c

171 lines
4.1 KiB
C

#include <curses.h>
#include "curspriv.h"
#include <termcap.h>
static WINDOW *twin; /* used by many routines */
/****************************************************************/
/* Gotoxy() moves the physical cursor to the desired address on */
/* The screen. We don't optimize here - on a PC, it takes more */
/* Time to optimize than to do things directly. */
/****************************************************************/
_PROTOTYPE(static void gotoxy, (int row, int col ));
_PROTOTYPE(static void newattr, (int ch ));
_PROTOTYPE(static void Putchar, (int ch ));
_PROTOTYPE(static void clrupdate, (WINDOW *scr ));
_PROTOTYPE(static void transformline, (int lineno ));
static void gotoxy(row, col)
int row, col;
{
poscur(row, col);
_cursvar.cursrow = row;
_cursvar.curscol = col;
}
/* Update attributes */
static void newattr(ch)
int ch;
{
extern char *me, *as, *ae, *mb, *md, *mr, *so, *us;
static int lastattr = 0;
if (lastattr != (ch &= ATR_MSK)) {
lastattr = ch;
tputs(me, 1, outc);
if (ae) tputs(ae, 1, outc);
if (ch & A_ALTCHARSET)
if (as) tputs(as, 1, outc);
if (ch & A_BLINK) tputs(mb, 1, outc);
if (ch & A_BOLD) tputs(md, 1, outc);
if (ch & A_REVERSE) tputs(mr, 1, outc);
if (ch & A_STANDOUT) tputs(so, 1, outc);
if (ch & A_UNDERLINE) tputs(us, 1, outc);
}
}
/* Putchar() writes a character, with attributes, to the physical
screen, but avoids writing to the lower right screen position.
Should it care about am?
*/
/* Output char with attribute */
static void Putchar(ch)
int ch;
{
if ((_cursvar.cursrow < LINES) || (_cursvar.curscol < COLS)) {
newattr(ch);
(void) putchar(ch);
}
}
/****************************************************************/
/* Clrupdate(scr) updates the screen by clearing it and then */
/* Redraw it in it's entirety. */
/****************************************************************/
static void clrupdate(scr)
WINDOW *scr;
{
register int *src;
register int *dst;
register int i;
register int j;
WINDOW *w;
w = curscr;
if (scr != w) { /* copy scr to curscr */
for (i = 0; i < LINES; i++) {
src = scr->_line[i];
dst = w->_line[i];
for (j = 0; j < COLS; j++) *dst++ = *src++;
} /* for */
} /* if */
newattr(scr->_attrs);
clrscr();
scr->_clear = FALSE;
for (i = 0; i < LINES; i++) { /* update physical screen */
src = w->_line[i];
j = 0;
while (j < COLS) {
if (*src != (' ' | ATR_NRM)) {
gotoxy(i, j);
while (j < COLS && (*src != (' ' | ATR_NRM))) {
Putchar(*src++);
j++;
}
} else {
src++;
j++;
}
} /* for */
} /* for */
fflush(stdout);
} /* clrupdate */
/****************************************************************/
/* Transformline() updates the given physical line to look */
/* Like the corresponding line in _cursvar.tmpwin. */
/****************************************************************/
static void transformline(register int lineno)
{
register int *dstp;
register int *srcp;
int x;
int endx;
x = twin->_minchng[lineno];
endx = twin->_maxchng[lineno];
dstp = curscr->_line[lineno] + x;
srcp = twin->_line[lineno] + x;
while (x <= endx) {
if (*dstp != *srcp) {
gotoxy(lineno, x);
while (x <= endx && (*dstp != *srcp)) {
Putchar(*srcp);
*dstp++ = *srcp++;
x++;
}
} else {
*dstp++ = *srcp++;
x++;
}
} /* for */
twin->_minchng[lineno] = _NO_CHANGE;
twin->_maxchng[lineno] = _NO_CHANGE;
} /* transformline */
/****************************************************************/
/* Doupdate() updates the physical screen to look like _curs- */
/* Var.tmpwin if curscr is not 'Clear-marked'. Otherwise it */
/* Updates the screen to look like curscr. */
/****************************************************************/
void doupdate()
{
int i;
twin = _cursvar.tmpwin;
if (curscr->_clear)
clrupdate(curscr);
else {
if (twin->_clear)
clrupdate(twin);
else {
for (i = 0; i < LINES; i++)
if (twin->_minchng[i] != _NO_CHANGE)
transformline(i);
}
}
curscr->_curx = twin->_curx;
curscr->_cury = twin->_cury;
gotoxy(curscr->_cury, curscr->_curx);
fflush(stdout);
} /* doupdate */