write.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. package door
  2. import (
  3. "bytes"
  4. "log"
  5. "regexp"
  6. "strconv"
  7. "strings"
  8. )
  9. var FindANSIColor *regexp.Regexp
  10. func init() {
  11. FindANSIColor = regexp.MustCompile("\x1b\\[([0-9;]*)m")
  12. }
  13. /*
  14. find \x1b[0;1;37;40m codes.
  15. return array of start/end positions in the given string.
  16. */
  17. func find_ansicolor(text string) [][]int {
  18. var color_codes [][]int
  19. // word, _ := regexp.Compile("\x1b\\[([0-9;]+)m")
  20. var colors [][]int = FindANSIColor.FindAllStringIndex(text, -1)
  21. // regexp seems to be ignoring the capture groups.
  22. // colors := word.FindAllSubmatchIndex([]byte(text), len(text))
  23. for _, pos := range colors {
  24. var txt string = text[pos[0]+2 : pos[1]-1]
  25. if txt == "" {
  26. txt = "0"
  27. }
  28. // log.Printf("Text: [%s]\n", txt)
  29. var codes []string = strings.Split(txt, ";")
  30. // log.Printf("Codes: [%#v]\n", codes)
  31. var code []int = make([]int, len(codes))
  32. for idx, c := range codes {
  33. var err error
  34. code[idx], err = strconv.Atoi(c)
  35. if err != nil {
  36. log.Printf("Atoi: %#v [%s]\n", err, c)
  37. }
  38. color_codes = append(color_codes, code)
  39. }
  40. }
  41. return color_codes
  42. }
  43. type ANSIColorParts struct {
  44. Bold bool
  45. Blink bool
  46. FG int
  47. BG int
  48. }
  49. // Parse an array of codes into their ANSIColorParts.
  50. func ParseColorArray(colorarray []int) ANSIColorParts {
  51. var acp ANSIColorParts
  52. acp.FG = -1
  53. acp.BG = -1
  54. for _, c := range colorarray {
  55. switch c {
  56. case 0:
  57. acp.FG = 7
  58. acp.BG = 0
  59. case 1:
  60. acp.Bold = true
  61. case 5:
  62. acp.Blink = true
  63. case 30, 31, 32, 33, 34, 35, 36, 37:
  64. acp.FG = c - 30
  65. case 40, 41, 42, 43, 44, 45, 46, 47:
  66. acp.BG = c - 40
  67. }
  68. }
  69. return acp
  70. }
  71. // Update the LastColor with the latest codes.
  72. func (d *Door) UpdateLastColor(output string, LastColor *[]int) {
  73. // use the last color information + whatever is in the string to
  74. // track the last color set
  75. var updated ANSIColorParts = ParseColorArray(*LastColor)
  76. var colors [][]int = find_ansicolor(output)
  77. for _, codes := range colors {
  78. if codes[0] == 0 {
  79. updated = ParseColorArray(codes)
  80. } else {
  81. newCode := ParseColorArray(codes)
  82. if newCode.Bold {
  83. updated.Bold = true
  84. }
  85. if newCode.Blink {
  86. updated.Blink = true
  87. }
  88. if (newCode.FG != -1) && (newCode.FG != updated.FG) {
  89. updated.FG = newCode.FG
  90. }
  91. if (newCode.BG != -1) && (newCode.BG != updated.BG) {
  92. updated.BG = newCode.BG
  93. }
  94. }
  95. }
  96. *LastColor = make([]int, 1)
  97. (*LastColor)[0] = 0
  98. if updated.Blink {
  99. *LastColor = append(*LastColor, 5)
  100. }
  101. if updated.Bold {
  102. *LastColor = append(*LastColor, 1)
  103. }
  104. if updated.FG != -1 {
  105. *LastColor = append(*LastColor, updated.FG+30)
  106. }
  107. if updated.BG != -1 {
  108. *LastColor = append(*LastColor, updated.BG+40)
  109. }
  110. }
  111. /*
  112. reading from a closed channel is easy to detect.
  113. res, ok := <-channel
  114. ok == false
  115. writing to a closed channel is a panic.
  116. Have the Write manage itself.
  117. */
  118. // For now, update is SavePos + output + RestorePos.
  119. // FUTURE: Access the screen struct to "save" Color+Attr+Pos.
  120. func (d *Door) Update(output []byte) {
  121. // sync.Pool ?
  122. var buffer bytes.Buffer
  123. buffer.WriteString(SAVE_POS)
  124. buffer.Write(output)
  125. buffer.WriteString(RESTORE_POS)
  126. d.Writer.Write(buffer.Bytes())
  127. // d.Write(SAVE_POS + output + RESTORE_POS)
  128. }
  129. func (d *Door) WriteS(output string) {
  130. d.Write([]byte(output))
  131. }
  132. func (d *Door) WriteA(a ...interface{}) {
  133. d.Writer.Mutex.Lock()
  134. defer d.Writer.Mutex.Unlock()
  135. for _, item := range a {
  136. switch item.(type) {
  137. case string:
  138. d.Writer.OSWrite([]byte(item.(string)))
  139. case []byte:
  140. d.Writer.OSWrite(item.([]byte))
  141. case *bytes.Buffer:
  142. d.Writer.OSWrite(item.(*bytes.Buffer).Bytes())
  143. default:
  144. log.Printf("Unknown/unsupported type: %T\n", item)
  145. }
  146. }
  147. }
  148. func (d *Door) Write(output []byte) {
  149. if len(output) == 0 {
  150. return
  151. }
  152. d.Writer.Write(output)
  153. /*
  154. d.
  155. d.writerMutex.Lock()
  156. defer d.writerMutex.Unlock()
  157. if d.WriterClosed {
  158. return
  159. }
  160. if strings.HasSuffix(output, RestorePos) {
  161. output += Color(d.LastColor)
  162. } else {
  163. d.UpdateLastColor(output, &d.LastColor)
  164. }
  165. d.OSWrite([]byte(output))
  166. */
  167. }
  168. // This uses go routine Writer. (Deprecated)
  169. // Write string to client.
  170. /*
  171. func (d *Door) WriteCh(output string) {
  172. if output == "" {
  173. // That was easy.
  174. return
  175. }
  176. defer func() {
  177. if err := recover(); err != nil {
  178. log.Println("Write failed.", err)
  179. }
  180. }()
  181. d.writerChannel <- output
  182. }
  183. */