wopr.go 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  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. // No change here.
  129. /*
  130. var buffer [16]byte = [16]byte{' ', ' ', ' ', ' ',
  131. ' ', ' ', ' ', ' ',
  132. ' ', ' ', ' ', ' ',
  133. ' ', ' ', ' ', ' '}
  134. */
  135. // bytes.Repeat([]byte(" "), 16)
  136. buffer[15-w.Index] = '\xfe'
  137. u.Write(buffer[:])
  138. // u.WriteString(string(buffer))
  139. }
  140. }
  141. ranimate.Update() // Text.Write(ranimate.UpdateF())
  142. w.RemainingPanel.Lines = append(w.RemainingPanel.Lines, ranimate)
  143. }
  144. func (w *WOPR) Inc() {
  145. w.Index++
  146. if w.Index == 16 {
  147. w.Index = 0
  148. w.ElapsedPanel.Update()
  149. w.RemainingPanel.Update()
  150. w.RemainingD = time.Duration(w.RemainingD.Seconds()-1) * time.Second
  151. w.ElapsedD = time.Duration(w.ElapsedD.Seconds()+1) * time.Second
  152. }
  153. }
  154. var woprBytesUsed int
  155. func (w *WOPR) Output() []byte {
  156. w.output.Reset()
  157. w.output.WriteString(SavePos)
  158. w.output.Write(w.ElapsedPanel.Output())
  159. w.output.Write(w.RemainingPanel.Output())
  160. w.output.WriteString(RestorePos)
  161. if DEBUG_WOPR {
  162. if w.output.Cap() > woprBytesUsed {
  163. woprBytesUsed = w.output.Cap()
  164. log.Printf("WOPR: now %d\n", woprBytesUsed)
  165. }
  166. }
  167. return w.output.Bytes()
  168. }
  169. func (w *WOPR) Animate(d *Door) {
  170. // til := time.Now().UnixNano() % int64(time.Second)
  171. // w.Index = int((til / int64(time.Microsecond)) / 62500)
  172. // either put the sync sleep in the go routine, or sleep and sync Index.
  173. // time.Sleep(time.Duration(int64(time.Second) - til)) // time.Now().UnixMilli()%1000) * time.Millisecond)
  174. // time.Second / 16
  175. // w.Ticker = time.NewTicker(time.Microsecond * time.Duration(62500))
  176. // w.Index = 0
  177. // Convert time.Time to time.Duration
  178. // This gives us consistency when resuming.
  179. w.ElapsedD = time.Since(w.Elapsed)
  180. w.RemainingD = time.Until(w.Remaining)
  181. w.StopIt = make(chan bool)
  182. go func(d *Door) {
  183. // til := time.Now().UnixNano() % int64(time.Second)
  184. sec16 := int64(time.Second) / 16
  185. // tilms := til % sec16
  186. w.Index = 0 // int(til / sec16)
  187. // log.Printf("til: %d, sec: %d, w.Index: %d\n", til, sec16, w.Index)
  188. // either put the sync sleep in the go routine, or sleep and sync Index.
  189. // time.Sleep(time.Second - time.Duration(til)) // sec16 - (til % sec16))) //int64(time.Second) - til)) // time.Now().UnixMilli()%1000) * time.Millisecond)
  190. // time.Second / 16
  191. w.Ticker = time.NewTicker(time.Duration(sec16))
  192. // var output bytes.Buffer
  193. for {
  194. select {
  195. case <-w.StopIt:
  196. return
  197. case <-w.Ticker.C:
  198. w.ElapsedPanel.Update()
  199. w.RemainingPanel.Update()
  200. var output []byte = w.Output()
  201. if !d.Writer.IsClosed() {
  202. d.Write(output)
  203. } else {
  204. w.Ticker.Stop()
  205. return
  206. }
  207. w.Inc()
  208. }
  209. }
  210. }(d)
  211. }
  212. func (w *WOPR) Stop() {
  213. w.Ticker.Stop()
  214. w.StopIt <- true
  215. }