space-ace.go 13 KB

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