main.go (view raw)
1package main
2
3import (
4 "fmt"
5 "log"
6 "os"
7 "strings"
8
9 todo "github.com/1set/todotxt"
10 "github.com/charmbracelet/bubbles/help"
11 "github.com/charmbracelet/bubbles/key"
12 "github.com/charmbracelet/bubbles/textinput"
13 tea "github.com/charmbracelet/bubbletea"
14)
15
16type InterfaceState int
17
18const (
19 List InterfaceState = iota
20 Add
21)
22
23type model struct {
24 interfaceState InterfaceState
25 config config
26 keys keyMap
27 help help.Model
28
29 tasks todo.TaskList
30 cursor int // which to-do list item our cursor is pointing at
31 selected map[int]struct{} // which to-do items are selected
32
33 textInput textinput.Model
34}
35
36func initialModel() model {
37
38 config := NewConfig()
39
40 // New task input
41 ti := textinput.New()
42 ti.Focus()
43 ti.CharLimit = 200
44 ti.Width = 200
45
46 if tasklist, err := todo.LoadFromPath(config.taskFilePath); err != nil {
47 log.Fatal(err)
48 return model{}
49 } else {
50 tasks := tasklist
51 tasks.Sort(todo.SortCompletedDateAsc, todo.SortDueDateAsc)
52 selected := make(map[int]struct{})
53
54 interfaceState := List
55 if len(tasks) == 0 {
56 interfaceState = Add
57 }
58
59 for i, t := range tasks {
60 if t.Completed {
61 selected[i] = struct{}{}
62 }
63 }
64
65 return model{
66 interfaceState: interfaceState,
67 config: config,
68 tasks: tasks,
69 selected: selected,
70 textInput: ti,
71 keys: keys,
72 help: help.New(),
73 }
74
75 }
76
77}
78
79func (m model) Init() tea.Cmd {
80 return textinput.Blink
81}
82
83func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
84 var cmd tea.Cmd
85
86 switch msg := msg.(type) {
87
88 case tea.WindowSizeMsg:
89 // If we set a width on the help menu it can gracefully truncate
90 // its view as needed.
91 m.help.Width = msg.Width
92
93 // Is it a key press?
94 case tea.KeyMsg:
95
96 switch m.interfaceState {
97
98 case List:
99
100 switch {
101 // Navigation
102 case key.Matches(msg, m.keys.Up):
103 if m.cursor > 0 {
104 m.cursor--
105 }
106
107 case key.Matches(msg, m.keys.Down):
108 if m.cursor < len(m.tasks)-1 {
109 m.cursor++
110 }
111
112 // Tasks management
113 case key.Matches(msg, m.keys.Priority):
114 m.tasks[m.cursor].Priority = strings.ToUpper(msg.String())
115
116 case key.Matches(msg, m.keys.ClearPriority):
117 m.tasks[m.cursor].Priority = ""
118
119 case key.Matches(msg, m.keys.Check):
120 _, ok := m.selected[m.cursor]
121
122 if ok {
123 delete(m.selected, m.cursor)
124 m.tasks[m.cursor].Reopen()
125 } else {
126 m.selected[m.cursor] = struct{}{}
127 m.tasks[m.cursor].Complete()
128 }
129
130 case key.Matches(msg, m.keys.SortByPriority):
131 m.tasks.Sort(todo.SortCompletedDateAsc, todo.SortPriorityAsc)
132 m.selected = make(map[int]struct{})
133 for i, t := range m.tasks {
134 if t.Completed {
135 m.selected[i] = struct{}{}
136 }
137 }
138
139 case key.Matches(msg, m.keys.SortByDate):
140 m.tasks.Sort(todo.SortCompletedDateAsc, todo.SortDueDateAsc)
141 m.selected = make(map[int]struct{})
142 for i, t := range m.tasks {
143 if t.Completed {
144 m.selected[i] = struct{}{}
145 }
146 }
147
148 case key.Matches(msg, m.keys.Clean):
149 m.selected = make(map[int]struct{})
150 doneTasks := m.tasks.Filter(todo.FilterCompleted)
151 doneFile, err := os.OpenFile(m.config.doneFilePath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
152 if err != nil {
153 log.Fatal("Can't write done tasks to " + m.config.doneFilePath)
154 }
155 defer doneFile.Close()
156 if writeErr := doneTasks.WriteToFile(doneFile); writeErr != nil {
157 log.Fatal(writeErr)
158 }
159 m.tasks = m.tasks.Filter(todo.FilterNotCompleted)
160 m.selected = make(map[int]struct{})
161 for i, t := range m.tasks {
162 if t.Completed {
163 m.selected[i] = struct{}{}
164 }
165 }
166
167 // Interface
168 case key.Matches(msg, m.keys.Add):
169 m.interfaceState = Add
170
171 case key.Matches(msg, m.keys.SaveQuit):
172 m.tasks.WriteToPath(m.config.taskFilePath)
173 return m, tea.Quit
174
175 case key.Matches(msg, m.keys.Quit):
176 return m, tea.Quit
177
178 case key.Matches(msg, m.keys.Help):
179 m.help.ShowAll = !m.help.ShowAll
180
181 }
182
183 return m, nil
184
185 case Add:
186 switch msg.Type {
187 case tea.KeyCtrlC, tea.KeyEsc:
188 return m, tea.Quit
189 case tea.KeyEnter:
190 inputValue := m.textInput.Value()
191 if inputValue != "" {
192 newTask, _ := todo.ParseTask(inputValue)
193 m.tasks.AddTask(newTask)
194 }
195 m.textInput.Reset()
196 m.interfaceState = List
197 }
198
199 m.textInput, cmd = m.textInput.Update(msg)
200
201 return m, cmd
202
203 }
204 }
205
206 return m, nil
207}
208
209func (m model) View() string {
210 output := "\n"
211
212 switch m.interfaceState {
213
214 case List:
215 if len(m.tasks) > 0 {
216
217 for i, task := range m.tasks {
218
219 cursor := " "
220 if m.cursor == i {
221 cursor = "→"
222 }
223
224 // Render the row
225 _, checked := m.selected[i]
226 styles := NewTextStyle()
227 output += fmt.Sprintf("%s %s\n", cursor, styles.getTaskStyle(task, checked))
228 }
229
230 } else {
231 output += "No tasks in file"
232 }
233
234 output += "\n"
235 output += m.help.View(m.keys)
236 case Add:
237 output += fmt.Sprintf("Nouvelle tâche:\n\n%s", m.textInput.View())
238 output += "\n\n Press ESC to quit.\n"
239 }
240
241 return output
242
243}
244
245func main() {
246 p := tea.NewProgram(initialModel(), tea.WithAltScreen())
247 if _, err := p.Run(); err != nil {
248 fmt.Printf("Alas, there's been an error: %v", err)
249 os.Exit(1)
250 }
251}