main.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598
  1. #include "door.h"
  2. #include "yaml-cpp/yaml.h"
  3. #include <chrono> // chrono::system_clock
  4. #include <ctime> // localtime
  5. #include <iomanip> // put_time
  6. #include <iostream>
  7. #include <random>
  8. #include <string>
  9. #include "db.h"
  10. #include "deck.h"
  11. #include "play.h"
  12. #include "scores.h"
  13. #include "utils.h"
  14. #include "version.h"
  15. #include "starfield.h"
  16. #include <algorithm> // transform
  17. // configuration here -- access via extern
  18. YAML::Node config;
  19. door::ANSIColor stringToANSIColor(std::string colorCode);
  20. std::function<std::ofstream &(void)> get_logger;
  21. std::function<void(void)> cls_display_starfield;
  22. std::function<int(void)> press_a_key;
  23. std::string return_current_time_and_date() {
  24. auto now = std::chrono::system_clock::now();
  25. auto in_time_t = std::chrono::system_clock::to_time_t(now);
  26. std::stringstream ss;
  27. // ss << std::put_time(std::localtime(&in_time_t), "%Y-%m-%d %X");
  28. ss << std::put_time(std::localtime(&in_time_t), "%Y-%m-%d %r");
  29. return ss.str();
  30. }
  31. door::renderFunction any_color = [](const std::string &txt) -> door::Render {
  32. door::Render r(txt);
  33. door::ANSIColor blue(door::COLOR::GREEN, door::ATTR::BOLD);
  34. door::ANSIColor cyan(door::COLOR::YELLOW, door::ATTR::BOLD);
  35. for (char const &c : txt) {
  36. if (isupper(c))
  37. r.append(blue);
  38. else
  39. r.append(cyan);
  40. }
  41. return r;
  42. };
  43. int press_any_key(door::Door &door) {
  44. static std::default_random_engine generator;
  45. static std::uniform_int_distribution<int> distribution(0, 1);
  46. int use_set = distribution(generator);
  47. std::string text = "Press a key to continue...";
  48. door << door::reset;
  49. any_color(text).output(door);
  50. // << text;
  51. int r;
  52. int t = 0;
  53. // alternate animation
  54. #define ALT_ANIMATION
  55. #ifdef ALT_ANIMATION
  56. std::vector<std::pair<int, int>> words = find_words(text);
  57. std::string work_text = text;
  58. int text_length = text.size();
  59. // current word
  60. unsigned int word = 0;
  61. std::string current_word = text.substr(words[word].first, words[word].second);
  62. unsigned int wpos = 0;
  63. int ms_sleep = 0;
  64. int sleep_ms = 250;
  65. t = 0;
  66. while ((r = door.sleep_ms_key(sleep_ms)) == TIMEOUT) {
  67. ms_sleep += sleep_ms;
  68. if (ms_sleep > 1000) {
  69. ms_sleep -= 1000;
  70. t++;
  71. if (t >= door.inactivity) {
  72. // restore text
  73. door << std::string(text_length, '\b');
  74. any_color(text).output(door);
  75. // << text;
  76. door << door::nl;
  77. return TIMEOUT;
  78. }
  79. }
  80. ++wpos;
  81. if (wpos == current_word.size()) {
  82. ++word;
  83. work_text = text;
  84. wpos = 0;
  85. if (word == words.size())
  86. word = 0;
  87. current_word = text.substr(words[word].first, words[word].second);
  88. }
  89. int c = current_word[wpos];
  90. while (!std::islower(c)) {
  91. // It is not lower case
  92. wpos++;
  93. if (wpos == current_word.size()) {
  94. // Ok, this word is done.
  95. wpos = 0;
  96. word++;
  97. work_text = text;
  98. if (word == words.size())
  99. word = 0;
  100. current_word = text.substr(words[word].first, words[word].second);
  101. wpos = 0;
  102. }
  103. c = current_word[wpos];
  104. }
  105. // Ok, I found something that's lower case.
  106. work_text[words[word].first + wpos] = std::toupper(c);
  107. // std::toupper(words[word].first + wpos);
  108. door << std::string(text_length, '\b');
  109. any_color(work_text).output(door);
  110. // << work_text;
  111. }
  112. // ok, restore the original string
  113. door << std::string(text_length, '\b');
  114. any_color(text).output(door);
  115. //<< text;
  116. return r;
  117. #endif
  118. #ifdef ALT_ANIMATION_2
  119. // This can be cleaned up to be simpler -- and do the same effect. We don't
  120. // need the words here, we're not really using them.
  121. std::vector<std::pair<int, int>> words = find_words(text);
  122. std::string work_text = text;
  123. int text_length = text.size();
  124. // current word
  125. unsigned int word = 0;
  126. std::string current_word = text.substr(words[word].first, words[word].second);
  127. unsigned int wpos = 0;
  128. // DOES NOT HONOR door.inactivity (AT THIS TIME)
  129. while ((r = door.sleep_ms_key(200)) == TIMEOUT) {
  130. ++wpos;
  131. if (wpos == current_word.size()) {
  132. ++word;
  133. wpos = 0;
  134. if (word == words.size())
  135. word = 0;
  136. current_word = text.substr(words[word].first, words[word].second);
  137. }
  138. int c = current_word[wpos];
  139. while (!std::islower(c)) {
  140. // It is not lower case
  141. wpos++;
  142. if (wpos == current_word.size()) {
  143. // Ok, this word is done.
  144. wpos = 0;
  145. word++;
  146. if (word == words.size())
  147. word = 0;
  148. current_word = text.substr(words[word].first, words[word].second);
  149. wpos = 0;
  150. }
  151. c = current_word[wpos];
  152. }
  153. // Ok, I found something that's lower case.
  154. work_text = text;
  155. work_text[words[word].first + wpos] = std::toupper(c);
  156. // std::toupper(words[word].first + wpos);
  157. door << std::string(text_length, '\b') << work_text;
  158. }
  159. // ok, restore the original string
  160. door << std::string(text_length, '\b') << text;
  161. return r;
  162. #endif
  163. // animation spinner that reverses itself. /-\| and blocks.
  164. t = 0;
  165. int loop = 0;
  166. const char *loop_chars[3][4] = {{"/", "-", "\\", "|"},
  167. {"\xde", "\xdc", "\xdd", "\xdf"},
  168. {"\u2590", "\u2584", "\u258c", "\u2580"}};
  169. std::array<int, 11> sleeps = {1000, 700, 500, 300, 200, 200,
  170. 200, 300, 500, 700, 1000};
  171. // start out at
  172. int sleep_number = 5;
  173. bool forward = true;
  174. if (door::unicode) {
  175. if (use_set == 1)
  176. use_set = 2;
  177. }
  178. door << loop_chars[use_set][loop % 4];
  179. ms_sleep = 0;
  180. bool check_speed = false;
  181. while ((r = door.sleep_ms_key(sleeps[sleep_number])) == TIMEOUT) {
  182. ms_sleep += sleeps[sleep_number];
  183. if (ms_sleep > 1000) {
  184. ms_sleep -= 1000;
  185. t++;
  186. if (t >= door.inactivity) {
  187. door << "\b \b";
  188. door << door::nl;
  189. return TIMEOUT;
  190. }
  191. }
  192. if (forward) {
  193. loop++;
  194. if (loop == 4) {
  195. loop = 0;
  196. check_speed = true;
  197. }
  198. } else {
  199. --loop;
  200. if (loop < 0) {
  201. loop = 3;
  202. check_speed = true;
  203. }
  204. }
  205. if (check_speed) {
  206. sleep_number++;
  207. if (sleep_number >= (int)sleeps.size()) {
  208. sleep_number = 0;
  209. forward = !forward;
  210. }
  211. check_speed = false;
  212. }
  213. door << "\b" << loop_chars[use_set][loop];
  214. }
  215. door << "\b \b";
  216. door << door::nl;
  217. return r;
  218. }
  219. // This does not seem to be working. I keep getting zero.
  220. int opt_from_string(std::string colorCode) {
  221. for (std::size_t pos = 0; pos != deck_colors.size(); ++pos) {
  222. // if (caseInsensitiveStringCompare(colorCode, deck_colors[pos]) == 0) {
  223. if (iequals(colorCode, deck_colors[pos])) {
  224. return pos;
  225. }
  226. }
  227. return 0;
  228. }
  229. int configure(door::Door &door, DBData &db) {
  230. // pre-warm the parser
  231. find_words(std::string(""));
  232. auto menu = make_config_menu();
  233. int r = 0;
  234. bool save_deckcolor = false;
  235. const char *deckcolor = "DeckColor";
  236. std::string newColor;
  237. while (r >= 0) {
  238. if (save_deckcolor) {
  239. door << menu;
  240. db.setSetting(deckcolor, newColor);
  241. save_deckcolor = false;
  242. }
  243. if (cls_display_starfield)
  244. cls_display_starfield();
  245. else
  246. door << door::reset << door::cls;
  247. r = menu.choose(door);
  248. if (r > 0) {
  249. door << door::reset << door::cls;
  250. char c = menu.which(r - 1);
  251. if (c == 'D') {
  252. // Ok, deck colors
  253. // get default
  254. std::string currentDefault = db.getSetting(deckcolor, "ALL");
  255. int currentOpt = opt_from_string(currentDefault);
  256. door << door::reset << door::cls;
  257. auto deck = make_deck_menu();
  258. deck.defaultSelection(currentOpt);
  259. if (cls_display_starfield)
  260. cls_display_starfield();
  261. else
  262. door << door::reset << door::cls;
  263. int newOpt = deck.choose(door);
  264. door << door::reset << door::cls;
  265. if (newOpt >= 0) {
  266. newOpt--;
  267. newColor = stringFromColorOptions(newOpt);
  268. if (newOpt != currentOpt) {
  269. door.log() << deckcolor << " was " << currentDefault << ", "
  270. << currentOpt << ". Now " << newColor << ", " << newOpt
  271. << std::endl;
  272. save_deckcolor = true;
  273. }
  274. }
  275. }
  276. if (c == 'V') {
  277. // view settings -- Sysop Configuration
  278. if (cls_display_starfield)
  279. cls_display_starfield();
  280. else
  281. door << door::reset << door::cls;
  282. door::Panel config_panel = make_sysop_config();
  283. // config_panel.set(1, 1);
  284. door << config_panel << door::reset << door::nl;
  285. r = press_a_key();
  286. if (r < 0)
  287. return r;
  288. door << door::reset << door::cls;
  289. }
  290. if (c == 'Q') {
  291. return r;
  292. }
  293. }
  294. }
  295. return r;
  296. }
  297. int main(int argc, char *argv[]) {
  298. door::Door door("space-ace", argc, argv);
  299. // store the door log so we can easily access it.
  300. get_logger = [&door]() -> ofstream & { return door.log(); };
  301. press_a_key = [&door]() -> int { return press_any_key(door); };
  302. std::random_device rd;
  303. std::mt19937 rng(rd());
  304. starfield mainfield(door, rng);
  305. cls_display_starfield = [&mainfield]() -> void {
  306. mainfield.display();
  307. //display_starfield(door, rng);
  308. };
  309. DBData spacedb;
  310. spacedb.setUser(door.username);
  311. if (file_exists("space-ace.yaml")) {
  312. config = YAML::LoadFile("space-ace.yaml");
  313. }
  314. bool update_config = false;
  315. // populate with "good" defaults
  316. if (!config["hands_per_day"]) {
  317. config["hands_per_day"] = 3;
  318. update_config = true;
  319. }
  320. if (!config["date_format"]) {
  321. config["date_format"] = "%B %d"; // Month day or "%b %d,%Y" Mon,d YYYY
  322. update_config = true;
  323. }
  324. if (!config["date_score"]) {
  325. config["date_score"] = "%m/%d/%Y"; // or "%Y/%0m/%0d";
  326. update_config = true;
  327. }
  328. if (!config["makeup_per_day"]) {
  329. config["makeup_per_day"] = 5;
  330. update_config = true;
  331. }
  332. if (!config["play_days_ahead"]) {
  333. config["play_days_ahead"] = 2;
  334. update_config = true;
  335. }
  336. if (!config["date_monthly"]) {
  337. config["date_monthly"] = "%B %Y";
  338. update_config = true;
  339. }
  340. if (!config["_seed"]) {
  341. // pick numbers to use for seed
  342. std::string seeds;
  343. seeds += std::to_string(rng());
  344. seeds += ",";
  345. seeds += std::to_string(rng());
  346. seeds += ",";
  347. seeds += std::to_string(rng());
  348. config["_seed"] = seeds;
  349. update_config = true;
  350. }
  351. // save configuration -- something was updated
  352. if (update_config) {
  353. std::ofstream fout("space-ace.yaml");
  354. fout << config << std::endl;
  355. }
  356. // retrieve lastcall
  357. time_t last_call = std::stol(spacedb.getSetting("LastCall", "0"));
  358. std::chrono::_V2::system_clock::time_point now =
  359. std::chrono::system_clock::now();
  360. // store now as lastcall
  361. time_t now_t = std::chrono::system_clock::to_time_t(now);
  362. spacedb.setSetting("LastCall", std::to_string(now_t));
  363. // run maint
  364. {
  365. std::chrono::_V2::system_clock::time_point maint_date = now;
  366. firstOfMonthDate(maint_date);
  367. if (spacedb.expireScores(
  368. std::chrono::system_clock::to_time_t(maint_date))) {
  369. if (get_logger)
  370. get_logger() << "maint completed" << std::endl;
  371. door << "Thanks for waiting..." << door::nl;
  372. }
  373. }
  374. // Have they used this door before?
  375. if (last_call != 0) {
  376. door << door::ANSIColor(door::COLOR::YELLOW, door::ATTR::BOLD)
  377. << "Welcome Back!" << door::nl;
  378. auto nowClock = std::chrono::system_clock::from_time_t(now_t);
  379. auto lastClock = std::chrono::system_clock::from_time_t(last_call);
  380. auto delta = nowClock - lastClock;
  381. // int days = chrono::duration_cast<chrono::days>(delta).count(); //
  382. // c++ 20
  383. int hours = chrono::duration_cast<chrono::hours>(delta).count();
  384. int days = hours / 24;
  385. int minutes = chrono::duration_cast<chrono::minutes>(delta).count();
  386. int secs = chrono::duration_cast<chrono::seconds>(delta).count();
  387. if (days > 1) {
  388. door << door::ANSIColor(door::COLOR::GREEN, door::ATTR::BOLD)
  389. << "It's been " << days << " days since you last played."
  390. << door::nl;
  391. } else {
  392. if (hours > 1) {
  393. door << door::ANSIColor(door::COLOR::CYAN) << "It's been " << hours
  394. << " hours since you last played." << door::nl;
  395. } else {
  396. if (minutes > 1) {
  397. door << door::ANSIColor(door::COLOR::CYAN) << "It's been " << minutes
  398. << " minutes since you last played." << door::nl;
  399. } else {
  400. door << door::ANSIColor(door::COLOR::YELLOW, door::ATTR::BOLD)
  401. << "It's been " << secs << " seconds since you last played."
  402. << door::nl;
  403. door << "It's like you never left." << door::nl;
  404. }
  405. }
  406. }
  407. if (press_a_key() < 0)
  408. return 0;
  409. }
  410. int mx, my; // Max screen width/height
  411. if (door.width == 0) {
  412. // screen detect failed, use sensible defaults
  413. door.width = mx = 80;
  414. door.height = my = 23;
  415. } else {
  416. mx = door.width;
  417. my = door.height;
  418. }
  419. // We assume here that the width and height aren't something crazy like
  420. // 10x15. :P (or 24x923!)
  421. display_starfield_space_ace(door, rng);
  422. // for testing inactivity timeout
  423. // door.inactivity = 10;
  424. door::Panel timeout = make_timeout(mx, my);
  425. door::Menu m = make_main_menu();
  426. door::Panel about = make_about(); // 8 lines
  427. door::Panel help = make_help();
  428. // center the about box
  429. about.set((mx - 60) / 2, (my - 9) / 2);
  430. // center the help box
  431. help.set((mx - 60) / 2, (my - 15) / 2);
  432. PlayCards pc(door, spacedb, rng);
  433. int r = 0;
  434. while ((r >= 0) and (r != 6)) {
  435. // starfield + menu ?
  436. display_starfield(door, rng);
  437. r = m.choose(door);
  438. // need to reset the colors. (whoops!)
  439. door << door::reset << door::cls; // door::nl;
  440. // OK! The screen is blank at this point!
  441. switch (r) {
  442. case 1: // play game
  443. {
  444. // PlayCards pc(door, spacedb, rng);
  445. r = pc.play();
  446. // r = play_cards(door, spacedb, rng);
  447. }; break;
  448. case 2: // view scores
  449. // door << door::cls;
  450. {
  451. Scores score(door, spacedb);
  452. score.display_scores(door);
  453. }
  454. r = press_a_key();
  455. break;
  456. case 3: // configure
  457. r = configure(door, spacedb);
  458. // r = press_a_key();
  459. break;
  460. case 4: // help
  461. display_starfield(door, rng);
  462. door << help << door::nl;
  463. r = press_a_key();
  464. break;
  465. case 5: // about
  466. display_starfield(door, rng);
  467. door << about << door::nl;
  468. r = press_a_key();
  469. break;
  470. case 6: // quit
  471. break;
  472. }
  473. }
  474. if (r < 0) {
  475. // DOOR_ERROR:
  476. if (r == TIMEOUT) {
  477. door.log() << "TIMEOUT" << std::endl;
  478. door << timeout << door::reset << door::nl << door::nl;
  479. } else {
  480. if (r == OUTOFTIME) {
  481. door.log() << "OUTTA TIME" << std::endl;
  482. door::Panel notime = make_notime(mx, my);
  483. door << notime << door::reset << door::nl;
  484. }
  485. }
  486. return 0;
  487. }
  488. door << door::nl;
  489. // door << door::reset << door::cls;
  490. display_starfield(door, rng);
  491. door << m << door::reset << door::nl;
  492. // Normal DOOR exit goes here...
  493. door << door::nl << "Returning you to the BBS, please wait..." << door::nl;
  494. press_a_key = nullptr;
  495. get_logger = nullptr;
  496. cls_display_starfield = nullptr;
  497. return 0;
  498. }