playcards.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531
  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. )
  12. func StringToANSIColor(colorCode string) string {
  13. colors := []string{
  14. "BLUE",
  15. "BROWN",
  16. "RED",
  17. "CYAN",
  18. "GREEN",
  19. "MAGENTA",
  20. "WHITE",
  21. }
  22. code := strings.ToUpper(colorCode)
  23. for _, c := range colors {
  24. if c == code {
  25. return door.ColorText(code)
  26. }
  27. }
  28. if code == "ALL" {
  29. rand.Seed(time.Now().UnixNano())
  30. pos := rand.Intn(len(colors))
  31. return door.ColorText(colors[pos])
  32. }
  33. return door.ColorText("WHITE")
  34. }
  35. type PlayCards struct {
  36. Door *door.Door
  37. DB *DBData
  38. Config *map[string]string
  39. RNG *rand.Rand
  40. Seeds []int32
  41. Total_hands int
  42. Play_card int
  43. Current_streak int
  44. Best_streak int
  45. Select_card int
  46. Score int
  47. Play_day time.Time
  48. Play_day_t int64
  49. Days_played int
  50. Hand int
  51. Day_status [42]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. Calendar door.Panel
  59. DeckPanel Deck
  60. }
  61. // Adjust date to 2:00 AM
  62. func NormalizeDate(date *time.Time) {
  63. offset := time.Duration(date.Second())*time.Second +
  64. time.Duration(date.Minute())*time.Minute +
  65. time.Duration(date.Hour()-2)*time.Hour +
  66. time.Duration(date.Nanosecond())*time.Nanosecond
  67. *date = date.Add(-offset)
  68. }
  69. func cmdLineRender(bracket string, inner string, outer string) func(string) string {
  70. cmdRender := func(input string) string {
  71. var result string
  72. inOuter := true
  73. var lastColor string
  74. for _, c := range input {
  75. if c == '[' {
  76. inOuter = false
  77. if lastColor != bracket {
  78. result += bracket
  79. lastColor = bracket
  80. }
  81. result += string(c)
  82. continue
  83. }
  84. if c == ']' {
  85. inOuter = true
  86. if lastColor != bracket {
  87. result += bracket
  88. lastColor = bracket
  89. }
  90. result += string(c)
  91. continue
  92. }
  93. if inOuter {
  94. if lastColor != outer {
  95. result += outer
  96. lastColor = outer
  97. }
  98. } else {
  99. if lastColor != inner {
  100. result += inner
  101. lastColor = inner
  102. }
  103. }
  104. result += string(c)
  105. }
  106. return result
  107. }
  108. return cmdRender
  109. }
  110. func (pc *PlayCards) InitValues() {
  111. hands := pc.DB.GetSetting("hands_per_day", "3")
  112. pc.Total_hands, _ = strconv.Atoi(hands)
  113. pc.Play_card = 28
  114. pc.Current_streak = 0
  115. pc.Best_streak = 0
  116. best := pc.DB.GetSetting("best_streak", "0")
  117. pc.Best_streak, _ = strconv.Atoi(best)
  118. pc.Select_card = 23
  119. pc.Score = 0
  120. }
  121. func (pc *PlayCards) Init() {
  122. // init_values()
  123. pc.InitValues()
  124. // PlayCards::PlayCards()
  125. pc.Play_day = time.Now()
  126. NormalizeDate(&pc.Play_day)
  127. pc.Play_day_t = pc.Play_day.Unix()
  128. pc.DeckPanel.Init()
  129. var last_played int64
  130. last := pc.DB.GetSetting("last_played", "0")
  131. last_played, _ = strconv.ParseInt(last, 10, 64)
  132. days := pc.DB.GetSetting("days_played", "0")
  133. pc.Days_played, _ = strconv.Atoi(days)
  134. if last_played != pc.Play_day_t {
  135. pc.DB.SetSetting("last_played", strconv.FormatInt(pc.Play_day_t, 10))
  136. pc.DB.SetSetting("days_played", "0")
  137. pc.Days_played = 0
  138. }
  139. pc.Seeds = make([]int32, 0)
  140. // config _seed
  141. parts := strings.Split((*pc.Config)["_seed"], ",")
  142. for _, seed := range parts {
  143. i, _ := strconv.ParseInt(seed, 10, 32)
  144. pc.Seeds = append(pc.Seeds, int32(i))
  145. }
  146. pc.Hand = 0
  147. pc.Total_hands = 0
  148. // spaceAceTriPeaks = make_tripeaks();
  149. {
  150. tripeakstext := " " + SPACEACE + " - Tri-Peaks Solitaire v" + SPACEACE_VERSION + " "
  151. pc.SpaceAceTriPeaks = door.Panel{Width: len(tripeakstext),
  152. Style: door.SINGLE,
  153. BorderColor: door.ColorText("CYAN ON BLACK"),
  154. }
  155. pc.SpaceAceTriPeaks.Lines = append(pc.SpaceAceTriPeaks.Lines,
  156. door.Line{Text: tripeakstext})
  157. }
  158. svRender := RenderStatusValue(door.ColorText("BOLD WHITE ON BLUE"),
  159. door.ColorText("BOLD YELLOW ON BLUE"))
  160. // score_panel = make_score_panel();
  161. {
  162. W := 25
  163. pc.ScorePanel = door.Panel{Width: W}
  164. text := "Name: " + pc.Door.Config.Real_name
  165. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  166. door.Line{Text: fmt.Sprintf("%*s", -W, text), RenderF: svRender})
  167. scoreUpdate := func() string {
  168. txt := fmt.Sprintf("Score: %d", pc.Score)
  169. txt += strings.Repeat(" ", W-len(txt))
  170. return txt
  171. }
  172. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  173. door.Line{Text: scoreUpdate(),
  174. UpdateF: scoreUpdate,
  175. RenderF: svRender,
  176. })
  177. timeUpdate := func() string {
  178. left := pc.Door.TimeLeft()
  179. used := pc.Door.TimeUsed()
  180. txt := fmt.Sprintf("Time used: %3d / %3d", used.Minutes(), left.Minutes())
  181. txt += strings.Repeat(" ", W-len(txt))
  182. return txt
  183. }
  184. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  185. door.Line{Text: timeUpdate(),
  186. UpdateF: timeUpdate,
  187. RenderF: svRender,
  188. })
  189. handUpdate := func() string {
  190. txt := fmt.Sprintf("Playing Hand %d of %d", pc.Hand, pc.Total_hands)
  191. txt += strings.Repeat(" ", W-len(txt))
  192. return txt
  193. }
  194. pc.ScorePanel.Lines = append(pc.ScorePanel.Lines,
  195. door.Line{Text: handUpdate(),
  196. UpdateF: handUpdate,
  197. RenderF: svRender,
  198. })
  199. }
  200. // streak_panel = make_streak_panel();
  201. {
  202. W := 20
  203. pc.StreakPanel = door.Panel{Width: W}
  204. dateUpdate := func() string {
  205. format, ok := (*pc.Config)["date_format"]
  206. if !ok {
  207. format = "January 2"
  208. }
  209. txt := fmt.Sprintf("Playing: %s", pc.Play_day.Format(format))
  210. txt += strings.Repeat(" ", W-len(txt))
  211. return txt
  212. }
  213. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  214. door.Line{Text: dateUpdate(),
  215. UpdateF: dateUpdate,
  216. RenderF: svRender,
  217. })
  218. currentUpdate := func() string {
  219. txt := fmt.Sprintf("Current Streak: %d", pc.Current_streak)
  220. txt += strings.Repeat(" ", W-len(txt))
  221. return txt
  222. }
  223. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  224. door.Line{Text: currentUpdate(),
  225. UpdateF: currentUpdate,
  226. RenderF: svRender,
  227. })
  228. longestUpdate := func() string {
  229. txt := fmt.Sprintf("Longest Streak: %d", pc.Best_streak)
  230. txt += strings.Repeat(" ", W-len(txt))
  231. return txt
  232. }
  233. pc.StreakPanel.Lines = append(pc.StreakPanel.Lines,
  234. door.Line{Text: longestUpdate(),
  235. UpdateF: longestUpdate,
  236. RenderF: svRender,
  237. })
  238. }
  239. // left_panel = make_left_panel();
  240. {
  241. W := 13
  242. pc.LeftPanel = door.Panel{Width: W}
  243. leftUpdate := func() string {
  244. txt := fmt.Sprintf("Cards left:%d", 51-pc.Play_card)
  245. txt += strings.Repeat(" ", W-len(txt))
  246. return txt
  247. }
  248. pc.LeftPanel.Lines = append(pc.LeftPanel.Lines,
  249. door.Line{Text: leftUpdate(),
  250. UpdateF: leftUpdate,
  251. RenderF: svRender,
  252. })
  253. }
  254. // cmd_panel = make_command_panel();
  255. {
  256. W := 76
  257. pc.CmdPanel = door.Panel{Width: W}
  258. var commands string
  259. if door.Unicode {
  260. commands = "[4/\u25c4] Left [6/\u25ba] Right [Space] Play Card [Enter] Draw [Q]uit [R]edraw [H]elp"
  261. commands += strings.Repeat(" ", W-len([]rune(commands)))
  262. } else {
  263. commands = "[4/\x11] Left [6/\x10] Right [Space] Play Card [Enter] Draw [Q]uit [R]edraw [H]elp"
  264. commands += strings.Repeat(" ", W-len(commands))
  265. }
  266. cmdRender := cmdLineRender(door.ColorText("BOLD YELLOW ON BLUE"),
  267. door.ColorText("BOLD CYAN ON BLUE"),
  268. door.ColorText("BOLD GREEN ON BLUE"))
  269. pc.CmdPanel.Lines = append(pc.CmdPanel.Lines,
  270. door.Line{Text: commands,
  271. RenderF: cmdRender})
  272. }
  273. // next_quit_panel = make_next_panel();
  274. {
  275. W := 50
  276. pc.NextQuitPanel = door.Panel{Width: W,
  277. Style: door.DOUBLE,
  278. BorderColor: door.ColorText("BOLD YELLOW ON BLUE"),
  279. }
  280. nextUpdate := func() string {
  281. var text string
  282. if pc.Select_card != -1 {
  283. text = "[C]ontinue this hand"
  284. }
  285. if pc.Hand < pc.Total_hands {
  286. text += " [N]ext Hand [Q]uit"
  287. } else {
  288. text += " [D]one [Q]uit"
  289. }
  290. text = " " + text + " "
  291. text += strings.Repeat(" ", W-len(text))
  292. return text
  293. }
  294. pc.NextQuitPanel.Lines = append(pc.NextQuitPanel.Lines,
  295. door.Line{Text: nextUpdate(),
  296. UpdateF: nextUpdate,
  297. RenderF: cmdLineRender(door.ColorText("BOLD YEL ON BLU"),
  298. door.ColorText("BOLD CYAN ON BLU"),
  299. door.ColorText("BOLD GREE ON BLUE"))})
  300. }
  301. // calendar = make_calendar();
  302. {
  303. W := 41
  304. pc.Calendar = door.Panel{Width: W,
  305. Style: door.DOUBLE,
  306. BorderColor: door.ColorText("CYAN ON BLACK")}
  307. calendarRender := func(text string) string {
  308. // var result string
  309. // var lastColor string
  310. result := door.Render{Line: text}
  311. digits := door.ColorText("CYAN ON BLACK")
  312. digits_play := door.ColorText("BOLD CYAN ON BLACK")
  313. spaces := door.ColorText("WHITE ON BLACK")
  314. open := door.ColorText("BOLD GREEN ON BLACK")
  315. hands := door.ColorText("BOLD YELLOW ON BLACK")
  316. full := door.ColorText("RED ON BLACK")
  317. nny := door.ColorText("BOLD BLACK ON BLACK")
  318. for days := 0; days < 7; days++ {
  319. var dayText string
  320. if days == 0 {
  321. result.Append(spaces, 1)
  322. }
  323. dayText = text[1+days*6 : (1+days*6)+3]
  324. if dayText[1] == ' ' {
  325. result.Append(spaces, 3)
  326. } else {
  327. // Something is here
  328. cday := dayText[2]
  329. if cday == 'o' || cday == 'h' {
  330. result.Append(digits_play, 2)
  331. } else {
  332. result.Append(digits, 2)
  333. }
  334. switch cday {
  335. case 'o':
  336. result.Append(open, 1)
  337. case 'h':
  338. result.Append(hands, 1)
  339. case 'x':
  340. result.Append(full, 1)
  341. case 'u':
  342. result.Append(nny, 1)
  343. }
  344. }
  345. if days == 6 {
  346. result.Append(spaces, 1)
  347. } else {
  348. result.Append(spaces, 3)
  349. }
  350. }
  351. return result.Result
  352. }
  353. for row := 0; row < 6; row++ {
  354. calendarUpdate := func() string {
  355. var text string
  356. for d := 0; d < 7; d++ {
  357. text += " "
  358. v := pc.Day_status[(row*7)+d]
  359. if v == 0 {
  360. text += " "
  361. } else {
  362. text += fmt.Sprintf("%-2d")
  363. status := pc.Day_status[v-1]
  364. switch status {
  365. case 0:
  366. text += "o"
  367. case 1:
  368. text += "h"
  369. case 2:
  370. text += "x"
  371. case 3:
  372. text += "u"
  373. }
  374. }
  375. if d == 6 {
  376. text += " "
  377. } else {
  378. text += " "
  379. }
  380. }
  381. return text
  382. }
  383. pc.Calendar.Lines = append(pc.Calendar.Lines,
  384. door.Line{Text: calendarUpdate(),
  385. UpdateF: calendarUpdate,
  386. RenderF: calendarRender})
  387. }
  388. }
  389. }
  390. func (pc *PlayCards) Play() {
  391. // this defaults (for now) to playing today (if unplayed).
  392. // Otherwise display calendar.
  393. pc.Play_day = time.Now()
  394. NormalizeDate(&pc.Play_day)
  395. pc.Play_day_t = pc.Play_day.Unix()
  396. // played := db.HandsPlayedOnDay(pc.Play_day_t)
  397. played := 0
  398. var r int
  399. if played == 0 {
  400. pc.Door.Write("Let's play today..." + door.CRNL)
  401. time.Sleep(time.Second)
  402. pc.Hand = 1
  403. r = pc.PlayCards()
  404. if r != 'D' {
  405. return
  406. }
  407. } else {
  408. if played < pc.Total_hands {
  409. // let's finish today...
  410. pc.Hand = played + 1
  411. r = pc.PlayCards()
  412. if r != 'D' {
  413. return
  414. }
  415. }
  416. }
  417. /*
  418. CALENDAR_UPDATE:
  419. pc.UpdateCalendarDays()
  420. pc.Calendar.Update()
  421. pc.Door.Write(pc.Calendar.Output())
  422. */
  423. }
  424. func int32toByteArray(i int32) (arr [4]byte) {
  425. binary.BigEndian.PutUint32(arr[0:4], uint32(i))
  426. return
  427. }
  428. func int64toByteArray(i int64) (arr [4]byte) {
  429. binary.BigEndian.PutUint64(arr[0:4], uint64(i))
  430. return
  431. }
  432. func (pc *PlayCards) PlayCards() int {
  433. pc.InitValues()
  434. var game_width int
  435. var game_height int = 20
  436. pos := CardPos[27]
  437. game_width = pos.X + 5
  438. MX := door.Width
  439. MY := door.Height
  440. off_x := (MX - game_width) / 2
  441. off_y := (MY - game_height) / 2
  442. // We can now position things properly centered
  443. pc.SpaceAceTriPeaks.X = (MX - pc.SpaceAceTriPeaks.Width) / 2
  444. pc.SpaceAceTriPeaks.Y = off_y
  445. off_y += 3
  446. currentDefault := pc.DB.GetSetting("DeckColor", "ALL")
  447. deck_color := StringToANSIColor(currentDefault)
  448. pc.DeckPanel.SetBackColor(deck_color)
  449. pc.Play_card = 28
  450. pc.Select_card = 23
  451. pc.Score = 0
  452. pc.Current_streak = 0
  453. // Use play day to seed RNG
  454. {
  455. // Secret Squirrel method of seeding the RNG
  456. sha1 := sha1.New()
  457. seeds := make([]byte, 0)
  458. for _, seed := range pc.Seeds {
  459. ba := int32toByteArray(seed)
  460. seeds = append(seeds, ba[:]...)
  461. // sha1.Sum(ba[:])
  462. }
  463. pd := int64toByteArray(pc.Play_day_t)
  464. seeds = append(seeds, pd[:]...)
  465. result := sha1.Sum(seeds)
  466. var seed int64
  467. seed = int64(binary.BigEndian.Uint64(result[0:8]))
  468. pc.RNG.Seed(seed)
  469. }
  470. return 0
  471. }