render.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747
  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. write(fd, lines[i], strlen(lines[i]));
  188. write(fd, "\r\n", 2);
  189. }
  190. // success
  191. reset_image();
  192. return 1;
  193. }
  194. int send_image(int fd, int x, int y) {
  195. int i;
  196. const char **lines = image_data;
  197. if (lines == NULL) {
  198. ZF_LOGE("No image data for send_image(%d,%d)", x, y);
  199. return 0;
  200. }
  201. for (i = 0; i < image_size; i++) {
  202. send_goto(fd, x, y);
  203. y++;
  204. write(fd, lines[i], strlen(lines[i]));
  205. }
  206. // success
  207. reset_image();
  208. return 1;
  209. }
  210. #ifdef UNWANTED
  211. int send_file(int fd, char *filename) {
  212. FILE *fp;
  213. char buffer[100];
  214. int read;
  215. fp = fopen(filename, "rb");
  216. if (fp == NULL) {
  217. ZF_LOGD("Failed to open %s", filename);
  218. return 0;
  219. }
  220. while ((read = fread(buffer, 1, sizeof(buffer), fp)) > 0) {
  221. write(fd, buffer, read);
  222. };
  223. fclose(fp);
  224. return 1;
  225. }
  226. int send_file(int fd, int x, int y, char *filename) {
  227. FILE *fp;
  228. char buffer[100];
  229. int read;
  230. fp = fopen(filename, "rb");
  231. if (fp == NULL) {
  232. ZF_LOGD("Failed to open %s", filename);
  233. return 0;
  234. }
  235. send_goto(fd, x, y);
  236. y++;
  237. while ((read = fread(buffer, 1, sizeof(buffer), fp)) > 0) {
  238. char *cp, *last_cp;
  239. buffer[read] = 0;
  240. last_cp = buffer;
  241. while ((cp = strchr(last_cp, '\n')) != NULL) {
  242. *cp = 0;
  243. write(fd, last_cp, strlen(last_cp));
  244. send_goto(fd, x, y);
  245. y++;
  246. last_cp = cp + 1;
  247. };
  248. write(fd, last_cp, strlen(last_cp));
  249. };
  250. fclose(fp);
  251. return 1;
  252. }
  253. #endif
  254. int digit(std::string str, size_t &pos, int digits = 1) {
  255. int i = 0;
  256. while (digits > 0) {
  257. if (std::isdigit(str[pos])) {
  258. i *= 10;
  259. i += str[pos] - '0';
  260. pos++;
  261. } else {
  262. ZF_LOGE("digit fail");
  263. }
  264. digits--;
  265. };
  266. return i;
  267. }
  268. static std::vector<console_details> console_history;
  269. /**
  270. * process_trigger( fd, str, &pos )
  271. *
  272. * This process a command trigger.
  273. * It has seen TRIGGER, and now it is
  274. * processing whatever comes after it.
  275. * It will perform the process, and
  276. * update pos to whatever is next.
  277. */
  278. void process_trigger(int fd, std::string str, size_t &pos) {
  279. char ch = str[pos];
  280. int i, x, y;
  281. pos++;
  282. switch (ch) {
  283. case 'D':
  284. i = digit(str, pos, 2);
  285. if ((i > 0) && (i < 80)) {
  286. ZF_LOGI("DEL %02d", i);
  287. for (x = 0; x < i; x++) {
  288. write(fd, "\b \b", 3);
  289. console_receive(&console, "\b \b");
  290. }
  291. };
  292. break;
  293. case 'C': {
  294. i = 0;
  295. if (str[pos] == 'S') {
  296. pos++;
  297. console_history.push_back(console);
  298. ZF_LOGI("saved color.");
  299. break;
  300. }
  301. if (str[pos] == 'R') {
  302. pos++;
  303. if (console_history.empty()) {
  304. ZF_LOGE("Trying to Color Restore from empty vector!");
  305. break;
  306. }
  307. console_details old_color = console_history.back();
  308. console_history.pop_back();
  309. const char *restore = color_restore(&old_color);
  310. write(fd, restore, strlen(restore));
  311. break;
  312. }
  313. i = digit(str, pos, 2);
  314. write_color(fd, i);
  315. } break;
  316. // no longer takes a filename. :)
  317. case 'F':
  318. case 'f': {
  319. int UsePos = 0;
  320. if (ch == 'f') {
  321. UsePos = 1;
  322. x = digit(str, pos, 2);
  323. y = digit(str, pos, 2);
  324. ZF_LOGI("file at (%d, %d)", x, y);
  325. }
  326. ZF_LOGD("IMAGE (%d lines)", image_size);
  327. if (UsePos) {
  328. send_image(fd, x, y);
  329. } else {
  330. send_image(fd);
  331. };
  332. } break;
  333. case 'G': {
  334. x = digit(str, pos, 2);
  335. y = digit(str, pos, 2);
  336. char buffer[20]; // row ; column H
  337. int slen;
  338. ZF_LOGD("GOTO (%d,%d)", x, y);
  339. slen = snprintf(buffer, sizeof(buffer), "\x1b[%d;%dH", y, x);
  340. if (slen >= (int)sizeof(buffer)) {
  341. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  342. buffer[0] = 0;
  343. }
  344. write(fd, buffer, strlen(buffer));
  345. } break;
  346. case 'R': {
  347. i = digit(str, pos);
  348. if ((i > 0) && (i < 10)) {
  349. ZF_LOGI("RENDER %d", i);
  350. current_render.effect = i;
  351. } else {
  352. current_render.effect = 0;
  353. }
  354. } break;
  355. case 'S': {
  356. i = digit(str, pos);
  357. if ((i > 0) && (i < 10)) {
  358. ZF_LOGI("SPEED %d", i);
  359. current_render.speed = i;
  360. } else {
  361. current_render.speed = 0;
  362. }
  363. } break;
  364. case 'P': {
  365. i = digit(str, pos);
  366. if ((i > 0) && (i < 10)) {
  367. ZF_LOGI("PAWS %d", i);
  368. // sleep(i);
  369. if (!render_overlimit) {
  370. sleep(i);
  371. };
  372. }
  373. } break;
  374. }
  375. }
  376. /**
  377. * process_trigger( fd, *cp )
  378. *
  379. * This process a command trigger.
  380. * It has seen TRIGGER, and now it is
  381. * processing whatever comes after it.
  382. * It will perform the process, and
  383. * return the char * of whatever is next.
  384. */
  385. const char *process_trigger(int fd, const char *cp) {
  386. char ch;
  387. int i, x, y;
  388. ch = *cp;
  389. cp++;
  390. switch (ch) {
  391. case 'D':
  392. i = 0;
  393. if (isdigit(*cp)) {
  394. i = (*cp) - '0';
  395. cp++;
  396. };
  397. if (isdigit(*cp)) {
  398. i *= 10;
  399. i += (*cp) - '0';
  400. cp++;
  401. };
  402. if ((i > 0) && (i < 80)) {
  403. ZF_LOGI("DEL %02d", i);
  404. for (x = 0; x < i; x++) {
  405. write(fd, "\b \b", 3);
  406. console_receive(&console, "\b \b");
  407. }
  408. };
  409. break;
  410. case 'C': {
  411. i = 0;
  412. if (*cp == 'R') {
  413. cp++;
  414. const char *restore = color_restore(&console);
  415. write(fd, restore, strlen(restore));
  416. break;
  417. }
  418. if (isdigit(*cp)) {
  419. i = (*cp) - '0';
  420. cp++;
  421. };
  422. if (isdigit(*cp)) {
  423. i *= 10;
  424. i += (*cp) - '0';
  425. cp++;
  426. };
  427. write_color(fd, i);
  428. } break;
  429. // no longer takes a filename. :)
  430. case 'F':
  431. case 'f': {
  432. int pos = 0;
  433. if (ch == 'f') {
  434. pos = 1;
  435. x = (*cp) - '0';
  436. cp++;
  437. x *= 10;
  438. x += (*cp) - '0';
  439. cp++;
  440. y = (*cp) - '0';
  441. cp++;
  442. y *= 10;
  443. y += (*cp) - '0';
  444. cp++;
  445. ZF_LOGI("file at (%d, %d)", x, y);
  446. }
  447. ZF_LOGD("IMAGE (%d lines)", image_size);
  448. if (pos) {
  449. send_image(fd, x, y);
  450. } else {
  451. send_image(fd);
  452. };
  453. } break;
  454. case 'G': {
  455. x = 0;
  456. if (isdigit(*cp)) {
  457. x = (*cp) - '0';
  458. cp++;
  459. };
  460. if (isdigit(*cp)) {
  461. x *= 10;
  462. x += (*cp) - '0';
  463. cp++;
  464. };
  465. y = 0;
  466. if (isdigit(*cp)) {
  467. y = (*cp) - '0';
  468. cp++;
  469. };
  470. if (isdigit(*cp)) {
  471. y *= 10;
  472. y += (*cp) - '0';
  473. cp++;
  474. };
  475. char buffer[20]; // row ; column H
  476. int slen;
  477. ZF_LOGD("GOTO (%d,%d)", x, y);
  478. slen = snprintf(buffer, sizeof(buffer), "\x1b[%d;%dH", y, x);
  479. if (slen >= (int)sizeof(buffer)) {
  480. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  481. buffer[0] = 0;
  482. }
  483. write(fd, buffer, strlen(buffer));
  484. } break;
  485. case 'R': {
  486. i = 0;
  487. if (isdigit(*cp)) {
  488. i = (*cp) - '0';
  489. cp++;
  490. };
  491. if ((i > 0) && (i < 10)) {
  492. ZF_LOGI("RENDER %d", i);
  493. current_render.effect = i;
  494. } else {
  495. ZF_LOGI("RENDER OFF");
  496. current_render.effect = 0;
  497. }
  498. } break;
  499. case 'S': {
  500. i = 0;
  501. if (isdigit(*cp)) {
  502. i = (*cp) - '0';
  503. cp++;
  504. };
  505. if ((i > 0) && (i < 10)) {
  506. ZF_LOGI("SPEED %d", i);
  507. current_render.speed = i;
  508. } else {
  509. ZF_LOGI("SPEED OFF");
  510. current_render.speed = 0;
  511. }
  512. } break;
  513. case 'P': {
  514. i = 0;
  515. if (isdigit(*cp)) {
  516. i = (*cp) - '0';
  517. cp++;
  518. };
  519. if ((i > 0) && (i < 10)) {
  520. ZF_LOGI("PAWS %d", i);
  521. // sleep(i);
  522. if (!render_overlimit) {
  523. sleep(i);
  524. };
  525. }
  526. } break;
  527. }
  528. return cp;
  529. }
  530. /**
  531. * render_effect( fd, ch )
  532. *
  533. * Displays the given character with whatever
  534. * rendering effect is currently active.
  535. * (If any).
  536. */
  537. void render_effect(int fd, char ch) {
  538. int effect = current_render.effect;
  539. int l;
  540. char space = ' ';
  541. char bs = '\b';
  542. termchar tc;
  543. tc = console_char(&console, ch);
  544. if (tc.in_ansi) {
  545. // We are inside ANSI command, so
  546. write(fd, &ch, 1);
  547. return;
  548. }
  549. switch (effect) {
  550. case 1:
  551. // CHAR + SPC + BS
  552. render_sleep();
  553. write(fd, &ch, 1);
  554. render_sleep();
  555. write(fd, &space, 1);
  556. render_sleep();
  557. render_sleep();
  558. write(fd, &bs, 1);
  559. break;
  560. case 2:
  561. // CHAR + 8 spaces + 8 BS
  562. // This might be too much.
  563. #define MOVE 4
  564. render_sleep();
  565. write(fd, &ch, 1);
  566. for (l = 0; l < MOVE; l++) {
  567. render_sleep();
  568. write(fd, &space, 1);
  569. }
  570. for (l = 0; l < MOVE; l++) {
  571. render_sleep();
  572. write(fd, &bs, 1);
  573. }
  574. break;
  575. case 0:
  576. default:
  577. // NORMAL
  578. render_sleep();
  579. write(fd, &ch, 1);
  580. break;
  581. }
  582. }
  583. #ifdef UNWANTED
  584. /**
  585. * render( fd, string_out )
  586. *
  587. * Render an entire string.
  588. * Handles TRIGGER.
  589. * Renders with effects.
  590. */
  591. void render(int fd, const char *string_out, int len) {
  592. const char *cp = string_out;
  593. const char *trigger = cp;
  594. time_t start = time(NULL);
  595. int elapsed;
  596. int over = 0;
  597. reset_render();
  598. // ZF_LOGV("render(%d, %s)", fd, repr(string_out));
  599. ZF_LOGV_MEM(string_out, len, "render(%d, %d bytes):", fd, len);
  600. // Check our time from time to time.
  601. // If we start running long, disable sleeps.
  602. while ((trigger = strnstr(cp, len - (cp - string_out), TRIGGER)) != NULL) {
  603. // There is special things to handle in here.
  604. while (cp != trigger) {
  605. elapsed = time(NULL) - start;
  606. if (elapsed > SLEEP_LIMIT) {
  607. render_overlimit = 1;
  608. current_render.speed = 0;
  609. };
  610. // write(fd, cp, 1 );
  611. render_effect(fd, *cp);
  612. cp++;
  613. };
  614. // ZF_LOGI( "at trigger: (%s)", cp);
  615. cp += strlen(TRIGGER);
  616. // Ok, we're pointing at the trigger -- do something.
  617. cp = process_trigger(fd, cp);
  618. // ZF_LOGI( "after trigger: (%s)", cp);
  619. };
  620. // We still might be under a rendering effect.
  621. while (cp < (string_out + len)) {
  622. elapsed = time(NULL) - start;
  623. if (elapsed > SLEEP_LIMIT) {
  624. render_overlimit = 1;
  625. current_render.speed = 0;
  626. };
  627. // write(fd, cp, 1);
  628. render_effect(fd, *cp);
  629. cp++;
  630. }
  631. }
  632. #endif
  633. /**
  634. * render( fd, string_out )
  635. *
  636. * Render an entire string.
  637. * Handles TRIGGER.
  638. * Renders with effects.
  639. */
  640. void render(int fd, std::string &string_out) {
  641. reset_render();
  642. time_t start = time(NULL);
  643. size_t pos = 0;
  644. int elapsed;
  645. size_t tpos;
  646. while ((tpos = string_out.find(TRIGGER, pos)) != std::string::npos) {
  647. while (pos != tpos) {
  648. elapsed = time(NULL) - start;
  649. if (elapsed > SLEEP_LIMIT) {
  650. render_overlimit = 1;
  651. current_render.speed = 0;
  652. }
  653. render_effect(fd, string_out[pos]);
  654. pos++;
  655. }
  656. pos += strlen(TRIGGER);
  657. process_trigger(fd, string_out, pos);
  658. }
  659. while (pos < string_out.size()) {
  660. elapsed = time(NULL) - start;
  661. if (elapsed > SLEEP_LIMIT) {
  662. render_overlimit = 1;
  663. current_render.speed = 0;
  664. }
  665. render_effect(fd, string_out[pos]);
  666. pos++;
  667. }
  668. }