line.go 5.3 KB

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