11. 键盘交互

11.1. 基础知识

没有强大的用户界面,GUI是不完整的。为了与用户交互,curses 程序应该能够灵敏地响应用户的按键或鼠标操作。我们先来处理按键。

正如你在几乎所有上述例子中看到的,从用户那里获取按键输入非常容易。一个获取按键输入的简单方法是使用getch()函数。当您有兴趣读取单个按键而不是完整的文本行(通常以回车符结尾)时,应该启用 cbreak 模式来读取按键。应该启用 keypad 以获取功能键、方向键等。有关详细信息,请参阅初始化部分。

getch()返回与按下的键对应的整数。如果它是一个普通字符,则整数值将等同于该字符。否则,它返回一个数字,该数字可以与在以下文件中定义的常量匹配curses.h。 例如,如果用户按下 F1 键,则返回的整数是 265。这可以使用 curses.h 中定义的宏 KEY_F() 来检查。这使得读取按键具有可移植性且易于管理。

例如,如果您像这样调用 getch()

    int ch;

    ch = getch();

getch() 将等待用户按下按键(除非您指定了超时),当用户按下按键时,将返回相应的整数。然后,您可以检查返回的值与 curses.h 中定义的常量进行匹配,以确定您想要的按键。

以下代码片段将完成这项工作。

    if(ch == KEY_LEFT)
        printw("Left arrow is pressed\n");

让我们编写一个小程序,创建一个可以使用向上和向下箭头导航的菜单。

11.2. 一个简单的按键用法示例

示例 10. 一个简单的按键用法示例

#include <stdio.h>
#include <ncurses.h>

#define WIDTH 30
#define HEIGHT 10 

int startx = 0;
int starty = 0;

char *choices[] = { 
			"Choice 1",
			"Choice 2",
			"Choice 3",
			"Choice 4",
			"Exit",
		  };
int n_choices = sizeof(choices) / sizeof(char *);
void print_menu(WINDOW *menu_win, int highlight);

int main()
{	WINDOW *menu_win;
	int highlight = 1;
	int choice = 0;
	int c;

	initscr();
	clear();
	noecho();
	cbreak();	/* Line buffering disabled. pass on everything */
	startx = (80 - WIDTH) / 2;
	starty = (24 - HEIGHT) / 2;
		
	menu_win = newwin(HEIGHT, WIDTH, starty, startx);
	keypad(menu_win, TRUE);
	mvprintw(0, 0, "Use arrow keys to go up and down, Press enter to select a choice");
	refresh();
	print_menu(menu_win, highlight);
	while(1)
	{	c = wgetch(menu_win);
		switch(c)
		{	case KEY_UP:
				if(highlight == 1)
					highlight = n_choices;
				else
					--highlight;
				break;
			case KEY_DOWN:
				if(highlight == n_choices)
					highlight = 1;
				else 
					++highlight;
				break;
			case 10:
				choice = highlight;
				break;
			default:
				mvprintw(24, 0, "Charcter pressed is = %3d Hopefully it can be printed as '%c'", c, c);
				refresh();
				break;
		}
		print_menu(menu_win, highlight);
		if(choice != 0)	/* User did a choice come out of the infinite loop */
			break;
	}	
	mvprintw(23, 0, "You chose choice %d with choice string %s\n", choice, choices[choice - 1]);
	clrtoeol();
	refresh();
	endwin();
	return 0;
}


void print_menu(WINDOW *menu_win, int highlight)
{
	int x, y, i;	

	x = 2;
	y = 2;
	box(menu_win, 0, 0);
	for(i = 0; i < n_choices; ++i)
	{	if(highlight == i + 1) /* High light the present choice */
		{	wattron(menu_win, A_REVERSE); 
			mvwprintw(menu_win, y, x, "%s", choices[i]);
			wattroff(menu_win, A_REVERSE);
		}
		else
			mvwprintw(menu_win, y, x, "%s", choices[i]);
		++y;
	}
	wrefresh(menu_win);
}