termui-tree/tree.go
2019-05-19 19:27:00 -04:00

147 lines
3.7 KiB
Go

package main
import (
"image"
"github.com/gizak/termui/v3"
)
const indentWidth = 4
// TreeWidget is a renderable tree
type TreeWidget struct {
termui.Block
rootNodes []*Node
selectedNode *Node
}
// Node represents an item in the tree widget
type Node struct {
Text string
children []*Node
tree *TreeWidget
parent *Node
next *Node
prev *Node
}
// NewTreeWidget makes a new tree widget
func NewTreeWidget(rootNodes []*Node) *TreeWidget {
tree := &TreeWidget{
Block: *termui.NewBlock(),
}
tree.selectedNode = rootNodes[0]
for _, rootNode := range rootNodes {
tree.AddRootNode(rootNode)
}
return tree
}
// Draw draws the tree widget
func (tree *TreeWidget) Draw(buffer *termui.Buffer) {
tree.Block.Draw(buffer)
for _, item := range tree.rootNodes {
tree.renderSingleTree(item, buffer)
}
}
// AddRootNode adds a new top-level node to the tree
func (tree *TreeWidget) AddRootNode(node *Node) {
tree.rootNodes = append(tree.rootNodes, node)
if len(tree.rootNodes) == 1 {
tree.selectedNode = node.children[0]
}
node.setTree(tree)
}
// SelectNext marks the next node as the currently selected node
func (tree *TreeWidget) SelectNext() {
if len(tree.selectedNode.children) > 0 {
tree.selectedNode = tree.selectedNode.children[0]
} else if tree.selectedNode.next != nil {
tree.selectedNode = tree.selectedNode.next
} else if tree.selectedNode.next == nil && tree.selectedNode.parent != nil && tree.selectedNode.parent.next != nil {
// Go up to the next adjacent node to a parent
tree.selectedNode = tree.selectedNode.parent.next
}
}
func (tree *TreeWidget) renderSingleTree(root *Node, buffer *termui.Buffer) {
baseWidth := tree.Inner.Max.X - tree.Inner.Min.X
depths := map[*Node]int{root: 0}
toVisit := []*Node{root}
row := tree.Inner.Min.Y
for len(toVisit) > 0 {
var node *Node
toVisit, node = toVisit[:len(toVisit)-1], toVisit[len(toVisit)-1]
currentDepth := depths[node]
// Add the children to explore next in reverse order.
// The next off the stack should be the direct descendent of the parent.
for i := len(node.children) - 1; i >= 0; i-- {
child := node.children[i]
toVisit = append(toVisit, child)
depths[child] = currentDepth + 1
}
// Render this level of tree
width := baseWidth - currentDepth*indentWidth
startingCol := tree.Inner.Min.X + currentDepth*indentWidth
row += node.render(buffer, row, startingCol, width)
}
}
// render renders this node in the term ui, and return the number of rows that were required to render it
func (node *Node) render(buffer *termui.Buffer, row, col, width int) int {
style := termui.Style{
Fg: termui.ColorRed,
Bg: termui.ColorClear,
}
if node.tree.selectedNode == node {
style.Fg, style.Bg = style.Bg, style.Fg
} else if node.tree.selectedNode.next == node {
style.Bg = termui.ColorBlue
}
listItem := termui.ParseStyles(node.Text, style)
wrappedItem := termui.WrapCells(listItem, uint(width))
x := col
rowOffset := 0
for _, cell := range wrappedItem {
// If we're wrapped, we don't need to render anything, just go to the next line
if cell.Rune == '\n' {
rowOffset++
x = col
} else {
buffer.SetCell(cell, image.Point{x, row + rowOffset})
x++
}
}
return rowOffset + 1
}
// AddChild marks the given node as a child, and itself as the child node's parent
func (node *Node) AddChild(child *Node) {
child.parent = node
if node.children == nil {
node.children = []*Node{child}
} else {
node.children = append(node.children, child)
}
child.tree = node.tree
if len(node.children) > 1 {
lastNode := node.children[len(node.children)-2]
lastNode.next = child
child.prev = lastNode
}
}
func (node *Node) setTree(tree *TreeWidget) {
node.tree = tree
for _, child := range node.children {
child.setTree(tree)
}
}