bar.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. #include "door.h"
  2. #include "utf8.h"
  3. #include <iomanip>
  4. #include <iostream>
  5. namespace door {
  6. Bar::Bar(int width, BarStyle s) : text(' ', width), line(text) {
  7. length = width;
  8. style = s;
  9. // set updater
  10. // set colorizer
  11. // profit
  12. }
  13. void Bar::update_bar(void) {
  14. unsigned long step_width;
  15. text.clear();
  16. switch (style) {
  17. case BarStyle::SOLID:
  18. case BarStyle::PERCENTAGE:
  19. case BarStyle::PERCENT_SPACE: {
  20. step_width = 100 * 100 / length;
  21. int steps = current_percent / step_width;
  22. // number of steps visible.
  23. // for now:
  24. if (door::unicode)
  25. for (int i = 0; i < steps; ++i)
  26. text.append("\u2588");
  27. else
  28. text.assign(steps, '\xdb');
  29. for (int x = steps; x < length; ++x)
  30. text.append(" ");
  31. // line.setText(text);
  32. /*
  33. cout << "percent " << std::setw(5) << current_percent << " : step_width "
  34. << std::setw(5) << step_width << std::setw(5) << steps << " of "
  35. << std::setw(5) << length << " ";
  36. */
  37. if ((style == BarStyle::PERCENTAGE) || (style == BarStyle::PERCENT_SPACE)) {
  38. // Put the % text in the text.
  39. std::string percent;
  40. percent = std::to_string(current_percent / 100);
  41. int pos = (length / 2) - 1;
  42. if (percent != "100") {
  43. percent.append(1, '%');
  44. if (percent.length() < 3)
  45. percent.insert(0, " ");
  46. }
  47. if (style == BarStyle::PERCENT_SPACE) {
  48. percent.insert(0, 1, ' ');
  49. percent.append(1, ' ');
  50. pos -= 1;
  51. }
  52. // cout << "[" << percent << "] " << std::setw(3) << pos << " ";
  53. // unicode ... is unipain.
  54. if (door::unicode) {
  55. // handle unicode here
  56. // Find start position, and ending position for the replacement.
  57. const char *cp = text.c_str();
  58. const char *end = cp;
  59. while (*end != 0)
  60. end++;
  61. for (int i = 0; i < pos; i++) {
  62. utf8::next(cp, end);
  63. }
  64. // find position using unicode position
  65. int unicode_pos = cp - text.c_str();
  66. const char *cp_len_start = cp;
  67. const char *cp_len = cp;
  68. for (int i = 0; i < (int)percent.length(); i++) {
  69. utf8::next(cp_len, end);
  70. }
  71. int unicode_len = cp_len - cp_len_start;
  72. // cout << " " << std::setw(3) << unicode_pos << " " << std::setw(3) <<
  73. // unicode_len << " ";
  74. text.replace(unicode_pos, unicode_len, percent);
  75. } else {
  76. text.replace(pos, percent.length(), percent);
  77. };
  78. }
  79. line.setText(text);
  80. break;
  81. };
  82. case BarStyle::HALF_STEP: {
  83. step_width = 100 * 100 / length;
  84. int steps = current_percent * 2 / step_width;
  85. /*
  86. cout << "percent " << std::setw(5) << current_percent << " : step_width "
  87. << std::setw(5) << step_width << std::setw(5) << steps << " of "
  88. << std::setw(5) << length << " ";
  89. */
  90. if (door::unicode)
  91. for (int i = 0; i < steps / 2; i++)
  92. text.append("\u2588");
  93. else
  94. text.assign(steps / 2, '\xdb');
  95. if (steps % 2 == 1) {
  96. if (door::unicode)
  97. text.append("\u258c");
  98. else
  99. text.append(1, '\xdd');
  100. steps++;
  101. }
  102. for (int x = steps; x < length * 2; x += 2)
  103. text.append(" ");
  104. line.setText(text);
  105. break;
  106. }
  107. case BarStyle::GRADIENT: {
  108. step_width = 100 * 100 / length;
  109. int steps = current_percent * 4 / step_width;
  110. /*
  111. cout << "percent " << std::setw(5) << current_percent << " : step_width "
  112. << std::setw(5) << step_width << std::setw(5) << steps << " of "
  113. << std::setw(5) << length << " ";
  114. */
  115. if (door::unicode)
  116. for (int i = 0; i < steps / 4; i++)
  117. text.append("\u2588");
  118. else
  119. text.assign(steps / 4, '\xdb');
  120. if (steps % 4 != 0) {
  121. // display the gradient
  122. switch (steps % 4) {
  123. case 1:
  124. if (door::unicode)
  125. text.append("\u2591");
  126. else
  127. text.append(1, '\xb0');
  128. break;
  129. case 2:
  130. if (door::unicode)
  131. text.append("\u2592");
  132. else
  133. text.append(1, '\xb1');
  134. break;
  135. case 3:
  136. if (door::unicode)
  137. text.append("\u2593");
  138. else
  139. text.append(1, '\xb2');
  140. break;
  141. }
  142. while (steps % 4 != 0)
  143. steps++;
  144. }
  145. for (int x = steps; x < length * 4; x += 4)
  146. text.append(" ");
  147. line.setText(text);
  148. break;
  149. }
  150. }
  151. // cout << "percent" << current_percent << " : " << steps << " of " << length
  152. // << " " << std::endl;
  153. }
  154. void Bar::set(float percent) {
  155. current_percent = (unsigned long)(percent * 100.0);
  156. update_bar();
  157. }
  158. void Bar::set(int value, int max) {
  159. // I'm thinking the ulong would be % * 100.
  160. unsigned long percentage = value * 10000 / max;
  161. // float percentage = ((float)pos / (float)max) * 100.0;
  162. set(percentage);
  163. }
  164. void Bar::set(unsigned long percent) {
  165. current_percent = percent;
  166. update_bar();
  167. }
  168. /**
  169. * Output Bar
  170. *
  171. * This outputs the Progress Bar's internal line.
  172. *
  173. * @param os std::ostream
  174. * @param b const Bar &
  175. * @return std::ostream&
  176. */
  177. std::ostream &operator<<(std::ostream &os, const Bar &b) {
  178. os << b.line;
  179. // reset?
  180. os << door::reset;
  181. return os;
  182. }
  183. BarLine::BarLine(const std::string &txt, int width)
  184. : Line(txt, width), barstyle{BarStyle::SOLID},
  185. current_percent{0}, length{width} {
  186. init();
  187. }
  188. BarLine::BarLine(const char *txt, int width)
  189. : Line(txt, width), barstyle{BarStyle::SOLID},
  190. current_percent{0}, length{width} {
  191. init();
  192. }
  193. BarLine::BarLine(const std::string &txt, int width, ANSIColor c)
  194. : Line(txt, width, c), barstyle{BarStyle::SOLID},
  195. current_percent{0}, length{width} {
  196. init();
  197. }
  198. BarLine::BarLine(const char *txt, int width, ANSIColor c)
  199. : Line(txt, width, c), barstyle{BarStyle::SOLID},
  200. current_percent{0}, length{width} {
  201. init();
  202. }
  203. void BarLine::init(void) {
  204. // set update function.
  205. door::updateFunction barLineUpdate = [this](void) -> std::string {
  206. if (!colorRange.empty()) {
  207. // Ok, there is a range, so test for it.
  208. ANSIColor ac;
  209. // unsigned long p;
  210. for (auto bc : colorRange) {
  211. if (current_percent <= bc.percent) {
  212. ac = bc.c;
  213. // p = bc.percent;
  214. break;
  215. };
  216. }
  217. // cout << "!" << current_percent << "," << p << " ";
  218. setColor(ac);
  219. }
  220. return this->update_bar();
  221. };
  222. setUpdater(barLineUpdate);
  223. update();
  224. }
  225. void BarLine::setStyle(BarStyle s) { barstyle = s; }
  226. void BarLine::set(int value, int max) {
  227. unsigned long percentage = value * 100 * 100 / max;
  228. set(percentage);
  229. }
  230. void BarLine::set(float percent) {
  231. unsigned long percentage = (unsigned long)(percent * 100.0);
  232. set(percentage);
  233. }
  234. void BarLine::set(unsigned long percent) {
  235. current_percent = percent;
  236. update_bar();
  237. }
  238. std::string BarLine::update_bar(void) {
  239. unsigned long step_width;
  240. std::string btext;
  241. switch (barstyle) {
  242. case BarStyle::SOLID:
  243. case BarStyle::PERCENTAGE:
  244. case BarStyle::PERCENT_SPACE: {
  245. step_width = 100 * 100 / length;
  246. int steps = current_percent / step_width;
  247. // number of steps visible.
  248. // for now:
  249. if (door::unicode)
  250. for (int i = 0; i < steps; ++i)
  251. btext.append("\u2588");
  252. else
  253. btext.assign(steps, '\xdb');
  254. for (int x = steps; x < length; ++x)
  255. btext.append(" ");
  256. // line.setText(text);
  257. /*
  258. cout << "percent " << std::setw(5) << current_percent << " : step_width "
  259. << std::setw(5) << step_width << std::setw(5) << steps << " of "
  260. << std::setw(5) << length << " ";
  261. */
  262. if ((barstyle == BarStyle::PERCENTAGE) ||
  263. (barstyle == BarStyle::PERCENT_SPACE)) {
  264. // Put the % text in the text.
  265. std::string percent;
  266. percent = std::to_string(current_percent / 100);
  267. int pos = (length / 2) - 1;
  268. if (percent != "100") {
  269. percent.append(1, '%');
  270. if (percent.length() < 3)
  271. percent.insert(0, " ");
  272. }
  273. if (barstyle == BarStyle::PERCENT_SPACE) {
  274. percent.insert(0, 1, ' ');
  275. percent.append(1, ' ');
  276. pos -= 1;
  277. }
  278. // cout << "[" << percent << "] " << std::setw(3) << pos << " ";
  279. // unicode ... is unipain.
  280. if (door::unicode) {
  281. // handle unicode here
  282. // Find start position, and ending position for the replacement.
  283. const char *cp = btext.c_str();
  284. const char *end = cp;
  285. while (*end != 0)
  286. end++;
  287. for (int i = 0; i < pos; i++) {
  288. utf8::next(cp, end);
  289. }
  290. // find position using unicode position
  291. int unicode_pos = cp - btext.c_str();
  292. const char *cp_len_start = cp;
  293. const char *cp_len = cp;
  294. for (int i = 0; i < (int)percent.length(); i++) {
  295. utf8::next(cp_len, end);
  296. }
  297. int unicode_len = cp_len - cp_len_start;
  298. // cout << " " << std::setw(3) << unicode_pos << " " << std::setw(3) <<
  299. // unicode_len << " ";
  300. btext.replace(unicode_pos, unicode_len, percent);
  301. } else {
  302. btext.replace(pos, percent.length(), percent);
  303. };
  304. }
  305. return btext;
  306. break;
  307. };
  308. case BarStyle::HALF_STEP: {
  309. step_width = 100 * 100 / length;
  310. int steps = current_percent * 2 / step_width;
  311. /*
  312. cout << "percent " << std::setw(5) << current_percent << " : step_width "
  313. << std::setw(5) << step_width << std::setw(5) << steps << " of "
  314. << std::setw(5) << length << " ";
  315. */
  316. if (door::unicode)
  317. for (int i = 0; i < steps / 2; i++)
  318. btext.append("\u2588");
  319. else
  320. btext.assign(steps / 2, '\xdb');
  321. if (steps % 2 == 1) {
  322. if (door::unicode)
  323. btext.append("\u258c");
  324. else
  325. btext.append(1, '\xdd');
  326. steps++;
  327. }
  328. for (int x = steps; x < length * 2; x += 2)
  329. btext.append(" ");
  330. return btext;
  331. break;
  332. }
  333. case BarStyle::GRADIENT: {
  334. step_width = 100 * 100 / length;
  335. int steps = current_percent * 4 / step_width;
  336. /*
  337. cout << "percent " << std::setw(5) << current_percent << " : step_width "
  338. << std::setw(5) << step_width << std::setw(5) << steps << " of "
  339. << std::setw(5) << length << " ";
  340. */
  341. if (door::unicode)
  342. for (int i = 0; i < steps / 4; i++)
  343. btext.append("\u2588");
  344. else
  345. btext.assign(steps / 4, '\xdb');
  346. if (steps % 4 != 0) {
  347. // display the gradient
  348. switch (steps % 4) {
  349. case 1:
  350. if (door::unicode)
  351. btext.append("\u2591");
  352. else
  353. btext.append(1, '\xb0');
  354. break;
  355. case 2:
  356. if (door::unicode)
  357. btext.append("\u2592");
  358. else
  359. btext.append(1, '\xb1');
  360. break;
  361. case 3:
  362. if (door::unicode)
  363. btext.append("\u2593");
  364. else
  365. btext.append(1, '\xb2');
  366. break;
  367. }
  368. while (steps % 4 != 0)
  369. steps++;
  370. }
  371. for (int x = steps; x < length * 4; x += 4)
  372. btext.append(" ");
  373. return btext;
  374. break;
  375. }
  376. }
  377. // cout << "percent" << current_percent << " : " << steps << " of " << length
  378. // << " " << std::endl;
  379. return btext;
  380. }
  381. void BarLine::setColorRange(vector<BarColorRange> bcr) { colorRange = bcr; }
  382. } // namespace door