329 lines
8.5 KiB
Go
329 lines
8.5 KiB
Go
// TODO: update table every second (to show correct relative time)
|
|
package pages
|
|
|
|
import (
|
|
"tasksquire/common"
|
|
"tasksquire/components/detailsviewer"
|
|
"tasksquire/components/table"
|
|
"tasksquire/taskwarrior"
|
|
|
|
"github.com/charmbracelet/bubbles/key"
|
|
tea "github.com/charmbracelet/bubbletea"
|
|
"github.com/charmbracelet/lipgloss"
|
|
)
|
|
|
|
type ReportPage struct {
|
|
common *common.Common
|
|
|
|
activeReport *taskwarrior.Report
|
|
activeContext *taskwarrior.Context
|
|
activeProject string
|
|
selectedTask *taskwarrior.Task
|
|
taskCursor int
|
|
|
|
tasks taskwarrior.Tasks
|
|
|
|
taskTable table.Model
|
|
|
|
// Details panel state
|
|
detailsPanelActive bool
|
|
detailsViewer *detailsviewer.DetailsViewer
|
|
|
|
subpage common.Component
|
|
}
|
|
|
|
func NewReportPage(com *common.Common, report *taskwarrior.Report) *ReportPage {
|
|
// return &ReportPage{
|
|
// common: com,
|
|
// activeReport: report,
|
|
// activeContext: com.TW.GetActiveContext(),
|
|
// activeProject: "",
|
|
// taskTable: table.New(com),
|
|
// }
|
|
|
|
p := &ReportPage{
|
|
common: com,
|
|
activeReport: report,
|
|
activeContext: com.TW.GetActiveContext(),
|
|
activeProject: "",
|
|
taskTable: table.New(com),
|
|
detailsPanelActive: false,
|
|
detailsViewer: detailsviewer.New(com),
|
|
}
|
|
|
|
return p
|
|
}
|
|
|
|
func (p *ReportPage) SetSize(width int, height int) {
|
|
p.common.SetSize(width, height)
|
|
|
|
baseHeight := height - p.common.Styles.Base.GetVerticalFrameSize()
|
|
baseWidth := width - p.common.Styles.Base.GetHorizontalFrameSize()
|
|
|
|
var tableHeight int
|
|
if p.detailsPanelActive {
|
|
// Allocate 60% for table, 40% for details panel
|
|
// Minimum 5 lines for details, minimum 10 lines for table
|
|
detailsHeight := max(min(baseHeight*2/5, baseHeight-10), 5)
|
|
tableHeight = baseHeight - detailsHeight - 1 // -1 for spacing
|
|
|
|
// Set component size (component handles its own border/padding)
|
|
p.detailsViewer.SetSize(baseWidth, detailsHeight)
|
|
} else {
|
|
tableHeight = baseHeight
|
|
}
|
|
|
|
p.taskTable.SetWidth(baseWidth)
|
|
p.taskTable.SetHeight(tableHeight)
|
|
}
|
|
|
|
// Helper functions
|
|
func max(a, b int) int {
|
|
if a > b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
func min(a, b int) int {
|
|
if a < b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
func (p *ReportPage) Init() tea.Cmd {
|
|
return tea.Batch(p.getTasks(), doTick())
|
|
}
|
|
|
|
func (p *ReportPage) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|
var cmds []tea.Cmd
|
|
switch msg := msg.(type) {
|
|
case tea.WindowSizeMsg:
|
|
p.SetSize(msg.Width, msg.Height)
|
|
// case BackMsg:
|
|
case tickMsg:
|
|
cmds = append(cmds, p.getTasks())
|
|
cmds = append(cmds, doTick())
|
|
return p, tea.Batch(cmds...)
|
|
case taskMsg:
|
|
p.tasks = taskwarrior.Tasks(msg)
|
|
p.populateTaskTable(p.tasks)
|
|
case UpdateReportMsg:
|
|
p.activeReport = msg
|
|
cmds = append(cmds, p.getTasks())
|
|
case UpdateContextMsg:
|
|
p.activeContext = msg
|
|
p.common.TW.SetContext(msg)
|
|
cmds = append(cmds, p.getTasks())
|
|
case UpdateProjectMsg:
|
|
p.activeProject = string(msg)
|
|
cmds = append(cmds, p.getTasks())
|
|
case TaskPickedMsg:
|
|
if msg.Task != nil && msg.Task.Status == "pending" {
|
|
p.common.TW.StopActiveTasks()
|
|
p.common.TW.StartTask(msg.Task)
|
|
}
|
|
cmds = append(cmds, p.getTasks())
|
|
case UpdatedTasksMsg:
|
|
cmds = append(cmds, p.getTasks())
|
|
case tea.KeyMsg:
|
|
// Handle ESC when details panel is active
|
|
if p.detailsPanelActive && key.Matches(msg, p.common.Keymap.Back) {
|
|
p.detailsPanelActive = false
|
|
p.detailsViewer.Blur()
|
|
p.SetSize(p.common.Width(), p.common.Height())
|
|
return p, nil
|
|
}
|
|
|
|
switch {
|
|
case key.Matches(msg, p.common.Keymap.Quit):
|
|
return p, tea.Quit
|
|
case key.Matches(msg, p.common.Keymap.SetReport):
|
|
p.subpage = NewReportPickerPage(p.common, p.activeReport)
|
|
cmd := p.subpage.Init()
|
|
p.common.PushPage(p)
|
|
return p.subpage, cmd
|
|
case key.Matches(msg, p.common.Keymap.SetContext):
|
|
p.subpage = NewContextPickerPage(p.common)
|
|
cmd := p.subpage.Init()
|
|
p.common.PushPage(p)
|
|
return p.subpage, cmd
|
|
case key.Matches(msg, p.common.Keymap.Add):
|
|
p.subpage = NewTaskEditorPage(p.common, taskwarrior.NewTask())
|
|
cmd := p.subpage.Init()
|
|
p.common.PushPage(p)
|
|
return p.subpage, cmd
|
|
case key.Matches(msg, p.common.Keymap.Edit):
|
|
p.subpage = NewTaskEditorPage(p.common, *p.selectedTask)
|
|
cmd := p.subpage.Init()
|
|
p.common.PushPage(p)
|
|
return p.subpage, cmd
|
|
case key.Matches(msg, p.common.Keymap.Ok):
|
|
p.common.TW.SetTaskDone(p.selectedTask)
|
|
return p, p.getTasks()
|
|
case key.Matches(msg, p.common.Keymap.Delete):
|
|
p.common.TW.DeleteTask(p.selectedTask)
|
|
return p, p.getTasks()
|
|
case key.Matches(msg, p.common.Keymap.SetProject):
|
|
p.subpage = NewProjectPickerPage(p.common, p.activeProject)
|
|
cmd := p.subpage.Init()
|
|
p.common.PushPage(p)
|
|
return p.subpage, cmd
|
|
case key.Matches(msg, p.common.Keymap.PickProjectTask):
|
|
p.subpage = NewProjectTaskPickerPage(p.common)
|
|
cmd := p.subpage.Init()
|
|
p.common.PushPage(p)
|
|
return p.subpage, cmd
|
|
case key.Matches(msg, p.common.Keymap.Tag):
|
|
if p.selectedTask != nil {
|
|
tag := p.common.TW.GetConfig().Get("uda.tasksquire.tag.default")
|
|
if p.selectedTask.HasTag(tag) {
|
|
p.selectedTask.RemoveTag(tag)
|
|
} else {
|
|
p.selectedTask.AddTag(tag)
|
|
}
|
|
p.common.TW.ImportTask(p.selectedTask)
|
|
return p, p.getTasks()
|
|
}
|
|
return p, nil
|
|
case key.Matches(msg, p.common.Keymap.Undo):
|
|
p.common.TW.Undo()
|
|
return p, p.getTasks()
|
|
case key.Matches(msg, p.common.Keymap.StartStop):
|
|
if p.selectedTask != nil && p.selectedTask.Status == "pending" {
|
|
if p.selectedTask.Start == "" {
|
|
p.common.TW.StopActiveTasks()
|
|
p.common.TW.StartTask(p.selectedTask)
|
|
} else {
|
|
p.common.TW.StopTask(p.selectedTask)
|
|
}
|
|
return p, p.getTasks()
|
|
}
|
|
case key.Matches(msg, p.common.Keymap.ViewDetails):
|
|
if p.selectedTask != nil {
|
|
// Toggle details panel
|
|
p.detailsPanelActive = !p.detailsPanelActive
|
|
if p.detailsPanelActive {
|
|
p.detailsViewer.SetTask(p.selectedTask)
|
|
p.detailsViewer.Focus()
|
|
} else {
|
|
p.detailsViewer.Blur()
|
|
}
|
|
p.SetSize(p.common.Width(), p.common.Height())
|
|
return p, nil
|
|
}
|
|
}
|
|
}
|
|
|
|
var cmd tea.Cmd
|
|
|
|
// Route keyboard messages to details viewer when panel is active
|
|
if p.detailsPanelActive {
|
|
var viewerCmd tea.Cmd
|
|
var viewerModel tea.Model
|
|
viewerModel, viewerCmd = p.detailsViewer.Update(msg)
|
|
p.detailsViewer = viewerModel.(*detailsviewer.DetailsViewer)
|
|
cmds = append(cmds, viewerCmd)
|
|
} else {
|
|
// Route to table when details panel not active
|
|
p.taskTable, cmd = p.taskTable.Update(msg)
|
|
cmds = append(cmds, cmd)
|
|
|
|
if p.tasks != nil && len(p.tasks) > 0 {
|
|
p.selectedTask = p.tasks[p.taskTable.Cursor()]
|
|
} else {
|
|
p.selectedTask = nil
|
|
}
|
|
}
|
|
|
|
return p, tea.Batch(cmds...)
|
|
}
|
|
|
|
func (p *ReportPage) View() string {
|
|
if p.tasks == nil || len(p.tasks) == 0 {
|
|
return p.common.Styles.Base.Render("No tasks found")
|
|
}
|
|
|
|
tableView := p.taskTable.View()
|
|
|
|
if !p.detailsPanelActive {
|
|
return tableView
|
|
}
|
|
|
|
// Combine table and details panel vertically
|
|
return lipgloss.JoinVertical(
|
|
lipgloss.Left,
|
|
tableView,
|
|
p.detailsViewer.View(),
|
|
)
|
|
}
|
|
|
|
func (p *ReportPage) populateTaskTable(tasks taskwarrior.Tasks) {
|
|
if len(tasks) == 0 {
|
|
return
|
|
}
|
|
|
|
selected := p.taskTable.Cursor()
|
|
|
|
if p.selectedTask != nil {
|
|
for i, task := range tasks {
|
|
if task.Uuid == p.selectedTask.Uuid {
|
|
selected = i
|
|
}
|
|
}
|
|
}
|
|
if selected > len(tasks)-1 {
|
|
selected = len(tasks) - 1
|
|
}
|
|
|
|
// Calculate proper dimensions based on whether details panel is active
|
|
baseHeight := p.common.Height() - p.common.Styles.Base.GetVerticalFrameSize()
|
|
baseWidth := p.common.Width() - p.common.Styles.Base.GetHorizontalFrameSize()
|
|
|
|
var tableHeight int
|
|
if p.detailsPanelActive {
|
|
// Allocate 60% for table, 40% for details panel
|
|
// Minimum 5 lines for details, minimum 10 lines for table
|
|
detailsHeight := max(min(baseHeight*2/5, baseHeight-10), 5)
|
|
tableHeight = baseHeight - detailsHeight - 1 // -1 for spacing
|
|
} else {
|
|
tableHeight = baseHeight
|
|
}
|
|
|
|
p.taskTable = table.New(
|
|
p.common,
|
|
table.WithReport(p.activeReport),
|
|
table.WithTasks(tasks),
|
|
table.WithFocused(true),
|
|
table.WithWidth(baseWidth),
|
|
table.WithHeight(tableHeight),
|
|
table.WithStyles(p.common.Styles.TableStyle),
|
|
)
|
|
|
|
if selected == 0 {
|
|
selected = p.taskTable.Cursor()
|
|
}
|
|
if selected < len(tasks) {
|
|
p.taskTable.SetCursor(selected)
|
|
} else {
|
|
p.taskTable.SetCursor(len(p.tasks) - 1)
|
|
}
|
|
|
|
// Refresh details content if panel is active
|
|
if p.detailsPanelActive && p.selectedTask != nil {
|
|
p.detailsViewer.SetTask(p.selectedTask)
|
|
}
|
|
}
|
|
|
|
func (p *ReportPage) getTasks() tea.Cmd {
|
|
return func() tea.Msg {
|
|
filters := []string{}
|
|
if p.activeProject != "" {
|
|
filters = append(filters, "project:"+p.activeProject)
|
|
}
|
|
tasks := p.common.TW.GetTasks(p.activeReport, filters...)
|
|
return taskMsg(tasks)
|
|
}
|
|
}
|