terminal.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788
  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 <sstream>
  11. #include <stdio.h> // snprintf
  12. #include <stdlib.h>
  13. #include <string.h>
  14. #include <vector>
  15. Terminal::Terminal() { this->init(); }
  16. void Terminal::init(void) {
  17. this->posx = this->posy = 0;
  18. this->in_ansi = 0;
  19. this->fgcolor = 7;
  20. this->bgcolor = 0;
  21. this->status = 0;
  22. this->saved_cursor_position.clear();
  23. }
  24. void Terminal::ansi_color(int color) {
  25. // ZF_LOGV("ansi_color(%d)", color);
  26. if (color == 0) {
  27. this->status = 0;
  28. this->fgcolor = 7;
  29. this->bgcolor = 0;
  30. return;
  31. }
  32. if ((color == 1) || (color == 2) || (color == 3) || (color == 4) ||
  33. (color == 5)) {
  34. this->status = color;
  35. return;
  36. }
  37. if ((color >= 30) && (color <= 37)) {
  38. this->fgcolor = color - 30;
  39. return;
  40. }
  41. if ((color >= 40) && (color <= 47)) {
  42. this->bgcolor = color - 40;
  43. return;
  44. }
  45. if (color == 39) {
  46. // default fg color
  47. this->fgcolor = 7;
  48. return;
  49. }
  50. if (color == 49) {
  51. // default bg color
  52. this->bgcolor = 0;
  53. return;
  54. }
  55. ZF_LOGD("ansi_color( %d ) is unknown to me.", color);
  56. }
  57. std::string Terminal::color_restore(void) {
  58. std::ostringstream oss;
  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 (this->status == 0) {
  62. oss << "\x1b[0;" << this->fgcolor + 30 << ";" << this->bgcolor + 40 << "m";
  63. } else {
  64. oss << "\x1b[0;" << this->status << ";" << this->fgcolor + 30 << ";"
  65. << this->bgcolor + 40 << "m";
  66. }
  67. std::string buffer = oss.str();
  68. return buffer;
  69. }
  70. Terminal::ANSI_TYPE Terminal::ansi_code(std::string ansi) {
  71. std::string::iterator cp = ansi.begin();
  72. std::string::iterator last = ansi.end() - 1;
  73. int number, number2;
  74. if (*cp == '\x1b') {
  75. ++cp;
  76. // Ok, that's expected.
  77. if (*cp == '[') {
  78. ++cp;
  79. // https://en.wikipedia.org/wiki/ANSI_escape_code#CSI_sequences
  80. switch (*last) {
  81. case 'A':
  82. // cursor up
  83. if (cp == last) {
  84. number = 1;
  85. } else {
  86. number = std::stoi(std::string(cp, last));
  87. if (number < 1) {
  88. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi.c_str()),
  89. number);
  90. number = 1;
  91. }
  92. };
  93. this->posy -= number;
  94. if (this->posy < 0) {
  95. this->posy = 0;
  96. ZF_LOGD(
  97. "console_ansi( %s ): attempt to move above top of screen (%d)",
  98. repr(ansi.c_str()), number);
  99. }
  100. return CURSOR;
  101. case 'B':
  102. // cursor down
  103. if (cp == last) {
  104. number = 1;
  105. } else {
  106. number = std::stoi(std::string(cp, last));
  107. if (number < 1) {
  108. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi.c_str()),
  109. number);
  110. number = 1;
  111. }
  112. };
  113. this->posy += number;
  114. // check range/"scroll"
  115. return CURSOR;
  116. case 'C':
  117. // cursor forward
  118. if (cp == last) {
  119. number = 1;
  120. } else {
  121. number = std::stoi(std::string(cp, last));
  122. if (number < 1) {
  123. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi.c_str()),
  124. number);
  125. number = 1;
  126. }
  127. };
  128. this->posx += number;
  129. // Well. According to the "spec", the screen limits are hard
  130. // If the cursor is already at the edge of the screen, this has no
  131. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  132. while (this->posx > 79) {
  133. this->posy++;
  134. // check range/"scroll"
  135. this->posx -= 79;
  136. }
  137. return CURSOR;
  138. case 'D':
  139. // cursor backwards
  140. if (cp == last) {
  141. number = 1;
  142. } else {
  143. number = std::stoi(std::string(cp, last));
  144. if (number < 1) {
  145. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi.c_str()),
  146. number);
  147. number = 0;
  148. }
  149. };
  150. this->posx -= number;
  151. // Well. According to the "spec", the screen limits are hard
  152. // If the cursor is already at the edge of the screen, this has no
  153. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  154. while (this->posx < 0) {
  155. this->posy--;
  156. if (this->posy < 0) {
  157. this->posy = 0;
  158. }
  159. this->posx += 79;
  160. }
  161. return CURSOR;
  162. case 'H':
  163. // cursor position
  164. if (*cp == ';') {
  165. // Missing first number
  166. number = 1;
  167. ++cp;
  168. if (cp == last) {
  169. // missing 2nd number as well?
  170. number2 = 1;
  171. } else {
  172. number2 = std::stoi(std::string(cp, last));
  173. }
  174. } else {
  175. // Ok, find the first number
  176. number = std::stoi(std::string(cp, last));
  177. // covert iterator to position
  178. std::size_t pos = ansi.find(';', std::distance(ansi.begin(), cp));
  179. if (pos == std::string::npos) {
  180. // Missing 2nd number
  181. number2 = 1;
  182. } else {
  183. // 2nd number found, maybe.
  184. cp = ansi.begin() + pos;
  185. ++cp;
  186. if (cp == last) {
  187. number2 = 1;
  188. } else {
  189. number2 = std::stoi(std::string(cp, last));
  190. }
  191. }
  192. }
  193. // Our positions start at zero, not one.
  194. this->posy = number - 1;
  195. this->posx = number2 - 1;
  196. return CURSOR;
  197. case 'J':
  198. // clear
  199. if (cp == last) {
  200. number = 0;
  201. } else {
  202. number = std::stoi(std::string(cp, last));
  203. };
  204. // clears ... part of the screen.
  205. if (number == 2) {
  206. this->posx = 0;
  207. this->posy = 0;
  208. };
  209. return CLEAR;
  210. case 'K':
  211. // clear line
  212. if (cp == last) {
  213. number = 0;
  214. } else {
  215. number = std::stoi(std::string(cp, last));
  216. };
  217. return CLEAR;
  218. case 'm':
  219. // color
  220. if (cp == last) {
  221. // nothing given, default to 0.
  222. number = 0;
  223. ansi_color(number);
  224. } else {
  225. while (cp != last) {
  226. number = std::stoi(std::string(cp, last));
  227. ansi_color(number);
  228. ++cp;
  229. while ((cp != last) && (isdigit(*cp))) {
  230. ++cp;
  231. };
  232. if (cp != last) {
  233. if (*cp == ';') {
  234. cp++;
  235. }
  236. }
  237. }
  238. }
  239. return COLOR;
  240. case 's':
  241. // save position
  242. saved_cursor_position.push_back(std::make_pair(this->posx, this->posy));
  243. return CURSOR;
  244. case 'u':
  245. // restore position
  246. if (saved_cursor_position.empty()) {
  247. ZF_LOGE("Restore cursor position from empty history.");
  248. } else {
  249. std::pair<int, int> pos = saved_cursor_position.back();
  250. this->posx = pos.first;
  251. this->posy = pos.second;
  252. saved_cursor_position.pop_back();
  253. }
  254. return CURSOR;
  255. case 't':
  256. case 'r':
  257. case 'h':
  258. case '!':
  259. // These are ones that I don't care about.
  260. case 'n': // This is terminal detect -- give me cursor position
  261. return OTHER;
  262. default:
  263. // unsure -- possibly not important
  264. ZF_LOGD("console_ansi( %s ): ???", repr(ansi.c_str()));
  265. return OTHER;
  266. }
  267. }
  268. }
  269. ZF_LOGD("console_ansi( %s ) : ???", repr(ansi.c_str()));
  270. return OTHER;
  271. }
  272. Terminal::termchar Terminal::putchar(char ch) {
  273. Terminal::termchar tc;
  274. if (this->in_ansi) {
  275. // Ok, append this char
  276. this->ansi.append(1, ch);
  277. if (isalpha(ch)) {
  278. // Ok! end of ANSI code, process it.
  279. tc.ansi = ansi_code(this->ansi);
  280. this->in_ansi = 0;
  281. this->ansi.clear();
  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. this->ansi.append(1, ch);
  293. this->in_ansi = 1;
  294. tc.in_ansi = 1;
  295. return tc;
  296. }
  297. // should I try reporting MOTION non-ANSI ?
  298. if (ch == '\r') {
  299. // Carriage return
  300. this->posx = 0;
  301. return tc;
  302. }
  303. if (ch == '\n') {
  304. this->posy++;
  305. // check range/"scroll"
  306. return tc;
  307. }
  308. if (ch == '\b') {
  309. // Backspace.
  310. if (this->posx > 0) {
  311. this->posx--;
  312. }
  313. return tc;
  314. }
  315. if (ch == '\f') {
  316. // form feed
  317. // treat as clear screen
  318. this->posx = 0;
  319. this->posy = 0;
  320. return tc;
  321. }
  322. /*
  323. I don't believe that anything else can possibly be here, other then an
  324. actual printable character. So!
  325. FUTURE: Store the screen text + colors
  326. */
  327. this->posx++;
  328. if (this->posx > 79) {
  329. this->posx = 0;
  330. this->posy++;
  331. // check range/"scroll"
  332. }
  333. return tc;
  334. }
  335. }
  336. void Terminal::putstring(std::string text) {
  337. // This gets noisy when called from render effect ^D
  338. // ZF_LOGI("console_char %lu chars", chars.size());
  339. for (auto strit = text.begin(); strit != text.end(); strit++)
  340. putchar(*strit);
  341. }
  342. // Old non-C++ ways
  343. void console_init(struct console_details *cdp) {
  344. cdp->posx = 0;
  345. cdp->posy = 0;
  346. cdp->savedx = 0;
  347. cdp->savedy = 0;
  348. cdp->in_ansi = 0;
  349. cdp->fgcolor = 7;
  350. cdp->bgcolor = 0;
  351. cdp->status = 0;
  352. }
  353. /*
  354. * Given the ansi number codes
  355. * Figure out fg, bg and status.
  356. */
  357. void ansi_color(struct console_details *cdp, int color) {
  358. // ZF_LOGV("ansi_color(%d)", color);
  359. if (color == 0) {
  360. cdp->status = 0;
  361. cdp->fgcolor = 7;
  362. cdp->bgcolor = 0;
  363. return;
  364. }
  365. if ((color == 1) || (color == 2) || (color == 3) || (color == 4) ||
  366. (color == 5)) {
  367. cdp->status = color;
  368. return;
  369. }
  370. if ((color >= 30) && (color <= 37)) {
  371. cdp->fgcolor = color - 30;
  372. return;
  373. }
  374. if ((color >= 40) && (color <= 47)) {
  375. cdp->bgcolor = color - 40;
  376. return;
  377. }
  378. if (color == 39) {
  379. // default fg color
  380. cdp->fgcolor = 7;
  381. return;
  382. }
  383. if (color == 49) {
  384. // default bg color
  385. cdp->bgcolor = 0;
  386. return;
  387. }
  388. ZF_LOGD("ansi_color( %d ) is unknown to me.", color);
  389. }
  390. const char *color_restore(struct console_details *cdp) {
  391. static char buffer[30];
  392. int slen;
  393. // possible optimize: If fg is 7, don't set (0 reset does that), if bg is 0,
  394. // don't set (0 does that)
  395. if (cdp->status == 0) {
  396. slen = snprintf(buffer, sizeof(buffer), "\x1b[0;3%d;4%dm", cdp->fgcolor,
  397. cdp->bgcolor);
  398. if (slen >= (int)sizeof(buffer)) {
  399. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  400. buffer[0] = 0;
  401. }
  402. } else {
  403. slen = snprintf(buffer, sizeof(buffer), "\x1b[0;%d;3%d;4%dm", cdp->status,
  404. cdp->fgcolor, cdp->bgcolor);
  405. if (slen >= (int)sizeof(buffer)) {
  406. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  407. buffer[0] = 0;
  408. }
  409. };
  410. return buffer;
  411. }
  412. /* store cursor position X,Y */
  413. std::vector<std::pair<int, int>> cursor_position_history;
  414. ANSI_TYPE console_ansi(struct console_details *cdp, const char *ansi) {
  415. const char *cp = ansi;
  416. const char *last = ansi + strlen(ansi) - 1;
  417. int number, number2;
  418. if (*cp == '\x1b') {
  419. cp++;
  420. // Ok, that's expected.
  421. if (*cp == '[') {
  422. cp++;
  423. // https://en.wikipedia.org/wiki/ANSI_escape_code#CSI_sequences
  424. switch (*last) {
  425. case 'A':
  426. // cursor up
  427. if (cp == last) {
  428. number = 1;
  429. } else {
  430. number = atoi(cp);
  431. if (number < 1) {
  432. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  433. number);
  434. number = 1;
  435. }
  436. };
  437. cdp->posy -= number;
  438. if (cdp->posy < 0) {
  439. cdp->posy = 0;
  440. ZF_LOGD(
  441. "console_ansi( %s ): attempt to move above top of screen (%d)",
  442. repr(ansi), number);
  443. }
  444. return CURSOR;
  445. case 'B':
  446. // cursor down
  447. if (cp == last) {
  448. number = 1;
  449. } else {
  450. number = atoi(cp);
  451. if (number < 1) {
  452. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  453. number);
  454. number = 1;
  455. }
  456. };
  457. cdp->posy += number;
  458. // check range/"scroll"
  459. return CURSOR;
  460. case 'C':
  461. // cursor forward
  462. if (cp == last) {
  463. number = 1;
  464. } else {
  465. number = atoi(cp);
  466. if (number < 1) {
  467. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  468. number);
  469. number = 1;
  470. }
  471. };
  472. cdp->posx += number;
  473. // Well. According to the "spec", the screen limits are hard
  474. // If the cursor is already at the edge of the screen, this has no
  475. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  476. while (cdp->posx > 79) {
  477. cdp->posy++;
  478. // check range/"scroll"
  479. cdp->posx -= 79;
  480. }
  481. return CURSOR;
  482. case 'D':
  483. // cursor backwards
  484. if (cp == last) {
  485. number = 1;
  486. } else {
  487. number = atoi(cp);
  488. if (number < 1) {
  489. ZF_LOGD("console_ansi( %s ): number error (%d)", repr(ansi),
  490. number);
  491. number = 0;
  492. }
  493. };
  494. cdp->posx -= number;
  495. // Well. According to the "spec", the screen limits are hard
  496. // If the cursor is already at the edge of the screen, this has no
  497. // effect. ^ This is *NOT* how any ANSI BBS terminal program acts!
  498. while (cdp->posx < 0) {
  499. cdp->posy--;
  500. if (cdp->posy < 0) {
  501. cdp->posy = 0;
  502. }
  503. cdp->posx += 79;
  504. }
  505. return CURSOR;
  506. case 'H':
  507. // cursor position
  508. if (*cp == ';') {
  509. // Missing first number
  510. number = 1;
  511. cp++;
  512. if (cp == last) {
  513. // missing 2nd number as well?
  514. number2 = 1;
  515. } else {
  516. number2 = atoi(cp);
  517. }
  518. } else {
  519. // Ok, find the first number
  520. number = atoi(cp);
  521. cp = strchr(cp, ';');
  522. if (cp == NULL) {
  523. // Missing 2nd number
  524. number2 = 1;
  525. } else {
  526. // 2nd number found, maybe.
  527. cp++;
  528. if (cp == last) {
  529. number2 = 1;
  530. } else {
  531. number2 = atoi(cp);
  532. }
  533. }
  534. }
  535. // Our positions start at zero, not one.
  536. cdp->posy = number - 1;
  537. cdp->posx = number2 - 1;
  538. return CURSOR;
  539. case 'J':
  540. // clear
  541. if (cp == last) {
  542. number = 0;
  543. } else {
  544. number = atoi(cp);
  545. };
  546. // clears ... part of the screen.
  547. if (number == 2) {
  548. cdp->posx = 0;
  549. cdp->posy = 0;
  550. };
  551. return CLEAR;
  552. case 'K':
  553. // clear line
  554. if (cp == last) {
  555. number = 0;
  556. } else {
  557. number = atoi(cp);
  558. };
  559. return CLEAR;
  560. case 'm':
  561. // color
  562. if (cp == last) {
  563. // nothing given, default to 0.
  564. number = 0;
  565. ansi_color(cdp, number);
  566. } else {
  567. while (cp != last) {
  568. number = atoi(cp);
  569. ansi_color(cdp, number);
  570. cp++;
  571. while ((cp != last) && (isdigit(*cp))) {
  572. cp++;
  573. };
  574. if (cp != last) {
  575. if (*cp == ';') {
  576. cp++;
  577. }
  578. }
  579. }
  580. }
  581. return COLOR;
  582. case 's':
  583. // save position
  584. cursor_position_history.push_back(std::make_pair(cdp->posx, cdp->posy));
  585. return CURSOR;
  586. case 'u':
  587. // restore position
  588. if (cursor_position_history.empty()) {
  589. ZF_LOGE("Restore cursor position from empty history.");
  590. } else {
  591. std::pair<int, int> pos = cursor_position_history.back();
  592. cdp->posx = pos.first;
  593. cdp->posy = pos.second;
  594. cursor_position_history.pop_back();
  595. }
  596. return CURSOR;
  597. case 't':
  598. case 'r':
  599. case 'h':
  600. case '!':
  601. // These are ones that I don't care about.
  602. case 'n': // This is terminal detect -- give me cursor position
  603. return OTHER;
  604. default:
  605. // unsure -- possibly not important
  606. ZF_LOGD("console_ansi( %s ): ???", repr(ansi));
  607. return OTHER;
  608. }
  609. }
  610. }
  611. ZF_LOGD("console_ansi( %s ) : ???", repr(ansi));
  612. return OTHER;
  613. }
  614. /*
  615. * console_char()
  616. * return whether or not we are still in_ansi
  617. *
  618. * in_ansi TRUE: START (receiving start of/fragment of ANSI)
  619. * CURSOR, COLOR, CLEAR, OTHER
  620. * Results of the last ANSI parse.
  621. * in_ansi FALSE: START Normal character.
  622. *
  623. */
  624. termchar console_char(struct console_details *cdp, char ch) {
  625. char *cp;
  626. termchar tc;
  627. if (cdp->in_ansi) {
  628. // Ok, append this char
  629. cp = cdp->ansi + strlen(cdp->ansi);
  630. *cp = ch;
  631. cp++;
  632. *cp = 0;
  633. if (isalpha(ch)) {
  634. // Ok! end of ANSI code, process it.
  635. tc.ansi = console_ansi(cdp, cdp->ansi);
  636. cdp->in_ansi = 0;
  637. cdp->ansi[0] = 0;
  638. tc.in_ansi = 1;
  639. return tc;
  640. }
  641. tc.ansi = START;
  642. tc.in_ansi = 1;
  643. return tc;
  644. } else {
  645. tc.ansi = START;
  646. tc.in_ansi = 0;
  647. if (ch == '\x1b') {
  648. cp = cdp->ansi;
  649. *cp = ch;
  650. cp++;
  651. *cp = 0;
  652. cdp->in_ansi = 1;
  653. tc.in_ansi = 1;
  654. return tc;
  655. }
  656. // should I try reporting MOTION non-ANSI ?
  657. if (ch == '\r') {
  658. // Carriage return
  659. cdp->posx = 0;
  660. return tc;
  661. }
  662. if (ch == '\n') {
  663. cdp->posy++;
  664. // check range/"scroll"
  665. return tc;
  666. }
  667. if (ch == '\b') {
  668. // Backspace.
  669. if (cdp->posx > 0) {
  670. cdp->posx--;
  671. }
  672. return tc;
  673. }
  674. if (ch == '\f') {
  675. // form feed
  676. // treat as clear screen
  677. cdp->posx = 0;
  678. cdp->posy = 0;
  679. return tc;
  680. }
  681. /*
  682. I don't believe that anything else can possibly be here, other then an
  683. actual printable character. So!
  684. FUTURE: Store the screen text + colors
  685. */
  686. cdp->posx++;
  687. if (cdp->posx > 79) {
  688. cdp->posx = 0;
  689. cdp->posy++;
  690. // check range/"scroll"
  691. }
  692. return tc;
  693. }
  694. }
  695. void console_receive(struct console_details *cdp, std::string chars) {
  696. // This gets noisy when called from render effect ^D
  697. // ZF_LOGI("console_char %lu chars", chars.size());
  698. for (auto strit = chars.begin(); strit != chars.end(); strit++)
  699. console_char(cdp, *strit);
  700. }