door.cpp 27 KB

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