123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242 |
- #include "door.h"
- #include <algorithm>
- #include <chrono>
- #include <ctype.h>
- #include <string.h>
- #include <string>
- #include <thread>
- #include <libgen.h> // basename
- // time/date output std::put_time()
- // https://en.cppreference.com/w/cpp/io/manip/put_time
- #include <ctime>
- #include <iomanip>
- // alarm signal
- #include <signal.h>
- #include <unistd.h>
- #include <iconv.h>
- #include <algorithm>
- #include <iostream>
- /**
- * @file
- * @brief Door
- */
- /*
- My strategy here has failed.
- If I set enigma to cp437, then it handles everything but the cp437
- symbols (diamonds/hearts/spades/clubs) correctly on the unicode side.
- [And my door thinks it's all cp437 always]
- If I set enigma to utf8, then it works right on the ssh terminal side.
- But cp437 turns to puke because it's trying to convert cp437 from
- utf8 to cp437. The symbols get '?'.
- I can't detect unicode (when set to utf8), but I can detect cp437
- (by sending the diamonds/hearts characters).
- But I can't get through the enigma translation system. If only iconv worked
- correctly with hearts/clubs symbols! Then I wouldn't need this broken
- work-around code.
- */
- namespace door {
- void to_lower(std::string &text) {
- transform(text.begin(), text.end(), text.begin(), ::tolower);
- }
- bool replace(std::string &str, const std::string &from, const std::string &to) {
- size_t start_pos = str.find(from);
- if (start_pos == std::string::npos)
- return false;
- str.replace(start_pos, from.length(), to);
- return true;
- }
- bool replace(std::string &str, const char *from, const char *to) {
- size_t start_pos = str.find(from);
- if (start_pos == std::string::npos)
- return false;
- str.replace(start_pos, strlen(from), to);
- return true;
- }
- static bool hangup = false;
- void sig_handler(int signal) {
- hangup = true;
- /*
- ofstream sigf;
- sigf.open("signal.log", std::ofstream::out | std::ofstream::app);
- sigf << "SNAP! GOT: " << signal << std::endl;
- sigf.close();
- */
- // 13 SIGPIPE -- ok, what do I do with this, eh?
- }
- class IConv {
- iconv_t ic;
- public:
- IConv(const char *to, const char *from);
- ~IConv();
- int convert(char *input, char *output, size_t outbufsize);
- };
- IConv::IConv(const char *to, const char *from) : ic(iconv_open(to, from)) {}
- IConv::~IConv() { iconv_close(ic); }
- int IConv::convert(char *input, char *output, size_t outbufsize) {
- size_t inbufsize = strlen(input);
- // size_t orig_size = outbufsize;
- // memset(output, 0, outbufsize);
- // https://www.gnu.org/savannah-checkouts/gnu/libiconv/documentation/libiconv-1.15/iconv.3.html
- int r = iconv(ic, &input, &inbufsize, &output, &outbufsize);
- *output = 0;
- return r;
- }
- static IConv converter("UTF-8", "CP437");
- void cp437toUnicode(std::string input, std::string &out) {
- char buffer[10240];
- char output[16384];
- strcpy(buffer, input.c_str());
- converter.convert(buffer, output, sizeof(output));
- out.assign(output);
- }
- void cp437toUnicode(const char *input, std::string &out) {
- char buffer[10240];
- char output[16384];
- strcpy(buffer, input);
- converter.convert(buffer, output, sizeof(output));
- out.assign(output);
- }
- bool unicode = false;
- bool full_cp437 = false;
- bool debug_capture = false;
- /**
- * Construct a new Door object using the commandline parameters
- * given to the main function.
- *
- * @example door-example.cpp
- */
- Door::Door(std::string dname, int argc, char *argv[])
- : std::ostream(this), doorname{dname},
- has_dropfile{false}, debugging{false}, seconds_elapsed{0},
- previous(COLOR::WHITE), track{true}, cx{1}, cy{1},
- inactivity{120}, node{1} {
- // Setup commandline options
- opt.addUsage("Door++ library by BUGZ (C) 2021");
- opt.addUsage("");
- opt.addUsage(" -h --help Displays this help");
- opt.addUsage(" -l --local Local Mode");
- opt.addUsage(" -d --dropfile [FILENAME] Load Dropfile");
- opt.addUsage(" -n --node N Set node number");
- // opt.addUsage(" -c --cp437 Force CP437");
- // opt.addUsage(" -b --bbsname NAME Set BBS Name");
- opt.addUsage(" -u --username NAME Set Username");
- opt.addUsage(" -t --timeleft N Set time left");
- opt.addUsage(" --maxtime N Set max time");
- opt.addUsage("");
- opt.setFlag("help", 'h');
- opt.setFlag("local", 'l');
- opt.setFlag("cp437", 'c');
- opt.setFlag("unicode");
- opt.setFlag("debuggering");
- opt.setOption("dropfile", 'd');
- // opt.setOption("bbsname", 'b');
- opt.setOption("username", 'u');
- opt.setOption("timeleft", 't');
- opt.setOption("maxtime");
- opt.processCommandArgs(argc, argv);
- if (!opt.hasOptions()) {
- opt.printUsage();
- exit(1);
- }
- if (opt.getFlag("help") || opt.getFlag('h')) {
- opt.printUsage();
- exit(1);
- }
- if (opt.getValue("username") != nullptr) {
- username = opt.getValue("username");
- handle = username;
- }
- if (opt.getFlag("debuggering")) {
- debugging = true;
- }
- if (opt.getValue("node") != nullptr) {
- node = atoi(opt.getValue("node"));
- }
- if (opt.getValue("timeleft") != nullptr) {
- time_left = atoi(opt.getValue("timeleft"));
- } else {
- // sensible default
- time_left = 25;
- }
- time_used = 0;
- /*
- if (opt.getValue("bbsname") != nullptr) {
- bbsname = opt.getValue("bbsname");
- }
- */
- std::string logFileName = dname + ".log";
- logf.open(logFileName.c_str(), std::ofstream::out | std::ofstream::app);
- parse_dropfile(opt.getValue("dropfile"));
- /*
- If the dropfile has time_left, we'll use it.
- Adjust time_left by maxtime value (if given).
- */
- if (opt.getValue("maxtime") != nullptr) {
- int maxtime = atoi(opt.getValue("maxtime"));
- if (time_left > maxtime) {
- logf << "Adjusting time from " << time_left << " to " << maxtime
- << std::endl;
- time_left = maxtime;
- }
- }
- if (opt.getFlag("local")) {
- if (username.empty()) {
- std::cout << "Local mode requires username to be set." << std::endl;
- opt.printUsage();
- exit(1);
- }
- } else {
- // we must have a dropfile, or else!
- if (!has_dropfile) {
- std::cout << "I require a dropfile. And a shrubbery." << std::endl;
- opt.printUsage();
- exit(1);
- }
- }
- // Set program name
- log() << "Door init" << std::endl;
- init();
- // door.sys doesn't give BBS name. system_name
- if (!debugging) {
- detect_unicode_and_screen();
- logf << "Screen " << width << " X " << height << " unicode " << unicode
- << " full_cp437 " << full_cp437 << std::endl;
- }
- if (opt.getFlag("cp437")) {
- unicode = false;
- }
- if (opt.getFlag("unicode")) {
- unicode = true;
- }
- }
- Door::~Door() {
- // restore default mode
- // tcsetattr(STDIN_FILENO, TCSANOW, &tio_default);
- log() << "dtor" << std::endl;
- tcsetattr(STDIN_FILENO, TCOFLUSH, &tio_default);
- signal(SIGHUP, SIG_DFL);
- signal(SIGPIPE, SIG_DFL);
- // stop time thread
- stop_thread.set_value();
- time_thread.join();
- log() << "done" << std::endl;
- logf.close();
- }
- // https://www.tutorialspoint.com/how-do-i-terminate-a-thread-in-cplusplus11
- void Door::time_thread_run(std::future<void> future) {
- while (future.wait_for(std::chrono::milliseconds(1)) ==
- std::future_status::timeout) {
- std::this_thread::sleep_for(std::chrono::seconds(1));
- ++seconds_elapsed;
- if (seconds_elapsed % 60 == 0) {
- if (time_left > 0)
- --time_left;
- ++time_used;
- }
- }
- }
- void Door::init(void) {
- // https://viewsourcecode.org/snaptoken/kilo/02.enteringRawMode.html
- // https://viewsourcecode.org/snaptoken/kilo/03.rawInputAndOutput.html
- // ok! I didn't know about VTIME ! That's something different!
- // enable terminal RAW mode
- struct termios tio_raw;
- tcgetattr(STDIN_FILENO, &tio_default);
- tio_raw = tio_default;
- cfmakeraw(&tio_raw);
- // local terminal magic
- tio_raw.c_cc[VMIN] = 0;
- tio_raw.c_cc[VTIME] = 1;
- bpos = 0;
- tcsetattr(STDIN_FILENO, TCSANOW, &tio_raw);
- startup = std::time(nullptr);
- signal(SIGHUP, sig_handler);
- signal(SIGPIPE, sig_handler);
- // time thread
- std::future<void> stop_future = stop_thread.get_future();
- time_thread =
- std::thread(&Door::time_thread_run, this, std::move(stop_future));
- }
- void Door::detect_unicode_and_screen(void) {
- unicode = false;
- full_cp437 = false;
- width = 0;
- height = 0;
- if (!isatty(STDIN_FILENO)) {
- // https://stackoverflow.com/questions/273261/force-telnet-client-into-character-mode
- *this << "\377\375\042\377\373\001"; // fix telnet client
- }
- // maybe I need to be trying to detect cp437 instead of trying to detect
- // unicde!
- *this << "\x1b[0;30;40m\x1b[2J\x1b[H"; // black on black, clrscr, go home
- *this << "\x03\x04" // hearts and diamonds does CP437 work?
- << "\x1b[6n"; // cursor pos
- *this << door::nl << "\u2615"
- << "\x1b[6n"; // hot beverage + cursor pos
- *this << "\x1b[999C\x1b[999B\x1b[6n"; // goto end of screen + cursor pos
- *this << reset << "\x1b[2J\x1b[H"; // reset, cls, go home
- this->flush();
- usleep(1000 * 1000);
- if (haskey()) {
- char buffer[101];
- int len;
- len = read(STDIN_FILENO, &buffer, sizeof(buffer) - 1);
- // logf << "read " << len << std::endl;
- if (len > 0) {
- buffer[len] = 0;
- int x;
- #ifdef DEBUG_OUTPUT
- for (x = 0; x < len; x++) {
- if (buffer[x] < 0x20)
- logf << std::hex << (int)buffer[x] << " ";
- else
- logf << buffer[x] << " ";
- }
- #endif
- for (x = 0; x < len; x++) {
- if (buffer[x] == 0)
- buffer[x] = ' ';
- }
- #ifdef DEBUG_OUTPUT
- logf << std::endl;
- logf << "BUFFER [" << (char *)buffer << "]" << std::endl;
- #endif
- // log detection results
- if (1) {
- std::string cleanbuffer = buffer;
- std::string esc = "\x1b";
- std::string esc_text = "^[";
- while (replace(cleanbuffer, esc, esc_text)) {
- };
- logf << "BUFFER [" << cleanbuffer << "]" << std::endl;
- }
- // 1;3R required on David's machine. I'm not sure why.
- // 1;3R also happens under VSCodium.
- // 1;4R is what I get from syncterm.
- if (((strstr(buffer, "1;1R") != nullptr) or
- (strstr(buffer, "1;3R") != nullptr)) and
- ((strstr(buffer, "2;2R") != nullptr) or
- (strstr(buffer, "2;3R") != nullptr))) {
- unicode = true;
- log() << "unicode enabled \u2615" << std::endl; // "U0001f926");
- } else {
- if (strstr(buffer, "1;3R") != nullptr) {
- full_cp437 = true;
- }
- }
- // Get the terminal screen size
- char *cp;
- cp = strrchr(buffer, '\x1b');
- if (cp != nullptr) {
- cp++;
- if (*cp == '[') {
- cp++;
- height = atoi(cp);
- cp = strchr(cp, ';');
- if (cp != nullptr) {
- cp++;
- width = atoi(cp);
- if (width > 900) {
- // something went wrong
- width = 0;
- height = 0;
- }
- } else {
- height = 0;
- }
- }
- }
- }
- } else {
- logf << "FAIL-WHALE, no response to terminal getposition." << std::endl;
- }
- }
- void Door::parse_dropfile(const char *filepath) {
- if (filepath == nullptr)
- return;
- // Ok, parse file here...
- std::ifstream file(filepath);
- std::string line;
- while (std::getline(file, line)) {
- // These are "DOS" files. Remove trailing \r.
- if (!line.empty() && line[line.size() - 1] == '\r')
- line.erase(line.size() - 1);
- dropfilelines.push_back(line);
- }
- file.close();
- std::string filename;
- {
- // converting const char * to char * for basename.
- char *temp = strdup(filepath);
- filename = basename(temp);
- free(temp);
- }
- to_lower(filename);
- // for now, just door.sys.
- if (filename == "door.sys") {
- // Ok, parse away!
- node = atoi(dropfilelines[3].c_str());
- username = dropfilelines[9];
- location = dropfilelines[10];
- time_left = atoi(dropfilelines[18].c_str());
- sysop = dropfilelines[34];
- handle = dropfilelines[35];
- } else {
- if (filename == "door32.sys") {
- // https://raw.githubusercontent.com/NuSkooler/ansi-bbs/master/docs/dropfile_formats/door32_sys.txt
- // dropfilelines[0] = Comm type (0=local, 1=serial, 2=telnet)
- // dropfilelines[1] = Comm or Socket handle
- // dropfilelines[2] = BaudRate
- // dropfilelines[3] = BBS Software Version
- username = dropfilelines[4];
- handle = dropfilelines[5];
- time_left = atoi(dropfilelines[6].c_str());
- // dropfilelines[7] = Emulation (0=Ascii, 1=ANSI, .. or above = ANSI)
- node = atoi(dropfilelines[8].c_str());
- } else {
- std::string msg = "Unknown dropfile: ";
- msg += filename;
- log() << msg << std::endl;
- *this << msg << std::endl;
- exit(2);
- }
- }
- log() << "node:" << node << " username: " << username << " handle: " << handle
- << " time: " << time_left << std::endl;
- has_dropfile = true;
- }
- /**
- * @brief Give ofstream handle for logging
- *
- * This appends the current date/time stamp into the logfile, and returns a
- * reference.
- *
- * Example:
- *
- * door.log() << "Something bad just happened." << std::endl;
- *
- * @return ofstream&
- */
- ofstream &Door::log(void) {
- std::time_t t = std::time(nullptr);
- std::tm tm = *std::localtime(&t);
- logf << std::put_time(&tm, "%c ");
- return logf;
- }
- bool Door::haskey(void) {
- fd_set socket_set;
- struct timeval tv;
- int select_ret = -1;
- if (hangup)
- return -2;
- if (time_left < 2)
- return -3;
- while (select_ret == -1) {
- FD_ZERO(&socket_set);
- FD_SET(STDIN_FILENO, &socket_set);
- tv.tv_sec = 0;
- tv.tv_usec = 1;
- select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
- if (select_ret == -1) {
- if (errno == EINTR)
- continue;
- log() << "hangup detected" << std::endl;
- hangup = true;
- return (-2);
- }
- if (select_ret == 0)
- return false;
- }
- return true;
- }
- /*
- low-lever read a key from terminal or stdin.
- Returns key, or
- -1 (no key available)
- -2 (read error)
- */
- signed int Door::getch(void) {
- fd_set socket_set;
- struct timeval tv;
- int select_ret = -1;
- int recv_ret;
- char key;
- if (door::hangup)
- return -2;
- if (time_left < 2)
- return -3;
- while (select_ret == -1) {
- FD_ZERO(&socket_set);
- FD_SET(STDIN_FILENO, &socket_set);
- // This delay isn't long enough for QModem in a DOSBOX.
- // doorway mode arrow keys aren't always caught.
- tv.tv_sec = 0;
- tv.tv_usec = 100;
- // tv.tv_usec = 500;
- select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
- // select(STDIN_FILENO + 1, &socket_set, NULL, NULL, bWait ? NULL : &tv);
- if (select_ret == -1) {
- if (errno == EINTR)
- continue;
- log() << "hangup detected" << std::endl;
- door::hangup = true;
- return (-2);
- }
- if (select_ret == 0)
- return (-1);
- }
- recv_ret = read(STDIN_FILENO, &key, 1);
- if (recv_ret != 1) {
- // possibly log this.
- log() << "hangup" << std::endl;
- hangup = true;
- return -2;
- }
- // debug weird keys/layouts.
- // log() << "read " << std::hex << (int)key << std::endl;
- return key;
- }
- void Door::unget(char c) {
- if (bpos < sizeof(buffer) - 1) {
- buffer[bpos] = c;
- bpos++;
- }
- }
- char Door::get(void) {
- if (bpos == 0)
- return 0;
- bpos--;
- return buffer[bpos];
- }
- signed int Door::getkey(void) {
- signed int c, c2;
- if (bpos != 0) {
- c = get();
- } else {
- c = getch();
- }
- if (c < 0)
- return c;
- /*
- We get 0x0d 0x00 for [Enter] (Syncterm)
- From David's syncterm, I'm getting 0x0d 0x0a.
- This strips out the null.
- */
- if (c == 0x0d) {
- c2 = getch();
- if ((c2 != 0) and (c2 >= 0) and (c2 != 0x0a)) {
- log() << "got " << (int)c2 << " so stuffing it into unget buffer."
- << std::endl;
- unget(c2);
- }
- return c;
- }
- if (c == 0) {
- // possibly "doorway mode"
- int tries = 0;
- c2 = getch();
- while (c2 < 0) {
- if (tries > 7) {
- log() << "ok, got " << c2 << " and " << tries << " so returning 0x00!"
- << std::endl;
- return c;
- }
- c2 = getch();
- ++tries;
- }
- if (tries > 0) {
- log() << "tries " << tries << std::endl;
- }
- switch (c2) {
- case 0x50:
- return XKEY_DOWN_ARROW;
- case 0x48:
- return XKEY_UP_ARROW;
- case 0x4b:
- return XKEY_LEFT_ARROW;
- case 0x4d:
- return XKEY_RIGHT_ARROW;
- case 0x47:
- return XKEY_HOME;
- case 0x4f:
- return XKEY_END;
- case 0x49:
- return XKEY_PGUP;
- case 0x51:
- return XKEY_PGDN;
- case 0x3b:
- return XKEY_F1;
- case 0x3c:
- return XKEY_F2;
- case 0x3d:
- return XKEY_F3;
- case 0x3e:
- return XKEY_F4;
- case 0x3f:
- return XKEY_F5;
- case 0x40:
- return XKEY_F6;
- case 0x41:
- return XKEY_F7;
- case 0x42:
- return XKEY_F8;
- case 0x43:
- return XKEY_F9;
- case 0x44:
- return XKEY_F10;
- /*
- case 0x45:
- return XKEY_F11;
- case 0x46:
- return XKEY_F12;
- */
- case 0x52:
- return XKEY_INSERT;
- case 0x53:
- return XKEY_DELETE;
- }
- logf << "\r\nDEBUG:\r\n0x00 + 0x" << std::hex << (int)c2;
- logf << "\r\n";
- logf.flush();
- }
- if (c == 0x1b) {
- // possible extended key
- c2 = getch();
- if (c2 < 0) {
- // nope, just plain ESC key
- return c;
- }
- // consume extended key values int extended buffer
- char extended[16];
- unsigned int pos = 0;
- extended[pos] = (char)c2;
- extended[pos + 1] = 0;
- pos++;
- while ((pos < sizeof(extended) - 1) and ((c2 = getch()) >= 0)) {
- // special case here where I'm sending out cursor location requests
- // and the \x1b[X;YR strings are getting buffered.
- if (c2 == 0x1b) {
- unget(c2);
- break;
- }
- extended[pos] = (char)c2;
- extended[pos + 1] = 0;
- pos++;
- }
- // convert extended buffer to special key
- if (extended[0] == '[') {
- switch (extended[1]) {
- case 'A':
- return XKEY_UP_ARROW;
- case 'B':
- return XKEY_DOWN_ARROW;
- case 'C':
- return XKEY_RIGHT_ARROW;
- case 'D':
- return XKEY_LEFT_ARROW;
- case 'H':
- return XKEY_HOME;
- case 'F':
- return XKEY_END; // terminal
- case 'K':
- return XKEY_END;
- case 'U':
- return XKEY_PGUP;
- case 'V':
- return XKEY_PGDN;
- case '@':
- return XKEY_INSERT;
- }
- if (extended[pos - 1] == '~') {
- // \x1b[digits~
- int number = atoi(extended + 1);
- switch (number) {
- case 2:
- return XKEY_INSERT; // terminal
- case 3:
- return XKEY_DELETE; // terminal
- case 5:
- return XKEY_PGUP; // terminal
- case 6:
- return XKEY_PGDN; // terminal
- case 15:
- return XKEY_F5; // terminal
- case 17:
- return XKEY_F6; // terminal
- case 18:
- return XKEY_F7; // terminal
- case 19:
- return XKEY_F8; // terminal
- case 20:
- return XKEY_F9; // terminal
- case 21:
- return XKEY_F10; // terminal
- case 23:
- return XKEY_F11;
- case 24:
- return XKEY_F12; // terminal
- }
- }
- }
- if (extended[0] == 'O') {
- switch (extended[1]) {
- case 'P':
- return XKEY_F1;
- case 'Q':
- return XKEY_F2;
- case 'R':
- return XKEY_F3;
- case 'S':
- return XKEY_F4;
- case 't':
- return XKEY_F5; // syncterm
- }
- }
- // unknown -- This needs to be logged
- logf << "\r\nDEBUG:\r\nESC + ";
- for (unsigned int x = 0; x < pos; x++) {
- char z = extended[x];
- if (iscntrl(z)) {
- logf << (int)z << " ";
- } else {
- logf << "'" << (char)z << "'"
- << " ";
- };
- }
- logf << "\r\n";
- logf.flush();
- return XKEY_UNKNOWN;
- }
- return c;
- }
- int Door::get_input(void) {
- signed int c;
- c = getkey();
- if (c < 0)
- return 0;
- return c;
- }
- /*
- The following code will wait for 1.5 second:
- #include <sys/select.h>
- #include <sys/time.h>
- #include <unistd.h>`
- int main() {
- struct timeval t;
- t.tv_sec = 1;
- t.tv_usec = 500000;
- select(0, NULL, NULL, NULL, &t);
- }
- */
- /**
- * @brief Waits secs seconds for a keypress.
- *
- * returns key, or -1 on timeout (seconds passed).
- * -2 hangup
- * -3 out of time
- *
- * @param secs
- * @return signed int
- */
- signed int Door::sleep_key(int secs) {
- fd_set socket_set;
- struct timeval tv;
- int select_ret = -1;
- /*
- int recv_ret;
- char key;
- */
- if (hangup)
- return -2;
- if (time_left < 2)
- return -3;
- while (select_ret == -1) {
- FD_ZERO(&socket_set);
- FD_SET(STDIN_FILENO, &socket_set);
- tv.tv_sec = secs;
- tv.tv_usec = 0;
- select_ret = select(STDIN_FILENO + 1, &socket_set, NULL, NULL, &tv);
- // select(STDIN_FILENO + 1, &socket_set, NULL, NULL, bWait ? NULL : &tv);
- if (select_ret == -1) {
- if (errno == EINTR)
- continue;
- hangup = true;
- log() << "hangup detected" << std::endl;
- return (-2);
- }
- if (select_ret == 0)
- return (-1);
- }
- return getkey();
- }
- /**
- * @brief Input a string of requested max length.
- *
- * This first sends out max number of spaces, and max number of backspaces. This
- * will setup the input area. (If you set a background color of blue, this
- * would allow that to be seen by the user.)
- *
- * It handles input, backspaces / deleting the characters / enter input and
- * timeout/hangup/out of time.
- *
- * @param max
- * @return std::string
- */
- std::string Door::input_string(int max) {
- std::string input;
- // draw the input area.
- *this << std::string(max, ' ');
- *this << std::string(max, '\x08');
- int c;
- while (true) {
- c = sleep_key(inactivity);
- if (c < 0) {
- input.clear();
- return input;
- }
- if (c > 0x1000)
- continue;
- if (isprint(c)) {
- if (int(input.length()) < max) {
- *this << char(c);
- input.append(1, c);
- } else {
- // bell
- *this << '\x07';
- }
- } else {
- switch (c) {
- case 0x08:
- case 0x7f:
- if (input.length() > 0) {
- *this << "\x08 \x08";
- // this->flush();
- input.erase(input.length() - 1);
- };
- break;
- case 0x0d:
- return input;
- }
- }
- }
- }
- /**
- * @brief Get one of these keys
- *
- * returns char, or < 0 if timeout.
- *
- * @param keys
- * @return char or < 0
- */
- int Door::get_one_of(const char *keys) {
- int c;
- while (true) {
- c = sleep_key(inactivity);
- if (c < 0)
- return c;
- if (c > 0x1000)
- continue;
- const char *key = strchr(keys, (char)toupper(c));
- if (key != nullptr) {
- return *key;
- }
- *this << '\x07';
- }
- return c;
- }
- /**
- * Take given buffer and output it.
- *
- * If debug_capture is enabled, we save everything to debug_buffer.
- * This is used by the tests.
- *
- * @param s const char *
- * @param n std::streamsize
- * @return std::streamsize
- */
- std::streamsize Door::xsputn(const char *s, std::streamsize n) {
- if (debug_capture) {
- debug_buffer.append(s, n);
- } else {
- static std::string buffer;
- buffer.append(s, n);
- // setp(&(*buffer.begin()), &(*buffer.end()));
- if (!hangup) {
- std::cout << buffer;
- std::cout.flush();
- }
- // Tracking character position could be a problem / local terminal unicode.
- if (track)
- cx += n;
- buffer.clear();
- }
- return n;
- }
- /**
- * Stores a character into the buffer.
- * This does still use the buffer.
- * @todo Replace this also with a direct call to od_disp_emu.
- *
- * @param c char
- * @return int
- */
- int Door::overflow(int c) {
- if (debug_capture) {
- debug_buffer.append(1, (char)c);
- } else {
- if (!hangup) {
- std::cout << (char)c;
- std::cout.flush();
- }
- }
- if (track)
- cx++;
- // setp(&(*buffer.begin()), &(*buffer.end()));
- return c;
- }
- /**
- * Construct a new Color Output:: Color Output object
- * We default to BLACK/BLACK (not really a valid color),
- * pos=0 and len=0.
- */
- ColorOutput::ColorOutput() : c(COLOR::BLACK, COLOR::BLACK) {
- pos = 0;
- len = 0;
- }
- /**
- * Reset pos and len to 0.
- */
- void ColorOutput::reset(void) {
- pos = 0;
- len = 0;
- }
- /**
- * Construct a new Render:: Render object
- *
- * Render consists of constant text,
- * and vector of ColorOutput
- *
- * @param txt Text
- */
- Render::Render(const std::string txt) : text{txt} {}
- /**
- * Output the Render.
- *
- * This consists of going through the vector, getting
- * the fragment (from pos and len), and outputting the
- * color and fragment.
- *
- * @param os
- */
- void Render::output(std::ostream &os) {
- for (auto out : outputs) {
- std::string fragment = text.substr(out.pos, out.len);
- os << out.c << fragment;
- }
- }
- /**
- * Create render output.
- *
- * Call this for each section you want to colorize.
- *
- * @param color
- * @param len
- */
- void Render::append(ANSIColor color, int len) {
- if (outputs.empty()) {
- ColorOutput co;
- co.c = color;
- co.pos = 0;
- co.len = len;
- outputs.push_back(co);
- return;
- }
- ColorOutput ¤t = outputs.back();
- if (current.c == color) {
- current.len += len;
- return;
- }
- // Ok, new entry
- // beware the unicode text
- ColorOutput co;
- co.pos = current.pos + current.len;
- co.c = color;
- co.len = len;
- outputs.push_back(co);
- }
- /**
- * Construct a new Clrscr:: Clrscr object
- *
- * This is used to clear the screen.
- */
- Clrscr::Clrscr() {}
- /**
- * Clear the screen using ANSI codes.
- *
- * Not all systems home the cursor after clearing the screen.
- * We automatically home the cursor as well.
- *
- * @param os std::ostream&
- * @param clr const Clrscr&
- * @return std::ostream&
- */
- std::ostream &operator<<(std::ostream &os, const Clrscr &clr) {
- Door *d = dynamic_cast<Door *>(&os);
- if (d != nullptr) {
- d->track = false;
- *d << "\x1b[2J"
- "\x1b[H";
- d->cx = 1;
- d->cy = 1;
- d->track = true;
- } else {
- os << "\x1b[2J"
- "\x1b[H";
- }
- return os;
- }
- Clrscr cls;
- /**
- * This is used to issue NL+CR
- *
- */
- NewLine::NewLine() {}
- /**
- * Output Newline + CarriageReturn
- * @param os std::ostream
- * @param nl const NewLine
- * @return std::ostream&
- */
- std::ostream &operator<<(std::ostream &os, const NewLine &nl) {
- Door *d = dynamic_cast<Door *>(&os);
- if (d != nullptr) {
- d->track = false;
- *d << "\r\n";
- d->cx = 1;
- d->cy++;
- d->track = true;
- } else {
- os << "\r\n";
- };
- return os;
- }
- NewLine nl;
- /**
- * Construct a new Goto:: Goto object
- *
- * @param xpos
- * @param ypos
- */
- Goto::Goto(int xpos, int ypos) {
- x = xpos;
- y = ypos;
- }
- void Goto::set(int xpos, int ypos) {
- x = xpos;
- y = ypos;
- }
- /**
- * Output the ANSI codes to position the cursor to the given y,x position.
- *
- * @todo Optimize the ANSI goto string output.
- * @todo Update the Door object so it know where the cursor
- * is positioned.
- *
- * @param os std::ostream
- * @param g const Goto
- * @return std::ostream&
- */
- std::ostream &operator<<(std::ostream &os, const Goto &g) {
- Door *d = dynamic_cast<Door *>(&os);
- if (d != nullptr) {
- d->track = false;
- *d << "\x1b[";
- if (g.y > 1)
- *d << std::to_string(g.y);
- if (g.x > 1) {
- os << ";";
- *d << std::to_string(g.x);
- }
- *d << "H";
- d->cx = g.x;
- d->cy = g.y;
- d->track = true;
- } else {
- os << "\x1b[" << std::to_string(g.y) << ";" << std::to_string(g.x) << "H";
- };
- return os;
- }
- const char SaveCursor[] = "\x1b[s";
- const char RestoreCursor[] = "\x1b[u";
- // EXAMPLES
- /// BlueYellow Render example function
- renderFunction rBlueYellow = [](const std::string &txt) -> Render {
- Render r(txt);
- ANSIColor blue(COLOR::BLUE, ATTR::BOLD);
- ANSIColor cyan(COLOR::YELLOW, ATTR::BOLD);
- for (char const &c : txt) {
- if (isupper(c))
- r.append(blue);
- else
- r.append(cyan);
- }
- return r;
- };
- } // namespace door
|