getkey.cpp 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. #include <iostream>
  2. // raw mode
  3. #include <termios.h>
  4. #include <unistd.h>
  5. #include "door.h"
  6. // let's try this!
  7. #include <signal.h>
  8. #include <unistd.h>
  9. /**
  10. * @file
  11. * @brief Key and door input routines
  12. */
  13. /*
  14. void done(int signal) {
  15. std::cout << "\r\nWORP WORP\r\n";
  16. std::cout.flush();
  17. }
  18. */
  19. #include <ctype.h>
  20. /**
  21. * @brief Original terminal termios defaults.
  22. */
  23. struct termios tio_default;
  24. /**
  25. * @brief Enable terminal raw mode.
  26. *
  27. * This sets up the linux console so the door library will work correctly in
  28. * local mode.
  29. */
  30. void raw(void) {
  31. // enable terminal RAW mode
  32. struct termios tio_raw;
  33. tcgetattr(STDIN_FILENO, &tio_default);
  34. tio_raw = tio_default;
  35. cfmakeraw(&tio_raw);
  36. // Ok! I need the extra sauce here
  37. tio_raw.c_cc[VMIN] = 0;
  38. tio_raw.c_cc[VTIME] = 1;
  39. // tio_raw.c_iflag &= ~(ICRNL | IXON);
  40. tcsetattr(STDIN_FILENO, TCSANOW, &tio_raw);
  41. }
  42. /**
  43. * @brief Reset the terminal termios to the original values.
  44. */
  45. void reset(void) { tcsetattr(STDIN_FILENO, TCOFLUSH, &tio_default); }
  46. /**
  47. * @brief used by output routines.
  48. *
  49. * Sending "\n" isn't enough.
  50. */
  51. #define CRNL "\r\n"
  52. /*
  53. NOTE: cr (from syncterm), gives 0x0d 0x00
  54. */
  55. /**
  56. * @brief low level getch key read
  57. *
  58. * This reads a key with a defined timeout value.
  59. * This is called by other routines to handle arrow keys, F-keys.
  60. * returns -1 on timeout (no key), -2 on error (connection closed)
  61. * @return signed int
  62. */
  63. signed int getch(void) {
  64. fd_set socket_set;
  65. struct timeval tv;
  66. int select_ret = -1;
  67. int recv_ret;
  68. char key;
  69. while (select_ret == -1) {
  70. FD_ZERO(&socket_set);
  71. FD_SET(STDIN_FILENO, &socket_set);
  72. tv.tv_sec = 0;
  73. tv.tv_usec = 100;
  74. select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
  75. // select(STDIN_FILENO + 1, &socket_set, NULL, NULL, bWait ? NULL : &tv);
  76. if (select_ret == -1) {
  77. if (errno == EINTR)
  78. continue;
  79. return (-2);
  80. }
  81. if (select_ret == 0)
  82. return (-1);
  83. }
  84. recv_ret = read(STDIN_FILENO, &key, 1);
  85. if (recv_ret != 1) {
  86. std::cout << "eof?" << CRNL;
  87. std::cout.flush();
  88. return -2;
  89. }
  90. return key;
  91. }
  92. /**
  93. * @brief pushback buffer to store keys we're not ready for yet.
  94. */
  95. char buffer[10];
  96. /**
  97. * @brief pushback buffer position
  98. */
  99. int bpos = 0;
  100. /**
  101. * @brief ungets (pushes key back)
  102. *
  103. * If we read ahead, and we can't use it, we push it back into the buffer for
  104. * next time.
  105. * @param c
  106. */
  107. void unget(char c) {
  108. if (bpos < sizeof(buffer) - 1) {
  109. buffer[bpos] = c;
  110. bpos++;
  111. }
  112. }
  113. /**
  114. * @brief get a key from the pushback buffer.
  115. *
  116. * @return char
  117. */
  118. char get(void) {
  119. if (bpos == 0) {
  120. return 0;
  121. }
  122. bpos--;
  123. char c = buffer[bpos];
  124. return c;
  125. }
  126. /**
  127. * @brief high level getkey
  128. *
  129. * This returns function keys, arrow keys, see XKEY_* defines.
  130. * returns -1 (no key avaiable) or -2 (hangup)
  131. * or XKEY_UNKNOWN (don't know what it is)
  132. *
  133. * @return signed int
  134. */
  135. signed int getkey(void) {
  136. signed int c, c2;
  137. // consume pushback buffer before reading more keys.
  138. if (bpos != 0) {
  139. c = get();
  140. } else {
  141. c = getch();
  142. };
  143. if (c < 0)
  144. return c;
  145. /*
  146. What happens: syncterm gives us 0x0d 0x00 on [Enter].
  147. This strips out the possible null.
  148. */
  149. if (c == 0x0d) {
  150. c2 = getch();
  151. if ((c2 != 0) and (c2 >= 0))
  152. unget(c2);
  153. return c;
  154. }
  155. if (c == 0x1b) {
  156. // possible extended key
  157. c2 = getch();
  158. if (c2 < 0) {
  159. // nope, just plain ESC
  160. return c;
  161. }
  162. char extended[10];
  163. int pos = 0;
  164. extended[pos] = (char)c2;
  165. extended[pos + 1] = 0;
  166. pos++;
  167. while ((pos < sizeof(extended) - 1) and ((c2 = getch()) >= 0)) {
  168. // handle special case when I'm smashing out cursor location requests
  169. // and \x1b[X;YR strings get buffered
  170. if (c2 == 0x1b) {
  171. unget(c2);
  172. break;
  173. }
  174. extended[pos] = (char)c2;
  175. extended[pos + 1] = 0;
  176. pos++;
  177. }
  178. // FUTURE: Display debug when we fail to identify the key
  179. #ifdef DEBUG_OUTPUT
  180. std::cout << CRNL "DEBUG:" CRNL "ESC + ";
  181. for (int x = 0; x < pos; x++) {
  182. char z = extended[x];
  183. if (iscntrl(z)) {
  184. std::cout << (int)z << " ";
  185. } else {
  186. std::cout << "'" << (char)z << "'"
  187. << " ";
  188. };
  189. }
  190. #endif
  191. if (extended[0] == '[') {
  192. switch (extended[1]) {
  193. case 'A':
  194. return XKEY_UP_ARROW;
  195. case 'B':
  196. return XKEY_DOWN_ARROW;
  197. case 'C':
  198. return XKEY_RIGHT_ARROW;
  199. case 'D':
  200. return XKEY_LEFT_ARROW;
  201. case 'H':
  202. return XKEY_HOME;
  203. case 'F':
  204. return XKEY_END; // terminal
  205. case 'K':
  206. return XKEY_END;
  207. case 'U':
  208. return XKEY_PGUP;
  209. case 'V':
  210. return XKEY_PGDN;
  211. case '@':
  212. return XKEY_INSERT;
  213. };
  214. if (extended[pos - 1] == '~') {
  215. // This ends with ~
  216. int number = atoi(extended + 1);
  217. switch (number) {
  218. case 2:
  219. return XKEY_INSERT; // terminal
  220. case 3:
  221. return XKEY_DELETE; // terminal
  222. case 5:
  223. return XKEY_PGUP; // terminal
  224. case 6:
  225. return XKEY_PGDN; // terminal
  226. case 15:
  227. return XKEY_F5; // terminal
  228. case 17:
  229. return XKEY_F6; // terminal
  230. case 18:
  231. return XKEY_F7; // terminal
  232. case 19:
  233. return XKEY_F8; // terminal
  234. case 20:
  235. return XKEY_F9; // terminal
  236. case 21:
  237. return XKEY_F10; // terminal
  238. case 23:
  239. return XKEY_F11;
  240. case 24:
  241. return XKEY_F12; // terminal
  242. }
  243. }
  244. }
  245. if (extended[0] == 'O') {
  246. switch (extended[1]) {
  247. case 'P':
  248. return XKEY_F1;
  249. case 'Q':
  250. return XKEY_F2;
  251. case 'R':
  252. return XKEY_F3;
  253. case 'S':
  254. return XKEY_F4;
  255. case 't':
  256. return XKEY_F5; // syncterm
  257. }
  258. }
  259. // unknown -- display debug output
  260. std::cout << CRNL "DEBUG:" CRNL "ESC + ";
  261. for (int x = 0; x < pos; x++) {
  262. char z = extended[x];
  263. if (iscntrl(z)) {
  264. std::cout << (int)z << " ";
  265. } else {
  266. std::cout << "'" << (char)z << "'"
  267. << " ";
  268. };
  269. }
  270. return XKEY_UNKNOWN;
  271. }
  272. return c;
  273. }