Configure key repeat delay to detect if key is being pressed

前端 未结 2 1160
春和景丽
春和景丽 2021-01-14 11:46

I am writing a program in C that uses ncurses to check if a key is being pressed. The problem is that there is a key repeat delay.

If I for example hold the key \'a\

相关标签:
2条回答
  • 2021-01-14 12:07

    With ncurses (any curses implementation), you would use getch rather than getchar. The latter is a C standard I/O input function.

    Someone suggested Create a function to check for key press in unix using ncurses, which contains an answer worth mentioning. It uses nodelay to eliminate the time normally spent in getch for successive bytes of an escape sequence. In curses, you always have a tradeoff between waiting or not, since an escape sequence may not arrive all in one read operation. The example shown there reports cases when no character is available, and pauses (sleeps) for a short time in that case.

    If you only want to see the characters which are read, you could eliminate that pause (but making your program use a lot of CPU time):

    #include <ncurses.h>
    
    int kbhit(void)
    {
        int ch = getch();
    
        if (ch != ERR) {
            ungetch(ch);
            return 1;
        } else {
            return 0;
        }
    }
    
    int main(void)
    {
        initscr();
    
        cbreak();
        noecho();
        nodelay(stdscr, TRUE);
    
        scrollok(stdscr, TRUE);
        while (1) {
            if (kbhit()) {
                printw("Key pressed! It was: %d\n", getch());
            }
        }
    }
    

    or (recognizing that there is a tradeoff), use napms to pause a short amount of time, but lessening the CPU time used:

    #include <ncurses.h>
    
    int kbhit(void)
    {
        int ch = getch();
    
        if (ch != ERR) {
            ungetch(ch);
            return 1;
        } else {
            return 0;
        }
    }
    
    int main(void)
    {
        initscr();
    
        cbreak();
        noecho();
        nodelay(stdscr, TRUE);
    
        scrollok(stdscr, TRUE);
        while (1) {
            if (kbhit()) {
                printw("Key pressed! It was: %d\n", getch());
            } else {
                napms(20);
            }
        }
    }
    
    0 讨论(0)
  • 2021-01-14 12:17

    Try this to ignore buffered key repeats:

    int key;
    if ((key = getch()) != ERR) {
      while (getch() == key);
    }
    

    In context:

    //example.c
    
    #include <ncurses.h>
    #include <unistd.h>
    
    int main(int argc, char *argv[]) {
      int pos_x = 0;
      int max_x = 0, max_y = 0;
      int key = 0;
      int on = 1;
    
      initscr();
      noecho();
      cbreak();
      curs_set(FALSE);
      keypad(stdscr, TRUE);
      nodelay(stdscr, TRUE);
      getmaxyx(stdscr,max_y,max_x);
    
      while(1) {
        clear();
        mvprintw(0, 0, "Press, hold and release L-R arrow keys. Press UP/DOWN to toggle function.");
        mvprintw(1, 0, "Skip buffered repeats: %s", (on ? "ON" : "OFF"));
        mvprintw(2, pos_x, "@");
        refresh();
        usleep(50000);
        getmaxyx(stdscr,max_y,max_x);
        key = getch();
    
        // skip buffered repeats                                                                                     
        if (on) {
          if (key != ERR) {
            while (getch() == key);
          }
        }
        //                                                                                                           
    
        switch (key) {
        case KEY_LEFT:
          pos_x += (pos_x > 0 ? -1 : 0); break;
        case KEY_RIGHT:
          pos_x += (pos_x < max_x - 1 ? 1 : 0); break;
        case KEY_UP:
          on = 1; break;
        case KEY_DOWN:
          on = 0; break;
        }
      }
      endwin();
    }
    

    Compile and run with gcc example.c -lncurses -oexample && ./example

    0 讨论(0)
提交回复
热议问题