playcards.go 18 KB

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