deck.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  1. package main
  2. import (
  3. "bytes"
  4. "math/rand"
  5. "red-green/door"
  6. "strings"
  7. )
  8. type Deck struct {
  9. Cards []door.Panel
  10. Backs []door.Panel
  11. Mark []door.Panel
  12. }
  13. func (d *Deck) SetBackColor(color string) {
  14. for x := 0; x < 5; x++ {
  15. for idx := range d.Backs[x].Lines {
  16. d.Backs[x].Lines[idx].DefaultColor = color
  17. }
  18. }
  19. }
  20. func (d *Deck) Init() {
  21. d.Cards = make([]door.Panel, 52)
  22. for x := 0; x < 52; x++ {
  23. d.Cards[x] = CardOf(x)
  24. }
  25. d.Backs = make([]door.Panel, 5)
  26. for x := 0; x < 5; x++ {
  27. d.Backs[x] = BackOf(x)
  28. }
  29. d.Mark = make([]door.Panel, 2)
  30. d.Mark[0] = MarkOf(0)
  31. d.Mark[1] = MarkOf(1)
  32. }
  33. func CardOf(c int) door.Panel {
  34. suit := GetSuit(c)
  35. rank := GetRank(c)
  36. var is_red bool = suit < 2
  37. var color string
  38. if is_red {
  39. color = door.ColorText("RED ON WHITE")
  40. } else {
  41. color = door.ColorText("BLACK ON WHITE")
  42. }
  43. p := door.Panel{
  44. X: 0,
  45. Y: 0,
  46. Width: 5}
  47. r := RankSymbol(rank)
  48. s := SuitSymbol(suit)
  49. p.Lines = append(p.Lines,
  50. door.Line{Text: string(r) + string(s) + " ",
  51. DefaultColor: color})
  52. p.Lines = append(p.Lines,
  53. door.Line{Text: " " + string(s) + " ",
  54. DefaultColor: color})
  55. p.Lines = append(p.Lines,
  56. door.Line{Text: " " + string(s) + string(r),
  57. DefaultColor: color})
  58. return p
  59. }
  60. func BackOf(level int) door.Panel {
  61. p := door.Panel{
  62. X: 0,
  63. Y: 0,
  64. Width: 5,
  65. }
  66. var back string
  67. if door.Unicode {
  68. back = strings.Repeat(string(BackSymbol(level)), 5)
  69. } else {
  70. // bytes get destroyed as string.
  71. br := make([]byte, 1)
  72. br[0] = byte(BackSymbol(level))
  73. back = string(bytes.Repeat(br, 5))
  74. }
  75. for x := 0; x < 3; x++ {
  76. p.Lines = append(p.Lines,
  77. door.Line{Text: back})
  78. }
  79. return p
  80. }
  81. func MarkOf(c int) door.Panel {
  82. p := door.Panel{Width: 1}
  83. color := door.ColorText("BLUE ON WHITE")
  84. // var m rune
  85. if c == 0 {
  86. p.Lines = append(p.Lines,
  87. door.Line{Text: " ",
  88. DefaultColor: color,
  89. })
  90. // m = ' '
  91. } else {
  92. if door.Unicode {
  93. p.Lines = append(p.Lines,
  94. door.Line{Text: "\u25a0",
  95. DefaultColor: color,
  96. })
  97. // m = '\u25a0'
  98. } else {
  99. /*
  100. var b [1]byte
  101. b[0] = 0xfe
  102. m = string(b[:]) // '\xfe'
  103. */
  104. var b [1]byte
  105. b[0] = 0xfe
  106. p.Lines = append(p.Lines,
  107. door.Line{Text: string(b[:]),
  108. DefaultColor: color})
  109. }
  110. }
  111. return p
  112. }
  113. func RankSymbol(c int) rune {
  114. const symbols = "A23456789TJQK"
  115. return rune(symbols[c])
  116. }
  117. func SuitSymbol(c int) rune {
  118. if door.Unicode {
  119. switch c {
  120. case 0:
  121. return '\u2665'
  122. case 1:
  123. return '\u2666'
  124. case 2:
  125. return '\u2663'
  126. case 3:
  127. return '\u2660'
  128. }
  129. } else {
  130. if door.Full_CP437 {
  131. switch c {
  132. case 0:
  133. return '\x03'
  134. case 1:
  135. return '\x04'
  136. case 2:
  137. return '\x05'
  138. case 3:
  139. return '\x06'
  140. }
  141. } else {
  142. switch c {
  143. case 0:
  144. return '*'
  145. case 1:
  146. return '^'
  147. case 2:
  148. return '%'
  149. case 3:
  150. return '$'
  151. }
  152. }
  153. }
  154. return '?'
  155. }
  156. func BackSymbol(level int) rune {
  157. if level == 0 {
  158. return ' '
  159. }
  160. if door.Unicode {
  161. switch level {
  162. case 1:
  163. return '\u2591'
  164. case 2:
  165. return '\u2592'
  166. case 3:
  167. return '\u2593'
  168. case 4:
  169. return '\u2588'
  170. }
  171. } else {
  172. switch level {
  173. case 1:
  174. return '\xb0'
  175. case 2:
  176. return '\xb1'
  177. case 3:
  178. return '\xb2'
  179. case 4:
  180. return '\xdb'
  181. }
  182. }
  183. return '?'
  184. }
  185. func GetRank(c int) int {
  186. return (c % 52) % 13
  187. }
  188. func GetSuit(c int) int {
  189. return (c % 52) / 13
  190. }
  191. func GetDeck(c int) int {
  192. return c / 52
  193. }
  194. type Pos struct {
  195. X int
  196. Y int
  197. Level int
  198. }
  199. func CalcCardPos(pos int) Pos {
  200. var result Pos
  201. const space = 3
  202. const height = 3
  203. if pos == 28 {
  204. result = CalcCardPos(23)
  205. result.Y += height + 1
  206. result.Level--
  207. return result
  208. } else {
  209. if pos == 29 {
  210. result = CalcCardPos(22)
  211. result.Y += height + 1
  212. result.Level--
  213. return result
  214. }
  215. }
  216. const CARD_WIDTH = 5
  217. var HALF_WIDTH int = 3
  218. HALF_WIDTH += space / 2
  219. const between = CARD_WIDTH + space
  220. if pos < 3 {
  221. result.Level = 1
  222. result.Y = (result.Level-1)*(height-1) + 1
  223. result.X = pos*(between*3) + between + HALF_WIDTH + space
  224. return result
  225. } else {
  226. pos -= 3
  227. }
  228. if pos < 6 {
  229. result.Level = 2
  230. result.Y = (result.Level-1)*(height-1) + 1
  231. group := pos / 2
  232. result.X = pos*between + (group * between) + CARD_WIDTH + space*2
  233. return result
  234. } else {
  235. pos -= 6
  236. }
  237. if pos < 9 {
  238. result.Level = 3
  239. result.Y = (result.Level-1)*(height-1) + 1
  240. result.X = pos*between + HALF_WIDTH + space
  241. return result
  242. } else {
  243. pos -= 9
  244. }
  245. if pos < 10 {
  246. result.Level = 4
  247. result.Y = (result.Level-1)*(height-1) + 1
  248. result.X = pos*between + space
  249. return result
  250. }
  251. // failure
  252. result.Level = -1
  253. result.X = -1
  254. result.Y = -1
  255. return result
  256. }
  257. // 0-29
  258. var CardPos []Pos
  259. var blocks [][]int
  260. func init() {
  261. CardPos = make([]Pos, 30)
  262. for x := 0; x < 30; x++ {
  263. CardPos[x] = CalcCardPos(x)
  264. }
  265. blocks = [][]int{
  266. {3, 4},
  267. {5, 6},
  268. {7, 8}, // end row 1
  269. {9, 10},
  270. {10, 11},
  271. {12, 13},
  272. {13, 14},
  273. {15, 16},
  274. {16, 17},
  275. {18, 19}, // end row 2
  276. {19, 20},
  277. {20, 21},
  278. {21, 22},
  279. {22, 23},
  280. {23, 24},
  281. {24, 25},
  282. {25, 26},
  283. {26, 27},
  284. }
  285. }
  286. // Which card(s) are unblocked by this card?
  287. func Unblocks(card int) []int {
  288. var result []int
  289. for idx := range blocks {
  290. if blocks[idx][0] == card || blocks[idx][1] == card {
  291. result = append(result, idx)
  292. }
  293. }
  294. return result
  295. }
  296. func CanPlay(card1 int, card2 int) bool {
  297. rank1 := GetRank(card1)
  298. rank2 := GetRank(card2)
  299. if (rank1+1)%13 == rank2 {
  300. return true
  301. }
  302. if rank1 == 0 {
  303. rank1 += 13
  304. }
  305. if rank1-1 == rank2 {
  306. return true
  307. }
  308. return false
  309. }
  310. type DeckType int8
  311. /*
  312. This shuffles the deck(s) of cards.
  313. The RNG must be seeded before calling.
  314. */
  315. func ShuffleCards(RNG *rand.Rand, decks int) []DeckType {
  316. var size int = decks * 52
  317. var result []DeckType = make([]DeckType, size)
  318. for x := 0; x < size; x++ {
  319. result[x] = DeckType(x)
  320. }
  321. RNG.Shuffle(size, func(i, j int) {
  322. result[i], result[j] = result[j], result[i]
  323. })
  324. return result
  325. }
  326. func MakeCardStates(decks int) []DeckType {
  327. var size int = decks * 52
  328. var result []DeckType = make([]DeckType, size)
  329. // defaults to 0
  330. return result
  331. }