summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--ui/context.go95
-rw-r--r--ui/drawable.go8
-rw-r--r--ui/grid.go74
3 files changed, 177 insertions, 0 deletions
diff --git a/ui/context.go b/ui/context.go
new file mode 100644
index 0000000..9f2e2fe
--- /dev/null
+++ b/ui/context.go
@@ -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
+	}
+}
diff --git a/ui/drawable.go b/ui/drawable.go
new file mode 100644
index 0000000..eb60463
--- /dev/null
+++ b/ui/drawable.go
@@ -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))
+}
diff --git a/ui/grid.go b/ui/grid.go
new file mode 100644
index 0000000..fd0cab7
--- /dev/null
+++ b/ui/grid.go
@@ -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)
+	}
+}