ansicolor.h 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. #ifndef ANSICOLOR_H
  2. #define ANSICOLOR_H
  3. #include <cstdint>
  4. #include <string>
  5. #define CSI "\x1b["
  6. /**
  7. * ANSI Color codes
  8. */
  9. /**
  10. * @brief The colors available under ANSI-BBS
  11. */
  12. enum class COLOR : std::int8_t {
  13. /// BLACK (0)
  14. BLACK,
  15. /// RED (1)
  16. RED,
  17. /// GREEN (2)
  18. GREEN,
  19. /// BROWN (3)
  20. BROWN,
  21. /// YELLOW (3)
  22. YELLOW = 3,
  23. /// BLUE (4)
  24. BLUE,
  25. /// MAGENTA (5)
  26. MAGENTA,
  27. /// CYAN (6)
  28. CYAN,
  29. /// WHITE (7)
  30. WHITE
  31. };
  32. /**
  33. * @brief ANSI-BBS text attributes
  34. */
  35. enum class ATTR : std::int8_t {
  36. /// RESET forces all attributes (and Colors) to be sent.
  37. RESET,
  38. /// BOLD is the same as BRIGHT.
  39. BOLD,
  40. /// BRIGHT is the same as BOLD.
  41. BRIGHT = 1,
  42. /// SLOW BLINK
  43. BLINK = 5,
  44. /// INVERSE is Background on Foreground.
  45. INVERSE = 7
  46. };
  47. // std::int32_t
  48. constexpr long strhash(const char *txt) {
  49. long result = 0;
  50. for (int x = 0; x < 3; ++x) {
  51. // char c = txt[x] - 'A';
  52. result = (result << 8) | txt[x];
  53. }
  54. return result;
  55. };
  56. #define A_BOLD 0x01
  57. #define A_BLINK 0x02
  58. #define A_INVERSE 0x04
  59. #define A_RESET 0x10
  60. /**
  61. * @class ANSIColor
  62. * This holds foreground, background and ANSI-BBS attribute
  63. * information.
  64. * The special attribute RESET forces attribute and color
  65. * output always.
  66. *
  67. * @brief Foreground, Background and Attributes
  68. *
  69. */
  70. class ANSIColor {
  71. /** Foreground color */
  72. COLOR fg;
  73. /** Background color */
  74. COLOR bg;
  75. int attr;
  76. public:
  77. ANSIColor();
  78. ANSIColor(ATTR a);
  79. ANSIColor(COLOR f);
  80. ANSIColor(COLOR f, ATTR a);
  81. ANSIColor(COLOR f, ATTR a1, ATTR a2);
  82. ANSIColor(COLOR f, COLOR b);
  83. ANSIColor(COLOR f, COLOR b, ATTR a);
  84. ANSIColor(COLOR f, COLOR b, ATTR a1, ATTR a2);
  85. ANSIColor(std::initializer_list<int> il);
  86. constexpr ANSIColor(const char *text)
  87. : fg(COLOR::WHITE),
  88. bg(COLOR::BLACK),
  89. attr(0) {
  90. const char *cp = text;
  91. bool use_on = false;
  92. while (*cp != 0) {
  93. long key = strhash(cp);
  94. switch (key) {
  95. case strhash("BRI"):
  96. attr |= A_BOLD;
  97. break;
  98. case strhash("BOL"):
  99. attr |= A_BOLD;
  100. break;
  101. case strhash("BLI"):
  102. attr |= A_BLINK;
  103. break;
  104. case strhash("INV"):
  105. attr |= A_INVERSE;
  106. break;
  107. case strhash("RES"):
  108. attr |= A_RESET;
  109. break;
  110. case strhash("ON "):
  111. use_on = true;
  112. break;
  113. case strhash("BLU"):
  114. if (use_on)
  115. bg = COLOR::BLUE;
  116. else
  117. bg = COLOR::BLUE;
  118. break;
  119. case strhash("RED"):
  120. if (use_on)
  121. bg = COLOR::RED;
  122. else
  123. fg = COLOR::RED;
  124. break;
  125. case strhash("GRE"):
  126. if (use_on)
  127. bg = COLOR::GREEN;
  128. else
  129. fg = COLOR::GREEN;
  130. break;
  131. case strhash("YEL"):
  132. if (use_on)
  133. bg = COLOR::YELLOW;
  134. else {
  135. fg = COLOR::YELLOW;
  136. attr |= A_BOLD;
  137. }
  138. // if (use_on) throw error!
  139. break;
  140. case strhash("BRO"):
  141. if (use_on)
  142. bg = COLOR::BROWN;
  143. else
  144. fg = COLOR::BROWN;
  145. break;
  146. case strhash("CYA"):
  147. if (use_on)
  148. bg = COLOR::CYAN;
  149. else
  150. fg = COLOR::CYAN;
  151. break;
  152. case strhash("MAG"):
  153. if (use_on)
  154. bg = COLOR::MAGENTA;
  155. else
  156. fg = COLOR::MAGENTA;
  157. break;
  158. case strhash("BLA"):
  159. if (use_on)
  160. bg = COLOR::BLACK;
  161. else
  162. fg = COLOR::BLACK;
  163. break;
  164. case strhash("WHI"):
  165. if (use_on)
  166. bg = COLOR::WHITE;
  167. else
  168. fg = COLOR::WHITE;
  169. break;
  170. }
  171. // skip to the space character
  172. while ((*cp != ' ') && (*cp != 0)) ++cp;
  173. // skip past the space character
  174. while (*cp == ' ') ++cp;
  175. }
  176. };
  177. /*
  178. ANSIColor(int c1);
  179. ANSIColor(int c1, int c2);
  180. */
  181. ANSIColor &Attr(ATTR a);
  182. bool operator==(const ANSIColor &c) const;
  183. bool operator!=(const ANSIColor &c) const;
  184. void setFg(COLOR f);
  185. void setFg(COLOR f, ATTR a);
  186. void setBg(COLOR b);
  187. /**
  188. * Get the foreground color
  189. * @return COLOR
  190. */
  191. COLOR getFg() { return fg; };
  192. /**
  193. * Get the background color
  194. * @return COLOR
  195. */
  196. COLOR getBg() { return bg; };
  197. void setAttr(ATTR a);
  198. std::string output(void) const;
  199. std::string operator()(void) const;
  200. };
  201. #ifdef NON_WORKING
  202. #include <vector>
  203. constexpr std::vector<int> from_text(const char *text) {
  204. std::vector<int> vi;
  205. const char *cp = text;
  206. bool use_on = false;
  207. while (*cp != 0) {
  208. int key = strhash(cp);
  209. switch (key) {
  210. case strhash("BRI"):
  211. vi.push_back(1);
  212. break;
  213. case strhash("BOL"):
  214. vi.push_back(1);
  215. break;
  216. case strhash("BLI"):
  217. vi.push_back(5);
  218. break;
  219. case strhash("INV"):
  220. vi.push_back(7);
  221. break;
  222. case strhash("ON "):
  223. use_on = true;
  224. break;
  225. case strhash("BLU"):
  226. if (use_on)
  227. ac.setBg(COLOR::BLUE);
  228. else
  229. ac.setFg(COLOR::BLUE);
  230. break;
  231. case strhash("RED"):
  232. if (use_on)
  233. ac.setBg(COLOR::RED);
  234. else
  235. ac.setFg(COLOR::RED);
  236. break;
  237. case strhash("GRE"):
  238. if (use_on)
  239. ac.setBg(COLOR::GREEN);
  240. else
  241. ac.setFg(COLOR::GREEN);
  242. break;
  243. case strhash("YEL"):
  244. // if (use_on) throw error!
  245. ac.setFg(COLOR::YELLOW);
  246. ac.Attr(ATTR::BOLD);
  247. break;
  248. case strhash("BRO"):
  249. if (use_on)
  250. ac.setBg(COLOR::BROWN);
  251. else
  252. ac.setFg(COLOR::BROWN);
  253. break;
  254. case strhash("CYA"):
  255. if (use_on)
  256. ac.setBg(COLOR::CYAN);
  257. else
  258. ac.setFg(COLOR::CYAN);
  259. break;
  260. case strhash("MAG"):
  261. if (use_on)
  262. ac.setBg(COLOR::MAGENTA);
  263. else
  264. ac.setFg(COLOR::MAGENTA);
  265. break;
  266. case strhash("BLA"):
  267. if (use_on)
  268. ac.setBg(COLOR::BLACK);
  269. else
  270. ac.setFg(COLOR::BLACK);
  271. break;
  272. case strhash("WHI"):
  273. if (use_on)
  274. ac.setBg(COLOR::WHITE);
  275. else
  276. ac.setFg(COLOR::WHITE);
  277. break;
  278. }
  279. // skip to the space character
  280. while ((*cp != ' ') && (*cp != 0)) ++cp;
  281. // skip past the space character
  282. while (*cp == ' ') ++cp;
  283. }
  284. return vi;
  285. }
  286. #endif
  287. /*
  288. Maybe make char, char, char (fg, bg, attr) and store those from
  289. the constexpr.
  290. This "works", in that it does construct an ANSIColor from text.
  291. However, it isn't stored as an ANSIColor, but calls the function
  292. at run-time. (I was hoping for an efficient way to construct
  293. ANSIColor from a string.)
  294. Maybe a template that constructs it from a initializer_list?
  295. */
  296. constexpr ANSIColor ac_from_text(const char *text) {
  297. ANSIColor ac(nullptr);
  298. const char *cp = text;
  299. bool use_on = false;
  300. while (*cp != 0) {
  301. int key = strhash(cp);
  302. switch (key) {
  303. case strhash("BRI"):
  304. ac.Attr(ATTR::BRIGHT);
  305. break;
  306. case strhash("BOL"):
  307. ac.Attr(ATTR::BOLD);
  308. break;
  309. case strhash("BLI"):
  310. ac.Attr(ATTR::BLINK);
  311. break;
  312. case strhash("INV"):
  313. ac.Attr(ATTR::INVERSE);
  314. break;
  315. case strhash("ON "):
  316. use_on = true;
  317. break;
  318. case strhash("BLU"):
  319. if (use_on)
  320. ac.setBg(COLOR::BLUE);
  321. else
  322. ac.setFg(COLOR::BLUE);
  323. break;
  324. case strhash("RED"):
  325. if (use_on)
  326. ac.setBg(COLOR::RED);
  327. else
  328. ac.setFg(COLOR::RED);
  329. break;
  330. case strhash("GRE"):
  331. if (use_on)
  332. ac.setBg(COLOR::GREEN);
  333. else
  334. ac.setFg(COLOR::GREEN);
  335. break;
  336. case strhash("YEL"):
  337. // if (use_on) throw error!
  338. ac.setFg(COLOR::YELLOW);
  339. ac.Attr(ATTR::BOLD);
  340. break;
  341. case strhash("BRO"):
  342. if (use_on)
  343. ac.setBg(COLOR::BROWN);
  344. else
  345. ac.setFg(COLOR::BROWN);
  346. break;
  347. case strhash("CYA"):
  348. if (use_on)
  349. ac.setBg(COLOR::CYAN);
  350. else
  351. ac.setFg(COLOR::CYAN);
  352. break;
  353. case strhash("MAG"):
  354. if (use_on)
  355. ac.setBg(COLOR::MAGENTA);
  356. else
  357. ac.setFg(COLOR::MAGENTA);
  358. break;
  359. case strhash("BLA"):
  360. if (use_on)
  361. ac.setBg(COLOR::BLACK);
  362. else
  363. ac.setFg(COLOR::BLACK);
  364. break;
  365. case strhash("WHI"):
  366. if (use_on)
  367. ac.setBg(COLOR::WHITE);
  368. else
  369. ac.setFg(COLOR::WHITE);
  370. break;
  371. }
  372. // skip to the space character
  373. while ((*cp != ' ') && (*cp != 0)) ++cp;
  374. // skip past the space character
  375. while (*cp == ' ') ++cp;
  376. }
  377. return ac;
  378. };
  379. extern ANSIColor reset;
  380. #endif