dynamo-browse/internal/dynamo-browse/ui/teamodels/dynamotableview/model.go
Leon Mika 982d3a9ca7
Issue 18: Added a popup to modify table columns (#31)
Added a new popup to modify the columns of the table. With this new popup, the user can:

- Show and hide columns
- Move columns around
- Add new columns which are derived from the value of an expression
- Delete columns

Also got the overlay mechanisms working.
2022-10-04 22:23:48 +11:00

231 lines
5.7 KiB
Go

package dynamotableview
import (
"github.com/charmbracelet/bubbles/key"
tea "github.com/charmbracelet/bubbletea"
"github.com/charmbracelet/lipgloss"
"github.com/lmika/audax/internal/dynamo-browse/controllers"
"github.com/lmika/audax/internal/dynamo-browse/models"
"github.com/lmika/audax/internal/dynamo-browse/models/columns"
"github.com/lmika/audax/internal/dynamo-browse/ui/keybindings"
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/dynamoitemview"
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/frame"
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/layout"
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/styles"
table "github.com/lmika/go-bubble-table"
"strings"
)
var (
activeHeaderStyle = lipgloss.NewStyle().
Bold(true).
Foreground(lipgloss.Color("#ffffff")).
Background(lipgloss.Color("#4479ff"))
)
type Setting interface {
IsReadOnly() bool
}
type ColumnsProvider interface {
Columns() *columns.Columns
}
type Model struct {
frameTitle frame.FrameTitle
table table.Model
w, h int
keyBinding *keybindings.TableKeyBinding
setting Setting
columnsProvider ColumnsProvider
// model state
isReadOnly bool
colOffset int
rows []table.Row
columns []columns.Column
resultSet *models.ResultSet
}
func New(keyBinding *keybindings.TableKeyBinding, columnsProvider ColumnsProvider, setting Setting, uiStyles styles.Styles) *Model {
frameTitle := frame.NewFrameTitle("No table", true, uiStyles.Frames)
isReadOnly := setting.IsReadOnly()
model := &Model{
isReadOnly: isReadOnly,
frameTitle: frameTitle,
keyBinding: keyBinding,
setting: setting,
columnsProvider: columnsProvider,
}
model.table = table.New(columnModel{model}, 100, 100)
model.table.SetRows([]table.Row{})
return model
}
func (m *Model) Init() tea.Cmd {
return nil
}
func (m *Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) {
case controllers.NewResultSet:
m.resultSet = msg.ResultSet
m.updateTable()
return m, m.postSelectedItemChanged
case controllers.ColumnsUpdated:
m.rebuildTable(&m.table)
return m, m.postSelectedItemChanged
case controllers.SettingsUpdated:
m.updateTableHeading()
return m, nil
case controllers.MoveLeftmostDisplayedColumnInTableViewBy:
m.setLeftmostDisplayedColumn(m.colOffset + int(msg))
return m, nil
case tea.KeyMsg:
switch {
// Table nav
case key.Matches(msg, m.keyBinding.MoveUp):
m.table.GoUp()
return m, m.postSelectedItemChanged
case key.Matches(msg, m.keyBinding.MoveDown):
m.table.GoDown()
return m, m.postSelectedItemChanged
case key.Matches(msg, m.keyBinding.PageUp):
m.table.GoPageUp()
return m, m.postSelectedItemChanged
case key.Matches(msg, m.keyBinding.PageDown):
m.table.GoPageDown()
return m, m.postSelectedItemChanged
case key.Matches(msg, m.keyBinding.Home):
m.table.GoTop()
return m, m.postSelectedItemChanged
case key.Matches(msg, m.keyBinding.End):
m.table.GoBottom()
return m, m.postSelectedItemChanged
case key.Matches(msg, m.keyBinding.ColLeft):
m.setLeftmostDisplayedColumn(m.colOffset - 1)
return m, nil
case key.Matches(msg, m.keyBinding.ColRight):
m.setLeftmostDisplayedColumn(m.colOffset + 1)
return m, nil
}
}
return m, nil
}
func (m *Model) setLeftmostDisplayedColumn(newCol int) {
if newCol < 0 {
m.colOffset = 0
} else if newCol >= len(m.columnsProvider.Columns().Columns) {
m.colOffset = len(m.columnsProvider.Columns().Columns) - 1
} else {
m.colOffset = newCol
}
m.table.UpdateView()
}
func (m *Model) View() string {
return lipgloss.JoinVertical(lipgloss.Top, m.frameTitle.View(), m.table.View())
}
func (m *Model) Resize(w, h int) layout.ResizingModel {
m.w, m.h = w, h
tblHeight := h - m.frameTitle.HeaderHeight()
m.table.SetSize(w, tblHeight)
m.frameTitle.Resize(w, h)
return m
}
func (m *Model) updateTableHeading() {
tableName := new(strings.Builder)
tableName.WriteString("Table: " + m.resultSet.TableInfo.Name)
if m.setting.IsReadOnly() {
tableName.WriteString(" [RO]")
}
m.frameTitle.SetTitle(tableName.String())
}
func (m *Model) updateTable() {
m.updateTableHeading()
m.colOffset = 0
m.rebuildTable(nil)
}
func (m *Model) rebuildTable(targetTbl *table.Model) {
var tbl table.Model
resultSet := m.resultSet
// Use the target table model if you can, but if it's nil or the number of rows is smaller than the
// existing table, create a new one
if targetTbl == nil || len(resultSet.Items()) > len(m.rows) {
tbl = table.New(columnModel{m}, m.w, m.h-m.frameTitle.HeaderHeight())
if targetTbl != nil {
tbl.GoBottom()
}
} else {
tbl = *targetTbl
}
m.columns = m.columnsProvider.Columns().VisibleColumns()
newRows := make([]table.Row, 0)
for i, r := range resultSet.Items() {
if resultSet.Hidden(i) {
continue
}
newRows = append(newRows, itemTableRow{
model: m,
resultSet: resultSet,
itemIndex: i,
item: r,
})
}
m.rows = newRows
tbl.SetRows(newRows)
m.table = tbl
}
func (m *Model) SelectedItemIndex() int {
selectedItem, ok := m.selectedItem()
if !ok {
return -1
}
return selectedItem.itemIndex
}
func (m *Model) selectedItem() (itemTableRow, bool) {
resultSet := m.resultSet
if resultSet != nil && len(m.rows) > 0 {
selectedItem, ok := m.table.SelectedRow().(itemTableRow)
if ok {
return selectedItem, true
}
}
return itemTableRow{}, false
}
func (m *Model) postSelectedItemChanged() tea.Msg {
item, ok := m.selectedItem()
if !ok {
return dynamoitemview.NewItemSelected{ResultSet: item.resultSet, Item: nil}
}
return dynamoitemview.NewItemSelected{ResultSet: item.resultSet, Item: item.item}
}
func (m *Model) Refresh() tea.Cmd {
m.table.SetRows(m.rows)
return m.postSelectedItemChanged
}