space-ace.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570
  1. package main
  2. import (
  3. "fmt"
  4. "math/rand"
  5. "os"
  6. "path/filepath"
  7. "red-green/door"
  8. "regexp"
  9. "strconv"
  10. "strings"
  11. "time"
  12. "unicode"
  13. _ "github.com/mattn/go-sqlite3"
  14. "github.com/seehuhn/mt19937"
  15. )
  16. var Config map[string]string
  17. func find_words(text string) [][]int {
  18. word, _ := regexp.Compile("([A-Za-z]+)")
  19. return word.FindAllStringIndex(text, -1)
  20. }
  21. func press_a_key(d *door.Door) int {
  22. green := door.ColorText("BOLD GREEN")
  23. blue := door.ColorText("BOLD BLUE")
  24. yellow := door.ColorText("BOLD YELLOW")
  25. any_caps := green
  26. any_lower := yellow
  27. any_color := func(text string) string {
  28. var output string
  29. var lastColor string
  30. for _, letter := range text {
  31. if unicode.IsUpper(letter) {
  32. if lastColor != any_caps {
  33. lastColor = any_caps
  34. output += any_caps
  35. }
  36. output += string(letter)
  37. } else {
  38. if lastColor != any_lower {
  39. lastColor = any_lower
  40. output += any_lower
  41. }
  42. output += string(letter)
  43. }
  44. }
  45. return output
  46. }
  47. text := "Press a key to continue..."
  48. work_text := []rune(text)
  49. d.Write(door.Reset + any_color(text))
  50. var r int = -1
  51. var t int
  52. sleep_ms := 250
  53. ms_sleep := 0
  54. words := find_words(text)
  55. var word int = 0
  56. var wpos int = 0
  57. current_word := text[words[word][0]:words[word][1]]
  58. t = 0
  59. r = d.WaitKey(0, int64(sleep_ms)*1000)
  60. for r == -1 {
  61. ms_sleep += sleep_ms
  62. if ms_sleep > 1000 {
  63. ms_sleep -= 1000
  64. t++
  65. if t >= int(door.Inactivity) {
  66. d.Write(strings.Repeat("\b", len(text)))
  67. d.Write(any_color(text) + door.CRNL)
  68. return -1
  69. }
  70. }
  71. wpos++
  72. if wpos == len(current_word) {
  73. word++
  74. wpos = 0
  75. work_text = []rune(text)
  76. if word == len(words) {
  77. word = 0
  78. if any_caps == green {
  79. any_caps = blue
  80. } else {
  81. any_caps = green
  82. }
  83. d.Write(strings.Repeat("\b", len(text)))
  84. d.Write(any_color(string(work_text)))
  85. current_word = text[words[word][0]:words[word][1]]
  86. goto READKEY
  87. }
  88. current_word = text[words[word][0]:words[word][1]]
  89. }
  90. {
  91. c := rune(current_word[wpos])
  92. for !unicode.IsLower(c) {
  93. wpos++
  94. if wpos == len(current_word) {
  95. wpos = 0
  96. word++
  97. work_text = []rune(text)
  98. if word == len(words) {
  99. word = 0
  100. }
  101. current_word = text[words[word][0]:words[word][1]]
  102. }
  103. c = rune(current_word[wpos])
  104. }
  105. // Ok, we found something that's lower case
  106. work_text[words[word][0]+wpos] = unicode.ToUpper(c)
  107. }
  108. d.Write(strings.Repeat("\b", len(text)))
  109. d.Write(any_color(string(work_text)))
  110. READKEY:
  111. r = d.WaitKey(0, int64(sleep_ms)*1000)
  112. }
  113. d.Write(strings.Repeat("\b", len(text)))
  114. d.Write(any_color(text))
  115. return r
  116. }
  117. func MainMenu() door.Menu {
  118. // Make the main menu
  119. m := door.Menu{Panel: door.Panel{Width: 25,
  120. X: 5,
  121. Y: 5,
  122. Style: door.DOUBLE,
  123. Title: "[ Main Menu: ]",
  124. TitleOffset: 3,
  125. BorderColor: door.ColorText("BRI CYAN ON BLA")}}
  126. m.SelectedR = door.MakeMenuRender(door.ColorText("BOLD CYAN"),
  127. door.ColorText("BOLD BLUE"),
  128. door.ColorText("BOLD CYAN"),
  129. door.ColorText("BOLD BLUE"))
  130. m.UnselectedR = door.MakeMenuRender(door.ColorText("BOLD YEL ON BLUE"),
  131. door.ColorText("BOLD WHI ON BLUE"),
  132. door.ColorText("BOLD YEL ON BLUE"),
  133. door.ColorText("BOLD CYAN ON BLUE"))
  134. m.AddSelection("P", "Play Cards")
  135. m.AddSelection("S", "View Scores")
  136. m.AddSelection("C", "Configure")
  137. m.AddSelection("H", "Help")
  138. m.AddSelection("A", "About this game")
  139. m.AddSelection("Q", "Quit")
  140. return m
  141. }
  142. func ConfigMenu() door.Menu {
  143. m := door.Menu{Panel: door.Panel{Width: 31,
  144. X: 5,
  145. Y: 5,
  146. Style: door.DOUBLE,
  147. Title: "[ Configuration Menu ]",
  148. TitleColor: door.ColorText("BRI CYAN ON BLUE"),
  149. BorderColor: door.ColorText("CYAN ON BLUE")}}
  150. m.SelectedR = door.MakeMenuRender(door.ColorText("BOLD CYAN"),
  151. door.ColorText("BOLD BLUE"),
  152. door.ColorText("BOLD CYAN"),
  153. door.ColorText("BOLD BLUE"))
  154. m.UnselectedR = door.MakeMenuRender(door.ColorText("BOLD YEL ON BLUE"),
  155. door.ColorText("BOLD WHI ON BLUE"),
  156. door.ColorText("BOLD YEL ON BLUE"),
  157. door.ColorText("BOLD CYAN ON BLUE"))
  158. m.AddSelection("D", "Deck Colors")
  159. m.AddSelection("V", "View Settings")
  160. m.AddSelection("Q", "Quit")
  161. return m
  162. }
  163. func display_information(d *door.Door) {
  164. d.Write(door.Clrscr)
  165. keyColor := door.ColorText("BRI GREEN")
  166. sepColor := door.ColorText("BRI YEL")
  167. valColor := door.ColorText("CYAN")
  168. nice_format := func(key string, value string) string {
  169. return fmt.Sprintf("%s%-20s %s: %s%s", keyColor, key, sepColor, valColor, value) + door.CRNL
  170. }
  171. d.Write(nice_format("BBS Software", d.Config.BBSID))
  172. d.Write(nice_format("Real Name", d.Config.Real_name))
  173. d.Write(nice_format("Handle", d.Config.Handle))
  174. d.Write(nice_format("User #", strconv.Itoa(d.Config.User_number)))
  175. d.Write(nice_format("Security Level", strconv.Itoa(d.Config.Security_level)))
  176. d.Write(nice_format("Node #", strconv.Itoa(d.Config.Node)))
  177. d.Write(nice_format("Unicode", strconv.FormatBool(door.Unicode)))
  178. d.Write(nice_format("CP437", strconv.FormatBool(door.CP437)))
  179. d.Write(nice_format("Screen Size", fmt.Sprintf("%d X %d", door.Width, door.Height)))
  180. }
  181. func panel_demo(d *door.Door) {
  182. width := 55
  183. fmtStr := "%-55s"
  184. p := door.Panel{X: 5, Y: 5, Width: width, Style: door.DOUBLE, BorderColor: door.ColorText("CYAN ON BLUE"), Title: "[ Panel Demo ]"}
  185. lineColor := door.ColorText("BRIGHT WHI ON BLUE")
  186. // Add lines to the panel
  187. for _, line := range []string{"The BBS Door Panel Demo", "(C) 2021 Red Green Software, https://red-green.com"} {
  188. if door.Unicode {
  189. line = strings.Replace(line, "(C)", "\u00a9", -1)
  190. }
  191. l := door.Line{Text: fmt.Sprintf(fmtStr, line), DefaultColor: lineColor}
  192. p.Lines = append(p.Lines, l)
  193. }
  194. p.Lines = append(p.Lines, p.Spacer())
  195. p.Lines = append(p.Lines, door.Line{Text: fmt.Sprintf(fmtStr, "Welcome to golang!"), DefaultColor: lineColor})
  196. d.Write(door.Clrscr)
  197. d.Write(p.Output() + door.CRNL)
  198. }
  199. func MakeColorsRender(brackets string, text_color string, colors map[string]string) func(string) string {
  200. renderF := func(text string) string {
  201. words := find_words(text)
  202. var option bool = true
  203. var color_word bool = false
  204. var result string
  205. var last_color string
  206. var word_color string
  207. var words_id = 0
  208. word_pair := words[words_id]
  209. normal := colors["ALL"]
  210. for tpos, c := range text {
  211. if option {
  212. if c == '[' || c == ']' {
  213. if last_color != brackets {
  214. result += brackets
  215. last_color = brackets
  216. }
  217. if c == ']' {
  218. option = false
  219. }
  220. } else {
  221. if last_color != text_color {
  222. result += text_color
  223. last_color = text_color
  224. }
  225. }
  226. result += string(c)
  227. } else {
  228. // Out of the options
  229. if color_word {
  230. if tpos < word_pair[1] {
  231. if last_color != word_color {
  232. result += word_color
  233. last_color = word_color
  234. } else {
  235. color_word = false
  236. if last_color != normal {
  237. result += normal
  238. last_color = normal
  239. }
  240. }
  241. result += string(c)
  242. }
  243. } else {
  244. // look for COLOR word
  245. if tpos >= word_pair[1] {
  246. words_id++
  247. if words_id >= len(words) {
  248. word_pair = []int{999, 999}
  249. } else {
  250. word_pair = words[words_id]
  251. }
  252. }
  253. if word_pair[0] == tpos {
  254. // start of word
  255. possible := text[word_pair[0]:word_pair[1]]
  256. color_code, has := colors[strings.ToUpper(possible)]
  257. // log.Printf("Match %s / found %s %#v\n", possible, color_code, has)
  258. if has {
  259. word_color = color_code
  260. } else {
  261. word_color = colors["ALL"]
  262. }
  263. if last_color != word_color {
  264. result += word_color
  265. last_color = word_color
  266. }
  267. }
  268. result += string(c)
  269. }
  270. }
  271. }
  272. return result
  273. }
  274. return renderF
  275. }
  276. var DeckColors []string
  277. func init() {
  278. DeckColors = []string{
  279. "All", "Brown", "Green", "Red", "Blue", "Cyan", "Magenta", "White"}
  280. }
  281. func DeckColorMenu() door.Menu {
  282. m := door.Menu{Panel: door.Panel{Width: 31,
  283. X: 5,
  284. Y: 5,
  285. Style: door.DOUBLE,
  286. Title: "[ Deck Menu ]",
  287. TitleColor: door.ColorText("BRI CYAN ON BLUE"),
  288. BorderColor: door.ColorText("CYAN ON BLUE")}}
  289. unselectMap := map[string]string{"BLUE": door.ColorText("BRI BLUE ON BLUE"),
  290. "BROWN": door.ColorText("BROWN ON BLUE"),
  291. "RED": door.ColorText("BRI RED ON BLUE"),
  292. "CYAN": door.ColorText("CYAN ON BLUE"),
  293. "GREEN": door.ColorText("BRI GREEN ON BLUE"),
  294. "MAGENTA": door.ColorText("BRI MAGENTA ON BLUE"),
  295. "WHITE": door.ColorText("BRI WHITE ON BLUE"),
  296. "ALL": door.ColorText("WHITE ON BLUE")}
  297. selectMap := map[string]string{"BLUE": door.ColorText("BRI BLUE ON BLACK"),
  298. "BROWN": door.ColorText("BROWN ON BLACK"),
  299. "RED": door.ColorText("BRI RED ON BLACK"),
  300. "CYAN": door.ColorText("CYAN ON BLACK"),
  301. "GREEN": door.ColorText("BRI GREEN ON BLACK"),
  302. "MAGENTA": door.ColorText("BRI MAGENTA ON BLACK"),
  303. "WHITE": door.ColorText("BRI WHITE ON BLACK"),
  304. "ALL": door.ColorText("WHITE ON BLACK")}
  305. m.SelectedR = MakeColorsRender(door.ColorText("BOLD CYAN"),
  306. door.ColorText("BOLD BLUE"),
  307. selectMap)
  308. m.UnselectedR = MakeColorsRender(door.ColorText("BOLD YEL ON BLUE"),
  309. door.ColorText("BOLD WHI ON BLUE"),
  310. unselectMap)
  311. for _, color := range DeckColors {
  312. m.AddSelection(color[:1], color)
  313. }
  314. return m
  315. }
  316. func RenderStatusValue(status string, value string) func(string) string {
  317. renderF := func(text string) string {
  318. var result string
  319. pos := strings.Index(text, ":")
  320. result = status + text[:pos] + value + text[pos:]
  321. return result
  322. }
  323. return renderF
  324. }
  325. // Display the SysOp settings in the Config/yaml file
  326. func DisplaySettings(d *door.Door) {
  327. d.Write(door.Clrscr + door.Reset)
  328. W := 35
  329. panel := door.Panel{Width: W,
  330. X: 5,
  331. Y: 5,
  332. Style: door.DOUBLE,
  333. BorderColor: door.ColorText("BOLD CYAN ON BLUE"),
  334. }
  335. l := door.Line{Text: fmt.Sprintf("%*s", W, "Game Settings - SysOp Configurable"),
  336. DefaultColor: door.ColorText("BOLD GREEN ON BLUE")}
  337. panel.Lines = append(panel.Lines, l)
  338. renderF := RenderStatusValue(door.ColorText("BOLD YELLOW ON BLUE"), door.ColorText("BOLD GREEN ON BLUE"))
  339. for key, value := range Config {
  340. if key[0] == '_' {
  341. continue
  342. }
  343. l = door.Line{Text: fmt.Sprintf("%20s : %*s", key, -(W - 23), value), RenderF: renderF}
  344. panel.Lines = append(panel.Lines, l)
  345. }
  346. d.Write(panel.Output() + door.Reset + door.CRNL)
  347. press_a_key(d)
  348. }
  349. func Configure(d *door.Door, db *DBData) {
  350. menu := ConfigMenu()
  351. // deckcolor := "DeckColor"
  352. // save_deckcolor := false
  353. var choice int
  354. for choice >= 0 {
  355. d.Write(door.Clrscr)
  356. choice = menu.Choose(d)
  357. if choice < 0 {
  358. break
  359. }
  360. option := menu.GetOption(choice)
  361. switch option {
  362. case 'D':
  363. // Deck color
  364. colormenu := DeckColorMenu()
  365. d.Write(door.Clrscr)
  366. _ = colormenu.Choose(d)
  367. press_a_key(d)
  368. case 'V':
  369. // View settings
  370. DisplaySettings(d)
  371. case 'Q':
  372. choice = -1
  373. }
  374. }
  375. }
  376. func main() {
  377. var message string
  378. // Get path to binary, and chdir to it.
  379. binaryPath, _ := os.Executable()
  380. binaryPath = filepath.Dir(binaryPath)
  381. _ = os.Chdir(binaryPath)
  382. fmt.Println("Starting space-ace")
  383. rng := rand.New(mt19937.New())
  384. rng.Seed(time.Now().UnixNano())
  385. d := door.Door{}
  386. d.Init("space-ace")
  387. db := DBData{}
  388. db.Open("space-database.db")
  389. defer db.Close()
  390. // Use Real_name or Handle? Or read config?
  391. db.User = d.Config.Real_name
  392. const config_filename = "space-ace.yaml"
  393. if FileExists(config_filename) {
  394. Config = LoadConfig(config_filename)
  395. } else {
  396. Config = make(map[string]string)
  397. }
  398. var update_config bool = false
  399. // Guessing at this point as to what settings I actually want.
  400. config_defaults := map[string]string{"hands_per_day": "3",
  401. "date_format": "%B %d",
  402. "date_score": "%m/%d/%Y",
  403. "makeup_per_day": "5",
  404. "play_days_ahead": "2",
  405. "date_monthly": "%B %Y"}
  406. // _seed
  407. _, ok := Config["_seed"]
  408. if !ok {
  409. // Initialize the seed
  410. Config["_seed"] = fmt.Sprintf("%d,%d,%d", rng.Int31(), rng.Int31(), rng.Int31())
  411. update_config = true
  412. }
  413. for key, value := range config_defaults {
  414. if SetConfigDefault(&Config, key, value) {
  415. update_config = true
  416. }
  417. }
  418. if update_config {
  419. SaveConfig(config_filename, Config)
  420. }
  421. s := StarField{}
  422. s.Regenerate()
  423. s.Display(&d)
  424. d.Write(door.Goto(1, 1) + door.Reset)
  425. // Get the last call value (if they have called before)
  426. last_call, _ := strconv.ParseInt(db.GetSetting("LastCall", "0"), 10, 64)
  427. now := time.Now()
  428. db.SetSetting("LastCall", fmt.Sprintf("%d", now.Unix()))
  429. // Check for maint run -- get FirstOfMonthDate, and see if
  430. // records are older then it is. (If so, yes -- run maint)!
  431. if last_call != 0 {
  432. d.Write("Welcome Back!" + door.CRNL)
  433. delta := now.Sub(time.Unix(last_call, 0))
  434. hours := delta.Hours()
  435. if hours > 24 {
  436. d.Write(fmt.Sprintf("It's been %0.1f days since you last played."+door.CRNL, hours/24))
  437. } else {
  438. if hours > 1 {
  439. d.Write(fmt.Sprintf("It's been %0.1f hours since you last played."+door.CRNL, hours))
  440. } else {
  441. minutes := delta.Minutes()
  442. d.Write(fmt.Sprintf("It's been %0.1f minutes since you last played."+door.CRNL, minutes))
  443. }
  444. }
  445. }
  446. left := d.TimeLeft()
  447. message = fmt.Sprintf("You have %0.2f minutes / %0.2f seconds remaining..."+door.CRNL, left.Minutes(), left.Seconds())
  448. d.Write(message)
  449. press_a_key(&d)
  450. mainmenu := MainMenu()
  451. var choice int
  452. for choice >= 0 {
  453. d.Write(door.Clrscr)
  454. s.Display(&d)
  455. choice = mainmenu.Choose(&d)
  456. if choice < 0 {
  457. break
  458. }
  459. option := mainmenu.GetOption(choice)
  460. // fmt.Printf("Choice: %d, Option: %c\n", choice, option)
  461. switch option {
  462. case 'P':
  463. // Play cards
  464. case 'S':
  465. // View Scores
  466. case 'C':
  467. // Configure
  468. Configure(&d, &db)
  469. case 'H':
  470. // Help
  471. press_a_key(&d)
  472. case 'A':
  473. // About
  474. press_a_key(&d)
  475. case 'Q':
  476. choice = -1
  477. }
  478. }
  479. d.Write(door.Reset + door.CRNL)
  480. message = fmt.Sprintf("Returning to %s ..."+door.CRNL, d.Config.BBSID)
  481. d.Write(message)
  482. // d.WaitKey(1, 0)
  483. left = d.TimeLeft()
  484. message = fmt.Sprintf("You had %0.2f minutes / %0.2f seconds remaining!"+door.CRNL, left.Minutes(), left.Seconds())
  485. d.Write(message)
  486. left = d.TimeUsed()
  487. d.Write(fmt.Sprintf("You used %0.2f seconds / %0.2f minutes."+door.CRNL, left.Seconds(), left.Minutes()))
  488. fmt.Println("Exiting space-ace")
  489. }