1 /*        $NetBSD: screen.c,v 1.5 2023/10/06 05:49:49 simonb Exp $    */
2 
3 /*
4  * Copyright (C) 1984-2023  Mark Nudelman
5  *
6  * You may distribute under the terms of either the GNU General Public
7  * License or the Less License, as specified in the README file.
8  *
9  * For more information, see the README file.
10  */
11 
12 
13 /*
14  * Routines which deal with the characteristics of the terminal.
15  * Uses termcap to be as terminal-independent as possible.
16  */
17 
18 #include "less.h"
19 #include "cmd.h"
20 
21 #if MSDOS_COMPILER
22 #include "pckeys.h"
23 #if MSDOS_COMPILER==MSOFTC
24 #include <graph.h>
25 #else
26 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
27 #include <conio.h>
28 #if MSDOS_COMPILER==DJGPPC
29 #include <pc.h>
30 extern int fd0;
31 #endif
32 #else
33 #if MSDOS_COMPILER==WIN32C
34 #include <windows.h>
35 #endif
36 #endif
37 #endif
38 #include <time.h>
39 
40 #ifndef FOREGROUND_BLUE
41 #define FOREGROUND_BLUE      0x0001
42 #endif
43 #ifndef FOREGROUND_GREEN
44 #define FOREGROUND_GREEN     0x0002
45 #endif
46 #ifndef FOREGROUND_RED
47 #define FOREGROUND_RED       0x0004
48 #endif
49 #ifndef FOREGROUND_INTENSITY
50 #define FOREGROUND_INTENSITY 0x0008
51 #endif
52 
53 #else
54 
55 #if HAVE_SYS_IOCTL_H
56 #include <sys/ioctl.h>
57 #endif
58 
59 #if HAVE_TERMIOS_H && HAVE_TERMIOS_FUNCS
60 #include <termios.h>
61 #else
62 #if HAVE_TERMIO_H
63 #include <termio.h>
64 #else
65 #if HAVE_SGSTAT_H
66 #include <sgstat.h>
67 #else
68 #include <sgtty.h>
69 #endif
70 #endif
71 #endif
72 
73 #if HAVE_NCURSESW_TERMCAP_H
74 #include <ncursesw/termcap.h>
75 #else
76 #if HAVE_NCURSES_TERMCAP_H
77 #include <ncurses/termcap.h>
78 #else
79 #if HAVE_TERMCAP_H
80 #include <termcap.h>
81 #endif
82 #endif
83 #endif
84 #ifdef _OSK
85 #include <signal.h>
86 #endif
87 #if OS2
88 #include <sys/signal.h>
89 #include "pckeys.h"
90 #endif
91 #if HAVE_SYS_STREAM_H
92 #include <sys/stream.h>
93 #endif
94 #if HAVE_SYS_PTEM_H
95 #include <sys/ptem.h>
96 #endif
97 
98 #endif /* MSDOS_COMPILER */
99 
100 /*
101  * Check for broken termios package that forces you to manually
102  * set the line discipline.
103  */
104 #ifdef __ultrix__
105 #define MUST_SET_LINE_DISCIPLINE 1
106 #else
107 #define MUST_SET_LINE_DISCIPLINE 0
108 #endif
109 
110 #if OS2
111 #define DEFAULT_TERM            "ansi"
112 static char *windowid;
113 #else
114 #define DEFAULT_TERM            "unknown"
115 #endif
116 
117 #if MSDOS_COMPILER==MSOFTC
118 static int videopages;
119 static long msec_loops;
120 static int flash_created = 0;
121 #define SET_FG_COLOR(fg)        _settextcolor(fg)
122 #define SET_BG_COLOR(bg)        _setbkcolor(bg)
123 #define SETCOLORS(fg,bg)        { SET_FG_COLOR(fg); SET_BG_COLOR(bg); }
124 #endif
125 
126 #if MSDOS_COMPILER==BORLANDC
127 static unsigned short *whitescreen;
128 static int flash_created = 0;
129 #endif
130 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
131 #define _settextposition(y,x)   gotoxy(x,y)
132 #define _clearscreen(m)         clrscr()
133 #define _outtext(s)             cputs(s)
134 #define SET_FG_COLOR(fg)        textcolor(fg)
135 #define SET_BG_COLOR(bg)        textbackground(bg)
136 #define SETCOLORS(fg,bg)        { SET_FG_COLOR(fg); SET_BG_COLOR(bg); }
137 extern int sc_height;
138 #endif
139 
140 #if MSDOS_COMPILER==WIN32C
141 #define UTF8_MAX_LENGTH 4
142 struct keyRecord
143 {
144           WCHAR unicode;
145           int ascii;
146           int scan;
147 } currentKey;
148 
149 static int keyCount = 0;
150 static WORD curr_attr;
151 static int pending_scancode = 0;
152 static char x11mousebuf[] = "[M???";    /* Mouse report, after ESC */
153 static int x11mousePos, x11mouseCount;
154 static int win_unget_pending = FALSE;
155 static int win_unget_data;
156 
157 static HANDLE con_out_save = INVALID_HANDLE_VALUE; /* previous console */
158 static HANDLE con_out_ours = INVALID_HANDLE_VALUE; /* our own */
159 HANDLE con_out = INVALID_HANDLE_VALUE;             /* current console */
160 
161 extern int utf_mode;
162 extern int quitting;
163 static void win32_init_term();
164 static void win32_deinit_term();
165 
166 #ifndef ENABLE_VIRTUAL_TERMINAL_PROCESSING
167 #define ENABLE_VIRTUAL_TERMINAL_PROCESSING 4
168 #endif
169 
170 #define FG_COLORS       (FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE | FOREGROUND_INTENSITY)
171 #define BG_COLORS       (BACKGROUND_RED | BACKGROUND_GREEN | BACKGROUND_BLUE | BACKGROUND_INTENSITY)
172 #define MAKEATTR(fg,bg)         ((WORD)((fg)|((bg)<<4)))
173 #define APPLY_COLORS()          { if (SetConsoleTextAttribute(con_out, curr_attr) == 0) \
174                                   error("SETCOLORS failed", NULL_PARG); }
175 #define SET_FG_COLOR(fg)        { curr_attr &= ~0x0f; curr_attr |= (fg); APPLY_COLORS(); }
176 #define SET_BG_COLOR(bg)        { curr_attr &= ~0xf0; curr_attr |= ((bg)<<4); APPLY_COLORS(); }
177 #define SETCOLORS(fg,bg)        { curr_attr = MAKEATTR(fg,bg); APPLY_COLORS(); }
178 #endif
179 
180 #if MSDOS_COMPILER
181 public int nm_fg_color;         /* Color of normal text */
182 public int nm_bg_color;
183 public int bo_fg_color;         /* Color of bold text */
184 public int bo_bg_color;
185 public int ul_fg_color;         /* Color of underlined text */
186 public int ul_bg_color;
187 public int so_fg_color;         /* Color of standout text */
188 public int so_bg_color;
189 public int bl_fg_color;         /* Color of blinking text */
190 public int bl_bg_color;
191 static int sy_fg_color;         /* Color of system text (before less) */
192 static int sy_bg_color;
193 public int sgr_mode;            /* Honor ANSI sequences rather than using above */
194 #if MSDOS_COMPILER==WIN32C
195 static DWORD init_output_mode;  /* The initial console output mode */
196 public int vt_enabled = -1;     /* Is virtual terminal processing available? */
197 #endif
198 #else
199 
200 /*
201  * Strings passed to tputs() to do various terminal functions.
202  */
203 static char
204           *sc_pad,                /* Pad string */
205           *sc_home,               /* Cursor home */
206           *sc_addline,            /* Add line, scroll down following lines */
207           *sc_lower_left,         /* Cursor to last line, first column */
208           *sc_return,             /* Cursor to beginning of current line */
209           *sc_move,               /* General cursor positioning */
210           *sc_clear,              /* Clear screen */
211           *sc_eol_clear,          /* Clear to end of line */
212           *sc_eos_clear,          /* Clear to end of screen */
213           *sc_s_in,               /* Enter standout (highlighted) mode */
214           *sc_s_out,              /* Exit standout mode */
215           *sc_u_in,               /* Enter underline mode */
216           *sc_u_out,              /* Exit underline mode */
217           *sc_b_in,               /* Enter bold mode */
218           *sc_b_out,              /* Exit bold mode */
219           *sc_bl_in,              /* Enter blink mode */
220           *sc_bl_out,             /* Exit blink mode */
221           *sc_visual_bell,        /* Visual bell (flash screen) sequence */
222           *sc_backspace,          /* Backspace cursor */
223           *sc_s_keypad,           /* Start keypad mode */
224           *sc_e_keypad,           /* End keypad mode */
225           *sc_s_mousecap,         /* Start mouse capture mode */
226           *sc_e_mousecap,         /* End mouse capture mode */
227           *sc_init,               /* Startup terminal initialization */
228           *sc_deinit;             /* Exit terminal de-initialization */
229 
230 static int attrcolor = -1;
231 #endif
232 
233 static int init_done = 0;
234 
235 public int auto_wrap;           /* Terminal does \r\n when write past margin */
236 public int ignaw;               /* Terminal ignores \n immediately after wrap */
237 public int erase_char;          /* The user's erase char */
238 public int erase2_char;         /* The user's other erase char */
239 public int kill_char;           /* The user's line-kill char */
240 public int werase_char;         /* The user's word-erase char */
241 public int sc_width, sc_height; /* Height & width of screen */
242 public int bo_s_width, bo_e_width;      /* Printing width of boldface seq */
243 public int ul_s_width, ul_e_width;      /* Printing width of underline seq */
244 public int so_s_width, so_e_width;      /* Printing width of standout seq */
245 public int bl_s_width, bl_e_width;      /* Printing width of blink seq */
246 public int above_mem, below_mem;        /* Memory retained above/below screen */
247 public int can_goto_line;               /* Can move cursor to any line */
248 public int clear_bg;            /* Clear fills with background color */
249 public int missing_cap = 0;     /* Some capability is missing */
250 public char *kent = NULL;       /* Keypad ENTER sequence */
251 public int term_init_done = FALSE;
252 public int full_screen = TRUE;
253 
254 static int attrmode = AT_NORMAL;
255 static int termcap_debug = -1;
256 static int no_alt_screen;       /* sc_init does not switch to alt screen */
257 extern int binattr;
258 extern int one_screen;
259 #if LESSTEST
260 extern char *ttyin_name;
261 #endif /*LESSTEST*/
262 
263 #if !MSDOS_COMPILER
264 static char *cheaper(char *t1, char *t2, char *def);
265 static void tmodes(char *incap, char *outcap, char **instr,
266     char **outstr, char *def_instr, char *def_outstr, char **spp);
267 #endif
268 
269 /*
270  * These two variables are sometimes defined in,
271  * and needed by, the termcap library.
272  */
273 #if MUST_DEFINE_OSPEED
274 extern short ospeed;    /* Terminal output baud rate */
275 extern char PC;         /* Pad character */
276 #endif
277 #ifdef _OSK
278 short ospeed;
279 char PC_, *UP, *BC;
280 #endif
281 
282 extern int quiet;               /* If VERY_QUIET, use visual bell for bell */
283 extern int no_vbell;
284 extern int no_back_scroll;
285 extern int swindow;
286 extern int no_init;
287 extern int quit_at_eof;
288 extern int more_mode;
289 extern int no_keypad;
290 extern int sigs;
291 extern int wscroll;
292 extern int screen_trashed;
293 extern int top_scroll;
294 extern int quit_if_one_screen;
295 extern int oldbot;
296 extern int mousecap;
297 extern int is_tty;
298 extern int use_color;
299 #if HILITE_SEARCH
300 extern int hilite_search;
301 #endif
302 #if MSDOS_COMPILER==WIN32C
303 extern HANDLE tty;
304 extern DWORD console_mode;
305 #ifndef ENABLE_EXTENDED_FLAGS
306 #define ENABLE_EXTENDED_FLAGS 0x80
307 #define ENABLE_QUICK_EDIT_MODE 0x40
308 #endif
309 #else
310 extern int tty;
311 #endif
312 
313 #if (HAVE_TERMIOS_H && HAVE_TERMIOS_FUNCS) || defined(TCGETA)
314 /*
315  * Set termio flags for use by less.
316  */
set_termio_flags(struct termios * s)317 static void set_termio_flags(
318 #if HAVE_TERMIOS_H && HAVE_TERMIOS_FUNCS
319           struct termios *s
320 #else
321           struct termio *s
322 #endif
323           )
324 {
325           s->c_lflag &= ~(0
326 #ifdef ICANON
327                     | ICANON
328 #endif
329 #ifdef ECHO
330                     | ECHO
331 #endif
332 #ifdef ECHOE
333                     | ECHOE
334 #endif
335 #ifdef ECHOK
336                     | ECHOK
337 #endif
338 #ifdef ECHONL
339                     | ECHONL
340 #endif
341           );
342 
343           s->c_oflag |= (0
344 #ifdef OXTABS
345                     | OXTABS
346 #else
347 #ifdef TAB3
348                     | TAB3
349 #else
350 #ifdef XTABS
351                     | XTABS
352 #endif
353 #endif
354 #endif
355 #ifdef OPOST
356                     | OPOST
357 #endif
358 #ifdef ONLCR
359                     | ONLCR
360 #endif
361           );
362 
363           s->c_oflag &= ~(0
364 #ifdef ONOEOT
365                     | ONOEOT
366 #endif
367 #ifdef OCRNL
368                     | OCRNL
369 #endif
370 #ifdef ONOCR
371                     | ONOCR
372 #endif
373 #ifdef ONLRET
374                     | ONLRET
375 #endif
376           );
377 }
378 #endif
379 
380 /*
381  * Change terminal to "raw mode", or restore to "normal" mode.
382  * "Raw mode" means
383  *      1. An outstanding read will complete on receipt of a single keystroke.
384  *      2. Input is not echoed.
385  *      3. On output, \n is mapped to \r\n.
386  *      4. \t is NOT expanded into spaces.
387  *      5. Signal-causing characters such as ctrl-C (interrupt),
388  *         etc. are NOT disabled.
389  * It doesn't matter whether an input \n is mapped to \r, or vice versa.
390  */
raw_mode(int on)391 public void raw_mode(int on)
392 {
393           static int curr_on = 0;
394 
395           if (on == curr_on)
396                               return;
397           erase2_char = '\b'; /* in case OS doesn't know about erase2 */
398 #if LESSTEST
399           if (ttyin_name != NULL)
400           {
401                     /* {{ For consistent conditions when running tests. }} */
402                     erase_char = '\b';
403                     kill_char = CONTROL('U');
404                     werase_char = CONTROL('W');
405           } else
406 #endif /*LESSTEST*/
407 #if HAVE_TERMIOS_H && HAVE_TERMIOS_FUNCS
408     {
409           struct termios s;
410           static struct termios save_term;
411           static int saved_term = 0;
412 
413           if (on)
414           {
415                     /*
416                      * Get terminal modes.
417                      */
418                     if (tcgetattr(tty, &s) < 0)
419                     {
420                               erase_char = '\b';
421                               kill_char = CONTROL('U');
422                               werase_char = CONTROL('W');
423                     } else
424                     {
425                               /*
426                                * Save modes and set certain variables dependent on modes.
427                                */
428                               if (!saved_term)
429                               {
430                                         save_term = s;
431                                         saved_term = 1;
432                               }
433 #if HAVE_OSPEED
434                               switch (cfgetospeed(&s))
435                               {
436 #ifdef B0
437                               case B0: ospeed = 0; break;
438 #endif
439 #ifdef B50
440                               case B50: ospeed = 1; break;
441 #endif
442 #ifdef B75
443                               case B75: ospeed = 2; break;
444 #endif
445 #ifdef B110
446                               case B110: ospeed = 3; break;
447 #endif
448 #ifdef B134
449                               case B134: ospeed = 4; break;
450 #endif
451 #ifdef B150
452                               case B150: ospeed = 5; break;
453 #endif
454 #ifdef B200
455                               case B200: ospeed = 6; break;
456 #endif
457 #ifdef B300
458                               case B300: ospeed = 7; break;
459 #endif
460 #ifdef B600
461                               case B600: ospeed = 8; break;
462 #endif
463 #ifdef B1200
464                               case B1200: ospeed = 9; break;
465 #endif
466 #ifdef B1800
467                               case B1800: ospeed = 10; break;
468 #endif
469 #ifdef B2400
470                               case B2400: ospeed = 11; break;
471 #endif
472 #ifdef B4800
473                               case B4800: ospeed = 12; break;
474 #endif
475 #ifdef B9600
476                               case B9600: ospeed = 13; break;
477 #endif
478 #ifdef EXTA
479                               case EXTA: ospeed = 14; break;
480 #endif
481 #ifdef EXTB
482                               case EXTB: ospeed = 15; break;
483 #endif
484 #ifdef B57600
485                               case B57600: ospeed = 16; break;
486 #endif
487 #ifdef B115200
488                               case B115200: ospeed = 17; break;
489 #endif
490                               default: ;
491                               }
492 #endif
493                               erase_char = s.c_cc[VERASE];
494 #ifdef VERASE2
495                               erase2_char = s.c_cc[VERASE2];
496 #endif
497                               kill_char = s.c_cc[VKILL];
498 #ifdef VWERASE
499                               werase_char = s.c_cc[VWERASE];
500 #else
501                               werase_char = CONTROL('W');
502 #endif
503 
504                               /*
505                                * Set the modes to the way we want them.
506                                */
507                               set_termio_flags(&s);
508                               s.c_cc[VMIN] = 1;
509                               s.c_cc[VTIME] = 0;
510 #ifdef VLNEXT
511                               s.c_cc[VLNEXT] = 0;
512 #endif
513 #ifdef VDSUSP
514                               s.c_cc[VDSUSP] = 0;
515 #endif
516 #ifdef VSTOP
517                               s.c_cc[VSTOP] = 0;
518 #endif
519 #ifdef VSTART
520                               s.c_cc[VSTART] = 0;
521 #endif
522 #if MUST_SET_LINE_DISCIPLINE
523                               /*
524                                * System's termios is broken; need to explicitly
525                                * request TERMIODISC line discipline.
526                                */
527                               s.c_line = TERMIODISC;
528 #endif
529                     }
530           } else
531           {
532                     /*
533                      * Restore saved modes.
534                      */
535                     s = save_term;
536           }
537 #if HAVE_FSYNC
538           fsync(tty);
539 #endif
540           tcsetattr(tty, TCSADRAIN, &s);
541 #if MUST_SET_LINE_DISCIPLINE
542           if (!on)
543           {
544                     /*
545                      * Broken termios *ignores* any line discipline
546                      * except TERMIODISC.  A different old line discipline
547                      * is therefore not restored, yet.  Restore the old
548                      * line discipline by hand.
549                      */
550                     ioctl(tty, TIOCSETD, &save_term.c_line);
551           }
552 #endif
553     }
554 #else
555 #ifdef TCGETA
556     {
557           struct termio s;
558           static struct termio save_term;
559           static int saved_term = 0;
560 
561           if (on)
562           {
563                     /*
564                      * Get terminal modes.
565                      */
566                     ioctl(tty, TCGETA, &s);
567 
568                     /*
569                      * Save modes and set certain variables dependent on modes.
570                      */
571                     if (!saved_term)
572                     {
573                               save_term = s;
574                               saved_term = 1;
575                     }
576 #if HAVE_OSPEED
577                     ospeed = s.c_cflag & CBAUD;
578 #endif
579                     erase_char = s.c_cc[VERASE];
580                     kill_char = s.c_cc[VKILL];
581 #ifdef VWERASE
582                     werase_char = s.c_cc[VWERASE];
583 #else
584                     werase_char = CONTROL('W');
585 #endif
586 
587                     /*
588                      * Set the modes to the way we want them.
589                      */
590                     set_termio_flags(&s);
591                     s.c_cc[VMIN] = 1;
592                     s.c_cc[VTIME] = 0;
593 #ifdef VSTOP
594                     s.c_cc[VSTOP] = 0;
595 #endif
596 #ifdef VSTART
597                     s.c_cc[VSTART] = 0;
598 #endif
599           } else
600           {
601                     /*
602                      * Restore saved modes.
603                      */
604                     s = save_term;
605           }
606           ioctl(tty, TCSETAW, &s);
607     }
608 #else
609 #ifdef TIOCGETP
610     {
611           struct sgttyb s;
612           static struct sgttyb save_term;
613           static int saved_term = 0;
614 
615           if (on)
616           {
617                     /*
618                      * Get terminal modes.
619                      */
620                     ioctl(tty, TIOCGETP, &s);
621 
622                     /*
623                      * Save modes and set certain variables dependent on modes.
624                      */
625                     if (!saved_term)
626                     {
627                               save_term = s;
628                               saved_term = 1;
629                     }
630 #if HAVE_OSPEED
631                     ospeed = s.sg_ospeed;
632 #endif
633                     erase_char = s.sg_erase;
634                     kill_char = s.sg_kill;
635                     werase_char = CONTROL('W');
636 
637                     /*
638                      * Set the modes to the way we want them.
639                      */
640                     s.sg_flags |= CBREAK;
641                     s.sg_flags &= ~(ECHO|XTABS);
642           } else
643           {
644                     /*
645                      * Restore saved modes.
646                      */
647                     s = save_term;
648           }
649           ioctl(tty, TIOCSETN, &s);
650     }
651 #else
652 #ifdef _OSK
653     {
654           struct sgbuf s;
655           static struct sgbuf save_term;
656           static int saved_term = 0;
657 
658           if (on)
659           {
660                     /*
661                      * Get terminal modes.
662                      */
663                     _gs_opt(tty, &s);
664 
665                     /*
666                      * Save modes and set certain variables dependent on modes.
667                      */
668                     if (!saved_term)
669                     {
670                               save_term = s;
671                               saved_term = 1;
672                     }
673                     erase_char = s.sg_bspch;
674                     kill_char = s.sg_dlnch;
675                     werase_char = CONTROL('W');
676 
677                     /*
678                      * Set the modes to the way we want them.
679                      */
680                     s.sg_echo = 0;
681                     s.sg_eofch = 0;
682                     s.sg_pause = 0;
683                     s.sg_psch = 0;
684           } else
685           {
686                     /*
687                      * Restore saved modes.
688                      */
689                     s = save_term;
690           }
691           _ss_opt(tty, &s);
692     }
693 #else
694           /* MS-DOS, Windows, or OS2 */
695 #if OS2
696           /* OS2 */
697           LSIGNAL(SIGINT, SIG_IGN);
698 #endif
699           erase_char = '\b';
700 #if MSDOS_COMPILER==DJGPPC
701           kill_char = CONTROL('U');
702           /*
703            * So that when we shell out or run another program, its
704            * stdin is in cooked mode.  We do not switch stdin to binary
705            * mode if fd0 is zero, since that means we were called before
706            * tty was reopened in open_getchr, in which case we would be
707            * changing the original stdin device outside less.
708            */
709           if (fd0 != 0)
710                     setmode(0, on ? O_BINARY : O_TEXT);
711 #else
712           kill_char = ESC;
713 #endif
714           werase_char = CONTROL('W');
715 #endif
716 #endif
717 #endif
718 #endif
719           curr_on = on;
720 }
721 
722 #if !MSDOS_COMPILER
723 /*
724  * Some glue to prevent calling termcap functions if tgetent() failed.
725  */
726 static int hardcopy;
727 
ltget_env(char * capname)728 static char * ltget_env(char *capname)
729 {
730           char name[64];
731 
732           if (termcap_debug)
733           {
734                     struct env { struct env *next; char *name; char *value; };
735                     static struct env *envs = NULL;
736                     struct env *p;
737                     for (p = envs;  p != NULL;  p = p->next)
738                               if (strcmp(p->name, capname) == 0)
739                                         return p->value;
740                     p = (struct env *) ecalloc(1, sizeof(struct env));
741                     p->name = save(capname);
742                     p->value = (char *) ecalloc(strlen(capname)+3, sizeof(char));
743                     sprintf(p->value, "<%s>", capname);
744                     p->next = envs;
745                     envs = p;
746                     return p->value;
747           }
748           SNPRINTF1(name, sizeof(name), "LESS_TERMCAP_%s", capname);
749           return (lgetenv(name));
750 }
751 
ltgetflag(char * capname)752 static int ltgetflag(char *capname)
753 {
754           char *s;
755 
756           if ((s = ltget_env(capname)) != NULL)
757                     return (*s != '\0' && *s != '0');
758           if (hardcopy)
759                     return (0);
760           return (tgetflag(capname));
761 }
762 
ltgetnum(char * capname)763 static int ltgetnum(char *capname)
764 {
765           char *s;
766 
767           if ((s = ltget_env(capname)) != NULL)
768                     return (atoi(s));
769           if (hardcopy)
770                     return (-1);
771           return (tgetnum(capname));
772 }
773 
ltgetstr(char * capname,char ** pp)774 static char * ltgetstr(char *capname, char **pp)
775 {
776           char *s;
777 
778           if ((s = ltget_env(capname)) != NULL)
779                     return (s);
780           if (hardcopy)
781                     return (NULL);
782           return (tgetstr(capname, pp));
783 }
784 #endif /* MSDOS_COMPILER */
785 
786 /*
787  * Get size of the output screen.
788  */
scrsize(void)789 public void scrsize(void)
790 {
791           char *s;
792           int sys_height;
793           int sys_width;
794 #if !MSDOS_COMPILER
795           int n;
796 #endif
797 
798 #define DEF_SC_WIDTH    80
799 #if MSDOS_COMPILER
800 #define DEF_SC_HEIGHT   25
801 #else
802 #define DEF_SC_HEIGHT   24
803 #endif
804 
805 
806           sys_width = sys_height = 0;
807 
808 #if LESSTEST
809           if (ttyin_name != NULL)
810 #endif /*LESSTEST*/
811           {
812 #if MSDOS_COMPILER==MSOFTC
813           {
814                     struct videoconfig w;
815                     _getvideoconfig(&w);
816                     sys_height = w.numtextrows;
817                     sys_width = w.numtextcols;
818           }
819 #else
820 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
821           {
822                     struct text_info w;
823                     gettextinfo(&w);
824                     sys_height = w.screenheight;
825                     sys_width = w.screenwidth;
826           }
827 #else
828 #if MSDOS_COMPILER==WIN32C
829           {
830                     CONSOLE_SCREEN_BUFFER_INFO scr;
831                     GetConsoleScreenBufferInfo(con_out, &scr);
832                     sys_height = scr.srWindow.Bottom - scr.srWindow.Top + 1;
833                     sys_width = scr.srWindow.Right - scr.srWindow.Left + 1;
834           }
835 #else
836 #if OS2
837           {
838                     int s[2];
839                     _scrsize(s);
840                     sys_width = s[0];
841                     sys_height = s[1];
842                     /*
843                      * When using terminal emulators for XFree86/OS2, the
844                      * _scrsize function does not work well.
845                      * Call the scrsize.exe program to get the window size.
846                      */
847                     windowid = getenv("WINDOWID");
848                     if (windowid != NULL)
849                     {
850                               FILE *fd = popen("scrsize", "rt");
851                               if (fd != NULL)
852                               {
853                                         int w, h;
854                                         fscanf(fd, "%i %i", &w, &h);
855                                         if (w > 0 && h > 0)
856                                         {
857                                                   sys_width = w;
858                                                   sys_height = h;
859                                         }
860                                         pclose(fd);
861                               }
862                     }
863           }
864 #else
865 #ifdef TIOCGWINSZ
866           {
867                     struct winsize w;
868                     if (ioctl(2, TIOCGWINSZ, &w) == 0)
869                     {
870                               if (w.ws_row > 0)
871                                         sys_height = w.ws_row;
872                               if (w.ws_col > 0)
873                                         sys_width = w.ws_col;
874                     }
875           }
876 #else
877 #ifdef WIOCGETD
878           {
879                     struct uwdata w;
880                     if (ioctl(2, WIOCGETD, &w) == 0)
881                     {
882                               if (w.uw_height > 0)
883                                         sys_height = w.uw_height / w.uw_vs;
884                               if (w.uw_width > 0)
885                                         sys_width = w.uw_width / w.uw_hs;
886                     }
887           }
888 #endif
889 #endif
890 #endif
891 #endif
892 #endif
893 #endif
894           }
895 
896           if (sys_height > 0)
897                     sc_height = sys_height;
898           else if ((s = lgetenv("LINES")) != NULL)
899                     sc_height = atoi(s);
900 #if !MSDOS_COMPILER
901           else if ((n = ltgetnum("li")) > 0)
902                     sc_height = n;
903 #endif
904           if ((s = lgetenv("LESS_LINES")) != NULL)
905           {
906                     int height = atoi(s);
907                     sc_height = (height < 0) ? sc_height + height : height;
908                     full_screen = FALSE;
909           }
910           if (sc_height <= 0)
911                     sc_height = DEF_SC_HEIGHT;
912 
913           if (sys_width > 0)
914                     sc_width = sys_width;
915           else if ((s = lgetenv("COLUMNS")) != NULL)
916                     sc_width = atoi(s);
917 #if !MSDOS_COMPILER
918           else if ((n = ltgetnum("co")) > 0)
919                     sc_width = n;
920 #endif
921           if ((s = lgetenv("LESS_COLUMNS")) != NULL)
922           {
923                     int width = atoi(s);
924                     sc_width = (width < 0) ? sc_width + width : width;
925           }
926           if (sc_width <= 0)
927                     sc_width = DEF_SC_WIDTH;
928 }
929 
930 #if MSDOS_COMPILER==MSOFTC
931 /*
932  * Figure out how many empty loops it takes to delay a millisecond.
933  */
get_clock(void)934 static void get_clock(void)
935 {
936           clock_t start;
937 
938           /*
939            * Get synchronized at the start of a tick.
940            */
941           start = clock();
942           while (clock() == start)
943                     ;
944           /*
945            * Now count loops till the next tick.
946            */
947           start = clock();
948           msec_loops = 0;
949           while (clock() == start)
950                     msec_loops++;
951           /*
952            * Convert from (loops per clock) to (loops per millisecond).
953            */
954           msec_loops *= CLOCKS_PER_SEC;
955           msec_loops /= 1000;
956 }
957 
958 /*
959  * Delay for a specified number of milliseconds.
960  */
delay(int msec)961 static void delay(int msec)
962 {
963           long i;
964 
965           while (msec-- > 0)
966           {
967                     for (i = 0;  i < msec_loops;  i++)
968                               (void) clock();
969           }
970 }
971 #endif
972 
973 /*
974  * Return the characters actually input by a "special" key.
975  */
special_key_str(int key)976 public char * special_key_str(int key)
977 {
978           static char tbuf[40];
979           char *s;
980 #if MSDOS_COMPILER || OS2
981           static char k_right[]           = { '\340', PCK_RIGHT, 0 };
982           static char k_left[]            = { '\340', PCK_LEFT, 0  };
983           static char k_ctl_right[]       = { '\340', PCK_CTL_RIGHT, 0  };
984           static char k_ctl_left[]        = { '\340', PCK_CTL_LEFT, 0  };
985           static char k_insert[]          = { '\340', PCK_INSERT, 0  };
986           static char k_delete[]          = { '\340', PCK_DELETE, 0  };
987           static char k_ctl_delete[]      = { '\340', PCK_CTL_DELETE, 0  };
988           static char k_ctl_backspace[]   = { '\177', 0 };
989           static char k_backspace[]       = { '\b', 0 };
990           static char k_home[]            = { '\340', PCK_HOME, 0 };
991           static char k_end[]             = { '\340', PCK_END, 0 };
992           static char k_up[]              = { '\340', PCK_UP, 0 };
993           static char k_down[]            = { '\340', PCK_DOWN, 0 };
994           static char k_backtab[]         = { '\340', PCK_SHIFT_TAB, 0 };
995           static char k_pagedown[]        = { '\340', PCK_PAGEDOWN, 0 };
996           static char k_pageup[]          = { '\340', PCK_PAGEUP, 0 };
997           static char k_f1[]              = { '\340', PCK_F1, 0 };
998 #endif
999 #if !MSDOS_COMPILER
1000           char *sp = tbuf;
1001 #endif
1002 
1003           switch (key)
1004           {
1005 #if OS2
1006           /*
1007            * If windowid is not NULL, assume less is executed in
1008            * the XFree86 environment.
1009            */
1010           case SK_RIGHT_ARROW:
1011                     s = windowid ? ltgetstr("kr", &sp) : k_right;
1012                     break;
1013           case SK_LEFT_ARROW:
1014                     s = windowid ? ltgetstr("kl", &sp) : k_left;
1015                     break;
1016           case SK_UP_ARROW:
1017                     s = windowid ? ltgetstr("ku", &sp) : k_up;
1018                     break;
1019           case SK_DOWN_ARROW:
1020                     s = windowid ? ltgetstr("kd", &sp) : k_down;
1021                     break;
1022           case SK_PAGE_UP:
1023                     s = windowid ? ltgetstr("kP", &sp) : k_pageup;
1024                     break;
1025           case SK_PAGE_DOWN:
1026                     s = windowid ? ltgetstr("kN", &sp) : k_pagedown;
1027                     break;
1028           case SK_HOME:
1029                     s = windowid ? ltgetstr("kh", &sp) : k_home;
1030                     break;
1031           case SK_END:
1032                     s = windowid ? ltgetstr("@7", &sp) : k_end;
1033                     break;
1034           case SK_DELETE:
1035                     s = windowid ? ltgetstr("kD", &sp) : k_delete;
1036                     if (s == NULL)
1037                     {
1038                                         tbuf[0] = '\177';
1039                                         tbuf[1] = '\0';
1040                                         s = tbuf;
1041                     }
1042                     break;
1043 #endif
1044 #if MSDOS_COMPILER
1045           case SK_RIGHT_ARROW:
1046                     s = k_right;
1047                     break;
1048           case SK_LEFT_ARROW:
1049                     s = k_left;
1050                     break;
1051           case SK_UP_ARROW:
1052                     s = k_up;
1053                     break;
1054           case SK_DOWN_ARROW:
1055                     s = k_down;
1056                     break;
1057           case SK_PAGE_UP:
1058                     s = k_pageup;
1059                     break;
1060           case SK_PAGE_DOWN:
1061                     s = k_pagedown;
1062                     break;
1063           case SK_HOME:
1064                     s = k_home;
1065                     break;
1066           case SK_END:
1067                     s = k_end;
1068                     break;
1069           case SK_DELETE:
1070                     s = k_delete;
1071                     break;
1072 #endif
1073 #if MSDOS_COMPILER || OS2
1074           case SK_INSERT:
1075                     s = k_insert;
1076                     break;
1077           case SK_CTL_LEFT_ARROW:
1078                     s = k_ctl_left;
1079                     break;
1080           case SK_CTL_RIGHT_ARROW:
1081                     s = k_ctl_right;
1082                     break;
1083           case SK_CTL_BACKSPACE:
1084                     s = k_ctl_backspace;
1085                     break;
1086           case SK_CTL_DELETE:
1087                     s = k_ctl_delete;
1088                     break;
1089           case SK_BACKSPACE:
1090                     s = k_backspace;
1091                     break;
1092           case SK_F1:
1093                     s = k_f1;
1094                     break;
1095           case SK_BACKTAB:
1096                     s = k_backtab;
1097                     break;
1098 #else
1099           case SK_RIGHT_ARROW:
1100                     s = ltgetstr("kr", &sp);
1101                     break;
1102           case SK_LEFT_ARROW:
1103                     s = ltgetstr("kl", &sp);
1104                     break;
1105           case SK_UP_ARROW:
1106                     s = ltgetstr("ku", &sp);
1107                     break;
1108           case SK_DOWN_ARROW:
1109                     s = ltgetstr("kd", &sp);
1110                     break;
1111           case SK_PAGE_UP:
1112                     s = ltgetstr("kP", &sp);
1113                     break;
1114           case SK_PAGE_DOWN:
1115                     s = ltgetstr("kN", &sp);
1116                     break;
1117           case SK_HOME:
1118                     s = ltgetstr("kh", &sp);
1119                     break;
1120           case SK_END:
1121                     s = ltgetstr("@7", &sp);
1122                     break;
1123           case SK_DELETE:
1124                     s = ltgetstr("kD", &sp);
1125                     if (s == NULL)
1126                     {
1127                                         tbuf[0] = '\177';
1128                                         tbuf[1] = '\0';
1129                                         s = tbuf;
1130                     }
1131                     break;
1132           case SK_BACKSPACE:
1133                     s = ltgetstr("kb", &sp);
1134                     if (s == NULL)
1135                     {
1136                                         tbuf[0] = '\b';
1137                                         tbuf[1] = '\0';
1138                                         s = tbuf;
1139                     }
1140                     break;
1141 #endif
1142           case SK_CONTROL_K:
1143                     tbuf[0] = CONTROL('K');
1144                     tbuf[1] = '\0';
1145                     s = tbuf;
1146                     break;
1147           default:
1148                     return (NULL);
1149           }
1150           return (s);
1151 }
1152 
1153 /*
1154  * Get terminal capabilities via termcap.
1155  */
get_term(void)1156 public void get_term(void)
1157 {
1158           termcap_debug = !isnullenv(lgetenv("LESS_TERMCAP_DEBUG"));
1159 #if MSDOS_COMPILER
1160           auto_wrap = 1;
1161           ignaw = 0;
1162           can_goto_line = 1;
1163           clear_bg = 1;
1164           /*
1165            * Set up default colors.
1166            * The xx_s_width and xx_e_width vars are already initialized to 0.
1167            */
1168 #if MSDOS_COMPILER==MSOFTC
1169           sy_bg_color = _getbkcolor();
1170           sy_fg_color = _gettextcolor();
1171           get_clock();
1172 #else
1173 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
1174     {
1175           struct text_info w;
1176           gettextinfo(&w);
1177           sy_bg_color = (w.attribute >> 4) & 0x0F;
1178           sy_fg_color = (w.attribute >> 0) & 0x0F;
1179     }
1180 #else
1181 #if MSDOS_COMPILER==WIN32C
1182     {
1183           CONSOLE_SCREEN_BUFFER_INFO scr;
1184 
1185           con_out_save = con_out = GetStdHandle(STD_OUTPUT_HANDLE);
1186           /*
1187            * Always open stdin in binary. Note this *must* be done
1188            * before any file operations have been done on fd0.
1189            */
1190           SET_BINARY(0);
1191           GetConsoleMode(con_out, &init_output_mode);
1192           GetConsoleScreenBufferInfo(con_out, &scr);
1193           curr_attr = scr.wAttributes;
1194           sy_bg_color = (curr_attr & BG_COLORS) >> 4; /* normalize */
1195           sy_fg_color = curr_attr & FG_COLORS;
1196     }
1197 #endif
1198 #endif
1199 #endif
1200           nm_fg_color = sy_fg_color;
1201           nm_bg_color = sy_bg_color;
1202           bo_fg_color = 11;
1203           bo_bg_color = 0;
1204           ul_fg_color = 9;
1205           ul_bg_color = 0;
1206           so_fg_color = 15;
1207           so_bg_color = 9;
1208           bl_fg_color = 15;
1209           bl_bg_color = 0;
1210           sgr_mode = 0;
1211 
1212           /*
1213            * Get size of the screen.
1214            */
1215           scrsize();
1216           pos_init();
1217 
1218 
1219 #else /* !MSDOS_COMPILER */
1220 {
1221           char *sp;
1222           char *t1, *t2;
1223           char *term;
1224           /*
1225            * Some termcap libraries assume termbuf is static
1226            * (accessible after tgetent returns).
1227            */
1228           static char termbuf[TERMBUF_SIZE];
1229           static char sbuf[TERMSBUF_SIZE];
1230 
1231 #if OS2
1232           /*
1233            * Make sure the termcap database is available.
1234            */
1235           sp = lgetenv("TERMCAP");
1236           if (isnullenv(sp))
1237           {
1238                     char *termcap;
1239                     if ((sp = homefile("termcap.dat")) != NULL)
1240                     {
1241                               termcap = (char *) ecalloc(strlen(sp)+9, sizeof(char));
1242                               sprintf(termcap, "TERMCAP=%s", sp);
1243                               free(sp);
1244                               putenv(termcap);
1245                     }
1246           }
1247 #endif
1248           /*
1249            * Find out what kind of terminal this is.
1250            */
1251           if ((term = lgetenv("TERM")) == NULL)
1252                     term = DEFAULT_TERM;
1253           hardcopy = 0;
1254           /* {{ Should probably just pass NULL instead of termbuf. }} */
1255           if (tgetent(termbuf, term) != TGETENT_OK)
1256                     hardcopy = 1;
1257           if (ltgetflag("hc"))
1258                     hardcopy = 1;
1259 
1260           /*
1261            * Get size of the screen.
1262            */
1263           scrsize();
1264           pos_init();
1265 
1266           auto_wrap = ltgetflag("am");
1267           ignaw = ltgetflag("xn");
1268           above_mem = ltgetflag("da");
1269           below_mem = ltgetflag("db");
1270           clear_bg = ltgetflag("ut");
1271           no_alt_screen = ltgetflag("NR");
1272 
1273           /*
1274            * Assumes termcap variable "sg" is the printing width of:
1275            * the standout sequence, the end standout sequence,
1276            * the underline sequence, the end underline sequence,
1277            * the boldface sequence, and the end boldface sequence.
1278            */
1279           if ((so_s_width = ltgetnum("sg")) < 0)
1280                     so_s_width = 0;
1281           so_e_width = so_s_width;
1282 
1283           bo_s_width = bo_e_width = so_s_width;
1284           ul_s_width = ul_e_width = so_s_width;
1285           bl_s_width = bl_e_width = so_s_width;
1286 
1287 #if HILITE_SEARCH
1288           if (so_s_width > 0 || so_e_width > 0)
1289                     /*
1290                      * Disable highlighting by default on magic cookie terminals.
1291                      * Turning on highlighting might change the displayed width
1292                      * of a line, causing the display to get messed up.
1293                      * The user can turn it back on with -g,
1294                      * but she won't like the results.
1295                      */
1296                     hilite_search = 0;
1297 #endif
1298 
1299           /*
1300            * Get various string-valued capabilities.
1301            */
1302           sp = sbuf;
1303 
1304 #if HAVE_OSPEED
1305           sc_pad = ltgetstr("pc", &sp);
1306           if (sc_pad != NULL)
1307                     PC = *sc_pad;
1308 #endif
1309 
1310           sc_s_keypad = ltgetstr("ks", &sp);
1311           if (sc_s_keypad == NULL)
1312                     sc_s_keypad = "";
1313           sc_e_keypad = ltgetstr("ke", &sp);
1314           if (sc_e_keypad == NULL)
1315                     sc_e_keypad = "";
1316           kent = ltgetstr("@8", &sp);
1317 
1318           sc_s_mousecap = ltgetstr("MOUSE_START", &sp);
1319           if (sc_s_mousecap == NULL)
1320                     sc_s_mousecap = ESCS "[?1000h" ESCS "[?1006h";
1321           sc_e_mousecap = ltgetstr("MOUSE_END", &sp);
1322           if (sc_e_mousecap == NULL)
1323                     sc_e_mousecap = ESCS "[?1006l" ESCS "[?1000l";
1324 
1325           /*
1326            * This loses for terminals with termcap entries with ti/te strings
1327            * that switch to/from an alternate screen, and we're in quit_at_eof
1328            * (eg, more(1)).
1329            */
1330           if (quit_at_eof != OPT_ONPLUS && !more_mode) {
1331                     sc_init = ltgetstr("ti", &sp);
1332                     sc_deinit = ltgetstr("te", &sp);
1333           } else {
1334                     sc_init = NULL;
1335                     sc_deinit = NULL;
1336           }
1337           if (sc_init == NULL)
1338                     sc_init = "";
1339           if (sc_deinit == NULL)
1340                     sc_deinit = "";
1341 
1342           sc_eol_clear = ltgetstr("ce", &sp);
1343           if (sc_eol_clear == NULL || *sc_eol_clear == '\0')
1344           {
1345                     missing_cap = 1;
1346                     sc_eol_clear = "";
1347           }
1348 
1349           sc_eos_clear = ltgetstr("cd", &sp);
1350           if (below_mem && (sc_eos_clear == NULL || *sc_eos_clear == '\0'))
1351           {
1352                     missing_cap = 1;
1353                     sc_eos_clear = "";
1354           }
1355 
1356           sc_clear = ltgetstr("cl", &sp);
1357           if (sc_clear == NULL || *sc_clear == '\0')
1358           {
1359                     missing_cap = 1;
1360                     sc_clear = "\n\n";
1361           }
1362 
1363           sc_move = ltgetstr("cm", &sp);
1364           if (sc_move == NULL || *sc_move == '\0')
1365           {
1366                     /*
1367                      * This is not an error here, because we don't
1368                      * always need sc_move.
1369                      * We need it only if we don't have home or lower-left.
1370                      */
1371                     sc_move = "";
1372                     can_goto_line = 0;
1373           } else
1374                     can_goto_line = 1;
1375 
1376           tmodes("so", "se", &sc_s_in, &sc_s_out, "", "", &sp);
1377           tmodes("us", "ue", &sc_u_in, &sc_u_out, sc_s_in, sc_s_out, &sp);
1378           tmodes("md", "me", &sc_b_in, &sc_b_out, sc_s_in, sc_s_out, &sp);
1379           tmodes("mb", "me", &sc_bl_in, &sc_bl_out, sc_s_in, sc_s_out, &sp);
1380 
1381           sc_visual_bell = ltgetstr("vb", &sp);
1382           if (sc_visual_bell == NULL)
1383                     sc_visual_bell = "";
1384 
1385           if (ltgetflag("bs"))
1386                     sc_backspace = "\b";
1387           else
1388           {
1389                     sc_backspace = ltgetstr("bc", &sp);
1390                     if (sc_backspace == NULL || *sc_backspace == '\0')
1391                               sc_backspace = "\b";
1392           }
1393 
1394           /*
1395            * Choose between using "ho" and "cm" ("home" and "cursor move")
1396            * to move the cursor to the upper left corner of the screen.
1397            */
1398           t1 = ltgetstr("ho", &sp);
1399           if (t1 == NULL)
1400                     t1 = "";
1401           if (*sc_move == '\0')
1402                     t2 = "";
1403           else
1404           {
1405                     strcpy(sp, tgoto(sc_move, 0, 0));
1406                     t2 = sp;
1407                     sp += strlen(sp) + 1;
1408           }
1409           sc_home = cheaper(t1, t2, "|\b^");
1410 
1411           /*
1412            * Choose between using "ll" and "cm"  ("lower left" and "cursor move")
1413            * to move the cursor to the lower left corner of the screen.
1414            */
1415           t1 = ltgetstr("ll", &sp);
1416           if (t1 == NULL || !full_screen)
1417                     t1 = "";
1418           if (*sc_move == '\0')
1419                     t2 = "";
1420           else
1421           {
1422                     strcpy(sp, tgoto(sc_move, 0, sc_height-1));
1423                     t2 = sp;
1424                     sp += strlen(sp) + 1;
1425           }
1426           sc_lower_left = cheaper(t1, t2, "\r");
1427 
1428           /*
1429            * Get carriage return string.
1430            */
1431           sc_return = ltgetstr("cr", &sp);
1432           if (sc_return == NULL)
1433                     sc_return = "\r";
1434 
1435           /*
1436            * Choose between using "al" or "sr" ("add line" or "scroll reverse")
1437            * to add a line at the top of the screen.
1438            */
1439           t1 = ltgetstr("al", &sp);
1440           if (t1 == NULL)
1441                     t1 = "";
1442           t2 = ltgetstr("sr", &sp);
1443           if (t2 == NULL)
1444                     t2 = "";
1445 #if OS2
1446           if (*t1 == '\0' && *t2 == '\0')
1447                     sc_addline = "";
1448           else
1449 #endif
1450           if (above_mem)
1451                     sc_addline = t1;
1452           else
1453                     sc_addline = cheaper(t1, t2, "");
1454           if (*sc_addline == '\0')
1455           {
1456                     /*
1457                      * Force repaint on any backward movement.
1458                      */
1459                     no_back_scroll = 1;
1460           }
1461 }
1462 #endif /* MSDOS_COMPILER */
1463 }
1464 
1465 #if !MSDOS_COMPILER
1466 /*
1467  * Return the cost of displaying a termcap string.
1468  * We use the trick of calling tputs, but as a char printing function
1469  * we give it inc_costcount, which just increments "costcount".
1470  * This tells us how many chars would be printed by using this string.
1471  * {{ Couldn't we just use strlen? }}
1472  */
1473 static int costcount;
1474 
1475 /*ARGSUSED*/
inc_costcount(int c)1476 static int inc_costcount(int c)
1477 {
1478           costcount++;
1479           return (c);
1480 }
1481 
cost(char * t)1482 static int cost(char *t)
1483 {
1484           costcount = 0;
1485           tputs(t, sc_height, inc_costcount);
1486           return (costcount);
1487 }
1488 
1489 /*
1490  * Return the "best" of the two given termcap strings.
1491  * The best, if both exist, is the one with the lower
1492  * cost (see cost() function).
1493  */
cheaper(char * t1,char * t2,char * def)1494 static char * cheaper(char *t1, char *t2, char *def)
1495 {
1496           if (*t1 == '\0' && *t2 == '\0')
1497           {
1498                     missing_cap = 1;
1499                     return (def);
1500           }
1501           if (*t1 == '\0')
1502                     return (t2);
1503           if (*t2 == '\0')
1504                     return (t1);
1505           if (cost(t1) < cost(t2))
1506                     return (t1);
1507           return (t2);
1508 }
1509 
tmodes(char * incap,char * outcap,char ** instr,char ** outstr,char * def_instr,char * def_outstr,char ** spp)1510 static void tmodes(char *incap, char *outcap, char **instr, char **outstr, char *def_instr, char *def_outstr, char **spp)
1511 {
1512           *instr = ltgetstr(incap, spp);
1513           if (*instr == NULL)
1514           {
1515                     /* Use defaults. */
1516                     *instr = def_instr;
1517                     *outstr = def_outstr;
1518                     return;
1519           }
1520 
1521           *outstr = ltgetstr(outcap, spp);
1522           if (*outstr == NULL)
1523                     /* No specific out capability; use "me". */
1524                     *outstr = ltgetstr("me", spp);
1525           if (*outstr == NULL)
1526                     /* Don't even have "me"; use a null string. */
1527                     *outstr = "";
1528 }
1529 
1530 #endif /* MSDOS_COMPILER */
1531 
1532 
1533 /*
1534  * Below are the functions which perform all the
1535  * terminal-specific screen manipulation.
1536  */
1537 
1538 
1539 #if MSDOS_COMPILER
1540 
1541 #if MSDOS_COMPILER==WIN32C
_settextposition(int row,int col)1542 static void _settextposition(int row, int col)
1543 {
1544           COORD cpos;
1545           CONSOLE_SCREEN_BUFFER_INFO csbi;
1546 
1547           GetConsoleScreenBufferInfo(con_out, &csbi);
1548           cpos.X = csbi.srWindow.Left + (col - 1);
1549           cpos.Y = csbi.srWindow.Top + (row - 1);
1550           SetConsoleCursorPosition(con_out, cpos);
1551 }
1552 #endif
1553 
1554 /*
1555  * Initialize the screen to the correct color at startup.
1556  */
initcolor(void)1557 static void initcolor(void)
1558 {
1559 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
1560           intensevideo();
1561 #endif
1562           SETCOLORS(nm_fg_color, nm_bg_color);
1563 #if 0
1564           /*
1565            * This clears the screen at startup.  This is different from
1566            * the behavior of other versions of less.  Disable it for now.
1567            */
1568           char *blanks;
1569           int row;
1570           int col;
1571 
1572           /*
1573            * Create a complete, blank screen using "normal" colors.
1574            */
1575           SETCOLORS(nm_fg_color, nm_bg_color);
1576           blanks = (char *) ecalloc(width+1, sizeof(char));
1577           for (col = 0;  col < sc_width;  col++)
1578                     blanks[col] = ' ';
1579           blanks[sc_width] = '\0';
1580           for (row = 0;  row < sc_height;  row++)
1581                     _outtext(blanks);
1582           free(blanks);
1583 #endif
1584 }
1585 #endif
1586 
1587 #if MSDOS_COMPILER==WIN32C
1588 
1589 /*
1590  * Enable virtual terminal processing, if available.
1591  */
win32_init_vt_term(void)1592 static void win32_init_vt_term(void)
1593 {
1594           DWORD output_mode;
1595 
1596           if (vt_enabled == 0 || (vt_enabled == 1 && con_out == con_out_ours))
1597                     return;
1598 
1599           GetConsoleMode(con_out, &output_mode);
1600           vt_enabled = SetConsoleMode(con_out,
1601                            output_mode | ENABLE_VIRTUAL_TERMINAL_PROCESSING);
1602           if (vt_enabled)
1603           {
1604               auto_wrap = 0;
1605               ignaw = 1;
1606           }
1607 }
1608 
win32_deinit_vt_term(void)1609 static void win32_deinit_vt_term(void)
1610 {
1611           if (vt_enabled == 1 && con_out == con_out_save)
1612                     SetConsoleMode(con_out, init_output_mode);
1613 }
1614 
1615 /*
1616  * Termcap-like init with a private win32 console.
1617  */
win32_init_term(void)1618 static void win32_init_term(void)
1619 {
1620           CONSOLE_SCREEN_BUFFER_INFO scr;
1621           COORD size;
1622 
1623           if (con_out_save == INVALID_HANDLE_VALUE)
1624                     return;
1625 
1626           GetConsoleScreenBufferInfo(con_out_save, &scr);
1627 
1628           if (con_out_ours == INVALID_HANDLE_VALUE)
1629           {
1630                     /*
1631                      * Create our own screen buffer, so that we
1632                      * may restore the original when done.
1633                      */
1634                     con_out_ours = CreateConsoleScreenBuffer(
1635                               GENERIC_WRITE | GENERIC_READ,
1636                               FILE_SHARE_WRITE | FILE_SHARE_READ,
1637                               (LPSECURITY_ATTRIBUTES) NULL,
1638                               CONSOLE_TEXTMODE_BUFFER,
1639                               (LPVOID) NULL);
1640           }
1641 
1642           size.X = scr.srWindow.Right - scr.srWindow.Left + 1;
1643           size.Y = scr.srWindow.Bottom - scr.srWindow.Top + 1;
1644           SetConsoleScreenBufferSize(con_out_ours, size);
1645           SetConsoleActiveScreenBuffer(con_out_ours);
1646           con_out = con_out_ours;
1647 }
1648 
1649 /*
1650  * Restore the startup console.
1651  */
win32_deinit_term(void)1652 static void win32_deinit_term(void)
1653 {
1654           if (con_out_save == INVALID_HANDLE_VALUE)
1655                     return;
1656           if (quitting)
1657                     (void) CloseHandle(con_out_ours);
1658           SetConsoleActiveScreenBuffer(con_out_save);
1659           con_out = con_out_save;
1660 }
1661 
1662 #endif
1663 
1664 #if !MSDOS_COMPILER
do_tputs(char * str,int affcnt,int (* f_putc)(int))1665 static void do_tputs(char *str, int affcnt, int (*f_putc)(int))
1666 {
1667 #if LESSTEST
1668           if (ttyin_name != NULL && f_putc == putchr)
1669                     putstr(str);
1670           else
1671 #endif /*LESSTEST*/
1672                     tputs(str, affcnt, f_putc);
1673 }
1674 
1675 /*
1676  * Like tputs but we handle $<...> delay strings here because
1677  * some implementations of tputs don't perform delays correctly.
1678  */
ltputs(char * str,int affcnt,int (* f_putc)(int))1679 static void ltputs(char *str, int affcnt, int (*f_putc)(int))
1680 {
1681           while (str != NULL && *str != '\0')
1682           {
1683 #if HAVE_STRSTR
1684                     char *obrac = strstr(str, "$<");
1685                     if (obrac != NULL)
1686                     {
1687                               char str2[64];
1688                               int slen = obrac - str;
1689                               if (slen < sizeof(str2))
1690                               {
1691                                         int delay;
1692                                         /* Output first part of string (before "$<"). */
1693                                         memcpy(str2, str, slen);
1694                                         str2[slen] = '\0';
1695                                         do_tputs(str2, affcnt, f_putc);
1696                                         str += slen + 2;
1697                                         /* Perform the delay. */
1698                                         delay = lstrtoi(str, &str, 10);
1699                                         if (*str == '*')
1700                                                   if (ckd_mul(&delay, delay, affcnt))
1701                                                             delay = INT_MAX;
1702                                         flush();
1703                                         sleep_ms(delay);
1704                                         /* Skip past closing ">" at end of delay string. */
1705                                         str = strstr(str, ">");
1706                                         if (str != NULL)
1707                                                   str++;
1708                                         continue;
1709                               }
1710                     }
1711 #endif
1712                     /* Pass the rest of the string to tputs and we're done. */
1713                     do_tputs(str, affcnt, f_putc);
1714                     break;
1715           }
1716 }
1717 #endif /* MSDOS_COMPILER */
1718 
1719 /*
1720  * Configure the terminal so mouse clicks and wheel moves
1721  * produce input to less.
1722  */
init_mouse(void)1723 public void init_mouse(void)
1724 {
1725 #if !MSDOS_COMPILER
1726           ltputs(sc_s_mousecap, sc_height, putchr);
1727 #else
1728 #if MSDOS_COMPILER==WIN32C
1729           SetConsoleMode(tty, ENABLE_PROCESSED_INPUT | ENABLE_MOUSE_INPUT
1730                                   | ENABLE_EXTENDED_FLAGS /* disable quick edit */);
1731 
1732 #endif
1733 #endif
1734 }
1735 
1736 /*
1737  * Configure the terminal so mouse clicks and wheel moves
1738  * are handled by the system (so text can be selected, etc).
1739  */
deinit_mouse(void)1740 public void deinit_mouse(void)
1741 {
1742 #if !MSDOS_COMPILER
1743           ltputs(sc_e_mousecap, sc_height, putchr);
1744 #else
1745 #if MSDOS_COMPILER==WIN32C
1746           SetConsoleMode(tty, ENABLE_PROCESSED_INPUT | ENABLE_EXTENDED_FLAGS
1747                                   | (console_mode & ENABLE_QUICK_EDIT_MODE));
1748 #endif
1749 #endif
1750 }
1751 
1752 /*
1753  * Initialize terminal
1754  */
init(void)1755 public void init(void)
1756 {
1757           clear_bot_if_needed();
1758 #if !MSDOS_COMPILER
1759           if (!(quit_if_one_screen && one_screen))
1760           {
1761                     if (!no_init)
1762                     {
1763                               ltputs(sc_init, sc_height, putchr);
1764                               /*
1765                                * Some terminals leave the cursor unmoved when switching
1766                                * to the alt screen. To avoid having the text appear at
1767                                * a seemingly random line on the alt screen, move to
1768                                * lower left if we are using an alt screen.
1769                                */
1770                               if (*sc_init != '\0' && *sc_deinit != '\0' && !no_alt_screen)
1771                                         lower_left();
1772                               term_init_done = 1;
1773                     }
1774                     if (!no_keypad)
1775                               ltputs(sc_s_keypad, sc_height, putchr);
1776                     if (mousecap)
1777                               init_mouse();
1778           }
1779           init_done = 1;
1780           if (top_scroll)
1781           {
1782                     int i;
1783 
1784                     /*
1785                      * This is nice to terminals with no alternate screen,
1786                      * but with saved scrolled-off-the-top lines.  This way,
1787                      * no previous line is lost, but we start with a whole
1788                      * screen to ourself.
1789                      */
1790                     for (i = 1; i < sc_height; i++)
1791                               putchr('\n');
1792           } else
1793                     line_left();
1794 #else
1795 #if MSDOS_COMPILER==WIN32C
1796           if (!(quit_if_one_screen && one_screen))
1797           {
1798                     if (!no_init)
1799                     {
1800                               win32_init_term();
1801                               term_init_done = 1;
1802                     }
1803                     if (mousecap)
1804                               init_mouse();
1805 
1806           }
1807           win32_init_vt_term();
1808 #endif
1809           init_done = 1;
1810           initcolor();
1811           flush();
1812 #endif
1813 }
1814 
1815 /*
1816  * Deinitialize terminal
1817  */
deinit(void)1818 public void deinit(void)
1819 {
1820           if (!init_done)
1821                     return;
1822 #if !MSDOS_COMPILER
1823           if (!(quit_if_one_screen && one_screen))
1824           {
1825                     if (mousecap)
1826                               deinit_mouse();
1827                     if (!no_keypad)
1828                               ltputs(sc_e_keypad, sc_height, putchr);
1829                     if (!no_init)
1830                               ltputs(sc_deinit, sc_height, putchr);
1831           }
1832 #else
1833           /* Restore system colors. */
1834           SETCOLORS(sy_fg_color, sy_bg_color);
1835 #if MSDOS_COMPILER==WIN32C
1836           win32_deinit_vt_term();
1837           if (!(quit_if_one_screen && one_screen))
1838           {
1839                     if (mousecap)
1840                               deinit_mouse();
1841                     if (!no_init)
1842                               win32_deinit_term();
1843           }
1844 #else
1845           /* Need clreol to make SETCOLORS take effect. */
1846           clreol();
1847 #endif
1848 #endif
1849           init_done = 0;
1850 }
1851 
1852 /*
1853  * Are we interactive (ie. writing to an initialized tty)?
1854  */
interactive(void)1855 public int interactive(void)
1856 {
1857           return (is_tty && init_done);
1858 }
1859 
assert_interactive(void)1860 static void assert_interactive(void)
1861 {
1862           if (interactive()) return;
1863           /* abort(); */
1864 }
1865 
1866 /*
1867  * Home cursor (move to upper left corner of screen).
1868  */
home(void)1869 public void home(void)
1870 {
1871           assert_interactive();
1872 #if !MSDOS_COMPILER
1873           ltputs(sc_home, 1, putchr);
1874 #else
1875           flush();
1876           _settextposition(1,1);
1877 #endif
1878 }
1879 
1880 #if LESSTEST
dump_screen(void)1881 public void dump_screen(void)
1882 {
1883           char dump_cmd[32];
1884           SNPRINTF1(dump_cmd, sizeof(dump_cmd), ESCS"0;0;%dR", sc_width * sc_height);
1885           ltputs(dump_cmd, sc_height, putchr);
1886           flush();
1887 }
1888 #endif /*LESSTEST*/
1889 
1890 /*
1891  * Add a blank line (called with cursor at home).
1892  * Should scroll the display down.
1893  */
add_line(void)1894 public void add_line(void)
1895 {
1896           assert_interactive();
1897 #if !MSDOS_COMPILER
1898           ltputs(sc_addline, sc_height, putchr);
1899 #else
1900           flush();
1901 #if MSDOS_COMPILER==MSOFTC
1902           _scrolltextwindow(_GSCROLLDOWN);
1903           _settextposition(1,1);
1904 #else
1905 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
1906           movetext(1,1, sc_width,sc_height-1, 1,2);
1907           gotoxy(1,1);
1908           clreol();
1909 #else
1910 #if MSDOS_COMPILER==WIN32C
1911     {
1912           CHAR_INFO fillchar;
1913           SMALL_RECT rcSrc, rcClip;
1914           COORD new_org;
1915           CONSOLE_SCREEN_BUFFER_INFO csbi;
1916 
1917           GetConsoleScreenBufferInfo(con_out,&csbi);
1918 
1919           /* The clip rectangle is the entire visible screen. */
1920           rcClip.Left = csbi.srWindow.Left;
1921           rcClip.Top = csbi.srWindow.Top;
1922           rcClip.Right = csbi.srWindow.Right;
1923           rcClip.Bottom = csbi.srWindow.Bottom;
1924 
1925           /* The source rectangle is the visible screen minus the last line. */
1926           rcSrc = rcClip;
1927           rcSrc.Bottom--;
1928 
1929           /* Move the top left corner of the source window down one row. */
1930           new_org.X = rcSrc.Left;
1931           new_org.Y = rcSrc.Top + 1;
1932 
1933           /* Fill the right character and attributes. */
1934           fillchar.Char.AsciiChar = ' ';
1935           curr_attr = MAKEATTR(nm_fg_color, nm_bg_color);
1936           fillchar.Attributes = curr_attr;
1937           ScrollConsoleScreenBuffer(con_out, &rcSrc, &rcClip, new_org, &fillchar);
1938           _settextposition(1,1);
1939     }
1940 #endif
1941 #endif
1942 #endif
1943 #endif
1944 }
1945 
1946 #if 0
1947 /*
1948  * Remove the n topmost lines and scroll everything below it in the
1949  * window upward.  This is needed to stop leaking the topmost line
1950  * into the scrollback buffer when we go down-one-line (in WIN32).
1951  */
1952 public void remove_top(int n)
1953 {
1954 #if MSDOS_COMPILER==WIN32C
1955           SMALL_RECT rcSrc, rcClip;
1956           CHAR_INFO fillchar;
1957           COORD new_org;
1958           CONSOLE_SCREEN_BUFFER_INFO csbi; /* to get buffer info */
1959 
1960           if (n >= sc_height - 1)
1961           {
1962                     clear();
1963                     home();
1964                     return;
1965           }
1966 
1967           flush();
1968 
1969           GetConsoleScreenBufferInfo(con_out, &csbi);
1970 
1971           /* Get the extent of all-visible-rows-but-the-last. */
1972           rcSrc.Left    = csbi.srWindow.Left;
1973           rcSrc.Top     = csbi.srWindow.Top + n;
1974           rcSrc.Right   = csbi.srWindow.Right;
1975           rcSrc.Bottom  = csbi.srWindow.Bottom;
1976 
1977           /* Get the clip rectangle. */
1978           rcClip.Left   = rcSrc.Left;
1979           rcClip.Top    = csbi.srWindow.Top;
1980           rcClip.Right  = rcSrc.Right;
1981           rcClip.Bottom = rcSrc.Bottom ;
1982 
1983           /* Move the source window up n rows. */
1984           new_org.X = rcSrc.Left;
1985           new_org.Y = rcSrc.Top - n;
1986 
1987           /* Fill the right character and attributes. */
1988           fillchar.Char.AsciiChar = ' ';
1989           curr_attr = MAKEATTR(nm_fg_color, nm_bg_color);
1990           fillchar.Attributes = curr_attr;
1991 
1992           ScrollConsoleScreenBuffer(con_out, &rcSrc, &rcClip, new_org, &fillchar);
1993 
1994           /* Position cursor on first blank line. */
1995           goto_line(sc_height - n - 1);
1996 #endif
1997 }
1998 #endif
1999 
2000 #if MSDOS_COMPILER==WIN32C
2001 /*
2002  * Clear the screen.
2003  */
win32_clear(void)2004 static void win32_clear(void)
2005 {
2006           /*
2007            * This will clear only the currently visible rows of the NT
2008            * console buffer, which means none of the precious scrollback
2009            * rows are touched making for faster scrolling.  Note that, if
2010            * the window has fewer columns than the console buffer (i.e.
2011            * there is a horizontal scrollbar as well), the entire width
2012            * of the visible rows will be cleared.
2013            */
2014           COORD topleft;
2015           DWORD nchars;
2016           DWORD winsz;
2017           CONSOLE_SCREEN_BUFFER_INFO csbi;
2018 
2019           /* get the number of cells in the current buffer */
2020           GetConsoleScreenBufferInfo(con_out, &csbi);
2021           winsz = csbi.dwSize.X * (csbi.srWindow.Bottom - csbi.srWindow.Top + 1);
2022           topleft.X = 0;
2023           topleft.Y = csbi.srWindow.Top;
2024 
2025           curr_attr = MAKEATTR(nm_fg_color, nm_bg_color);
2026           FillConsoleOutputCharacter(con_out, ' ', winsz, topleft, &nchars);
2027           FillConsoleOutputAttribute(con_out, curr_attr, winsz, topleft, &nchars);
2028 }
2029 
2030 /*
2031  * Remove the n topmost lines and scroll everything below it in the
2032  * window upward.
2033  */
win32_scroll_up(int n)2034 public void win32_scroll_up(int n)
2035 {
2036           SMALL_RECT rcSrc, rcClip;
2037           CHAR_INFO fillchar;
2038           COORD topleft;
2039           COORD new_org;
2040           DWORD nchars;
2041           DWORD size;
2042           CONSOLE_SCREEN_BUFFER_INFO csbi;
2043 
2044           if (n <= 0)
2045                     return;
2046 
2047           if (n >= sc_height - 1)
2048           {
2049                     win32_clear();
2050                     _settextposition(1,1);
2051                     return;
2052           }
2053 
2054           /* Get the extent of what will remain visible after scrolling. */
2055           GetConsoleScreenBufferInfo(con_out, &csbi);
2056           rcSrc.Left    = csbi.srWindow.Left;
2057           rcSrc.Top     = csbi.srWindow.Top + n;
2058           rcSrc.Right   = csbi.srWindow.Right;
2059           rcSrc.Bottom  = csbi.srWindow.Bottom;
2060 
2061           /* Get the clip rectangle. */
2062           rcClip.Left   = rcSrc.Left;
2063           rcClip.Top    = csbi.srWindow.Top;
2064           rcClip.Right  = rcSrc.Right;
2065           rcClip.Bottom = rcSrc.Bottom ;
2066 
2067           /* Move the source text to the top of the screen. */
2068           new_org.X = rcSrc.Left;
2069           new_org.Y = rcClip.Top;
2070 
2071           /* Fill the right character and attributes. */
2072           fillchar.Char.AsciiChar = ' ';
2073           fillchar.Attributes = MAKEATTR(nm_fg_color, nm_bg_color);
2074 
2075           /* Scroll the window. */
2076           SetConsoleTextAttribute(con_out, fillchar.Attributes);
2077           ScrollConsoleScreenBuffer(con_out, &rcSrc, &rcClip, new_org, &fillchar);
2078 
2079           /* Clear remaining lines at bottom. */
2080           topleft.X = csbi.dwCursorPosition.X;
2081           topleft.Y = rcSrc.Bottom - n;
2082           size = (n * csbi.dwSize.X) + (rcSrc.Right - topleft.X);
2083           FillConsoleOutputCharacter(con_out, ' ', size, topleft,
2084                     &nchars);
2085           FillConsoleOutputAttribute(con_out, fillchar.Attributes, size, topleft,
2086                     &nchars);
2087           SetConsoleTextAttribute(con_out, curr_attr);
2088 
2089           /* Move cursor n lines up from where it was. */
2090           csbi.dwCursorPosition.Y -= n;
2091           SetConsoleCursorPosition(con_out, csbi.dwCursorPosition);
2092 }
2093 #endif
2094 
2095 /*
2096  * Move cursor to lower left corner of screen.
2097  */
lower_left(void)2098 public void lower_left(void)
2099 {
2100           assert_interactive();
2101 #if !MSDOS_COMPILER
2102           ltputs(sc_lower_left, 1, putchr);
2103 #else
2104           flush();
2105           _settextposition(sc_height, 1);
2106 #endif
2107 }
2108 
2109 /*
2110  * Move cursor to left position of current line.
2111  */
line_left(void)2112 public void line_left(void)
2113 {
2114           assert_interactive();
2115 #if !MSDOS_COMPILER
2116           ltputs(sc_return, 1, putchr);
2117 #else
2118           {
2119                     int row;
2120                     flush();
2121 #if MSDOS_COMPILER==WIN32C
2122                     {
2123                               CONSOLE_SCREEN_BUFFER_INFO scr;
2124                               GetConsoleScreenBufferInfo(con_out, &scr);
2125                               row = scr.dwCursorPosition.Y - scr.srWindow.Top + 1;
2126                     }
2127 #else
2128 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
2129                               row = wherey();
2130 #else
2131                     {
2132                               struct rccoord tpos = _gettextposition();
2133                               row = tpos.row;
2134                     }
2135 #endif
2136 #endif
2137                     _settextposition(row, 1);
2138           }
2139 #endif
2140 }
2141 
2142 /*
2143  * Check if the console size has changed and reset internals
2144  * (in lieu of SIGWINCH for WIN32).
2145  */
check_winch(void)2146 public void check_winch(void)
2147 {
2148 #if MSDOS_COMPILER==WIN32C
2149           CONSOLE_SCREEN_BUFFER_INFO scr;
2150           COORD size;
2151 
2152           if (con_out == INVALID_HANDLE_VALUE)
2153                     return;
2154 
2155           flush();
2156           GetConsoleScreenBufferInfo(con_out, &scr);
2157           size.Y = scr.srWindow.Bottom - scr.srWindow.Top + 1;
2158           size.X = scr.srWindow.Right - scr.srWindow.Left + 1;
2159           if (size.Y != sc_height || size.X != sc_width)
2160           {
2161                     sc_height = size.Y;
2162                     sc_width = size.X;
2163                     if (!no_init && con_out_ours == con_out)
2164                               SetConsoleScreenBufferSize(con_out, size);
2165                     pos_init();
2166                     wscroll = (sc_height + 1) / 2;
2167                     screen_trashed = 1;
2168           }
2169 #endif
2170 }
2171 
2172 /*
2173  * Goto a specific line on the screen.
2174  */
goto_line(int sindex)2175 public void goto_line(int sindex)
2176 {
2177           assert_interactive();
2178 #if !MSDOS_COMPILER
2179           ltputs(tgoto(sc_move, 0, sindex), 1, putchr);
2180 #else
2181           flush();
2182           _settextposition(sindex+1, 1);
2183 #endif
2184 }
2185 
2186 #if MSDOS_COMPILER==MSOFTC || MSDOS_COMPILER==BORLANDC
2187 /*
2188  * Create an alternate screen which is all white.
2189  * This screen is used to create a "flash" effect, by displaying it
2190  * briefly and then switching back to the normal screen.
2191  * {{ Yuck!  There must be a better way to get a visual bell. }}
2192  */
create_flash(void)2193 static void create_flash(void)
2194 {
2195 #if MSDOS_COMPILER==MSOFTC
2196           struct videoconfig w;
2197           char *blanks;
2198           int row, col;
2199 
2200           _getvideoconfig(&w);
2201           videopages = w.numvideopages;
2202           if (videopages < 2)
2203           {
2204                     at_enter(AT_STANDOUT);
2205                     at_exit();
2206           } else
2207           {
2208                     _setactivepage(1);
2209                     at_enter(AT_STANDOUT);
2210                     blanks = (char *) ecalloc(w.numtextcols, sizeof(char));
2211                     for (col = 0;  col < w.numtextcols;  col++)
2212                               blanks[col] = ' ';
2213                     for (row = w.numtextrows;  row > 0;  row--)
2214                               _outmem(blanks, w.numtextcols);
2215                     _setactivepage(0);
2216                     _setvisualpage(0);
2217                     free(blanks);
2218                     at_exit();
2219           }
2220 #else
2221 #if MSDOS_COMPILER==BORLANDC
2222           int n;
2223 
2224           whitescreen = (unsigned short *)
2225                     malloc(sc_width * sc_height * sizeof(short));
2226           if (whitescreen == NULL)
2227                     return;
2228           for (n = 0;  n < sc_width * sc_height;  n++)
2229                     whitescreen[n] = 0x7020;
2230 #endif
2231 #endif
2232           flash_created = 1;
2233 }
2234 #endif /* MSDOS_COMPILER */
2235 
2236 /*
2237  * Output the "visual bell", if there is one.
2238  */
vbell(void)2239 public void vbell(void)
2240 {
2241           if (no_vbell)
2242                     return;
2243 #if !MSDOS_COMPILER
2244           if (*sc_visual_bell == '\0')
2245                     return;
2246           ltputs(sc_visual_bell, sc_height, putchr);
2247 #else
2248 #if MSDOS_COMPILER==DJGPPC
2249           ScreenVisualBell();
2250 #else
2251 #if MSDOS_COMPILER==MSOFTC
2252           /*
2253            * Create a flash screen on the second video page.
2254            * Switch to that page, then switch back.
2255            */
2256           if (!flash_created)
2257                     create_flash();
2258           if (videopages < 2)
2259                     return;
2260           _setvisualpage(1);
2261           delay(100);
2262           _setvisualpage(0);
2263 #else
2264 #if MSDOS_COMPILER==BORLANDC
2265           unsigned short *currscreen;
2266 
2267           /*
2268            * Get a copy of the current screen.
2269            * Display the flash screen.
2270            * Then restore the old screen.
2271            */
2272           if (!flash_created)
2273                     create_flash();
2274           if (whitescreen == NULL)
2275                     return;
2276           currscreen = (unsigned short *)
2277                     malloc(sc_width * sc_height * sizeof(short));
2278           if (currscreen == NULL) return;
2279           gettext(1, 1, sc_width, sc_height, currscreen);
2280           puttext(1, 1, sc_width, sc_height, whitescreen);
2281           delay(100);
2282           puttext(1, 1, sc_width, sc_height, currscreen);
2283           free(currscreen);
2284 #else
2285 #if MSDOS_COMPILER==WIN32C
2286           /* paint screen with an inverse color */
2287           clear();
2288 
2289           /* leave it displayed for 100 msec. */
2290           Sleep(100);
2291 
2292           /* restore with a redraw */
2293           repaint();
2294 #endif
2295 #endif
2296 #endif
2297 #endif
2298 #endif
2299 }
2300 
2301 /*
2302  * Make a noise.
2303  */
beep(void)2304 static void beep(void)
2305 {
2306 #if !MSDOS_COMPILER
2307           putchr(CONTROL('G'));
2308 #else
2309 #if MSDOS_COMPILER==WIN32C
2310           MessageBeep(0);
2311 #else
2312           write(1, "\7", 1);
2313 #endif
2314 #endif
2315 }
2316 
2317 /*
2318  * Ring the terminal bell.
2319  */
bell(void)2320 public void bell(void)
2321 {
2322           if (quiet == VERY_QUIET)
2323                     vbell();
2324           else
2325                     beep();
2326 }
2327 
2328 /*
2329  * Clear the screen.
2330  */
clear(void)2331 public void clear(void)
2332 {
2333           assert_interactive();
2334 #if !MSDOS_COMPILER
2335           ltputs(sc_clear, sc_height, putchr);
2336 #else
2337           flush();
2338 #if MSDOS_COMPILER==WIN32C
2339           win32_clear();
2340 #else
2341           _clearscreen(_GCLEARSCREEN);
2342 #endif
2343 #endif
2344 }
2345 
2346 /*
2347  * Clear from the cursor to the end of the cursor's line.
2348  * {{ This must not move the cursor. }}
2349  */
clear_eol(void)2350 public void clear_eol(void)
2351 {
2352           /* assert_interactive();*/
2353 #if !MSDOS_COMPILER
2354           ltputs(sc_eol_clear, 1, putchr);
2355 #else
2356 #if MSDOS_COMPILER==MSOFTC
2357           short top, left;
2358           short bot, right;
2359           struct rccoord tpos;
2360 
2361           flush();
2362           /*
2363            * Save current state.
2364            */
2365           tpos = _gettextposition();
2366           _gettextwindow(&top, &left, &bot, &right);
2367           /*
2368            * Set a temporary window to the current line,
2369            * from the cursor's position to the right edge of the screen.
2370            * Then clear that window.
2371            */
2372           _settextwindow(tpos.row, tpos.col, tpos.row, sc_width);
2373           _clearscreen(_GWINDOW);
2374           /*
2375            * Restore state.
2376            */
2377           _settextwindow(top, left, bot, right);
2378           _settextposition(tpos.row, tpos.col);
2379 #else
2380 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
2381           flush();
2382           clreol();
2383 #else
2384 #if MSDOS_COMPILER==WIN32C
2385           DWORD           nchars;
2386           COORD           cpos;
2387           CONSOLE_SCREEN_BUFFER_INFO scr;
2388 
2389           flush();
2390           memset(&scr, 0, sizeof(scr));
2391           GetConsoleScreenBufferInfo(con_out, &scr);
2392           cpos.X = scr.dwCursorPosition.X;
2393           cpos.Y = scr.dwCursorPosition.Y;
2394           curr_attr = MAKEATTR(nm_fg_color, nm_bg_color);
2395           FillConsoleOutputAttribute(con_out, curr_attr,
2396                     scr.dwSize.X - cpos.X, cpos, &nchars);
2397           FillConsoleOutputCharacter(con_out, ' ',
2398                     scr.dwSize.X - cpos.X, cpos, &nchars);
2399 #endif
2400 #endif
2401 #endif
2402 #endif
2403 }
2404 
2405 /*
2406  * Clear the current line.
2407  * Clear the screen if there's off-screen memory below the display.
2408  */
clear_eol_bot(void)2409 static void clear_eol_bot(void)
2410 {
2411           assert_interactive();
2412 #if MSDOS_COMPILER
2413           clear_eol();
2414 #else
2415           if (below_mem)
2416                     ltputs(sc_eos_clear, 1, putchr);
2417           else
2418                     ltputs(sc_eol_clear, 1, putchr);
2419 #endif
2420 }
2421 
2422 /*
2423  * Clear the bottom line of the display.
2424  * Leave the cursor at the beginning of the bottom line.
2425  */
clear_bot(void)2426 public void clear_bot(void)
2427 {
2428           /*
2429            * If we're in a non-normal attribute mode, temporarily exit
2430            * the mode while we do the clear.  Some terminals fill the
2431            * cleared area with the current attribute.
2432            */
2433           if (oldbot)
2434                     lower_left();
2435           else
2436                     line_left();
2437 
2438           if (attrmode == AT_NORMAL)
2439                     clear_eol_bot();
2440           else
2441           {
2442                     int saved_attrmode = attrmode;
2443 
2444                     at_exit();
2445                     clear_eol_bot();
2446                     at_enter(saved_attrmode);
2447           }
2448 }
2449 
2450 /*
2451  * Color string may be "x[y]" where x and y are 4-bit color chars,
2452  * or "N[.M]" where N and M are decimal integers>
2453  * Any of x,y,N,M may also be "-" to mean "unchanged".
2454  */
2455 
2456 /*
2457  * Parse a 4-bit color char.
2458  */
parse_color4(char ch)2459 static int parse_color4(char ch)
2460 {
2461           switch (ch)
2462           {
2463           case 'k': return 0;
2464           case 'r': return CV_RED;
2465           case 'g': return CV_GREEN;
2466           case 'y': return CV_RED|CV_GREEN;
2467           case 'b': return CV_BLUE;
2468           case 'm': return CV_RED|CV_BLUE;
2469           case 'c': return CV_GREEN|CV_BLUE;
2470           case 'w': return CV_RED|CV_GREEN|CV_BLUE;
2471           case 'K': return 0|CV_BRIGHT;
2472           case 'R': return CV_RED|CV_BRIGHT;
2473           case 'G': return CV_GREEN|CV_BRIGHT;
2474           case 'Y': return CV_RED|CV_GREEN|CV_BRIGHT;
2475           case 'B': return CV_BLUE|CV_BRIGHT;
2476           case 'M': return CV_RED|CV_BLUE|CV_BRIGHT;
2477           case 'C': return CV_GREEN|CV_BLUE|CV_BRIGHT;
2478           case 'W': return CV_RED|CV_GREEN|CV_BLUE|CV_BRIGHT;
2479           case '-': return CV_NOCHANGE;
2480           default:  return CV_ERROR;
2481           }
2482 }
2483 
2484 /*
2485  * Parse a color as a decimal integer.
2486  */
parse_color6(char ** ps)2487 static int parse_color6(char **ps)
2488 {
2489           if (**ps == '-')
2490           {
2491                     (*ps)++;
2492                     return CV_NOCHANGE;
2493           } else
2494           {
2495                     char *ops = *ps;
2496                     int color = lstrtoi(ops, ps, 10);
2497                     if (color < 0 || *ps == ops)
2498                               return CV_ERROR;
2499                     return color;
2500           }
2501 }
2502 
2503 /*
2504  * Parse a color pair and return the foreground/background values.
2505  * Return type of color specifier:
2506  *  CV_4BIT: fg/bg values are OR of CV_{RGB} bits.
2507  *  CV_6BIT: fg/bg values are integers entered by user.
2508  */
parse_color(char * str,int * p_fg,int * p_bg)2509 public COLOR_TYPE parse_color(char *str, int *p_fg, int *p_bg)
2510 {
2511           int fg;
2512           int bg;
2513           COLOR_TYPE type = CT_NULL;
2514 
2515           if (str == NULL || *str == '\0')
2516                     return CT_NULL;
2517           if (*str == '+')
2518                     str++; /* ignore leading + */
2519 
2520           fg = parse_color4(str[0]);
2521           bg = parse_color4((strlen(str) < 2) ? '-' : str[1]);
2522           if (fg != CV_ERROR && bg != CV_ERROR)
2523                     type = CT_4BIT;
2524           else
2525           {
2526                     fg = parse_color6(&str);
2527                     bg = (fg != CV_ERROR && *str++ == '.') ? parse_color6(&str) : CV_NOCHANGE;
2528                     if (fg != CV_ERROR && bg != CV_ERROR)
2529                               type = CT_6BIT;
2530           }
2531           if (p_fg != NULL) *p_fg = fg;
2532           if (p_bg != NULL) *p_bg = bg;
2533           return type;
2534 }
2535 
2536 #if !MSDOS_COMPILER
2537 
sgr_color(int color)2538 static int sgr_color(int color)
2539 {
2540           switch (color)
2541           {
2542           case 0:                                    return 30;
2543           case CV_RED:                               return 31;
2544           case CV_GREEN:                             return 32;
2545           case CV_RED|CV_GREEN:                      return 33;
2546           case CV_BLUE:                              return 34;
2547           case CV_RED|CV_BLUE:                       return 35;
2548           case CV_GREEN|CV_BLUE:                     return 36;
2549           case CV_RED|CV_GREEN|CV_BLUE:              return 37;
2550 
2551           case CV_BRIGHT:                            return 90;
2552           case CV_RED|CV_BRIGHT:                     return 91;
2553           case CV_GREEN|CV_BRIGHT:                   return 92;
2554           case CV_RED|CV_GREEN|CV_BRIGHT:            return 93;
2555           case CV_BLUE|CV_BRIGHT:                    return 94;
2556           case CV_RED|CV_BLUE|CV_BRIGHT:             return 95;
2557           case CV_GREEN|CV_BLUE|CV_BRIGHT:           return 96;
2558           case CV_RED|CV_GREEN|CV_BLUE|CV_BRIGHT:    return 97;
2559 
2560           default: return color;
2561           }
2562 }
2563 
tput_fmt(char * fmt,int color,int (* f_putc)(int))2564 static void tput_fmt(char *fmt, int color, int (*f_putc)(int))
2565 {
2566           char buf[INT_STRLEN_BOUND(int)+16];
2567           if (color == attrcolor)
2568                     return;
2569           SNPRINTF1(buf, sizeof(buf), fmt, color);
2570           ltputs(buf, 1, f_putc);
2571           attrcolor = color;
2572 }
2573 
tput_color(char * str,int (* f_putc)(int))2574 static void tput_color(char *str, int (*f_putc)(int))
2575 {
2576           int fg;
2577           int bg;
2578 
2579           if (str != NULL && strcmp(str, "*") == 0)
2580           {
2581                     /* Special case: reset to normal */
2582                     tput_fmt(ESCS"[m", -1, f_putc);
2583                     return;
2584           }
2585           switch (parse_color(str, &fg, &bg))
2586           {
2587           case CT_4BIT:
2588                     if (fg >= 0)
2589                               tput_fmt(ESCS"[%dm", sgr_color(fg), f_putc);
2590                     if (bg >= 0)
2591                               tput_fmt(ESCS"[%dm", sgr_color(bg)+10, f_putc);
2592                     break;
2593           case CT_6BIT:
2594                     if (fg >= 0)
2595                               tput_fmt(ESCS"[38;5;%dm", fg, f_putc);
2596                     if (bg >= 0)
2597                               tput_fmt(ESCS"[48;5;%dm", bg, f_putc);
2598                     break;
2599           default:
2600                     break;
2601           }
2602 }
2603 
tput_inmode(char * mode_str,int attr,int attr_bit,int (* f_putc)(int))2604 static void tput_inmode(char *mode_str, int attr, int attr_bit, int (*f_putc)(int))
2605 {
2606           char *color_str;
2607           if ((attr & attr_bit) == 0)
2608                     return;
2609           color_str = get_color_map(attr_bit);
2610           if (color_str == NULL || *color_str == '\0' || *color_str == '+')
2611           {
2612                     ltputs(mode_str, 1, f_putc);
2613                     if (color_str == NULL || *color_str++ != '+')
2614                               return;
2615           }
2616           /* Color overrides mode string */
2617           tput_color(color_str, f_putc);
2618 }
2619 
tput_outmode(char * mode_str,int attr_bit,int (* f_putc)(int))2620 static void tput_outmode(char *mode_str, int attr_bit, int (*f_putc)(int))
2621 {
2622           if ((attrmode & attr_bit) == 0)
2623                     return;
2624           ltputs(mode_str, 1, f_putc);
2625 }
2626 
2627 #else /* MSDOS_COMPILER */
2628 
2629 #if MSDOS_COMPILER==WIN32C
WIN32put_fmt(char * fmt,int color)2630 static int WIN32put_fmt(char *fmt, int color)
2631 {
2632           char buf[INT_STRLEN_BOUND(int)+16];
2633           int len = SNPRINTF1(buf, sizeof(buf), fmt, color);
2634           WIN32textout(buf, len);
2635           return TRUE;
2636 }
2637 #endif
2638 
win_set_color(int attr)2639 static int win_set_color(int attr)
2640 {
2641           int fg;
2642           int bg;
2643           int out = FALSE;
2644           char *str = get_color_map(attr);
2645           if (str == NULL || str[0] == '\0')
2646                     return FALSE;
2647           switch (parse_color(str, &fg, &bg))
2648           {
2649           case CT_4BIT:
2650                     if (fg >= 0 && bg >= 0)
2651                     {
2652                               SETCOLORS(fg, bg);
2653                               out = TRUE;
2654                     } else if (fg >= 0)
2655                     {
2656                               SET_FG_COLOR(fg);
2657                               out = TRUE;
2658                     } else if (bg >= 0)
2659                     {
2660                               SET_BG_COLOR(bg);
2661                               out = TRUE;
2662                     }
2663                     break;
2664 #if MSDOS_COMPILER==WIN32C
2665           case CT_6BIT:
2666                     if (vt_enabled)
2667                     {
2668                               if (fg > 0)
2669                                         out = WIN32put_fmt(ESCS"[38;5;%dm", fg);
2670                               if (bg > 0)
2671                                         out = WIN32put_fmt(ESCS"[48;5;%dm", bg);
2672                     }
2673                     break;
2674 #endif
2675           default:
2676                     break;
2677           }
2678           return out;
2679 }
2680 
2681 #endif /* MSDOS_COMPILER */
2682 
at_enter(int attr)2683 public void at_enter(int attr)
2684 {
2685           attr = apply_at_specials(attr);
2686 #if !MSDOS_COMPILER
2687           /* The one with the most priority is last.  */
2688           tput_inmode(sc_u_in, attr, AT_UNDERLINE, putchr);
2689           tput_inmode(sc_b_in, attr, AT_BOLD, putchr);
2690           tput_inmode(sc_bl_in, attr, AT_BLINK, putchr);
2691           /* Don't use standout and color at the same time. */
2692           if (use_color && (attr & AT_COLOR))
2693                     tput_color(get_color_map(attr), putchr);
2694           else
2695                     tput_inmode(sc_s_in, attr, AT_STANDOUT, putchr);
2696 #else
2697           flush();
2698           /* The one with the most priority is first.  */
2699           if ((attr & AT_COLOR) && use_color)
2700           {
2701                     win_set_color(attr);
2702           } else if (attr & AT_STANDOUT)
2703           {
2704                     SETCOLORS(so_fg_color, so_bg_color);
2705           } else if (attr & AT_BLINK)
2706           {
2707                     SETCOLORS(bl_fg_color, bl_bg_color);
2708           } else if (attr & AT_BOLD)
2709           {
2710                     SETCOLORS(bo_fg_color, bo_bg_color);
2711           } else if (attr & AT_UNDERLINE)
2712           {
2713                     SETCOLORS(ul_fg_color, ul_bg_color);
2714           }
2715 #endif
2716           attrmode = attr;
2717 }
2718 
at_exit(void)2719 public void at_exit(void)
2720 {
2721 #if !MSDOS_COMPILER
2722           /* Undo things in the reverse order we did them.  */
2723           tput_color("*", putchr);
2724           tput_outmode(sc_s_out, AT_STANDOUT, putchr);
2725           tput_outmode(sc_bl_out, AT_BLINK, putchr);
2726           tput_outmode(sc_b_out, AT_BOLD, putchr);
2727           tput_outmode(sc_u_out, AT_UNDERLINE, putchr);
2728 #else
2729           flush();
2730           SETCOLORS(nm_fg_color, nm_bg_color);
2731 #endif
2732           attrmode = AT_NORMAL;
2733 }
2734 
at_switch(int attr)2735 public void at_switch(int attr)
2736 {
2737           int new_attrmode = apply_at_specials(attr);
2738           int ignore_modes = AT_ANSI;
2739 
2740           if ((new_attrmode & ~ignore_modes) != (attrmode & ~ignore_modes))
2741           {
2742                     at_exit();
2743                     at_enter(attr);
2744           }
2745 }
2746 
is_at_equiv(int attr1,int attr2)2747 public int is_at_equiv(int attr1, int attr2)
2748 {
2749           attr1 = apply_at_specials(attr1);
2750           attr2 = apply_at_specials(attr2);
2751 
2752           return (attr1 == attr2);
2753 }
2754 
apply_at_specials(int attr)2755 public int apply_at_specials(int attr)
2756 {
2757           if (attr & AT_BINARY)
2758                     attr |= binattr;
2759           if (attr & AT_HILITE)
2760                     attr |= AT_STANDOUT;
2761           attr &= ~(AT_BINARY|AT_HILITE);
2762 
2763           return attr;
2764 }
2765 
2766 /*
2767  * Output a plain backspace, without erasing the previous char.
2768  */
putbs(void)2769 public void putbs(void)
2770 {
2771           if (termcap_debug)
2772                     putstr("<bs>");
2773           else
2774           {
2775 #if !MSDOS_COMPILER
2776           ltputs(sc_backspace, 1, putchr);
2777 #else
2778           int row, col;
2779 
2780           flush();
2781           {
2782 #if MSDOS_COMPILER==MSOFTC
2783                     struct rccoord tpos;
2784                     tpos = _gettextposition();
2785                     row = tpos.row;
2786                     col = tpos.col;
2787 #else
2788 #if MSDOS_COMPILER==BORLANDC || MSDOS_COMPILER==DJGPPC
2789                     row = wherey();
2790                     col = wherex();
2791 #else
2792 #if MSDOS_COMPILER==WIN32C
2793                     CONSOLE_SCREEN_BUFFER_INFO scr;
2794                     GetConsoleScreenBufferInfo(con_out, &scr);
2795                     row = scr.dwCursorPosition.Y - scr.srWindow.Top + 1;
2796                     col = scr.dwCursorPosition.X - scr.srWindow.Left + 1;
2797 #endif
2798 #endif
2799 #endif
2800           }
2801           if (col <= 1)
2802                     return;
2803           _settextposition(row, col-1);
2804 #endif /* MSDOS_COMPILER */
2805           }
2806 }
2807 
2808 #if MSDOS_COMPILER==WIN32C
2809 /*
2810  * Determine whether an input character is waiting to be read.
2811  */
win32_kbhit(void)2812 public int win32_kbhit(void)
2813 {
2814           INPUT_RECORD ip;
2815           DWORD read;
2816 
2817           if (keyCount > 0 || win_unget_pending)
2818                     return (TRUE);
2819 
2820           currentKey.ascii = 0;
2821           currentKey.scan = 0;
2822 
2823           if (x11mouseCount > 0)
2824           {
2825                     currentKey.ascii = x11mousebuf[x11mousePos++];
2826                     --x11mouseCount;
2827                     keyCount = 1;
2828                     return (TRUE);
2829           }
2830 
2831           /*
2832            * Wait for a real key-down event, but
2833            * ignore SHIFT and CONTROL key events.
2834            */
2835           do
2836           {
2837                     PeekConsoleInputW(tty, &ip, 1, &read);
2838                     if (read == 0)
2839                               return (FALSE);
2840                     ReadConsoleInputW(tty, &ip, 1, &read);
2841                     /* generate an X11 mouse sequence from the mouse event */
2842                     if (mousecap && ip.EventType == MOUSE_EVENT &&
2843                         ip.Event.MouseEvent.dwEventFlags != MOUSE_MOVED)
2844                     {
2845                               x11mousebuf[3] = X11MOUSE_OFFSET + ip.Event.MouseEvent.dwMousePosition.X + 1;
2846                               x11mousebuf[4] = X11MOUSE_OFFSET + ip.Event.MouseEvent.dwMousePosition.Y + 1;
2847                               switch (ip.Event.MouseEvent.dwEventFlags)
2848                               {
2849                               case 0: /* press or release */
2850                                         if (ip.Event.MouseEvent.dwButtonState == 0)
2851                                                   x11mousebuf[2] = X11MOUSE_OFFSET + X11MOUSE_BUTTON_REL;
2852                                         else if (ip.Event.MouseEvent.dwButtonState & (FROM_LEFT_3RD_BUTTON_PRESSED | FROM_LEFT_4TH_BUTTON_PRESSED))
2853                                                   continue;
2854                                         else
2855                                                   x11mousebuf[2] = X11MOUSE_OFFSET + X11MOUSE_BUTTON1 + ((int)ip.Event.MouseEvent.dwButtonState << 1);
2856                                         break;
2857                               case MOUSE_WHEELED:
2858                                         x11mousebuf[2] = X11MOUSE_OFFSET + (((int)ip.Event.MouseEvent.dwButtonState < 0) ? X11MOUSE_WHEEL_DOWN : X11MOUSE_WHEEL_UP);
2859                                         break;
2860                               default:
2861                                         continue;
2862                               }
2863                               x11mousePos = 0;
2864                               x11mouseCount = 5;
2865                               currentKey.ascii = ESC;
2866                               keyCount = 1;
2867                               return (TRUE);
2868                     }
2869           } while (ip.EventType != KEY_EVENT ||
2870                     ip.Event.KeyEvent.bKeyDown != TRUE ||
2871                     (ip.Event.KeyEvent.wVirtualScanCode == 0 && ip.Event.KeyEvent.uChar.UnicodeChar == 0) ||
2872                     ((ip.Event.KeyEvent.dwControlKeyState & (RIGHT_ALT_PRESSED|LEFT_CTRL_PRESSED)) == (RIGHT_ALT_PRESSED|LEFT_CTRL_PRESSED) && ip.Event.KeyEvent.uChar.UnicodeChar == 0) ||
2873                     ip.Event.KeyEvent.wVirtualKeyCode == VK_KANJI ||
2874                     ip.Event.KeyEvent.wVirtualKeyCode == VK_SHIFT ||
2875                     ip.Event.KeyEvent.wVirtualKeyCode == VK_CONTROL ||
2876                     ip.Event.KeyEvent.wVirtualKeyCode == VK_MENU);
2877 
2878           currentKey.unicode = ip.Event.KeyEvent.uChar.UnicodeChar;
2879           currentKey.ascii = ip.Event.KeyEvent.uChar.AsciiChar;
2880           currentKey.scan = ip.Event.KeyEvent.wVirtualScanCode;
2881           keyCount = ip.Event.KeyEvent.wRepeatCount;
2882 
2883           if (ip.Event.KeyEvent.dwControlKeyState &
2884                     (LEFT_ALT_PRESSED | RIGHT_ALT_PRESSED))
2885           {
2886                     switch (currentKey.scan)
2887                     {
2888                     case PCK_ALT_E:     /* letter 'E' */
2889                               currentKey.ascii = 0;
2890                               break;
2891                     }
2892           } else if (ip.Event.KeyEvent.dwControlKeyState &
2893                     (LEFT_CTRL_PRESSED | RIGHT_CTRL_PRESSED))
2894           {
2895                     switch (currentKey.scan)
2896                     {
2897                     case PCK_RIGHT: /* right arrow */
2898                               currentKey.scan = PCK_CTL_RIGHT;
2899                               break;
2900                     case PCK_LEFT: /* left arrow */
2901                               currentKey.scan = PCK_CTL_LEFT;
2902                               break;
2903                     case PCK_DELETE: /* delete */
2904                               currentKey.scan = PCK_CTL_DELETE;
2905                               break;
2906                     }
2907           } else if (ip.Event.KeyEvent.dwControlKeyState & SHIFT_PRESSED)
2908           {
2909                     switch (currentKey.scan)
2910                     {
2911                     case PCK_SHIFT_TAB: /* tab */
2912                               currentKey.ascii = 0;
2913                               break;
2914                     }
2915           }
2916 
2917           return (TRUE);
2918 }
2919 
2920 /*
2921  * Read a character from the keyboard.
2922  *
2923  * Known issues:
2924  * - WIN32getch API should be int like libc (with unsigned char values or -1).
2925  * - The unicode code below can return 0 - incorrectly indicating scan code.
2926  * - UTF16-LE surrogate pairs don't work (and return 0).
2927  * - If win32_kbhit returns true then WIN32getch should never block, but it
2928  *   will block till the next keypress if it's numlock/capslock scan code.
2929  */
WIN32getch(void)2930 public char WIN32getch(void)
2931 {
2932           char ascii;
2933           static unsigned char utf8[UTF8_MAX_LENGTH];
2934           static int utf8_size = 0;
2935           static int utf8_next_byte = 0;
2936 
2937           if (win_unget_pending)
2938           {
2939                     win_unget_pending = FALSE;
2940                     return (char) win_unget_data;
2941           }
2942 
2943           // Return the rest of multibyte character from the prior call
2944           if (utf8_next_byte < utf8_size)
2945           {
2946                     ascii = utf8[utf8_next_byte++];
2947                     return ascii;
2948           }
2949           utf8_size = 0;
2950 
2951           if (pending_scancode)
2952           {
2953                     pending_scancode = 0;
2954                     return ((char)(currentKey.scan & 0x00FF));
2955           }
2956 
2957           do {
2958                     while (!win32_kbhit())
2959                     {
2960                               Sleep(20);
2961                               if (ABORT_SIGS())
2962                                         return ('\003');
2963                     }
2964                     keyCount--;
2965                     // If multibyte character, return its first byte
2966                     if (currentKey.unicode > 0x7f)
2967                     {
2968                               utf8_size = WideCharToMultiByte(CP_UTF8, 0, &currentKey.unicode, 1, (LPSTR) &utf8, sizeof(utf8), NULL, NULL);
2969                               if (utf8_size == 0)
2970                                         return '\0';
2971                               ascii = utf8[0];
2972                               utf8_next_byte = 1;
2973                     } else
2974                               ascii = currentKey.ascii;
2975                     /*
2976                      * On PC's, the extended keys return a 2 byte sequence beginning
2977                      * with '00', so if the ascii code is 00, the next byte will be
2978                      * the lsb of the scan code.
2979                      */
2980                     pending_scancode = (ascii == 0x00);
2981           } while (pending_scancode &&
2982                     (currentKey.scan == PCK_CAPS_LOCK || currentKey.scan == PCK_NUM_LOCK));
2983 
2984           return ascii;
2985 }
2986 
2987 /*
2988  * Make the next call to WIN32getch return ch without changing the queue state.
2989  */
WIN32ungetch(int ch)2990 public void WIN32ungetch(int ch)
2991 {
2992           win_unget_pending = TRUE;
2993           win_unget_data = ch;
2994 }
2995 #endif
2996 
2997 #if MSDOS_COMPILER
2998 /*
2999  */
WIN32setcolors(int fg,int bg)3000 public void WIN32setcolors(int fg, int bg)
3001 {
3002           SETCOLORS(fg, bg);
3003 }
3004 
3005 /*
3006  */
WIN32textout(char * text,int len)3007 public void WIN32textout(char *text, int len)
3008 {
3009 #if MSDOS_COMPILER==WIN32C
3010           DWORD written;
3011           if (utf_mode == 2)
3012           {
3013                     /*
3014                      * We've got UTF-8 text in a non-UTF-8 console.  Convert it to
3015                      * wide and use WriteConsoleW.
3016                      */
3017                     WCHAR wtext[1024];
3018                     len = MultiByteToWideChar(CP_UTF8, 0, text, len, wtext,
3019                                                     sizeof(wtext)/sizeof(*wtext));
3020                     WriteConsoleW(con_out, wtext, len, &written, NULL);
3021           } else
3022                     WriteConsole(con_out, text, len, &written, NULL);
3023 #else
3024           char c = text[len];
3025           text[len] = '\0';
3026           cputs(text);
3027           text[len] = c;
3028 #endif
3029 }
3030 #endif
3031 
3032