You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tson/gui/gui.go

265 lines
5.8 KiB
Go

package gui
import (
"bytes"
"encoding/json"
"fmt"
"io/ioutil"
"log"
"os"
"strconv"
"strings"
"github.com/gdamore/tcell"
"github.com/rivo/tview"
)
type Gui struct {
Tree *Tree
App *tview.Application
Pages *tview.Pages
}
func New() *Gui {
g := &Gui{
Tree: NewTree(),
App: tview.NewApplication(),
Pages: tview.NewPages(),
}
return g
}
func (g *Gui) Run(i interface{}) error {
g.Tree.UpdateView(g, i)
g.Tree.SetKeybindings(g)
grid := tview.NewGrid().
AddItem(g.Tree, 0, 0, 1, 1, 0, 0, true)
g.Pages.AddAndSwitchToPage("main", grid, true)
if err := g.App.SetRoot(g.Pages, true).Run(); err != nil {
log.Println(err)
return err
}
return nil
}
func (g *Gui) Modal(p tview.Primitive, width, height int) tview.Primitive {
return tview.NewGrid().
SetColumns(0, width, 0).
SetRows(0, height, 0).
AddItem(p, 1, 1, 1, 1, 0, 0, true)
}
func (g *Gui) Message(message, page string, doneFunc func()) {
doneLabel := "ok"
modal := tview.NewModal().
SetText(message).
AddButtons([]string{doneLabel}).
SetDoneFunc(func(buttonIndex int, buttonLabel string) {
g.Pages.RemovePage("message")
g.Pages.SwitchToPage(page).ShowPage("main")
if buttonLabel == doneLabel {
doneFunc()
}
})
g.Pages.AddAndSwitchToPage("message", g.Modal(modal, 80, 29), true).ShowPage("main")
}
func (g *Gui) Input(text, label string, width int, doneFunc func(text string)) {
input := tview.NewInputField().SetText(text)
input.SetBorder(true)
input.SetLabel(label).SetLabelWidth(width).SetDoneFunc(func(key tcell.Key) {
if key == tcell.KeyEnter {
doneFunc(input.GetText())
g.Pages.RemovePage("input")
}
})
g.Pages.AddAndSwitchToPage("input", g.Modal(input, 0, 3), true).ShowPage("main")
}
func (g *Gui) Form(fieldLabel []string, doneLabel, title, pageName string,
height int, doneFunc func(values map[string]string) error) {
if g.Pages.HasPage(pageName) {
g.Pages.ShowPage(pageName)
return
}
form := tview.NewForm()
for _, label := range fieldLabel {
form.AddInputField(label, "", 0, nil, nil)
}
form.AddButton(doneLabel, func() {
values := make(map[string]string)
for _, label := range fieldLabel {
item := form.GetFormItemByLabel(label)
switch item.(type) {
case *tview.InputField:
input, ok := item.(*tview.InputField)
if ok {
values[label] = os.ExpandEnv(input.GetText())
}
}
}
if err := doneFunc(values); err != nil {
g.Message(err.Error(), pageName, func() {})
return
}
g.Pages.RemovePage(pageName)
}).
AddButton("cancel", func() {
g.Pages.RemovePage(pageName)
})
form.SetBorder(true).SetTitle(title).
SetTitleAlign(tview.AlignLeft)
g.Pages.AddAndSwitchToPage(pageName, g.Modal(form, 0, height), true).ShowPage("main")
}
func (g *Gui) LoadJSON() {
labels := []string{"file"}
g.Form(labels, "read", "read from file", "read_from_file", 7, func(values map[string]string) error {
file := values[labels[0]]
b, err := ioutil.ReadFile(file)
if err != nil {
log.Println(fmt.Sprintf("can't read file: %s", err))
return err
}
var i interface{}
if err := json.Unmarshal(b, &i); err != nil {
log.Println(fmt.Sprintf("can't unmarshal json: %s", err))
return err
}
g.Tree.UpdateView(g, i)
return nil
})
}
func (g *Gui) Search() {
pageName := "search"
if g.Pages.HasPage(pageName) {
g.Pages.ShowPage(pageName)
} else {
input := tview.NewInputField()
input.SetBorder(true).SetTitle("search").SetTitleAlign(tview.AlignLeft)
input.SetChangedFunc(func(text string) {
root := *g.Tree.OriginRoot
g.Tree.SetRoot(&root)
if text != "" {
root := g.Tree.GetRoot()
root.SetChildren(g.walk(root.GetChildren(), text))
}
})
input.SetLabel("word").SetLabelWidth(5).SetDoneFunc(func(key tcell.Key) {
if key == tcell.KeyEnter {
g.Pages.HidePage(pageName)
}
})
g.Pages.AddAndSwitchToPage(pageName, g.Modal(input, 0, 3), true).ShowPage("main")
}
}
func (g *Gui) walk(nodes []*tview.TreeNode, text string) []*tview.TreeNode {
var newNodes []*tview.TreeNode
for _, child := range nodes {
log.Println(child.GetText())
if strings.Index(strings.ToLower(child.GetText()), text) != -1 {
newNodes = append(newNodes, child)
} else {
newNodes = append(newNodes, g.walk(child.GetChildren(), text)...)
}
}
return newNodes
}
func (g *Gui) SaveJSON() {
labels := []string{"file"}
g.Form(labels, "save", "save to file", "save_to_file", 7, func(values map[string]string) error {
file := values[labels[0]]
file = os.ExpandEnv(file)
var buf bytes.Buffer
enc := json.NewEncoder(&buf)
enc.SetIndent("", " ")
if err := enc.Encode(g.makeJSON(g.Tree.GetRoot())); err != nil {
log.Println(fmt.Sprintf("can't marshal json: %s", err))
return err
}
if err := ioutil.WriteFile(file, buf.Bytes(), 0666); err != nil {
log.Println(fmt.Sprintf("can't create file: %s", err))
return err
}
return nil
})
}
func (g *Gui) makeJSON(node *tview.TreeNode) interface{} {
ref := node.GetReference().(Reference)
children := node.GetChildren()
switch ref.JSONType {
case Object:
i := make(map[string]interface{})
for _, n := range children {
i[n.GetText()] = g.makeJSON(n)
}
return i
case Array:
var i []interface{}
for _, n := range children {
i = append(i, g.makeJSON(n))
}
return i
case Key:
v := node.GetChildren()[0]
if v.GetReference().(Reference).JSONType == Value {
return g.parseValue(v)
}
return map[string]interface{}{
node.GetText(): g.makeJSON(v),
}
}
return g.parseValue(node)
}
func (g *Gui) parseValue(node *tview.TreeNode) interface{} {
v := node.GetText()
ref := node.GetReference().(Reference)
switch ref.ValueType {
case Int:
i, _ := strconv.Atoi(v)
return i
case Float:
f, _ := strconv.ParseFloat(v, 64)
return f
case Boolean:
b, _ := strconv.ParseBool(v)
return b
case Null:
return nil
}
return v
}