terminal.cpp 9.1 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) || (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 >= (int)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 >= (int)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. const char *cp = ansi;
  80. const char *last = ansi + strlen(ansi) - 1;
  81. int number, number2;
  82. if (*cp == '\x1b') {
  83. cp++;
  84. // Ok, that's expected.
  85. if (*cp == '[') {
  86. cp++;
  87. // https://en.wikipedia.org/wiki/ANSI_escape_code#CSI_sequences
  88. switch (*last) {
  89. case 'A':
  90. // cursor up
  91. if (cp == last) {
  92. number = 1;
  93. } else {
  94. number = atoi(cp);
  95. if (number < 1) {
  96. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  97. number);
  98. number = 1;
  99. }
  100. };
  101. cdp->posy -= number;
  102. if (cdp->posy < 0) {
  103. cdp->posy = 0;
  104. ZF_LOGD(
  105. "console_ansi( %s ): attempt to move above top of screen (%d)",
  106. repr(ansi), number);
  107. }
  108. return CURSOR;
  109. case 'B':
  110. // cursor down
  111. if (cp == last) {
  112. number = 1;
  113. } else {
  114. number = atoi(cp);
  115. if (number < 1) {
  116. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  117. number);
  118. number = 1;
  119. }
  120. };
  121. cdp->posy += number;
  122. // check range/"scroll"
  123. return CURSOR;
  124. case 'C':
  125. // cursor forward
  126. if (cp == last) {
  127. number = 1;
  128. } else {
  129. number = atoi(cp);
  130. if (number < 1) {
  131. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  132. number);
  133. number = 1;
  134. }
  135. };
  136. cdp->posx += number;
  137. // Well. According to the "spec", the screen limits are hard
  138. // If the cursor is already at the edge of the screen, this has no
  139. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  140. while (cdp->posx > 79) {
  141. cdp->posy++;
  142. // check range/"scroll"
  143. cdp->posx -= 79;
  144. }
  145. return CURSOR;
  146. case 'D':
  147. // cursor backwards
  148. if (cp == last) {
  149. number = 1;
  150. } else {
  151. number = atoi(cp);
  152. if (number < 1) {
  153. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  154. number);
  155. number = 0;
  156. }
  157. };
  158. cdp->posx -= number;
  159. // Well. According to the "spec", the screen limits are hard
  160. // If the cursor is already at the edge of the screen, this has no
  161. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  162. while (cdp->posx < 0) {
  163. cdp->posy--;
  164. if (cdp->posy < 0) {
  165. cdp->posy = 0;
  166. }
  167. cdp->posx += 79;
  168. }
  169. return CURSOR;
  170. case 'H':
  171. // cursor position
  172. if (*cp == ';') {
  173. // Missing first number
  174. number = 1;
  175. cp++;
  176. if (cp == last) {
  177. // missing 2nd number as well?
  178. number2 = 1;
  179. } else {
  180. number2 = atoi(cp);
  181. }
  182. } else {
  183. // Ok, find the first number
  184. number = atoi(cp);
  185. cp = strchr(cp, ';');
  186. if (cp == NULL) {
  187. // Missing 2nd number
  188. number2 = 1;
  189. } else {
  190. // 2nd number found, maybe.
  191. cp++;
  192. if (cp == last) {
  193. number2 = 1;
  194. } else {
  195. number2 = atoi(cp);
  196. }
  197. }
  198. }
  199. // Our positions start at zero, not one.
  200. cdp->posx = number - 1;
  201. cdp->posy = number2 - 1;
  202. return CURSOR;
  203. case 'J':
  204. // clear
  205. if (cp == last) {
  206. number = 0;
  207. } else {
  208. number = atoi(cp);
  209. };
  210. // clears ... part of the screen.
  211. if (number == 2) {
  212. cdp->posx = 0;
  213. cdp->posy = 0;
  214. };
  215. return CLEAR;
  216. case 'K':
  217. // clear line
  218. if (cp == last) {
  219. number = 0;
  220. } else {
  221. number = atoi(cp);
  222. };
  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. 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. return OTHER;
  253. default:
  254. // unsure -- possibly not important
  255. ZF_LOGD("console_ansi( %s ): ???", repr(ansi));
  256. return OTHER;
  257. }
  258. }
  259. }
  260. ZF_LOGD("console_ansi( %s ) : ???", repr(ansi));
  261. return OTHER;
  262. }
  263. /*
  264. * console_char()
  265. * return whether or not we are still in_ansi
  266. */
  267. termchar console_char(struct console_details *cdp, char ch) {
  268. char *cp;
  269. termchar tc;
  270. if (cdp->in_ansi) {
  271. // Ok, append this char
  272. cp = cdp->ansi + strlen(cdp->ansi);
  273. *cp = ch;
  274. cp++;
  275. *cp = 0;
  276. if (isalpha(ch)) {
  277. // Ok!
  278. tc.ansi = console_ansi(cdp, cdp->ansi);
  279. cdp->in_ansi = 0;
  280. cdp->ansi[0] = 0;
  281. tc.in_ansi = 1;
  282. return tc;
  283. }
  284. tc.ansi = START;
  285. tc.in_ansi = 1;
  286. return tc;
  287. } else {
  288. tc.ansi = START;
  289. tc.in_ansi = 0;
  290. if (ch == '\x1b') {
  291. cp = cdp->ansi;
  292. *cp = ch;
  293. cp++;
  294. *cp = 0;
  295. cdp->in_ansi = 1;
  296. tc.in_ansi = 1;
  297. return tc;
  298. }
  299. // should I try reporting MOTION non-ANSI ?
  300. if (ch == '\r') {
  301. // Carriage return
  302. cdp->posx = 0;
  303. return tc;
  304. }
  305. if (ch == '\n') {
  306. cdp->posy++;
  307. // check range/"scroll"
  308. return tc;
  309. }
  310. if (ch == '\b') {
  311. // Backspace.
  312. if (cdp->posx > 0) {
  313. cdp->posx--;
  314. }
  315. return tc;
  316. }
  317. if (ch == '\f') {
  318. // form feed
  319. // treat as clear screen
  320. cdp->posx = 0;
  321. cdp->posy = 0;
  322. return tc;
  323. }
  324. /*
  325. I don't believe that anything else can possibly be here, other then an
  326. actual printable character. So!
  327. */
  328. cdp->posx++;
  329. if (cdp->posx > 79) {
  330. cdp->posx = 0;
  331. cdp->posy++;
  332. // check range/"scroll"
  333. }
  334. return tc;
  335. }
  336. }
  337. #ifdef UNWANTED
  338. void console_string(struct console_details *cdp, const char *chars) {
  339. int x;
  340. for (x = 0; x < strlen(chars); x++) {
  341. console_char(cdp, chars[x]);
  342. }
  343. }
  344. #endif
  345. // extern void log_flush(void);
  346. void console_receive(struct console_details *cdp, std::string chars) {
  347. ZF_LOGI("console_char %lu chars", chars.size());
  348. /* // the c way
  349. for (int x = 0; x < chars.size(); x++) {
  350. console_char(cdp, chars[x]);
  351. }
  352. */
  353. // the C++ way
  354. for (auto strit = chars.begin(); strit != chars.end(); strit++)
  355. console_char(cdp, *strit);
  356. }
  357. #ifdef UNWANTED
  358. void console_receive(struct console_details *cdp, const char *chars, int len) {
  359. int x;
  360. for (x = 0; x < len; x++) {
  361. console_char(cdp, chars[x]);
  362. }
  363. }
  364. #endif