/****************************************************************************
* Copyright (c) 2006 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 *
* "Software"), to deal in the Software without restriction, including *
* without limitation the rights to use, copy, modify, merge, publish, *
* distribute, distribute with modifications, sublicense, and/or sell *
* copies of the Software, and to permit persons to whom the Software is *
* furnished to do so, subject to the following conditions: *
* *
* The above copyright notice and this permission notice shall be included *
* in all copies or substantial portions of the Software. *
* *
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS *
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF *
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. *
* IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, *
* DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR *
* OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR *
* THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
* *
* Except as contained in this notice, the name(s) of the above copyright *
* holders shall not be used in advertising or otherwise to promote the *
* sale, use or other dealings in this Software without prior written *
* authorization. *
****************************************************************************/
/*
* Author: Thomas E. Dickey, 2006
*
* $Id: foldkeys.c,v 1.4 2010/11/14 01:00:53 tom Exp $
*
* Demonstrate a method for altering key definitions at runtime.
*
* This program reads the key definitions, merging those which have xterm-style
* modifiers into their equivalents which have no modifiers. It does this
* merging only for the keys which are defined in the terminal description.
*/
#include <test.priv.h>
#if defined(NCURSES_VERSION) && NCURSES_EXT_FUNCS
#if TIME_WITH_SYS_TIME
# include <sys/time.h>
# include <time.h>
#else
# if HAVE_SYS_TIME_H
# include <sys/time.h>
# else
# include <time.h>
# endif
#endif
#define MY_LOGFILE "demo_foldkeys.log"
#define MY_KEYS (KEY_MAX + 1)
/*
* Log the most recently-written line to our logfile
*/
static void
log_last_line(WINDOW *win)
{
FILE *fp;
int y, x, n;
char temp[256];
if ((fp = fopen(MY_LOGFILE, "a")) != 0) {
int need = sizeof(temp) - 1;
if (need > COLS)
need = COLS;
getyx(win, y, x);
wmove(win, y - 1, 0);
n = winnstr(win, temp, need);
while (n-- > 0) {
if (isspace(UChar(temp[n])))
temp[n] = '\0';
else
break;
}
wmove(win, y, x);
fprintf(fp, "%s\n", temp);
fclose(fp);
}
}
/*
* ncurses has no API for telling what the actual last key-code is. That is
* a secret because the codes past KEY_MAX are computed at run-time and may
* differ depending on the previous calls to newterm(), etc. It is unlikely
* that one could have more than a thousand key definitions...
*/
#define MAX_KEYS 2000
typedef struct {
const char *name;
const char *value;
int code;
int state;
} KeyInfo;
static void
demo_foldkeys(void)
{
KeyInfo info[MAX_KEYS];
int info_len = 0;
int merged = 0;
int code;
int j, k;
/*
* Tell ncurses that we want to use function keys. That will make it add
* any user-defined keys that appear in the terminfo.
*/
keypad(stdscr, TRUE);
/*
* List the predefined keys using the strnames[] array.
*/
for (code = 0; code < STRCOUNT; ++code) {
NCURSES_CONST char *name = strnames[code];
NCURSES_CONST char *value = tigetstr(name);
if (value != 0 && value != (NCURSES_CONST char *) -1) {
info[info_len].name = strnames[code];
info[info_len].code = key_defined(value);
info[info_len].value = value;
info[info_len].state = 0;
if (info[info_len].code > 0)
++info_len;
}
}
/*
* We can get the names for user-defined keys from keyname(). It returns
* a name like KEY_foo for the predefined keys, which tigetstr() does not
* understand.
*/
for (code = KEY_MAX; code < MAX_KEYS; ++code) {
NCURSES_CONST char *name = keyname(code);
if (name != 0) {
info[info_len].name = name;
info[info_len].code = code;
info[info_len].value = tigetstr(name);
info[info_len].state = 0;
++info_len;
}
}
printw("Initially %d key definitions\n", info_len);
/*
* Look for keys that have xterm-style modifiers.
*/
for (j = 0; j < info_len; ++j) {
int first, second;
char final[2];
char *value;
if (info[j].state == 0
&& sscanf(info[j].value,
"\033[%d;%d%c",
&first,
&second,
final) == 3
&& *final != ';'
&& (value = strdup(info[j].value)) != 0) {
sprintf(value, "\033[%d%c", first, *final);
for (k = 0; k < info_len; ++k) {
if (info[k].state == 0
&& !strcmp(info[k].value, value)) {
info[j].state = 1;
break;
}
}
if (info[j].state == 0) {
sprintf(value, "\033O%c", *final);
for (k = 0; k < info_len; ++k) {
if (info[k].state == 0
&& !strcmp(info[k].value, value)) {
info[j].state = 1;
break;
}
}
}
if (info[j].state == 1) {
if ((define_key(info[j].value, info[k].code)) != ERR) {
printw("map %s to %s\n", info[j].value, info[k].value);
keyok(info[j].code, FALSE);
++merged;
} else {
printw("? cannot define_key %d:%s\n", j, info[j].value);
}
} else {
printw("? cannot merge %d:%s\n", j, info[j].value);
}
free(value);
}
}
printw("Merged to %d key definitions\n", info_len - merged);
}
int
main(int argc GCC_UNUSED, char *argv[]GCC_UNUSED)
{
int ch;
#if HAVE_GETTIMEOFDAY
int secs, msecs;
struct timeval current, previous;
#endif
unlink(MY_LOGFILE);
newterm(0, stdout, stdin);
(void) cbreak(); /* take input chars one at a time, no wait for \n */
(void) noecho(); /* don't echo input */
scrollok(stdscr, TRUE);
keypad(stdscr, TRUE);
move(0, 0);
demo_foldkeys();
#if HAVE_GETTIMEOFDAY
gettimeofday(&previous, 0);
#endif
while ((ch = getch()) != ERR) {
bool escaped = (ch >= MY_KEYS);
const char *name = keyname(escaped ? (ch - MY_KEYS) : ch);
#if HAVE_GETTIMEOFDAY
gettimeofday(¤t, 0);
secs = (int) (current.tv_sec - previous.tv_sec);
msecs = (int) ((current.tv_usec - previous.tv_usec) / 1000);
if (msecs < 0) {
msecs += 1000;
--secs;
}
if (msecs >= 1000) {
secs += msecs / 1000;
msecs %= 1000;
}
printw("%6d.%03d ", secs, msecs);
previous = current;
#endif
printw("Keycode %d, name %s%s\n",
ch,
escaped ? "ESC-" : "",
name != 0 ? name : "<null>");
log_last_line(stdscr);
clrtoeol();
if (ch == 'q')
break;
}
endwin();
ExitProgram(EXIT_SUCCESS);
}
#else
int
main(void)
{
printf("This program requires the ncurses library\n");
ExitProgram(EXIT_FAILURE);
}
#endif