play.cpp 53 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698
  1. #include "play.h"
  2. #include "db.h"
  3. #include "deck.h"
  4. #include "utils.h"
  5. #include "version.h"
  6. #include <ctime>
  7. #include <iomanip> // put_time
  8. #include <sstream>
  9. /**
  10. * @brief This config option allow any card to be played on any other card.
  11. *
  12. * This is for testing BONUS and scoring, etc. This disables saving of streak.
  13. *
  14. */
  15. #define CHEATER "_CHEAT_YOUR_ASS_OFF"
  16. PlayCards::PlayCards(door::Door &d, DBData &dbd, std::mt19937 &r)
  17. : door{d}, db{dbd}, rng{r} {
  18. init_values();
  19. play_day = std::chrono::system_clock::now();
  20. normalizeDate(play_day);
  21. time_t play_day_t = std::chrono::system_clock::to_time_t(play_day);
  22. // check last_played
  23. time_t last_played;
  24. {
  25. std::string last_played_str = dbd.getSetting("last_played", "0");
  26. last_played = std::stoi(last_played_str);
  27. std::string days_played_str = dbd.getSetting("days_played", "0");
  28. days_played = std::stoi(days_played_str);
  29. }
  30. if (last_played != play_day_t) {
  31. // Ok, they haven't played today, so:
  32. get_logger() << "reset days_played = 0" << std::endl;
  33. dbd.setSetting("last_played", std::to_string(play_day_t));
  34. dbd.setSetting("days_played", std::to_string(0));
  35. days_played = 0;
  36. }
  37. {
  38. std::string seed = config["_seed"].as<std::string>();
  39. if (get_logger) {
  40. get_logger() << "seed: " << seed << std::endl;
  41. }
  42. std::vector<std::string> parts = split(seed, ',');
  43. for (auto &p : parts) {
  44. seeds.push_back(std::stol(p));
  45. if (get_logger) {
  46. get_logger() << "seed " << p << std::endl;
  47. }
  48. }
  49. }
  50. /*
  51. * TODO: Check the date with the db. Have they already played up today? If
  52. * so, display calendar and find a day they can play. Or, play missed hands
  53. * for today.
  54. */
  55. hand = 0;
  56. total_hands = 0;
  57. spaceAceTriPeaks = make_tripeaks();
  58. score_panel = make_score_panel();
  59. streak_panel = make_streak_panel();
  60. left_panel = make_left_panel();
  61. cmd_panel = make_command_panel();
  62. next_quit_panel = make_next_panel();
  63. calendar = make_calendar();
  64. /*
  65. int mx = door.width;
  66. int my = door.height;
  67. */
  68. }
  69. PlayCards::~PlayCards() {}
  70. void PlayCards::init_values(void) {
  71. // beware of hand=1 ! We might not be playing the first hand here!
  72. // hand = 1;
  73. if (config["hands_per_day"]) {
  74. total_hands = config["hands_per_day"].as<int>();
  75. } else
  76. total_hands = 3;
  77. play_card = 28;
  78. current_streak = 0;
  79. best_streak = 0;
  80. {
  81. std::string best;
  82. best = db.getSetting("best_streak", "0");
  83. best_streak = std::stoi(best);
  84. }
  85. select_card = 23;
  86. score = 0;
  87. }
  88. /**
  89. * @brief Display the bonus text, when you remove a peak.
  90. *
  91. */
  92. void PlayCards::bonus(void) {
  93. door << door::ANSIColor(door::COLOR::YELLOW, door::ATTR::BOLD) << "BONUS";
  94. }
  95. int PlayCards::press_a_key(void) {
  96. door << door::reset << "Press a key to continue...";
  97. int r = door.sleep_key(door.inactivity);
  98. door << door::nl;
  99. return r;
  100. }
  101. /**
  102. * @brief Play
  103. *
  104. * This will display the calendar (if needed), otherwise takes player into
  105. * play_cards using now() as play_day.
  106. * \sa PlayCards::play_cards()
  107. *
  108. * @return int
  109. */
  110. int PlayCards::play(void) {
  111. play_day = std::chrono::system_clock::now();
  112. normalizeDate(play_day);
  113. if (config["hands_per_day"]) {
  114. total_hands = config["hands_per_day"].as<int>();
  115. } else
  116. total_hands = 3;
  117. // check to see if played already today
  118. time_t play_day_t = std::chrono::system_clock::to_time_t(play_day);
  119. int played = db.handsPlayedOnDay(play_day_t);
  120. #ifdef DEBUG_OUTPUT
  121. if (get_logger) {
  122. get_logger() << "played today (" << play_day_t << ")= " << played
  123. << std::endl;
  124. }
  125. #endif
  126. int r;
  127. if (played == 0) {
  128. // playing today
  129. door << "Let's play today..." << door::nl;
  130. std::this_thread::sleep_for(std::chrono::seconds(1));
  131. hand = 1;
  132. r = play_cards();
  133. if (r != 'D')
  134. return r;
  135. } else {
  136. if (played < total_hands) {
  137. door << "Let's finish today..." << door::nl;
  138. std::this_thread::sleep_for(std::chrono::seconds(1));
  139. hand = played + 1;
  140. r = play_cards();
  141. if (r != 'D')
  142. return r;
  143. }
  144. }
  145. // Ok, we need to select a day.
  146. // calendar = make_calendar();
  147. // if (false) {
  148. CALENDAR_UPDATE:
  149. if (get_logger) {
  150. get_logger() << "update calendar days" << std::endl;
  151. }
  152. update_calendar_days();
  153. calendar->update();
  154. //}
  155. if (cls_display_starfield)
  156. cls_display_starfield();
  157. else
  158. door << door::reset << door::cls;
  159. door << *calendar;
  160. bool has_playable_day = false;
  161. for (int x = 0; x < 31; ++x) {
  162. int status = calendar_day_status[x];
  163. if ((status == 0) or (status == 1)) {
  164. has_playable_day = true;
  165. break;
  166. }
  167. }
  168. if (!has_playable_day) {
  169. door << door::nl;
  170. door << "Sorry, there are no days available to play." << door::nl;
  171. r = press_a_key();
  172. if (r < 0)
  173. return r;
  174. return 'Q';
  175. }
  176. door << door::nl;
  177. door << "Please choose day : " << door::SaveCursor;
  178. AGAIN:
  179. std::string toplay = door.input_string(3);
  180. door << door::RestoreCursor;
  181. int number;
  182. try {
  183. number = std::stoi(toplay);
  184. } catch (std::exception &e) {
  185. number = 0;
  186. }
  187. if (number == 0)
  188. return ' ';
  189. int status;
  190. if (number <= 31) {
  191. status = calendar_day_status[number - 1];
  192. if (status == 0) {
  193. // play full day -- how do I figure out the date for this?
  194. hand = 1;
  195. play_day_t = calendar_day_t[number - 1];
  196. play_day = std::chrono::system_clock::from_time_t(play_day_t);
  197. r = play_cards();
  198. if (r == 'D')
  199. goto CALENDAR_UPDATE;
  200. return r;
  201. }
  202. if (status == 1) {
  203. // play half day
  204. play_day_t = calendar_day_t[number - 1];
  205. play_day = std::chrono::system_clock::from_time_t(play_day_t);
  206. played = db.handsPlayedOnDay(play_day_t);
  207. if (played < total_hands) {
  208. hand = played + 1;
  209. r = play_cards();
  210. if (r == 'D')
  211. goto CALENDAR_UPDATE;
  212. return r;
  213. }
  214. }
  215. goto AGAIN;
  216. };
  217. return ' ';
  218. }
  219. /**
  220. * @brief play_cards
  221. *
  222. * Play cards for the given play_day and hand.
  223. *
  224. * This should be called by play with the correct #play_day set.
  225. * \sa PlayCards::play()
  226. *
  227. * @return int
  228. */
  229. int PlayCards::play_cards(void) {
  230. init_values();
  231. // Calculate the game size
  232. int game_width;
  233. int game_height = 20;
  234. {
  235. int cx, cy;
  236. cardPos(27, cx, cy);
  237. game_width = cx + 5;
  238. }
  239. int mx = door.width;
  240. int my = door.height;
  241. off_x = (mx - game_width) / 2;
  242. off_y = (my - game_height) / 2;
  243. // int true_off_y = off_y;
  244. // we can now position things properly centered
  245. int tp_off_x = (mx - spaceAceTriPeaks->getWidth()) / 2;
  246. spaceAceTriPeaks->set(tp_off_x, off_y);
  247. off_y += 3; // adjust for tripeaks panel
  248. std::string currentDefault = db.getSetting("DeckColor", "ALL");
  249. next_hand:
  250. // Make sure we pick the deck color here. We want it to (possibly) change
  251. // between hands.
  252. get_logger() << "DeckColor shows as " << currentDefault << std::endl;
  253. deck_color = stringToANSIColor(currentDefault);
  254. dp.setBack(deck_color);
  255. // dp = Deck(deck_color);
  256. play_card = 28;
  257. select_card = 23;
  258. score = 0;
  259. current_streak = 0;
  260. // Use play_day to seed the rng
  261. {
  262. time_t tt = std::chrono::system_clock::to_time_t(play_day);
  263. tm local_tm = *localtime(&tt);
  264. std::vector<int> rand_seed_seq = seeds;
  265. rand_seed_seq.push_back(local_tm.tm_year + 1900);
  266. rand_seed_seq.push_back(local_tm.tm_mon + 1);
  267. rand_seed_seq.push_back(local_tm.tm_mday);
  268. rand_seed_seq.push_back(hand);
  269. // std::seed_seq seq{local_tm.tm_year + 1900, local_tm.tm_mon + 1,
  270. // local_tm.tm_mday, hand};
  271. std::seed_seq seq(rand_seed_seq.begin(), rand_seed_seq.end());
  272. deck = shuffleCards(seq, 1);
  273. state = makeCardStates();
  274. }
  275. /*
  276. door::Panel score_panel = make_score_panel();
  277. door::Panel streak_panel = make_streak_panel();
  278. door::Panel left_panel = make_left_panel();
  279. door::Panel cmd_panel = make_command_panel();
  280. */
  281. {
  282. int off_yp = off_y + 11;
  283. int cx, cy;
  284. int left_panel_x, right_panel_x;
  285. // find position of card, to position the panels
  286. cardPos(18, cx, cy);
  287. left_panel_x = cx;
  288. cardPos(15, cx, cy);
  289. right_panel_x = cx;
  290. score_panel->set(left_panel_x + off_x, off_yp);
  291. streak_panel->set(right_panel_x + off_x, off_yp);
  292. cmd_panel->set(left_panel_x + off_x, off_yp + 5);
  293. // next panel position (top = card 10, centered)
  294. cardPos(10, cx, cy);
  295. int next_off_x = (mx - next_quit_panel->getWidth()) / 2;
  296. next_quit_panel->set(next_off_x, cy + off_y);
  297. }
  298. bool dealing = true;
  299. int r = 0;
  300. redraw(dealing);
  301. dealing = false;
  302. left_panel->update(door);
  303. door << door::reset;
  304. shared_panel c;
  305. bool in_game = true;
  306. bool save_streak = false;
  307. while (in_game) {
  308. // time might have updated, so update score panel too.
  309. score_panel->update(door);
  310. // do the save here -- try to hide the database lag
  311. if (save_streak) {
  312. save_streak = false;
  313. std::string best = std::to_string(best_streak);
  314. db.setSetting("best_streak", best);
  315. }
  316. r = door.sleep_key(door.inactivity);
  317. if (r > 0) {
  318. // not a timeout or expire.
  319. if (r < 0x1000) // not a function key
  320. r = std::toupper(r);
  321. switch (r) {
  322. case '\x0d':
  323. if (current_streak > best_streak) {
  324. best_streak = current_streak;
  325. if (!config[CHEATER]) {
  326. save_streak = true;
  327. }
  328. streak_panel->update(door);
  329. }
  330. if (play_card < 51) {
  331. play_card++;
  332. current_streak = 0;
  333. streak_panel->update(door);
  334. // update the cards left_panel
  335. left_panel->update(door);
  336. // Ok, deal next card from the pile.
  337. int cx, cy, level;
  338. if (play_card == 51) {
  339. cardPosLevel(29, cx, cy, level);
  340. level = 0; // out of cards
  341. c = dp.back(level);
  342. c->set(cx + off_x, cy + off_y);
  343. door << *c;
  344. }
  345. cardPos(28, cx, cy);
  346. c = dp.card(deck.at(play_card));
  347. c->set(cx + off_x, cy + off_y);
  348. door << *c;
  349. }
  350. break;
  351. case 'R':
  352. redraw(false);
  353. break;
  354. case 'Q':
  355. // possibly prompt here for [N]ext hand or [Q]uit ?
  356. if (current_streak > best_streak) {
  357. best_streak = current_streak;
  358. if (!config[CHEATER]) {
  359. save_streak = true;
  360. }
  361. streak_panel->update(door);
  362. }
  363. next_quit_panel->update();
  364. door << *next_quit_panel;
  365. // use some other variable here for what we get from the get_one_of.
  366. if (hand < total_hands) {
  367. r = door.get_one_of("CNQ");
  368. } else {
  369. r = door.get_one_of("CDQ");
  370. }
  371. if (r == 'C') {
  372. // continue
  373. redraw(false);
  374. break;
  375. }
  376. if ((r == 'D') or (r == 'Q')) {
  377. // Ok, we are calling it quits.
  378. // save score if > 0
  379. if (score >= 50) {
  380. time_t right_now = std::chrono::system_clock::to_time_t(
  381. std::chrono::system_clock::now());
  382. db.saveScore(right_now,
  383. std::chrono::system_clock::to_time_t(play_day), hand,
  384. 0, score);
  385. }
  386. in_game = false;
  387. // r = 'Q';
  388. }
  389. if (r == 'N') {
  390. // no. If you want to play the next hand, we have to save this.
  391. get_logger() << "SCORE: " << score << std::endl;
  392. time_t right_now = std::chrono::system_clock::to_time_t(
  393. std::chrono::system_clock::now());
  394. db.saveScore(right_now,
  395. std::chrono::system_clock::to_time_t(play_day), hand, 0,
  396. score);
  397. hand++;
  398. goto next_hand;
  399. }
  400. // in_game = false;
  401. break;
  402. case ' ':
  403. case '5':
  404. // can we play this card?
  405. /*
  406. get_logger() << "canPlay( " << select_card << ":"
  407. << deck1.at(select_card) << "/"
  408. << d.getRank(deck1.at(select_card)) << " , "
  409. << play_card << "/" <<
  410. d.getRank(deck1.at(play_card))
  411. << ") = "
  412. << d.canPlay(deck1.at(select_card),
  413. deck1.at(play_card))
  414. << std::endl;
  415. */
  416. if (dp.canPlay(deck.at(select_card), deck.at(play_card)) or
  417. config[CHEATER]) {
  418. // if (true) {
  419. // yes we can.
  420. ++current_streak;
  421. // update best_streak when they draw the next card.
  422. /*
  423. if (current_streak > best_streak) {
  424. best_streak = current_streak;
  425. if (!config[CHEATER]) {
  426. save_streak = true;
  427. }
  428. }
  429. */
  430. streak_panel->update(door);
  431. score += 10;
  432. if (current_streak > 1)
  433. score += current_streak * 5;
  434. score_panel->update(door);
  435. /*
  436. if (get_logger)
  437. get_logger() << "score_panel update : " << score << std::endl;
  438. */
  439. // play card!
  440. state.at(select_card) = 2;
  441. {
  442. // swap the select card with play_card
  443. int temp = deck.at(select_card);
  444. deck.at(select_card) = deck.at(play_card);
  445. deck.at(play_card) = temp;
  446. // select_card is -- invalidated here! find "new" card.
  447. int cx, cy;
  448. // erase/clear select_card
  449. std::vector<int> check = dp.unblocks(select_card);
  450. bool left = false, right = false;
  451. for (const int c : check) {
  452. std::pair<int, int> blockers = dp.blocks[c];
  453. if (blockers.first == select_card)
  454. right = true;
  455. if (blockers.second == select_card)
  456. left = true;
  457. }
  458. dp.removeCard(door, select_card, off_x, off_y, left, right);
  459. /* // old way of doing this that leaves holes.
  460. cardPosLevel(select_card, cx, cy, level);
  461. c = d.back(0);
  462. c->set(cx + off_x, cy + off_y);
  463. door << *c;
  464. */
  465. // redraw play card #28. (Which is the "old" select_card)
  466. cardPos(28, cx, cy);
  467. c = dp.card(deck.at(play_card));
  468. c->set(cx + off_x, cy + off_y);
  469. door << *c;
  470. // Did we unhide a card here?
  471. // std::vector<int> check = d.unblocks(select_card);
  472. /*
  473. get_logger() << "select_card = " << select_card
  474. << " unblocks: " << check.size() << std::endl;
  475. */
  476. int new_card_shown = -1;
  477. if (!check.empty()) {
  478. for (const int to_check : check) {
  479. std::pair<int, int> blockers = dp.blocks[to_check];
  480. /*
  481. get_logger()
  482. << "Check: " << to_check << " " << blockers.first << ","
  483. << blockers.second << " " << state.at(blockers.first)
  484. << "," << state.at(blockers.second) << std::endl;
  485. */
  486. if ((state.at(blockers.first) == 2) and
  487. (state.at(blockers.second) == 2)) {
  488. // WOOT! Card uncovered.
  489. /*
  490. get_logger() << "showing: " << to_check << std::endl;
  491. */
  492. state.at(to_check) = 1;
  493. cardPos(to_check, cx, cy);
  494. c = dp.card(deck.at(to_check));
  495. c->set(cx + off_x, cy + off_y);
  496. door << *c;
  497. new_card_shown = to_check;
  498. }
  499. }
  500. } else {
  501. // top card cleared
  502. // get_logger() << "top card cleared?" << std::endl;
  503. // display something at select_card position
  504. int cx, cy;
  505. cardPos(select_card, cx, cy);
  506. door << door::Goto(cx + off_x, cy + off_y);
  507. bonus();
  508. score += 100;
  509. state.at(select_card) = 3; // handle this in the "redraw"
  510. score_panel->update(door);
  511. }
  512. // Find new "number" for select_card to be.
  513. if (new_card_shown != -1) {
  514. select_card = new_card_shown;
  515. } else {
  516. // select_card++;
  517. int new_select = findClosestActiveCard(state, select_card);
  518. if (new_select != -1) {
  519. select_card = new_select;
  520. } else {
  521. get_logger() << "Winner!" << std::endl;
  522. select_card = -1; // winner marker?
  523. // bonus for cards left
  524. int bonus = 15 * (51 - play_card);
  525. score += 15 * (51 - play_card);
  526. score_panel->update(door);
  527. // maybe display this somewhere?
  528. // door << " BONUS: " << bonus << door::nl;
  529. get_logger()
  530. << "SCORE: " << score << ", " << bonus << std::endl;
  531. // if (!config[CHEATER]) {
  532. time_t right_now = std::chrono::system_clock::to_time_t(
  533. std::chrono::system_clock::now());
  534. db.saveScore(right_now,
  535. std::chrono::system_clock::to_time_t(play_day),
  536. hand, 1, score);
  537. //}
  538. next_quit_panel->update();
  539. door << *next_quit_panel;
  540. if (hand < total_hands) {
  541. r = door.get_one_of("NQ");
  542. } else {
  543. r = door.get_one_of("DQ");
  544. }
  545. if (r == 'N') {
  546. hand++;
  547. goto next_hand;
  548. }
  549. in_game = false;
  550. if (r == 'D') {
  551. // done?
  552. // maybe r = 'Q'; ?
  553. }
  554. if (r == 'Q') {
  555. r = 'Q';
  556. }
  557. /*
  558. press_a_key(door);
  559. if (hand < total_hands) {
  560. hand++;
  561. // current_streak = 0;
  562. door << door::reset << door::cls;
  563. goto next_hand;
  564. }
  565. r = 'Q';
  566. */
  567. // in_game = false;
  568. }
  569. }
  570. // update the select_card marker!
  571. cardPos(select_card, cx, cy);
  572. c = dp.marker(1);
  573. c->set(cx + off_x + 2, cy + off_y + 2);
  574. door << *c;
  575. }
  576. }
  577. break;
  578. case XKEY_LEFT_ARROW:
  579. case '4': {
  580. int new_select = findNextActiveCard(true, state, select_card);
  581. /*
  582. int new_active = active_card - 1;
  583. while (new_active >= 0) {
  584. if (state.at(new_active) == 1)
  585. break;
  586. --new_active;
  587. }*/
  588. if (new_select >= 0) {
  589. int cx, cy;
  590. cardPos(select_card, cx, cy);
  591. c = dp.marker(0);
  592. c->set(cx + off_x + 2, cy + off_y + 2);
  593. door << *c;
  594. select_card = new_select;
  595. cardPos(select_card, cx, cy);
  596. c = dp.marker(1);
  597. c->set(cx + off_x + 2, cy + off_y + 2);
  598. door << *c;
  599. }
  600. } break;
  601. case XKEY_RIGHT_ARROW:
  602. case '6': {
  603. int new_select = findNextActiveCard(false, state, select_card);
  604. /*
  605. int new_active = active_card + 1;
  606. while (new_active < 28) {
  607. if (state.at(new_active) == 1)
  608. break;
  609. ++new_active;
  610. }
  611. */
  612. if (new_select >= 0) { //(new_active < 28) {
  613. int cx, cy;
  614. cardPos(select_card, cx, cy);
  615. c = dp.marker(0);
  616. c->set(cx + off_x + 2, cy + off_y + 2);
  617. door << *c;
  618. select_card = new_select;
  619. cardPos(select_card, cx, cy);
  620. c = dp.marker(1);
  621. c->set(cx + off_x + 2, cy + off_y + 2);
  622. door << *c;
  623. }
  624. }
  625. break;
  626. }
  627. } else
  628. in_game = false;
  629. }
  630. if (r == 'Q') {
  631. // continue, play next hand (if applicable), or quit?
  632. // if score < 50, don't bother saving.
  633. // continue -- eat r & redraw.
  634. // quit, save score and exit (unless score is zero).
  635. }
  636. return r;
  637. }
  638. /**
  639. * @brief Redraw the entire play cards screen.
  640. *
  641. * If dealing then show delays when displaying cards, or turning them over.
  642. *
  643. * Otherwise, the user has requested a redraw -- and don't delay.
  644. *
  645. * @param dealing
  646. */
  647. void PlayCards::redraw(bool dealing) {
  648. shared_panel c;
  649. display_starfield(door, rng);
  650. // door << door::reset << door::cls;
  651. door << *spaceAceTriPeaks;
  652. {
  653. // step 1:
  654. // draw the deck "source"
  655. int cx, cy, level;
  656. cardPosLevel(29, cx, cy, level);
  657. if (play_card == 51)
  658. level = 0; // out of cards!
  659. c = dp.back(level);
  660. c->set(cx + off_x, cy + off_y);
  661. // p3 is heigh below
  662. left_panel->set(cx + off_x, cy + off_y + height);
  663. // how do I update these? (hand >1)
  664. score_panel->update();
  665. left_panel->update();
  666. streak_panel->update();
  667. cmd_panel->update();
  668. door << *score_panel << *left_panel << *streak_panel << *cmd_panel;
  669. door << *c;
  670. if (dealing)
  671. std::this_thread::sleep_for(std::chrono::seconds(1));
  672. }
  673. for (int x = 0; x < (dealing ? 28 : 29); x++) {
  674. int cx, cy, level;
  675. cardPosLevel(x, cx, cy, level);
  676. // This is hardly visible.
  677. // door << door::Goto(cx + off_x - 1, cy + off_y + 1);
  678. if (dealing)
  679. std::this_thread::sleep_for(std::chrono::milliseconds(75));
  680. if (dealing) {
  681. c = dp.back(level);
  682. c->set(cx + off_x, cy + off_y);
  683. door << *c;
  684. } else {
  685. // redrawing -- draw the cards with their correct "state"
  686. int s = state.at(x);
  687. switch (s) {
  688. case 0:
  689. c = dp.back(level);
  690. c->set(cx + off_x, cy + off_y);
  691. door << *c;
  692. break;
  693. case 1:
  694. // cardPosLevel(x, space, height, cx, cy, level);
  695. if (x == 28)
  696. c = dp.card(deck.at(play_card));
  697. else
  698. c = dp.card(deck.at(x));
  699. c->set(cx + off_x, cy + off_y);
  700. door << *c;
  701. break;
  702. case 2:
  703. // no card to draw. :)
  704. break;
  705. case 3:
  706. // peak cleared, draw bonus
  707. door << door::Goto(cx + off_x, cy + off_y);
  708. bonus();
  709. break;
  710. }
  711. }
  712. }
  713. if (dealing)
  714. for (int x = 18; x < 29; x++) {
  715. int cx, cy;
  716. state.at(x) = 1;
  717. cardPos(x, cx, cy);
  718. // door << door::Goto(cx + off_x - 1, cy + off_y + 1);
  719. std::this_thread::sleep_for(std::chrono::milliseconds(200));
  720. c = dp.card(deck.at(x));
  721. c->set(cx + off_x, cy + off_y);
  722. door << *c;
  723. }
  724. {
  725. int cx, cy;
  726. cardPos(select_card, cx, cy);
  727. c = dp.marker(1);
  728. c->set(cx + off_x + 2, cy + off_y + 2);
  729. door << *c;
  730. }
  731. }
  732. door::renderFunction statusValue(door::ANSIColor status,
  733. door::ANSIColor value) {
  734. door::renderFunction rf = [status,
  735. value](const std::string &txt) -> door::Render {
  736. door::Render r(txt);
  737. size_t pos = txt.find(':');
  738. if (pos == std::string::npos) {
  739. for (char const &c : txt) {
  740. if (std::isdigit(c))
  741. r.append(value);
  742. else
  743. r.append(status);
  744. }
  745. } else {
  746. pos++;
  747. r.append(status, pos);
  748. r.append(value, txt.length() - pos);
  749. }
  750. return r;
  751. };
  752. return rf;
  753. }
  754. /**
  755. * @brief make the score panel
  756. * This displays: Name, Score, Time Used, Hands Played.
  757. *
  758. * @return std::unique_ptr<door::Panel>
  759. */
  760. std::unique_ptr<door::Panel> PlayCards::make_score_panel() {
  761. const int W = 25;
  762. std::unique_ptr<door::Panel> p = std::make_unique<door::Panel>(W);
  763. p->setStyle(door::BorderStyle::NONE);
  764. door::ANSIColor statusColor(door::COLOR::WHITE, door::COLOR::BLUE,
  765. door::ATTR::BOLD);
  766. door::ANSIColor valueColor(door::COLOR::YELLOW, door::COLOR::BLUE,
  767. door::ATTR::BOLD);
  768. door::renderFunction svRender = statusValue(statusColor, valueColor);
  769. // or use renderStatus as defined above.
  770. // We'll stick with these for now.
  771. {
  772. std::string userString = "Name: ";
  773. userString += door.username;
  774. std::unique_ptr<door::Line> username =
  775. std::make_unique<door::Line>(userString, W);
  776. username->setRender(svRender);
  777. p->addLine(std::move(username));
  778. }
  779. {
  780. door::updateFunction scoreUpdate = [this](void) -> std::string {
  781. std::string text = "Score: ";
  782. text.append(std::to_string(score));
  783. return text;
  784. };
  785. std::string scoreString = scoreUpdate();
  786. std::unique_ptr<door::Line> scoreline =
  787. std::make_unique<door::Line>(scoreString, W);
  788. scoreline->setRender(svRender);
  789. scoreline->setUpdater(scoreUpdate);
  790. p->addLine(std::move(scoreline));
  791. }
  792. {
  793. door::updateFunction timeUpdate = [this](void) -> std::string {
  794. std::stringstream ss;
  795. std::string text;
  796. ss << "Time used: " << setw(3) << door.time_used << " / " << setw(3)
  797. << door.time_left;
  798. text = ss.str();
  799. return text;
  800. };
  801. std::string timeString = timeUpdate();
  802. door::Line time(timeString, W);
  803. time.setRender(svRender);
  804. time.setUpdater(timeUpdate);
  805. p->addLine(std::make_unique<door::Line>(time));
  806. }
  807. {
  808. door::updateFunction handUpdate = [this](void) -> std::string {
  809. std::string text = "Playing Hand ";
  810. text.append(std::to_string(hand));
  811. text.append(" of ");
  812. text.append(std::to_string(total_hands));
  813. return text;
  814. };
  815. std::string handString = handUpdate();
  816. door::Line hands(handString, W);
  817. hands.setRender(svRender);
  818. hands.setUpdater(handUpdate);
  819. p->addLine(std::make_unique<door::Line>(hands));
  820. }
  821. return p;
  822. }
  823. std::unique_ptr<door::Panel> PlayCards::make_streak_panel(void) {
  824. const int W = 20;
  825. std::unique_ptr<door::Panel> p = std::make_unique<door::Panel>(W);
  826. p->setStyle(door::BorderStyle::NONE);
  827. door::ANSIColor statusColor(door::COLOR::WHITE, door::COLOR::BLUE,
  828. door::ATTR::BOLD);
  829. door::ANSIColor valueColor(door::COLOR::YELLOW, door::COLOR::BLUE,
  830. door::ATTR::BOLD);
  831. door::renderFunction svRender = statusValue(statusColor, valueColor);
  832. {
  833. door::updateFunction dateUpdate = [this](void) -> std::string {
  834. std::string text = "Playing: ";
  835. auto in_time_t = std::chrono::system_clock::to_time_t(play_day);
  836. std::stringstream ss;
  837. if (config["date_format"]) {
  838. std::string fmt = config["date_format"].as<std::string>();
  839. ss << std::put_time(std::localtime(&in_time_t), fmt.c_str());
  840. } else
  841. ss << std::put_time(std::localtime(&in_time_t), "%B %d");
  842. text.append(ss.str());
  843. return text;
  844. };
  845. std::string text = dateUpdate();
  846. door::Line current(text, W);
  847. current.setRender(svRender);
  848. current.setUpdater(dateUpdate);
  849. p->addLine(std::make_unique<door::Line>(current));
  850. }
  851. {
  852. door::updateFunction currentUpdate = [this](void) -> std::string {
  853. std::string text = "Current Streak: ";
  854. text.append(std::to_string(current_streak));
  855. return text;
  856. };
  857. std::string currentString = currentUpdate();
  858. door::Line current(currentString, W);
  859. current.setRender(svRender);
  860. current.setUpdater(currentUpdate);
  861. p->addLine(std::make_unique<door::Line>(current));
  862. }
  863. {
  864. door::updateFunction currentUpdate = [this](void) -> std::string {
  865. std::string text = "Longest Streak: ";
  866. text.append(std::to_string(best_streak));
  867. return text;
  868. };
  869. std::string currentString = currentUpdate();
  870. door::Line current(currentString, W);
  871. current.setRender(svRender);
  872. current.setUpdater(currentUpdate);
  873. p->addLine(std::make_unique<door::Line>(current));
  874. }
  875. return p;
  876. }
  877. std::unique_ptr<door::Panel> PlayCards::make_left_panel(void) {
  878. const int W = 13;
  879. std::unique_ptr<door::Panel> p = std::make_unique<door::Panel>(W);
  880. p->setStyle(door::BorderStyle::NONE);
  881. door::ANSIColor statusColor(door::COLOR::WHITE, door::COLOR::BLUE,
  882. door::ATTR::BOLD);
  883. door::ANSIColor valueColor(door::COLOR::YELLOW, door::COLOR::BLUE,
  884. door::ATTR::BOLD);
  885. door::renderFunction svRender = statusValue(statusColor, valueColor);
  886. {
  887. door::updateFunction cardsleftUpdate = [this](void) -> std::string {
  888. std::string text = "Cards left:";
  889. text.append(std::to_string(51 - play_card));
  890. return text;
  891. };
  892. std::string cardsleftString = "Cards left:--";
  893. door::Line cardsleft(cardsleftString, W);
  894. cardsleft.setRender(svRender);
  895. cardsleft.setUpdater(cardsleftUpdate);
  896. p->addLine(std::make_unique<door::Line>(cardsleft));
  897. }
  898. return p;
  899. }
  900. door::renderFunction commandLineRender(door::ANSIColor bracket,
  901. door::ANSIColor inner,
  902. door::ANSIColor outer) {
  903. door::renderFunction rf = [bracket, inner,
  904. outer](const std::string &txt) -> door::Render {
  905. door::Render r(txt);
  906. bool inOuter = true;
  907. for (char const &c : txt) {
  908. if (c == '[') {
  909. inOuter = false;
  910. r.append(bracket);
  911. continue;
  912. }
  913. if (c == ']') {
  914. inOuter = true;
  915. r.append(bracket);
  916. continue;
  917. }
  918. if (inOuter)
  919. r.append(outer);
  920. else
  921. r.append(inner);
  922. }
  923. return r;
  924. };
  925. return rf;
  926. }
  927. std::unique_ptr<door::Panel> PlayCards::make_command_panel(void) {
  928. const int W = 76;
  929. std::unique_ptr<door::Panel> p = make_unique<door::Panel>(W);
  930. p->setStyle(door::BorderStyle::NONE);
  931. std::string commands;
  932. if (door::unicode) {
  933. commands = "[4/\u25c4] Left [6/\u25ba] Right [Space] Play Card [Enter] "
  934. "Draw [Q]uit "
  935. "[R]edraw [H]elp";
  936. } else {
  937. commands =
  938. "[4/\x11] Left [6/\x10] Right [Space] Play Card [Enter] Draw [Q]uit "
  939. "[R]edraw [H]elp";
  940. }
  941. door::ANSIColor bracketColor(door::COLOR::YELLOW, door::COLOR::BLUE,
  942. door::ATTR::BOLD);
  943. door::ANSIColor outerColor(door::COLOR::CYAN, door::COLOR::BLUE,
  944. door::ATTR::BOLD);
  945. door::ANSIColor innerColor(door::COLOR::GREEN, door::COLOR::BLUE,
  946. door::ATTR::BOLD);
  947. door::renderFunction cmdRender =
  948. commandLineRender(bracketColor, innerColor, outerColor);
  949. door::Line cmd(commands, W);
  950. cmd.setRender(cmdRender);
  951. p->addLine(std::make_unique<door::Line>(cmd));
  952. return p;
  953. }
  954. std::unique_ptr<door::Panel> PlayCards::make_next_panel(void) {
  955. const int W = 50; // 50;
  956. std::unique_ptr<door::Panel> p = make_unique<door::Panel>(W);
  957. door::ANSIColor panelColor(door::COLOR::YELLOW, door::COLOR::GREEN,
  958. door::ATTR::BOLD);
  959. p->setStyle(door::BorderStyle::DOUBLE);
  960. p->setColor(panelColor);
  961. door::updateFunction nextUpdate = [this](void) -> std::string {
  962. std::string text;
  963. if (select_card != -1)
  964. text = "[C]ontinue this hand";
  965. if (hand < total_hands)
  966. text += " [N]ext Hand [Q]uit";
  967. else
  968. text += " [D]one [Q]uit";
  969. return text;
  970. };
  971. std::string text;
  972. text = nextUpdate();
  973. door::ANSIColor bracketColor(door::COLOR::YELLOW, door::COLOR::BLUE,
  974. door::ATTR::BOLD);
  975. door::ANSIColor innerColor(door::COLOR::CYAN, door::COLOR::BLUE,
  976. door::ATTR::BOLD);
  977. door::ANSIColor outerColor(door::COLOR::GREEN, door::COLOR::BLUE,
  978. door::ATTR::BOLD);
  979. door::renderFunction textRender =
  980. commandLineRender(bracketColor, innerColor, outerColor);
  981. door::Line nextLine(text, W);
  982. nextLine.setRender(textRender);
  983. nextLine.setPadding(" ", panelColor);
  984. nextLine.setUpdater(nextUpdate);
  985. nextLine.fit();
  986. p->addLine(std::make_unique<door::Line>(nextLine));
  987. return p;
  988. }
  989. std::unique_ptr<door::Panel> PlayCards::make_tripeaks(void) {
  990. std::string tripeaksText(" " SPACEACE
  991. " - Tri-Peaks Solitaire v" SPACEACE_VERSION " ");
  992. std::unique_ptr<door::Panel> spaceAceTriPeaks =
  993. std::make_unique<door::Panel>(tripeaksText.size());
  994. spaceAceTriPeaks->setStyle(door::BorderStyle::SINGLE);
  995. spaceAceTriPeaks->setColor(
  996. door::ANSIColor(door::COLOR::CYAN, door::COLOR::BLACK));
  997. spaceAceTriPeaks->addLine(
  998. std::make_unique<door::Line>(tripeaksText, tripeaksText.size()));
  999. return spaceAceTriPeaks;
  1000. }
  1001. std::unique_ptr<door::Panel> PlayCards::make_month(std::string month) {
  1002. const int W = 3;
  1003. std::unique_ptr<door::Panel> p = make_unique<door::Panel>(W);
  1004. door::ANSIColor panelColor(door::COLOR::YELLOW, door::COLOR::BLACK,
  1005. door::ATTR::BOLD);
  1006. p->setStyle(door::BorderStyle::DOUBLE);
  1007. p->setColor(panelColor);
  1008. for (auto c : month) {
  1009. std::string text = " ";
  1010. text += c;
  1011. text += " ";
  1012. door::Line line(text);
  1013. p->addLine(std::make_unique<door::Line>(line));
  1014. }
  1015. return p;
  1016. }
  1017. std::unique_ptr<door::Panel> PlayCards::make_weekdays(void) {
  1018. const int W = 41;
  1019. std::string text = " SUN MON TUE WED THU FRI SAT ";
  1020. std::unique_ptr<door::Panel> p = make_unique<door::Panel>(W);
  1021. door::ANSIColor panelColor(door::COLOR::CYAN, door::COLOR::BLACK,
  1022. door::ATTR::BOLD);
  1023. p->setStyle(door::BorderStyle::DOUBLE);
  1024. p->setColor(panelColor);
  1025. door::Line line(text);
  1026. p->addLine(std::make_unique<door::Line>(line));
  1027. return p;
  1028. }
  1029. std::unique_ptr<door::Panel> PlayCards::make_calendar_panel(void) {
  1030. const int W = 41;
  1031. std::unique_ptr<door::Panel> p = make_unique<door::Panel>(W);
  1032. p->setStyle(door::BorderStyle::DOUBLE);
  1033. p->setColor(door::ANSIColor(door::COLOR::CYAN, door::COLOR::BLACK));
  1034. door::renderFunction calendarRender =
  1035. [](const std::string &txt) -> door::Render {
  1036. door::Render r(txt);
  1037. // normal digits color
  1038. door::ANSIColor digits(door::COLOR::CYAN, door::COLOR::BLACK);
  1039. // digits/days that can be played
  1040. door::ANSIColor digits_play(door::COLOR::CYAN, door::COLOR::BLACK,
  1041. door::ATTR::BOLD);
  1042. // spaces color
  1043. door::ANSIColor spaces(door::COLOR::WHITE, door::COLOR::BLACK);
  1044. // o - open days
  1045. door::ANSIColor open(door::COLOR::GREEN, door::COLOR::BLACK,
  1046. door::ATTR::BOLD);
  1047. // h - hands can be played
  1048. door::ANSIColor hands(door::COLOR::YELLOW, door::COLOR::BLACK,
  1049. door::ATTR::BOLD);
  1050. // x - already played
  1051. door::ANSIColor full(door::COLOR::RED, door::COLOR::BLACK);
  1052. // u - No, Not Yet! unavailable
  1053. door::ANSIColor nny(door::COLOR::BLACK, door::COLOR::BLACK,
  1054. door::ATTR::BOLD);
  1055. int days;
  1056. /*
  1057. if (get_logger) {
  1058. get_logger() << "Line [" << txt << "]" << std::endl;
  1059. }
  1060. */
  1061. // B _ BBBB _ BBBB _ BBBB _ BBBB _ BBBB _ BBBB _ B
  1062. for (days = 0; days < 7; ++days) {
  1063. std::string dayText;
  1064. if (days == 0) {
  1065. r.append(spaces);
  1066. // dayText = txt.substr(1, 3);
  1067. } // else {
  1068. /*
  1069. if (get_logger)
  1070. get_logger() << days << " " << 1 + (days * 6) << std::endl;
  1071. */
  1072. dayText = txt.substr(1 + (days * 6), 3);
  1073. // }
  1074. /*
  1075. if (get_logger) {
  1076. get_logger() << days << " "
  1077. << "[" << dayText << "] " << std::endl;
  1078. }
  1079. */
  1080. if (dayText[1] == ' ') {
  1081. // Ok, nothing there!
  1082. r.append(spaces, 3);
  1083. } else {
  1084. // Something is there
  1085. char cday = dayText[2];
  1086. if ((cday == 'o') or (cday == 'h'))
  1087. r.append(digits_play, 2);
  1088. else
  1089. r.append(digits, 2);
  1090. switch (dayText[2]) {
  1091. case 'o':
  1092. r.append(open);
  1093. break;
  1094. case 'h':
  1095. r.append(hands);
  1096. break;
  1097. case 'x':
  1098. r.append(full);
  1099. break;
  1100. case 'u':
  1101. r.append(nny);
  1102. break;
  1103. }
  1104. }
  1105. if (days == 6)
  1106. r.append(spaces);
  1107. else
  1108. r.append(spaces, 3);
  1109. }
  1110. return r;
  1111. };
  1112. int row;
  1113. for (row = 0; row < 6; ++row) {
  1114. door::updateFunction calendarUpdate = [this, row]() -> std::string {
  1115. std::string text;
  1116. for (int d = 0; d < 7; ++d) {
  1117. text += " ";
  1118. int v = this->calendar_panel_days[(row * 7) + d];
  1119. if (v == 0)
  1120. text += " ";
  1121. else {
  1122. std::string number = std::to_string(v);
  1123. if (number.length() == 1)
  1124. text += " ";
  1125. text += number;
  1126. // Now, translate the status
  1127. int status = this->calendar_day_status[v - 1];
  1128. switch (status) {
  1129. case 0:
  1130. text += "o";
  1131. break;
  1132. case 1:
  1133. text += "h";
  1134. break;
  1135. case 2:
  1136. text += "x";
  1137. break;
  1138. case 3:
  1139. text += "u";
  1140. break;
  1141. }
  1142. }
  1143. if (d == 6)
  1144. text += " ";
  1145. else
  1146. text += " ";
  1147. }
  1148. return text;
  1149. };
  1150. std::string text;
  1151. text = calendarUpdate();
  1152. door::Line line(text, W);
  1153. line.setRender(calendarRender);
  1154. line.setUpdater(calendarUpdate);
  1155. p->addLine(std::make_unique<door::Line>(line));
  1156. }
  1157. return p;
  1158. }
  1159. /**
  1160. * @brief make calendar
  1161. * We assume the calendar is for this month now()
  1162. * Jaunary
  1163. * February
  1164. * March
  1165. * April
  1166. * May
  1167. * June
  1168. * July
  1169. * August
  1170. * September
  1171. * October
  1172. * November
  1173. * December
  1174. * 123456789012345678901234567890123456789012345678901234567890
  1175. * ▒▒░▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀░▒▒
  1176. * N ▒▒▌SUN MON TUE WED THU FRI SAT▐▒▒
  1177. * O ▒▒░───▄───▄───▄───▄───▄───▄───░▒▒
  1178. * V ▒▒▌ 1x│ 2x│ 3o│ 4o│ 5o│ 6o│ 7o▐▒▒ X = Day Completed
  1179. * E ▒▒▌ 8o│ 9x│10o│11o│12x│13x│14o▐▒▒ O = Day Playable
  1180. * M ▒▒▌15x│16u│17u│18u│19u│20u│21u▐▒▒ U = Day Unavailable
  1181. * B ▒▒▌22u│23u│24u│25u│26u│27u│28u▐▒▒ H = Day Uncompleted
  1182. * E ▒▒▌29u│30u│ │ │ │ │ ▐▒▒
  1183. * R ▒▒▌ │ │ │ │ │ │ ▐▒▒
  1184. * ▒▒░▄▄▄█▄▄▄█▄▄▄█▄▄▄█▄▄▄█▄▄▄█▄▄▄░▒▒
  1185. *
  1186. * 123456789012345678901234567890123456789012345678901234567890123456789012345
  1187. *
  1188. * ▒▒░▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀▀░▒▒
  1189. * N ▒▒▌ SUN MON TUE WED THU FRI SAT ▐▒▒
  1190. * O ▒▒░─────▄─────▄─────▄─────▄─────▄─────▄─────░▒▒
  1191. * V ▒▒▌ 1x │ 2x │ 3o │ 4o │ 5o │ 6o │ 7o ▐▒▒ X = Day Completed
  1192. * E ▒▒▌ 8o │ 9x │ 10o │ 11o │ 12x │ 13x │ 14o ▐▒▒ O = Day Playable
  1193. * M ▒▒▌ 15x │ 16u │ 17u │ 18u │ 19u │ 20u │ 21u ▐▒▒ U = Day Unavailable
  1194. * B ▒▒▌ 22u │ 23u │ 24u │ 25u │ 26u │ 27u │ 28u ▐▒▒ H = Day Uncompleted
  1195. * E ▒▒▌ 29u │ 30u │ │ │ │ │ ▐▒▒
  1196. * R ▒▒▌ │ │ │ │ │ │ ▐▒▒
  1197. * ▒▒░▄▄▄▄▄█▄▄▄▄▄█▄▄▄▄▄█▄▄▄▄▄█▄▄▄▄▄█▄▄▄▄▄█▄▄▄▄▄░▒▒
  1198. * X Extra Days Allowed Per Day
  1199. *
  1200. * 123456789012345678901234567890123456789012345678901234567890123456789012345
  1201. * ╔═══╗ ╔═════════════════════════════════════════╗
  1202. * ║ N ║ ║ SUN MON TUE WED THU FRI SAT ║
  1203. * ║ O ║ ╚═════════════════════════════════════════╝
  1204. * ║ V ║ ╔═════════════════════════════════════════╗
  1205. * ║ E ║ ║ 1x 2x 3o 4o 5o 6o 7o ║
  1206. * ║ M ║ ║ 8x 9x 10o 11o 12o 13o 14o ║
  1207. * ║ B ║ ║ 15h 16h 17h 18h 19u 20u 21u ║
  1208. * ║ E ║ ║ 22u 23u 24u 25u 26u 27u 28u ║
  1209. * ║ R ║ ║ 29u 30u 31u ║
  1210. * ╚═══╝ ║ ║
  1211. * ╚═════════════════════════════════════════╝
  1212. *
  1213. *
  1214. * Sunday
  1215. * Monday
  1216. * Tuesday
  1217. * Wednesday
  1218. * Thursday
  1219. * Friday
  1220. * Saturday
  1221. *
  1222. * 12345678901234567890123456789012345678901234567890123456789012345678901234567890
  1223. *
  1224. *╔═══╗╔═══════════════════════════════════════════════════════════════════════╗
  1225. *║ N ║║ Sunday Monday Tuesday Wednesday Thursday Friday Saturday
  1226. *║ ║ O
  1227. *║╚═══════════════════════════════════════════════════════════════════════╝
  1228. * 1 ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗
  1229. * ║ 1 x ║ ║ 2 x ║ ║ 3 o ║ ║ 4 o ║ ║ 5 o ║ ║ 6 o ║ ║ 7 o ║
  1230. * ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝
  1231. * 4 ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗
  1232. * ║ 8 x ║ ║ 9 x ║ ║ 10 o ║ ║ 11 o ║ ║ 12 o ║ ║ 13 o ║ ║ 14 o ║
  1233. * ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝
  1234. * 7 ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗
  1235. * ║ 15 x ║ ║ 16 x ║ ║ 17 o ║ ║ 18 o ║ ║ 19 o ║ ║ 20 o ║ ║ 21 o ║
  1236. * ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝
  1237. * 10 ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗ ╔══════╗
  1238. * ║ 22 x ║ ║ 23 x ║ ║ 24 o ║ ║ 25 o ║ ║ 26 o ║ ║ 27 o ║ ║ 28 o ║
  1239. * ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝ ╚══════╝
  1240. * 13 ╔══════╗ ╔══════╗ ╔══════╗
  1241. * ║ 29 x ║ ║ 30 u ║ ║ 31 u ║
  1242. * ╚══════╝ ╚══════╝ ╚══════╝
  1243. *
  1244. * +2 +12 +22 +32 +42 +52 +62
  1245. *
  1246. *
  1247. *
  1248. *
  1249. *
  1250. *
  1251. *
  1252. *
  1253. *
  1254. *
  1255. * 123456789012345678901234567890123456789012345678901234567890123456789012345
  1256. * ╔═══╗ ╔═════════════════════════════════════════════════════╗
  1257. * ║ N ║ ║ SUN MON TUE WED THU FRI SAT ║
  1258. * ║ O ║ ╚═════════════════════════════════════════════════════╝
  1259. * ║ V ║ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗
  1260. * ║ E ║ ║ 1x ║ ║ 2x ║ ║ 3o ║ ║ 4o ║ ║ 5o ║ ║ 6o ║ ║ 7o ║
  1261. * ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝
  1262. * ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗
  1263. * ║ 8x ║ ║ 9x ║ ║ 10o ║ ║ 11o ║ ║ 12o ║ ║ 13o ║ ║ 14o ║
  1264. * ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝
  1265. * ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗
  1266. * ║ 15x ║ ║ 16x ║ ║ 17o ║ ║ 18o ║ ║ 19o ║ ║ 20o ║ ║ 21o ║
  1267. * ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝
  1268. * ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗ ╔═════╗
  1269. * ║ 22x ║ ║ 23x ║ ║ 24o ║ ║ 25o ║ ║ 26o ║ ║ 27o ║ ║ 28o ║
  1270. * ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝ ╚═════╝
  1271. * ╔═════╗ ╔═════╗ ╔═════╗
  1272. * ║ 29x ║ ║ 30u ║ ║ 31u ║
  1273. * ╚═════╝ ╚═════╝ ╚═════╝
  1274. *
  1275. * ^ Actually single lines, not double lines here ^
  1276. *
  1277. * ║ M ║ ║ 8x 9x 10o 11o 12o 13o 14o ║
  1278. * ║ B ║ ║ 15h 16h 17h 18h 19u 20u 21u ║
  1279. * ║ E ║ ║ 22u 23u 24u 25u 26u 27u 28u ║
  1280. * ║ R ║ ║ 29u 30u 31u ║
  1281. * ╚═══╝ ║ ║
  1282. * ╚═════════════════════════════════════════╝
  1283. *
  1284. *
  1285. *
  1286. * @return std::unique_ptr<door::Panel>
  1287. */
  1288. std::string
  1289. PlayCards::current_month(std::chrono::_V2::system_clock::time_point now) {
  1290. time_t now_t = std::chrono::system_clock::to_time_t(now);
  1291. std::tm *now_tm = localtime(&now_t);
  1292. ostringstream os;
  1293. std::string text;
  1294. os << std::put_time(now_tm, "%B");
  1295. text = os.str();
  1296. return text;
  1297. }
  1298. /**
  1299. * @brief Update the calendar with days played so far.
  1300. *
  1301. * This is before redisplaying the calendar.
  1302. *
  1303. */
  1304. void PlayCards::update_calendar_days(void) {
  1305. time_t month_t = calendar_day_t[0];
  1306. std::tm month_lt;
  1307. localtime_r(&month_t, &month_lt);
  1308. int this_month = month_lt.tm_mon;
  1309. // int this_day = month_lt.tm_mday;
  1310. int this_year = month_lt.tm_year + 1900;
  1311. // calendar_day_status.fill(0);
  1312. auto last_played = db.whenPlayed();
  1313. /*
  1314. int play_days_ahead = 0;
  1315. if (config["play_days_ahead"]) {
  1316. play_days_ahead = config["play_days_ahead"].as<int>();
  1317. }
  1318. */
  1319. // until maint is setup, we need to verify that the month and year is
  1320. // correct.
  1321. for (auto played : last_played) {
  1322. #ifdef DEBUG_OUTPUT
  1323. get_logger() << "played " << played.first << " hands: " << played.second
  1324. << std::endl;
  1325. #endif
  1326. time_t play_t = played.first;
  1327. std::tm played_tm;
  1328. localtime_r(&play_t, &played_tm);
  1329. #ifdef DEBUG_OUTPUT
  1330. if (get_logger) {
  1331. get_logger() << played_tm.tm_mon + 1 << "/" << played_tm.tm_mday << "/"
  1332. << played_tm.tm_year + 1900 << " : " << played.second << " "
  1333. << play_t << std::endl;
  1334. }
  1335. #endif
  1336. if ((played_tm.tm_mon == this_month) &&
  1337. (played_tm.tm_year + 1900 == this_year)) {
  1338. // Ok!
  1339. int hands = played.second;
  1340. if (hands < total_hands) {
  1341. calendar_day_status[played_tm.tm_mday - 1] = 1;
  1342. } else {
  1343. if (hands >= total_hands) {
  1344. calendar_day_status[played_tm.tm_mday - 1] = 2;
  1345. }
  1346. }
  1347. }
  1348. }
  1349. }
  1350. /**
  1351. * make_calendar
  1352. *
  1353. * This needs the screen size information so it can place the
  1354. * panels in the correct location.
  1355. */
  1356. std::unique_ptr<door::Screen> PlayCards::make_calendar() {
  1357. std::unique_ptr<door::Screen> s = std::make_unique<door::Screen>();
  1358. /*
  1359. * This has potential of jumping ahead if player is on after midnight.
  1360. * I'd rather this be stable. When they start the game, is where
  1361. * we count things. (Also last day of month + midnight -- means
  1362. * we need maint ran! No!)
  1363. */
  1364. std::chrono::system_clock::time_point month =
  1365. std::chrono::system_clock::now();
  1366. // std::chrono::_V2::system_clock::time_point month =
  1367. // std::chrono::system_clock::now();
  1368. time_t month_t = std::chrono::system_clock::to_time_t(month);
  1369. // adjust to first day of the month
  1370. std::tm month_lt;
  1371. localtime_r(&month_t, &month_lt);
  1372. int this_month = month_lt.tm_mon;
  1373. int this_day = month_lt.tm_mday;
  1374. int this_year = month_lt.tm_year + 1900;
  1375. firstOfMonthDate(month);
  1376. month_t = std::chrono::system_clock::to_time_t(month);
  1377. calendar_day_t.fill(0);
  1378. calendar_day_t[0] = month_t;
  1379. #ifdef DEBUG_OUTPUT
  1380. get_logger() << "1st of Month is "
  1381. << std::put_time(std::localtime(&month_t), "%c %Z") << std::endl;
  1382. #endif
  1383. localtime_r(&month_t, &month_lt);
  1384. const int FIRST_WEEKDAY = month_lt.tm_wday; // 0-6
  1385. #ifdef DEBUG_OUTPUT
  1386. get_logger() << "1st of the Month starts on " << FIRST_WEEKDAY << std::endl;
  1387. #endif
  1388. // find the last day of this month.
  1389. auto month_l = month;
  1390. time_t month_last_day_t = month_t;
  1391. std::tm mld_tm;
  1392. do {
  1393. month_l += 24h;
  1394. normalizeDate(month_l);
  1395. month_last_day_t = std::chrono::system_clock::to_time_t(month_l);
  1396. localtime_r(&month_last_day_t, &mld_tm);
  1397. if (mld_tm.tm_mday != 1) {
  1398. month_last_day = mld_tm.tm_mday;
  1399. calendar_day_t[month_last_day - 1] = month_last_day_t;
  1400. }
  1401. } while (mld_tm.tm_mday != 1);
  1402. #ifdef DEBUG_OUTPUT
  1403. get_logger() << "Last day is " << month_last_day << std::endl;
  1404. #endif
  1405. calendar_panel_days.fill(0);
  1406. int row = 0;
  1407. for (int x = 0; x < month_last_day; x++) {
  1408. int dow = (x + FIRST_WEEKDAY) % 7;
  1409. if ((x != 0) and (dow == 0))
  1410. row++;
  1411. /*
  1412. get_logger() << "x = " << x << " dow = " << dow << " row = " << row
  1413. << std::endl;
  1414. */
  1415. // we actually want x+1 (1 to month_last_day)
  1416. // get_logger() << row * 7 + dow << " = " << x + 1 << std::endl;
  1417. calendar_panel_days[row * 7 + dow] = x + 1;
  1418. }
  1419. calendar_day_status.fill(0);
  1420. // update_calendar_days(month_t);
  1421. auto last_played = db.whenPlayed();
  1422. int play_days_ahead = 0;
  1423. if (config["play_days_ahead"]) {
  1424. play_days_ahead = config["play_days_ahead"].as<int>();
  1425. }
  1426. // until maint is setup, we need to verify that the month and year is
  1427. // correct.
  1428. for (auto played : last_played) {
  1429. #ifdef DEBUG_OUTPUT
  1430. get_logger() << "played " << played.first << " hands: " << played.second
  1431. << std::endl;
  1432. #endif
  1433. time_t play_t = played.first;
  1434. std::tm played_tm;
  1435. localtime_r(&play_t, &played_tm);
  1436. #ifdef DEBUG_OUTPUT
  1437. if (get_logger) {
  1438. get_logger() << played_tm.tm_mon + 1 << "/" << played_tm.tm_mday << "/"
  1439. << played_tm.tm_year + 1900 << " : " << played.second << " "
  1440. << play_t << std::endl;
  1441. }
  1442. #endif
  1443. if ((played_tm.tm_mon == this_month) &&
  1444. (played_tm.tm_year + 1900 == this_year)) {
  1445. // Ok!
  1446. int hands = played.second;
  1447. #ifdef DEBUG_OUTPUT
  1448. if (get_logger) {
  1449. get_logger() << "hands " << hands << " total " << total_hands
  1450. << std::endl;
  1451. }
  1452. #endif
  1453. if (hands < total_hands) {
  1454. calendar_day_status[played_tm.tm_mday - 1] = 1;
  1455. } else {
  1456. if (hands >= total_hands) {
  1457. calendar_day_status[played_tm.tm_mday - 1] = 2;
  1458. }
  1459. }
  1460. }
  1461. }
  1462. #ifdef DEBUG_OUTPUT
  1463. if (get_logger) {
  1464. get_logger() << "last day " << month_last_day << " today " << this_day
  1465. << " plays ahead " << play_days_ahead << std::endl;
  1466. }
  1467. #endif
  1468. // mark all days ahead as NNY.
  1469. for (int d = 0; d < 31; ++d) {
  1470. if (this_day + play_days_ahead - 1 < d) {
  1471. calendar_day_status[d] = 3;
  1472. }
  1473. }
  1474. #ifdef DEBUG_OUTPUT
  1475. {
  1476. // output all of the calendar information
  1477. ofstream &of = get_logger();
  1478. of << "Calendar_panel_days:" << std::endl;
  1479. for (int x = 0; x < (int)calendar_panel_days.size(); ++x) {
  1480. of << std::setw(2) << calendar_panel_days[x] << ":";
  1481. int c = calendar_panel_days[x];
  1482. if (c == 0)
  1483. of << " ";
  1484. else {
  1485. of << calendar_day_status[c - 1];
  1486. };
  1487. of << " ";
  1488. if ((x != 0) and (((x + 1) % 7) == 0)) {
  1489. of << std::endl;
  1490. }
  1491. }
  1492. of << std::endl;
  1493. }
  1494. #endif
  1495. std::string current = current_month(month);
  1496. string_toupper(current);
  1497. if (current.length() < 6) {
  1498. current.insert(0, " ");
  1499. current += " ";
  1500. }
  1501. std::unique_ptr<door::Panel> p = make_month(current);
  1502. p->set(3, 3);
  1503. s->addPanel(std::move(p));
  1504. p = make_weekdays();
  1505. p->set(8, 3);
  1506. s->addPanel(std::move(p));
  1507. p = make_calendar_panel();
  1508. p->set(8, 6);
  1509. s->addPanel(std::move(p));
  1510. return s;
  1511. }