ansicolor.cpp 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. #include "door.h"
  2. #include <string>
  3. /**
  4. * @file
  5. * @brief ANSIColor
  6. */
  7. namespace door {
  8. /**
  9. * Construct a new ANSIColor::ANSIColor object
  10. * with sensible defaults (White on Black).
  11. *
  12. */
  13. ANSIColor::ANSIColor()
  14. : fg(COLOR::WHITE), bg(COLOR::BLACK), reset(0), bold(0), blink(0),
  15. inverse(0) {}
  16. /**
  17. * Construct a new ANSIColor::ANSIColor object
  18. * with attribute set.
  19. *
  20. * @param[in] a ATTR
  21. */
  22. ANSIColor::ANSIColor(ATTR a) : ANSIColor() { Attr(a); }
  23. /**
  24. * Construct a new ANSIColor::ANSIColor object
  25. * with a foreground color.
  26. *
  27. * @param[in] f COLOR
  28. */
  29. ANSIColor::ANSIColor(COLOR f) : ANSIColor() { fg = f; }
  30. /**
  31. * Construct a new ANSIColor::ANSIColor object
  32. * with a foreground color and attribute.
  33. *
  34. * @param[in] f COLOR
  35. * @param[in] a ATTR
  36. */
  37. ANSIColor::ANSIColor(COLOR f, ATTR a) : ANSIColor() {
  38. fg = f;
  39. Attr(a);
  40. }
  41. /**
  42. * Construct a new ANSIColor::ANSIColor object
  43. * with a foreground color and attributes.
  44. *
  45. * @param[in] f COLOR
  46. * @param[in] a1 ATTR
  47. * @param[in] a2 ATTR
  48. */
  49. ANSIColor::ANSIColor(COLOR f, ATTR a1, ATTR a2) : ANSIColor() {
  50. fg = f;
  51. Attr(a1);
  52. Attr(a2);
  53. }
  54. /**
  55. * Construct a new ANSIColor::ANSIColor object
  56. * with a foreground and background color.
  57. *
  58. * @param[in] f foreground COLOR
  59. * @param[in] b background COLOR
  60. */
  61. ANSIColor::ANSIColor(COLOR f, COLOR b) : ANSIColor() {
  62. fg = f;
  63. bg = b;
  64. }
  65. /**
  66. * Construct a new ANSIColor::ANSIColor object
  67. * with a foreground color, background color,
  68. * and attribute.
  69. *
  70. * @param[in] f foreground COLOR
  71. * @param[in] b background COLOR
  72. * @param[in] a ATTR
  73. */
  74. ANSIColor::ANSIColor(COLOR f, COLOR b, ATTR a) : ANSIColor() {
  75. fg = f;
  76. bg = b;
  77. Attr(a);
  78. }
  79. /**
  80. * Construct a new ANSIColor::ANSIColor object
  81. * with foreground, background color and attributes.
  82. *
  83. * @param[in] f foreground COLOR
  84. * @param[in] b background COLOR
  85. * @param[in] a1 ATTR
  86. * @param[in] a2 ATTR
  87. */
  88. ANSIColor::ANSIColor(COLOR f, COLOR b, ATTR a1, ATTR a2) : ANSIColor() {
  89. fg = f;
  90. bg = b;
  91. Attr(a1);
  92. Attr(a2);
  93. }
  94. /**
  95. * Set attribute. We return the object so
  96. * calls can be chained.
  97. *
  98. * @param[in] a ATTR
  99. * @return ANSIColor&
  100. */
  101. ANSIColor &ANSIColor::Attr(ATTR a) {
  102. switch (a) {
  103. case ATTR::RESET:
  104. reset = 1;
  105. break;
  106. case ATTR::BOLD:
  107. bold = 1;
  108. break;
  109. case ATTR::BLINK:
  110. blink = 1;
  111. break;
  112. case ATTR::INVERSE:
  113. inverse = 1;
  114. break;
  115. }
  116. return *this;
  117. }
  118. /**
  119. * Equal operator.
  120. *
  121. * This compares colors and attributes, but ignores reset.
  122. *
  123. * @param[in] c const ANSIColor &
  124. * @return bool
  125. */
  126. bool ANSIColor::operator==(const ANSIColor &c) const {
  127. return ((fg == c.fg) and (bg == c.bg) and (bold == c.bold) and
  128. (blink == c.blink) and (inverse == c.inverse));
  129. }
  130. /**
  131. * Not-equal operator.
  132. *
  133. * This compares colors and attributes, but ignores reset.
  134. *
  135. * @param[in] c const ANSIColor &
  136. * @return bool
  137. */
  138. bool ANSIColor::operator!=(const ANSIColor &c) const {
  139. return !((fg == c.fg) and (bg == c.bg) and (bold == c.bold) and
  140. (blink == c.blink) and (inverse == c.inverse));
  141. }
  142. /**
  143. * @brief Set foreground color
  144. *
  145. * @param[in] f foreground COLOR
  146. */
  147. void ANSIColor::setFg(COLOR f) {
  148. fg = f;
  149. reset = 0;
  150. bold = 0;
  151. blink = 0;
  152. inverse = 0;
  153. }
  154. /**
  155. * @brief Set foreground color and attribute
  156. *
  157. * @param[in] f foreground COLOR
  158. * @param[in] a ATTR
  159. */
  160. void ANSIColor::setFg(COLOR f, ATTR a) {
  161. fg = f;
  162. attr(a);
  163. }
  164. /**
  165. * @brief Set background color
  166. *
  167. * @param[in] b background COLOR
  168. */
  169. void ANSIColor::setBg(COLOR b) { bg = b; }
  170. /**
  171. * @brief Set attribute
  172. *
  173. * This clears all the attributes before setting the selected ATTR.
  174. *
  175. * @param[in] a ATTR
  176. */
  177. void ANSIColor::attr(ATTR a) {
  178. // first, clear all attributes
  179. reset = 0;
  180. bold = 0;
  181. blink = 0;
  182. inverse = 0;
  183. Attr(a);
  184. }
  185. /**
  186. * Output the full ANSI codes for attributes and color.
  187. * This does not look at the previous values.
  188. */
  189. std::string ANSIColor::output(void) const {
  190. std::string clr(CSI);
  191. // check for special cases
  192. if (reset and (fg == COLOR::BLACK) and (bg == COLOR::BLACK)) {
  193. clr += "0m";
  194. return clr;
  195. }
  196. if (reset and (fg == COLOR::WHITE) and (bg == COLOR::BLACK)) {
  197. clr += "0m";
  198. return clr;
  199. }
  200. if (reset) {
  201. clr += "0;";
  202. }
  203. if (bold) {
  204. if (blink) {
  205. clr += "5;";
  206. }
  207. clr += "1;";
  208. } else {
  209. if (!reset)
  210. clr += "0;";
  211. if (blink) {
  212. clr += "5;";
  213. }
  214. }
  215. clr += std::to_string(30 + (int)fg) + ";";
  216. clr += std::to_string(40 + (int)bg) + "m";
  217. return clr;
  218. }
  219. /**
  220. * @brief Output debug string for ANSIColor
  221. *
  222. * @return std::string
  223. */
  224. std::string ANSIColor::debug(void) {
  225. std::string output;
  226. output = "ANSIColor FG";
  227. output += std::to_string((int)fg);
  228. output += ", BG";
  229. output += std::to_string((int)bg);
  230. output += ", B";
  231. output += std::to_string(bold);
  232. output += ", R";
  233. output += std::to_string(reset);
  234. return output;
  235. }
  236. /**
  237. * Output only what ANSI attributes and colors have changed.
  238. * This uses the previous ANSIColor value to determine what
  239. * has changed.
  240. *
  241. * This sets previous to the current upon completion.
  242. */
  243. std::string ANSIColor::output(ANSIColor &previous) const {
  244. std::string clr(CSI);
  245. // color output optimization
  246. // check for special cases
  247. if (reset and (fg == COLOR::BLACK) and (bg == COLOR::BLACK)) {
  248. clr += "0m";
  249. previous = *this;
  250. previous.reset = 0;
  251. return clr;
  252. }
  253. bool temp_reset = false;
  254. if ((!blink) and (blink != previous.blink)) {
  255. temp_reset = true;
  256. }
  257. if ((reset) or (temp_reset)) {
  258. // current has RESET, so default to always sending colors.
  259. if (temp_reset) {
  260. clr += "0m";
  261. }
  262. // this fixes the extra \x1b that shows up with reset.
  263. if (clr.compare(CSI) == 0)
  264. clr.clear();
  265. clr += output();
  266. /*
  267. std::ofstream logf;
  268. logf.open("ansicolor.log", std::ofstream::out | std::ofstream::app);
  269. logf << "clr = [" << clr << "]" << std::endl;
  270. logf.close();
  271. */
  272. previous = *this;
  273. previous.reset = 0;
  274. return clr;
  275. }
  276. if (*this == previous) {
  277. clr.clear();
  278. return clr;
  279. }
  280. // resume "optimization"
  281. if (bold != previous.bold) {
  282. // not the same, so handle this.
  283. if (bold) {
  284. if (blink) {
  285. clr += "5;";
  286. }
  287. clr += "1;";
  288. if (fg != previous.fg)
  289. clr += std::to_string((int)fg + 30) + ";";
  290. if (bg != previous.bg)
  291. clr += std::to_string((int)bg + 40) + ";";
  292. } else {
  293. clr += "0;";
  294. if (blink) {
  295. clr += "5;";
  296. }
  297. // RESET to turn OFF BOLD, clears previous
  298. if (fg != COLOR::WHITE)
  299. clr += std::to_string((int)fg + 30) + ";";
  300. if (bg != COLOR::BLACK)
  301. clr += std::to_string((int)bg + 40) + ";";
  302. }
  303. } else {
  304. // not bold.
  305. if (blink) {
  306. clr += "5;";
  307. }
  308. if (fg != previous.fg)
  309. clr += std::to_string((int)fg + 30) + ";";
  310. if (bg != previous.bg)
  311. clr += std::to_string((int)bg + 40) + ";";
  312. };
  313. // Remove ';' if last character
  314. std::string::iterator si = clr.end() - 1;
  315. if (*si == ';') {
  316. clr.erase(si);
  317. }
  318. if (clr.compare(CSI) == 0)
  319. clr.clear();
  320. else
  321. clr += "m";
  322. // final step, set previous to current and return the string;
  323. previous = *this;
  324. return clr;
  325. };
  326. /**
  327. * This converts ANSI \ref COLOR and \ref ATTR to ANSI codes
  328. * understood by the \ref Door output class.
  329. */
  330. std::ostream &operator<<(std::ostream &os, const ANSIColor &c) {
  331. std::string out;
  332. Door *d = dynamic_cast<Door *>(&os);
  333. if (d != nullptr) {
  334. d->track = false;
  335. out = c.output(d->previous);
  336. // if (!out.empty())
  337. if (out.compare("\x1b[") == 0)
  338. std::abort();
  339. *d << out;
  340. d->track = true;
  341. } else {
  342. // "dumb" version that can't remember anything/ doesn't optimize color
  343. // output.
  344. std::string out = c.output();
  345. os << out;
  346. }
  347. return os;
  348. }
  349. ANSIColor reset(ATTR::RESET);
  350. } // namespace door