door.cpp 24 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103
  1. #include "door.h"
  2. #include <algorithm>
  3. #include <chrono>
  4. #include <ctype.h>
  5. #include <string.h>
  6. #include <string>
  7. #include <thread>
  8. #include <libgen.h> // basename
  9. // time/date output std::put_time()
  10. // https://en.cppreference.com/w/cpp/io/manip/put_time
  11. #include <ctime>
  12. #include <iomanip>
  13. // alarm signal
  14. #include <signal.h>
  15. #include <unistd.h>
  16. #include <iconv.h>
  17. #include <algorithm>
  18. #include <iostream>
  19. void to_lower(std::string &text) {
  20. transform(text.begin(), text.end(), text.begin(), ::tolower);
  21. }
  22. namespace door {
  23. static bool hangup = false;
  24. void sig_handler(int signal) {
  25. hangup = true;
  26. /*
  27. ofstream sigf;
  28. sigf.open("signal.log", std::ofstream::out | std::ofstream::app);
  29. sigf << "SNAP! GOT: " << signal << std::endl;
  30. sigf.close();
  31. */
  32. // 13 SIGPIPE -- ok, what do I do with this, eh?
  33. }
  34. class IConv {
  35. iconv_t ic;
  36. public:
  37. IConv(const char *to, const char *from);
  38. ~IConv();
  39. int convert(char *input, char *output, size_t outbufsize);
  40. };
  41. IConv::IConv(const char *to, const char *from) : ic(iconv_open(to, from)) {}
  42. IConv::~IConv() { iconv_close(ic); }
  43. int IConv::convert(char *input, char *output, size_t outbufsize) {
  44. size_t inbufsize = strlen(input);
  45. // size_t orig_size = outbufsize;
  46. // memset(output, 0, outbufsize);
  47. // https://www.gnu.org/savannah-checkouts/gnu/libiconv/documentation/libiconv-1.15/iconv.3.html
  48. int r = iconv(ic, &input, &inbufsize, &output, &outbufsize);
  49. *output = 0;
  50. return r;
  51. }
  52. static IConv converter("UTF-8", "CP437");
  53. void cp437toUnicode(std::string input, std::string &out) {
  54. char buffer[10240];
  55. char output[16384];
  56. strcpy(buffer, input.c_str());
  57. converter.convert(buffer, output, sizeof(output));
  58. out.assign(output);
  59. }
  60. void cp437toUnicode(const char *input, std::string &out) {
  61. char buffer[10240];
  62. char output[16384];
  63. strcpy(buffer, input);
  64. converter.convert(buffer, output, sizeof(output));
  65. out.assign(output);
  66. }
  67. bool unicode = false;
  68. bool debug_capture = false;
  69. /**
  70. * Construct a new Door object using the commandline parameters
  71. * given to the main function.
  72. *
  73. * @example door_example.cpp
  74. */
  75. Door::Door(std::string dname, int argc, char *argv[])
  76. : std::ostream(this), doorname{dname}, has_dropfile{false},
  77. seconds_elapsed{0}, previous(COLOR::WHITE), track{true}, cx{1}, cy{1},
  78. inactivity{120}, node{1} {
  79. // Setup commandline options
  80. opt.addUsage("Door++ library by BUGZ (C) 2021");
  81. opt.addUsage("");
  82. opt.addUsage(" -h --help Displays this help");
  83. opt.addUsage(" -l --local Local Mode");
  84. opt.addUsage(" -d --dropfile [FILENAME] Load Dropfile");
  85. opt.addUsage(" -n --node N Set node number");
  86. // opt.addUsage(" -b --bbsname NAME Set BBS Name");
  87. opt.addUsage(" -u --username NAME Set Username");
  88. opt.addUsage(" -t --timeleft N Set time left");
  89. opt.addUsage(" --maxtime N Set max time");
  90. opt.addUsage("");
  91. opt.setFlag("help", 'h');
  92. opt.setFlag("local", 'l');
  93. opt.setOption("dropfile", 'd');
  94. // opt.setOption("bbsname", 'b');
  95. opt.setOption("username", 'u');
  96. opt.setOption("timeleft", 't');
  97. opt.setOption("maxtime");
  98. opt.processCommandArgs(argc, argv);
  99. if (!opt.hasOptions()) {
  100. opt.printUsage();
  101. exit(1);
  102. }
  103. if (opt.getFlag("help") || opt.getFlag('h')) {
  104. opt.printUsage();
  105. exit(1);
  106. }
  107. if (opt.getValue("username") != nullptr) {
  108. username = opt.getValue("username");
  109. }
  110. if (opt.getValue("node") != nullptr) {
  111. node = atoi(opt.getValue("node"));
  112. }
  113. if (opt.getValue("timeleft") != nullptr) {
  114. time_left = atoi(opt.getValue("timeleft"));
  115. } else {
  116. // sensible default
  117. time_left = 25;
  118. }
  119. /*
  120. if (opt.getValue("bbsname") != nullptr) {
  121. bbsname = opt.getValue("bbsname");
  122. }
  123. */
  124. parse_dropfile(opt.getValue("dropfile"));
  125. /*
  126. If the dropfile has time_left, we'll use it.
  127. Adjust time_left by maxtime value (if given).
  128. */
  129. if (opt.getValue("maxtime") != nullptr) {
  130. int maxtime = atoi(opt.getValue("maxtime"));
  131. if (time_left > maxtime) {
  132. logf << "Adjusting time from " << time_left << " to " << maxtime
  133. << std::endl;
  134. time_left = maxtime;
  135. }
  136. }
  137. if (opt.getFlag("local")) {
  138. if (username.empty()) {
  139. std::cout << "Local mode requires username to be set." << std::endl;
  140. opt.printUsage();
  141. exit(1);
  142. }
  143. } else {
  144. // we must have a dropfile, or else!
  145. if (!has_dropfile) {
  146. std::cout << "I require a dropfile. And a shrubbery." << std::endl;
  147. opt.printUsage();
  148. exit(1);
  149. }
  150. }
  151. // Set program name
  152. std::string logFileName = dname + ".log";
  153. logf.open(logFileName.c_str(), std::ofstream::out | std::ofstream::app);
  154. log("Door init");
  155. init();
  156. // door.sys doesn't give BBS name. system_name
  157. detect_unicode_and_screen();
  158. logf << "Screen " << width << " X " << height << std::endl;
  159. }
  160. Door::~Door() {
  161. // restore default mode
  162. // tcsetattr(STDIN_FILENO, TCSANOW, &tio_default);
  163. log("dtor");
  164. tcsetattr(STDIN_FILENO, TCOFLUSH, &tio_default);
  165. signal(SIGHUP, SIG_DFL);
  166. signal(SIGPIPE, SIG_DFL);
  167. // time thread
  168. stop_thread.set_value();
  169. time_thread.join();
  170. log("done");
  171. logf.close();
  172. }
  173. // https://www.tutorialspoint.com/how-do-i-terminate-a-thread-in-cplusplus11
  174. void Door::time_thread_run(std::future<void> future) {
  175. while (future.wait_for(std::chrono::milliseconds(1)) ==
  176. std::future_status::timeout) {
  177. // std::cout << "Executing the thread....." << std::endl;
  178. std::this_thread::sleep_for(std::chrono::seconds(1));
  179. seconds_elapsed++;
  180. // log("TICK");
  181. // logf << "TICK " << seconds_elapsed << std::endl;
  182. if (seconds_elapsed % 60 == 0) {
  183. if (time_left > 0)
  184. time_left--;
  185. }
  186. }
  187. }
  188. void Door::init(void) {
  189. // https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
  190. // https://viewsourcecode.org/snaptoken/kilo/03.rawInputAndOutput.html
  191. // ok! I didn't know about VTIME ! That's something different!
  192. // enable terminal RAW mode
  193. struct termios tio_raw;
  194. tcgetattr(STDIN_FILENO, &tio_default);
  195. tio_raw = tio_default;
  196. cfmakeraw(&tio_raw);
  197. // local terminal magic
  198. tio_raw.c_cc[VMIN] = 0;
  199. tio_raw.c_cc[VTIME] = 1;
  200. bpos = 0;
  201. tcsetattr(STDIN_FILENO, TCSANOW, &tio_raw);
  202. startup = std::time(nullptr);
  203. signal(SIGHUP, sig_handler);
  204. signal(SIGPIPE, sig_handler);
  205. // time thread
  206. std::future<void> stop_future = stop_thread.get_future();
  207. time_thread =
  208. std::thread(&Door::time_thread_run, this, std::move(stop_future));
  209. }
  210. void Door::detect_unicode_and_screen(void) {
  211. unicode = false;
  212. width = 0;
  213. height = 0;
  214. if (!isatty(STDIN_FILENO)) {
  215. // https://stackoverflow.com/questions/273261/force-telnet-client-into-character-mode
  216. *this << "\377\375\042\377\373\001"; // fix telnet client
  217. }
  218. *this << "\x1b[0;30;40m\x1b[2J\x1b[H"; // black on black, clrscr, go home
  219. *this << "\u2615"
  220. << "\x1b[6n"; // hot beverage + cursor pos
  221. *this << "\x1b[999C\x1b[999B\x1b[6n"; // goto end of screen + cursor pos
  222. *this << "\x1b[H"; // go home
  223. this->flush();
  224. usleep(1000 * 1000);
  225. if (haskey()) {
  226. char buffer[101];
  227. int len;
  228. len = read(STDIN_FILENO, &buffer, sizeof(buffer) - 1);
  229. // logf << "read " << len << std::endl;
  230. if (len > 0) {
  231. buffer[len] = 0;
  232. int x;
  233. /*
  234. for (x = 0; x < len; x++) {
  235. if (buffer[x] < 0x20)
  236. logf << std::hex << (int)buffer[x] << " ";
  237. else
  238. logf << buffer[x] << " ";
  239. }
  240. */
  241. for (x = 0; x < len; x++) {
  242. if (buffer[x] == 0)
  243. buffer[x] = ' ';
  244. }
  245. /*
  246. logf << std::endl;
  247. logf << "BUFFER [" << (char *)buffer << "]" << std::endl;
  248. */
  249. // logf << "BUFFER [" << (char *)buffer << "]" << std::endl;
  250. // this did not work -- because of the null characters in the buffer.
  251. // 1;3R required on David's machine. I'm not sure why.
  252. // 1;3R also happens under VSCodium.
  253. // 1;4R is what I get from syncterm.
  254. if ((strstr(buffer, "1;2R") != nullptr) or
  255. (strstr(buffer, "1;3R") != nullptr)) {
  256. unicode = true;
  257. log("unicode enabled \u2615"); // "U0001f926");
  258. }
  259. // Get the terminal screen size
  260. // \x1b[1;2R\x1b[41;173R
  261. // log(buffer);
  262. char *cp;
  263. cp = strchr(buffer, '\x1b');
  264. if (cp != nullptr) {
  265. cp = strchr(cp + 1, '\x1b');
  266. } else {
  267. log("Failed terminal size detection. See buffer:");
  268. log(buffer);
  269. return;
  270. }
  271. if (cp != nullptr) {
  272. cp++;
  273. if (*cp == '[') {
  274. cp++;
  275. height = atoi(cp);
  276. cp = strchr(cp, ';');
  277. if (cp != nullptr) {
  278. cp++;
  279. width = atoi(cp);
  280. } else {
  281. height = 0;
  282. }
  283. }
  284. }
  285. }
  286. } else {
  287. logf << "FAIL-WHALE, no response to terminal getposition." << std::endl;
  288. }
  289. }
  290. void Door::parse_dropfile(const char *filepath) {
  291. if (filepath == nullptr)
  292. return;
  293. // Ok, parse file here...
  294. std::ifstream file(filepath);
  295. std::string line;
  296. while (std::getline(file, line)) {
  297. // These are "DOS" files. Remove trailing \r.
  298. if (!line.empty() && line[line.size() - 1] == '\r')
  299. line.erase(line.size() - 1);
  300. dropfilelines.push_back(line);
  301. }
  302. file.close();
  303. std::string filename;
  304. {
  305. // converting const char * to char * for basename.
  306. char *temp = strdup(filepath);
  307. filename = basename(temp);
  308. free(temp);
  309. }
  310. to_lower(filename);
  311. // for now, just door.sys.
  312. if (filename == "door.sys") {
  313. // Ok, parse away!
  314. node = atoi(dropfilelines[3].c_str());
  315. username = dropfilelines[9];
  316. location = dropfilelines[10];
  317. time_left = atoi(dropfilelines[18].c_str());
  318. sysop = dropfilelines[34];
  319. handle = dropfilelines[35];
  320. } else {
  321. std::string msg = "Unknown dropfile: ";
  322. msg += filename;
  323. log(msg);
  324. *this << msg << std::endl;
  325. exit(2);
  326. }
  327. has_dropfile = true;
  328. }
  329. void Door::log(std::string output) {
  330. // todo: have logging
  331. std::time_t t = std::time(nullptr);
  332. std::tm tm = *std::localtime(&t);
  333. logf << std::put_time(&tm, "%c ") << output << std::endl;
  334. }
  335. bool Door::haskey(void) {
  336. fd_set socket_set;
  337. struct timeval tv;
  338. int select_ret = -1;
  339. if (hangup)
  340. return -2;
  341. if (time_left < 2)
  342. return -3;
  343. while (select_ret == -1) {
  344. FD_ZERO(&socket_set);
  345. FD_SET(STDIN_FILENO, &socket_set);
  346. tv.tv_sec = 0;
  347. tv.tv_usec = 1;
  348. select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
  349. if (select_ret == -1) {
  350. if (errno == EINTR)
  351. continue;
  352. log("hangup detected");
  353. hangup = true;
  354. return (-2);
  355. }
  356. if (select_ret == 0)
  357. return false;
  358. }
  359. return true;
  360. }
  361. /*
  362. low-lever read a key from terminal or stdin.
  363. Returns key, or
  364. -1 (no key available)
  365. -2 (read error)
  366. */
  367. signed int Door::getch(void) {
  368. fd_set socket_set;
  369. struct timeval tv;
  370. int select_ret = -1;
  371. int recv_ret;
  372. char key;
  373. if (door::hangup)
  374. return -2;
  375. if (time_left < 2)
  376. return -3;
  377. while (select_ret == -1) {
  378. FD_ZERO(&socket_set);
  379. FD_SET(STDIN_FILENO, &socket_set);
  380. tv.tv_sec = 0;
  381. tv.tv_usec = 100;
  382. select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
  383. // select(STDIN_FILENO + 1, &socket_set, NULL, NULL, bWait ? NULL : &tv);
  384. if (select_ret == -1) {
  385. if (errno == EINTR)
  386. continue;
  387. log("hangup detected");
  388. door::hangup = true;
  389. return (-2);
  390. }
  391. if (select_ret == 0)
  392. return (-1);
  393. }
  394. recv_ret = read(STDIN_FILENO, &key, 1);
  395. if (recv_ret != 1) {
  396. // possibly log this.
  397. log("hangup");
  398. hangup = true;
  399. return -2;
  400. }
  401. return key;
  402. }
  403. void Door::unget(char c) {
  404. if (bpos < sizeof(buffer) - 1) {
  405. buffer[bpos] = c;
  406. bpos++;
  407. }
  408. }
  409. char Door::get(void) {
  410. if (bpos == 0)
  411. return 0;
  412. bpos--;
  413. return buffer[bpos];
  414. }
  415. signed int Door::getkey(void) {
  416. signed int c, c2;
  417. if (bpos != 0) {
  418. c = get();
  419. } else {
  420. c = getch();
  421. }
  422. if (c < 0)
  423. return c;
  424. /*
  425. We get 0x0d 0x00 for [Enter] (Syncterm)
  426. This strips out the null.
  427. */
  428. if (c == 0x0d) {
  429. c2 = getch();
  430. if ((c2 != 0) and (c2 >= 0))
  431. unget(c2);
  432. return c;
  433. }
  434. if (c == 0x1b) {
  435. // possible extended key
  436. c2 = getch();
  437. if (c2 < 0) {
  438. // nope, just plain ESC key
  439. return c;
  440. }
  441. // consume extended key values int extended buffer
  442. char extended[16];
  443. unsigned int pos = 0;
  444. extended[pos] = (char)c2;
  445. extended[pos + 1] = 0;
  446. pos++;
  447. while ((pos < sizeof(extended) - 1) and ((c2 = getch()) >= 0)) {
  448. // special case here where I'm sending out cursor location requests
  449. // and the \x1b[X;YR strings are getting buffered.
  450. if (c2 == 0x1b) {
  451. unget(c2);
  452. break;
  453. }
  454. extended[pos] = (char)c2;
  455. extended[pos + 1] = 0;
  456. pos++;
  457. }
  458. // convert extended buffer to special key
  459. if (extended[0] == '[') {
  460. switch (extended[1]) {
  461. case 'A':
  462. return XKEY_UP_ARROW;
  463. case 'B':
  464. return XKEY_DOWN_ARROW;
  465. case 'C':
  466. return XKEY_RIGHT_ARROW;
  467. case 'D':
  468. return XKEY_LEFT_ARROW;
  469. case 'H':
  470. return XKEY_HOME;
  471. case 'F':
  472. return XKEY_END; // terminal
  473. case 'K':
  474. return XKEY_END;
  475. case 'U':
  476. return XKEY_PGUP;
  477. case 'V':
  478. return XKEY_PGDN;
  479. case '@':
  480. return XKEY_INSERT;
  481. }
  482. if (extended[pos - 1] == '~') {
  483. // \x1b[digits~
  484. int number = atoi(extended + 1);
  485. switch (number) {
  486. case 2:
  487. return XKEY_INSERT; // terminal
  488. case 3:
  489. return XKEY_DELETE; // terminal
  490. case 5:
  491. return XKEY_PGUP; // terminal
  492. case 6:
  493. return XKEY_PGDN; // terminal
  494. case 15:
  495. return XKEY_F5; // terminal
  496. case 17:
  497. return XKEY_F6; // terminal
  498. case 18:
  499. return XKEY_F7; // terminal
  500. case 19:
  501. return XKEY_F8; // terminal
  502. case 20:
  503. return XKEY_F9; // terminal
  504. case 21:
  505. return XKEY_F10; // terminal
  506. case 23:
  507. return XKEY_F11;
  508. case 24:
  509. return XKEY_F12; // terminal
  510. }
  511. }
  512. }
  513. if (extended[0] == 'O') {
  514. switch (extended[1]) {
  515. case 'P':
  516. return XKEY_F1;
  517. case 'Q':
  518. return XKEY_F2;
  519. case 'R':
  520. return XKEY_F3;
  521. case 'S':
  522. return XKEY_F4;
  523. case 't':
  524. return XKEY_F5; // syncterm
  525. }
  526. }
  527. // unknown -- This needs to be logged
  528. logf << "\r\nDEBUG:\r\nESC + ";
  529. for (unsigned int x = 0; x < pos; x++) {
  530. char z = extended[x];
  531. if (iscntrl(z)) {
  532. logf << (int)z << " ";
  533. } else {
  534. logf << "'" << (char)z << "'"
  535. << " ";
  536. };
  537. }
  538. logf << "\r\n";
  539. logf.flush();
  540. return XKEY_UNKNOWN;
  541. }
  542. return c;
  543. }
  544. int Door::get_input(void) {
  545. signed int c;
  546. c = getkey();
  547. if (c < 0)
  548. return 0;
  549. return c;
  550. /*
  551. tODInputEvent event;
  552. od_get_input(&event, OD_NO_TIMEOUT, GETIN_NORMAL);
  553. */
  554. }
  555. /*
  556. The following code will wait for 1.5 second:
  557. #include <sys/select.h>
  558. #include <sys/time.h>
  559. #include <unistd.h>`
  560. int main() {
  561. struct timeval t;
  562. t.tv_sec = 1;
  563. t.tv_usec = 500000;
  564. select(0, NULL, NULL, NULL, &t);
  565. }
  566. */
  567. signed int Door::sleep_key(int secs) {
  568. fd_set socket_set;
  569. struct timeval tv;
  570. int select_ret = -1;
  571. /*
  572. int recv_ret;
  573. char key;
  574. */
  575. if (hangup)
  576. return -2;
  577. if (time_left < 2)
  578. return -3;
  579. while (select_ret == -1) {
  580. FD_ZERO(&socket_set);
  581. FD_SET(STDIN_FILENO, &socket_set);
  582. tv.tv_sec = secs;
  583. tv.tv_usec = 0;
  584. select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
  585. // select(STDIN_FILENO + 1, &socket_set, NULL, NULL, bWait ? NULL : &tv);
  586. if (select_ret == -1) {
  587. if (errno == EINTR)
  588. continue;
  589. hangup = true;
  590. log("hangup detected");
  591. return (-2);
  592. }
  593. if (select_ret == 0)
  594. return (-1);
  595. }
  596. return getkey();
  597. }
  598. std::string Door::input_string(int max) {
  599. std::string input;
  600. // draw the input area.
  601. *this << std::string(max, ' ');
  602. *this << std::string(max, '\x08');
  603. int c;
  604. while (true) {
  605. c = sleep_key(inactivity);
  606. if (c < 0) {
  607. input.clear();
  608. return input;
  609. }
  610. if (c > 0x1000)
  611. continue;
  612. if (isprint(c)) {
  613. if (int(input.length()) < max) {
  614. *this << char(c);
  615. input.append(1, c);
  616. } else {
  617. // bell
  618. *this << '\x07';
  619. }
  620. } else {
  621. switch (c) {
  622. case 0x08:
  623. case 0x7f:
  624. if (input.length() > 0) {
  625. *this << "\x08 \x08";
  626. // this->flush();
  627. input.erase(input.length() - 1);
  628. };
  629. break;
  630. case 0x0d:
  631. return input;
  632. }
  633. }
  634. }
  635. }
  636. /**
  637. * Take given buffer and output it.
  638. *
  639. * This originally stored output in the buffer. We now directly output
  640. * via the OpenDoor od_disp_emu.
  641. *
  642. * @param s const char *
  643. * @param n std::streamsize
  644. * @return std::streamsize
  645. */
  646. std::streamsize Door::xsputn(const char *s, std::streamsize n) {
  647. if (debug_capture) {
  648. debug_buffer.append(s, n);
  649. } else {
  650. static std::string buffer;
  651. buffer.append(s, n);
  652. // setp(&(*buffer.begin()), &(*buffer.end()));
  653. if (!hangup) {
  654. std::cout << buffer;
  655. std::cout.flush();
  656. }
  657. // od_disp_emu(buffer.c_str(), TRUE);
  658. // Tracking character position could be a problem / local terminal unicode.
  659. if (track)
  660. cx += n;
  661. buffer.clear();
  662. }
  663. return n;
  664. }
  665. /**
  666. * Stores a character into the buffer.
  667. * This does still use the buffer.
  668. * @todo Replace this also with a direct call to od_disp_emu.
  669. *
  670. * @param c char
  671. * @return int
  672. */
  673. int Door::overflow(char c) {
  674. /*
  675. char temp[2];
  676. temp[0] = c;
  677. temp[1] = 0;
  678. */
  679. // buffer.push_back(c);
  680. // od_disp_emu(temp, TRUE);
  681. if (debug_capture) {
  682. debug_buffer.append(1, c);
  683. } else {
  684. if (!hangup) {
  685. std::cout << c;
  686. std::cout.flush();
  687. }
  688. }
  689. if (track)
  690. cx++;
  691. // setp(&(*buffer.begin()), &(*buffer.end()));
  692. return c;
  693. }
  694. /**
  695. * Construct a new Color Output:: Color Output object
  696. * We default to BLACK/BLACK (not really a valid color),
  697. * pos=0 and len=0.
  698. */
  699. ColorOutput::ColorOutput() : c(COLOR::BLACK, COLOR::BLACK) {
  700. pos = 0;
  701. len = 0;
  702. }
  703. /**
  704. * Reset pos and len to 0.
  705. */
  706. void ColorOutput::reset(void) {
  707. pos = 0;
  708. len = 0;
  709. }
  710. /**
  711. * Construct a new Render:: Render object
  712. *
  713. * Render consists of constant text,
  714. * and vector of ColorOutput
  715. *
  716. * @param txt Text
  717. */
  718. Render::Render(const std::string txt) : text{txt} {}
  719. /**
  720. * Output the Render.
  721. *
  722. * This consists of going through the vector, getting
  723. * the fragment (from pos and len), and outputting the
  724. * color and fragment.
  725. *
  726. * @param os
  727. */
  728. void Render::output(std::ostream &os) {
  729. for (auto out : outputs) {
  730. std::string fragment = text.substr(out.pos, out.len);
  731. os << out.c << fragment;
  732. }
  733. }
  734. /**
  735. * Construct a new Clrscr:: Clrscr object
  736. *
  737. * This is used to clear the screen.
  738. */
  739. Clrscr::Clrscr() {}
  740. /**
  741. * Clear the screen using ANSI codes.
  742. *
  743. * Not all systems home the cursor after clearing the screen.
  744. * We automatically home the cursor as well.
  745. *
  746. * @param os std::ostream&
  747. * @param clr const Clrscr&
  748. * @return std::ostream&
  749. */
  750. std::ostream &operator<<(std::ostream &os, const Clrscr &clr) {
  751. Door *d = dynamic_cast<Door *>(&os);
  752. if (d != nullptr) {
  753. d->track = false;
  754. *d << "\x1b[2J"
  755. "\x1b[H";
  756. d->cx = 1;
  757. d->cy = 1;
  758. d->track = true;
  759. } else {
  760. os << "\x1b[2J"
  761. "\x1b[H";
  762. }
  763. return os;
  764. }
  765. Clrscr cls;
  766. /**
  767. * This is used to issue NL+CR
  768. *
  769. */
  770. NewLine::NewLine() {}
  771. /**
  772. * Output Newline + CarriageReturn
  773. * @param os std::ostream
  774. * @param nl const NewLine
  775. * @return std::ostream&
  776. */
  777. std::ostream &operator<<(std::ostream &os, const NewLine &nl) {
  778. Door *d = dynamic_cast<Door *>(&os);
  779. if (d != nullptr) {
  780. d->track = false;
  781. *d << "\r\n";
  782. d->cx = 1;
  783. d->cy++;
  784. d->track = true;
  785. } else {
  786. os << "\r\n";
  787. };
  788. return os;
  789. }
  790. NewLine nl;
  791. /**
  792. * Construct a new Goto:: Goto object
  793. *
  794. * @param xpos
  795. * @param ypos
  796. */
  797. Goto::Goto(int xpos, int ypos) {
  798. x = xpos;
  799. y = ypos;
  800. }
  801. /**
  802. * Output the ANSI codes to position the cursor to the given y,x position.
  803. *
  804. * @todo Optimize the ANSI goto string output.
  805. * @todo Update the Door object so it know where the cursor
  806. * is positioned.
  807. *
  808. * @param os std::ostream
  809. * @param g const Goto
  810. * @return std::ostream&
  811. */
  812. std::ostream &operator<<(std::ostream &os, const Goto &g) {
  813. Door *d = dynamic_cast<Door *>(&os);
  814. if (d != nullptr) {
  815. d->track = false;
  816. *d << "\x1b[";
  817. if (g.y > 1)
  818. *d << std::to_string(g.y);
  819. os << ";";
  820. if (g.x > 1)
  821. *d << std::to_string(g.x);
  822. *d << "H";
  823. d->cx = g.x;
  824. d->cy = g.y;
  825. d->track = true;
  826. } else {
  827. os << "\x1b[" << std::to_string(g.y) << ";" << std::to_string(g.x) << "H";
  828. };
  829. return os;
  830. }
  831. // EXAMPLES
  832. /// BlueYellow Render example function
  833. renderFunction rBlueYellow = [](const std::string &txt) -> Render {
  834. Render r(txt);
  835. ColorOutput co;
  836. bool uc = true;
  837. ANSIColor blue(COLOR::BLUE, ATTR::BOLD);
  838. ANSIColor cyan(COLOR::YELLOW, ATTR::BOLD);
  839. co.pos = 0;
  840. co.len = 0;
  841. co.c = blue;
  842. // d << blue;
  843. int tpos = 0;
  844. for (char const &c : txt) {
  845. if (uc) {
  846. if (!isupper(c)) {
  847. // possible color change
  848. if (co.len != 0) {
  849. r.outputs.push_back(co);
  850. co.reset();
  851. co.pos = tpos;
  852. }
  853. co.c = cyan;
  854. // d << cyan;
  855. uc = false;
  856. }
  857. } else {
  858. if (isupper(c)) {
  859. if (co.len != 0) {
  860. r.outputs.push_back(co);
  861. co.reset();
  862. co.pos = tpos;
  863. }
  864. co.c = blue;
  865. // d << blue;
  866. uc = true;
  867. }
  868. }
  869. co.len++;
  870. tpos++;
  871. // d << c;
  872. }
  873. if (co.len != 0) {
  874. r.outputs.push_back(co);
  875. }
  876. return r;
  877. };
  878. door::renderFunction renderStatusValue(door::ANSIColor status,
  879. door::ANSIColor value) {
  880. door::renderFunction rf = [status,
  881. value](const std::string &txt) -> door::Render {
  882. door::Render r(txt);
  883. door::ColorOutput co;
  884. co.pos = 0;
  885. co.len = 0;
  886. co.c = status;
  887. size_t pos = txt.find(':');
  888. if (pos == std::string::npos) {
  889. // failed to find - use entire string as status color.
  890. co.len = txt.length();
  891. r.outputs.push_back(co);
  892. } else {
  893. pos++; // Have : in status color
  894. co.len = pos;
  895. r.outputs.push_back(co);
  896. co.reset();
  897. co.pos = pos;
  898. co.c = value;
  899. co.len = txt.length() - pos;
  900. r.outputs.push_back(co);
  901. }
  902. return r;
  903. };
  904. return rf;
  905. }
  906. door::renderFunction rStatusValue = [](const std::string &txt) -> door::Render {
  907. door::Render r(txt);
  908. door::ColorOutput co;
  909. // default colors STATUS: value
  910. door::ANSIColor status(door::COLOR::BLUE, door::ATTR::BOLD);
  911. door::ANSIColor value(door::COLOR::YELLOW, door::ATTR::BOLD);
  912. co.pos = 0;
  913. co.len = 0;
  914. co.c = status;
  915. size_t pos = txt.find(':');
  916. if (pos == std::string::npos) {
  917. // failed to find - use entire string as status color.
  918. co.len = txt.length();
  919. r.outputs.push_back(co);
  920. } else {
  921. pos++; // Have : in status color
  922. co.len = pos;
  923. r.outputs.push_back(co);
  924. co.reset();
  925. co.pos = pos;
  926. co.c = value;
  927. co.len = txt.length() - pos;
  928. r.outputs.push_back(co);
  929. }
  930. return r;
  931. };
  932. /*
  933. std::function<void(Door &d, std::string &txt)> BlueYellow2 =
  934. [](Door &d, std::string &txt) -> void {
  935. bool uc = true;
  936. ANSIColor blue(COLOR::BLACK, COLOR::CYAN);
  937. ANSIColor cyan(COLOR::YELLOW, COLOR::BLUE, ATTR::BOLD);
  938. d << blue;
  939. for (char const &c : txt) {
  940. if (uc) {
  941. if (c == ':') {
  942. d << cyan;
  943. uc = false;
  944. }
  945. }
  946. d << c;
  947. }
  948. };
  949. std::function<void(Door &d, std::string &txt)> Aweful =
  950. [](Door &d, std::string &txt) -> void {
  951. for (char const &c : txt) {
  952. // Color clr((Colors)((c % 14) + 1), Colors::BLACK, 0);
  953. // Use only BRIGHT/LIGHT colors.
  954. ANSIColor clr((COLOR)(c % 8), ATTR::BOLD);
  955. d << clr << c;
  956. }
  957. };
  958. */
  959. } // namespace door