playcards.go 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182
  1. package main
  2. import (
  3. "crypto/sha1"
  4. "encoding/binary"
  5. "fmt"
  6. "log"
  7. "math/rand"
  8. "red-green/door"
  9. "strconv"
  10. "strings"
  11. "time"
  12. )
  13. func StringToANSIColor(colorCode string) string {
  14. colors := []string{
  15. "BLUE",
  16. "BROWN",
  17. "RED",
  18. "CYAN",
  19. "GREEN",
  20. "MAGENTA",
  21. "WHITE",
  22. }
  23. code := strings.ToUpper(colorCode)
  24. for _, c := range colors {
  25. if c == code {
  26. return door.ColorText(code)
  27. }
  28. }
  29. if code == "ALL" {
  30. rand.Seed(time.Now().UnixNano())
  31. pos := rand.Intn(len(colors))
  32. return door.ColorText(colors[pos])
  33. }
  34. return door.ColorText("WHITE")
  35. }
  36. type PlayCards struct {
  37. Door *door.Door
  38. DB *DBData
  39. Config *map[string]string
  40. RNG *rand.Rand
  41. Seeds []int32
  42. Total_hands int
  43. Play_card int
  44. Current_streak int
  45. Best_streak int
  46. Select_card int
  47. Score int
  48. Play_day time.Time
  49. Play_day_t int64
  50. Days_played int
  51. Hand int
  52. SpaceAceTriPeaks door.Panel
  53. ScorePanel door.Panel
  54. StreakPanel door.Panel
  55. LeftPanel door.Panel
  56. CmdPanel door.Panel
  57. NextQuitPanel door.Panel
  58. DeckPanel Deck
  59. Deck []DeckType // [51]DeckType t
  60. State []DeckType // [51]DeckType
  61. Off_X int
  62. Off_Y int
  63. Calendar_day_t [31]int64
  64. Calendar door.Screen
  65. Calendar_panel_days [42]int // Where is each day positioned on the Calendar?
  66. Calendar_day_status [31]int
  67. }
  68. // Possibly change Deck to [51]DeckType. This game always only has 1 deck.
  69. // Possibly change State [51]DeckType to [51]int8.
  70. // There are few states to track.
  71. func cmdLineRender(bracket string, inner string, outer string) func(string) string {
  72. return func(input string) string {
  73. var r door.Render = door.Render{Line: input}
  74. inOuter := true
  75. for _, c := range input {
  76. if c == '[' {
  77. inOuter = false
  78. r.Append(bracket, 1)
  79. continue
  80. }
  81. if c == ']' {
  82. inOuter = true
  83. r.Append(bracket, 1)
  84. continue
  85. }
  86. if inOuter {
  87. r.Append(outer, 1)
  88. } else {
  89. r.Append(inner, 1)
  90. }
  91. }
  92. return r.Result
  93. }
  94. }
  95. func (pc *PlayCards) InitValues() {
  96. hands := pc.DB.GetSetting("hands_per_day", "3")
  97. pc.Total_hands, _ = strconv.Atoi(hands)
  98. pc.Play_card = 28
  99. pc.Current_streak = 0
  100. pc.Best_streak = 0
  101. best := pc.DB.GetSetting("best_streak", "0")
  102. pc.Best_streak, _ = strconv.Atoi(best)
  103. pc.Select_card = 23
  104. pc.Score = 0
  105. }
  106. func (pc *PlayCards) Init() {
  107. // init_values()
  108. pc.InitValues()
  109. // PlayCards::PlayCards()
  110. pc.Play_day = time.Now()
  111. NormalizeDate(&pc.Play_day)
  112. pc.Play_day_t = pc.Play_day.Unix()
  113. pc.DeckPanel.Init()
  114. var last_played int64
  115. last := pc.DB.GetSetting("last_played", "0")
  116. last_played, _ = strconv.ParseInt(last, 10, 64)
  117. days := pc.DB.GetSetting("days_played", "0")
  118. pc.Days_played, _ = strconv.Atoi(days)
  119. if last_played != pc.Play_day_t {
  120. pc.DB.SetSetting("last_played", strconv.FormatInt(pc.Play_day_t, 10))
  121. pc.DB.SetSetting("days_played", "0")
  122. pc.Days_played = 0
  123. }
  124. pc.Seeds = make([]int32, 0)
  125. // config _seed
  126. parts := strings.Split((*pc.Config)["_seed"], ",")
  127. for _, seed := range parts {
  128. i, _ := strconv.ParseInt(seed, 10, 32)
  129. pc.Seeds = append(pc.Seeds, int32(i))
  130. }
  131. pc.Hand = 0
  132. pc.Total_hands = 0
  133. // spaceAceTriPeaks = make_tripeaks();
  134. {
  135. tripeakstext := " " + SPACEACE + " - Tri-Peaks Solitaire v" + SPACEACE_VERSION + " "
  136. pc.SpaceAceTriPeaks = door.Panel{Width: len(tripeakstext),
  137. Style: door.SINGLE,
  138. BorderColor: door.ColorText("CYAN ON BLACK"),
  139. }
  140. pc.SpaceAceTriPeaks.Lines = append(pc.SpaceAceTriPeaks.Lines,
  141. door.Line{Text: tripeakstext})
  142. }
  143. svRender := RenderStatusValue(door.ColorText("BOLD WHITE ON BLUE"),
  144. door.ColorText("BOLD YELLOW ON BLUE"))
  145. // score_panel = make_score_panel();
  146. {
  147. W := 25
  148. pc.ScorePanel = door.Panel{Width: W}
  149. text := "Name: " + pc.Door.Config.Real_name
  150. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  151. door.Line{Text: fmt.Sprintf("%*s", -W, text), RenderF: svRender})
  152. scoreUpdate := func() string {
  153. txt := fmt.Sprintf("Score: %d", pc.Score)
  154. txt += strings.Repeat(" ", W-len(txt))
  155. return txt
  156. }
  157. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  158. door.Line{Text: scoreUpdate(),
  159. UpdateF: scoreUpdate,
  160. RenderF: svRender,
  161. })
  162. timeUpdate := func() string {
  163. var left int = int(pc.Door.TimeLeft().Minutes())
  164. var used int = int(pc.Door.TimeUsed().Minutes())
  165. txt := fmt.Sprintf("Time used: %3d / %3d", used, left)
  166. txt += strings.Repeat(" ", W-len(txt))
  167. return txt
  168. }
  169. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  170. door.Line{Text: timeUpdate(),
  171. UpdateF: timeUpdate,
  172. RenderF: svRender,
  173. })
  174. handUpdate := func() string {
  175. txt := fmt.Sprintf("Playing Hand %d of %d", pc.Hand, pc.Total_hands)
  176. txt += strings.Repeat(" ", W-len(txt))
  177. return txt
  178. }
  179. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  180. door.Line{Text: handUpdate(),
  181. UpdateF: handUpdate,
  182. RenderF: svRender,
  183. })
  184. }
  185. // streak_panel = make_streak_panel();
  186. {
  187. W := 20
  188. pc.StreakPanel = door.Panel{Width: W}
  189. dateUpdate := func() string {
  190. format, ok := (*pc.Config)["date_format"]
  191. if !ok {
  192. format = "January 2"
  193. }
  194. txt := fmt.Sprintf("Playing: %s", pc.Play_day.Format(format))
  195. txt += strings.Repeat(" ", W-len(txt))
  196. return txt
  197. }
  198. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  199. door.Line{Text: dateUpdate(),
  200. UpdateF: dateUpdate,
  201. RenderF: svRender,
  202. })
  203. currentUpdate := func() string {
  204. txt := fmt.Sprintf("Current Streak: %d", pc.Current_streak)
  205. txt += strings.Repeat(" ", W-len(txt))
  206. return txt
  207. }
  208. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  209. door.Line{Text: currentUpdate(),
  210. UpdateF: currentUpdate,
  211. RenderF: svRender,
  212. })
  213. longestUpdate := func() string {
  214. txt := fmt.Sprintf("Longest Streak: %d", pc.Best_streak)
  215. txt += strings.Repeat(" ", W-len(txt))
  216. return txt
  217. }
  218. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  219. door.Line{Text: longestUpdate(),
  220. UpdateF: longestUpdate,
  221. RenderF: svRender,
  222. })
  223. }
  224. // left_panel = make_left_panel();
  225. {
  226. W := 13
  227. pc.LeftPanel = door.Panel{Width: W}
  228. leftUpdate := func() string {
  229. txt := fmt.Sprintf("Cards left:%d", 51-pc.Play_card)
  230. txt += strings.Repeat(" ", W-len(txt))
  231. return txt
  232. }
  233. pc.LeftPanel.Lines = append(pc.LeftPanel.Lines,
  234. door.Line{Text: leftUpdate(),
  235. UpdateF: leftUpdate,
  236. RenderF: svRender,
  237. })
  238. }
  239. // cmd_panel = make_command_panel();
  240. {
  241. W := 76
  242. pc.CmdPanel = door.Panel{Width: W}
  243. var commands string
  244. if door.Unicode {
  245. commands = "[4/\u25c4] Left [6/\u25ba] Right [Space] Play Card [Enter] Draw [Q]uit [R]edraw [H]elp"
  246. commands += strings.Repeat(" ", W-len([]rune(commands)))
  247. } else {
  248. commands = "[4/\x11] Left [6/\x10] Right [Space] Play Card [Enter] Draw [Q]uit [R]edraw [H]elp"
  249. commands += strings.Repeat(" ", W-len(commands))
  250. }
  251. cmdRender := cmdLineRender(door.ColorText("BOLD YELLOW ON BLUE"),
  252. door.ColorText("BOLD CYAN ON BLUE"),
  253. door.ColorText("BOLD GREEN ON BLUE"))
  254. pc.CmdPanel.Lines = append(pc.CmdPanel.Lines,
  255. door.Line{Text: commands,
  256. RenderF: cmdRender})
  257. }
  258. // next_quit_panel = make_next_panel();
  259. {
  260. W := 50
  261. pc.NextQuitPanel = door.Panel{Width: W,
  262. Style: door.DOUBLE,
  263. BorderColor: door.ColorText("BOLD YELLOW ON BLUE"),
  264. }
  265. nextUpdate := func() string {
  266. var text string
  267. if pc.Select_card != -1 {
  268. text = "[C]ontinue this hand"
  269. }
  270. if pc.Hand < pc.Total_hands {
  271. text += " [N]ext Hand [Q]uit"
  272. } else {
  273. text += " [D]one [Q]uit"
  274. }
  275. text = " " + text + " "
  276. text += strings.Repeat(" ", W-len(text))
  277. return text
  278. }
  279. pc.NextQuitPanel.Lines = append(pc.NextQuitPanel.Lines,
  280. door.Line{Text: nextUpdate(),
  281. UpdateF: nextUpdate,
  282. RenderF: cmdLineRender(door.ColorText("BOLD YEL ON BLU"),
  283. door.ColorText("BOLD CYAN ON BLU"),
  284. door.ColorText("BOLD GREE ON BLUE"))})
  285. }
  286. // calendar = make_calendar();
  287. {
  288. pc.Calendar = door.Screen{}
  289. var month time.Time = time.Now()
  290. var today_day int = month.Day()
  291. FirstOfMonthDate(&month)
  292. // clear out
  293. for x := range pc.Calendar_day_t {
  294. pc.Calendar_day_t[x] = 0
  295. }
  296. // pc.Calendar_day_t = make([]int64, 31)
  297. pc.Calendar_day_t[0] = month.Unix()
  298. var First_Weekday int = int(month.Weekday())
  299. var month_last_day = 0
  300. var month_end time.Time = month
  301. for {
  302. month_end = month_end.AddDate(0, 0, 1)
  303. NormalizeDate(&month_end)
  304. if month_end.Day() == 1 {
  305. break
  306. } else {
  307. pc.Calendar_day_t[month_end.Day()-1] = month_end.Unix()
  308. month_last_day = month_end.Day()
  309. }
  310. }
  311. // clear out
  312. for x := range pc.Calendar_panel_days {
  313. pc.Calendar_panel_days[x] = 0
  314. }
  315. // pc.Calendar_panel_days = make([]int, 6*7)
  316. var row int = 0
  317. for x := 0; x < month_last_day; x++ {
  318. var dow int = (x + First_Weekday) % 7
  319. if x != 0 && dow == 0 {
  320. row++
  321. }
  322. pc.Calendar_panel_days[row*7+dow] = x + 1
  323. }
  324. // clear out
  325. for x := range pc.Calendar_day_status {
  326. pc.Calendar_day_status[x] = 0
  327. }
  328. // pc.Calendar_day_status = make([]int, 31)
  329. last_played := pc.DB.WhenPlayed()
  330. for played, hands := range last_played {
  331. if played >= month.Unix() {
  332. // Ok, it is within the range
  333. var played_day int = time.Unix(played, 0).Day()
  334. if hands < pc.Total_hands {
  335. pc.Calendar_day_status[played_day-1] = 1
  336. } else {
  337. pc.Calendar_day_status[played_day-1] = 2
  338. }
  339. }
  340. }
  341. // Get play_days_ahead from config
  342. var play_days_ahead int = 0
  343. if playdays, has := (*pc.Config)["play_days_ahead"]; has {
  344. play_days_ahead, _ = strconv.Atoi(playdays)
  345. }
  346. // Mark all days ahead as NNY.
  347. for d := 0; d < 31; d++ {
  348. if today_day+play_days_ahead-1 < d {
  349. pc.Calendar_day_status[d] = 3
  350. }
  351. }
  352. // Get current month as string
  353. var current string = strings.ToUpper(CurrentMonth(month))
  354. if len(current) < 6 {
  355. current = " " + current + " "
  356. }
  357. // FUTURE: Replace this with TDF rendering of Month.
  358. // make_month
  359. var MonthPanel door.Panel = door.Panel{X: 3,
  360. Y: 3,
  361. Width: 3,
  362. Style: door.DOUBLE,
  363. BorderColor: door.ColorText("BOLD YELLOW ON BLACK"),
  364. }
  365. for _, c := range current {
  366. MonthPanel.Lines = append(MonthPanel.Lines,
  367. door.Line{Text: fmt.Sprintf(" %c ", c)})
  368. }
  369. pc.Calendar.AddPanel(MonthPanel)
  370. // make_weekday
  371. var WeekdayPanel door.Panel = door.Panel{X: 8,
  372. Y: 3,
  373. Width: 41,
  374. Style: door.DOUBLE,
  375. BorderColor: door.ColorText("BOLD CYAN ON BLACK"),
  376. }
  377. WeekdayPanel.Lines = append(WeekdayPanel.Lines,
  378. door.Line{Text: " SUN MON TUE WED THU FRI SAT "})
  379. pc.Calendar.AddPanel(WeekdayPanel)
  380. // make_calendar_panel
  381. var CalendarPanel door.Panel
  382. {
  383. W := 41
  384. CalendarPanel = door.Panel{Width: W,
  385. Style: door.DOUBLE,
  386. BorderColor: door.ColorText("CYAN ON BLACK")}
  387. calendarRender := func(text string) string {
  388. // var result string
  389. // var lastColor string
  390. result := door.Render{Line: text}
  391. digits := door.ColorText("CYAN ON BLACK")
  392. digits_play := door.ColorText("BOLD CYAN ON BLACK")
  393. spaces := door.ColorText("WHITE ON BLACK")
  394. open := door.ColorText("BOLD GREEN ON BLACK")
  395. hands := door.ColorText("BOLD YELLOW ON BLACK")
  396. full := door.ColorText("RED ON BLACK")
  397. nny := door.ColorText("BOLD BLACK ON BLACK")
  398. for days := 0; days < 7; days++ {
  399. var dayText string
  400. if days == 0 {
  401. result.Append(spaces, 1)
  402. }
  403. dayText = text[1+days*6 : (1+days*6)+3]
  404. // if dayText[1] == ' ' {
  405. if dayText[0] == ' ' {
  406. result.Append(spaces, 3)
  407. } else {
  408. // Something is here
  409. cday := dayText[2]
  410. if cday == 'o' || cday == 'h' {
  411. result.Append(digits_play, 2)
  412. } else {
  413. result.Append(digits, 2)
  414. }
  415. switch cday {
  416. case 'o':
  417. result.Append(open, 1)
  418. case 'h':
  419. result.Append(hands, 1)
  420. case 'x':
  421. result.Append(full, 1)
  422. case 'u':
  423. result.Append(nny, 1)
  424. }
  425. }
  426. if days == 6 {
  427. result.Append(spaces, 1)
  428. } else {
  429. result.Append(spaces, 3)
  430. }
  431. }
  432. return result.Result
  433. }
  434. for row := 0; row < 6; row++ {
  435. // Do this, or it doesn't get captured correctly.
  436. _row := row
  437. // _pc := pc
  438. calendarUpdate := func() string {
  439. var text string
  440. for d := 0; d < 7; d++ {
  441. text += " "
  442. v := pc.Calendar_panel_days[(_row*7)+d]
  443. if v == 0 {
  444. text += " "
  445. } else {
  446. text += fmt.Sprintf("%-2d", v)
  447. status := pc.Calendar_day_status[v-1]
  448. switch status {
  449. case 0:
  450. text += "o"
  451. case 1:
  452. text += "h"
  453. case 2:
  454. text += "x"
  455. case 3:
  456. text += "u"
  457. }
  458. }
  459. if d == 6 {
  460. text += " "
  461. } else {
  462. text += " "
  463. }
  464. }
  465. return text
  466. }
  467. log.Printf("line %d: [%s]\n", row, calendarUpdate())
  468. CalendarPanel.Lines = append(CalendarPanel.Lines,
  469. door.Line{Text: calendarUpdate(),
  470. UpdateF: calendarUpdate,
  471. RenderF: calendarRender})
  472. }
  473. }
  474. CalendarPanel.X = 8
  475. CalendarPanel.Y = 6
  476. pc.Calendar.AddPanel(CalendarPanel)
  477. }
  478. // end make_calendar
  479. }
  480. func (pc *PlayCards) Play() int {
  481. // this defaults (for now) to playing today (if unplayed).
  482. // Otherwise display calendar.
  483. pc.Play_day = time.Now()
  484. NormalizeDate(&pc.Play_day)
  485. pc.Play_day_t = pc.Play_day.Unix()
  486. played := pc.DB.HandsPlayedOnDay(pc.Play_day_t)
  487. log.Printf("HandsPlayedOnDay(%d), %d", pc.Play_day_t, played)
  488. var r int
  489. if played == 0 {
  490. pc.Door.Write(door.CRNL + door.Reset + "Let's play today..." + door.CRNL)
  491. time.Sleep(time.Second * time.Duration(2))
  492. pc.Hand = 1
  493. r = pc.PlayCards()
  494. if r != 'D' {
  495. return r
  496. }
  497. } else {
  498. if played < pc.Total_hands {
  499. // let's finish today...
  500. pc.Door.Write(door.CRNL + door.Reset + "Let's finish today..." + door.CRNL)
  501. time.Sleep(time.Second * time.Duration(2))
  502. pc.Hand = played + 1
  503. r = pc.PlayCards()
  504. if r != 'D' {
  505. return r
  506. }
  507. }
  508. }
  509. CALENDAR_UPDATE:
  510. // pc.UpdateCalendarDays()
  511. month_t := pc.Calendar_day_t[0]
  512. last_played := pc.DB.WhenPlayed()
  513. for played, hands := range last_played {
  514. if played >= month_t {
  515. // Ok, it is within the range
  516. var played_day int = time.Unix(played, 0).Day()
  517. if hands < pc.Total_hands {
  518. pc.Calendar_day_status[played_day-1] = 1
  519. } else {
  520. pc.Calendar_day_status[played_day-1] = 2
  521. }
  522. }
  523. }
  524. log.Printf("%#v\n", pc.Calendar)
  525. log.Println("Calendar.Update()")
  526. pc.Calendar.Update()
  527. log.Println("Calendar.Output()")
  528. pc.Door.Write(pc.Calendar.Output())
  529. var has_playable_day bool = false
  530. for x := 0; x < 31; x++ {
  531. status := pc.Calendar_day_status[x]
  532. if status == 0 || status == 1 {
  533. has_playable_day = true
  534. break
  535. }
  536. }
  537. if !has_playable_day {
  538. pc.Door.Write(door.CRNL + "Sorry, there are no days available to play." + door.CRNL)
  539. r = press_a_key(pc.Door)
  540. if r < 0 {
  541. return r
  542. } else {
  543. return 'Q'
  544. }
  545. }
  546. log.Println("Choose Day")
  547. pc.Door.Write(door.CRNL + "Please choose a day : " + door.SavePos)
  548. AGAIN:
  549. log.Println("Input")
  550. var toplay string = pc.Door.Input(3)
  551. log.Printf("Input was: %s\n", toplay)
  552. pc.Door.Write(door.RestorePos)
  553. var number int
  554. var err error
  555. number, err = strconv.Atoi(toplay)
  556. if err != nil {
  557. number = 0
  558. }
  559. if number == 0 {
  560. return ' '
  561. }
  562. var status int
  563. if number <= 31 {
  564. status = pc.Calendar_day_status[number-1]
  565. if status == 0 {
  566. // play full day
  567. pc.Hand = 1
  568. pc.Play_day_t = pc.Calendar_day_t[number-1]
  569. pc.Play_day = time.Unix(pc.Play_day_t, 0)
  570. r = pc.PlayCards()
  571. if r == 'D' {
  572. goto CALENDAR_UPDATE
  573. }
  574. return r
  575. }
  576. if status == 1 {
  577. // Play half day
  578. pc.Play_day_t = pc.Calendar_day_t[number-1]
  579. pc.Play_day = time.Unix(pc.Play_day_t, 0)
  580. played := pc.DB.HandsPlayedOnDay(pc.Play_day_t)
  581. if played < pc.Total_hands {
  582. pc.Hand = played + 1
  583. r = pc.PlayCards()
  584. if r == 'D' {
  585. goto CALENDAR_UPDATE
  586. }
  587. return r
  588. }
  589. }
  590. goto AGAIN
  591. }
  592. return ' '
  593. }
  594. func (pc *PlayCards) PlayCards() int {
  595. pc.InitValues()
  596. var game_width int
  597. var game_height int = 20
  598. // pos := &CardPos[27]
  599. game_width = CardPos[27].X + 5
  600. MX := door.Width
  601. MY := door.Height
  602. pc.Off_X = (MX - game_width) / 2
  603. pc.Off_Y = (MY - game_height) / 2
  604. // _ = off_x
  605. // We can now position things properly centered
  606. pc.SpaceAceTriPeaks.X = (MX - pc.SpaceAceTriPeaks.Width) / 2
  607. pc.SpaceAceTriPeaks.Y = pc.Off_Y
  608. pc.Off_Y += 3
  609. currentDefault := pc.DB.GetSetting("DeckColor", "ALL")
  610. Next_Hand:
  611. deck_color := StringToANSIColor(currentDefault)
  612. pc.DeckPanel.SetBackColor(deck_color)
  613. pc.Play_card = 28
  614. pc.Select_card = 23
  615. pc.Score = 0
  616. pc.Current_streak = 0
  617. // Use play day to seed RNG
  618. {
  619. // Secret Squirrel method of seeding the RNG
  620. seed_seq := make([]byte, 0)
  621. for _, seed := range pc.Seeds {
  622. ba := Int32toByteArray(seed)
  623. seed_seq = append(seed_seq, ba[:]...)
  624. // sha1.Sum(ba[:])
  625. }
  626. pd := Int64toByteArray(pc.Play_day_t)
  627. seed_seq = append(seed_seq, pd[:]...)
  628. // We also need the hand # that we're playing.
  629. seed_seq = append(seed_seq, byte(pc.Hand))
  630. result := sha1.Sum(seed_seq)
  631. var seed int64 = int64(binary.BigEndian.Uint64(result[0:8]))
  632. pc.RNG.Seed(seed)
  633. // I'm seeing changes in the seed_seq bytes, but the seed is the same number.
  634. // log.Printf("%#v\nSeed %d\nLen %d\nresult %#v\n", seed_seq, seed, len(seed_seq), result)
  635. pc.Deck = ShuffleCards(pc.RNG, 1)
  636. pc.State = MakeCardStates(1)
  637. }
  638. // Position the panels
  639. {
  640. off_yp := pc.Off_Y + 11
  641. left_panel_x := CardPos[18].X
  642. right_panel_x := CardPos[15].X
  643. pc.ScorePanel.X = left_panel_x + pc.Off_X
  644. pc.ScorePanel.Y = off_yp
  645. pc.StreakPanel.X = right_panel_x + pc.Off_X
  646. pc.StreakPanel.Y = off_yp
  647. pc.CmdPanel.X = left_panel_x + pc.Off_X
  648. pc.CmdPanel.Y = off_yp + 5
  649. next_off_x := (MX - pc.NextQuitPanel.Width) / 2
  650. pc.NextQuitPanel.X = next_off_x
  651. pc.NextQuitPanel.Y = CardPos[10].Y + pc.Off_Y
  652. }
  653. var Dealing bool = true
  654. var r int = 0
  655. pc.Redraw(Dealing)
  656. Dealing = false
  657. pc.LeftPanel.Update()
  658. pc.Door.Write(door.Reset)
  659. var c *door.Panel
  660. var in_game bool = true
  661. var save_streak bool = false
  662. var waiting int = 0
  663. for in_game {
  664. var output string
  665. output = pc.ScorePanel.Update()
  666. if output != "" {
  667. pc.Door.Write(output)
  668. {
  669. // Redisplay Mark / Selected Card
  670. Pos := &CardPos[pc.Select_card]
  671. c = &pc.DeckPanel.Mark[1]
  672. c.X = Pos.X + pc.Off_X + 2
  673. c.Y = Pos.Y + pc.Off_Y + 2
  674. pc.Door.Write(c.Output())
  675. }
  676. }
  677. if save_streak {
  678. save_streak = false
  679. pc.DB.SetSetting("best_streak", strconv.Itoa(pc.Best_streak))
  680. }
  681. waiting = 0
  682. for waiting < int(door.Inactivity) {
  683. r = pc.Door.WaitKey(1, 0)
  684. if r == -1 {
  685. // TIMEOUT is expected here
  686. waiting++
  687. output = pc.ScorePanel.Update()
  688. if output != "" {
  689. pc.Door.Write(output)
  690. {
  691. // Redisplay Mark / Selected Card
  692. Pos := &CardPos[pc.Select_card]
  693. c = &pc.DeckPanel.Mark[1]
  694. c.X = Pos.X + pc.Off_X + 2
  695. c.Y = Pos.Y + pc.Off_Y + 2
  696. pc.Door.Write(c.Output())
  697. }
  698. }
  699. } else {
  700. break
  701. }
  702. }
  703. if r > 0 {
  704. // Not a timeout
  705. /*
  706. if r < 0x1000 {
  707. // not a function key
  708. r = int(unicode.ToUpper(rune(r)))
  709. }
  710. */
  711. switch r {
  712. case '\x0d':
  713. // Next Card
  714. if pc.Current_streak > pc.Best_streak {
  715. pc.Best_streak = pc.Current_streak
  716. save_streak = true
  717. pc.Door.Write(pc.StreakPanel.Update())
  718. }
  719. if pc.Play_card < 51 {
  720. pc.Play_card++
  721. pc.Current_streak = 0
  722. pc.Door.Write(pc.StreakPanel.Update())
  723. pc.Door.Write(pc.LeftPanel.Update())
  724. // Deal the next card
  725. if pc.Play_card == 51 {
  726. // out of cards
  727. cpos := &CardPos[29]
  728. c = &pc.DeckPanel.Backs[0]
  729. c.X = cpos.X + pc.Off_X
  730. c.Y = cpos.Y + pc.Off_Y
  731. pc.Door.Write(c.Output())
  732. }
  733. cpos := &CardPos[28]
  734. c = &pc.DeckPanel.Cards[pc.Deck[pc.Play_card]]
  735. c.X = cpos.X + pc.Off_X
  736. c.Y = cpos.Y + pc.Off_Y
  737. pc.Door.Write(c.Output())
  738. }
  739. case 'R', 'r':
  740. pc.Redraw(false)
  741. case 'Q', 'q':
  742. // Possibly prompt here for [N]ext hand or [Q]uit
  743. if pc.Current_streak > pc.Best_streak {
  744. pc.Best_streak = pc.Current_streak
  745. pc.Door.Write(pc.StreakPanel.Update())
  746. }
  747. pc.NextQuitPanel.Update()
  748. pc.Door.Write(pc.NextQuitPanel.Output())
  749. if pc.State[26] == 2 {
  750. pc.Door.Write(door.ColorText("BLACK"))
  751. } else {
  752. pc.Door.Write(door.Reset)
  753. }
  754. if pc.Hand < pc.Total_hands {
  755. r = pc.Door.GetOneOf("CNQ")
  756. } else {
  757. r = pc.Door.GetOneOf("CDQ")
  758. }
  759. if r == 'C' {
  760. // Continue
  761. pc.Redraw(false)
  762. } else {
  763. if r == 'D' || r == 'Q' {
  764. if pc.Score >= 50 {
  765. pc.DB.SaveScore(time.Now().Unix(), pc.Play_day_t, pc.Hand, 0, pc.Score)
  766. }
  767. in_game = false
  768. } else {
  769. if r == 'N' {
  770. pc.DB.SaveScore(time.Now().Unix(), pc.Play_day_t, pc.Hand, 0, pc.Score)
  771. pc.Hand++
  772. goto Next_Hand
  773. }
  774. }
  775. }
  776. case ' ', '5':
  777. if CanPlay(pc.Deck[pc.Select_card],
  778. pc.Deck[pc.Play_card]) {
  779. pc.Current_streak++
  780. pc.Door.Write(pc.StreakPanel.Update())
  781. pc.Score += 10
  782. if pc.Current_streak > 1 {
  783. pc.Score += pc.Current_streak * 5
  784. }
  785. pc.Door.Write(pc.ScorePanel.Update())
  786. // Play card
  787. pc.State[pc.Select_card] = 2
  788. {
  789. // Swap out the select card with the play card
  790. temp := pc.Deck[pc.Select_card]
  791. pc.Deck[pc.Select_card] = pc.Deck[pc.Play_card]
  792. pc.Deck[pc.Play_card] = temp
  793. // Select card is invalidated here. Find new card.
  794. check := Unblocks(pc.Select_card)
  795. var left bool = false
  796. var right bool = false
  797. for _, chk := range check {
  798. blk := Blocks[chk]
  799. if blk[0] == pc.Select_card {
  800. right = true
  801. }
  802. if blk[1] == pc.Select_card {
  803. left = true
  804. }
  805. }
  806. cardback_color := pc.DeckPanel.Backs[1].Lines[0].DefaultColor
  807. pc.Door.Write(RemoveCard(pc.Select_card, cardback_color, pc.Off_X, pc.Off_Y, left, right))
  808. // Redraw play card #28 ("old" select_card)
  809. cpos := &CardPos[28]
  810. c = &pc.DeckPanel.Cards[pc.Deck[pc.Play_card]]
  811. c.X = cpos.X + pc.Off_X
  812. c.Y = cpos.Y + pc.Off_Y
  813. pc.Door.Write(c.Output())
  814. // Did we unhide a card?
  815. var new_card_shown int = -1
  816. if len(check) > 0 {
  817. for _, chk := range check {
  818. blk := Blocks[chk]
  819. if pc.State[blk[0]] == 2 &&
  820. pc.State[blk[1]] == 2 {
  821. pc.State[chk] = 1
  822. cpos := &CardPos[chk]
  823. c = &pc.DeckPanel.Cards[pc.Deck[chk]]
  824. c.X = cpos.X + pc.Off_X
  825. c.Y = cpos.Y + pc.Off_Y
  826. pc.Door.Write(c.Output())
  827. new_card_shown = chk
  828. }
  829. }
  830. } else {
  831. // top card cleared
  832. cpos := &CardPos[pc.Select_card]
  833. pc.Door.Write(door.Goto(cpos.X+pc.Off_X, cpos.Y+pc.Off_Y) + Bonus())
  834. pc.Score += 100
  835. pc.State[pc.Select_card] = 3 // Handle in "redraw"
  836. pc.Door.Write(pc.ScorePanel.Update())
  837. }
  838. // Find new number for select_card.
  839. if new_card_shown != -1 {
  840. pc.Select_card = new_card_shown
  841. } else {
  842. new_select := FindClosestActiveCard(&pc.State, pc.Select_card)
  843. if new_select != -1 {
  844. pc.Select_card = new_select
  845. } else {
  846. log.Println("Winner")
  847. pc.Select_card = -1
  848. pc.Score += 15 * (51 - pc.Play_card)
  849. pc.Door.Write(pc.ScorePanel.Update())
  850. // Save Score
  851. pc.DB.SaveScore(time.Now().Unix(), pc.Play_day_t, pc.Hand, 1, pc.Score)
  852. pc.NextQuitPanel.Update()
  853. pc.Door.Write(pc.NextQuitPanel.Output())
  854. if pc.State[26] == 2 {
  855. pc.Door.Write(door.ColorText("BLACK"))
  856. } else {
  857. pc.Door.Write(door.Reset)
  858. }
  859. if pc.Hand < pc.Total_hands {
  860. r = pc.Door.GetOneOf("NQ")
  861. } else {
  862. r = pc.Door.GetOneOf("DQ")
  863. }
  864. if r == 'N' {
  865. pc.Hand++
  866. goto Next_Hand
  867. }
  868. in_game = false
  869. // if r == 'D' ?
  870. if r == 'Q' {
  871. // this seemed odd. We quit the game ?
  872. r = 'Q'
  873. }
  874. break
  875. }
  876. }
  877. // Update the select_card marker
  878. Pos := &CardPos[pc.Select_card]
  879. c = &pc.DeckPanel.Mark[1]
  880. c.X = Pos.X + pc.Off_X + 2
  881. c.Y = Pos.Y + pc.Off_Y + 2
  882. pc.Door.Write(c.Output() + door.Reset)
  883. }
  884. }
  885. case door.XKEY_LEFT_ARROW, '4':
  886. var new_select int = FindNextActiveCard(true, &pc.State, pc.Select_card)
  887. if new_select >= 0 {
  888. Pos := &CardPos[pc.Select_card]
  889. c = &pc.DeckPanel.Mark[0]
  890. c.X = Pos.X + pc.Off_X + 2
  891. c.Y = Pos.Y + pc.Off_Y + 2
  892. pc.Door.Write(c.Output())
  893. pc.Select_card = new_select
  894. Pos = &CardPos[pc.Select_card]
  895. c = &pc.DeckPanel.Mark[1]
  896. c.X = Pos.X + pc.Off_X + 2
  897. c.Y = Pos.Y + pc.Off_Y + 2
  898. pc.Door.Write(c.Output() + door.Reset)
  899. }
  900. case door.XKEY_RIGHT_ARROW, '6':
  901. var new_select int = FindNextActiveCard(false, &pc.State, pc.Select_card)
  902. if new_select >= 0 {
  903. Pos := &CardPos[pc.Select_card]
  904. c = &pc.DeckPanel.Mark[0]
  905. c.X = Pos.X + pc.Off_X + 2
  906. c.Y = Pos.Y + pc.Off_Y + 2
  907. pc.Door.Write(c.Output())
  908. pc.Select_card = new_select
  909. Pos = &CardPos[pc.Select_card]
  910. c = &pc.DeckPanel.Mark[1]
  911. c.X = Pos.X + pc.Off_X + 2
  912. c.Y = Pos.Y + pc.Off_Y + 2
  913. pc.Door.Write(c.Output() + door.Reset)
  914. }
  915. }
  916. } else {
  917. in_game = false
  918. }
  919. }
  920. return r
  921. }
  922. func (pc *PlayCards) Redraw(Dealing bool) {
  923. // stars.display()
  924. pc.Door.Write(door.Clrscr + door.Reset)
  925. pc.Door.Write(pc.SpaceAceTriPeaks.Output())
  926. var c *door.Panel
  927. {
  928. // Step 1: Draw the deck "source"
  929. pos := &CardPos[29]
  930. if pc.Play_card == 51 {
  931. // out of cards
  932. pos.Level = 0
  933. }
  934. c = &pc.DeckPanel.Backs[pos.Level]
  935. c.X = pos.X + pc.Off_X
  936. c.Y = pos.Y + pc.Off_Y
  937. pc.LeftPanel.X = pos.X + pc.Off_X
  938. pc.LeftPanel.Y = pos.Y + pc.Off_Y + 3 // const height see deck
  939. pc.ScorePanel.Update()
  940. pc.LeftPanel.Update()
  941. pc.StreakPanel.Update()
  942. pc.CmdPanel.Update()
  943. pc.Door.Write(pc.ScorePanel.Output())
  944. pc.Door.Write(pc.LeftPanel.Output())
  945. pc.Door.Write(pc.StreakPanel.Output())
  946. pc.Door.Write(pc.CmdPanel.Output())
  947. pc.Door.Write(c.Output())
  948. if Dealing {
  949. time.Sleep(time.Second)
  950. }
  951. }
  952. var x_max int = 29
  953. if Dealing {
  954. x_max = 28
  955. }
  956. for x := 0; x < x_max; x++ {
  957. pos := &CardPos[x]
  958. if Dealing {
  959. time.Sleep(time.Duration(75) * time.Millisecond)
  960. }
  961. // log.Printf("Redraw(%d, %d)", x, pos.Level)
  962. if Dealing {
  963. c = &pc.DeckPanel.Backs[pos.Level]
  964. c.X = pos.X + pc.Off_X
  965. c.Y = pos.Y + pc.Off_Y
  966. pc.Door.Write(c.Output())
  967. } else {
  968. switch pc.State[x] {
  969. case 0:
  970. c = &pc.DeckPanel.Backs[pos.Level]
  971. c.X = pos.X + pc.Off_X
  972. c.Y = pos.Y + pc.Off_Y
  973. pc.Door.Write(c.Output())
  974. case 1:
  975. if x == 28 {
  976. c = &pc.DeckPanel.Cards[pc.Deck[pc.Play_card]]
  977. } else {
  978. c = &pc.DeckPanel.Cards[pc.Deck[x]]
  979. }
  980. c.X = pos.X + pc.Off_X
  981. c.Y = pos.Y + pc.Off_Y
  982. pc.Door.Write(c.Output())
  983. case 2:
  984. // no card to draw.
  985. case 3:
  986. // peak cleared, draw bonus
  987. var output string = door.Goto(pos.X+pc.Off_X, pos.Y+pc.Off_Y)
  988. output += Bonus()
  989. pc.Door.Write(output)
  990. }
  991. }
  992. }
  993. if Dealing {
  994. for x := 18; x < 29; x++ {
  995. pc.State[x] = 1
  996. // CardPos[x] X Y Level
  997. Pos := &CardPos[x]
  998. time.Sleep(time.Duration(200) * time.Millisecond)
  999. c = &pc.DeckPanel.Cards[pc.Deck[x]]
  1000. c.X = Pos.X + pc.Off_X
  1001. c.Y = Pos.Y + pc.Off_Y
  1002. pc.Door.Write(c.Output())
  1003. }
  1004. }
  1005. {
  1006. Pos := &CardPos[pc.Select_card]
  1007. c = &pc.DeckPanel.Mark[1]
  1008. c.X = Pos.X + pc.Off_X + 2
  1009. c.Y = Pos.Y + pc.Off_Y + 2
  1010. pc.Door.Write(c.Output())
  1011. }
  1012. }
  1013. func Bonus() string {
  1014. return door.ColorText("BOLD YELLOW") + "BONUS"
  1015. }
  1016. func (pc *PlayCards) UpdateCalendarDays() {
  1017. }