| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377 | #include "door.h"#include "utf8.h"/** * @file * @brief Line */namespace door {#ifdef EXPERIMENTALBasicLine::BasicLine(std::string txt) : text{txt}, hasColor{false} {}BasicLine::BasicLine(std::string txt, ANSIColor c)    : text{txt}, hasColor{true}, color{c} {}bool BasicLine::hasRender(void) {  if (render)    return true;  return false;}void BasicLine::setText(std::string txt) { text = txt; }void BasicLine::setColor(ANSIColor c) {  color = c;  hasColor = true;}void BasicLine::setRender(renderFunction rf) { render = rf; }void BasicLine::setUpdater(updateFunction uf) { updater = uf; }/** * Update BasicLine, if we have an updater. * * If we have an updater, call it.  If the text is different, * update setText() and return true. * Otherwise false. * * This doesn't detect changes (like if the render has been changed, for * example) * * @return bool */bool BasicLine::update(void) {  if (updater) {    std::string temp = updater();    if (temp == text)      return false;    setText(temp);    return true;  }  return false;}/** * Output Line * * This looks for padding and paddingColor. * This uses the render function if set. * * @param os std::ostream * @param l const BasicLine & * @return std::ostream& */std::ostream &operator<<(std::ostream &os, const BasicLine &l) {  if (l.render) {    // This has a renderer.  Use it.    Render r = l.render(l.text);    r.output(os);  } else {    if (l.hasColor) {      os << l.color;    };    os << l.text;  }  return os;}MultiLine::MultiLine(){};void MultiLine::append(std::shared_ptr<BasicLine> bl) { lines.push_back(bl); }bool MultiLine::update() {  bool updated = false;  for (auto line : lines) {    if (line->update())      updated = true;  }  return updated;}/** * Output Line * * This looks for padding and paddingColor. * This uses the render function if set. * * @param os std::ostream * @param ml const MultiLine & * @return std::ostream& */std::ostream &operator<<(std::ostream &os, const MultiLine &ml) {  for (auto line : ml.lines) {    os << *line;  }  return os;}#endif/** * Construct a new Line:: Line object with * string and total width. * * @param txt std::string * @param width int */Line::Line(const std::string &txt, int w) : text{txt}, width{w} {  hasColor = false;}Line::Line(const std::string &txt, int w, ANSIColor c)    : text{txt}, color{c}, width{w} {  hasColor = true;}Line::Line(const char *txt, int w, ANSIColor c)    : text{txt}, color{c}, width{w} {  hasColor = true;}Line::Line(const std::string &txt, int w, renderFunction rf)    : text{txt}, render{rf}, width{w} {  hasColor = false;}Line::Line(const char *txt, int w, renderFunction rf)    : text{txt}, render{rf}, width{w} {  hasColor = false;}/** * Construct a new Line:: Line object with * const char * and total width * * @param txt const char * * @param width int */Line::Line(const char *txt, int w) : text{txt}, width{w} { hasColor = false; }/** * Construct a new Line:: Line object from an * existing Line * * @param rhs const Line& */Line::Line(const Line &rhs)    : text{rhs.text}, hasColor{rhs.hasColor}, color{rhs.color},      padding{rhs.padding}, paddingColor{rhs.paddingColor} {  if (rhs.render) {    render = rhs.render;  }  if (rhs.updater) {    updater = rhs.updater;  }  width = rhs.width;}Line::Line(Line &&rhs)    : text{rhs.text}, hasColor{rhs.hasColor}, color{rhs.color},      padding{rhs.padding}, paddingColor{rhs.paddingColor} {  if (rhs.render)    render = rhs.render;  if (rhs.updater)    updater = rhs.updater;  width = rhs.width;}/** * Has a render function been set? * * @return bool */bool Line::hasRender(void) {  if (render) {    return true;  } else {    return false;  }}/** * Return total length of Line * * text.length + 2 * padding length * * @return int */int Line::length(void) {  if (!padding.empty()) {    if (unicode) {      return utf8::distance(padding.begin(), padding.end()) * 2 +             utf8::distance(text.begin(), text.end());    } else {      return padding.length() * 2 + text.length();    }  }  if (unicode) {    return utf8::distance(text.begin(), text.end());  } else {    return text.length();  }}/** * Make text the given width by padding string with spaces. * * @param width int */void Line::fit(void) {  int need;  if (door::unicode)    need = width - utf8::distance(text.begin(), text.end());  else    need = width - text.length();  need -= padding.length() * 2;  if (need > 0) {    text.append(std::string(need, ' '));  }}/** * Set Line text. * @param txt std::string */void Line::setText(std::string &txt) { text = txt; }/** * Set Line text. * @param txt const char * */void Line::setText(const char *txt) { text = txt; }/** * set padding (color and text) * * @param padstring std::string * @param padColor ANSIColor */void Line::setPadding(std::string &padstring, ANSIColor padColor) {  padding = padstring;  paddingColor = padColor;}/** * set padding (color and text) * * @param padstring const char * * @param padColor ANSIColor */void Line::setPadding(const char *padstring, ANSIColor padColor) {  padding = padstring;  paddingColor = padColor;}/** * set color * * @param c ANSIColor */void Line::setColor(ANSIColor c) {  color = c;  hasColor = true;}/** * set render * * Set the renderFunction to use for this Line.  This * replaces the colorizer. * @param rf renderFunction */void Line::setRender(renderFunction rf) { render = rf; }/** * set updater function * * This can update the line text when called. * @todo Define an updateFunction. * @param newUpdater updateFunction */void Line::setUpdater(updateFunction newUpdater) { updater = newUpdater; }std::string Line::debug(void) {  std::string desc;  desc = "Line(";  desc += text;  desc += "): ";  if (updater) {    desc += "[U]";  }  if (render) {    desc += "[R]";  }  return desc;}/** * Call updater, report if the text was actually changed. * * @return bool */bool Line::update(void) {  if (updater) {    std::string newText = updater();    // int line_len;    int need;    if (unicode) {      // line_len = utf8::distance(text.begin(), text.end());      need = width - utf8::distance(newText.begin(), newText.end());    } else {      // line_len = text.length();      need = width - newText.length();    }    need -= padding.length() * 2;    if (need > 0) {      newText.append(std::string(need, ' '));    }    if (newText != text) {      text = newText;      return true;    }  }  return false;}/** * Output Line * * This looks for padding and paddingColor. * This uses the render function if set. * * @param os std::ostream * @param l const Line & * @return std::ostream& */std::ostream &operator<<(std::ostream &os, const Line &l) {  // Door *d = dynamic_cast<Door *>(&os);  if (!l.padding.empty()) {    os << l.paddingColor << l.padding;  }  if (l.render) {    // This has a renderer.  Use it.    Render r = l.render(l.text);    r.output(os);  } else {    if (l.hasColor) {      os << l.color;    };    os << l.text;  }  if (!l.padding.empty()) {    os << l.paddingColor << l.padding;  }  return os;}} // namespace door
 |