line.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. package door
  2. import (
  3. "bytes"
  4. "log"
  5. "unicode"
  6. )
  7. /*
  8. door.Line - Display a line of text
  9. Example:
  10. var basicLine door.Line = {Text: "Welcome",
  11. DefaultColor: door.Color("BRIGHT YELLOW"),
  12. }
  13. d.Write(basicLine.Output() + door.Reset + door.CRNL)
  14. This outputs Welcome in Bright/Bold Yellow.
  15. Example Render:
  16. var renderAlphaDigit ColorRender = func(text string) string {
  17. var r door.Render{Text: text}
  18. var alpha string = door.ColorText("BOLD YELLOW")
  19. var digit string = door.ColorText("BOLD GREEN")
  20. var other string = door.Reset
  21. for _, letter := range text {
  22. if unicode.IsAlpha(letter) {
  23. r.Append(alpha, 1)
  24. } else {
  25. if unicode.IsDigit(letter) {
  26. r.Append(digit, 1)
  27. } else {
  28. r.Append(other, 1)
  29. }
  30. }
  31. }
  32. return r.Result
  33. }
  34. var renderLine door.Line = {Text: "Render - 12345",
  35. RenderF: renderAlphaDigit,
  36. }
  37. d.Write(renderLine.Output() + door.Reset + door.CRNL)
  38. This outputs "Render" in Yellow, "12345" in Green, and " - " in White.
  39. Example Update:
  40. var updateTime() string {
  41. var now time.Time = time.Now()
  42. var result string = now.Format("3:04:05 PM")
  43. return result
  44. }
  45. var timeLine door.Line = {Text: updateTime(),
  46. UpdateF: updateTime,
  47. }
  48. d.Write(timeLine.Output() + door.CRNL)
  49. time.Sleep(time.Second)
  50. // Check timeLine for update
  51. if timeLine.Update() {
  52. // Yes, there's an update to the time, output it.
  53. d.Write(timeLine.Output() + door.CRNL)
  54. }
  55. if timeLine.Update() {
  56. // This isn't called. There were no changes.
  57. d.Write(timeLine.Output() + door.CRNL)
  58. }
  59. This outputs the Current time in 12 hour format. It pauses for a second,
  60. and outputs the new time.
  61. */
  62. /*
  63. Line of text to display.
  64. */
  65. type Line struct {
  66. Text *bytes.Buffer // Text to be displayed
  67. update *bytes.Buffer // Buffer for updates
  68. DefaultColor string // Default Color to use
  69. RenderF ColorRender // Render function (displays string with colors)
  70. render *bytes.Buffer // Buffer for rendering
  71. UpdateF Updater // Update function updates the text
  72. Width int // Line length
  73. }
  74. func NewLine(text string) Line {
  75. return Line{Text: bytes.NewBuffer([]byte(text))}
  76. }
  77. /*
  78. Line Update - This calls the UpdateF if present.
  79. Returns true if the line has been updated, and there's an Update function.
  80. */
  81. func (l *Line) Update() bool {
  82. if l.Text == nil {
  83. l.Text = &bytes.Buffer{}
  84. }
  85. if l.UpdateF == nil {
  86. return false
  87. }
  88. if l.update == nil {
  89. l.update = &bytes.Buffer{}
  90. }
  91. l.update.Reset()
  92. l.UpdateF(l.update)
  93. l.LineLength(l.update)
  94. if bytes.Compare(l.update.Bytes(), l.Text.Bytes()) != 0 {
  95. l.Text.Reset()
  96. l.Text.Write(l.update.Bytes())
  97. return true
  98. }
  99. return false
  100. }
  101. // If a line Width has been set, make sure we match it.
  102. func (l *Line) LineLength(text *bytes.Buffer) {
  103. if l.Width == 0 {
  104. return
  105. }
  106. var length int
  107. if Unicode {
  108. var ubuff *bytes.Buffer = bytes.NewBuffer(text.Bytes())
  109. var e error
  110. var r rune
  111. for {
  112. r, _, e = ubuff.ReadRune()
  113. if e != nil {
  114. break
  115. }
  116. length += UnicodeWidth(r)
  117. }
  118. } else {
  119. length = text.Len()
  120. }
  121. if length > l.Width {
  122. log.Printf("ERROR: Line Width %d: Have %d\n", l.Width, length)
  123. } else {
  124. for length < l.Width {
  125. text.WriteByte(' ')
  126. length++
  127. }
  128. // *text += strings.Repeat(" ", l.Width-length)
  129. }
  130. }
  131. /*
  132. Line Output - returns a string with ANSI Color codes.
  133. If there is no RenderF, we use the DefaultColor. Otherwise we pass the text
  134. to RenderF and return what it returns.
  135. */
  136. func (l *Line) Output() []byte {
  137. if l.UpdateF == nil {
  138. l.LineLength(l.Text)
  139. }
  140. if l.render == nil {
  141. l.render = &bytes.Buffer{}
  142. }
  143. if l.RenderF == nil {
  144. l.render.Reset()
  145. l.render.WriteString(l.DefaultColor)
  146. l.render.Write(l.Text.Bytes())
  147. return l.render.Bytes()
  148. // return l.DefaultColor + l.Text
  149. } else {
  150. return l.RenderF(l.render, l.Text.Bytes())
  151. }
  152. }
  153. func RenderBlueYellow(output *bytes.Buffer, text []byte) []byte {
  154. output.Reset()
  155. // var output = RenderPool.Get().(*strings.Builder)
  156. // output.Reset()
  157. var blue string = ColorText("BOLD BLUE")
  158. var yellow string = ColorText("BOLD YELLOW")
  159. var last *string
  160. for _, letter := range text {
  161. if unicode.IsUpper(rune(letter)) {
  162. if last != &blue {
  163. output.WriteString(blue)
  164. last = &blue
  165. }
  166. } else {
  167. if last != &yellow {
  168. output.WriteString(yellow)
  169. last = &yellow
  170. }
  171. }
  172. output.WriteByte(letter)
  173. }
  174. // var result = output.String()
  175. // RenderPool.Put(output)
  176. // return result
  177. return output.Bytes()
  178. }