play.cpp 52 KB

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