lines.cpp 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. #include "door.h"
  2. #include "utf8.h"
  3. namespace door {
  4. #ifdef EXPERIMENTAL
  5. BasicLine::BasicLine(std::string txt) : text{txt}, hasColor{false} {}
  6. BasicLine::BasicLine(std::string txt, ANSIColor c)
  7. : text{txt}, hasColor{true}, color{c} {}
  8. bool BasicLine::hasRender(void) {
  9. if (render)
  10. return true;
  11. return false;
  12. }
  13. void BasicLine::setText(std::string txt) { text = txt; }
  14. void BasicLine::setColor(ANSIColor c) {
  15. color = c;
  16. hasColor = true;
  17. }
  18. void BasicLine::setRender(renderFunction rf) { render = rf; }
  19. void BasicLine::setUpdater(updateFunction uf) { updater = uf; }
  20. /**
  21. * Update BasicLine, if we have an updater.
  22. *
  23. * If we have an updater, call it. If the text is different,
  24. * update setText() and return true.
  25. * Otherwise false.
  26. *
  27. * This doesn't detect changes (like if the render has been changed, for
  28. * example)
  29. *
  30. * @return bool
  31. */
  32. bool BasicLine::update(void) {
  33. if (updater) {
  34. std::string temp = updater();
  35. if (temp == text)
  36. return false;
  37. setText(temp);
  38. return true;
  39. }
  40. return false;
  41. }
  42. /**
  43. * Output Line
  44. *
  45. * This looks for padding and paddingColor.
  46. * This uses the render function if set.
  47. *
  48. * @param os std::ostream
  49. * @param l const BasicLine &
  50. * @return std::ostream&
  51. */
  52. std::ostream &operator<<(std::ostream &os, const BasicLine &l) {
  53. if (l.render) {
  54. // This has a renderer. Use it.
  55. Render r = l.render(l.text);
  56. r.output(os);
  57. } else {
  58. if (l.hasColor) {
  59. os << l.color;
  60. };
  61. os << l.text;
  62. }
  63. return os;
  64. }
  65. MultiLine::MultiLine(){};
  66. void MultiLine::append(std::shared_ptr<BasicLine> bl) { lines.push_back(bl); }
  67. bool MultiLine::update() {
  68. bool updated = false;
  69. for (auto line : lines) {
  70. if (line->update())
  71. updated = true;
  72. }
  73. return updated;
  74. }
  75. /**
  76. * Output Line
  77. *
  78. * This looks for padding and paddingColor.
  79. * This uses the render function if set.
  80. *
  81. * @param os std::ostream
  82. * @param ml const MultiLine &
  83. * @return std::ostream&
  84. */
  85. std::ostream &operator<<(std::ostream &os, const MultiLine &ml) {
  86. for (auto line : ml.lines) {
  87. os << *line;
  88. }
  89. return os;
  90. }
  91. #endif
  92. /**
  93. * Construct a new Line:: Line object with
  94. * string and total width.
  95. *
  96. * @param txt std::string
  97. * @param width int
  98. */
  99. Line::Line(const std::string &txt, int w) : text{txt}, width{w} {
  100. hasColor = false;
  101. }
  102. Line::Line(const std::string &txt, int w, ANSIColor c)
  103. : text{txt}, color{c}, width{w} {
  104. hasColor = true;
  105. }
  106. Line::Line(const char *txt, int w, ANSIColor c)
  107. : text{txt}, color{c}, width{w} {
  108. hasColor = true;
  109. }
  110. Line::Line(const std::string &txt, int w, renderFunction rf)
  111. : text{txt}, render{rf}, width{w} {
  112. hasColor = false;
  113. }
  114. Line::Line(const char *txt, int w, renderFunction rf)
  115. : text{txt}, render{rf}, width{w} {
  116. hasColor = false;
  117. }
  118. /**
  119. * Construct a new Line:: Line object with
  120. * const char * and total width
  121. *
  122. * @param txt const char *
  123. * @param width int
  124. */
  125. Line::Line(const char *txt, int w) : text{txt}, width{w} { hasColor = false; }
  126. /**
  127. * Construct a new Line:: Line object from an
  128. * existing Line
  129. *
  130. * @param rhs const Line&
  131. */
  132. Line::Line(const Line &rhs)
  133. : text{rhs.text}, hasColor{rhs.hasColor}, color{rhs.color},
  134. padding{rhs.padding}, paddingColor{rhs.paddingColor} {
  135. if (rhs.render) {
  136. render = rhs.render;
  137. }
  138. if (rhs.updater) {
  139. updater = rhs.updater;
  140. }
  141. width = rhs.width;
  142. }
  143. /**
  144. * Has a render function been set?
  145. *
  146. * @return bool
  147. */
  148. bool Line::hasRender(void) {
  149. if (render) {
  150. return true;
  151. } else {
  152. return false;
  153. }
  154. }
  155. /**
  156. * Return total length of Line
  157. *
  158. * text.length + 2 * padding length
  159. *
  160. * @return int
  161. */
  162. int Line::length(void) {
  163. if (!padding.empty()) {
  164. if (unicode) {
  165. return utf8::distance(padding.begin(), padding.end()) * 2 +
  166. utf8::distance(text.begin(), text.end());
  167. } else {
  168. return padding.length() * 2 + text.length();
  169. }
  170. }
  171. if (unicode) {
  172. return utf8::distance(text.begin(), text.end());
  173. } else {
  174. return text.length();
  175. }
  176. }
  177. /**
  178. * Make text the given width by padding string with spaces.
  179. *
  180. * @param width int
  181. */
  182. void Line::fit(void) {
  183. int need;
  184. if (door::unicode)
  185. need = width - utf8::distance(text.begin(), text.end());
  186. else
  187. need = width - text.length();
  188. need -= padding.length() * 2;
  189. if (need > 0) {
  190. text.append(std::string(need, ' '));
  191. }
  192. }
  193. /**
  194. * Set Line text.
  195. * @param txt std::string
  196. */
  197. void Line::setText(std::string &txt) { text = txt; }
  198. /**
  199. * Set Line text.
  200. * @param txt const char *
  201. */
  202. void Line::setText(const char *txt) { text = txt; }
  203. /**
  204. * set padding (color and text)
  205. *
  206. * @param padstring std::string
  207. * @param padColor ANSIColor
  208. */
  209. void Line::setPadding(std::string &padstring, ANSIColor padColor) {
  210. padding = padstring;
  211. paddingColor = padColor;
  212. }
  213. /**
  214. * set padding (color and text)
  215. *
  216. * @param padstring const char *
  217. * @param padColor ANSIColor
  218. */
  219. void Line::setPadding(const char *padstring, ANSIColor padColor) {
  220. padding = padstring;
  221. paddingColor = padColor;
  222. }
  223. /**
  224. * set color
  225. *
  226. * @param c ANSIColor
  227. */
  228. void Line::setColor(ANSIColor c) {
  229. color = c;
  230. hasColor = true;
  231. }
  232. /**
  233. * set render
  234. *
  235. * Set the renderFunction to use for this Line. This
  236. * replaces the colorizer.
  237. * @param rf renderFunction
  238. */
  239. void Line::setRender(renderFunction rf) { render = rf; }
  240. /**
  241. * set updater function
  242. *
  243. * This can update the line text when called.
  244. * @todo Define an updateFunction.
  245. * @param newUpdater updateFunction
  246. */
  247. void Line::setUpdater(updateFunction newUpdater) { updater = newUpdater; }
  248. std::string Line::debug(void) {
  249. std::string desc;
  250. desc = "Line(";
  251. desc += text;
  252. desc += "): ";
  253. if (updater) {
  254. desc += "[U]";
  255. }
  256. if (render) {
  257. desc += "[R]";
  258. }
  259. return desc;
  260. }
  261. /**
  262. * Call updater, report if the text was actually changed.
  263. *
  264. * @return bool
  265. */
  266. bool Line::update(void) {
  267. if (updater) {
  268. std::string newText = updater();
  269. // int line_len;
  270. int need;
  271. if (unicode) {
  272. // line_len = utf8::distance(text.begin(), text.end());
  273. need = width - utf8::distance(newText.begin(), newText.end());
  274. } else {
  275. // line_len = text.length();
  276. need = width - newText.length();
  277. }
  278. need -= padding.length() * 2;
  279. if (need > 0) {
  280. newText.append(std::string(need, ' '));
  281. }
  282. if (newText != text) {
  283. text = newText;
  284. return true;
  285. }
  286. }
  287. return false;
  288. }
  289. /**
  290. * Output Line
  291. *
  292. * This looks for padding and paddingColor.
  293. * This uses the render function if set.
  294. *
  295. * @param os std::ostream
  296. * @param l const Line &
  297. * @return std::ostream&
  298. */
  299. std::ostream &operator<<(std::ostream &os, const Line &l) {
  300. // Door *d = dynamic_cast<Door *>(&os);
  301. if (!l.padding.empty()) {
  302. os << l.paddingColor << l.padding;
  303. }
  304. if (l.render) {
  305. // This has a renderer. Use it.
  306. Render r = l.render(l.text);
  307. r.output(os);
  308. } else {
  309. if (l.hasColor) {
  310. os << l.color;
  311. };
  312. os << l.text;
  313. }
  314. if (!l.padding.empty()) {
  315. os << l.paddingColor << l.padding;
  316. }
  317. return os;
  318. }
  319. } // namespace door