1
0
Fork 0
mirror of https://codeberg.org/noisytoot/notnotdnethack.git synced 2024-11-14 13:26:10 +00:00
notnotdnethack/win/tty/termcap.c
Ron Nazarov 378f50efd6
Always enable TEXTCOLOR and STATUS_COLORS and require terminfo
Always enabled: TERMLIB ASCIIGRAPH TERMINFO TEXTCOLOR STATUS_COLORS

Removed: NO_TERMS
2024-05-06 18:48:53 +01:00

950 lines
22 KiB
C

/* SCCS Id: @(#)termcap.c 3.4 2000/07/10 */
/* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
/* NetHack may be freely redistributed. See license for details. */
#include "hack.h"
#if defined (TTY_GRAPHICS)
#include "wintty.h"
#include "tcap.h"
#define Tgetstr(key) (tgetstr(key,&tbufptr))
static char * s_atr2str(int);
static char * e_atr2str(int);
void cmov(int, int);
void nocmov(int, int);
# if !defined(UNIX)
static void analyze_seq(char *, int *, int *);
# endif
static void init_hilite(void);
static void kill_hilite(void);
/* (see tcap.h) -- nh_CM, nh_ND, nh_CD, nh_HI,nh_HE, nh_US,nh_UE,
ul_hack */
struct tc_lcl_data tc_lcl_data = { 0, 0, 0, 0,0, 0,0, FALSE };
int allow_bgcolor = 0;
static char *HO, *CL, *CE, *UP, *XD, *BC, *SO, *SE, *TI, *TE;
static char *VS, *VE;
static char *ME;
static char *MR;
#if 0
static char *MB, *MH;
static char *MD; /* may already be in use below */
#endif
static char *MD;
static int SG;
static char PC = '\0';
static char tbuf[512];
char *hilites[CLR_MAX]; /* terminal escapes for the various colors */
static char *KS = (char *)0, *KE = (char *)0; /* keypad sequences */
static char nullstr[] = "";
extern boolean HE_resets_AS;
void
tty_startup(int *wid, int *hgt)
{
register int i;
register const char *term;
register char *tptr;
char *tbufptr, *pc;
term = getenv("TERM");
if (!term)
#ifndef ANSI_DEFAULT
error("Can't get TERM.");
#else
{
HO = "\033[H";
/* nh_CD = "\033[J"; */
CE = "\033[K"; /* the ANSI termcap */
nh_CM = "\033[%i%d;%dH";
UP = "\033[A";
nh_ND = "\033[C";
XD = "\033[B";
BC = "\033[D";
nh_HI = SO = "\033[1m";
allow_bgcolor = 1;
nh_US = "\033[4m";
MR = "\033[7m";
TI = nh_HE = ME = SE = nh_UE = "\033[0m";
/* strictly, SE should be 2, and nh_UE should be 24,
but we can't trust all ANSI emulators to be
that complete. -3. */
AS = "\016";
AE = "\017";
TE = VS = VE = nullstr;
for (i = 0; i < CLR_MAX / 2; i++)
if (i != CLR_BLACK) {
hilites[i|BRIGHT] = (char *) alloc(sizeof("\033[1;3%dm"));
Sprintf(hilites[i|BRIGHT], "\033[1;3%dm", i);
if (i != CLR_GRAY)
{
hilites[i] = (char *) alloc(sizeof("\033[0;3%dm"));
Sprintf(hilites[i], "\033[0;3%dm", i);
}
}
*wid = CO;
*hgt = LI;
CL = "\033[2J"; /* last thing set */
return;
}
#endif /* ANSI_DEFAULT */
tptr = (char *) alloc(1024);
tbufptr = tbuf;
if(!strncmp(term, "5620", 4))
flags.null = FALSE; /* this should be a termcap flag */
if(tgetent(tptr, term) < 1) {
char buf[BUFSZ];
(void) strncpy(buf, term,
(BUFSZ - 1) - (sizeof("Unknown terminal type: . ")));
buf[BUFSZ-1] = '\0';
error("Unknown terminal type: %s.", term);
}
if ((pc = Tgetstr("pc")) != 0)
PC = *pc;
if(!(BC = Tgetstr("le"))) /* both termcap and terminfo use le */
error("Terminal must backspace.");
# ifdef MINIMAL_TERM
HO = (char *)0;
# else
HO = Tgetstr("ho");
# endif
/*
* LI and CO are set in ioctl.c via a TIOCGWINSZ if available. If
* the kernel has values for either we should use them rather than
* the values from TERMCAP ...
*/
if (!CO) CO = tgetnum("co");
if (!LI) LI = tgetnum("li");
if(CO < COLNO || LI < ROWNO+3)
setclipped();
nh_ND = Tgetstr("nd");
if(tgetflag("os"))
error("NetHack can't have OS.");
if(tgetflag("ul"))
ul_hack = TRUE;
CE = Tgetstr("ce");
UP = Tgetstr("up");
/* It seems that xd is no longer supported, and we should use
a linefeed instead; unfortunately this requires resetting
CRMOD, and many output routines will have to be modified
slightly. Let's leave that till the next release. */
XD = Tgetstr("xd");
/* not: XD = Tgetstr("do"); */
if(!(nh_CM = Tgetstr("cm"))) {
if(!UP && !HO)
error("NetHack needs CM or UP or HO.");
tty_raw_print("Playing NetHack on terminals without CM is suspect.");
tty_wait_synch();
}
SO = Tgetstr("so");
SE = Tgetstr("se");
nh_US = Tgetstr("us");
nh_UE = Tgetstr("ue");
SG = tgetnum("sg"); /* -1: not fnd; else # of spaces left by so */
if(!SO || !SE || (SG > 0)) SO = SE = nh_US = nh_UE = nullstr;
TI = Tgetstr("ti");
TE = Tgetstr("te");
VS = VE = nullstr;
VS = Tgetstr("eA"); /* enable graphics */
KS = Tgetstr("ks"); /* keypad start (special mode) */
KE = Tgetstr("ke"); /* keypad end (ordinary mode [ie, digits]) */
MR = Tgetstr("mr"); /* reverse */
# if 0
MB = Tgetstr("mb"); /* blink */
MD = Tgetstr("md"); /* boldface */
MH = Tgetstr("mh"); /* dim */
# endif
ME = Tgetstr("me"); /* turn off all attributes */
if (!ME || (SE == nullstr)) ME = SE; /* default to SE value */
/* Get rid of padding numbers for nh_HI and nh_HE. Hope they
* aren't really needed!!! nh_HI and nh_HE are outputted to the
* pager as a string - so how can you send it NULs???
* -jsb
*/
nh_HI = (char *) alloc((unsigned)(strlen(SO)+1));
nh_HE = (char *) alloc((unsigned)(strlen(ME)+1));
i = 0;
while (digit(SO[i])) i++;
Strcpy(nh_HI, &SO[i]);
i = 0;
while (digit(ME[i])) i++;
Strcpy(nh_HE, &ME[i]);
AS = Tgetstr("as");
AE = Tgetstr("ae");
nh_CD = Tgetstr("cd");
allow_bgcolor = 1;
MD = Tgetstr("md");
init_hilite();
*wid = CO;
*hgt = LI;
CL = "\033[2J"; /* last thing set */
/*if (!(CL = Tgetstr("cl")))
error("NetHack needs CL.");*/
if ((int)(tbufptr - tbuf) > (int)(sizeof tbuf))
error("TERMCAP entry too big...\n");
free((void *)tptr);
}
/* note: at present, this routine is not part of the formal window interface */
/* deallocate resources prior to final termination */
void
tty_shutdown(void)
{
kill_hilite();
/* we don't attempt to clean up individual termcap variables [yet?] */
return;
}
void
tty_number_pad(int state)
{
switch (state) {
case -1: /* activate keypad mode (escape sequences) */
if (KS && *KS) xputs(KS);
break;
case 1: /* activate numeric mode for keypad (digits) */
if (KE && *KE) xputs(KE);
break;
case 0: /* don't need to do anything--leave terminal as-is */
default:
break;
}
}
extern void (*decgraphics_mode_callback)(void); /* defined in drawing.c */
static void tty_decgraphics_termcap_fixup(void);
/*
We call this routine whenever DECgraphics mode is enabled, even if it
has been previously set, in case the user manages to reset the fonts.
The actual termcap fixup only needs to be done once, but we can't
call xputs() from the option setting or graphics assigning routines,
so this is a convenient hook.
*/
static void
tty_decgraphics_termcap_fixup(void)
{
static char ctrlN[] = "\016";
static char ctrlO[] = "\017";
static char appMode[] = "\033=";
static char numMode[] = "\033>";
/* these values are missing from some termcaps */
if (!AS) AS = ctrlN; /* ^N (shift-out [graphics font]) */
if (!AE) AE = ctrlO; /* ^O (shift-in [regular font]) */
if (!KS) KS = appMode; /* ESC= (application keypad mode) */
if (!KE) KE = numMode; /* ESC> (numeric keypad mode) */
/*
* Select the line-drawing character set as the alternate font.
* Do not select NA ASCII as the primary font since people may
* reasonably be using the UK character set.
*/
if (iflags.DECgraphics) xputs("\033)0");
/* some termcaps suffer from the bizarre notion that resetting
video attributes should also reset the chosen character set */
{
const char *nh_he = nh_HE, *ae = AE;
int he_limit, ae_length;
if (digit(*ae)) { /* skip over delay prefix, if any */
do ++ae; while (digit(*ae));
if (*ae == '.') { ++ae; if (digit(*ae)) ++ae; }
if (*ae == '*') ++ae;
}
/* can't use nethack's case-insensitive strstri() here, and some old
systems don't have strstr(), so use brute force substring search */
ae_length = strlen(ae), he_limit = strlen(nh_he);
while (he_limit >= ae_length) {
if (strncmp(nh_he, ae, ae_length) == 0) {
HE_resets_AS = TRUE;
break;
}
++nh_he, --he_limit;
}
}
}
void
tty_start_screen(void)
{
xputs(TI);
xputs(VS);
if (iflags.DECgraphics) tty_decgraphics_termcap_fixup();
/* set up callback in case option is not set yet but toggled later */
decgraphics_mode_callback = tty_decgraphics_termcap_fixup;
if (iflags.num_pad) tty_number_pad(1); /* make keypad send digits */
}
void
tty_end_screen(void)
{
clear_screen();
xputs(VE);
xputs(TE);
}
/* Cursor movements */
void
nocmov(int x, int y)
{
if ((int) ttyDisplay->cury > y) {
if(UP) {
while ((int) ttyDisplay->cury > y) { /* Go up. */
xputs(UP);
ttyDisplay->cury--;
}
} else if(nh_CM) {
cmov(x, y);
} else if(HO) {
home();
tty_curs(BASE_WINDOW, x+1, y);
} /* else impossible("..."); */
} else if ((int) ttyDisplay->cury < y) {
if(XD) {
while((int) ttyDisplay->cury < y) {
xputs(XD);
ttyDisplay->cury++;
}
} else if(nh_CM) {
cmov(x, y);
} else {
while((int) ttyDisplay->cury < y) {
xputc('\n');
ttyDisplay->curx = 0;
ttyDisplay->cury++;
}
}
}
if ((int) ttyDisplay->curx < x) { /* Go to the right. */
if(!nh_ND) cmov(x, y); else /* bah */
/* should instead print what is there already */
while ((int) ttyDisplay->curx < x) {
xputs(nh_ND);
ttyDisplay->curx++;
}
} else if ((int) ttyDisplay->curx > x) {
while ((int) ttyDisplay->curx > x) { /* Go to the left. */
xputs(BC);
ttyDisplay->curx--;
}
}
}
void
cmov(int x, int y)
{
xputs(tgoto(nh_CM, x, y));
ttyDisplay->cury = y;
ttyDisplay->curx = x;
}
void
xputc(char c)
{
(void) putchar(c);
}
void
xputs(const char *s)
{
tputs(s, 1, (int (*)())xputc);
}
void
cl_end(void)
{
if(CE)
xputs(CE);
else { /* no-CE fix - free after Harold Rynes */
/* this looks terrible, especially on a slow terminal
but is better than nothing */
register int cx = ttyDisplay->curx+1;
while(cx < CO) {
xputc(' ');
cx++;
}
tty_curs(BASE_WINDOW, (int)ttyDisplay->curx+1,
(int)ttyDisplay->cury);
}
}
void
clear_screen(void)
{
/* note: if CL is null, then termcap initialization failed,
so don't attempt screen-oriented I/O during final cleanup.
*/
if (CL) {
xputs(CL);
home();
}
}
void
home(void)
{
if(HO)
xputs(HO);
else if(nh_CM)
xputs(tgoto(nh_CM, 0, 0));
else
tty_curs(BASE_WINDOW, 1, 0); /* using UP ... */
ttyDisplay->curx = ttyDisplay->cury = 0;
}
void
standoutbeg(void)
{
if(SO) xputs(SO);
}
void
standoutend(void)
{
if(SE) xputs(SE);
}
#if 0 /* if you need one of these, uncomment it (here and in extern.h) */
void
revbeg(void)
{
if(MR) xputs(MR);
}
void
boldbeg(void)
{
if(MD) xputs(MD);
}
void
blinkbeg(void)
{
if(MB) xputs(MB);
}
void
dimbeg(void)
/* not in most termcap entries */
{
if(MH) xputs(MH);
}
void
m_end(void)
{
if(ME) xputs(ME);
}
#endif
void
backsp(void)
{
xputs(BC);
}
void
tty_nhbell(void)
{
if (flags.silent) return;
(void) putchar('\007'); /* curx does not change */
(void) fflush(stdout);
}
void
graph_on(void) {
if (AS) xputs(AS);
}
void
graph_off(void) {
if (AE) xputs(AE);
}
static const short tmspc10[] = { /* from termcap */
0, 2000, 1333, 909, 743, 666, 500, 333, 166, 83, 55, 41, 20, 10, 5
};
/* delay 50 ms */
void
tty_delay_output(int delay)
{
if (iflags.debug_fuzzer)
return;
#ifdef TIMED_DELAY
if (flags.nap) {
(void) fflush(stdout);
static int length[] = { 0, 10, 50 };
msleep(length[delay]); /* sleep based on passed in delay */
return;
}
#endif
/* BUG: if the padding character is visible, as it is on the 5620
then this looks terrible. */
if(flags.null)
/* cbosgd!cbcephus!pds for SYS V R2 */
tputs("$<50>", 1, (int (*)())xputc);
else if(ospeed > 0 && ospeed < SIZE(tmspc10) && nh_CM) {
/* delay by sending cm(here) an appropriate number of times */
register int cmlen = strlen(tgoto(nh_CM, ttyDisplay->curx,
ttyDisplay->cury));
register int i = 500 + tmspc10[ospeed]/2;
while(i > 0) {
cmov((int)ttyDisplay->curx, (int)ttyDisplay->cury);
i -= cmlen*tmspc10[ospeed];
}
}
}
void
cl_eos(void) /* free after Robert Viduya */
{ /* must only be called with curx = 1 */
if(nh_CD)
xputs(nh_CD);
else {
register int cy = ttyDisplay->cury+1;
while(cy <= LI-2) {
cl_end();
xputc('\n');
cy++;
}
cl_end();
tty_curs(BASE_WINDOW, (int)ttyDisplay->curx+1,
(int)ttyDisplay->cury);
}
}
# if defined(UNIX)
/*
* Sets up color highlighting, using terminfo(4) escape sequences.
*
* Having never seen a terminfo system without curses, we assume this
* inclusion is safe. On systems with color terminfo, it should define
* the 8 COLOR_FOOs, and avoid us having to guess whether this particular
* terminfo uses BGR or RGB for its indexes.
*
* If we don't get the definitions, then guess. Original color terminfos
* used BGR for the original Sf (setf, Standard foreground) codes, but
* there was a near-total lack of user documentation, so some subsequent
* terminfos, such as early Linux ncurses and SCO UNIX, used RGB. Possibly
* as a result of the confusion, AF (setaf, ANSI Foreground) codes were
* introduced, but this caused yet more confusion. Later Linux ncurses
* have BGR Sf, RGB AF, and RGB COLOR_FOO, which appears to be the SVR4
* standard. We could switch the colors around when using Sf with ncurses,
* which would help things on later ncurses and hurt things on early ncurses.
* We'll try just preferring AF and hoping it always agrees with COLOR_FOO,
* and falling back to Sf if AF isn't defined.
*
* In any case, treat black specially so we don't try to display black
* characters on the assumed black background.
*/
/* `curses' is aptly named; various versions don't like these
macros used elsewhere within nethack; fortunately they're
not needed beyond this point, so we don't need to worry
about reconstructing them after the header file inclusion. */
#undef delay_output
#undef TRUE
#undef FALSE
#define m_move curses_m_move /* Some curses.h decl m_move(), not used here */
#include <curses.h>
# ifndef COLOR_BLACK /* trust include file */
#define COLOR_BLACK 0
#define COLOR_BLUE 1
#define COLOR_GREEN 2
#define COLOR_CYAN 3
#define COLOR_RED 4
#define COLOR_MAGENTA 5
#define COLOR_YELLOW 6
#define COLOR_WHITE 7
# endif
/* Mapping data for the six terminfo colors that resolve to pairs of nethack
* colors. Black and white are handled specially.
*/
const struct {int ti_color, nh_color, nh_bright_color;} ti_map[6] =
{
{COLOR_RED,CLR_RED,CLR_ORANGE},
{COLOR_GREEN,CLR_GREEN,CLR_BRIGHT_GREEN},
{COLOR_YELLOW,CLR_BROWN,CLR_YELLOW},
{COLOR_BLUE,CLR_BLUE,CLR_BRIGHT_BLUE},
{COLOR_MAGENTA,CLR_MAGENTA,CLR_BRIGHT_MAGENTA},
{COLOR_CYAN,CLR_CYAN,CLR_BRIGHT_CYAN}
};
static void
init_hilite(void)
{
register int c;
char *setf, *scratch;
int length_md;
int colors = tgetnum("Co");
if (colors < 8
|| (MD == NULL) || (strlen(MD) == 0)
|| ((setf = tgetstr("AF", (char **)0)) == (char *)0
&& (setf = tgetstr("Sf", (char **)0)) == (char *)0))
{
/* Fallback when colors not available
* It's arbitrary to collapse all colors except gray
* together, but that's what the previous code did.
*/
hilites[CLR_BLACK] = nh_HI;
hilites[CLR_RED] = nh_HI;
hilites[CLR_GREEN] = nh_HI;
hilites[CLR_BROWN] = nh_HI;
hilites[CLR_BLUE] = nh_HI;
hilites[CLR_MAGENTA] = nh_HI;
hilites[CLR_CYAN] = nh_HI;
hilites[CLR_GRAY] = "";
hilites[NO_COLOR] = "";
hilites[CLR_ORANGE] = nh_HI;
hilites[CLR_BRIGHT_GREEN] = nh_HI;
hilites[CLR_YELLOW] = nh_HI;
hilites[CLR_BRIGHT_BLUE] = nh_HI;
hilites[CLR_BRIGHT_MAGENTA] = nh_HI;
hilites[CLR_BRIGHT_CYAN] = nh_HI;
hilites[CLR_WHITE] = nh_HI;
return;
}
length_md = strlen(MD);
c = 6;
while (c--)
{
scratch = tparm(setf,ti_map[c].ti_color);
hilites[ti_map[c].nh_color] = (char *) alloc(strlen(scratch) + 1);
Strcpy(hilites[ti_map[c].nh_color], scratch);
if (colors >= 16) {
/* Use proper bright colors if terminal supports them. */
scratch = tparm(setf,ti_map[c].ti_color|BRIGHT);
hilites[ti_map[c].nh_bright_color] = (char *) alloc(strlen(scratch) + 1);
Strcpy(hilites[ti_map[c].nh_bright_color], scratch);
} else {
/* For terminals supporting only 8 colors, use bold + color for
* bright colors. */
hilites[ti_map[c].nh_bright_color] = (char *) alloc(strlen(scratch) + length_md + 1);
Strcpy(hilites[ti_map[c].nh_bright_color], MD);
Strcat(hilites[ti_map[c].nh_bright_color], scratch);
}
}
if (colors >= 16) {
scratch = tparm(setf,COLOR_WHITE|BRIGHT);
hilites[CLR_WHITE] = (char *) alloc(strlen(scratch) + 1);
Strcpy(hilites[CLR_WHITE],scratch);
} else {
scratch = tparm(setf,COLOR_WHITE);
hilites[CLR_WHITE] = (char *) alloc(strlen(scratch) + length_md + 1);
Strcpy(hilites[CLR_WHITE],MD);
Strcat(hilites[CLR_WHITE],scratch);
}
hilites[CLR_GRAY] = "";
hilites[NO_COLOR] = "";
if (iflags.wc2_darkgray)
{
/* On many terminals, esp. those using classic PC CGA/EGA/VGA
* textmode, specifying "hilight" and "black" simultaneously
* produces a dark shade of gray that is visible against a
* black background. We can use it to represent black objects.
*/
if (colors >= 16) {
scratch = tparm(setf,COLOR_BLACK|BRIGHT);
hilites[CLR_BLACK] = (char *) alloc(strlen(scratch) + 1);
Strcpy(hilites[CLR_BLACK],scratch);
} else {
scratch = tparm(setf,COLOR_BLACK);
hilites[CLR_BLACK] = (char *) alloc(strlen(scratch) + length_md + 1);
Strcpy(hilites[CLR_BLACK],MD);
Strcat(hilites[CLR_BLACK],scratch);
}
}
else
{
/* But it's concievable that hilighted black-on-black could
* still be invisible on many others. We substitute blue for
* black.
*/
hilites[CLR_BLACK] = hilites[CLR_BLUE];
}
}
static void
kill_hilite(void)
{
/* if colors weren't usable, no freeing needed */
if (hilites[CLR_BLACK] == nh_HI)
return;
if (hilites[CLR_BLACK] != hilites[CLR_BLUE])
free(hilites[CLR_BLACK]);
/* CLR_BLUE overlaps CLR_BRIGHT_BLUE, do not free */
/* CLR_GREEN overlaps CLR_BRIGHT_GREEN, do not free */
/* CLR_CYAN overlaps CLR_BRIGHT_CYAN, do not free */
/* CLR_RED overlaps CLR_ORANGE, do not free */
/* CLR_MAGENTA overlaps CLR_BRIGHT_MAGENTA, do not free */
/* CLR_BROWN overlaps CLR_YELLOW, do not free */
/* CLR_GRAY is a constant "", do not free */
/* NO_COLOR is a constant "", do not free */
free(hilites[CLR_BRIGHT_BLUE]);
free(hilites[CLR_BRIGHT_GREEN]);
free(hilites[CLR_BRIGHT_CYAN]);
free(hilites[CLR_YELLOW]);
free(hilites[CLR_ORANGE]);
free(hilites[CLR_BRIGHT_MAGENTA]);
free(hilites[CLR_WHITE]);
}
# else /* UNIX && TERMINFO */
/* find the foreground and background colors set by nh_HI or nh_HE */
static void
analyze_seq (char *str, int *fg, int *bg)
{
register int c, code;
int len;
*fg = *bg = NO_COLOR;
c = (str[0] == '\233') ? 1 : 2; /* index of char beyond esc prefix */
len = strlen(str) - 1; /* length excluding attrib suffix */
if ((c != 1 && (str[0] != '\033' || str[1] != '[')) ||
(len - c) < 1 || str[len] != 'm')
return;
while (c < len) {
if ((code = atoi(&str[c])) == 0) { /* reset */
/* this also catches errors */
*fg = *bg = NO_COLOR;
} else if (code == 1) { /* bold */
*fg |= BRIGHT;
# if 0
/* I doubt we'll ever resort to using blinking characters,
unless we want a pulsing glow for something. But, in case
we do... - 3. */
} else if (code == 5) { /* blinking */
*fg |= BLINK;
} else if (code == 25) { /* stop blinking */
*fg &= ~BLINK;
# endif
} else if (code == 7 || code == 27) { /* reverse */
code = *fg & ~BRIGHT;
*fg = *bg | (*fg & BRIGHT);
*bg = code;
} else if (code >= 30 && code <= 37) { /* hi_foreground RGB */
*fg = code - 30;
} else if (code >= 40 && code <= 47) { /* hi_background RGB */
*bg = code - 40;
}
while (digit(str[++c]));
c++;
}
}
/*
* Sets up highlighting sequences, using ANSI escape sequences (highlight code
* found in print.c). The nh_HI and nh_HE sequences (usually from SO) are
* scanned to find foreground and background colors.
*/
static void
init_hilite(void)
{
register int c;
int backg, foreg, hi_backg, hi_foreg;
for (c = 0; c < SIZE(hilites); c++)
hilites[c] = nh_HI;
hilites[CLR_GRAY] = hilites[NO_COLOR] = (char *)0;
analyze_seq(nh_HI, &hi_foreg, &hi_backg);
analyze_seq(nh_HE, &foreg, &backg);
for (c = 0; c < SIZE(hilites); c++)
/* avoid invisibility */
if ((backg & ~BRIGHT) != c) {
if (c == foreg)
hilites[c] = (char *)0;
else if (c != hi_foreg || backg != hi_backg) {
hilites[c] = (char *) alloc(sizeof("\033[%d;3%d;4%dm"));
Sprintf(hilites[c], "\033[%d", !!(c & BRIGHT));
if ((c | BRIGHT) != (foreg | BRIGHT))
Sprintf(eos(hilites[c]), ";3%d", c & ~BRIGHT);
if (backg != CLR_BLACK)
Sprintf(eos(hilites[c]), ";4%d", backg & ~BRIGHT);
Strcat(hilites[c], "m");
}
}
}
static void
kill_hilite(void)
{
register int c;
for (c = 0; c < CLR_MAX / 2; c++) {
if (hilites[c|BRIGHT] == hilites[c]) hilites[c|BRIGHT] = 0;
if (hilites[c] && (hilites[c] != nh_HI))
free((void *) hilites[c]), hilites[c] = 0;
if (hilites[c|BRIGHT] && (hilites[c|BRIGHT] != nh_HI))
free((void *) hilites[c|BRIGHT]), hilites[c|BRIGHT] = 0;
}
return;
}
# endif /* UNIX */
static char nulstr[] = "";
static char *
s_atr2str(int n)
{
switch (n) {
case ATR_ULINE:
if(nh_US) return nh_US;
case ATR_BOLD:
case ATR_BLINK:
if (MD) return MD;
return nh_HI;
case ATR_INVERSE:
return MR;
}
return nulstr;
}
static char *
e_atr2str(int n)
{
switch (n) {
case ATR_ULINE:
if(nh_UE) return nh_UE;
case ATR_BOLD:
case ATR_BLINK:
return nh_HE;
case ATR_INVERSE:
return ME;
}
return nulstr;
}
void
term_start_attr(int attr)
{
if (attr) {
xputs(s_atr2str(attr));
}
}
void
term_end_attr(int attr)
{
if(attr) {
xputs(e_atr2str(attr));
}
}
void
term_start_raw_bold(void)
{
xputs(nh_HI);
}
void
term_end_raw_bold(void)
{
xputs(nh_HE);
}
void
term_start_bgcolor(int color)
{
if (allow_bgcolor) {
char tmp[8];
Sprintf(tmp, "\033[%dm", ((color % 8) + 40));
xputs(tmp);
} else {
xputs(e_atr2str(ATR_INVERSE));
}
}
void
term_end_color(void)
{
xputs(nh_HE);
}
void
term_start_color(int color)
{
xputs(hilites[color]);
}
int
has_color(int color)
{
#ifdef CURSES_GRAPHICS
/* XXX has_color() should be added to windowprocs */
/* iflags.wc_color is set to false and the option disabled if the
terminal cannot display color */
if (windowprocs.name != NULL &&
!strcmpi(windowprocs.name, "curses")) return iflags.wc_color;
#endif
return hilites[color] != (char *)0;
}
#endif /* TTY_GRAPHICS */
/*termcap.c*/