X-Git-Url: http://ncurses.scripts.mit.edu/?p=ncurses.git;a=blobdiff_plain;f=test%2Ffilter.c;h=6cd9206b274d44b1d6646e839c6cba1dabfa52d0;hp=d5bd10a850bb9c4d4d84153e9d3bfadc3e888f22;hb=HEAD;hpb=79185cda09da0a0eef36ab0daabe3b739a2757cb diff --git a/test/filter.c b/test/filter.c index d5bd10a8..b7444569 100644 --- a/test/filter.c +++ b/test/filter.c @@ -1,5 +1,6 @@ /**************************************************************************** - * Copyright (c) 1998-2014,2016 Free Software Foundation, Inc. * + * Copyright 2019-2020,2022 Thomas E. Dickey * + * Copyright 1998-2016,2017 Free Software Foundation, Inc. * * * * Permission is hereby granted, free of charge, to any person obtaining a * * copy of this software and associated documentation files (the * @@ -29,7 +30,7 @@ /* * Author: Thomas E. Dickey 1998 * - * $Id: filter.c,v 1.22 2016/03/13 00:41:43 tom Exp $ + * $Id: filter.c,v 1.38 2022/12/04 00:40:11 tom Exp $ * * An example of the 'filter()' function in ncurses, this program prompts * for commands and executes them (like a command shell). It illustrates @@ -41,6 +42,7 @@ * reset_shell_mode() and reset_prog_mode() functions, we could invoke endwin() * and refresh(), but that does not work any better. */ +#define NEED_KEY_EVENT #include #if HAVE_FILTER @@ -68,10 +70,10 @@ show_prompt(int underline, bool clocked) int skip, y, x; int margin; - sprintf(buffer, "%02d:%02d:%02d", - my->tm_hour, - my->tm_min, - my->tm_sec); + _nc_SPRINTF(buffer, _nc_SLIMIT(sizeof(buffer)) "%02d:%02d:%02d", + my->tm_hour, + my->tm_min, + my->tm_sec); if (limit > 9) { skip = 0; @@ -100,12 +102,11 @@ static int new_command(char *buffer, int length, int underline, bool clocked, bool polled) { int code = OK; - int limit; if (polled) { bool done = FALSE; bool first = TRUE; - int y, x; + int y = 0, x = 0; int n; int mark = 0; int used = 0; @@ -113,6 +114,7 @@ new_command(char *buffer, int length, int underline, bool clocked, bool polled) timeout(20); /* no one types 50CPS... */ while (!done) { + int limit; int ch = getch(); buffer[used] = '\0'; @@ -242,39 +244,87 @@ new_command(char *buffer, int length, int underline, bool clocked, bool polled) */ #ifdef KEY_EVENT if (code == KEY_EVENT) - strcpy(buffer, "# event!"); + _nc_STRCPY(buffer, "# event!", length); #endif #ifdef KEY_RESIZE if (code == KEY_RESIZE) { - strcpy(buffer, "# resize!"); + _nc_STRCPY(buffer, "# resize!", length); getch(); } #endif } attroff(underline); attroff(A_BOLD); - printw("\n"); + refresh(); return code; } +#ifdef NCURSES_VERSION +/* + * Cancel xterm's alternate-screen mode (from dialog -TD) + */ +#define isprivate(s) ((s) != 0 && strstr(s, "\033[?") != 0) +static void +cancel_altscreen(void) +{ + if (isatty(fileno(stdout)) + && key_mouse != 0 /* xterm and kindred */ + && isprivate(enter_ca_mode) + && isprivate(exit_ca_mode)) { + /* + * initscr() or newterm() already wrote enter_ca_mode as a side effect + * of initializing the screen. It would be nice to not even do that, + * but we do not really have access to the correct copy of the + * terminfo description until those functions have been invoked. + */ + (void) refresh(); + (void) putp(exit_ca_mode); + (void) fflush(stdout); + /* + * Prevent ncurses from switching "back" to the normal screen when + * exiting from this program. That would move the cursor to the + * original location saved in xterm. Normally curses sets the cursor + * position to the first line after the display, but the alternate + * screen switching is done after that point. + * + * Cancelling the strings altogether also works around the buggy + * implementation of alternate-screen in rxvt, etc., which clear more + * of the display than they should. + */ + enter_ca_mode = 0; + exit_ca_mode = 0; + } +} +#endif + static void -usage(void) +usage(int ok) { static const char *msg[] = { "Usage: filter [options]" ,"" + ,USAGE_COMMON ,"Options:" - ," -c show current time on prompt line with \"Command\"" - ," -i use initscr() rather than newterm()" - ," -p poll for individual characters rather than using getnstr" +#ifdef NCURSES_VERSION + ," -a suppress xterm alternate-screen by amending smcup/rmcup" +#endif + ," -c show current time on prompt line with \"Command\"" +#if HAVE_USE_DEFAULT_COLORS + ," -d invoke use_default_colors" +#endif + ," -i use initscr() rather than newterm()" + ," -p poll for individual characters rather than using getnstr" }; unsigned n; for (n = 0; n < SIZEOF(msg); n++) fprintf(stderr, "%s\n", msg[n]); - ExitProgram(EXIT_FAILURE); + ExitProgram(ok ? EXIT_SUCCESS : EXIT_FAILURE); } +/* *INDENT-OFF* */ +VERSION_COMMON() +/* *INDENT-ON* */ int main(int argc, char *argv[]) @@ -282,25 +332,45 @@ main(int argc, char *argv[]) int ch; char buffer[80]; int underline; +#ifdef NCURSES_VERSION + bool a_option = FALSE; +#endif bool c_option = FALSE; +#if HAVE_USE_DEFAULT_COLORS + bool d_option = FALSE; +#endif bool i_option = FALSE; bool p_option = FALSE; setlocale(LC_ALL, ""); - while ((ch = getopt(argc, argv, "cip")) != -1) { + while ((ch = getopt(argc, argv, OPTS_COMMON "adcip")) != -1) { switch (ch) { +#ifdef NCURSES_VERSION + case 'a': + a_option = TRUE; + break; +#endif case 'c': c_option = TRUE; break; +#if HAVE_USE_DEFAULT_COLORS + case 'd': + d_option = TRUE; + break; +#endif case 'i': i_option = TRUE; break; case 'p': p_option = TRUE; break; + case OPTS_VERSION: + show_version(argv); + ExitProgram(EXIT_SUCCESS); default: - usage(); + usage(ch == OPTS_USAGE); + /* NOTREACHED */ } } @@ -310,8 +380,16 @@ main(int argc, char *argv[]) if (i_option) { initscr(); } else { - (void) newterm((char *) 0, stdout, stdin); + if (newterm((char *) 0, stdout, stdin) == 0) { + fprintf(stderr, "cannot initialize terminal\n"); + ExitProgram(EXIT_FAILURE); + } + } +#ifdef NCURSES_VERSION + if (a_option) { + cancel_altscreen(); } +#endif cbreak(); keypad(stdscr, TRUE); @@ -319,7 +397,7 @@ main(int argc, char *argv[]) int background = COLOR_BLACK; start_color(); #if HAVE_USE_DEFAULT_COLORS - if (use_default_colors() != ERR) + if (d_option && (use_default_colors() != ERR)) background = -1; #endif init_pair(1, COLOR_CYAN, (short) background); @@ -342,7 +420,7 @@ main(int argc, char *argv[]) erase(); refresh(); } - printw("done"); + clear(); refresh(); endwin(); ExitProgram(EXIT_SUCCESS);