playcards.go 12 KB

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