mirror of
https://github.com/bcicen/ctop.git
synced 2024-08-30 18:23:19 +00:00
193 lines
3.5 KiB
Go
193 lines
3.5 KiB
Go
package main
|
|
|
|
import (
|
|
"math"
|
|
|
|
"github.com/bcicen/ctop/connector"
|
|
"github.com/bcicen/ctop/container"
|
|
ui "github.com/gizak/termui"
|
|
)
|
|
|
|
type GridCursor struct {
|
|
selectedID string // id of currently selected container
|
|
filtered container.Containers
|
|
cSource connector.Connector
|
|
isScrolling bool // toggled when actively scrolling
|
|
}
|
|
|
|
func (gc *GridCursor) Len() int { return len(gc.filtered) }
|
|
|
|
func (gc *GridCursor) Selected() *container.Container {
|
|
idx := gc.Idx()
|
|
if idx < gc.Len() {
|
|
return gc.filtered[idx]
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Refresh containers from source
|
|
func (gc *GridCursor) RefreshContainers() (lenChanged bool) {
|
|
oldLen := gc.Len()
|
|
|
|
// Containers filtered by display bool
|
|
gc.filtered = container.Containers{}
|
|
var cursorVisible bool
|
|
for _, c := range gc.cSource.All() {
|
|
if c.Display {
|
|
if c.Id == gc.selectedID {
|
|
cursorVisible = true
|
|
}
|
|
gc.filtered = append(gc.filtered, c)
|
|
}
|
|
}
|
|
|
|
if oldLen != gc.Len() {
|
|
lenChanged = true
|
|
}
|
|
|
|
if !cursorVisible {
|
|
gc.Reset()
|
|
}
|
|
if gc.selectedID == "" {
|
|
gc.Reset()
|
|
}
|
|
return lenChanged
|
|
}
|
|
|
|
// Set an initial cursor position, if possible
|
|
func (gc *GridCursor) Reset() {
|
|
for _, c := range gc.cSource.All() {
|
|
c.Widgets.UnHighlight()
|
|
}
|
|
if gc.Len() > 0 {
|
|
gc.selectedID = gc.filtered[0].Id
|
|
gc.filtered[0].Widgets.Highlight()
|
|
}
|
|
}
|
|
|
|
// Return current cursor index
|
|
func (gc *GridCursor) Idx() int {
|
|
for n, c := range gc.filtered {
|
|
if c.Id == gc.selectedID {
|
|
return n
|
|
}
|
|
}
|
|
gc.Reset()
|
|
return 0
|
|
}
|
|
|
|
func (gc *GridCursor) ScrollPage() {
|
|
// skip scroll if no need to page
|
|
if gc.Len() < cGrid.MaxRows() {
|
|
cGrid.Offset = 0
|
|
return
|
|
}
|
|
|
|
idx := gc.Idx()
|
|
|
|
// page down
|
|
if idx >= cGrid.Offset+cGrid.MaxRows() {
|
|
cGrid.Offset++
|
|
cGrid.Align()
|
|
}
|
|
// page up
|
|
if idx < cGrid.Offset {
|
|
cGrid.Offset--
|
|
cGrid.Align()
|
|
}
|
|
|
|
}
|
|
|
|
func (gc *GridCursor) Up() {
|
|
gc.isScrolling = true
|
|
defer func() { gc.isScrolling = false }()
|
|
|
|
idx := gc.Idx()
|
|
if idx <= 0 { // already at top
|
|
return
|
|
}
|
|
active := gc.filtered[idx]
|
|
next := gc.filtered[idx-1]
|
|
|
|
active.Widgets.UnHighlight()
|
|
gc.selectedID = next.Id
|
|
next.Widgets.Highlight()
|
|
|
|
gc.ScrollPage()
|
|
ui.Render(cGrid)
|
|
}
|
|
|
|
func (gc *GridCursor) Down() {
|
|
gc.isScrolling = true
|
|
defer func() { gc.isScrolling = false }()
|
|
|
|
idx := gc.Idx()
|
|
if idx >= gc.Len()-1 { // already at bottom
|
|
return
|
|
}
|
|
active := gc.filtered[idx]
|
|
next := gc.filtered[idx+1]
|
|
|
|
active.Widgets.UnHighlight()
|
|
gc.selectedID = next.Id
|
|
next.Widgets.Highlight()
|
|
|
|
gc.ScrollPage()
|
|
ui.Render(cGrid)
|
|
}
|
|
|
|
func (gc *GridCursor) PgUp() {
|
|
idx := gc.Idx()
|
|
if idx <= 0 { // already at top
|
|
return
|
|
}
|
|
|
|
nextidx := int(math.Max(0.0, float64(idx-cGrid.MaxRows())))
|
|
if gc.pgCount() > 0 {
|
|
cGrid.Offset = int(math.Max(float64(cGrid.Offset-cGrid.MaxRows()),
|
|
float64(0)))
|
|
}
|
|
|
|
active := gc.filtered[idx]
|
|
next := gc.filtered[nextidx]
|
|
|
|
active.Widgets.UnHighlight()
|
|
gc.selectedID = next.Id
|
|
next.Widgets.Highlight()
|
|
|
|
cGrid.Align()
|
|
ui.Render(cGrid)
|
|
}
|
|
|
|
func (gc *GridCursor) PgDown() {
|
|
idx := gc.Idx()
|
|
if idx >= gc.Len()-1 { // already at bottom
|
|
return
|
|
}
|
|
|
|
nextidx := int(math.Min(float64(gc.Len()-1), float64(idx+cGrid.MaxRows())))
|
|
if gc.pgCount() > 0 {
|
|
cGrid.Offset = int(math.Min(float64(cGrid.Offset+cGrid.MaxRows()),
|
|
float64(gc.Len()-cGrid.MaxRows())))
|
|
}
|
|
|
|
active := gc.filtered[idx]
|
|
next := gc.filtered[nextidx]
|
|
|
|
active.Widgets.UnHighlight()
|
|
gc.selectedID = next.Id
|
|
next.Widgets.Highlight()
|
|
|
|
cGrid.Align()
|
|
ui.Render(cGrid)
|
|
}
|
|
|
|
// number of pages at current row count and term height
|
|
func (gc *GridCursor) pgCount() int {
|
|
pages := gc.Len() / cGrid.MaxRows()
|
|
if gc.Len()%cGrid.MaxRows() > 0 {
|
|
pages++
|
|
}
|
|
return pages
|
|
}
|