2018-02-16 00:05:07 -05:00
|
|
|
package ui
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
2020-11-30 22:07:03 +00:00
|
|
|
"github.com/gdamore/tcell/v2"
|
|
|
|
"github.com/gdamore/tcell/v2/views"
|
2018-06-11 20:04:21 -04:00
|
|
|
"github.com/mattn/go-runewidth"
|
2018-02-16 00:05:07 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
// A context allows you to draw in a sub-region of the terminal
|
|
|
|
type Context struct {
|
2019-12-20 13:21:32 -05:00
|
|
|
screen tcell.Screen
|
|
|
|
viewport *views.ViewPort
|
|
|
|
x, y int
|
|
|
|
onPopover func(*Popover)
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
|
2018-02-26 22:41:54 -05:00
|
|
|
func (ctx *Context) X() int {
|
2018-06-01 09:58:00 +02:00
|
|
|
x, _, _, _ := ctx.viewport.GetPhysical()
|
|
|
|
return x
|
2018-02-26 22:41:54 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Context) Y() int {
|
2018-06-01 09:58:00 +02:00
|
|
|
_, y, _, _ := ctx.viewport.GetPhysical()
|
|
|
|
return y
|
2018-02-26 22:41:54 -05:00
|
|
|
}
|
|
|
|
|
2018-02-16 00:05:07 -05:00
|
|
|
func (ctx *Context) Width() int {
|
2018-06-01 09:58:00 +02:00
|
|
|
width, _ := ctx.viewport.Size()
|
|
|
|
return width
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Context) Height() int {
|
2018-06-01 09:58:00 +02:00
|
|
|
_, height := ctx.viewport.Size()
|
|
|
|
return height
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
|
2019-12-20 13:21:32 -05:00
|
|
|
func NewContext(width, height int, screen tcell.Screen, p func(*Popover)) *Context {
|
2018-06-01 09:58:00 +02:00
|
|
|
vp := views.NewViewPort(screen, 0, 0, width, height)
|
2019-12-20 13:21:32 -05:00
|
|
|
return &Context{screen, vp, 0, 0, p}
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Context) Subcontext(x, y, width, height int) *Context {
|
2018-06-01 09:58:00 +02:00
|
|
|
vp_width, vp_height := ctx.viewport.Size()
|
2018-06-11 20:04:21 -04:00
|
|
|
if x < 0 || y < 0 {
|
2018-06-01 09:58:00 +02:00
|
|
|
panic(fmt.Errorf("Attempted to create context with negative offset"))
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
2018-06-11 20:04:21 -04:00
|
|
|
if x+width > vp_width || y+height > vp_height {
|
2018-06-01 09:58:00 +02:00
|
|
|
panic(fmt.Errorf("Attempted to create context larger than parent"))
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
2018-06-01 09:58:00 +02:00
|
|
|
vp := views.NewViewPort(ctx.viewport, x, y, width, height)
|
2019-12-20 13:21:32 -05:00
|
|
|
return &Context{ctx.screen, vp, ctx.x + x, ctx.y + y, ctx.onPopover}
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
|
2018-06-01 09:58:00 +02:00
|
|
|
func (ctx *Context) SetCell(x, y int, ch rune, style tcell.Style) {
|
|
|
|
width, height := ctx.viewport.Size()
|
|
|
|
if x >= width || y >= height {
|
2018-02-16 00:05:07 -05:00
|
|
|
panic(fmt.Errorf("Attempted to draw outside of context"))
|
|
|
|
}
|
2018-06-01 09:58:00 +02:00
|
|
|
crunes := []rune{}
|
|
|
|
ctx.viewport.SetContent(x, y, ch, crunes, style)
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
|
2018-06-01 09:58:00 +02:00
|
|
|
func (ctx *Context) Printf(x, y int, style tcell.Style,
|
2018-02-17 19:42:29 -05:00
|
|
|
format string, a ...interface{}) int {
|
2018-06-01 09:58:00 +02:00
|
|
|
width, height := ctx.viewport.Size()
|
2018-02-16 00:05:07 -05:00
|
|
|
|
2018-06-01 09:58:00 +02:00
|
|
|
if x >= width || y >= height {
|
2018-02-16 00:05:07 -05:00
|
|
|
panic(fmt.Errorf("Attempted to draw outside of context"))
|
|
|
|
}
|
|
|
|
|
|
|
|
str := fmt.Sprintf(format, a...)
|
|
|
|
|
|
|
|
old_x := x
|
|
|
|
|
|
|
|
newline := func() bool {
|
|
|
|
x = old_x
|
|
|
|
y++
|
2018-06-01 09:58:00 +02:00
|
|
|
return y < height
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
for _, ch := range str {
|
|
|
|
switch ch {
|
|
|
|
case '\n':
|
|
|
|
if !newline() {
|
2018-02-17 19:42:29 -05:00
|
|
|
return runewidth.StringWidth(str)
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
case '\r':
|
|
|
|
x = old_x
|
|
|
|
default:
|
2018-06-01 09:58:00 +02:00
|
|
|
crunes := []rune{}
|
|
|
|
ctx.viewport.SetContent(x, y, ch, crunes, style)
|
2018-02-17 19:42:29 -05:00
|
|
|
x += runewidth.RuneWidth(ch)
|
2018-06-11 20:04:21 -04:00
|
|
|
if x == old_x+width {
|
2018-02-16 00:05:07 -05:00
|
|
|
if !newline() {
|
2018-02-17 19:42:29 -05:00
|
|
|
return runewidth.StringWidth(str)
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-02-17 19:42:29 -05:00
|
|
|
|
|
|
|
return runewidth.StringWidth(str)
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
|
|
|
|
2018-06-01 09:58:00 +02:00
|
|
|
func (ctx *Context) Fill(x, y, width, height int, rune rune, style tcell.Style) {
|
|
|
|
vp := views.NewViewPort(ctx.viewport, x, y, width, height)
|
|
|
|
vp.Fill(rune, style)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Context) SetCursor(x, y int) {
|
2019-01-14 08:07:24 -05:00
|
|
|
ctx.screen.ShowCursor(ctx.x+x, ctx.y+y)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Context) HideCursor() {
|
|
|
|
ctx.screen.HideCursor()
|
2018-02-16 00:05:07 -05:00
|
|
|
}
|
2019-12-20 13:21:32 -05:00
|
|
|
|
|
|
|
func (ctx *Context) Popover(x, y, width, height int, d Drawable) {
|
|
|
|
ctx.onPopover(&Popover{
|
|
|
|
x: ctx.x + x,
|
|
|
|
y: ctx.y + y,
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
content: d,
|
|
|
|
})
|
|
|
|
}
|