terminal.cpp 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411
  1. /*
  2. Terminal tracking
  3. Actually, I believe I only really need to track the color information.
  4. Everything else, I'm not sure I really care about. (NNY!)
  5. */
  6. #include <ctype.h>
  7. #include <stdio.h> // snprintf
  8. #include <stdlib.h>
  9. #include <string.h>
  10. #include "terminal.h"
  11. #include "utils.h"
  12. #include "zf_log.h"
  13. void console_init(struct console_details *cdp) {
  14. cdp->posx = 0;
  15. cdp->posy = 0;
  16. cdp->savedx = 0;
  17. cdp->savedy = 0;
  18. cdp->in_ansi = 0;
  19. cdp->fgcolor = 7;
  20. cdp->bgcolor = 0;
  21. cdp->status = 0;
  22. }
  23. void ansi_color(struct console_details *cdp, int color) {
  24. // ZF_LOGV("ansi_color(%d)", color);
  25. if (color == 0) {
  26. cdp->status = 0;
  27. cdp->fgcolor = 7;
  28. cdp->bgcolor = 0;
  29. return;
  30. }
  31. if (color == 1) {
  32. cdp->status = 1;
  33. return;
  34. }
  35. if (color == 5) {
  36. cdp->status = 5;
  37. return;
  38. }
  39. if ((color >= 30) && (color <= 37)) {
  40. cdp->fgcolor = color - 30;
  41. return;
  42. }
  43. if ((color >= 40) && (color <= 47)) {
  44. cdp->bgcolor = color - 40;
  45. return;
  46. }
  47. if (color == 39) {
  48. // default fg color
  49. cdp->fgcolor = 7;
  50. return;
  51. }
  52. if (color == 49) {
  53. // default bg color
  54. cdp->bgcolor = 0;
  55. return;
  56. }
  57. ZF_LOGD("ansi_color( %d ) is unknown to me.", color);
  58. }
  59. const char *color_restore(struct console_details *cdp) {
  60. static char buffer[30];
  61. int slen;
  62. if (cdp->status == 0) {
  63. slen = snprintf(buffer, sizeof(buffer), "\x1b[0;3%d;4%dm", cdp->fgcolor,
  64. cdp->bgcolor);
  65. if (slen >= sizeof(buffer)) {
  66. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  67. buffer[0] = 0;
  68. }
  69. } else {
  70. slen = snprintf(buffer, sizeof(buffer), "\x1b[0;%d;3%d;4%dm", cdp->status,
  71. cdp->fgcolor, cdp->bgcolor);
  72. if (slen >= sizeof(buffer)) {
  73. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  74. buffer[0] = 0;
  75. }
  76. };
  77. return buffer;
  78. }
  79. ANSI_TYPE console_ansi(struct console_details *cdp, const char *ansi) {
  80. int understood = 0;
  81. const char *cp = ansi;
  82. const char *last = ansi + strlen(ansi) - 1;
  83. int number, number2;
  84. if (*cp == '\x1b') {
  85. cp++;
  86. // Ok, that's expected.
  87. if (*cp == '[') {
  88. cp++;
  89. // https://en.wikipedia.org/wiki/ANSI_escape_code#CSI_sequences
  90. switch (*last) {
  91. case 'A':
  92. // cursor up
  93. if (cp == last) {
  94. number = 1;
  95. } else {
  96. number = atoi(cp);
  97. if (number < 1) {
  98. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  99. number);
  100. number = 1;
  101. }
  102. };
  103. cdp->posy -= number;
  104. if (cdp->posy < 0) {
  105. cdp->posy = 0;
  106. ZF_LOGD(
  107. "console_ansi( %s ): attempt to move above top of screen (%d)",
  108. repr(ansi), number);
  109. }
  110. understood = 1;
  111. return CURSOR;
  112. case 'B':
  113. // cursor down
  114. if (cp == last) {
  115. number = 1;
  116. } else {
  117. number = atoi(cp);
  118. if (number < 1) {
  119. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  120. number);
  121. number = 1;
  122. }
  123. };
  124. cdp->posy += number;
  125. // check range/"scroll"
  126. understood = 1;
  127. return CURSOR;
  128. case 'C':
  129. // cursor forward
  130. if (cp == last) {
  131. number = 1;
  132. } else {
  133. number = atoi(cp);
  134. if (number < 1) {
  135. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  136. number);
  137. number = 1;
  138. }
  139. };
  140. cdp->posx += number;
  141. // Well. According to the "spec", the screen limits are hard
  142. // If the cursor is already at the edge of the screen, this has no
  143. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  144. while (cdp->posx > 79) {
  145. cdp->posy++;
  146. // check range/"scroll"
  147. cdp->posx -= 79;
  148. }
  149. understood = 1;
  150. return CURSOR;
  151. case 'D':
  152. // cursor backwards
  153. if (cp == last) {
  154. number = 1;
  155. } else {
  156. number = atoi(cp);
  157. if (number < 1) {
  158. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  159. number);
  160. number = 0;
  161. }
  162. };
  163. cdp->posx -= number;
  164. // Well. According to the "spec", the screen limits are hard
  165. // If the cursor is already at the edge of the screen, this has no
  166. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  167. while (cdp->posx < 0) {
  168. cdp->posy--;
  169. if (cdp->posy < 0) {
  170. cdp->posy = 0;
  171. }
  172. cdp->posx += 79;
  173. }
  174. understood = 1;
  175. return CURSOR;
  176. case 'H':
  177. // cursor position
  178. if (*cp == ';') {
  179. // Missing first number
  180. number = 1;
  181. cp++;
  182. if (cp == last) {
  183. // missing 2nd number as well?
  184. number2 = 1;
  185. } else {
  186. number2 = atoi(cp);
  187. }
  188. } else {
  189. // Ok, find the first number
  190. number = atoi(cp);
  191. cp = strchr(cp, ';');
  192. if (cp == NULL) {
  193. // Missing 2nd number
  194. number2 = 1;
  195. } else {
  196. // 2nd number found, maybe.
  197. cp++;
  198. if (cp == last) {
  199. number2 = 1;
  200. } else {
  201. number2 = atoi(cp);
  202. }
  203. }
  204. }
  205. // Our positions start at zero, not one.
  206. cdp->posx = number - 1;
  207. cdp->posy = number2 - 1;
  208. understood = 1;
  209. return CURSOR;
  210. case 'J':
  211. // clear
  212. if (cp == last) {
  213. number = 0;
  214. } else {
  215. number = atoi(cp);
  216. };
  217. // clears ... part of the screen.
  218. if (number == 2) {
  219. cdp->posx = 0;
  220. cdp->posy = 0;
  221. };
  222. understood = 1;
  223. return CLEAR;
  224. case 'm':
  225. // color
  226. if (cp == last) {
  227. // nothing given, default to 0.
  228. number = 0;
  229. ansi_color(cdp, number);
  230. } else {
  231. while (cp != last) {
  232. number = atoi(cp);
  233. ansi_color(cdp, number);
  234. cp++;
  235. while ((cp != last) && (isdigit(*cp))) {
  236. cp++;
  237. };
  238. if (cp != last) {
  239. if (*cp == ';') {
  240. cp++;
  241. }
  242. }
  243. }
  244. }
  245. understood = 1;
  246. return COLOR;
  247. case 't':
  248. case 'r':
  249. case 'h':
  250. case '!':
  251. // These are ones that I don't care about.
  252. case 'n': // This is terminal detect -- give me cursor position
  253. understood = 1;
  254. return OTHER;
  255. default:
  256. // unsure -- possibly not important
  257. ZF_LOGD("console_ansi( %s ): ???", repr(ansi));
  258. understood = 0;
  259. return OTHER;
  260. }
  261. }
  262. }
  263. }
  264. /*
  265. * console_char()
  266. * return whether or not we are still in_ansi
  267. */
  268. termchar console_char(struct console_details *cdp, char ch) {
  269. char *cp;
  270. termchar tc;
  271. if (cdp->in_ansi) {
  272. // Ok, append this char
  273. cp = cdp->ansi + strlen(cdp->ansi);
  274. *cp = ch;
  275. cp++;
  276. *cp = 0;
  277. if (isalpha(ch)) {
  278. // Ok!
  279. tc.ansi = console_ansi(cdp, cdp->ansi);
  280. cdp->in_ansi = 0;
  281. cdp->ansi[0] = 0;
  282. tc.in_ansi = 1;
  283. return tc;
  284. }
  285. tc.ansi = START;
  286. tc.in_ansi = 1;
  287. return tc;
  288. } else {
  289. tc.ansi = START;
  290. tc.in_ansi = 0;
  291. if (ch == '\x1b') {
  292. cp = cdp->ansi;
  293. *cp = ch;
  294. cp++;
  295. *cp = 0;
  296. cdp->in_ansi = 1;
  297. tc.in_ansi = 1;
  298. return tc;
  299. }
  300. // should I try reporting MOTION non-ANSI ?
  301. if (ch == '\r') {
  302. // Carriage return
  303. cdp->posx = 0;
  304. return tc;
  305. }
  306. if (ch == '\n') {
  307. cdp->posy++;
  308. // check range/"scroll"
  309. return tc;
  310. }
  311. if (ch == '\b') {
  312. // Backspace.
  313. if (cdp->posx > 0) {
  314. cdp->posx--;
  315. }
  316. return tc;
  317. }
  318. if (ch == '\f') {
  319. // form feed
  320. // treat as clear screen
  321. cdp->posx = 0;
  322. cdp->posy = 0;
  323. return tc;
  324. }
  325. /*
  326. I don't believe that anything else can possibly be here, other then an
  327. actual printable character. So!
  328. */
  329. cdp->posx++;
  330. if (cdp->posx > 79) {
  331. cdp->posx = 0;
  332. cdp->posy++;
  333. // check range/"scroll"
  334. }
  335. return tc;
  336. }
  337. }
  338. #ifdef UNWANTED
  339. void console_string(struct console_details *cdp, const char *chars) {
  340. int x;
  341. for (x = 0; x < strlen(chars); x++) {
  342. console_char(cdp, chars[x]);
  343. }
  344. }
  345. #endif
  346. // extern void log_flush(void);
  347. void console_receive(struct console_details *cdp, std::string chars) {
  348. ZF_LOGI("console_char %lu chars", chars.size());
  349. /* // the c way
  350. for (int x = 0; x < chars.size(); x++) {
  351. console_char(cdp, chars[x]);
  352. }
  353. */
  354. // the C++ way
  355. for (auto strit = chars.begin(); strit != chars.end(); strit++)
  356. console_char(cdp, *strit);
  357. }
  358. #ifdef UNWANTED
  359. void console_receive(struct console_details *cdp, const char *chars, int len) {
  360. int x;
  361. for (x = 0; x < len; x++) {
  362. console_char(cdp, chars[x]);
  363. }
  364. }
  365. #endif