Merge pull request #14 from lmika/feature/issue-11
Issue 11: Added the ability to resize the table and item view
This commit is contained in:
commit
1109f2c9ee
|
@ -6,6 +6,10 @@ import (
|
||||||
"github.com/lmika/audax/internal/dynamo-browse/models"
|
"github.com/lmika/audax/internal/dynamo-browse/models"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type SetTableItemView struct {
|
||||||
|
ViewIndex int
|
||||||
|
}
|
||||||
|
|
||||||
type NewResultSet struct {
|
type NewResultSet struct {
|
||||||
ResultSet *models.ResultSet
|
ResultSet *models.ResultSet
|
||||||
currentFilter string
|
currentFilter string
|
||||||
|
|
|
@ -15,10 +15,22 @@ import (
|
||||||
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/statusandprompt"
|
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/statusandprompt"
|
||||||
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/styles"
|
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/styles"
|
||||||
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/tableselect"
|
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/tableselect"
|
||||||
|
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/utils"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
"log"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
ViewModeTablePrimary = 0
|
||||||
|
ViewModeTableItemEqual = 1
|
||||||
|
ViewModeItemPrimary = 2
|
||||||
|
ViewModeItemOnly = 3
|
||||||
|
ViewModeTableOnly = 4
|
||||||
|
|
||||||
|
ViewModeCount = 5
|
||||||
|
)
|
||||||
|
|
||||||
type Model struct {
|
type Model struct {
|
||||||
tableReadController *controllers.TableReadController
|
tableReadController *controllers.TableReadController
|
||||||
tableWriteController *controllers.TableWriteController
|
tableWriteController *controllers.TableWriteController
|
||||||
|
@ -27,9 +39,12 @@ type Model struct {
|
||||||
statusAndPrompt *statusandprompt.StatusAndPrompt
|
statusAndPrompt *statusandprompt.StatusAndPrompt
|
||||||
tableSelect *tableselect.Model
|
tableSelect *tableselect.Model
|
||||||
|
|
||||||
|
mainViewIndex int
|
||||||
|
|
||||||
root tea.Model
|
root tea.Model
|
||||||
tableView *dynamotableview.Model
|
tableView *dynamotableview.Model
|
||||||
itemView *dynamoitemview.Model
|
itemView *dynamoitemview.Model
|
||||||
|
mainView tea.Model
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewModel(
|
func NewModel(
|
||||||
|
@ -42,7 +57,7 @@ func NewModel(
|
||||||
|
|
||||||
dtv := dynamotableview.New(uiStyles)
|
dtv := dynamotableview.New(uiStyles)
|
||||||
div := dynamoitemview.New(itemRendererService, uiStyles)
|
div := dynamoitemview.New(itemRendererService, uiStyles)
|
||||||
mainView := layout.NewVBox(layout.LastChildFixedAt(13), dtv, div)
|
mainView := layout.NewVBox(layout.LastChildFixedAt(14), dtv, div)
|
||||||
|
|
||||||
itemEdit := dynamoitemedit.NewModel(mainView)
|
itemEdit := dynamoitemedit.NewModel(mainView)
|
||||||
statusAndPrompt := statusandprompt.New(itemEdit, "", uiStyles.StatusAndPrompt)
|
statusAndPrompt := statusandprompt.New(itemEdit, "", uiStyles.StatusAndPrompt)
|
||||||
|
@ -131,6 +146,7 @@ func NewModel(
|
||||||
root: root,
|
root: root,
|
||||||
tableView: dtv,
|
tableView: dtv,
|
||||||
itemView: div,
|
itemView: div,
|
||||||
|
mainView: mainView,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -140,6 +156,9 @@ func (m Model) Init() tea.Cmd {
|
||||||
|
|
||||||
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||||
switch msg := msg.(type) {
|
switch msg := msg.(type) {
|
||||||
|
case controllers.SetTableItemView:
|
||||||
|
cmd := m.setMainViewIndex(msg.ViewIndex)
|
||||||
|
return m, cmd
|
||||||
case controllers.ResultSetUpdated:
|
case controllers.ResultSetUpdated:
|
||||||
return m, m.tableView.Refresh()
|
return m, m.tableView.Refresh()
|
||||||
case tea.KeyMsg:
|
case tea.KeyMsg:
|
||||||
|
@ -161,6 +180,14 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||||
return m, m.tableReadController.Filter
|
return m, m.tableReadController.Filter
|
||||||
case "backspace":
|
case "backspace":
|
||||||
return m, m.tableReadController.ViewBack
|
return m, m.tableReadController.ViewBack
|
||||||
|
case "w":
|
||||||
|
return m, func() tea.Msg {
|
||||||
|
return controllers.SetTableItemView{ViewIndex: utils.Cycle(m.mainViewIndex, 1, ViewModeCount)}
|
||||||
|
}
|
||||||
|
case "W":
|
||||||
|
return m, func() tea.Msg {
|
||||||
|
return controllers.SetTableItemView{ViewIndex: utils.Cycle(m.mainViewIndex, -1, ViewModeCount)}
|
||||||
|
}
|
||||||
//case "e":
|
//case "e":
|
||||||
// m.itemEdit.Visible()
|
// m.itemEdit.Visible()
|
||||||
// return m, nil
|
// return m, nil
|
||||||
|
@ -180,3 +207,28 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||||
func (m Model) View() string {
|
func (m Model) View() string {
|
||||||
return m.root.View()
|
return m.root.View()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *Model) setMainViewIndex(viewIndex int) tea.Cmd {
|
||||||
|
log.Printf("setting view index = %v", viewIndex)
|
||||||
|
|
||||||
|
var newMainView tea.Model
|
||||||
|
switch viewIndex {
|
||||||
|
case ViewModeTablePrimary:
|
||||||
|
newMainView = layout.NewVBox(layout.LastChildFixedAt(14), m.tableView, m.itemView)
|
||||||
|
case ViewModeTableItemEqual:
|
||||||
|
newMainView = layout.NewVBox(layout.EqualSize(), m.tableView, m.itemView)
|
||||||
|
case ViewModeItemPrimary:
|
||||||
|
newMainView = layout.NewVBox(layout.FirstChildFixedAt(7), m.tableView, m.itemView)
|
||||||
|
case ViewModeItemOnly:
|
||||||
|
newMainView = layout.NewZStack(m.itemView, m.tableView)
|
||||||
|
case ViewModeTableOnly:
|
||||||
|
newMainView = layout.NewZStack(m.tableView, m.tableView)
|
||||||
|
default:
|
||||||
|
newMainView = m.mainView
|
||||||
|
}
|
||||||
|
|
||||||
|
m.mainViewIndex = viewIndex
|
||||||
|
m.mainView = newMainView
|
||||||
|
m.itemEdit.SetSubmodel(m.mainView)
|
||||||
|
return m.tableView.Refresh()
|
||||||
|
}
|
||||||
|
|
|
@ -107,3 +107,8 @@ func (m *Model) Resize(w, h int) layout.ResizingModel {
|
||||||
func (m *Model) Visible() {
|
func (m *Model) Visible() {
|
||||||
m.visible = true
|
m.visible = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *Model) SetSubmodel(submodel tea.Model) {
|
||||||
|
m.submodel = submodel
|
||||||
|
m.Resize(m.w, m.h)
|
||||||
|
}
|
||||||
|
|
|
@ -24,6 +24,21 @@ func (l equalSize) childSize(idx, cnt, available int) int {
|
||||||
return childrenHeight
|
return childrenHeight
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func FirstChildFixedAt(size int) BoxSize {
|
||||||
|
return firstChildFixedAt{size}
|
||||||
|
}
|
||||||
|
|
||||||
|
type firstChildFixedAt struct {
|
||||||
|
firstChildSize int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l firstChildFixedAt) childSize(idx, cnt, available int) int {
|
||||||
|
if idx == 0 {
|
||||||
|
return l.firstChildSize
|
||||||
|
}
|
||||||
|
return (equalSize{}).childSize(idx, cnt-1, available-l.firstChildSize)
|
||||||
|
}
|
||||||
|
|
||||||
func LastChildFixedAt(size int) BoxSize {
|
func LastChildFixedAt(size int) BoxSize {
|
||||||
return lastChildFixedAt{size}
|
return lastChildFixedAt{size}
|
||||||
}
|
}
|
||||||
|
|
61
internal/dynamo-browse/ui/teamodels/layout/zstack.go
Normal file
61
internal/dynamo-browse/ui/teamodels/layout/zstack.go
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
package layout
|
||||||
|
|
||||||
|
import (
|
||||||
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
|
"github.com/lmika/audax/internal/dynamo-browse/ui/teamodels/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ZStack struct {
|
||||||
|
visibleModel tea.Model
|
||||||
|
focusedModel tea.Model
|
||||||
|
otherModels []tea.Model
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewZStack(visibleModel tea.Model, focusedModel tea.Model, otherModels ...tea.Model) ZStack {
|
||||||
|
return ZStack{
|
||||||
|
visibleModel: visibleModel,
|
||||||
|
focusedModel: focusedModel,
|
||||||
|
otherModels: otherModels,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vb ZStack) Init() tea.Cmd {
|
||||||
|
var cc utils.CmdCollector
|
||||||
|
cc.Collect(vb.visibleModel, vb.visibleModel.Init())
|
||||||
|
cc.Collect(vb.focusedModel, vb.focusedModel.Init())
|
||||||
|
for _, c := range vb.otherModels {
|
||||||
|
cc.Collect(c, c.Init())
|
||||||
|
}
|
||||||
|
return cc.Cmd()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vb ZStack) Update(msg tea.Msg) (m tea.Model, cmd tea.Cmd) {
|
||||||
|
switch msg.(type) {
|
||||||
|
case tea.KeyMsg:
|
||||||
|
// Only the focused model gets keyboard events
|
||||||
|
vb.focusedModel, cmd = vb.focusedModel.Update(msg)
|
||||||
|
return vb, cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
// All other messages go to each model
|
||||||
|
var cc utils.CmdCollector
|
||||||
|
vb.visibleModel = cc.Collect(vb.visibleModel.Update(msg))
|
||||||
|
vb.focusedModel = cc.Collect(vb.focusedModel.Update(msg))
|
||||||
|
for i, c := range vb.otherModels {
|
||||||
|
vb.otherModels[i] = cc.Collect(c.Update(msg))
|
||||||
|
}
|
||||||
|
return vb, cc.Cmd()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vb ZStack) View() string {
|
||||||
|
return vb.visibleModel.View()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (vb ZStack) Resize(w, h int) ResizingModel {
|
||||||
|
vb.visibleModel = Resize(vb.visibleModel, w, h)
|
||||||
|
vb.focusedModel = Resize(vb.focusedModel, w, h)
|
||||||
|
for i := range vb.otherModels {
|
||||||
|
vb.otherModels[i] = Resize(vb.otherModels[i], w, h)
|
||||||
|
}
|
||||||
|
return vb
|
||||||
|
}
|
|
@ -6,3 +6,17 @@ func Max(x, y int) int {
|
||||||
}
|
}
|
||||||
return y
|
return y
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Cycle(n int, by int, max int) int {
|
||||||
|
by = by % max
|
||||||
|
if by > 0 {
|
||||||
|
return (n + by) % max
|
||||||
|
} else if by < 0 {
|
||||||
|
wn := n + by
|
||||||
|
if wn < 0 {
|
||||||
|
return max + wn
|
||||||
|
}
|
||||||
|
return wn
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue