123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850 |
- #ifndef DOOR_H
- #define DOOR_H
- #include "anyoption.h"
- #include <cstdint>
- #include <ctime>
- #include <fstream>
- #include <functional>
- #include <future>
- #include <iostream>
- #include <list>
- #include <memory>
- #include <ostream>
- #include <vector>
- // raw mode
- #include <termios.h>
- #include <unistd.h>
- #define CSI "\x1b["
- // getkey definitions
- #define XKEY_START 0x1000
- #define XKEY_UP_ARROW 0x1001
- #define XKEY_DOWN_ARROW 0x1002
- #define XKEY_RIGHT_ARROW 0x1003
- #define XKEY_LEFT_ARROW 0x1004
- #define XKEY_HOME 0x1010
- #define XKEY_END 0x1011
- #define XKEY_PGUP 0x1012
- #define XKEY_PGDN 0x1023
- #define XKEY_INSERT 0x1024
- #define XKEY_DELETE 0x7f
- #define XKEY_F1 0x1021
- #define XKEY_F2 0x1022
- #define XKEY_F3 0x1023
- #define XKEY_F4 0x1024
- #define XKEY_F5 0x1025
- #define XKEY_F6 0x1026
- #define XKEY_F7 0x1027
- #define XKEY_F8 0x1028
- #define XKEY_F9 0x1029
- #define XKEY_F10 0x102a
- #define XKEY_F11 0x102b
- #define XKEY_F12 0x102c
- #define XKEY_UNKNOWN 0x1111
- #define TIMEOUT -1
- #define HANGUP -2
- #define OUTOFTIME -3
- /**
- * @brief The BBS door project.
- * This is an attempt at writing a C++ BBS door toolkit.
- */
- namespace door {
- extern bool unicode;
- extern bool full_cp437;
- extern bool debug_capture;
- extern std::list<char> pushback;
- /*
- Translate CP437 strings to unicode for output.
- if (door::unicode) {
- // perform translation
- }
- */
- void cp437toUnicode(std::string input, std::string &out);
- void cp437toUnicode(const char *input, std::string &out);
- /*
- door 2.0
- */
- /**
- * ANSI Color codes
- */
- /**
- * @brief The colors available under ANSI-BBS
- */
- enum class COLOR : std::int8_t {
- /// BLACK (0)
- BLACK,
- /// RED (1)
- RED,
- /// GREEN (2)
- GREEN,
- /// BROWN (3)
- BROWN,
- /// YELLOW (3)
- YELLOW = 3,
- /// BLUE (4)
- BLUE,
- /// MAGENTA (5)
- MAGENTA,
- /// CYAN (6)
- CYAN,
- /// WHITE (7)
- WHITE
- };
- /**
- * @brief ANSI-BBS text attributes
- */
- enum class ATTR : std::int8_t {
- /// RESET forces all attributes (and Colors) to be sent.
- RESET,
- /// BOLD is the same as BRIGHT.
- BOLD,
- /// BRIGHT is the same as BOLD.
- BRIGHT = 1,
- /// SLOW BLINK
- BLINK = 5,
- /// INVERSE is Background on Foreground.
- INVERSE = 7
- };
- /**
- * @class ANSIColor
- * This holds foreground, background and ANSI-BBS attribute
- * information.
- * The special attribute RESET forces attribute and color
- * output always.
- *
- * @brief Foreground, Background and Attributes
- *
- */
- class ANSIColor {
- /** Foreground color */
- COLOR fg;
- /** Background color */
- COLOR bg;
- // Track attributes (ATTR)
- /** reset flag / always send color and attributes */
- unsigned int reset : 1;
- /** bold / bright flag */
- unsigned int bold : 1;
- /** blink slow blinking text */
- unsigned int blink : 1;
- /** inverse */
- unsigned int inverse : 1;
- public:
- ANSIColor();
- ANSIColor(ATTR a);
- ANSIColor(COLOR f);
- ANSIColor(COLOR f, ATTR a);
- ANSIColor(COLOR f, ATTR a1, ATTR a2);
- ANSIColor(COLOR f, COLOR b);
- ANSIColor(COLOR f, COLOR b, ATTR a);
- ANSIColor(COLOR f, COLOR b, ATTR a1, ATTR a2);
- ANSIColor &Attr(ATTR a);
- bool operator==(const ANSIColor &c) const;
- bool operator!=(const ANSIColor &c) const;
- void setFg(COLOR f);
- void setFg(COLOR f, ATTR a);
- void setBg(COLOR b);
- /**
- * Get the foreground color
- * @return COLOR
- */
- COLOR getFg() { return fg; };
- /**
- * Get the background color
- * @return COLOR
- */
- COLOR getBg() { return bg; };
- void attr(ATTR a);
- std::string output(void) const;
- std::string debug(void);
- std::string output(ANSIColor &previous) const;
- friend std::ostream &operator<<(std::ostream &os, const ANSIColor &c);
- };
- /**
- * @class Door
- *
- * This handles output to the caller, via ostream.
- */
- class Door : public std::ostream, private std::streambuf {
- private:
- std::streamsize xsputn(const char *s, std::streamsize n) override;
- int overflow(int c) override;
- /** The name used for logfile */
- std::string doorname;
- void parse_dropfile(const char *filepath);
- void init(void);
- std::time_t startup;
- /** Initial terminal defaults. */
- struct termios tio_default;
- signed int getch(void);
- signed int getkey_or_pushback(void);
- /** Did we read a dropfile? */
- bool has_dropfile;
- bool debugging;
- /** Name of the dropfile. */
- std::string dropfilename;
- /** Contents of the dropfile. */
- vector<std::string> dropfilelines;
- /** Logfile */
- ofstream logf;
- void detect_unicode_and_screen(void);
- /** Allow us to stop the time_thread. */
- std::promise<void> stop_thread;
- /** Used by time_thread to know when a minute has passed. */
- int seconds_elapsed;
- void time_thread_run(std::future<void> future);
- /** Thread used to update time_left and time_used. */
- std::thread time_thread;
- public:
- Door(std::string dname, int argc, char *argv[]);
- Door(Door &) = delete;
- virtual ~Door();
- ofstream &log(void);
- /** Commandline options parser. */
- AnyOption opt;
- /** Buffer that holds the output for testing. */
- std::string debug_buffer;
- /**
- * Previous ANSI-BBS colors and attributes sent.
- * This is used to optimize ANSI Color output.
- * \see ANSIColor::output()
- */
- ANSIColor previous;
- /** \todo Enable tracking cursor position. */
- bool track;
- /** \todo Current cursor X position. */
- int cx;
- /** \todo Current cursor Y position. */
- int cy;
- /** Detected screen width. \ref Door::detect_unicode_and_screen */
- int width;
- /** Detected screen height. */
- int height;
- /**
- * @brief Number of seconds before timing out.
- *
- * When prompting for user input, this is the number of seconds they have to
- * respond before we give up and timeout on them. (Default 120/2 minutes)
- */
- int inactivity;
- /** BBS Dropfile username */
- std::string username;
- /** BBS Dropfile handle */
- std::string handle;
- /** BBS Dropfile location */
- std::string location;
- /** BBS Dropfile sysop name */
- std::string sysop;
- // std::string bbsname;
- /** BBS Dropfile node number */
- int node;
- /** time left in minutes */
- atomic<int> time_left;
- /** time used in minutes */
- atomic<int> time_used;
- signed int getkey(void);
- bool haskey(void);
- signed int sleep_key(int secs);
- signed int sleep_ms_key(int msecs);
- std::string input_string(int max);
- int get_one_of(const char *keys);
- };
- // Use this to define the deprecated colorizer [POC]
- // typedef std::function<void(Door &, std::string &)> colorFunction;
- /**
- * @class ColorOutput
- * This works with \ref Render to create the output. This consists
- * of ANSIColor and text position + length.
- *
- * @brief This holds an ANSIColor and text position + length
- *
- */
- class ColorOutput {
- public:
- ColorOutput();
- void reset(void);
- /// Color to use for this fragment
- ANSIColor c;
- /// Starting position of Render.text
- int pos;
- /// Length
- int len;
- };
- /*
- No, don't do this.
- Instead, return an iterator/generator.
- */
- /**
- * @class Render
- * This holds the string, and a vector that contains ColorOutput parts.
- *
- * @see Render::output()
- *
- * @brief Rendering a string with ANSIColor
- *
- */
- class Render {
- /// Complete text to be rendered.
- std::string text;
- public:
- Render(const std::string txt);
- /// Vector of ColorOutput object.
- std::vector<ColorOutput> outputs;
- void append(ANSIColor color, int len = 1);
- void output(std::ostream &os);
- };
- /**
- * This defines the render output function. Given the line text, we output the
- * color codes needs to display the line.
- *
- * ~~~{.cpp}
- * door::ANSIColor upperColor, lowerColor;
- *
- * door::RenderFunction render = [upperColor, lowerColor]
- * (const std::string &text) -> door::Render {
- * door::Render r(text);
- * for (char const &c : text) {
- * if (std::isupper(c))
- * r.append(upperColor);
- * else
- * r.append(lowerColor);
- * }
- * return r;
- * };
- * ~~~
- *
- * @brief Render output function
- *
- */
- typedef std::function<Render(const std::string &)> renderFunction;
- /**
- * This defines the update function.
- *
- * This updates the text.
- *
- * ~~~{.cpp}
- * int score = 0;
- *
- * door::updateFunction updater = [](void) -> std::string {
- * std::string text = "Score: ";
- * text += std::to_string(score);
- * return text;
- * };
- *
- * fancyLine.setUpdater(updater);
- * ~~~
- */
- typedef std::function<std::string(void)> updateFunction;
- /**
- * @class Clrscr
- * Clear the screen
- * @brief Clear the screen
- */
- class Clrscr {
- public:
- Clrscr(void);
- friend std::ostream &operator<<(std::ostream &os, const Clrscr &clr);
- };
- /**
- * Clear the BBS terminal.
- *
- */
- extern Clrscr cls;
- /**
- * @class NewLine
- * Carriage return + Newline
- * @brief CR+LF
- */
- class NewLine {
- public:
- NewLine(void);
- friend std::ostream &operator<<(std::ostream &os, const NewLine &nl);
- };
- /**
- * CRLF
- */
- extern NewLine nl;
- /**
- * This resets the colors to normal state.
- *
- * @brief reset colors to normal
- */
- extern ANSIColor reset;
- /// @deprecated Not used
- enum class Justify { NONE, LEFT, RIGHT, CENTER };
- /**
- * @class Goto
- * This handles outputting ANSI codes to position the cursor on the screen.
- *
- * @brief ANSI Goto X, Y position
- */
- class Goto {
- /// X-Position
- int x;
- /// Y-Position
- int y;
- public:
- Goto(int xpos, int ypos);
- /**
- * Default Goto constructor copier
- */
- Goto(const Goto &) = default;
- void set(int xpos, int ypos);
- friend std::ostream &operator<<(std::ostream &os, const Goto &g);
- };
- extern const char SaveCursor[];
- extern const char RestoreCursor[];
- #ifdef EXPERIMENTAL
- /* should we try to derive a base class, so you can have multilines of
- * multilines? */
- class LineBase {
- public:
- virtual ~LineBase() = default;
- virtual bool update(void) = 0;
- // friend std::ostream &operator<<(std::ostream &os, const LineBase &lb) = 0;
- };
- class BasicLine {
- protected:
- std::string text;
- bool hasColor;
- ANSIColor color;
- /// renderFunction to use when rendering Line.
- renderFunction render;
- /// updateFunction to use when updating.
- updateFunction updater;
- public:
- BasicLine(std::string txt);
- BasicLine(std::string txt, ANSIColor c);
- BasicLine(const BasicLine &rhs) = default;
- virtual ~BasicLine() = default;
- bool hasRender(void);
- void setText(std::string txt);
- void setColor(ANSIColor c);
- void setRender(renderFunction rf);
- void setUpdater(updateFunction uf);
- bool update(void);
- friend std::ostream &operator<<(std::ostream &os, const BasicLine &l);
- };
- class MultiLine {
- protected:
- std::vector<std::shared_ptr<BasicLine>> lines;
- public:
- MultiLine();
- void append(std::shared_ptr<BasicLine> bl);
- bool update(void);
- friend std::ostream &operator<<(std::ostream &os, const MultiLine &l);
- };
- #endif
- /**
- * @class Line
- * This holds text and ANSIColor information, and knows how to
- * send them out to the Door.
- * @brief Text and ANSIColor
- */
- class Line {
- protected:
- /// Text of the line
- std::string text;
- /// Do we have color?
- bool hasColor;
- /// Line color
- ANSIColor color;
- /// Padding characters
- std::string padding;
- /// Padding color
- ANSIColor paddingColor;
- /// renderFunction to use when rendering Line.
- renderFunction render;
- /// updateFunction to use when updating.
- updateFunction updater;
- int width;
- /**
- * @param width int
- */
- // void makeWidth(int width);
- public:
- Line(const std::string &txt, int width = 0);
- Line(const char *txt, int width = 0);
- Line(const std::string &txt, int width, ANSIColor c);
- Line(const char *txt, int width, ANSIColor c);
- Line(const std::string &txt, int width, renderFunction rf);
- Line(const char *txt, int width, renderFunction rf);
- Line(const Line &rhs);
- Line(Line &&rhs);
- // ~Line();
- bool hasRender(void);
- int length(void); // const;
- void fit(void);
- /**
- * @param padstring std::string &
- * @param padColor ANSIColor
- */
- void setPadding(std::string &padstring, ANSIColor padColor);
- /**
- * @param padstring const char *
- * @param padColor ANSIColor
- */
- void setPadding(const char *padstring, ANSIColor padcolor);
- void setText(std::string &txt);
- void setText(const char *txt);
- const char *getText(void) { return text.c_str(); };
- void setColor(ANSIColor c);
- void setRender(renderFunction rf);
- void setUpdater(updateFunction uf);
- bool update(void);
- std::string debug(void);
- /**
- * @todo This might be a problem, because const Line wouldn't
- * allow me to track "updates". I.E. I send the line, I'd
- * need to change the line's State to "nothing changed".
- * Then, if something did change, the next update request would
- * be able to know that yes, this does indeed need to be sent.
- *
- * @bug This also might cause problems if I display a shared
- * BasicLine (in multiple places), and then update it. It
- * would only update in the first place (the others wouldn't
- * show it needs an update).
- */
- friend std::ostream &operator<<(std::ostream &os, const Line &l);
- };
- /// Example BlueYellow renderFunction
- extern renderFunction rBlueYellow;
- /*
- Progress Bar Styles:
- solid chars.
- half step chars.
- gradient (1/4 %25, %50, %75) chars.
- percentage (solid chars, percentage is XX% or 100)
- percent_space (solid chars, percentage is " XX% " or " 100 ")
- */
- enum class BarStyle { SOLID, HALF_STEP, GRADIENT, PERCENTAGE, PERCENT_SPACE };
- /**
- * BarColorRange
- *
- * vector<door::BarColorRange> colorRange = {
- * {2500, door::ANSIColor(door::COLOR::RED)},
- * {5000, door::ANSIColor(door::COLOR::BROWN)},
- * {7500, door::ANSIColor(door::COLOR::YELLOW, door::ATTR::BOLD)},
- * {9500, door::ANSIColor(door::COLOR::GREEN)},
- * {10100, door::ANSIColor(door::COLOR::GREEN, door::ATTR::BOLD)}};
- * BarLine.setColorRange(colorRange);
- *
- */
- struct BarColorRange {
- unsigned long percent;
- ANSIColor c;
- };
- class BarLine : public Line {
- protected:
- BarStyle barstyle;
- unsigned long current_percent;
- void init(void);
- std::string update_bar(void);
- int length;
- vector<BarColorRange> colorRange;
- public:
- BarLine(int width);
- BarLine(int width, BarStyle style);
- BarLine(int width, BarStyle style, ANSIColor c);
-
- void watch(float &percent);
- void watch(int &value, int &max);
-
- void setStyle(BarStyle s);
- void set(int value, int max);
- void set(float percent);
- void set(unsigned long percent);
- void setColorRange(vector<BarColorRange> bcr);
- // friend std::ostream &operator<<(std::ostream &os, const BarLine &b);
- };
- /**
- * The different Borders supported by Panel.
- *
- */
- enum class BorderStyle {
- /// NONE (0)
- NONE,
- /// SINGLE (1)
- SINGLE,
- /// DOUBLE (2)
- DOUBLE,
- /// SINGLE top DOUBLE side (3)
- SINGLE_DOUBLE,
- /// DOUBLE top SINGLE side (4)
- DOUBLE_SINGLE,
- /// BLANK (5)
- BLANK
- };
- class Panel {
- protected:
- int x;
- int y;
- int width; // or padding ?
- BorderStyle border_style;
- ANSIColor border_color;
- /**
- * @todo Fix this to use shared_ptr.
- * I don't think unique_ptr is the right way to go with this. I want to reuse
- * things, and that means shared_ptr!
- *
- */
- std::vector<std::unique_ptr<Line>> lines;
- bool hidden;
- // when you show panel, should it mark it as
- // redisplay everything?? maybe??
- bool shown_once; // ?? maybe shown_once_already ?
- std::unique_ptr<Line> title;
- int offset;
- public:
- Panel(int x, int y, int width);
- Panel(int width);
- // Panel(const Panel &);
- Panel(Panel &) = delete; // default;
- Panel(Panel &&ref);
- void set(int x, int y);
- void get(int &x, int &y) {
- x = this->x;
- y = this->y;
- };
- void setTitle(std::unique_ptr<Line> T, int off = 1);
- void setStyle(BorderStyle bs);
- void setColor(ANSIColor c);
- int getWidth(void) { return width; };
- int getHeight(void) {
- if (border_style == BorderStyle::NONE)
- return lines.size();
- else
- return lines.size() + 2;
- };
- void hide(void);
- void show(void);
- void addLine(std::unique_ptr<Line> l);
- // bool delLine(std::shared_ptr<Line> l); // ?
- /*
- void display(void);
- void update(void);
- */
- /**
- * @brief Updates a panel.
- *
- * returns True if something was changed (and cursor has moved)
- * False, nothing to do, cursor is ok.
- *
- * @param d
- * @return true
- * @return false
- */
- bool update(Door &d);
- void update(Door &d, int line);
- void update(void);
- door::Goto gotoEnd(void);
- std::unique_ptr<Line> spacer_line(bool single);
- void lineSetBack(ANSIColor back);
- friend std::ostream &operator<<(std::ostream &os, const Panel &p);
- };
- /*
- Menu - defaults to double lines.
- Has colorize for selected item / non-selected.
- Arrow keys + ENTER, or keypress to select an item.
- [O] Option Displayed Here
- [ + ] = c1
- O = c2
- Remaining UC TEXT = c3
- Remaining LC text = c4
- // Colors for CS and CU (color selected, color unselected)
- */
- class Menu : public Panel {
- protected:
- unsigned int chosen;
- std::vector<char> options;
- renderFunction selectedRender;
- renderFunction unselectedRender;
- /*
- std::function<void(Door &d, std::string &)> selectedColorizer;
- std::function<void(Door &d, std::string &)> unselectedColorizer;
- */
- public:
- static renderFunction defaultSelectedRender;
- static renderFunction defaultUnselectedRender;
- /*
- static std::function<void(Door &d, std::string &)> defaultSelectedColorizer;
- static std::function<void(Door &d, std::string &)> defaultUnselectedColorizer;
- */
- Menu(int x, int y, int width);
- Menu(int width);
- // Menu(const Menu &);
- Menu(const Menu &) = delete;
- Menu(Menu &&);
- void addSelection(char c, const char *line);
- void addSelection(char c, const char *line, updateFunction update);
- void defaultSelection(int d);
- void setRender(bool selected, renderFunction render);
- int choose(Door &door);
- char which(int d);
- static renderFunction makeRender(ANSIColor c1, ANSIColor c2, ANSIColor c3,
- ANSIColor c4);
- };
- class Screen {
- protected:
- // bool hidden;
- /**
- * @brief vector of panels.
- */
- std::vector<std::unique_ptr<Panel>> panels;
- public:
- Screen(void);
- Screen(Screen &) = default;
- void addPanel(std::unique_ptr<Panel> p);
- /*
- bool delPanel(std::shared_ptr<Panel> p);
- void hide(void);
- void show(void);
- */
- bool update(Door &d);
- void update(void);
- friend std::ostream &operator<<(std::ostream &os, const Screen &s);
- };
- /*
- screen - contains panels.
- - default to 1,1 X 80,24
- - refresh(style) could redraw panels by order they were added,
- or could redraw panels from top to bottom, left to right.
- crazy ideas:
- hide panels / z-order
- how to handle panel on top of other panels?
- Can I have you win + show animated final score calculations?
- panel - has X,Y and width, optional length. contains lines.
- length could be simply number of "lines".
- - has optional border. double/single/Ds/Sd TOPbottom
- - has optional title.
- - has optional footer.
- addLine()
- append() - Appends another line to current line.
- set(X,Y) - set a "line" at a given X,Y position.
- menu - another type of panel, contains menu options/lines.
- lightmenu - like above, but allows arrow keys to select menu options.
- line - contains text.
- (Maybe a "dirty" flag is needed here?)
- - has optional (width)
- - has optional (justify - L, R, Center)
- - has optional padding (# of blank chars)
- - has color (of text)
- - has formatter/coloring function (to colorize the text)
- Example would be one that sets capital letters to one color, lower to another.
- Another example would be one that displays Score: XXX, where Score is one
- color, : is another, and XXX is yet another. Properly padded, of course.
- - has "lambda" function to update the value? (Maybe?)
- Idea would be that I could update the score, and panel.update(). It would
- call all the line.update() functions and only update anything that has
- changed.
- Crazy ideas:
- Can I delete a line, and have it automatically removed from a panel?
- lightline - text, changes format/coloring if focus/nofocus is set?
- */
- } // namespace door
- #endif
|