wopr.go 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. package door
  2. import (
  3. "bytes"
  4. "fmt"
  5. "log"
  6. "strings"
  7. "time"
  8. )
  9. /*
  10. WOPR
  11. [ GAME ]
  12. [ TIME ELAPSED ]
  13. [ XX HRS ]
  14. [ XX MIN XX SEC ]
  15. [■ ] ->
  16. [ GAME ]
  17. [ TIME REMAINING ]
  18. [ XX HRS ]
  19. [ XX MIN XX SEC ]
  20. [ ■] <-
  21. Border SINGLE
  22. Ok, Width = 16
  23. 1000 / 16 = 62.5 ms = 62500 us
  24. Width 16 (at pos 1, update time/sec increment)
  25. Black on Cyan
  26. */
  27. type WOPR struct {
  28. ElapsedPanel Panel
  29. Elapsed time.Time
  30. ElapsedD time.Duration
  31. RemainingPanel Panel
  32. Remaining time.Time
  33. RemainingD time.Duration
  34. Color string
  35. Index int
  36. Ticker *time.Ticker
  37. StopIt chan bool
  38. output *bytes.Buffer
  39. }
  40. const DEBUG_WOPR bool = true
  41. func (w *WOPR) Clear() []byte {
  42. var output bytes.Buffer
  43. output.Write(w.ElapsedPanel.Clear())
  44. output.Write(w.RemainingPanel.Clear())
  45. return output.Bytes()
  46. }
  47. // Initialize, Set X, Y on Panels, Animate()
  48. // Initialize, and create panels
  49. func (w *WOPR) Init(elapsed time.Time, remaining time.Time, color string) {
  50. if color == "" {
  51. color = ColorText("BLACK ON CYAN")
  52. }
  53. w.output = &bytes.Buffer{}
  54. w.Elapsed = elapsed
  55. w.Remaining = remaining
  56. w.Index = 0
  57. w.ElapsedPanel = Panel{Width: 16,
  58. BorderColor: color,
  59. Style: SINGLE}
  60. w.ElapsedPanel.Lines = append(w.ElapsedPanel.Lines, NewLine(" GAME "))
  61. w.ElapsedPanel.Lines = append(w.ElapsedPanel.Lines, NewLine(" TIME ELAPSED "))
  62. var ehour Line = Line{}
  63. ehour.UpdateF = func(u *bytes.Buffer) {
  64. var hours int = int(w.ElapsedD.Hours())
  65. u.Reset()
  66. fmt.Fprintf(u, " %02d HRS ", hours%100)
  67. }
  68. ehour.Update()
  69. // ehour.Text.Write(ehour.update)
  70. w.ElapsedPanel.Lines = append(w.ElapsedPanel.Lines, ehour)
  71. var eminsec Line = Line{}
  72. eminsec.UpdateF = func(u *bytes.Buffer) {
  73. var mins int = int(w.ElapsedD.Minutes()) % 60
  74. var secs int = int(w.ElapsedD.Seconds()) % 60
  75. u.Reset()
  76. fmt.Fprintf(u, " %02d MIN %02d SEC ", mins, secs)
  77. }
  78. eminsec.Update() // Text.Write(eminsec.UpdateF())
  79. w.ElapsedPanel.Lines = append(w.ElapsedPanel.Lines, eminsec)
  80. var eanimate Line = Line{}
  81. eanimate.UpdateF = func(u *bytes.Buffer) {
  82. u.Reset()
  83. // Left to Right
  84. if Unicode {
  85. var buffer []rune = []rune(strings.Repeat(" ", 16))
  86. buffer[w.Index] = '\u25a0'
  87. u.WriteString(string(buffer))
  88. } else {
  89. var buffer []byte = bytes.Repeat([]byte(" "), 16)
  90. buffer[w.Index] = '\xfe'
  91. u.WriteString(string(buffer))
  92. }
  93. }
  94. eanimate.Update() // Text.Write(eanimate.UpdateF())
  95. w.ElapsedPanel.Lines = append(w.ElapsedPanel.Lines, eanimate)
  96. w.RemainingPanel = Panel{Width: 16,
  97. BorderColor: color,
  98. Style: SINGLE}
  99. w.RemainingPanel.Lines = append(w.RemainingPanel.Lines, NewLine(" GAME "))
  100. w.RemainingPanel.Lines = append(w.RemainingPanel.Lines, NewLine(" TIME REMAINING "))
  101. var rhour Line = Line{}
  102. rhour.UpdateF = func(u *bytes.Buffer) {
  103. var hours int = int(w.RemainingD.Hours())
  104. u.Reset()
  105. fmt.Fprintf(u, " %02d HRS ", hours%100)
  106. }
  107. rhour.Update() // Text.Write(rhour.UpdateF())
  108. w.RemainingPanel.Lines = append(w.RemainingPanel.Lines, rhour)
  109. var rminsec Line = Line{}
  110. rminsec.UpdateF = func(u *bytes.Buffer) {
  111. var mins int = int(w.RemainingD.Minutes()) % 60
  112. var secs int = int(w.RemainingD.Seconds()) % 60
  113. u.Reset()
  114. fmt.Fprintf(u, " %02d MIN %02d SEC ", mins, secs)
  115. }
  116. rminsec.Update() // Text.Write(rminsec.UpdateF())
  117. w.RemainingPanel.Lines = append(w.RemainingPanel.Lines, rminsec)
  118. var ranimate Line = Line{}
  119. ranimate.UpdateF = func(u *bytes.Buffer) {
  120. u.Reset()
  121. // Left to Right
  122. if Unicode {
  123. var buffer []rune = []rune(strings.Repeat(" ", 16))
  124. buffer[15-w.Index] = '\u25a0'
  125. u.WriteString(string(buffer))
  126. } else {
  127. var buffer []byte = bytes.Repeat([]byte(" "), 16)
  128. buffer[15-w.Index] = '\xfe'
  129. u.WriteString(string(buffer))
  130. }
  131. }
  132. ranimate.Update() // Text.Write(ranimate.UpdateF())
  133. w.RemainingPanel.Lines = append(w.RemainingPanel.Lines, ranimate)
  134. }
  135. func (w *WOPR) Inc() {
  136. w.Index++
  137. if w.Index == 16 {
  138. w.Index = 0
  139. w.ElapsedPanel.Update()
  140. w.RemainingPanel.Update()
  141. w.RemainingD = time.Duration(w.RemainingD.Seconds()-1) * time.Second
  142. w.ElapsedD = time.Duration(w.ElapsedD.Seconds()+1) * time.Second
  143. }
  144. }
  145. func (w *WOPR) Animate(d *Door) {
  146. // til := time.Now().UnixNano() % int64(time.Second)
  147. // w.Index = int((til / int64(time.Microsecond)) / 62500)
  148. // either put the sync sleep in the go routine, or sleep and sync Index.
  149. // time.Sleep(time.Duration(int64(time.Second) - til)) // time.Now().UnixMilli()%1000) * time.Millisecond)
  150. // time.Second / 16
  151. // w.Ticker = time.NewTicker(time.Microsecond * time.Duration(62500))
  152. // w.Index = 0
  153. // Convert time.Time to time.Duration
  154. // This gives us consistency when resuming.
  155. w.ElapsedD = time.Since(w.Elapsed)
  156. w.RemainingD = time.Until(w.Remaining)
  157. w.StopIt = make(chan bool)
  158. go func(d *Door, output *bytes.Buffer) {
  159. // til := time.Now().UnixNano() % int64(time.Second)
  160. sec16 := int64(time.Second) / 16
  161. // tilms := til % sec16
  162. w.Index = 0 // int(til / sec16)
  163. // log.Printf("til: %d, sec: %d, w.Index: %d\n", til, sec16, w.Index)
  164. // either put the sync sleep in the go routine, or sleep and sync Index.
  165. // time.Sleep(time.Second - time.Duration(til)) // sec16 - (til % sec16))) //int64(time.Second) - til)) // time.Now().UnixMilli()%1000) * time.Millisecond)
  166. // time.Second / 16
  167. w.Ticker = time.NewTicker(time.Duration(sec16))
  168. // var output bytes.Buffer
  169. for {
  170. select {
  171. case <-w.StopIt:
  172. return
  173. case <-w.Ticker.C:
  174. w.ElapsedPanel.Update()
  175. w.RemainingPanel.Update()
  176. var using int
  177. using = output.Cap()
  178. output.Reset()
  179. output.WriteString(SavePos)
  180. output.Write(w.ElapsedPanel.Output())
  181. output.Write(w.RemainingPanel.Output())
  182. output.WriteString(RestorePos)
  183. if DEBUG_WOPR {
  184. if output.Cap() > using {
  185. using = output.Cap()
  186. log.Printf("WOPR: now %d\n", using)
  187. }
  188. }
  189. if !d.Writer.IsClosed() {
  190. d.Write(output.Bytes())
  191. output.Reset()
  192. } else {
  193. w.Ticker.Stop()
  194. return
  195. }
  196. w.Inc()
  197. }
  198. }
  199. }(d, w.output)
  200. }
  201. func (w *WOPR) Stop() {
  202. w.Ticker.Stop()
  203. w.StopIt <- true
  204. }