Обнаружить событие клавиатуры в C
Как я могу обнаружить событие клавиатуры на языке C без использования сторонних библиотек? Должен ли я использовать обработку сигналов?
5 ответов
Там нет стандартного способа, но они должны начать вас.
Окна:
getch();
Unix:
Используйте этот код из книги У. Ричарда Стивенса по программированию в Unix, чтобы перевести терминал в режим raw, а затем используйте read().
static struct termios save_termios;
static int term_saved;
int tty_raw(int fd) { /* RAW! mode */
struct termios buf;
if (tcgetattr(fd, &save_termios) < 0) /* get the original state */
return -1;
buf = save_termios;
buf.c_lflag &= ~(ECHO | ICANON | IEXTEN | ISIG);
/* echo off, canonical mode off, extended input
processing off, signal chars off */
buf.c_iflag &= ~(BRKINT | ICRNL | ISTRIP | IXON);
/* no SIGINT on BREAK, CR-toNL off, input parity
check off, don't strip the 8th bit on input,
ouput flow control off */
buf.c_cflag &= ~(CSIZE | PARENB);
/* clear size bits, parity checking off */
buf.c_cflag |= CS8;
/* set 8 bits/char */
buf.c_oflag &= ~(OPOST);
/* output processing off */
buf.c_cc[VMIN] = 1; /* 1 byte at a time */
buf.c_cc[VTIME] = 0; /* no timer on input */
if (tcsetattr(fd, TCSAFLUSH, &buf) < 0)
return -1;
term_saved = 1;
return 0;
}
int tty_reset(int fd) { /* set it to normal! */
if (term_saved)
if (tcsetattr(fd, TCSAFLUSH, &save_termios) < 0)
return -1;
return 0;
}
Вы действительно должны использовать сторонние библиотеки. Определенно нет никакого независимого от платформы способа сделать это в ANSI C. Обработка сигналов - не тот способ.
Стандарт C, к сожалению, не имеет никаких средств для обнаружения событий клавиатуры. Вы должны полагаться на специфичные для платформы расширения. Обработка сигналов не поможет вам.
Немного опоздал с ответом, но:
Основная идея состоит в том, чтобы перевести терминал в неканонический режим, чтобы вы могли читать каждое нажатие клавиши индивидуально.
#include <stdio.h>
#include <termios.h>
#include <unistd.h>
void setNonCanonicalMode() {
struct termios term;
tcgetattr(STDIN_FILENO, &term);
term.c_lflag &= ~(ICANON | ECHO);
tcsetattr(STDIN_FILENO, TCSANOW, &term);
}
void resetCanonicalMode() {
struct termios term;
tcgetattr(STDIN_FILENO, &term);
term.c_lflag |= ICANON | ECHO;
tcsetattr(STDIN_FILENO, TCSANOW, &term);
}
Этого должно быть достаточно.