// TODO: update table every second (to show correct relative time) package pages import ( "log/slog" "tasksquire/common" "tasksquire/components/table" "tasksquire/taskwarrior" "github.com/charmbracelet/bubbles/key" // "github.com/charmbracelet/bubbles/table" tea "github.com/charmbracelet/bubbletea" ) type ReportPage struct { common *common.Common activeReport *taskwarrior.Report activeContext *taskwarrior.Context activeProject string selectedTask *taskwarrior.Task tasks taskwarrior.Tasks taskTable table.Model subpage tea.Model subpageActive bool } func NewReportPage(com *common.Common, report *taskwarrior.Report) *ReportPage { return &ReportPage{ common: com, activeReport: report, activeContext: com.TW.GetActiveContext(), activeProject: "", } } func (p *ReportPage) SetSize(width int, height int) { p.common.SetSize(width, height) slog.Info("FramSize", "vert", p.common.Styles.Main.GetVerticalFrameSize(), "horz", p.common.Styles.Main.GetHorizontalFrameSize()) p.taskTable.SetWidth(width - p.common.Styles.Main.GetVerticalFrameSize()) p.taskTable.SetHeight(height - p.common.Styles.Main.GetHorizontalFrameSize()) } func (p *ReportPage) Init() tea.Cmd { return p.getTasks() } 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: p.subpageActive = false 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 UpdatedTasksMsg: cmds = append(cmds, p.getTasks()) case tea.KeyMsg: 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) p.subpage.Init() p.subpageActive = true p.common.PushPage(p) return p.subpage, nil case key.Matches(msg, p.common.Keymap.SetContext): p.subpage = NewContextPickerPage(p.common) p.subpage.Init() p.subpageActive = true p.common.PushPage(p) return p.subpage, nil case key.Matches(msg, p.common.Keymap.Add): p.subpage = NewTaskEditorPage(p.common, taskwarrior.Task{}) p.subpage.Init() p.subpageActive = true p.common.PushPage(p) return p.subpage, nil case key.Matches(msg, p.common.Keymap.Edit): p.subpage = NewTaskEditorPage(p.common, *p.selectedTask) p.subpage.Init() p.subpageActive = true p.common.PushPage(p) return p.subpage, nil 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.SetProject): p.subpage = NewProjectPickerPage(p.common, p.activeProject) p.subpage.Init() p.subpageActive = true p.common.PushPage(p) return p.subpage, nil case key.Matches(msg, p.common.Keymap.Undo): p.common.TW.Undo() return p, p.getTasks() } } var cmd tea.Cmd 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 { // return p.common.Styles.Main.Render(p.taskTable.View()) + "\n" if p.tasks == nil || len(p.tasks) == 0 { return p.common.Styles.Main.Render("No tasks found") } return p.common.Styles.Main.Render(p.taskTable.View()) } func (p *ReportPage) populateTaskTable(tasks taskwarrior.Tasks) { if len(tasks) == 0 { return } selected := 0 if p.selectedTask != nil { for i, task := range tasks { if task.Uuid == p.selectedTask.Uuid { selected = i } } } p.taskTable = table.New( table.WithReport(p.activeReport), table.WithTasks(tasks), table.WithFocused(true), table.WithWidth(p.common.Width()-p.common.Styles.Main.GetVerticalFrameSize()), table.WithHeight(p.common.Height()-p.common.Styles.Main.GetHorizontalFrameSize()-10), 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) } } 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) } } type TaskMsg taskwarrior.Tasks