all repos — todo.txt-go @ b258016622cf9ea6bcb18f4850b601b16dd54185

Setup project
Tim Izzo tim@octree.ch
Thu, 20 Mar 2025 08:42:41 +0100
commit

b258016622cf9ea6bcb18f4850b601b16dd54185

6 files changed, 351 insertions(+), 0 deletions(-)

jump to
A .gitignore

@@ -0,0 +1,20 @@

+# Binaries for programs and plugins +*.exe +*.exe~ +*.dll +*.so +*.dylib + +# Test binary, built with `go test -c` +*.test + +# Output of the go coverage tool, specifically when used with LiteIDE +*.out + +# Dependency directories +vendor/ + +# Go workspace file +go.work + +*.txt
A README.md

@@ -0,0 +1,8 @@

+# TODO.txt in Go + +## Ressources + +- <https://github.com/charmbracelet/bubbletea> +- <https://github.com/1set/todotxt/blob/master/task.go> +- <https://github.com/todotxt/todo.txt> +- <https://github.com/freitass/todo.txt-vim>
A go.mod

@@ -0,0 +1,33 @@

+module 5ika.ch/todo-txt + +go 1.24.1 + +require ( + github.com/charmbracelet/bubbles v0.20.0 + github.com/charmbracelet/bubbletea v1.3.4 +) + +require ( + github.com/1set/gut v0.0.0-20201117175203-a82363231997 // indirect + github.com/atotto/clipboard v0.1.4 // indirect +) + +require ( + github.com/1set/todotxt v0.0.4 + github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect + github.com/charmbracelet/lipgloss v1.0.0 + github.com/charmbracelet/x/ansi v0.8.0 // indirect + github.com/charmbracelet/x/term v0.2.1 // indirect + github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect + github.com/lucasb-eyer/go-colorful v1.2.0 // indirect + github.com/mattn/go-isatty v0.0.20 // indirect + github.com/mattn/go-localereader v0.0.1 // indirect + github.com/mattn/go-runewidth v0.0.16 // indirect + github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 // indirect + github.com/muesli/cancelreader v0.2.2 // indirect + github.com/muesli/termenv v0.15.2 // indirect + github.com/rivo/uniseg v0.4.7 // indirect + golang.org/x/sync v0.11.0 // indirect + golang.org/x/sys v0.30.0 // indirect + golang.org/x/text v0.3.8 // indirect +)
A go.sum

@@ -0,0 +1,45 @@

+github.com/1set/gut v0.0.0-20201117175203-a82363231997 h1:za2jSkE1Rx56hTzBko3ZZ4gA/nq+rA/jVovWuAF4jyo= +github.com/1set/gut v0.0.0-20201117175203-a82363231997/go.mod h1:DpCCAL0dgBMQdiqPUIIRpdU9zNcIZwJjW+L/8Mb30mw= +github.com/1set/todotxt v0.0.4 h1:A8DpMwGxctq7oT3s/2uC26RkLMKHRnG1E7x/cD8IeSI= +github.com/1set/todotxt v0.0.4/go.mod h1:FILY0dppcGIZzNbr4WUZ1d2bd1YWTCllmGIuXt3MP9Y= +github.com/atotto/clipboard v0.1.4 h1:EH0zSVneZPSuFR11BlR9YppQTVDbh5+16AmcJi4g1z4= +github.com/atotto/clipboard v0.1.4/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI= +github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k= +github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8= +github.com/charmbracelet/bubbles v0.20.0 h1:jSZu6qD8cRQ6k9OMfR1WlM+ruM8fkPWkHvQWD9LIutE= +github.com/charmbracelet/bubbles v0.20.0/go.mod h1:39slydyswPy+uVOHZ5x/GjwVAFkCsV8IIVy+4MhzwwU= +github.com/charmbracelet/bubbletea v1.3.4 h1:kCg7B+jSCFPLYRA52SDZjr51kG/fMUEoPoZrkaDHyoI= +github.com/charmbracelet/bubbletea v1.3.4/go.mod h1:dtcUCyCGEX3g9tosuYiut3MXgY/Jsv9nKVdibKKRRXo= +github.com/charmbracelet/lipgloss v1.0.0 h1:O7VkGDvqEdGi93X+DeqsQ7PKHDgtQfF8j8/O2qFMQNg= +github.com/charmbracelet/lipgloss v1.0.0/go.mod h1:U5fy9Z+C38obMs+T+tJqst9VGzlOYGj4ri9reL3qUlo= +github.com/charmbracelet/x/ansi v0.8.0 h1:9GTq3xq9caJW8ZrBTe0LIe2fvfLR/bYXKTx2llXn7xE= +github.com/charmbracelet/x/ansi v0.8.0/go.mod h1:wdYl/ONOLHLIVmQaxbIYEC/cRKOQyjTkowiI4blgS9Q= +github.com/charmbracelet/x/term v0.2.1 h1:AQeHeLZ1OqSXhrAWpYUtZyX1T3zVxfpZuEQMIQaGIAQ= +github.com/charmbracelet/x/term v0.2.1/go.mod h1:oQ4enTYFV7QN4m0i9mzHrViD7TQKvNEEkHUMCmsxdUg= +github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f h1:Y/CXytFA4m6baUTXGLOoWe4PQhGxaX0KpnayAqC48p4= +github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f/go.mod h1:vw97MGsxSvLiUE2X8qFplwetxpGLQrlU1Q9AUEIzCaM= +github.com/lucasb-eyer/go-colorful v1.2.0 h1:1nnpGOrhyZZuNyfu1QjKiUICQ74+3FNCN69Aj6K7nkY= +github.com/lucasb-eyer/go-colorful v1.2.0/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0= +github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= +github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= +github.com/mattn/go-localereader v0.0.1 h1:ygSAOl7ZXTx4RdPYinUpg6W99U8jWvWi9Ye2JC/oIi4= +github.com/mattn/go-localereader v0.0.1/go.mod h1:8fBrzywKY7BI3czFoHkuzRoWE9C+EiG4R1k4Cjx5p88= +github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc= +github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= +github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 h1:ZK8zHtRHOkbHy6Mmr5D264iyp3TiX5OmNcI5cIARiQI= +github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6/go.mod h1:CJlz5H+gyd6CUWT45Oy4q24RdLyn7Md9Vj2/ldJBSIo= +github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA= +github.com/muesli/cancelreader v0.2.2/go.mod h1:3XuTXfFS2VjM+HTLZY9Ak0l6eUKfijIfMUZ4EgX0QYo= +github.com/muesli/termenv v0.15.2 h1:GohcuySI0QmI3wN8Ok9PtKGkgkFIk7y6Vpb5PvrY+Wo= +github.com/muesli/termenv v0.15.2/go.mod h1:Epx+iuz8sNs7mNKhxzH4fWXGNpZwUaJKRS1noLXviQ8= +github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= +github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ= +github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= +golang.org/x/sync v0.11.0 h1:GGz8+XQP4FvTTrjZPzNKTMFtSXH80RAzG+5ghFPgK9w= +golang.org/x/sync v0.11.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= +golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc= +golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/text v0.3.8 h1:nAL+RVCQ9uMn3vJZbV+MRnydTJFPf8qqY42YiA6MrqY= +golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ=
A main.go

@@ -0,0 +1,205 @@

+package main + +import ( + "fmt" + "log" + "os" + + todo "github.com/1set/todotxt" + "github.com/charmbracelet/bubbles/textinput" + tea "github.com/charmbracelet/bubbletea" +) + +type InterfaceState int + +const ( + List InterfaceState = iota + Add +) + +type model struct { + interfaceState InterfaceState + + tasks todo.TaskList + cursor int // which to-do list item our cursor is pointing at + selected map[int]struct{} // which to-do items are selected + + textInput textinput.Model +} + +func initialModel() model { + + ti := textinput.New() + ti.Focus() + ti.CharLimit = 200 + ti.Width = 200 + + if tasklist, err := todo.LoadFromPath("todo.txt"); err != nil { + log.Fatal(err) + // TODO - Handle error, create file if not exists + return model{ + textInput: ti, + } + } else { + // tasks := tasklist.Filter(todo.FilterNotCompleted) + tasks := tasklist + _ = tasks.Sort(todo.SortPriorityAsc, todo.SortProjectAsc) + selected := make(map[int]struct{}) + + for i, t := range tasks { + if t.Completed { + selected[i] = struct{}{} + } + } + + return model{ + tasks: tasks, + selected: selected, + textInput: ti, + } + + } + +} + +func (m model) Init() tea.Cmd { + return textinput.Blink +} + +func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) { + var cmd tea.Cmd + + switch msg := msg.(type) { + + // Is it a key press? + case tea.KeyMsg: + + switch m.interfaceState { + + case List: + + switch msg.String() { + + // These keys should exit the program. + case "ctrl+c", "q": + m.tasks.WriteToPath("todo.txt") + return m, tea.Quit + + // Set focused task priority + case "A": + m.tasks[m.cursor].Priority = "A" + case "B": + m.tasks[m.cursor].Priority = "B" + case "C": + m.tasks[m.cursor].Priority = "C" + + // Remove completed tasks + case "u": + m.tasks = m.tasks.Filter(todo.FilterNotCompleted) + m.selected = make(map[int]struct{}) + for i, t := range m.tasks { + if t.Completed { + m.selected[i] = struct{}{} + } + } + + // The "up" and "k" keys move the cursor up + case "up", "k": + if m.cursor > 0 { + m.cursor-- + } + + // The "down" and "j" keys move the cursor down + case "down", "j": + if m.cursor < len(m.tasks) { + m.cursor++ + } + + // The "enter" key and the spacebar (a literal space) toggle + // the selected state for the item that the cursor is pointing at. + case "enter", " ": + _, ok := m.selected[m.cursor] + if ok { + delete(m.selected, m.cursor) + m.tasks[m.cursor].Reopen() + } else { + m.selected[m.cursor] = struct{}{} + m.tasks[m.cursor].Complete() + } + + // Switch interface to add task interface + case "a": + m.interfaceState = Add + + } + + return m, nil + + case Add: + switch msg.Type { + case tea.KeyCtrlC, tea.KeyEsc: + return m, tea.Quit + case tea.KeyEnter: + newTask, _ := todo.ParseTask(m.textInput.Value()) + m.tasks.AddTask(newTask) + m.textInput.Reset() + m.interfaceState = List + } + + m.textInput, cmd = m.textInput.Update(msg) + + return m, cmd + + } + } + + // Return the updated model to the Bubble Tea runtime for processing. + // Note that we're not returning a command. + return m, nil +} + +func (m model) View() string { + output := "My tasks list\n\n" + + switch m.interfaceState { + + case List: + + // Iterate over our tasks + for i, task := range m.tasks { + + cursor := " " + if m.cursor == i { + cursor = ">" + } + + // Is this choice selected? + checked := " " // not selected + if _, ok := m.selected[i]; ok { + checked = "x" // selected! + } + + // Render the row + styles := NewTextStyle() + priorityStyle := getPriorityStyle(styles, task.Priority) + output += fmt.Sprintf("%s [%s] %s %s\n", cursor, checked, priorityStyle.Render(task.Priority), task.Todo) + + } + + output += "\nPress q to quit.\n" + case Add: + output += fmt.Sprintf("Nouvelle tâche:\n\n%s", m.textInput.View()) + output += "\n\n Press ESC to quit.\n" + } + + return output + +} + +func main() { + p := tea.NewProgram(initialModel()) + if _, err := p.Run(); err != nil { + fmt.Printf("Alas, there's been an error: %v", err) + os.Exit(1) + } +}
A style.go

@@ -0,0 +1,40 @@

+package main + +import "github.com/charmbracelet/lipgloss" + +type Style struct { + Priority PriorityStyle +} + +type PriorityStyle struct { + A lipgloss.Style + B lipgloss.Style + C lipgloss.Style +} + +func NewTextStyle() Style { + + priorityStyle := PriorityStyle{ + A: lipgloss.NewStyle().Foreground(lipgloss.Color("#f38ba8")), + B: lipgloss.NewStyle().Foreground(lipgloss.Color("#fab387")), + C: lipgloss.NewStyle().Foreground(lipgloss.Color("#94e2d5")), + } + + style := Style{ + Priority: priorityStyle, + } + return style +} + +func getPriorityStyle(style Style, priority string) lipgloss.Style { + switch priority { + case "A": + return style.Priority.A + case "B": + return style.Priority.B + case "C": + return style.Priority.C + default: + return lipgloss.NewStyle() + } +}