render.cpp 16 KB

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