// 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) } }