playcards.go 28 KB

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