dispatchers.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. #include "dispatchers.h"
  2. #include <boost/format.hpp>
  3. #include <cctype>
  4. #include "boxes.h"
  5. #include "logging.h"
  6. Dispatch::Dispatch(Director &d) : director{d} {};
  7. Dispatch::~Dispatch(){};
  8. void Dispatch::to_server(const std::string &send) { director.to_server(send); }
  9. void Dispatch::to_client(const std::string &send) { director.to_client(send); }
  10. const std::string &Dispatch::get_prompt(void) {
  11. return director.current_prompt;
  12. }
  13. void Dispatch::setNotify(notifyFunc nf) { notify_ = nf; }
  14. void Dispatch::notify(void) {
  15. if (director.post) {
  16. director.post(notify_);
  17. }
  18. }
  19. void Dispatch::chain_client_input(const std::string &input) {
  20. if (chain) {
  21. chain->chain_client_input(input);
  22. } else {
  23. client_input(input);
  24. }
  25. }
  26. void Dispatch::chain_server_line(const std::string &line, const std::string &raw_line) {
  27. if (chain) {
  28. chain->chain_server_line(line, raw_line);
  29. } else {
  30. server_line(line, raw_line);
  31. }
  32. }
  33. void Dispatch::chain_server_prompt(const std::string &prompt) {
  34. if (chain) {
  35. chain->chain_server_prompt(prompt);
  36. } else {
  37. server_prompt(prompt);
  38. }
  39. }
  40. void Dispatch::server_line(const std::string &line, const std::string &raw_line) {}
  41. void Dispatch::server_prompt(const std::string &prompt) {}
  42. void Dispatch::client_input(const std::string &input) {}
  43. MainDispatch::MainDispatch(Director &d) : Dispatch{d}, id{d}, md{d} {
  44. BUGZ_LOG(warning) << "MainDispatch()";
  45. }
  46. MainDispatch::~MainDispatch() { BUGZ_LOG(warning) << "~MainDispatch()"; }
  47. void MainDispatch::activate(void) {
  48. count = 0;
  49. old_prompt = get_prompt();
  50. /*
  51. ╔══════════════════════════════╗
  52. ║ TradeWars Proxy Active ║
  53. ╚══════════════════════════════╝
  54. -=>
  55. */
  56. Boxes box(30, 1, true);
  57. box.boxcolor = "\x1b[1;33;44m";
  58. box.textcolor = "\x1b[1;33;44m";
  59. to_client("\n\r");
  60. to_client(box.top());
  61. std::string output = " TradeWars Proxy \x1b[5mActive\x1b[0;1;33;44m ";
  62. to_client(box.row(output));
  63. to_client(box.bottom());
  64. // to_client("\n\r\x1b[1;34mWELCOME! You are now in the proxy zone...\n\r");
  65. id.prompt = "\x1b[0m \x1b[1;33;44m-=>\x1b[0m \x1b[1;37;44m";
  66. id.max_length = 15;
  67. id.setNotify([this]() { this->have_input(); });
  68. id.activate();
  69. }
  70. void MainDispatch::have_input(void) {
  71. ++count;
  72. std::string output =
  73. str(boost::format("Your Input (%2%): [%1%]\n\r") % id.input % count);
  74. to_client("\x1b[0m");
  75. to_client(output);
  76. if (id.input == "?") {
  77. // Maybe? Maybe not.
  78. }
  79. if (id.input == "menu") {
  80. md.menu_box_color = "\x1b[1;33;44m";
  81. md.menu_text_color = "\x1b[1;37;44m";
  82. md.menu_title = "Proxy Menu";
  83. md.menu_options_color = "\x1b[1;36;40m";
  84. // md.menu_prompt = " --==>> ";
  85. // bold white to white --- black to green
  86. // md.menu_prompt = "\x1b[1;37;47m\xdb\xb2\xb1\xb0\x1b[0;30;47m RED GREEN
  87. // \x1b[30;42m\xdb\xb2\xb1\xb0\x1b[0m : ";
  88. // md.menu_prompt = "\x1b[0;31;47m\xdb\xb2\xb1\xb0\x1b[0;30;47m RED
  89. // GREEN\x1b[37;42m\xdb\xb2\xb1\xb0\x1b[0m : ";
  90. const char *CP437_GRADIENT = "\xdb\xb2\xb1\xb0 "; // 100, 75, 50, 25, 0
  91. md.menu_prompt =
  92. "\x1b[0;31;40m\xdb\xb2\xb1\xb0 \x1b[31;40mRED "
  93. "\x1b[32;40mGREEN\x1b[30;42m\xdb\xb2\xb1\xb0 \x1b[0m : ";
  94. md.lazy = true;
  95. md.menu = {{"A", "Apple"}, {"B", "Blue"}, {"R", "Rabbit"}, {"Z", "ZOOO!"}};
  96. md.setNotify([this]() { this->menu_choice(); });
  97. md.activate();
  98. return;
  99. }
  100. if (id.input == "menu2") {
  101. md.lazy = false;
  102. md.setNotify([this]() { this->menu_choice(); });
  103. md.activate();
  104. return;
  105. }
  106. if (id.input.empty()) {
  107. // if (count >= 5) {
  108. auto lines = Boxes::alert(" Returning you to the game... ", "",
  109. "\x1b[1;32m", 30, 1, true);
  110. // I'm not setting the box color, so the last color bleeds over.
  111. to_client("\x1b[0m");
  112. for (auto line : lines) {
  113. to_client(line);
  114. };
  115. to_client("Returning you to the game...\n\r");
  116. deactivate();
  117. } else {
  118. // prompt it again, sam.
  119. id.setNotify([this]() { this->have_input(); });
  120. id.activate();
  121. }
  122. }
  123. void MainDispatch::menu_choice(void) {
  124. if (md.input.empty()) {
  125. to_client("Menu abort.\n\r");
  126. } else {
  127. std::string text = "Back from menu [";
  128. text.append(md.input);
  129. text.append("] was your selection.\n\r");
  130. to_client(text);
  131. }
  132. id.max_length = 15;
  133. id.setNotify([this]() { this->have_input(); });
  134. id.activate();
  135. }
  136. void MainDispatch::deactivate(void) {
  137. // Since we're the main thing there --
  138. // sess->show_client = true;
  139. // sess->talk_direct = true;
  140. notify();
  141. }
  142. void MainDispatch::server_line(const std::string &line, const std::string &raw_line) {
  143. BUGZ_LOG(info) << "MDSL: " << line;
  144. to_client("SL: ");
  145. to_client(line);
  146. to_client("\n\r");
  147. }
  148. void MainDispatch::server_prompt(const std::string &prompt) {
  149. BUGZ_LOG(info) << "MDSP: " << prompt;
  150. }
  151. #ifdef NEVERMORE
  152. void MainDispatch::client_input(const std::string &input) {
  153. // I don't care what the old prompt looked liked at this point.
  154. BUGZ_LOG(warning) << "Got: " << input; // << " prompt=" << get_prompt();
  155. // Serious problem when the input = "\x1b" ESC. The output gets gummed/locked
  156. // up.
  157. if (input == "\x1b") {
  158. return;
  159. }
  160. ++count;
  161. std::string output = str(boost::format("MSG %1%: [%2%]\n\r") % count % input);
  162. to_client(output);
  163. if (count >= 5) {
  164. to_client("And we're outta here!\n\r");
  165. deactivate();
  166. }
  167. }
  168. #endif
  169. InputDispatch::InputDispatch(Director &d) : Dispatch(d) {
  170. BUGZ_LOG(warning) << "InputDispatch()";
  171. }
  172. InputDispatch::~InputDispatch() { BUGZ_LOG(warning) << "~InputDispatch()"; }
  173. void InputDispatch::activate(void) {
  174. input.clear();
  175. to_client(prompt);
  176. }
  177. void InputDispatch::deactivate(void) { notify(); }
  178. void InputDispatch::server_line(const std::string &line, const std::string &raw_line) {
  179. // TO FIX:
  180. // clear user input and input prompt, display server_line,
  181. // and re-display prompt.
  182. }
  183. // void InputDispatch::server_prompt(const std::string &prompt) {}
  184. void InputDispatch::client_input(const std::string &cinput) {
  185. // BUGZ_LOG(info) << "InputDispatch::client_input(" << cinput << ")";
  186. for (const char ch : cinput) {
  187. if (isprint(ch)) {
  188. // Ok!
  189. if (input.length() < max_length) {
  190. to_client(std::string(1, ch));
  191. input += ch;
  192. }
  193. } else if ((ch == '\b') || (ch == 0x7f)) {
  194. // Backspace or rubout
  195. if (input.length() > 0) {
  196. to_client("\b \b");
  197. input.erase(input.size() - 1);
  198. }
  199. } else if (ch == '\r') {
  200. // Ok, we're done!
  201. BUGZ_LOG(info) << "InputDispatch done: " << input;
  202. to_client("\x1b[0m\n\r");
  203. deactivate();
  204. }
  205. }
  206. }
  207. /**
  208. * Menu Dispatch
  209. *
  210. * Two types of menus:
  211. * lazy: display the menu name and show prompt. ? shows menu.
  212. * non-lazy: displays menu + prompts.
  213. *
  214. *
  215. */
  216. MenuDispatch::MenuDispatch(Director &d) : Dispatch(d) {
  217. BUGZ_LOG(warning) << "MenuDispatch()";
  218. }
  219. MenuDispatch::~MenuDispatch() { BUGZ_LOG(warning) << "~MenuDispatch()"; }
  220. void MenuDispatch::activate(void) {
  221. calculate_widths();
  222. input.clear();
  223. BUGZ_LOG(warning) << "MenuDispatch::activate() " << max_width << ", "
  224. << max_option_width;
  225. if (lazy)
  226. menubox();
  227. else
  228. help();
  229. to_client(menu_prompt);
  230. }
  231. void MenuDispatch::deactivate(void) { notify(); }
  232. void MenuDispatch::help(void) {
  233. size_t max = max_width;
  234. Boxes mbox(max, 1, true);
  235. mbox.boxcolor = menu_box_color;
  236. if (lazy) {
  237. // just the menu
  238. mbox.textcolor = menu_options_color;
  239. to_client(mbox.top());
  240. for (auto const menu_item : menu) {
  241. std::string text = " ";
  242. text.append(menu_item.first);
  243. text.append(" - ");
  244. text.append(menu_item.second);
  245. while (text.length() < max) text.append(1, ' ');
  246. to_client(mbox.row(text));
  247. }
  248. to_client(mbox.bottom());
  249. } else {
  250. // full menu
  251. mbox.textcolor = menu_text_color;
  252. to_client(mbox.top());
  253. std::string title = centered(max, menu_title);
  254. BUGZ_LOG(debug) << "help max=" << max << " [" << title << "]";
  255. to_client(mbox.row(title));
  256. to_client(mbox.middle());
  257. mbox.textcolor = menu_options_color;
  258. for (auto const menu_item : menu) {
  259. std::string text = " ";
  260. text.append(menu_item.first);
  261. text.append(" - ");
  262. text.append(menu_item.second);
  263. while (text.length() < max) text.append(1, ' ');
  264. to_client(mbox.row(text));
  265. }
  266. to_client(mbox.bottom());
  267. }
  268. }
  269. std::string MenuDispatch::centered(int length, const std::string &s) {
  270. std::string text = s;
  271. size_t leftovers = length - text.length();
  272. int count = leftovers / 2;
  273. if (count > 0) {
  274. text.insert(0, count, ' ');
  275. text.append(count, ' ');
  276. }
  277. if (leftovers % 1 == 1) text.append(1, ' ');
  278. return text;
  279. }
  280. void MenuDispatch::menubox(void) {
  281. // just the menu box
  282. std::string title = centered(max_width, menu_title);
  283. /*
  284. int leftovers = max - menu_title.length();
  285. while (leftovers > 2) {
  286. title.insert(0, 1, " ");
  287. title.append(1, " ");
  288. leftovers -= 2;
  289. };
  290. if (leftovers == 1)
  291. title.append(1, " ");
  292. */
  293. BUGZ_LOG(debug) << "menubox max=" << max_width << " [" << title << "]";
  294. auto abox =
  295. Boxes::alert(title, menu_box_color, menu_text_color, max_width, 1, true);
  296. for (auto line : abox) {
  297. to_client(line);
  298. }
  299. }
  300. void MenuDispatch::calculate_widths(void) {
  301. max_width = menu_title.length() + 2;
  302. max_option_width = 0;
  303. for (auto key : menu) {
  304. size_t menu_line_length =
  305. 1 + key.first.length() + 3 + key.second.length() + 1;
  306. if (menu_line_length > max_width) max_width = menu_line_length;
  307. if (key.first.length() > max_option_width)
  308. max_option_width = key.first.length();
  309. }
  310. instant = max_option_width == 1;
  311. }
  312. void MenuDispatch::server_line(const std::string &line, const std::string &raw_line) {
  313. // TODO:
  314. // Clear prompt, display raw server line, restore prompt.
  315. }
  316. void MenuDispatch::client_input(const std::string &cinput) {
  317. for (auto const ch : cinput) {
  318. // not likely that we'd have more then one,
  319. // but deal with it correctly.
  320. if (ch == '\r') {
  321. // enter
  322. if (instant) return;
  323. for (auto const mnu : menu) {
  324. if (mnu.first == input) {
  325. to_client("\x1b[0m\n\r");
  326. deactivate();
  327. return;
  328. }
  329. }
  330. // input wasn't found ?
  331. while (input.length() > 0) {
  332. to_client("\b \b");
  333. input.erase(input.length() - 1);
  334. }
  335. return; // don't continue ...
  336. }
  337. if (ch == '\x1b') {
  338. // [ESC] - erase the input string
  339. while (input.length() > 0) {
  340. to_client("\b \b");
  341. input.erase(input.length() - 1);
  342. }
  343. // Exit - allow escape from menu
  344. deactivate();
  345. return;
  346. }
  347. if (ch == '\b') {
  348. if (input.length() > 0) {
  349. to_client("\b \b");
  350. input.erase(input.length() - 1);
  351. }
  352. }
  353. if (ch == '?') {
  354. to_client(cinput); // display what they entered.
  355. to_client("\x1b[0m\n\r");
  356. help();
  357. to_client(menu_prompt);
  358. return;
  359. }
  360. if (isprint(ch)) {
  361. char c = ch;
  362. if (!case_sensitive) c = toupper(ch);
  363. // ok, it's a printable character
  364. if (input.length() < max_option_width) {
  365. // ok, there's room.
  366. to_client(std::string(1, c));
  367. input.append(1, c);
  368. }
  369. if (instant) {
  370. for (auto const mnu : menu) {
  371. if (mnu.first == input) {
  372. to_client("\x1b[0m\n\r");
  373. deactivate();
  374. return;
  375. }
  376. }
  377. }
  378. }
  379. }
  380. }
  381. /*
  382. * CoreDispatch: This is an example class that does dispatch.
  383. * Copy this and make changes from there...
  384. */
  385. CoreDispatch::CoreDispatch(Director &d) : Dispatch(d) {
  386. BUGZ_LOG(warning) << "CoreDispatch()";
  387. }
  388. void CoreDispatch::activate(void) {
  389. // save things, set things
  390. }
  391. void CoreDispatch::deactivate(void) {
  392. // restore things
  393. notify();
  394. }
  395. void CoreDispatch::server_line(const std::string &line, const std::string &raw_line) {}
  396. void CoreDispatch::server_prompt(const std::string &prompt) {}
  397. void CoreDispatch::client_input(const std::string &input) {
  398. BUGZ_LOG(warning) << "Got: " << input << " prompt=" << get_prompt();
  399. }