playcards.go 12 KB

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