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.

173 lines
4.1 KiB
Go

package tui
import (
"fmt"
"log"
"strconv"
"time"
"github.com/rivo/tview"
_ "github.com/gdamore/tcell/v2"
)
type Device struct {
Id uint32
Type string
Status string
Data string
Index uint32
}
type TUI struct {
*Display
//*LocalView
*TUIClient
SelectedReactor <-chan uint32
Err chan error
}
func NewTUI(ip string, port int, ch chan error) *TUI {
//r := make(map[uint32]*Reactor)
t := &TUI{}
//l := new(LocalView)
//l.Reactors = r
//t.LocalView = l
t.Err = ch
client := NewTUIClient(ip, port)
t.TUIClient = client
return t
}
func (t *TUI) Start() {
// setup tview app and wait for user connection in standin modal
if err := t.TUIClient.Start(); err != nil {
t.Err <- err
}
go t.Monitor()
t.CreateDisplay()
t.Display.Start()
//go t.Refresh()
}
func (t *TUI) CreateDisplay() {
c := make(chan uint32)
t.Display = NewDisplay(c)
t.SelectedReactor = c
t.Flex.AddItem(t.ReactorList,0,1,true).
AddItem(t.DevicePages,0,2,false)
}
func (t *TUI) Monitor() {
// orchestrates updates and grpc requests
timer := make(chan struct{})
go func(signal chan struct{}){
for {
signal <- struct{}{}
time.Sleep(1 * time.Second)
}
}(timer)
for {
select {
//case reactor := <-t.SelectedReactor:
// reactor has been selected in tui, grabbing devs
//t.App.QueueUpdateDraw(func() {
// t.UpdateDevices(reactor)
//})
//devs := t.GetReactorDevices(reactor)
//t.DisplayReactorDevices(devs)
case <-timer:
// time to ping for status
t.App.QueueUpdateDraw(func() {
t.UpdateDevices()
})
}
}
}
func (t *TUI) UpdateDevices(r ...uint32) {
// get devices for the reactor and update the tui
var id uint32
if len(r) > 0 {
id = r[0]
}
devs, err := t.TUIClient.GetDevices(id)
if err != nil {
log.Fatal(err)
}
if len(r) > 0 {
// reactor specificed
r := make(map[uint32]*Device)
d := make(map[uint32]*Device)
for id, dev := range devs {
if dev.Type == "Reactor" {
r[id] = dev
} else {
d[id] = dev
}
}
t.DisplayDevices(d)
t.DisplayReactors(r)
} else {
t.DisplayReactors(devs)
}
}
// display struct and logic
type Display struct {
App *tview.Application
Flex *tview.Flex
ReactorList *tview.List
DevicePages *tview.Pages
SelectedReactor chan<- uint32
}
func NewDisplay(ch chan uint32) *Display {
d := &Display{}
d.App = tview.NewApplication()
d.Flex = tview.NewFlex()
d.ReactorList = tview.NewList().ShowSecondaryText(false)
d.DevicePages = tview.NewPages()
d.ReactorList.SetTitle("Reactors").SetBorder(true)
d.DevicePages.SetTitle("Devices").SetBorder(true)
d.SelectedReactor = ch
return d
}
func (d *Display) Start() {
if err := d.App.SetRoot(d.Flex, true).Run(); err != nil {
d.App.Stop()
log.Fatal(err)
}
}
func (d *Display) DisplayDevices(devs map[uint32]*Device, rid ...uint32) {
for _, dev := range devs {
if dev.Type == "Reactor"{
//d.ReactorList.InsertItem(int(dev.Ind
} else if len(rid) > 0 {
//
fmt.Println("selected")
}
}
}
func (d *Display) DisplayReactors(r map[uint32]*Device) {
// function to display reactor list to table
for id, reactor := range r {
txt := fmt.Sprintf("%v %v%v", reactor.Id, reactor.Status,reactor.Data)
if d.ReactorList.GetItemCount() > int(reactor.Index) {
d.ReactorList.RemoveItem(int(reactor.Index))
}
d.ReactorList.InsertItem(int(reactor.Index),txt,string(id),rune(49+reactor.Index),nil)
}
}
func (d *Display) SelectReactor(index int, main, id string, r rune) {
// called when reactor in list in selected
if id, err := strconv.Atoi(id); err != nil {
log.Fatal(err)
} else {
d.SelectedReactor <-uint32(id)
}
}