playcards.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824
  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. "unicode"
  13. )
  14. func StringToANSIColor(colorCode string) string {
  15. colors := []string{
  16. "BLUE",
  17. "BROWN",
  18. "RED",
  19. "CYAN",
  20. "GREEN",
  21. "MAGENTA",
  22. "WHITE",
  23. }
  24. code := strings.ToUpper(colorCode)
  25. for _, c := range colors {
  26. if c == code {
  27. return door.ColorText(code)
  28. }
  29. }
  30. if code == "ALL" {
  31. rand.Seed(time.Now().UnixNano())
  32. pos := rand.Intn(len(colors))
  33. return door.ColorText(colors[pos])
  34. }
  35. return door.ColorText("WHITE")
  36. }
  37. type PlayCards struct {
  38. Door *door.Door
  39. DB *DBData
  40. Config *map[string]string
  41. RNG *rand.Rand
  42. Seeds []int32
  43. Total_hands int
  44. Play_card int
  45. Current_streak int
  46. Best_streak int
  47. Select_card int
  48. Score int
  49. Play_day time.Time
  50. Play_day_t int64
  51. Days_played int
  52. Hand int
  53. Day_status [42]int
  54. SpaceAceTriPeaks door.Panel
  55. ScorePanel door.Panel
  56. StreakPanel door.Panel
  57. LeftPanel door.Panel
  58. CmdPanel door.Panel
  59. NextQuitPanel door.Panel
  60. Calendar door.Panel
  61. DeckPanel Deck
  62. Deck []DeckType
  63. State []DeckType
  64. Off_X int
  65. Off_Y int
  66. }
  67. // Adjust date to 2:00 AM
  68. func NormalizeDate(date *time.Time) {
  69. offset := time.Duration(date.Second())*time.Second +
  70. time.Duration(date.Minute())*time.Minute +
  71. time.Duration(date.Hour()-2)*time.Hour +
  72. time.Duration(date.Nanosecond())*time.Nanosecond
  73. *date = date.Add(-offset)
  74. }
  75. func cmdLineRender(bracket string, inner string, outer string) func(string) string {
  76. cmdRender := func(input string) string {
  77. var result string
  78. inOuter := true
  79. var lastColor string
  80. for _, c := range input {
  81. if c == '[' {
  82. inOuter = false
  83. if lastColor != bracket {
  84. result += bracket
  85. lastColor = bracket
  86. }
  87. result += string(c)
  88. continue
  89. }
  90. if c == ']' {
  91. inOuter = true
  92. if lastColor != bracket {
  93. result += bracket
  94. lastColor = bracket
  95. }
  96. result += string(c)
  97. continue
  98. }
  99. if inOuter {
  100. if lastColor != outer {
  101. result += outer
  102. lastColor = outer
  103. }
  104. } else {
  105. if lastColor != inner {
  106. result += inner
  107. lastColor = inner
  108. }
  109. }
  110. result += string(c)
  111. }
  112. return result
  113. }
  114. return cmdRender
  115. }
  116. func (pc *PlayCards) InitValues() {
  117. hands := pc.DB.GetSetting("hands_per_day", "3")
  118. pc.Total_hands, _ = strconv.Atoi(hands)
  119. pc.Play_card = 28
  120. pc.Current_streak = 0
  121. pc.Best_streak = 0
  122. best := pc.DB.GetSetting("best_streak", "0")
  123. pc.Best_streak, _ = strconv.Atoi(best)
  124. pc.Select_card = 23
  125. pc.Score = 0
  126. }
  127. func (pc *PlayCards) Init() {
  128. // init_values()
  129. pc.InitValues()
  130. // PlayCards::PlayCards()
  131. pc.Play_day = time.Now()
  132. NormalizeDate(&pc.Play_day)
  133. pc.Play_day_t = pc.Play_day.Unix()
  134. pc.DeckPanel.Init()
  135. var last_played int64
  136. last := pc.DB.GetSetting("last_played", "0")
  137. last_played, _ = strconv.ParseInt(last, 10, 64)
  138. days := pc.DB.GetSetting("days_played", "0")
  139. pc.Days_played, _ = strconv.Atoi(days)
  140. if last_played != pc.Play_day_t {
  141. pc.DB.SetSetting("last_played", strconv.FormatInt(pc.Play_day_t, 10))
  142. pc.DB.SetSetting("days_played", "0")
  143. pc.Days_played = 0
  144. }
  145. pc.Seeds = make([]int32, 0)
  146. // config _seed
  147. parts := strings.Split((*pc.Config)["_seed"], ",")
  148. for _, seed := range parts {
  149. i, _ := strconv.ParseInt(seed, 10, 32)
  150. pc.Seeds = append(pc.Seeds, int32(i))
  151. }
  152. pc.Hand = 0
  153. pc.Total_hands = 0
  154. // spaceAceTriPeaks = make_tripeaks();
  155. {
  156. tripeakstext := " " + SPACEACE + " - Tri-Peaks Solitaire v" + SPACEACE_VERSION + " "
  157. pc.SpaceAceTriPeaks = door.Panel{Width: len(tripeakstext),
  158. Style: door.SINGLE,
  159. BorderColor: door.ColorText("CYAN ON BLACK"),
  160. }
  161. pc.SpaceAceTriPeaks.Lines = append(pc.SpaceAceTriPeaks.Lines,
  162. door.Line{Text: tripeakstext})
  163. }
  164. svRender := RenderStatusValue(door.ColorText("BOLD WHITE ON BLUE"),
  165. door.ColorText("BOLD YELLOW ON BLUE"))
  166. // score_panel = make_score_panel();
  167. {
  168. W := 25
  169. pc.ScorePanel = door.Panel{Width: W}
  170. text := "Name: " + pc.Door.Config.Real_name
  171. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  172. door.Line{Text: fmt.Sprintf("%*s", -W, text), RenderF: svRender})
  173. scoreUpdate := func() string {
  174. txt := fmt.Sprintf("Score: %d", pc.Score)
  175. txt += strings.Repeat(" ", W-len(txt))
  176. return txt
  177. }
  178. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  179. door.Line{Text: scoreUpdate(),
  180. UpdateF: scoreUpdate,
  181. RenderF: svRender,
  182. })
  183. timeUpdate := func() string {
  184. var left int = int(pc.Door.TimeLeft().Minutes())
  185. var used int = int(pc.Door.TimeUsed().Minutes())
  186. txt := fmt.Sprintf("Time used: %3d / %3d", used, left)
  187. txt += strings.Repeat(" ", W-len(txt))
  188. return txt
  189. }
  190. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  191. door.Line{Text: timeUpdate(),
  192. UpdateF: timeUpdate,
  193. RenderF: svRender,
  194. })
  195. handUpdate := func() string {
  196. txt := fmt.Sprintf("Playing Hand %d of %d", pc.Hand, pc.Total_hands)
  197. txt += strings.Repeat(" ", W-len(txt))
  198. return txt
  199. }
  200. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  201. door.Line{Text: handUpdate(),
  202. UpdateF: handUpdate,
  203. RenderF: svRender,
  204. })
  205. }
  206. // streak_panel = make_streak_panel();
  207. {
  208. W := 20
  209. pc.StreakPanel = door.Panel{Width: W}
  210. dateUpdate := func() string {
  211. format, ok := (*pc.Config)["date_format"]
  212. if !ok {
  213. format = "January 2"
  214. }
  215. txt := fmt.Sprintf("Playing: %s", pc.Play_day.Format(format))
  216. txt += strings.Repeat(" ", W-len(txt))
  217. return txt
  218. }
  219. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  220. door.Line{Text: dateUpdate(),
  221. UpdateF: dateUpdate,
  222. RenderF: svRender,
  223. })
  224. currentUpdate := func() string {
  225. txt := fmt.Sprintf("Current Streak: %d", pc.Current_streak)
  226. txt += strings.Repeat(" ", W-len(txt))
  227. return txt
  228. }
  229. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  230. door.Line{Text: currentUpdate(),
  231. UpdateF: currentUpdate,
  232. RenderF: svRender,
  233. })
  234. longestUpdate := func() string {
  235. txt := fmt.Sprintf("Longest Streak: %d", pc.Best_streak)
  236. txt += strings.Repeat(" ", W-len(txt))
  237. return txt
  238. }
  239. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  240. door.Line{Text: longestUpdate(),
  241. UpdateF: longestUpdate,
  242. RenderF: svRender,
  243. })
  244. }
  245. // left_panel = make_left_panel();
  246. {
  247. W := 13
  248. pc.LeftPanel = door.Panel{Width: W}
  249. leftUpdate := func() string {
  250. txt := fmt.Sprintf("Cards left:%d", 51-pc.Play_card)
  251. txt += strings.Repeat(" ", W-len(txt))
  252. return txt
  253. }
  254. pc.LeftPanel.Lines = append(pc.LeftPanel.Lines,
  255. door.Line{Text: leftUpdate(),
  256. UpdateF: leftUpdate,
  257. RenderF: svRender,
  258. })
  259. }
  260. // cmd_panel = make_command_panel();
  261. {
  262. W := 76
  263. pc.CmdPanel = door.Panel{Width: W}
  264. var commands string
  265. if door.Unicode {
  266. commands = "[4/\u25c4] Left [6/\u25ba] Right [Space] Play Card [Enter] Draw [Q]uit [R]edraw [H]elp"
  267. commands += strings.Repeat(" ", W-len([]rune(commands)))
  268. } else {
  269. commands = "[4/\x11] Left [6/\x10] Right [Space] Play Card [Enter] Draw [Q]uit [R]edraw [H]elp"
  270. commands += strings.Repeat(" ", W-len(commands))
  271. }
  272. cmdRender := cmdLineRender(door.ColorText("BOLD YELLOW ON BLUE"),
  273. door.ColorText("BOLD CYAN ON BLUE"),
  274. door.ColorText("BOLD GREEN ON BLUE"))
  275. pc.CmdPanel.Lines = append(pc.CmdPanel.Lines,
  276. door.Line{Text: commands,
  277. RenderF: cmdRender})
  278. }
  279. // next_quit_panel = make_next_panel();
  280. {
  281. W := 50
  282. pc.NextQuitPanel = door.Panel{Width: W,
  283. Style: door.DOUBLE,
  284. BorderColor: door.ColorText("BOLD YELLOW ON BLUE"),
  285. }
  286. nextUpdate := func() string {
  287. var text string
  288. if pc.Select_card != -1 {
  289. text = "[C]ontinue this hand"
  290. }
  291. if pc.Hand < pc.Total_hands {
  292. text += " [N]ext Hand [Q]uit"
  293. } else {
  294. text += " [D]one [Q]uit"
  295. }
  296. text = " " + text + " "
  297. text += strings.Repeat(" ", W-len(text))
  298. return text
  299. }
  300. pc.NextQuitPanel.Lines = append(pc.NextQuitPanel.Lines,
  301. door.Line{Text: nextUpdate(),
  302. UpdateF: nextUpdate,
  303. RenderF: cmdLineRender(door.ColorText("BOLD YEL ON BLU"),
  304. door.ColorText("BOLD CYAN ON BLU"),
  305. door.ColorText("BOLD GREE ON BLUE"))})
  306. }
  307. // calendar = make_calendar();
  308. {
  309. W := 41
  310. pc.Calendar = door.Panel{Width: W,
  311. Style: door.DOUBLE,
  312. BorderColor: door.ColorText("CYAN ON BLACK")}
  313. calendarRender := func(text string) string {
  314. // var result string
  315. // var lastColor string
  316. result := door.Render{Line: text}
  317. digits := door.ColorText("CYAN ON BLACK")
  318. digits_play := door.ColorText("BOLD CYAN ON BLACK")
  319. spaces := door.ColorText("WHITE ON BLACK")
  320. open := door.ColorText("BOLD GREEN ON BLACK")
  321. hands := door.ColorText("BOLD YELLOW ON BLACK")
  322. full := door.ColorText("RED ON BLACK")
  323. nny := door.ColorText("BOLD BLACK ON BLACK")
  324. for days := 0; days < 7; days++ {
  325. var dayText string
  326. if days == 0 {
  327. result.Append(spaces, 1)
  328. }
  329. dayText = text[1+days*6 : (1+days*6)+3]
  330. if dayText[1] == ' ' {
  331. result.Append(spaces, 3)
  332. } else {
  333. // Something is here
  334. cday := dayText[2]
  335. if cday == 'o' || cday == 'h' {
  336. result.Append(digits_play, 2)
  337. } else {
  338. result.Append(digits, 2)
  339. }
  340. switch cday {
  341. case 'o':
  342. result.Append(open, 1)
  343. case 'h':
  344. result.Append(hands, 1)
  345. case 'x':
  346. result.Append(full, 1)
  347. case 'u':
  348. result.Append(nny, 1)
  349. }
  350. }
  351. if days == 6 {
  352. result.Append(spaces, 1)
  353. } else {
  354. result.Append(spaces, 3)
  355. }
  356. }
  357. return result.Result
  358. }
  359. for row := 0; row < 6; row++ {
  360. calendarUpdate := func() string {
  361. var text string
  362. for d := 0; d < 7; d++ {
  363. text += " "
  364. v := pc.Day_status[(row*7)+d]
  365. if v == 0 {
  366. text += " "
  367. } else {
  368. text += fmt.Sprintf("%-2d", v)
  369. status := pc.Day_status[v-1]
  370. switch status {
  371. case 0:
  372. text += "o"
  373. case 1:
  374. text += "h"
  375. case 2:
  376. text += "x"
  377. case 3:
  378. text += "u"
  379. }
  380. }
  381. if d == 6 {
  382. text += " "
  383. } else {
  384. text += " "
  385. }
  386. }
  387. return text
  388. }
  389. pc.Calendar.Lines = append(pc.Calendar.Lines,
  390. door.Line{Text: calendarUpdate(),
  391. UpdateF: calendarUpdate,
  392. RenderF: calendarRender})
  393. }
  394. }
  395. }
  396. func (pc *PlayCards) Play() {
  397. // this defaults (for now) to playing today (if unplayed).
  398. // Otherwise display calendar.
  399. pc.Play_day = time.Now()
  400. NormalizeDate(&pc.Play_day)
  401. pc.Play_day_t = pc.Play_day.Unix()
  402. // played := db.HandsPlayedOnDay(pc.Play_day_t)
  403. played := 0
  404. var r int
  405. if played == 0 {
  406. pc.Door.Write("Let's play today..." + door.CRNL)
  407. time.Sleep(time.Second)
  408. pc.Hand = 1
  409. r = pc.PlayCards()
  410. if r != 'D' {
  411. return
  412. }
  413. } else {
  414. if played < pc.Total_hands {
  415. // let's finish today...
  416. pc.Hand = played + 1
  417. r = pc.PlayCards()
  418. if r != 'D' {
  419. return
  420. }
  421. }
  422. }
  423. /*
  424. CALENDAR_UPDATE:
  425. pc.UpdateCalendarDays()
  426. pc.Calendar.Update()
  427. pc.Door.Write(pc.Calendar.Output())
  428. */
  429. }
  430. func int32toByteArray(i int32) (arr [4]byte) {
  431. binary.BigEndian.PutUint32(arr[0:4], uint32(i))
  432. return
  433. }
  434. func int64toByteArray(i int64) (arr [8]byte) {
  435. binary.BigEndian.PutUint64(arr[0:8], uint64(i))
  436. return
  437. }
  438. func (pc *PlayCards) PlayCards() int {
  439. pc.InitValues()
  440. var game_width int
  441. var game_height int = 20
  442. pos := CardPos[27]
  443. game_width = pos.X + 5
  444. MX := door.Width
  445. MY := door.Height
  446. pc.Off_X = (MX - game_width) / 2
  447. pc.Off_Y = (MY - game_height) / 2
  448. // _ = off_x
  449. // We can now position things properly centered
  450. pc.SpaceAceTriPeaks.X = (MX - pc.SpaceAceTriPeaks.Width) / 2
  451. pc.SpaceAceTriPeaks.Y = pc.Off_Y
  452. pc.Off_Y += 3
  453. currentDefault := pc.DB.GetSetting("DeckColor", "ALL")
  454. Next_Hand:
  455. deck_color := StringToANSIColor(currentDefault)
  456. pc.DeckPanel.SetBackColor(deck_color)
  457. pc.Play_card = 28
  458. pc.Select_card = 23
  459. pc.Score = 0
  460. pc.Current_streak = 0
  461. // Use play day to seed RNG
  462. {
  463. // Secret Squirrel method of seeding the RNG
  464. Sha1 := sha1.New()
  465. seed_seq := make([]byte, 0)
  466. for _, seed := range pc.Seeds {
  467. ba := int32toByteArray(seed)
  468. seed_seq = append(seed_seq, ba[:]...)
  469. // sha1.Sum(ba[:])
  470. }
  471. pd := int64toByteArray(pc.Play_day_t)
  472. seed_seq = append(seed_seq, pd[:]...)
  473. // We also need the hand # that we're playing.
  474. seed_seq = append(seed_seq, byte(pc.Hand))
  475. Sha1.Write(seed_seq)
  476. result := Sha1.Sum(nil)
  477. var seed int64 = int64(binary.BigEndian.Uint64(result[0:8]))
  478. pc.RNG.Seed(seed)
  479. // I'm seeing changes in the seed_seq bytes, but the seed is the same number.
  480. log.Printf("%#v\nSeed %d\nLen %d\nresult %#v\n", seed_seq, seed, len(seed_seq), result)
  481. pc.Deck = ShuffleCards(pc.RNG, 1)
  482. pc.State = MakeCardStates(1)
  483. }
  484. // Position the panels
  485. {
  486. off_yp := pc.Off_Y + 11
  487. left_panel_x := CardPos[18].X
  488. right_panel_x := CardPos[15].X
  489. pc.ScorePanel.X = left_panel_x + pc.Off_X
  490. pc.ScorePanel.Y = off_yp
  491. pc.StreakPanel.X = right_panel_x + pc.Off_X
  492. pc.StreakPanel.Y = off_yp
  493. pc.CmdPanel.X = left_panel_x + pc.Off_X
  494. pc.CmdPanel.Y = off_yp + 5
  495. next_off_x := (MX - pc.NextQuitPanel.Width) / 2
  496. pc.NextQuitPanel.X = next_off_x
  497. pc.NextQuitPanel.Y = CardPos[10].Y + pc.Off_Y
  498. }
  499. var Dealing bool = true
  500. var r int = 0
  501. pc.Redraw(Dealing)
  502. Dealing = false
  503. pc.LeftPanel.Update()
  504. pc.Door.Write(door.Reset)
  505. var c *door.Panel
  506. var in_game bool = true
  507. var save_streak bool = false
  508. var waiting int = 0
  509. for in_game {
  510. var output string
  511. output = pc.ScorePanel.Update()
  512. if output != "" {
  513. pc.Door.Write(output)
  514. {
  515. // Redisplay Mark / Selected Card
  516. Pos := &CardPos[pc.Select_card]
  517. c = &pc.DeckPanel.Mark[1]
  518. c.X = Pos.X + pc.Off_X + 2
  519. c.Y = Pos.Y + pc.Off_Y + 2
  520. pc.Door.Write(c.Output())
  521. }
  522. }
  523. if save_streak {
  524. save_streak = false
  525. pc.DB.SetSetting("best_streak", strconv.Itoa(pc.Best_streak))
  526. }
  527. waiting = 0
  528. for waiting < int(door.Inactivity) {
  529. r = pc.Door.WaitKey(1, 0)
  530. if r == -1 {
  531. // TIMEOUT is expected here
  532. waiting++
  533. output = pc.ScorePanel.Update()
  534. if output != "" {
  535. pc.Door.Write(output)
  536. {
  537. // Redisplay Mark / Selected Card
  538. Pos := &CardPos[pc.Select_card]
  539. c = &pc.DeckPanel.Mark[1]
  540. c.X = Pos.X + pc.Off_X + 2
  541. c.Y = Pos.Y + pc.Off_Y + 2
  542. pc.Door.Write(c.Output())
  543. }
  544. }
  545. } else {
  546. break
  547. }
  548. }
  549. if r > 0 {
  550. // Not a timeout
  551. if r < 0x1000 {
  552. // not a function key
  553. switch unicode.ToUpper(rune(r)) {
  554. case '\x0d':
  555. // Next Card
  556. if pc.Current_streak > pc.Best_streak {
  557. pc.Best_streak = pc.Current_streak
  558. save_streak = true
  559. pc.Door.Write(pc.StreakPanel.Update())
  560. }
  561. if pc.Play_card < 51 {
  562. pc.Play_card++
  563. pc.Current_streak = 0
  564. pc.Door.Write(pc.StreakPanel.Update())
  565. pc.Door.Write(pc.LeftPanel.Update())
  566. // Deal the next card
  567. if pc.Play_card == 51 {
  568. // out of cards
  569. cpos := &CardPos[29]
  570. c = &pc.DeckPanel.Backs[0]
  571. c.X = cpos.X + pc.Off_X
  572. c.Y = cpos.Y + pc.Off_Y
  573. pc.Door.Write(c.Output())
  574. }
  575. cpos := &CardPos[28]
  576. c = &pc.DeckPanel.Cards[pc.Deck[pc.Play_card]]
  577. c.X = cpos.X + pc.Off_X
  578. c.Y = cpos.Y + pc.Off_Y
  579. pc.Door.Write(c.Output())
  580. }
  581. case 'R':
  582. pc.Redraw(false)
  583. case 'Q':
  584. // Possibly prompt here for [N]ext hand or [Q]uit
  585. if pc.Current_streak > pc.Best_streak {
  586. pc.Best_streak = pc.Current_streak
  587. pc.Door.Write(pc.StreakPanel.Update())
  588. }
  589. pc.NextQuitPanel.Update()
  590. pc.Door.Write(pc.NextQuitPanel.Output())
  591. if pc.State[26] == 2 {
  592. pc.Door.Write(door.ColorText("BLACK"))
  593. } else {
  594. pc.Door.Write(door.Reset)
  595. }
  596. if pc.Hand < pc.Total_hands {
  597. r = pc.Door.GetOneOf("CNQ")
  598. } else {
  599. r = pc.Door.GetOneOf("CDQ")
  600. }
  601. if r == 'C' {
  602. // Continue
  603. pc.Redraw(false)
  604. } else {
  605. if r == 'D' || r == 'Q' {
  606. if pc.Score >= 50 {
  607. // pc.DB.SaveScore(now, pc.Time_T, .. pc.Score)
  608. _ = pc.Score
  609. }
  610. in_game = false
  611. } else {
  612. if r == 'N' {
  613. // pc.DB.SaveScore(...)
  614. _ = pc.Score
  615. pc.Hand++
  616. goto Next_Hand
  617. }
  618. }
  619. }
  620. }
  621. }
  622. }
  623. }
  624. return 0
  625. }
  626. func (pc *PlayCards) Redraw(Dealing bool) {
  627. // stars.display()
  628. pc.Door.Write(door.Clrscr + door.Reset)
  629. pc.Door.Write(pc.SpaceAceTriPeaks.Output())
  630. var c *door.Panel
  631. {
  632. // Step 1: Draw the deck "source"
  633. pos := CardPos[29]
  634. if pc.Play_card == 51 {
  635. // out of cards
  636. pos.Level = 0
  637. }
  638. c = &pc.DeckPanel.Backs[pos.Level]
  639. c.X = pos.X + pc.Off_X
  640. c.Y = pos.Y + pc.Off_Y
  641. pc.LeftPanel.X = pos.X + pc.Off_X
  642. pc.LeftPanel.Y = pos.Y + pc.Off_Y + 3 // const height see deck
  643. pc.ScorePanel.Update()
  644. pc.LeftPanel.Update()
  645. pc.StreakPanel.Update()
  646. pc.CmdPanel.Update()
  647. pc.Door.Write(pc.ScorePanel.Output())
  648. pc.Door.Write(pc.LeftPanel.Output())
  649. pc.Door.Write(pc.StreakPanel.Output())
  650. pc.Door.Write(pc.CmdPanel.Output())
  651. pc.Door.Write(c.Output())
  652. if Dealing {
  653. time.Sleep(time.Second)
  654. }
  655. }
  656. var x_max int = 29
  657. if Dealing {
  658. x_max = 28
  659. }
  660. for x := 0; x < x_max; x++ {
  661. pos := CardPos[x]
  662. if Dealing {
  663. time.Sleep(time.Duration(75) * time.Millisecond)
  664. }
  665. if Dealing {
  666. c = &pc.DeckPanel.Backs[pos.Level]
  667. c.X = pos.X + pc.Off_X
  668. c.Y = pos.Y + pc.Off_Y
  669. pc.Door.Write(c.Output())
  670. } else {
  671. switch pc.State[x] {
  672. case 0:
  673. c = &pc.DeckPanel.Backs[pos.Level]
  674. c.X = pos.X + pc.Off_X
  675. c.Y = pos.Y + pc.Off_Y
  676. pc.Door.Write(c.Output())
  677. case 1:
  678. if x == 28 {
  679. c = &pc.DeckPanel.Cards[pc.Deck[pc.Play_card]]
  680. } else {
  681. c = &pc.DeckPanel.Cards[pc.Deck[x]]
  682. }
  683. c.X = pos.X + pc.Off_X
  684. c.Y = pos.Y + pc.Off_Y
  685. pc.Door.Write(c.Output())
  686. case 2:
  687. // no card to draw.
  688. case 3:
  689. // peak cleared, draw bonus
  690. var output string = door.Goto(pos.X+pc.Off_X, pos.Y+pc.Off_Y)
  691. output += Bonus()
  692. pc.Door.Write(output)
  693. }
  694. }
  695. }
  696. if Dealing {
  697. for x := 18; x < 29; x++ {
  698. pc.State[x] = 1
  699. // CardPos[x] X Y Level
  700. Pos := &CardPos[x]
  701. time.Sleep(time.Duration(200) * time.Millisecond)
  702. c = &pc.DeckPanel.Cards[pc.Deck[x]]
  703. c.X = Pos.X + pc.Off_X
  704. c.Y = Pos.Y + pc.Off_Y
  705. pc.Door.Write(c.Output())
  706. }
  707. }
  708. {
  709. Pos := &CardPos[pc.Select_card]
  710. c = &pc.DeckPanel.Mark[1]
  711. c.X = Pos.X + pc.Off_X + 2
  712. c.Y = Pos.Y + pc.Off_Y + 2
  713. pc.Door.Write(c.Output())
  714. }
  715. }
  716. func Bonus() string {
  717. return door.ColorText("BOLD YELLOW") + "BONUS"
  718. }