terminal.cpp 9.2 KB

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