Initial pass on new frontend design

This commit is contained in:
Drew DeVault 2018-02-16 00:05:07 -05:00
parent 18493180bd
commit 8c8c21f3ff
3 changed files with 177 additions and 0 deletions

95
ui/context.go Normal file
View 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
View 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
View 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)
}
}