main.cpp 15 KB

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