Initial pass on new frontend design
This commit is contained in:
parent
18493180bd
commit
8c8c21f3ff
3 changed files with 177 additions and 0 deletions
95
ui/context.go
Normal file
95
ui/context.go
Normal file
|
@ -0,0 +1,95 @@
|
|||
package ui
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
|
||||
"github.com/nsf/termbox-go"
|
||||
)
|
||||
|
||||
// A context allows you to draw in a sub-region of the terminal
|
||||
type Context struct {
|
||||
x int
|
||||
y int
|
||||
width int
|
||||
height int
|
||||
}
|
||||
|
||||
func (ctx *Context) Width() int {
|
||||
return ctx.width
|
||||
}
|
||||
|
||||
func (ctx *Context) Height() int {
|
||||
return ctx.height
|
||||
}
|
||||
|
||||
func NewContext() *Context {
|
||||
width, height := termbox.Size()
|
||||
return &Context{0, 0, width, height}
|
||||
}
|
||||
|
||||
func (ctx *Context) Subcontext(x, y, width, height int) *Context {
|
||||
if x+width > ctx.width || y+height > ctx.height {
|
||||
panic(fmt.Errorf("Attempted to create context larger than parent"))
|
||||
}
|
||||
return &Context{
|
||||
x: ctx.x + x,
|
||||
y: ctx.y + y,
|
||||
width: width,
|
||||
height: height,
|
||||
}
|
||||
}
|
||||
|
||||
func (ctx *Context) SetCell(x, y int, ch rune, fg, bg termbox.Attribute) {
|
||||
if x >= ctx.width || y >= ctx.height {
|
||||
panic(fmt.Errorf("Attempted to draw outside of context"))
|
||||
}
|
||||
termbox.SetCell(ctx.x+x, ctx.y+y, ch, fg, bg)
|
||||
}
|
||||
|
||||
func (ctx *Context) Printf(x, y int, ref termbox.Cell,
|
||||
format string, a ...interface{}) {
|
||||
|
||||
if x >= ctx.width || y >= ctx.height {
|
||||
panic(fmt.Errorf("Attempted to draw outside of context"))
|
||||
}
|
||||
|
||||
str := fmt.Sprintf(format, a...)
|
||||
|
||||
x += ctx.x
|
||||
y += ctx.y
|
||||
old_x := x
|
||||
|
||||
newline := func() bool {
|
||||
x = old_x
|
||||
y++
|
||||
return y < ctx.height
|
||||
}
|
||||
for _, ch := range str {
|
||||
switch ch {
|
||||
case '\n':
|
||||
if !newline() {
|
||||
return
|
||||
}
|
||||
case '\r':
|
||||
x = old_x
|
||||
default:
|
||||
termbox.SetCell(x, y, ch, ref.Fg, ref.Bg)
|
||||
x++
|
||||
if x == old_x+ctx.width {
|
||||
if !newline() {
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (ctx *Context) Fill(x, y, width, height int, ref termbox.Cell) {
|
||||
_x := x
|
||||
for ; y < height && y < ctx.height; y++ {
|
||||
for ; x < width && x < ctx.width; x++ {
|
||||
ctx.SetCell(x, y, ref.Ch, ref.Fg, ref.Bg)
|
||||
}
|
||||
x = _x
|
||||
}
|
||||
}
|
8
ui/drawable.go
Normal file
8
ui/drawable.go
Normal file
|
@ -0,0 +1,8 @@
|
|||
package ui
|
||||
|
||||
type Drawable interface {
|
||||
// Called when this renderable should draw itself
|
||||
Draw(ctx Context)
|
||||
// Specifies a function to call when this cell needs to be redrawn
|
||||
OnInvalidate(callback func(d Drawable))
|
||||
}
|
74
ui/grid.go
Normal file
74
ui/grid.go
Normal file
|
@ -0,0 +1,74 @@
|
|||
package ui
|
||||
|
||||
import "fmt"
|
||||
|
||||
type Grid struct {
|
||||
Rows []DimSpec
|
||||
Columns []DimSpec
|
||||
Cells []*GridCell
|
||||
onInvalidate func(d Drawable)
|
||||
}
|
||||
|
||||
const (
|
||||
SIZE_EXACT = iota
|
||||
SIZE_WEIGHT = iota
|
||||
)
|
||||
|
||||
// Specifies the layout of a single row or column
|
||||
type DimSpec struct {
|
||||
// One of SIZE_EXACT or SIZE_WEIGHT
|
||||
Strategy uint
|
||||
// If Strategy = SIZE_EXACT, this is the number of cells this dim shall
|
||||
// occupy. If SIZE_WEIGHT, the space left after all exact dims are measured
|
||||
// is distributed amonst the remaining dims weighted by this value.
|
||||
Size *uint
|
||||
}
|
||||
|
||||
type GridCell struct {
|
||||
Row uint
|
||||
Column uint
|
||||
RowSpan uint
|
||||
ColSpan uint
|
||||
Content Drawable
|
||||
invalid bool
|
||||
}
|
||||
|
||||
func (grid *Grid) Draw(ctx Context) {
|
||||
// TODO
|
||||
}
|
||||
|
||||
func (grid *Grid) OnInvalidate(onInvalidate func(d Drawable)) {
|
||||
grid.onInvalidate = onInvalidate
|
||||
}
|
||||
|
||||
func (grid *Grid) AddChild(cell *GridCell) {
|
||||
grid.Cells = append(grid.Cells, cell)
|
||||
cell.Content.OnInvalidate(grid.cellInvalidated)
|
||||
cell.invalid = true
|
||||
}
|
||||
|
||||
func (grid *Grid) RemoveChild(cell *GridCell) {
|
||||
for i, _cell := range grid.Cells {
|
||||
if _cell == cell {
|
||||
grid.Cells = append(grid.Cells[:i], grid.Cells[i+1:]...)
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (grid *Grid) cellInvalidated(drawable Drawable) {
|
||||
var cell *GridCell
|
||||
for _, cell = range grid.Cells {
|
||||
if cell.Content == drawable {
|
||||
break
|
||||
}
|
||||
cell = nil
|
||||
}
|
||||
if cell == nil {
|
||||
panic(fmt.Errorf("Attempted to invalidate unknown cell"))
|
||||
}
|
||||
cell.invalid = true
|
||||
if grid.onInvalidate != nil {
|
||||
grid.onInvalidate(grid)
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue