C按键

时间:2016-01-18 17:20:15

标签: c terminal key

我正在尝试在C中实现一种终端。我正在尝试在上下键压力下构建终端历史记录,但如果按下这些键,光标将向上(或向下)一行。如何“控制”这个功能?

编辑: 我们来举个例子:

fgets(cmd, sizeof(cmd), stdin);

    if (cmd[0] == '\033')
        if (cmd[2] == 'A')
            printf("up-arrow pressed!\n");

这段代码太可怕了,但这是一个很好的起点。有用。问题是如果你按向上箭头键(光标),它会在上排移动并打印“向上箭头按下!”。如何在没有外部库的情况下避免光标上下移动?

2 个答案:

答案 0 :(得分:1)

对于启动器,一个如何初始化终端和一些基本代码的示例:

#include <stdio.h>
#include <termios.h>

int main (void) {

        struct termios t;
        tcgetattr(0, &t);
        t.c_lflag &= ~(ECHO | ICANON);
        tcsetattr(0, TCSANOW, &t);

        while (1) {
                char ch = getchar();
                if (ch == '\x1b') {
                        do {
                                ch = getchar();
                        } while (!isalpha(ch));
                        printf("Got escape sequence: ");
                        switch (ch) {
                                case 'A':
                                        printf("Cursor up\n");
                                        break;
                                case 'B':
                                        printf("Cursor down\n");
                                        break;
                                case 'D':
                                        printf("Cursor left\n");
                                        break;
                                case 'C':
                                        printf("Cursor right\n");
                                        break;
                                default:
                                        printf("%c\n", ch);
                        }
                        // handle escape sequence
                } else {
                        printf("Got %c\n", ch);
                }
        }

        return 0;
}

现在你必须输出输入的字符,并在收到cursor- *或其他一些控制序列(而不是debug - printf s时,通过发送相应的转义序列来移动以上)。您必须密切跟踪当前的屏幕布局。

正如我在评论中已经提到的,这是一项非常繁琐的工作,除了学习(或者可能是嵌入式系统)之外,与使用libreadline或类似的API相比,我没有看到任何好处。

答案 1 :(得分:1)

除了使用^C之外,真正的应用程序将提供退出程序。这是一个示例,修改@ ctx的示例以执行此操作(并改进转义序列的处理),当读取连续的两个转义时退出:

#include <stdio.h>
#include <termios.h>

int main (void) {

        struct termios t, save;
        int last;
        tcgetattr(0, &t);
        save = t;
        t.c_lflag &= ~(ECHO | ICANON);
        tcsetattr(0, TCSANOW, &t);

        while (1) {
                int ch = getchar();
                if (ch == '\033') {
                    if (last == ch)
                            break;
                        do {
                            last = ch;
                            ch = getchar();
                        } while (ispunct(ch) || isdigit(ch));
                        if (isalpha(ch)) {
                            printf("Got escape sequence: ");
                            switch (ch) {
                                case 'A':
                                        printf("Cursor up\n");
                                        break;
                                case 'B':
                                        printf("Cursor down\n");
                                        break;
                                case 'D':
                                        printf("Cursor left\n");
                                        break;
                                case 'C':
                                        printf("Cursor right\n");
                                        break;
                                default:
                                    printf("%c\n", ch);
                                    break;
                                    break;
                            }
                        }
                } else if (ch > 0) {
                        printf("Got %c\n", ch);
                }
            last = ch;
        }

        tcsetattr(0, TCSANOW, &save);
        return 0;
}

在curses中,等效(使用不同的消息)更简单:

#include <curses.h>

int main (void) {
    int last = 0;

    filter();
    initscr();
    noecho();
    cbreak();
    keypad(stdscr, TRUE);

    while (1) {
            int ch = getch();
            if (ch >= 0) {
                    if (ch >= KEY_MIN) {
                            printf("Got special key %s\r\n", keyname(ch));
                    } else {
                            if (ch == '\033' && last == ch)
                                    break;
                            printf("Got %s\r\n", unctrl(ch));
                    }
                    last = ch;
                    fflush(stdout);
            }
        }

        endwin();
        return 0;
}