123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936 |
- #include "door.h"
- #include <set>
- #include <string.h>
- namespace door {
- Panel::Panel(int xp, int yp, int panelWidth) : border_color() {
- x = xp;
- y = yp;
- width = panelWidth;
- hidden = false;
- border_style = BorderStyle::NONE;
-
- }
- Panel::Panel(int panelWidth) : border_color() {
- x = 0;
- y = 0;
- width = panelWidth;
- hidden = false;
- border_style = BorderStyle::NONE;
- }
- Panel::Panel(Panel &&ref) {
- x = ref.x;
- y = ref.y;
- width = ref.width;
- hidden = ref.hidden;
- border_style = ref.border_style;
- title = std::move(ref.title);
- offset = ref.offset;
- lines = std::move(ref.lines);
- }
- void Panel::set(int xp, int yp) {
- x = xp;
- y = yp;
- }
- void Panel::setTitle(std::unique_ptr<Line> t, int off) {
- title = std::move(t);
- offset = off;
- }
- void Panel::setStyle(BorderStyle bs) { border_style = bs; }
- void Panel::setColor(ANSIColor c) { border_color = c; }
- void Panel::hide(void) { hidden = true; }
- void Panel::show(void) { hidden = false; }
- void Panel::addLine(std::unique_ptr<Line> l) {
- l->fit();
- lines.push_back(std::move(l));
- }
- struct box_styles {
-
- const char *tl;
-
- const char *tr;
-
- const char *top;
-
- const char *side;
-
- const char *bl;
-
- const char *br;
-
- const char *ml;
-
- const char *mr;
- };
- struct box_styles UBOXES[] = {{"\u250c", "\u2510", "\u2500", "\u2502", "\u2514",
- "\u2518", "\u251c", "\u2524"},
- {"\u2554", "\u2557", "\u2550", "\u2551", "\u255a",
- "\u255d", "\u2560", "\u2563"},
- {"\u2553", "\u2556", "\u2500", "\u2551", "\u2559",
- "\u255c", "\u255f", "\u2562"},
- {"\u2552", "\u2555", "\u2550", "\u2502", "\u2558",
- "\u255b", "\u255e", "\u2561"}};
- struct box_styles BOXES[] = {
-
- {
- "\xda",
- "\xbf",
- "\xc4",
- "\xb3",
- "\xc0",
- "\xd9",
- "\xc3",
- "\xb4",
- },
-
- {
- "\xc9",
- "\xbb",
- "\xcd",
- "\xba",
- "\xc8",
- "\xbc",
- "\xcc",
- "\xb9",
- },
-
- {
- "\xd6",
- "\xb7",
- "\xc4",
- "\xba",
- "\xd3",
- "\xbd",
- "\xc7",
- "\xb6",
- },
-
- {
- "\xd5",
- "\xb8",
- "\xcd",
- "\xb3",
- "\xd4",
- "\xbe",
- "\xc6",
- "\xb5",
- },
- };
- const char *JOIN[2][2][2] = {{
- {"\xc3", "\xb4"},
- {"\xc6", "\xb5"}
- },
- {
- {"\xc7", "\xb6"},
- {"\xcc", "\xb9"},
- }};
- const char *UJOIN[2][2][2] = {{
- {"\u251c", "\u2524"},
- {"\u255e", "\u2561"}
- },
- {
- {"\u255f", "\u2562"},
- {"\u2560", "\u2563"},
- }};
- bool Panel::update(Door &d) {
- int row = y;
- int style = (int)border_style;
- if (style > 0)
- ++row;
- bool updated = false;
- for (auto &line : lines) {
- if (line->update()) {
-
- updated = true;
- int col = x;
- if (style > 0)
- ++col;
- d << door::Goto(col, row);
- d << *line;
- }
- ++row;
- }
- return updated;
- }
- void Panel::update(Door &d, int line) {
- int row = y;
- int style = (int)border_style;
- if (style > 0)
- ++row;
-
- auto &l = lines[line];
- row += line;
- int col = x;
- if (style > 0)
- ++col;
- d << door::Goto(col, row);
- d << *l;
- }
- void Panel::update(void) {
- for (auto &line : lines) {
- line->update();
- }
- }
- door::Goto Panel::gotoEnd(void) {
- int row = y;
- int style = (int)border_style;
- if (style > 0)
- ++row;
- row += lines.size();
- int col = x;
- if (style > 0)
- col += 2;
- col += width;
- return door::Goto(col, row);
- }
- void Panel::lineSetBack(ANSIColor back) {
- for (auto &line : lines) {
- line->setColor(back);
- }
- }
- std::unique_ptr<Line> Panel::spacer_line(bool single) {
- std::string spacer_text;
- if (door::unicode) {
- for (int x = 0; x < width; ++x) {
- if (single)
- spacer_text.append(UBOXES[0].top);
- else
- spacer_text.append(UBOXES[1].top);
- }
- } else {
- if (single)
- spacer_text = std::string(width, BOXES[0].top[0]);
- else
- spacer_text = std::string(width, BOXES[1].top[0]);
- }
- std::unique_ptr<Line> line = make_unique<Line>(spacer_text, width);
- return line;
- }
- std::ostream &operator<<(std::ostream &os, const Panel &p) {
- if (p.hidden)
- return os;
-
- int style = (int)p.border_style;
- struct box_styles s;
-
-
-
- if (style > 0) {
-
- if (style < 5) {
- if (unicode)
- s = UBOXES[style - 1];
- else
- s = BOXES[style - 1];
- } else {
- s.bl = s.br = s.mr = s.ml = " ";
- s.top = s.side = " ";
- s.tl = s.tr = " ";
- }
- }
-
-
-
- int row = p.y;
- if (style > 0) {
-
- os << door::Goto(p.x, row);
- os << p.border_color << s.tl;
- if (p.title) {
- for (int c = 0; c < p.offset; c++)
- os << s.top;
- os << *(p.title);
- os << p.border_color;
- int left = p.width - (p.offset + (p.title)->length());
- if (left > 0) {
- for (int c = 0; c < left; c++)
- os << s.top;
- };
- os << s.tr;
- } else {
- for (int c = 0; c < p.width; c++)
- os << s.top;
- os << s.tr;
- };
-
- ++row;
- };
- for (auto &line : p.lines) {
- os << door::Goto(p.x, row);
- bool join = false;
- int line_is = -1;
- int border_is = -1;
-
- {
- const char *line_text = line->getText();
- if (door::unicode) {
- if (strncmp(UBOXES[0].top, line_text, strlen(UBOXES[0].top)) == 0) {
- join = true;
- line_is = 0;
- }
- if (strncmp(UBOXES[1].top, line_text, strlen(UBOXES[1].top)) == 0) {
- join = true;
- line_is = 1;
- }
- } else {
- if (BOXES[0].top[0] == line_text[0]) {
- join = true;
- line_is = 0;
- }
- if (BOXES[1].top[0] == line_text[0]) {
- join = true;
- line_is = 1;
- }
- }
- }
- if (style > 0) {
- if (join) {
- switch (p.border_style) {
- case door::BorderStyle::SINGLE:
- case door::BorderStyle::DOUBLE_SINGLE:
- border_is = 0;
- break;
- case door::BorderStyle::DOUBLE:
- case door::BorderStyle::SINGLE_DOUBLE:
- border_is = 1;
- break;
- default:
- break;
- }
- os << p.border_color;
- if (door::unicode)
- os << UJOIN[border_is][line_is][0];
- else
- os << JOIN[border_is][line_is][0];
- } else {
- os << p.border_color << s.side;
- };
- };
-
- os << *line;
- if (style > 0) {
- if (join) {
- os << p.border_color;
- if (door::unicode)
- os << UJOIN[border_is][line_is][1];
- else
- os << JOIN[border_is][line_is][1];
- } else
- os << p.border_color << s.side;
- };
-
- row++;
-
-
- }
-
- if (style > 0) {
- os << door::Goto(p.x, row);
- os << p.border_color << s.bl;
- for (int c = 0; c < p.width; c++)
- os << s.top;
-
- os << s.br;
- };
-
-
- return os;
- }
- renderFunction Menu::defaultSelectedRender = Menu::makeRender(
- ANSIColor(COLOR::BLUE, COLOR::WHITE), ANSIColor(COLOR::BLUE, COLOR::WHITE),
- ANSIColor(COLOR::BLUE, COLOR::WHITE), ANSIColor(COLOR::BLUE, COLOR::WHITE));
- renderFunction Menu::defaultUnselectedRender =
- Menu::makeRender(ANSIColor(COLOR::WHITE, COLOR::BLUE, ATTR::BOLD),
- ANSIColor(COLOR::WHITE, COLOR::BLUE, ATTR::BOLD),
- ANSIColor(COLOR::WHITE, COLOR::BLUE, ATTR::BOLD),
- ANSIColor(COLOR::YELLOW, COLOR::BLUE, ATTR::BOLD));
- Menu::Menu(int x, int y, int width) : Panel(x, y, width) {
- setStyle(BorderStyle::DOUBLE);
-
-
- setRender(true, defaultSelectedRender);
-
- setRender(false, defaultUnselectedRender);
-
-
- chosen = 0;
- }
- Menu::Menu(int width) : Panel(width) {
- setStyle(BorderStyle::DOUBLE);
- setRender(true, defaultSelectedRender);
- setRender(false, defaultUnselectedRender);
- chosen = 0;
- }
- Menu::Menu(Menu &&ref) : Panel(ref.x, ref.y, ref.width) {
- x = ref.x;
- y = ref.y;
- width = ref.width;
- border_style = ref.border_style;
- setRender(true, ref.selectedRender);
- setRender(false, ref.unselectedRender);
- options = ref.options;
- lines = std::move(ref.lines);
- chosen = ref.chosen;
- }
- void Menu::addSelection(char c, const char *line) {
- std::string menuline;
- menuline.reserve(5 + strlen(line));
- menuline = "[ ] ";
- menuline[1] = c;
- menuline += line;
-
-
- addLine(std::make_unique<Line>(menuline, width));
- options.push_back(c);
- }
- void Menu::addSelection(char c, const char *line, updateFunction update) {
- std::string menuline;
- menuline.reserve(5 + strlen(line));
- menuline = "[ ] ";
- menuline[1] = c;
- updateFunction fullUpdate = [menuline, update](void) -> std::string {
- std::string text = menuline;
- text += update();
- return text;
- };
- menuline += line;
- std::unique_ptr<Line> l = std::make_unique<Line>(menuline, width);
- l->setUpdater(fullUpdate);
-
- addLine(std::move(l));
- options.push_back(c);
- }
- void Menu::defaultSelection(int d) { chosen = d; }
- char Menu::which(int d) { return options[d]; }
- void Menu::setRender(bool selected, renderFunction render) {
- if (selected)
- selectedRender = render;
- else
- unselectedRender = render;
- }
- renderFunction Menu::makeRender(ANSIColor c1, ANSIColor c2, ANSIColor c3,
- ANSIColor c4) {
- renderFunction render = [c1, c2, c3, c4](const std::string &txt) -> Render {
- Render r(txt);
- bool option = true;
- for (char const &c : txt) {
- if (option) {
- if (c == '[' or c == ']') {
- r.append(c1);
- option = (c == '[');
- } else {
- r.append(c2);
- }
- } else {
- if (isupper(c))
- r.append(c3);
- else
- r.append(c4);
- }
- }
- return r;
- };
- return render;
- }
- int Menu::choose(Door &door) {
-
-
- bool updated = true;
- bool update_and_exit = false;
- std::set<int> changed;
- while (true) {
- if (updated) {
- for (unsigned int x = 0; x < lines.size(); ++x) {
- if (x == chosen) {
- lines[x]->setRender(
- selectedRender);
- } else {
- lines[x]->setRender(
- unselectedRender);
- }
- }
-
- if (changed.empty())
- door << *this;
- else {
-
- for (auto si : changed) {
-
- update(door, si);
- }
-
-
-
- door << gotoEnd();
- }
-
-
- };
- if (update_and_exit)
- return chosen + 1;
-
-
- updated = false;
-
- int event = door.sleep_key(door.inactivity);
- if (event < 0) {
-
- return event;
- }
- unsigned int previous_choice = chosen;
- changed.clear();
- bool use_numberpad = true;
-
- for (const char &c : options) {
- if ((c == '8') or (c == '2'))
- use_numberpad = false;
- }
- switch (event) {
- case '8':
- if (!use_numberpad)
- break;
- case XKEY_UP_ARROW:
- if (chosen > 0) {
- chosen--;
- updated = true;
- }
- break;
- case '2':
- if (!use_numberpad)
- break;
- case XKEY_DOWN_ARROW:
- if (chosen < lines.size() - 1) {
- chosen++;
- updated = true;
- }
- break;
- case XKEY_HOME:
- if (chosen != 0) {
- chosen = 0;
- updated = true;
- }
- break;
- case XKEY_END:
- if (chosen != lines.size() - 1) {
- chosen = lines.size() - 1;
- updated = true;
- }
- }
- if (event == 0x0d) {
-
- return chosen + 1;
- }
- for (unsigned int x = 0; x < lines.size(); x++) {
- if (toupper(options[x]) == toupper(event)) {
-
- if (chosen == x) {
- return x + 1;
- }
-
-
- updated = true;
- chosen = x;
- update_and_exit = true;
- }
- }
- if (previous_choice != chosen) {
- changed.insert(previous_choice);
- changed.insert(chosen);
- }
- }
- return 0;
- }
- Screen::Screen(){};
- void Screen::addPanel(std::unique_ptr<Panel> p) {
- panels.push_back(std::move(p));
- }
- bool Screen::update(Door &d) {
- bool updated = false;
- for (auto &panel : panels) {
- if (panel->update(d))
- updated = true;
- }
- return updated;
- }
- void Screen::update(void) {
- for (auto &panel : panels) {
- panel->update();
- }
- }
- std::ostream &operator<<(std::ostream &os, const Screen &s) {
-
- for (auto &panel : s.panels) {
- os << *panel;
- };
-
-
- return os;
- }
- }
|