render.cpp 18 KB

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