tiki/view/tiki_plugin_view_test.go
2026-03-23 23:58:26 -04:00

229 lines
6.1 KiB
Go

package view
import (
"fmt"
"testing"
"github.com/boolean-maybe/tiki/config"
"github.com/boolean-maybe/tiki/controller"
"github.com/boolean-maybe/tiki/model"
"github.com/boolean-maybe/tiki/plugin"
"github.com/boolean-maybe/tiki/store"
"github.com/boolean-maybe/tiki/task"
)
func TestPluginViewRefreshResetsNonSelectedLaneScrollOffset(t *testing.T) {
taskStore := store.NewInMemoryStore()
pluginConfig := model.NewPluginConfig("TestPlugin")
pluginConfig.SetLaneLayout([]int{1, 1}, nil)
pluginDef := &plugin.TikiPlugin{
BasePlugin: plugin.BasePlugin{
Name: "TestPlugin",
},
Lanes: []plugin.TikiLane{
{Name: "Lane0", Columns: 1},
{Name: "Lane1", Columns: 1},
},
}
tasks := make([]*task.Task, 10)
for i := range tasks {
tasks[i] = &task.Task{
ID: fmt.Sprintf("T-%d", i),
Title: fmt.Sprintf("Task %d", i),
Status: task.StatusReady,
Type: task.TypeStory,
}
}
pv := NewPluginView(taskStore, pluginConfig, pluginDef, func(lane int) []*task.Task {
return tasks
}, nil, controller.PluginViewActions(), true)
itemHeight := config.TaskBoxHeight
for _, lb := range pv.laneBoxes {
lb.SetRect(0, 0, 80, itemHeight*5)
}
// select last task in lane 0 to force scroll offset
pluginConfig.SetSelectedLane(0)
pluginConfig.SetSelectedIndexForLane(0, len(tasks)-1)
pv.refresh()
if pv.laneBoxes[0].scrollOffset == 0 {
t.Fatalf("expected lane 0 scroll offset > 0 after selecting last item")
}
// non-selected lane 1 must have scroll offset 0
if pv.laneBoxes[1].scrollOffset != 0 {
t.Fatalf("expected non-selected lane 1 scroll offset 0, got %d", pv.laneBoxes[1].scrollOffset)
}
// switch selection to lane 1, scroll it, then verify lane 0 resets
pluginConfig.SetSelectedLane(1)
pluginConfig.SetSelectedIndexForLane(1, len(tasks)-1)
pv.refresh()
if pv.laneBoxes[1].scrollOffset == 0 {
t.Fatalf("expected lane 1 scroll offset > 0 after selecting last item")
}
if pv.laneBoxes[0].scrollOffset != 0 {
t.Fatalf("expected non-selected lane 0 scroll offset 0, got %d", pv.laneBoxes[0].scrollOffset)
}
}
func TestPluginViewGridLayout_RowCount(t *testing.T) {
tests := []struct {
name string
numTasks int
columns int
expectedRows int
}{
{"zero tasks", 0, 1, 0},
{"6 tasks / 2 cols", 6, 2, 3},
{"5 tasks / 3 cols", 5, 3, 2},
{"1 task / 1 col", 1, 1, 1},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
taskStore := store.NewInMemoryStore()
pluginConfig := model.NewPluginConfig("TestPlugin")
pluginConfig.SetLaneLayout([]int{tt.columns}, nil)
pluginDef := &plugin.TikiPlugin{
BasePlugin: plugin.BasePlugin{Name: "TestPlugin"},
Lanes: []plugin.TikiLane{{Name: "Lane", Columns: tt.columns}},
}
tasks := make([]*task.Task, tt.numTasks)
for i := range tasks {
tasks[i] = &task.Task{
ID: fmt.Sprintf("T-%d", i),
Title: fmt.Sprintf("Task %d", i),
Status: task.StatusReady,
Type: task.TypeStory,
}
}
pv := NewPluginView(taskStore, pluginConfig, pluginDef, func(lane int) []*task.Task {
return tasks
}, nil, controller.PluginViewActions(), true)
pv.refresh()
got := len(pv.laneBoxes[0].items)
if got != tt.expectedRows {
t.Errorf("rows = %d, want %d", got, tt.expectedRows)
}
})
}
}
func TestPluginViewGridLayout_SelectedRow(t *testing.T) {
tests := []struct {
name string
numTasks int
columns int
selectedIndex int
expectedSelectedRow int
}{
{"index 0, 2 cols", 4, 2, 0, 0},
{"index 2, 2 cols", 4, 2, 2, 1},
{"index 4, 3 cols", 6, 3, 4, 1},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
taskStore := store.NewInMemoryStore()
pluginConfig := model.NewPluginConfig("TestPlugin")
pluginConfig.SetLaneLayout([]int{tt.columns}, nil)
pluginDef := &plugin.TikiPlugin{
BasePlugin: plugin.BasePlugin{Name: "TestPlugin"},
Lanes: []plugin.TikiLane{{Name: "Lane", Columns: tt.columns}},
}
tasks := make([]*task.Task, tt.numTasks)
for i := range tasks {
tasks[i] = &task.Task{
ID: fmt.Sprintf("T-%d", i),
Title: fmt.Sprintf("Task %d", i),
Status: task.StatusReady,
Type: task.TypeStory,
}
}
pv := NewPluginView(taskStore, pluginConfig, pluginDef, func(lane int) []*task.Task {
return tasks
}, nil, controller.PluginViewActions(), true)
pluginConfig.SetSelectedLane(0)
pluginConfig.SetSelectedIndexForLane(0, tt.selectedIndex)
pv.refresh()
got := pv.laneBoxes[0].selectionIndex
if got != tt.expectedSelectedRow {
t.Errorf("selectionIndex = %d, want %d", got, tt.expectedSelectedRow)
}
})
}
}
func TestPluginViewRefreshPreservesScrollOffset(t *testing.T) {
taskStore := store.NewInMemoryStore()
pluginConfig := model.NewPluginConfig("TestPlugin")
pluginConfig.SetLaneLayout([]int{1}, nil)
pluginDef := &plugin.TikiPlugin{
BasePlugin: plugin.BasePlugin{
Name: "TestPlugin",
},
Lanes: []plugin.TikiLane{
{Name: "Lane", Columns: 1},
},
}
tasks := make([]*task.Task, 10)
for i := range tasks {
tasks[i] = &task.Task{
ID: fmt.Sprintf("T-%d", i),
Title: fmt.Sprintf("Task %d", i),
Status: task.StatusReady,
Type: task.TypeStory,
}
}
pv := NewPluginView(taskStore, pluginConfig, pluginDef, func(lane int) []*task.Task {
return tasks
}, nil, controller.PluginViewActions(), true)
if len(pv.laneBoxes) != 1 {
t.Fatalf("expected 1 lane box, got %d", len(pv.laneBoxes))
}
lane := pv.laneBoxes[0]
itemHeight := config.TaskBoxHeight
lane.SetRect(0, 0, 80, itemHeight*5)
pluginConfig.SetSelectedIndexForLane(0, len(tasks)-1)
pv.refresh()
expectedScrollOffset := len(tasks) - 5
if lane.scrollOffset != expectedScrollOffset {
t.Fatalf("expected scrollOffset %d, got %d", expectedScrollOffset, lane.scrollOffset)
}
laneBefore := lane
pluginConfig.SetSelectedIndexForLane(0, len(tasks)-2)
pv.refresh()
if pv.laneBoxes[0] != laneBefore {
t.Fatalf("expected lane list to be reused across refresh")
}
if lane.scrollOffset != expectedScrollOffset {
t.Fatalf("expected scrollOffset to remain %d, got %d", expectedScrollOffset, lane.scrollOffset)
}
}