panel.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. package door
  2. import (
  3. "bytes"
  4. "log"
  5. "strings"
  6. )
  7. type BorderStyle int
  8. const (
  9. NONE BorderStyle = iota
  10. SINGLE
  11. DOUBLE
  12. DOUBLE_SINGLE
  13. SINGLE_DOUBLE
  14. )
  15. type Panel struct {
  16. X int
  17. Y int
  18. Width int
  19. Style BorderStyle
  20. BorderColor string
  21. Lines []*Line
  22. Title string
  23. TitleColor string
  24. TitleOffset int
  25. output *bytes.Buffer
  26. }
  27. /*
  28. func (p *Panel) Range(withBorder bool) (sx, sy, ex, ey int) {
  29. sx = p.X
  30. sy = p.Y
  31. ex = p.X + p.Width
  32. ey = p.Y + len(p.Lines)
  33. if p.Style > 0 {
  34. if withBorder {
  35. ex += 2
  36. ey += 2
  37. } else {
  38. sx += 1
  39. sy += 1
  40. }
  41. }
  42. return
  43. }
  44. */
  45. func (p *Panel) Length() int {
  46. return len(p.Lines)
  47. }
  48. func (p *Panel) RightBottomPos() (rx, by int) {
  49. rx = p.X + p.Width
  50. by = p.Y + len(p.Lines)
  51. if p.HasBorder() {
  52. rx += 1
  53. by += 1
  54. }
  55. return
  56. }
  57. func (p *Panel) HasBorder() bool {
  58. return int(p.Style) > 0
  59. }
  60. /*
  61. Is X,Y within the Panel?
  62. */
  63. func (p *Panel) Within(x, y int) (bool, int, int) {
  64. if x < p.X || y < p.Y {
  65. return false, 0, 0
  66. }
  67. mx, my := p.RightBottomPos()
  68. if x > mx || y > my {
  69. return false, 0, 0
  70. }
  71. // Ok, it must be within:
  72. return true, x - p.X, y - p.Y
  73. }
  74. /*
  75. // Panel Clicked?
  76. // If onBorder == true, returns true, 0 for border click.
  77. // Otherwise true, line clicked on (starting with 1)/
  78. func (p *Panel) Clicked(m Mouse, onBorder bool) (bool, int) {
  79. var sx, sy, ex, ey = p.Range(onBorder)
  80. var mx = int(m.X)
  81. var my = int(m.Y)
  82. if (mx >= sx) && (mx <= ex) {
  83. if (my >= sy) && (my <= ey) {
  84. if onBorder {
  85. if my == sy || my == ey {
  86. return true, 0
  87. }
  88. return true, my - sy
  89. } else {
  90. return true, (my - sy) + 1
  91. }
  92. }
  93. }
  94. return false, 0
  95. }
  96. */
  97. // Clear out the panel
  98. func (p *Panel) Clear() []byte {
  99. var style int = int(p.Style)
  100. if p.output == nil {
  101. p.output = &bytes.Buffer{}
  102. }
  103. p.output.Reset()
  104. var row int = p.Y
  105. var blank []byte
  106. if p.HasBorder() {
  107. blank = bytes.Repeat([]byte{' '}, p.Width+2)
  108. p.output.Write(Goto(p.X, row))
  109. p.output.Write(blank)
  110. row++
  111. } else {
  112. blank = bytes.Repeat([]byte{' '}, p.Width)
  113. }
  114. for _ = range p.Lines {
  115. p.output.Write(Goto(p.X, row))
  116. p.output.Write(blank)
  117. row++
  118. }
  119. if style > 0 {
  120. p.output.Write(Goto(p.X, row))
  121. p.output.Write(blank)
  122. }
  123. return p.output.Bytes()
  124. }
  125. // Output the panel
  126. func (p *Panel) Output() []byte {
  127. var style int = int(p.Style)
  128. var box_style *BoxStyle
  129. if p.output == nil {
  130. p.output = &bytes.Buffer{}
  131. }
  132. p.output.Reset()
  133. if p.HasBorder() {
  134. box_style = &BOXES[style-1]
  135. }
  136. var row int = p.Y
  137. if p.HasBorder() {
  138. // Top line / border
  139. // p.output.Write(Goto(p.X, row))
  140. GotoW(p.X, row, p.output)
  141. p.output.WriteString(p.BorderColor)
  142. p.output.WriteString(box_style.Top_Left)
  143. if p.Title != "" {
  144. if p.TitleOffset+len(p.Title) > p.Width {
  145. log.Panicf("Panel (not wide enough) Width %d : Title size %d + offset %d = %d\n",
  146. p.Width, len(p.Title), p.TitleOffset, p.TitleOffset+len(p.Title))
  147. }
  148. p.output.WriteString(strings.Repeat(box_style.Top, p.TitleOffset))
  149. p.output.WriteString(p.TitleColor + p.Title + p.BorderColor)
  150. }
  151. p.output.WriteString(strings.Repeat(box_style.Top, p.Width-(p.TitleOffset+len(p.Title))) + box_style.Top_Right)
  152. row++
  153. }
  154. for _, line := range p.Lines {
  155. // p.output.Write(Goto(p.X, row))
  156. GotoW(p.X, row, p.output)
  157. line.LineLength(line.Text)
  158. var joined bool = false
  159. if p.HasBorder() {
  160. top := box_style.Top
  161. // This only works in non-unicode mode
  162. if bytes.Compare(line.Text.Bytes()[0:len(top)], []byte(top)) == 0 {
  163. // Yes, this line needs to be joined...
  164. p.output.WriteString(p.BorderColor + box_style.Middle_Left)
  165. p.output.Write(line.Output())
  166. p.output.WriteString(p.BorderColor + box_style.Middle_Right)
  167. joined = true
  168. }
  169. }
  170. if !joined {
  171. if p.HasBorder() {
  172. p.output.WriteString(p.BorderColor)
  173. p.output.WriteString(box_style.Side)
  174. }
  175. p.output.Write(line.Output())
  176. if style > 0 {
  177. p.output.WriteString(p.BorderColor)
  178. p.output.WriteString(box_style.Side)
  179. }
  180. }
  181. row++
  182. }
  183. if p.HasBorder() {
  184. // Bottom / border
  185. // p.output.Write(Goto(p.X, row))
  186. GotoW(p.X, row, p.output)
  187. p.output.WriteString(p.BorderColor)
  188. p.output.WriteString(box_style.Bottom_Left)
  189. p.output.WriteString(strings.Repeat(box_style.Top, p.Width) + box_style.Bottom_Right)
  190. }
  191. return p.output.Bytes()
  192. }
  193. // Output anything that has updated
  194. func (p *Panel) Update() []byte {
  195. if p.output == nil {
  196. p.output = &bytes.Buffer{}
  197. }
  198. p.output.Reset()
  199. var style int = int(p.Style)
  200. var row, col int
  201. row = p.Y
  202. col = p.X
  203. if style > 0 {
  204. row++
  205. col++
  206. }
  207. for idx := range p.Lines {
  208. if p.Lines[idx].Update() {
  209. // Yes, line was updated
  210. p.output.Write(Goto(col, row))
  211. p.output.Write(p.Lines[idx].Output())
  212. }
  213. row++
  214. }
  215. return p.output.Bytes()
  216. }
  217. // Output the updated line
  218. func (p *Panel) UpdateLine(index int) []byte {
  219. if p.output == nil {
  220. p.output = &bytes.Buffer{}
  221. }
  222. p.output.Reset()
  223. // var style int = int(p.Style)
  224. p.Lines[index].Update()
  225. var row, col int
  226. row = p.Y + index
  227. col = p.X
  228. if p.HasBorder() {
  229. row++
  230. col++
  231. }
  232. var line *Line = p.Lines[index]
  233. p.output.Write(Goto(col, row))
  234. p.output.Write(line.Output())
  235. return p.output.Bytes()
  236. }
  237. // Position Cursor at the "end" of the panel
  238. func (p *Panel) GotoEnd() []byte {
  239. rx, by := p.RightBottomPos()
  240. return Goto(rx+1, by)
  241. }
  242. // Is the top line of this style a single line?
  243. func Single(bs BorderStyle) bool {
  244. switch bs {
  245. case SINGLE, SINGLE_DOUBLE:
  246. return true
  247. default:
  248. return false
  249. }
  250. }
  251. // Create a spacer line that will be connected maybe to the sides.
  252. func (p *Panel) Spacer() *Line {
  253. var l *Line = &Line{Text: &bytes.Buffer{}}
  254. var pos int
  255. if Single(p.Style) {
  256. pos = 0
  257. } else {
  258. pos = 1
  259. }
  260. l.Text.WriteString(strings.Repeat(BOXES[pos].Top, p.Width))
  261. return l
  262. }
  263. func (p *Panel) Center() {
  264. p.CenterX()
  265. p.CenterY()
  266. }
  267. func (p *Panel) CenterX() {
  268. p.X = (Width - p.Width) / 2
  269. }
  270. func (p *Panel) CenterY() {
  271. p.Y = (Height - len(p.Lines)) / 2
  272. }