space-ace.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715
  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/seehuhn/mt19937"
  14. )
  15. var SPACEACE string
  16. var SPACEACE_VERSION string
  17. var SPACEACE_COPYRIGHT string
  18. func init() {
  19. SPACEACE = "Space Ace"
  20. SPACEACE_VERSION = "0.0.2"
  21. SPACEACE_COPYRIGHT = "(C) 2021 Bugz, Red-Green Software"
  22. }
  23. var Config map[string]string
  24. func find_words(text string) [][]int {
  25. word, _ := regexp.Compile("([A-Za-z]+)")
  26. return word.FindAllStringIndex(text, -1)
  27. }
  28. func press_a_key(d *door.Door) int {
  29. green := door.ColorText("BOLD GREEN")
  30. blue := door.ColorText("BOLD BLUE")
  31. yellow := door.ColorText("BOLD YELLOW")
  32. any_caps := green
  33. any_lower := yellow
  34. any_color := func(text string) string {
  35. var output string
  36. var lastColor string
  37. for _, letter := range text {
  38. if unicode.IsUpper(letter) {
  39. if lastColor != any_caps {
  40. lastColor = any_caps
  41. output += any_caps
  42. }
  43. output += string(letter)
  44. } else {
  45. if lastColor != any_lower {
  46. lastColor = any_lower
  47. output += any_lower
  48. }
  49. output += string(letter)
  50. }
  51. }
  52. return output
  53. }
  54. text := "Press a key to continue..."
  55. work_text := []rune(text)
  56. d.Write(door.Reset + any_color(text))
  57. var r int = -1
  58. var t int
  59. sleep_ms := 250
  60. ms_sleep := 0
  61. words := find_words(text)
  62. var word int = 0
  63. var wpos int = 0
  64. current_word := text[words[word][0]:words[word][1]]
  65. t = 0
  66. r = d.WaitKey(0, int64(sleep_ms)*1000)
  67. for r == -1 {
  68. ms_sleep += sleep_ms
  69. if ms_sleep > 1000 {
  70. ms_sleep -= 1000
  71. t++
  72. if t >= int(door.Inactivity) {
  73. d.Write(strings.Repeat("\b", len(text)))
  74. d.Write(any_color(text) + door.CRNL)
  75. return -1
  76. }
  77. }
  78. wpos++
  79. if wpos == len(current_word) {
  80. word++
  81. wpos = 0
  82. work_text = []rune(text)
  83. if word == len(words) {
  84. word = 0
  85. if any_caps == green {
  86. any_caps = blue
  87. } else {
  88. any_caps = green
  89. }
  90. d.Write(strings.Repeat("\b", len(text)))
  91. d.Write(any_color(string(work_text)))
  92. current_word = text[words[word][0]:words[word][1]]
  93. goto READKEY
  94. }
  95. current_word = text[words[word][0]:words[word][1]]
  96. }
  97. {
  98. c := rune(current_word[wpos])
  99. for !unicode.IsLower(c) {
  100. wpos++
  101. if wpos == len(current_word) {
  102. wpos = 0
  103. word++
  104. work_text = []rune(text)
  105. if word == len(words) {
  106. word = 0
  107. }
  108. current_word = text[words[word][0]:words[word][1]]
  109. }
  110. c = rune(current_word[wpos])
  111. }
  112. // Ok, we found something that's lower case
  113. work_text[words[word][0]+wpos] = unicode.ToUpper(c)
  114. }
  115. d.Write(strings.Repeat("\b", len(text)))
  116. d.Write(any_color(string(work_text)))
  117. READKEY:
  118. r = d.WaitKey(0, int64(sleep_ms)*1000)
  119. }
  120. d.Write(strings.Repeat("\b", len(text)))
  121. d.Write(any_color(text))
  122. return r
  123. }
  124. func MainMenu() door.Menu {
  125. // Make the main menu
  126. m := door.Menu{Panel: door.Panel{Width: 25,
  127. X: 5,
  128. Y: 5,
  129. Style: door.DOUBLE,
  130. Title: "[ Main Menu: ]",
  131. TitleOffset: 3,
  132. BorderColor: door.ColorText("BRI CYAN ON BLA")}}
  133. m.SelectedR = door.MakeMenuRender(door.ColorText("BOLD CYAN"),
  134. door.ColorText("BOLD BLUE"),
  135. door.ColorText("BOLD CYAN"),
  136. door.ColorText("BOLD BLUE"))
  137. m.UnselectedR = door.MakeMenuRender(door.ColorText("BOLD YEL ON BLUE"),
  138. door.ColorText("BOLD WHI ON BLUE"),
  139. door.ColorText("BOLD YEL ON BLUE"),
  140. door.ColorText("BOLD CYAN ON BLUE"))
  141. m.AddSelection("P", "Play Cards")
  142. m.AddSelection("S", "View Scores")
  143. m.AddSelection("C", "Configure")
  144. m.AddSelection("H", "Help")
  145. m.AddSelection("A", "About this game")
  146. m.AddSelection("Q", "Quit")
  147. return m
  148. }
  149. func ConfigMenu() door.Menu {
  150. m := door.Menu{Panel: door.Panel{Width: 31,
  151. X: 5,
  152. Y: 5,
  153. Style: door.DOUBLE,
  154. Title: "[ Configuration Menu ]",
  155. TitleColor: door.ColorText("BRI CYAN ON BLUE"),
  156. BorderColor: door.ColorText("CYAN ON BLUE")}}
  157. m.SelectedR = door.MakeMenuRender(door.ColorText("BOLD CYAN"),
  158. door.ColorText("BOLD BLUE"),
  159. door.ColorText("BOLD CYAN"),
  160. door.ColorText("BOLD BLUE"))
  161. m.UnselectedR = door.MakeMenuRender(door.ColorText("BOLD YEL ON BLUE"),
  162. door.ColorText("BOLD WHI ON BLUE"),
  163. door.ColorText("BOLD YEL ON BLUE"),
  164. door.ColorText("BOLD CYAN ON BLUE"))
  165. m.AddSelection("D", "Deck Colors")
  166. m.AddSelection("V", "View Settings")
  167. m.AddSelection("Q", "Quit")
  168. return m
  169. }
  170. func display_information(d *door.Door) {
  171. d.Write(door.Clrscr)
  172. keyColor := door.ColorText("BRI GREEN")
  173. sepColor := door.ColorText("BRI YEL")
  174. valColor := door.ColorText("CYAN")
  175. nice_format := func(key string, value string) string {
  176. return fmt.Sprintf("%s%-20s %s: %s%s", keyColor, key, sepColor, valColor, value) + door.CRNL
  177. }
  178. d.Write(nice_format("BBS Software", d.Config.BBSID))
  179. d.Write(nice_format("Real Name", d.Config.Real_name))
  180. d.Write(nice_format("Handle", d.Config.Handle))
  181. d.Write(nice_format("User #", strconv.Itoa(d.Config.User_number)))
  182. d.Write(nice_format("Security Level", strconv.Itoa(d.Config.Security_level)))
  183. d.Write(nice_format("Node #", strconv.Itoa(d.Config.Node)))
  184. d.Write(nice_format("Unicode", strconv.FormatBool(door.Unicode)))
  185. d.Write(nice_format("CP437", strconv.FormatBool(door.CP437)))
  186. d.Write(nice_format("Screen Size", fmt.Sprintf("%d X %d", door.Width, door.Height)))
  187. }
  188. func panel_demo(d *door.Door) {
  189. width := 55
  190. fmtStr := "%-55s"
  191. p := door.Panel{X: 5, Y: 5, Width: width, Style: door.DOUBLE, BorderColor: door.ColorText("CYAN ON BLUE"), Title: "[ Panel Demo ]"}
  192. lineColor := door.ColorText("BRIGHT WHI ON BLUE")
  193. // Add lines to the panel
  194. for _, line := range []string{"The BBS Door Panel Demo", "(C) 2021 Red Green Software, https://red-green.com"} {
  195. if door.Unicode {
  196. line = strings.Replace(line, "(C)", "\u00a9", -1)
  197. }
  198. l := door.Line{Text: fmt.Sprintf(fmtStr, line), DefaultColor: lineColor}
  199. p.Lines = append(p.Lines, l)
  200. }
  201. p.Lines = append(p.Lines, p.Spacer())
  202. p.Lines = append(p.Lines, door.Line{Text: fmt.Sprintf(fmtStr, "Welcome to golang!"), DefaultColor: lineColor})
  203. d.Write(door.Clrscr)
  204. d.Write(p.Output() + door.CRNL)
  205. }
  206. func MakeColorsRender(brackets string, text_color string, colors map[string]string) func(string) string {
  207. renderF := func(text string) string {
  208. words := find_words(text)
  209. var option bool = true
  210. var color_word bool = false
  211. var result string
  212. var last_color string
  213. var word_color string
  214. var words_id = 0
  215. word_pair := words[words_id]
  216. normal := colors["ALL"]
  217. for tpos, c := range text {
  218. if option {
  219. if c == '[' || c == ']' {
  220. if last_color != brackets {
  221. result += brackets
  222. last_color = brackets
  223. }
  224. if c == ']' {
  225. option = false
  226. }
  227. } else {
  228. if last_color != text_color {
  229. result += text_color
  230. last_color = text_color
  231. }
  232. }
  233. result += string(c)
  234. } else {
  235. // Out of the options
  236. if color_word {
  237. if tpos < word_pair[1] {
  238. if last_color != word_color {
  239. result += word_color
  240. last_color = word_color
  241. } else {
  242. color_word = false
  243. if last_color != normal {
  244. result += normal
  245. last_color = normal
  246. }
  247. }
  248. result += string(c)
  249. }
  250. } else {
  251. // look for COLOR word
  252. if tpos >= word_pair[1] {
  253. words_id++
  254. if words_id >= len(words) {
  255. word_pair = []int{999, 999}
  256. } else {
  257. word_pair = words[words_id]
  258. }
  259. }
  260. if word_pair[0] == tpos {
  261. // start of word
  262. possible := text[word_pair[0]:word_pair[1]]
  263. color_code, has := colors[strings.ToUpper(possible)]
  264. // log.Printf("Match %s / found %s %#v\n", possible, color_code, has)
  265. if has {
  266. word_color = color_code
  267. } else {
  268. word_color = colors["ALL"]
  269. }
  270. if last_color != word_color {
  271. result += word_color
  272. last_color = word_color
  273. }
  274. }
  275. result += string(c)
  276. }
  277. }
  278. }
  279. return result
  280. }
  281. return renderF
  282. }
  283. var DeckColors []string
  284. func init() {
  285. DeckColors = []string{
  286. "All", "Brown", "Green", "Red", "Blue", "Cyan", "Magenta", "White"}
  287. }
  288. func DeckColorMenu(current string) door.Menu {
  289. m := door.Menu{Panel: door.Panel{Width: 31,
  290. X: 5,
  291. Y: 5,
  292. Style: door.DOUBLE,
  293. Title: "[ Deck Menu ]",
  294. TitleColor: door.ColorText("BRI CYAN ON BLUE"),
  295. BorderColor: door.ColorText("CYAN ON BLUE")}}
  296. unselectMap := map[string]string{"BLUE": door.ColorText("BRI BLUE ON BLUE"),
  297. "BROWN": door.ColorText("BROWN ON BLUE"),
  298. "RED": door.ColorText("BRI RED ON BLUE"),
  299. "CYAN": door.ColorText("CYAN ON BLUE"),
  300. "GREEN": door.ColorText("BRI GREEN ON BLUE"),
  301. "MAGENTA": door.ColorText("BRI MAGENTA ON BLUE"),
  302. "WHITE": door.ColorText("BRI WHITE ON BLUE"),
  303. "ALL": door.ColorText("BRI WHITE ON BLUE")}
  304. selectMap := map[string]string{"BLUE": door.ColorText("BRI BLUE ON BLACK"),
  305. "BROWN": door.ColorText("BROWN ON BLACK"),
  306. "RED": door.ColorText("BRI RED ON BLACK"),
  307. "CYAN": door.ColorText("CYAN ON BLACK"),
  308. "GREEN": door.ColorText("BRI GREEN ON BLACK"),
  309. "MAGENTA": door.ColorText("BRI MAGENTA ON BLACK"),
  310. "WHITE": door.ColorText("BRI WHITE ON BLACK"),
  311. "ALL": door.ColorText("BRI WHITE ON BLACK")}
  312. m.SelectedR = MakeColorsRender(door.ColorText("BOLD CYAN"),
  313. door.ColorText("BOLD BLUE"),
  314. selectMap)
  315. m.UnselectedR = MakeColorsRender(door.ColorText("BOLD YEL ON BLUE"),
  316. door.ColorText("BOLD WHI ON BLUE"),
  317. unselectMap)
  318. m.Chosen = 0
  319. for idx, color := range DeckColors {
  320. m.AddSelection(color[:1], color)
  321. if color == current {
  322. m.Chosen = idx
  323. }
  324. }
  325. return m
  326. }
  327. /*
  328. door::renderFunction statusValue(door::ANSIColor status,
  329. door::ANSIColor value) {
  330. door::renderFunction rf = [status,
  331. value](const std::string &txt) -> door::Render {
  332. door::Render r(txt);
  333. size_t pos = txt.find(':');
  334. if (pos == std::string::npos) {
  335. for (char const &c : txt) {
  336. if (std::isdigit(c))
  337. r.append(value);
  338. else
  339. r.append(status);
  340. }
  341. } else {
  342. pos++;
  343. r.append(status, pos);
  344. r.append(value, txt.length() - pos);
  345. }
  346. return r;
  347. };
  348. return rf;
  349. }
  350. */
  351. func RenderStatusValue(status string, value string) func(string) string {
  352. renderF := func(text string) string {
  353. pos := strings.Index(text, ":")
  354. if pos != -1 {
  355. return status + text[:pos] + value + text[pos:]
  356. } else {
  357. var r door.Render = door.Render{Line: text}
  358. for _, letter := range text {
  359. if unicode.IsDigit(letter) {
  360. r.Append(value, 1)
  361. } else {
  362. r.Append(status, 1)
  363. }
  364. }
  365. return r.Result
  366. }
  367. }
  368. return renderF
  369. }
  370. // Display the SysOp settings in the Config/yaml file
  371. func DisplaySettings(d *door.Door) {
  372. d.Write(door.Clrscr + door.Reset)
  373. W := 35
  374. panel := door.Panel{Width: W,
  375. X: 5,
  376. Y: 5,
  377. Style: door.DOUBLE,
  378. BorderColor: door.ColorText("BOLD CYAN ON BLUE"),
  379. }
  380. l := door.Line{Text: fmt.Sprintf("%*s", W, "Game Settings - SysOp Configurable"),
  381. DefaultColor: door.ColorText("BOLD GREEN ON BLUE")}
  382. panel.Lines = append(panel.Lines, l)
  383. renderF := RenderStatusValue(door.ColorText("BOLD YELLOW ON BLUE"), door.ColorText("BOLD GREEN ON BLUE"))
  384. for key, value := range Config {
  385. if key[0] == '_' {
  386. continue
  387. }
  388. key = strings.Replace(key, "_", " ", -1)
  389. l = door.Line{Text: fmt.Sprintf("%20s : %*s", key, -(W - 23), value), RenderF: renderF}
  390. panel.Lines = append(panel.Lines, l)
  391. }
  392. d.Write(panel.Output() + door.Reset + door.CRNL)
  393. press_a_key(d)
  394. }
  395. func Configure(d *door.Door, db *DBData) {
  396. menu := ConfigMenu()
  397. // deckcolor := "DeckColor"
  398. // save_deckcolor := false
  399. var choice int
  400. for choice >= 0 {
  401. d.Write(door.Clrscr)
  402. choice = menu.Choose(d)
  403. if choice < 0 {
  404. break
  405. }
  406. option := menu.GetOption(choice)
  407. switch option {
  408. case 'D':
  409. // Deck color
  410. current_color := db.GetSetting("DeckColor", "All")
  411. colormenu := DeckColorMenu(current_color)
  412. d.Write(door.Clrscr)
  413. c := colormenu.Choose(d)
  414. if c > 0 {
  415. db.SetSetting("DeckColor", DeckColors[c-1])
  416. }
  417. d.Write(door.CRNL)
  418. press_a_key(d)
  419. case 'V':
  420. // View settings
  421. DisplaySettings(d)
  422. case 'Q':
  423. choice = -1
  424. }
  425. }
  426. }
  427. func Help() door.Panel {
  428. W := 60
  429. center_x := (door.Width - W) / 2
  430. center_y := (door.Height - 16) / 2
  431. help := door.Panel{X: center_x,
  432. Y: center_y,
  433. Width: W,
  434. Style: door.DOUBLE,
  435. BorderColor: door.ColorText("BOLD YELLOW ON BLUE")}
  436. help.Lines = append(help.Lines, door.Line{Text: fmt.Sprintf("%*s", -W, "Help"),
  437. DefaultColor: door.ColorText("BOLD CYAN ON BLUE")})
  438. help.Lines = append(help.Lines, help.Spacer())
  439. copyright := SPACEACE + " v" + SPACEACE_VERSION
  440. help.Lines = append(help.Lines,
  441. door.Line{Text: fmt.Sprintf("%*s", -W, copyright),
  442. DefaultColor: door.ColorText("BOLD WHITE ON BLUE")})
  443. copyright = SPACEACE_COPYRIGHT
  444. if door.Unicode {
  445. copyright = strings.Replace(copyright, "(C)", "\u00a9", -1)
  446. }
  447. help.Lines = append(help.Lines,
  448. door.Line{Text: fmt.Sprintf("%*s", -W, copyright),
  449. DefaultColor: door.ColorText("BOLD WHITE ON BLUE")})
  450. for _, text := range []string{"",
  451. "Use Left/Right arrow keys, or 4/6 keys to move marker.",
  452. "The marker wraps around the sides of the screen.", "",
  453. "Select card to play with Space or 5.",
  454. "A card can play if it is higher or lower in rank by 1.",
  455. "",
  456. "Enter draws another card.",
  457. "",
  458. "Example: A Jack could play either a Ten or a Queen.",
  459. "Example: A King could play either an Ace or a Queen.",
  460. "",
  461. "The more cards in your streak, the more points earn.",
  462. } {
  463. help.Lines = append(help.Lines, door.Line{Text: fmt.Sprintf("%*s", -W, text)})
  464. }
  465. return help
  466. }
  467. func About() door.Panel {
  468. W := 60
  469. center_x := (door.Width - W) / 2
  470. center_y := (door.Height - 16) / 2
  471. about := door.Panel{X: center_x,
  472. Y: center_y,
  473. Width: W,
  474. Style: door.SINGLE_DOUBLE,
  475. BorderColor: door.ColorText("BOLD YELLOW ON BLUE"),
  476. }
  477. about.Lines = append(about.Lines, door.Line{Text: fmt.Sprintf("%*s", -W, "About This Door"),
  478. DefaultColor: door.ColorText("BOLD CYAN ON BLUE")})
  479. about.Lines = append(about.Lines, about.Spacer())
  480. copyright := SPACEACE + " v" + SPACEACE_VERSION
  481. about.Lines = append(about.Lines, door.Line{Text: fmt.Sprintf("%*s", -W, copyright)})
  482. copyright = SPACEACE_COPYRIGHT
  483. if door.Unicode {
  484. copyright = strings.Replace(copyright, "(C)", "\u00a9", -1)
  485. }
  486. about.Lines = append(about.Lines,
  487. door.Line{Text: fmt.Sprintf("%*s", -W, copyright),
  488. DefaultColor: door.ColorText("BOLD WHITE ON BLUE")})
  489. for _, text := range []string{"",
  490. "This door was written by Bugz.",
  491. "",
  492. "It is written in Go, understands CP437 and unicode, adapts",
  493. "to screen sizes, uses door32.sys, supports TheDraw Fonts,",
  494. "and runs on Linux and Windows."} {
  495. about.Lines = append(about.Lines, door.Line{Text: fmt.Sprintf("%*s", -W, text)})
  496. }
  497. return about
  498. }
  499. func main() {
  500. var message string
  501. // Get path to binary, and chdir to it.
  502. binaryPath, _ := os.Executable()
  503. binaryPath = filepath.Dir(binaryPath)
  504. _ = os.Chdir(binaryPath)
  505. fmt.Println("Starting space-ace")
  506. rng := rand.New(mt19937.New())
  507. rng.Seed(time.Now().UnixNano())
  508. d := door.Door{}
  509. d.Init("space-ace")
  510. db := DBData{}
  511. db.Open("space-database.db")
  512. defer db.Close()
  513. // Use Real_name or Handle? Or read config?
  514. db.User = d.Config.Real_name
  515. const config_filename = "space-ace.yaml"
  516. if FileExists(config_filename) {
  517. Config = LoadConfig(config_filename)
  518. } else {
  519. Config = make(map[string]string)
  520. }
  521. var update_config bool = false
  522. // Guessing at this point as to what settings I actually want.
  523. config_defaults := map[string]string{"hands_per_day": "3",
  524. "date_format": "January 2",
  525. "date_score": "01/02/2006",
  526. "makeup_per_day": "5",
  527. "play_days_ahead": "2",
  528. "date_monthly": "January 2006"}
  529. // _seed
  530. _, ok := Config["_seed"]
  531. if !ok {
  532. // Initialize the seed
  533. Config["_seed"] = fmt.Sprintf("%d,%d,%d", rng.Int31(), rng.Int31(), rng.Int31())
  534. update_config = true
  535. }
  536. for key, value := range config_defaults {
  537. if SetConfigDefault(&Config, key, value) {
  538. update_config = true
  539. }
  540. }
  541. if update_config {
  542. SaveConfig(config_filename, Config)
  543. }
  544. starfield := StarField{}
  545. starfield.Regenerate(rng)
  546. starfield.Display(&d)
  547. d.Write(door.Goto(1, 1) + door.Reset)
  548. // Get the last call value (if they have called before)
  549. last_call, _ := strconv.ParseInt(db.GetSetting("LastCall", "0"), 10, 64)
  550. now := time.Now()
  551. db.SetSetting("LastCall", fmt.Sprintf("%d", now.Unix()))
  552. // Check for maint run -- get FirstOfMonthDate, and see if
  553. // records are older then it is. (If so, yes -- run maint)!
  554. db.ExpireScores(0)
  555. if last_call != 0 {
  556. d.Write("Welcome Back!" + door.CRNL)
  557. delta := now.Sub(time.Unix(last_call, 0))
  558. hours := delta.Hours()
  559. if hours > 24 {
  560. d.Write(fmt.Sprintf("It's been %0.1f days since you last played."+door.CRNL, hours/24))
  561. } else {
  562. if hours > 1 {
  563. d.Write(fmt.Sprintf("It's been %0.1f hours since you last played."+door.CRNL, hours))
  564. } else {
  565. minutes := delta.Minutes()
  566. d.Write(fmt.Sprintf("It's been %0.1f minutes since you last played."+door.CRNL, minutes))
  567. }
  568. }
  569. }
  570. left := d.TimeLeft()
  571. message = fmt.Sprintf("You have %0.2f minutes / %0.2f seconds remaining..."+door.CRNL, left.Minutes(), left.Seconds())
  572. d.Write(message)
  573. press_a_key(&d)
  574. mainmenu := MainMenu()
  575. var choice int
  576. for choice >= 0 {
  577. d.Write(door.Clrscr)
  578. starfield.Display(&d)
  579. choice = mainmenu.Choose(&d)
  580. if choice < 0 {
  581. break
  582. }
  583. option := mainmenu.GetOption(choice)
  584. // fmt.Printf("Choice: %d, Option: %c\n", choice, option)
  585. switch option {
  586. case 'P':
  587. // Play cards
  588. pc := PlayCards{Door: &d,
  589. DB: &db,
  590. Config: &Config,
  591. RNG: rng,
  592. }
  593. pc.Init()
  594. pc.Play()
  595. case 'S':
  596. // View Scores
  597. case 'C':
  598. // Configure
  599. Configure(&d, &db)
  600. case 'H':
  601. // Help
  602. h := Help()
  603. d.Write(door.Clrscr + h.Output() + door.Reset + door.CRNL)
  604. press_a_key(&d)
  605. case 'A':
  606. // About
  607. a := About()
  608. d.Write(door.Clrscr + a.Output() + door.Reset + door.CRNL)
  609. press_a_key(&d)
  610. case 'Q':
  611. choice = -1
  612. }
  613. }
  614. d.Write(door.Reset + door.CRNL)
  615. message = fmt.Sprintf("Returning to %s ..."+door.CRNL, d.Config.BBSID)
  616. d.Write(message)
  617. // d.WaitKey(1, 0)
  618. left = d.TimeLeft()
  619. message = fmt.Sprintf("You had %0.2f minutes / %0.2f seconds remaining!"+door.CRNL, left.Minutes(), left.Seconds())
  620. d.Write(message)
  621. left = d.TimeUsed()
  622. d.Write(fmt.Sprintf("You used %0.2f seconds / %0.2f minutes."+door.CRNL, left.Seconds(), left.Minutes()))
  623. fmt.Println("Exiting space-ace")
  624. }