lines.cpp 6.9 KB

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