play.cpp 56 KB

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