render.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752
  1. #include "render.h"
  2. #include "terminal.h"
  3. #include "zf_log.h"
  4. #include <ctype.h>
  5. #include <stdio.h>
  6. #include <string.h>
  7. #include <time.h>
  8. #include <unistd.h> // usleep
  9. #include <vector>
  10. extern const char *strnstr(const char *source, int len, const char *needle);
  11. extern const char *repr(const char *data);
  12. extern struct console_details console;
  13. struct render current_render;
  14. int render_overlimit = 0;
  15. void reset_render(void) {
  16. current_render.speed = 0;
  17. current_render.effect = 0;
  18. render_overlimit = 0;
  19. }
  20. // or possibly a vector, and pop the image pointer off
  21. // allowing for multiple images.
  22. const char **image_data = NULL;
  23. int image_size = 0;
  24. void render_image(const char **lines, int size) {
  25. image_data = lines;
  26. image_size = size;
  27. }
  28. void reset_image(void) {
  29. image_data = NULL;
  30. image_size = 0;
  31. }
  32. int ms_sleep(unsigned int ms) {
  33. int result = 0;
  34. struct timespec ts = {ms / 1000, (ms % 1000) * 1000000L};
  35. do {
  36. struct timespec ts_sleep = ts;
  37. result = nanosleep(&ts_sleep, &ts);
  38. } while ((-1 == result));
  39. return result;
  40. }
  41. void render_sleep(void) {
  42. if (render_overlimit)
  43. return;
  44. if (current_render.speed) { // * 100 still too slow.
  45. ms_sleep(current_render.speed * 10);
  46. }
  47. }
  48. /*
  49. Well SNAP! Mystic numbers don't remotely match ANSI color codes.
  50. 00 : Sets the current foreground to Black 0;30
  51. 01 : Sets the current foreground to Dark Blue 0;34
  52. 02 : Sets the current foreground to Dark Green 0;32
  53. 03 : Sets the current foreground to Dark Cyan 0;36
  54. 04 : Sets the current foreground to Dark Red 0;31
  55. 05 : Sets the current foreground to Dark Magenta 0;35
  56. 06 : Sets the current foreground to Brown 0;33
  57. 07 : Sets the current foreground to Grey 0;37
  58. 08 : Sets the current foreground to Dark Grey 1;30
  59. 09 : Sets the current foreground to Light Blue 1;34
  60. 10 : Sets the current foreground to Light Green 1;32
  61. 11 : Sets the current foreground to Light Cyan 1;36
  62. 12 : Sets the current foreground to Light Red 1;31
  63. 13 : Sets the current foreground to Light Magenta 1;35
  64. 14 : Sets the current foreground to Yellow 1;33
  65. 15 : Sets the current foreground to White 1;37
  66. 16 : Sets the current background to Black 40
  67. 17 : Sets the current background to Blue 44
  68. 18 : Sets the current background to Green 42
  69. 19 : Sets the current background to Cyan 46
  70. 20 : Sets the current background to Red 41
  71. 21 : Sets the current background to Magenta 45
  72. 22 : Sets the current background to Brown 43
  73. 23 : Sets the current background to Grey 47
  74. 24 : Sets the current background to black with blinking foreground 5;40
  75. 25 : Sets the current background to blue with blinking foreground 5;44
  76. 26 : Sets the current background to green with blinking foreground 5;42
  77. 27 : Sets the current background to cyan with blinking foreground 5;46
  78. 28 : Sets the current background to red with blinking foreground 5;41
  79. 29 : Sets the current background to magenta with blinking foreground 5;45
  80. 30 : Sets the current background to brown with blinking foreground 5;43
  81. 31 : Sets the current background to grey with blinking foreground 5;47
  82. Other things that Mystic does ...
  83. [A## - Move the cursor up ## lines
  84. [B## - Move the cursor down ## lines
  85. [C## - Move the cursor forward (to the right) ## columns
  86. [D## - Move the cursor backwards (to the left) ## columns
  87. [K - Clear from the current cursor position to the end of the line
  88. [L - Move cursor and erase data backwards from current column to column ##
  89. [X## - Move cursor to X coordinate ##
  90. [Y## - Move cursor to Y coordinate ##
  91. BS - Sends 1 destructive backspace sequence (ASCII 8-32-8)
  92. CL - Clears the screen (ANSI 1,1 locate and [2J or ASCII 12)
  93. CR - Send a carrage return and line feed (move to next line)
  94. RA - Restore the saved text attribute color
  95. RS - Restore the saved user's terminal screen
  96. SA - Save the current text attribute color
  97. SS - Save the entire user's terminal screen
  98. */
  99. // Covert MYSTIC color to (Proper) ANSI COLOR.
  100. const int MYSTIC[] = {0, 4, 2, 6, 1, 5, 3, 7};
  101. // ANSI_color = MYSTIC[ odd_mystic_color % 8 ]
  102. void write_color(int fd, int color) {
  103. char buffer[12];
  104. int slen;
  105. switch (color) {
  106. case 0:
  107. case 1:
  108. case 2:
  109. case 3:
  110. case 4:
  111. case 5:
  112. case 6:
  113. case 7:
  114. slen = snprintf(buffer, sizeof(buffer), "\x1b[0;3%dm", MYSTIC[color]);
  115. if (slen >= (int)sizeof(buffer)) {
  116. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  117. buffer[0] = 0;
  118. }
  119. break;
  120. case 8:
  121. case 9:
  122. case 10:
  123. case 11:
  124. case 12:
  125. case 13:
  126. case 14:
  127. case 15:
  128. slen = snprintf(buffer, sizeof(buffer), "\x1b[0;1;3%dm", MYSTIC[color - 8]);
  129. if (slen >= (int)sizeof(buffer)) {
  130. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  131. buffer[0] = 0;
  132. }
  133. break;
  134. case 16:
  135. case 17:
  136. case 18:
  137. case 19:
  138. case 20:
  139. case 21:
  140. case 22:
  141. case 23:
  142. slen = snprintf(buffer, sizeof(buffer), "\x1b[4%dm", MYSTIC[color - 16]);
  143. if (slen >= (int)sizeof(buffer)) {
  144. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  145. buffer[0] = 0;
  146. }
  147. break;
  148. case 24:
  149. case 25:
  150. case 26:
  151. case 27:
  152. case 28:
  153. case 29:
  154. case 30:
  155. case 31:
  156. slen = snprintf(buffer, sizeof(buffer), "\x1b[5;4%dm", MYSTIC[color - 24]);
  157. if (slen >= (int)sizeof(buffer)) {
  158. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  159. buffer[0] = 0;
  160. }
  161. break;
  162. default:
  163. buffer[0] = 0;
  164. break;
  165. }
  166. ZF_LOGD("write_color( %d ): %s", color, repr(buffer));
  167. write(fd, buffer, strlen(buffer));
  168. }
  169. void send_goto(int fd, int x, int y) {
  170. char gbuffer[16];
  171. int slen;
  172. slen = snprintf(gbuffer, sizeof(gbuffer), "\x1b[%d;%dH", y, x);
  173. if (slen >= (int)sizeof(gbuffer)) {
  174. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(gbuffer));
  175. gbuffer[0] = 0;
  176. }
  177. write(fd, gbuffer, strlen(gbuffer));
  178. }
  179. int send_image(int fd) {
  180. int i;
  181. const char **lines = image_data;
  182. if (lines == NULL) {
  183. ZF_LOGE("No image data for send_image.");
  184. return 0;
  185. }
  186. for (i = 0; i < image_size; i++) {
  187. std::string line(lines[i]);
  188. line.append("\r\n");
  189. render(fd, line);
  190. /*
  191. write(fd, lines[i], strlen(lines[i]));
  192. write(fd, "\r\n", 2);
  193. */
  194. }
  195. // success
  196. reset_image();
  197. return 1;
  198. }
  199. int send_image(int fd, int x, int y) {
  200. int i;
  201. const char **lines = image_data;
  202. if (lines == NULL) {
  203. ZF_LOGE("No image data for send_image(%d,%d)", x, y);
  204. return 0;
  205. }
  206. for (i = 0; i < image_size; i++) {
  207. send_goto(fd, x, y);
  208. y++;
  209. write(fd, lines[i], strlen(lines[i]));
  210. }
  211. // success
  212. reset_image();
  213. return 1;
  214. }
  215. #ifdef UNWANTED
  216. int send_file(int fd, char *filename) {
  217. FILE *fp;
  218. char buffer[100];
  219. int read;
  220. fp = fopen(filename, "rb");
  221. if (fp == NULL) {
  222. ZF_LOGD("Failed to open %s", filename);
  223. return 0;
  224. }
  225. while ((read = fread(buffer, 1, sizeof(buffer), fp)) > 0) {
  226. write(fd, buffer, read);
  227. };
  228. fclose(fp);
  229. return 1;
  230. }
  231. int send_file(int fd, int x, int y, char *filename) {
  232. FILE *fp;
  233. char buffer[100];
  234. int read;
  235. fp = fopen(filename, "rb");
  236. if (fp == NULL) {
  237. ZF_LOGD("Failed to open %s", filename);
  238. return 0;
  239. }
  240. send_goto(fd, x, y);
  241. y++;
  242. while ((read = fread(buffer, 1, sizeof(buffer), fp)) > 0) {
  243. char *cp, *last_cp;
  244. buffer[read] = 0;
  245. last_cp = buffer;
  246. while ((cp = strchr(last_cp, '\n')) != NULL) {
  247. *cp = 0;
  248. write(fd, last_cp, strlen(last_cp));
  249. send_goto(fd, x, y);
  250. y++;
  251. last_cp = cp + 1;
  252. };
  253. write(fd, last_cp, strlen(last_cp));
  254. };
  255. fclose(fp);
  256. return 1;
  257. }
  258. #endif
  259. int digit(std::string str, size_t &pos, int digits = 1) {
  260. int i = 0;
  261. while (digits > 0) {
  262. if (std::isdigit(str[pos])) {
  263. i *= 10;
  264. i += str[pos] - '0';
  265. pos++;
  266. } else {
  267. ZF_LOGE("digit fail");
  268. }
  269. digits--;
  270. };
  271. return i;
  272. }
  273. static std::vector<console_details> console_history;
  274. /**
  275. * process_trigger( fd, str, &pos )
  276. *
  277. * This process a command trigger.
  278. * It has seen TRIGGER, and now it is
  279. * processing whatever comes after it.
  280. * It will perform the process, and
  281. * update pos to whatever is next.
  282. */
  283. void process_trigger(int fd, std::string str, size_t &pos) {
  284. char ch = str[pos];
  285. int i, x, y;
  286. pos++;
  287. switch (ch) {
  288. case 'D':
  289. i = digit(str, pos, 2);
  290. if ((i > 0) && (i < 80)) {
  291. ZF_LOGI("DEL %02d", i);
  292. for (x = 0; x < i; x++) {
  293. write(fd, "\b \b", 3);
  294. console_receive(&console, "\b \b");
  295. }
  296. };
  297. break;
  298. case 'C': {
  299. i = 0;
  300. if (str[pos] == 'S') {
  301. pos++;
  302. console_history.push_back(console);
  303. ZF_LOGI("saved color.");
  304. break;
  305. }
  306. if (str[pos] == 'R') {
  307. pos++;
  308. if (console_history.empty()) {
  309. ZF_LOGE("Trying to Color Restore from empty vector!");
  310. break;
  311. }
  312. console_details old_color = console_history.back();
  313. console_history.pop_back();
  314. const char *restore = color_restore(&old_color);
  315. write(fd, restore, strlen(restore));
  316. break;
  317. }
  318. i = digit(str, pos, 2);
  319. write_color(fd, i);
  320. } break;
  321. // no longer takes a filename. :)
  322. case 'F':
  323. case 'f': {
  324. int UsePos = 0;
  325. if (ch == 'f') {
  326. UsePos = 1;
  327. x = digit(str, pos, 2);
  328. y = digit(str, pos, 2);
  329. ZF_LOGI("file at (%d, %d)", x, y);
  330. }
  331. ZF_LOGD("IMAGE (%d lines)", image_size);
  332. if (UsePos) {
  333. send_image(fd, x, y);
  334. } else {
  335. send_image(fd);
  336. };
  337. } break;
  338. case 'G': {
  339. x = digit(str, pos, 2);
  340. y = digit(str, pos, 2);
  341. char buffer[20]; // row ; column H
  342. int slen;
  343. ZF_LOGD("GOTO (%d,%d)", x, y);
  344. slen = snprintf(buffer, sizeof(buffer), "\x1b[%d;%dH", y, x);
  345. if (slen >= (int)sizeof(buffer)) {
  346. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  347. buffer[0] = 0;
  348. }
  349. write(fd, buffer, strlen(buffer));
  350. } break;
  351. case 'R': {
  352. i = digit(str, pos);
  353. if ((i > 0) && (i < 10)) {
  354. ZF_LOGI("RENDER %d", i);
  355. current_render.effect = i;
  356. } else {
  357. current_render.effect = 0;
  358. }
  359. } break;
  360. case 'S': {
  361. i = digit(str, pos);
  362. if ((i > 0) && (i < 10)) {
  363. ZF_LOGI("SPEED %d", i);
  364. current_render.speed = i;
  365. } else {
  366. current_render.speed = 0;
  367. }
  368. } break;
  369. case 'P': {
  370. i = digit(str, pos);
  371. if ((i > 0) && (i < 10)) {
  372. ZF_LOGI("PAWS %d", i);
  373. // sleep(i);
  374. if (!render_overlimit) {
  375. sleep(i);
  376. };
  377. }
  378. } break;
  379. }
  380. }
  381. /**
  382. * process_trigger( fd, *cp )
  383. *
  384. * This process a command trigger.
  385. * It has seen TRIGGER, and now it is
  386. * processing whatever comes after it.
  387. * It will perform the process, and
  388. * return the char * of whatever is next.
  389. */
  390. const char *process_trigger(int fd, const char *cp) {
  391. char ch;
  392. int i, x, y;
  393. ch = *cp;
  394. cp++;
  395. switch (ch) {
  396. case 'D':
  397. i = 0;
  398. if (isdigit(*cp)) {
  399. i = (*cp) - '0';
  400. cp++;
  401. };
  402. if (isdigit(*cp)) {
  403. i *= 10;
  404. i += (*cp) - '0';
  405. cp++;
  406. };
  407. if ((i > 0) && (i < 80)) {
  408. ZF_LOGI("DEL %02d", i);
  409. for (x = 0; x < i; x++) {
  410. write(fd, "\b \b", 3);
  411. console_receive(&console, "\b \b");
  412. }
  413. };
  414. break;
  415. case 'C': {
  416. i = 0;
  417. if (*cp == 'R') {
  418. cp++;
  419. const char *restore = color_restore(&console);
  420. write(fd, restore, strlen(restore));
  421. break;
  422. }
  423. if (isdigit(*cp)) {
  424. i = (*cp) - '0';
  425. cp++;
  426. };
  427. if (isdigit(*cp)) {
  428. i *= 10;
  429. i += (*cp) - '0';
  430. cp++;
  431. };
  432. write_color(fd, i);
  433. } break;
  434. // no longer takes a filename. :)
  435. case 'F':
  436. case 'f': {
  437. int pos = 0;
  438. if (ch == 'f') {
  439. pos = 1;
  440. x = (*cp) - '0';
  441. cp++;
  442. x *= 10;
  443. x += (*cp) - '0';
  444. cp++;
  445. y = (*cp) - '0';
  446. cp++;
  447. y *= 10;
  448. y += (*cp) - '0';
  449. cp++;
  450. ZF_LOGI("file at (%d, %d)", x, y);
  451. }
  452. ZF_LOGD("IMAGE (%d lines)", image_size);
  453. if (pos) {
  454. send_image(fd, x, y);
  455. } else {
  456. send_image(fd);
  457. };
  458. } break;
  459. case 'G': {
  460. x = 0;
  461. if (isdigit(*cp)) {
  462. x = (*cp) - '0';
  463. cp++;
  464. };
  465. if (isdigit(*cp)) {
  466. x *= 10;
  467. x += (*cp) - '0';
  468. cp++;
  469. };
  470. y = 0;
  471. if (isdigit(*cp)) {
  472. y = (*cp) - '0';
  473. cp++;
  474. };
  475. if (isdigit(*cp)) {
  476. y *= 10;
  477. y += (*cp) - '0';
  478. cp++;
  479. };
  480. char buffer[20]; // row ; column H
  481. int slen;
  482. ZF_LOGD("GOTO (%d,%d)", x, y);
  483. slen = snprintf(buffer, sizeof(buffer), "\x1b[%d;%dH", y, x);
  484. if (slen >= (int)sizeof(buffer)) {
  485. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  486. buffer[0] = 0;
  487. }
  488. write(fd, buffer, strlen(buffer));
  489. } break;
  490. case 'R': {
  491. i = 0;
  492. if (isdigit(*cp)) {
  493. i = (*cp) - '0';
  494. cp++;
  495. };
  496. if ((i > 0) && (i < 10)) {
  497. ZF_LOGI("RENDER %d", i);
  498. current_render.effect = i;
  499. } else {
  500. ZF_LOGI("RENDER OFF");
  501. current_render.effect = 0;
  502. }
  503. } break;
  504. case 'S': {
  505. i = 0;
  506. if (isdigit(*cp)) {
  507. i = (*cp) - '0';
  508. cp++;
  509. };
  510. if ((i > 0) && (i < 10)) {
  511. ZF_LOGI("SPEED %d", i);
  512. current_render.speed = i;
  513. } else {
  514. ZF_LOGI("SPEED OFF");
  515. current_render.speed = 0;
  516. }
  517. } break;
  518. case 'P': {
  519. i = 0;
  520. if (isdigit(*cp)) {
  521. i = (*cp) - '0';
  522. cp++;
  523. };
  524. if ((i > 0) && (i < 10)) {
  525. ZF_LOGI("PAWS %d", i);
  526. // sleep(i);
  527. if (!render_overlimit) {
  528. sleep(i);
  529. };
  530. }
  531. } break;
  532. }
  533. return cp;
  534. }
  535. /**
  536. * render_effect( fd, ch )
  537. *
  538. * Displays the given character with whatever
  539. * rendering effect is currently active.
  540. * (If any).
  541. */
  542. void render_effect(int fd, char ch) {
  543. int effect = current_render.effect;
  544. int l;
  545. char space = ' ';
  546. char bs = '\b';
  547. termchar tc;
  548. tc = console_char(&console, ch);
  549. if (tc.in_ansi) {
  550. // We are inside ANSI command, so
  551. write(fd, &ch, 1);
  552. return;
  553. }
  554. switch (effect) {
  555. case 1:
  556. // CHAR + SPC + BS
  557. render_sleep();
  558. write(fd, &ch, 1);
  559. render_sleep();
  560. write(fd, &space, 1);
  561. render_sleep();
  562. render_sleep();
  563. write(fd, &bs, 1);
  564. break;
  565. case 2:
  566. // CHAR + 8 spaces + 8 BS
  567. // This might be too much.
  568. #define MOVE 4
  569. render_sleep();
  570. write(fd, &ch, 1);
  571. for (l = 0; l < MOVE; l++) {
  572. render_sleep();
  573. write(fd, &space, 1);
  574. }
  575. for (l = 0; l < MOVE; l++) {
  576. render_sleep();
  577. write(fd, &bs, 1);
  578. }
  579. break;
  580. case 0:
  581. default:
  582. // NORMAL
  583. render_sleep();
  584. write(fd, &ch, 1);
  585. break;
  586. }
  587. }
  588. #ifdef UNWANTED
  589. /**
  590. * render( fd, string_out )
  591. *
  592. * Render an entire string.
  593. * Handles TRIGGER.
  594. * Renders with effects.
  595. */
  596. void render(int fd, const char *string_out, int len) {
  597. const char *cp = string_out;
  598. const char *trigger = cp;
  599. time_t start = time(NULL);
  600. int elapsed;
  601. int over = 0;
  602. reset_render();
  603. // ZF_LOGV("render(%d, %s)", fd, repr(string_out));
  604. ZF_LOGV_MEM(string_out, len, "render(%d, %d bytes):", fd, len);
  605. // Check our time from time to time.
  606. // If we start running long, disable sleeps.
  607. while ((trigger = strnstr(cp, len - (cp - string_out), TRIGGER)) != NULL) {
  608. // There is special things to handle in here.
  609. while (cp != trigger) {
  610. elapsed = time(NULL) - start;
  611. if (elapsed > SLEEP_LIMIT) {
  612. render_overlimit = 1;
  613. current_render.speed = 0;
  614. };
  615. // write(fd, cp, 1 );
  616. render_effect(fd, *cp);
  617. cp++;
  618. };
  619. // ZF_LOGI( "at trigger: (%s)", cp);
  620. cp += strlen(TRIGGER);
  621. // Ok, we're pointing at the trigger -- do something.
  622. cp = process_trigger(fd, cp);
  623. // ZF_LOGI( "after trigger: (%s)", cp);
  624. };
  625. // We still might be under a rendering effect.
  626. while (cp < (string_out + len)) {
  627. elapsed = time(NULL) - start;
  628. if (elapsed > SLEEP_LIMIT) {
  629. render_overlimit = 1;
  630. current_render.speed = 0;
  631. };
  632. // write(fd, cp, 1);
  633. render_effect(fd, *cp);
  634. cp++;
  635. }
  636. }
  637. #endif
  638. /**
  639. * render( fd, string_out )
  640. *
  641. * Render an entire string.
  642. * Handles TRIGGER.
  643. * Renders with effects.
  644. */
  645. void render(int fd, std::string &string_out) {
  646. // reset_render();
  647. time_t start = time(NULL);
  648. size_t pos = 0;
  649. int elapsed;
  650. size_t tpos;
  651. while ((tpos = string_out.find(TRIGGER, pos)) != std::string::npos) {
  652. while (pos != tpos) {
  653. elapsed = time(NULL) - start;
  654. if (elapsed > SLEEP_LIMIT) {
  655. render_overlimit = 1;
  656. current_render.speed = 0;
  657. }
  658. render_effect(fd, string_out[pos]);
  659. pos++;
  660. }
  661. pos += strlen(TRIGGER);
  662. process_trigger(fd, string_out, pos);
  663. }
  664. while (pos < string_out.size()) {
  665. elapsed = time(NULL) - start;
  666. if (elapsed > SLEEP_LIMIT) {
  667. render_overlimit = 1;
  668. current_render.speed = 0;
  669. }
  670. render_effect(fd, string_out[pos]);
  671. pos++;
  672. }
  673. }