terminal.cpp 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430
  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 "terminal.h"
  7. #include "utils.h"
  8. #include "zf_log.h"
  9. #include <ctype.h>
  10. #include <stdio.h> // snprintf
  11. #include <stdlib.h>
  12. #include <string.h>
  13. #include <vector>
  14. void console_init(struct console_details *cdp) {
  15. cdp->posx = 0;
  16. cdp->posy = 0;
  17. cdp->savedx = 0;
  18. cdp->savedy = 0;
  19. cdp->in_ansi = 0;
  20. cdp->fgcolor = 7;
  21. cdp->bgcolor = 0;
  22. cdp->status = 0;
  23. }
  24. void ansi_color(struct console_details *cdp, int color) {
  25. // ZF_LOGV("ansi_color(%d)", color);
  26. if (color == 0) {
  27. cdp->status = 0;
  28. cdp->fgcolor = 7;
  29. cdp->bgcolor = 0;
  30. return;
  31. }
  32. if ((color == 1) || (color == 2) || (color == 3) || (color == 4) ||
  33. (color == 5)) {
  34. cdp->status = color;
  35. return;
  36. }
  37. if ((color >= 30) && (color <= 37)) {
  38. cdp->fgcolor = color - 30;
  39. return;
  40. }
  41. if ((color >= 40) && (color <= 47)) {
  42. cdp->bgcolor = color - 40;
  43. return;
  44. }
  45. if (color == 39) {
  46. // default fg color
  47. cdp->fgcolor = 7;
  48. return;
  49. }
  50. if (color == 49) {
  51. // default bg color
  52. cdp->bgcolor = 0;
  53. return;
  54. }
  55. ZF_LOGD("ansi_color( %d ) is unknown to me.", color);
  56. }
  57. const char *color_restore(struct console_details *cdp) {
  58. static char buffer[30];
  59. int slen;
  60. // possible optimize: If fg is 7, don't set (0 reset does that), if bg is 0,
  61. // don't set (0 does that)
  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 >= (int)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 >= (int)sizeof(buffer)) {
  73. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  74. buffer[0] = 0;
  75. }
  76. };
  77. return buffer;
  78. }
  79. std::vector<std::pair<int, int>> cursor_position_history;
  80. ANSI_TYPE console_ansi(struct console_details *cdp, const char *ansi) {
  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. 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. return CURSOR;
  126. case 'C':
  127. // cursor forward
  128. if (cp == last) {
  129. number = 1;
  130. } else {
  131. number = atoi(cp);
  132. if (number < 1) {
  133. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  134. number);
  135. number = 1;
  136. }
  137. };
  138. cdp->posx += number;
  139. // Well. According to the "spec", the screen limits are hard
  140. // If the cursor is already at the edge of the screen, this has no
  141. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  142. while (cdp->posx > 79) {
  143. cdp->posy++;
  144. // check range/"scroll"
  145. cdp->posx -= 79;
  146. }
  147. return CURSOR;
  148. case 'D':
  149. // cursor backwards
  150. if (cp == last) {
  151. number = 1;
  152. } else {
  153. number = atoi(cp);
  154. if (number < 1) {
  155. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  156. number);
  157. number = 0;
  158. }
  159. };
  160. cdp->posx -= number;
  161. // Well. According to the "spec", the screen limits are hard
  162. // If the cursor is already at the edge of the screen, this has no
  163. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  164. while (cdp->posx < 0) {
  165. cdp->posy--;
  166. if (cdp->posy < 0) {
  167. cdp->posy = 0;
  168. }
  169. cdp->posx += 79;
  170. }
  171. return CURSOR;
  172. case 'H':
  173. // cursor position
  174. if (*cp == ';') {
  175. // Missing first number
  176. number = 1;
  177. cp++;
  178. if (cp == last) {
  179. // missing 2nd number as well?
  180. number2 = 1;
  181. } else {
  182. number2 = atoi(cp);
  183. }
  184. } else {
  185. // Ok, find the first number
  186. number = atoi(cp);
  187. cp = strchr(cp, ';');
  188. if (cp == NULL) {
  189. // Missing 2nd number
  190. number2 = 1;
  191. } else {
  192. // 2nd number found, maybe.
  193. cp++;
  194. if (cp == last) {
  195. number2 = 1;
  196. } else {
  197. number2 = atoi(cp);
  198. }
  199. }
  200. }
  201. // Our positions start at zero, not one.
  202. cdp->posy = number - 1;
  203. cdp->posx = number2 - 1;
  204. return CURSOR;
  205. case 'J':
  206. // clear
  207. if (cp == last) {
  208. number = 0;
  209. } else {
  210. number = atoi(cp);
  211. };
  212. // clears ... part of the screen.
  213. if (number == 2) {
  214. cdp->posx = 0;
  215. cdp->posy = 0;
  216. };
  217. return CLEAR;
  218. case 'K':
  219. // clear line
  220. if (cp == last) {
  221. number = 0;
  222. } else {
  223. number = atoi(cp);
  224. };
  225. return CLEAR;
  226. case 'm':
  227. // color
  228. if (cp == last) {
  229. // nothing given, default to 0.
  230. number = 0;
  231. ansi_color(cdp, number);
  232. } else {
  233. while (cp != last) {
  234. number = atoi(cp);
  235. ansi_color(cdp, number);
  236. cp++;
  237. while ((cp != last) && (isdigit(*cp))) {
  238. cp++;
  239. };
  240. if (cp != last) {
  241. if (*cp == ';') {
  242. cp++;
  243. }
  244. }
  245. }
  246. }
  247. return COLOR;
  248. case 's':
  249. // save position
  250. cursor_position_history.push_back(std::make_pair(cdp->posx, cdp->posy));
  251. return CURSOR;
  252. case 'u':
  253. // restore position
  254. if (cursor_position_history.empty()) {
  255. ZF_LOGE("Restore cursor position from empty history.");
  256. } else {
  257. std::pair<int, int> pos = cursor_position_history.back();
  258. cdp->posx = pos.first;
  259. cdp->posy = pos.second;
  260. cursor_position_history.pop_back();
  261. }
  262. return CURSOR;
  263. case 't':
  264. case 'r':
  265. case 'h':
  266. case '!':
  267. // These are ones that I don't care about.
  268. case 'n': // This is terminal detect -- give me cursor position
  269. return OTHER;
  270. default:
  271. // unsure -- possibly not important
  272. ZF_LOGD("console_ansi( %s ): ???", repr(ansi));
  273. return OTHER;
  274. }
  275. }
  276. }
  277. ZF_LOGD("console_ansi( %s ) : ???", repr(ansi));
  278. return OTHER;
  279. }
  280. /*
  281. * console_char()
  282. * return whether or not we are still in_ansi
  283. */
  284. termchar console_char(struct console_details *cdp, char ch) {
  285. char *cp;
  286. termchar tc;
  287. if (cdp->in_ansi) {
  288. // Ok, append this char
  289. cp = cdp->ansi + strlen(cdp->ansi);
  290. *cp = ch;
  291. cp++;
  292. *cp = 0;
  293. if (isalpha(ch)) {
  294. // Ok!
  295. tc.ansi = console_ansi(cdp, cdp->ansi);
  296. cdp->in_ansi = 0;
  297. cdp->ansi[0] = 0;
  298. tc.in_ansi = 1;
  299. return tc;
  300. }
  301. tc.ansi = START;
  302. tc.in_ansi = 1;
  303. return tc;
  304. } else {
  305. tc.ansi = START;
  306. tc.in_ansi = 0;
  307. if (ch == '\x1b') {
  308. cp = cdp->ansi;
  309. *cp = ch;
  310. cp++;
  311. *cp = 0;
  312. cdp->in_ansi = 1;
  313. tc.in_ansi = 1;
  314. return tc;
  315. }
  316. // should I try reporting MOTION non-ANSI ?
  317. if (ch == '\r') {
  318. // Carriage return
  319. cdp->posx = 0;
  320. return tc;
  321. }
  322. if (ch == '\n') {
  323. cdp->posy++;
  324. // check range/"scroll"
  325. return tc;
  326. }
  327. if (ch == '\b') {
  328. // Backspace.
  329. if (cdp->posx > 0) {
  330. cdp->posx--;
  331. }
  332. return tc;
  333. }
  334. if (ch == '\f') {
  335. // form feed
  336. // treat as clear screen
  337. cdp->posx = 0;
  338. cdp->posy = 0;
  339. return tc;
  340. }
  341. /*
  342. I don't believe that anything else can possibly be here, other then an
  343. actual printable character. So!
  344. */
  345. cdp->posx++;
  346. if (cdp->posx > 79) {
  347. cdp->posx = 0;
  348. cdp->posy++;
  349. // check range/"scroll"
  350. }
  351. return tc;
  352. }
  353. }
  354. #ifdef UNWANTED
  355. void console_string(struct console_details *cdp, const char *chars) {
  356. int x;
  357. for (x = 0; x < strlen(chars); x++) {
  358. console_char(cdp, chars[x]);
  359. }
  360. }
  361. #endif
  362. // extern void log_flush(void);
  363. void console_receive(struct console_details *cdp, std::string chars) {
  364. // This gets noisy when called from render effect ^D
  365. // ZF_LOGI("console_char %lu chars", chars.size());
  366. /* // the c way
  367. for (int x = 0; x < chars.size(); x++) {
  368. console_char(cdp, chars[x]);
  369. }
  370. */
  371. // the C++ way
  372. for (auto strit = chars.begin(); strit != chars.end(); strit++)
  373. console_char(cdp, *strit);
  374. }
  375. #ifdef UNWANTED
  376. void console_receive(struct console_details *cdp, const char *chars, int len) {
  377. int x;
  378. for (x = 0; x < len; x++) {
  379. console_char(cdp, chars[x]);
  380. }
  381. }
  382. #endif