playcards.go 12 KB

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