terminal.cpp 9.1 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) {
  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. ZF_LOGD("console_ansi( %s ) : ???", repr(ansi));
  264. return OTHER;
  265. }
  266. /*
  267. * console_char()
  268. * return whether or not we are still in_ansi
  269. */
  270. termchar console_char(struct console_details *cdp, char ch) {
  271. char *cp;
  272. termchar tc;
  273. if (cdp->in_ansi) {
  274. // Ok, append this char
  275. cp = cdp->ansi + strlen(cdp->ansi);
  276. *cp = ch;
  277. cp++;
  278. *cp = 0;
  279. if (isalpha(ch)) {
  280. // Ok!
  281. tc.ansi = console_ansi(cdp, cdp->ansi);
  282. cdp->in_ansi = 0;
  283. cdp->ansi[0] = 0;
  284. tc.in_ansi = 1;
  285. return tc;
  286. }
  287. tc.ansi = START;
  288. tc.in_ansi = 1;
  289. return tc;
  290. } else {
  291. tc.ansi = START;
  292. tc.in_ansi = 0;
  293. if (ch == '\x1b') {
  294. cp = cdp->ansi;
  295. *cp = ch;
  296. cp++;
  297. *cp = 0;
  298. cdp->in_ansi = 1;
  299. tc.in_ansi = 1;
  300. return tc;
  301. }
  302. // should I try reporting MOTION non-ANSI ?
  303. if (ch == '\r') {
  304. // Carriage return
  305. cdp->posx = 0;
  306. return tc;
  307. }
  308. if (ch == '\n') {
  309. cdp->posy++;
  310. // check range/"scroll"
  311. return tc;
  312. }
  313. if (ch == '\b') {
  314. // Backspace.
  315. if (cdp->posx > 0) {
  316. cdp->posx--;
  317. }
  318. return tc;
  319. }
  320. if (ch == '\f') {
  321. // form feed
  322. // treat as clear screen
  323. cdp->posx = 0;
  324. cdp->posy = 0;
  325. return tc;
  326. }
  327. /*
  328. I don't believe that anything else can possibly be here, other then an
  329. actual printable character. So!
  330. */
  331. cdp->posx++;
  332. if (cdp->posx > 79) {
  333. cdp->posx = 0;
  334. cdp->posy++;
  335. // check range/"scroll"
  336. }
  337. return tc;
  338. }
  339. }
  340. #ifdef UNWANTED
  341. void console_string(struct console_details *cdp, const char *chars) {
  342. int x;
  343. for (x = 0; x < strlen(chars); x++) {
  344. console_char(cdp, chars[x]);
  345. }
  346. }
  347. #endif
  348. // extern void log_flush(void);
  349. void console_receive(struct console_details *cdp, std::string chars) {
  350. ZF_LOGI("console_char %lu chars", chars.size());
  351. /* // the c way
  352. for (int x = 0; x < chars.size(); x++) {
  353. console_char(cdp, chars[x]);
  354. }
  355. */
  356. // the C++ way
  357. for (auto strit = chars.begin(); strit != chars.end(); strit++)
  358. console_char(cdp, *strit);
  359. }
  360. #ifdef UNWANTED
  361. void console_receive(struct console_details *cdp, const char *chars, int len) {
  362. int x;
  363. for (x = 0; x < len; x++) {
  364. console_char(cdp, chars[x]);
  365. }
  366. }
  367. #endif