summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPaul Bütow <1224732+snonux@users.noreply.github.com>2025-06-20 20:59:49 +0300
committerGitHub <noreply@github.com>2025-06-20 20:59:49 +0300
commit3670a1b978e84f74d210b2432fa6b10d3b29d4ce (patch)
treed51d28b85aef6f74a6ec4f0525dcdc4af2f6ea95
parentbd44e7e81e9808c875340806eccefc043e27a8b7 (diff)
parent15fea05b5244f39e686f30770482b99687ac66c7 (diff)
Merge pull request #37 from snonux/codex/add-hotkey-to-set-random-due-date
Add random due date hotkey
-rw-r--r--internal/ui/table.go16
-rw-r--r--internal/ui/table_test.go56
2 files changed, 72 insertions, 0 deletions
diff --git a/internal/ui/table.go b/internal/ui/table.go
index fc1540c..bd1d4ca 100644
--- a/internal/ui/table.go
+++ b/internal/ui/table.go
@@ -2,6 +2,7 @@ package ui
import (
"fmt"
+ "math/rand"
"strconv"
"strings"
"time"
@@ -16,6 +17,10 @@ import (
"tasksamurai/internal/task"
)
+func init() {
+ rand.Seed(time.Now().UnixNano())
+}
+
// Model wraps a Bubble Tea table.Model to display tasks.
type Model struct {
tbl atable.Model
@@ -236,6 +241,16 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
return m, nil
}
}
+ case "r":
+ if row := m.tbl.SelectedRow(); row != nil {
+ idStr := ansi.Strip(row[0])
+ if id, err := strconv.Atoi(idStr); err == nil {
+ days := rand.Intn(31) + 7
+ due := time.Now().AddDate(0, 0, days).Format("2006-01-02")
+ task.SetDueDate(id, due)
+ m.reload()
+ }
+ }
case "a":
if row := m.tbl.SelectedRow(); row != nil {
idStr := ansi.Strip(row[0])
@@ -281,6 +296,7 @@ func (m Model) View() string {
"s: toggle start/stop",
"D: mark task done",
"d: set due date",
+ "r: random due date",
"a: annotate task",
"A: replace annotations",
"q: quit",
diff --git a/internal/ui/table_test.go b/internal/ui/table_test.go
index 061b418..f02d587 100644
--- a/internal/ui/table_test.go
+++ b/internal/ui/table_test.go
@@ -5,6 +5,7 @@ import (
"path/filepath"
"strings"
"testing"
+ "time"
tea "github.com/charmbracelet/bubbletea"
)
@@ -223,3 +224,58 @@ func TestDueDateHotkey(t *testing.T) {
t.Fatalf("due not set: %q", data)
}
}
+
+func TestRandomDueDateHotkey(t *testing.T) {
+ tmp := t.TempDir()
+ taskPath := filepath.Join(tmp, "task")
+ dueFile := filepath.Join(tmp, "due.txt")
+
+ script := "#!/bin/sh\n" +
+ "if echo \"$@\" | grep -q export; then\n" +
+ " echo '{\"id\":1,\"uuid\":\"x\",\"description\":\"d\",\"status\":\"pending\",\"entry\":\"\",\"priority\":\"\",\"urgency\":0}'\n" +
+ " exit 0\n" +
+ "fi\n" +
+ "echo \"$@\" > " + dueFile + "\n"
+
+ if err := os.WriteFile(taskPath, []byte(script), 0o755); err != nil {
+ t.Fatal(err)
+ }
+
+ origPath := os.Getenv("PATH")
+ os.Setenv("PATH", tmp+":"+origPath)
+ t.Cleanup(func() { os.Setenv("PATH", origPath) })
+
+ os.Setenv("TASKDATA", tmp)
+ os.Setenv("TASKRC", "/dev/null")
+ t.Cleanup(func() {
+ os.Unsetenv("TASKDATA")
+ os.Unsetenv("TASKRC")
+ })
+
+ m, err := New("")
+ if err != nil {
+ t.Fatalf("New: %v", err)
+ }
+
+ mv, _ := m.Update(tea.KeyMsg{Type: tea.KeyRunes, Runes: []rune{'r'}})
+ m = mv.(Model)
+
+ data, err := os.ReadFile(dueFile)
+ if err != nil {
+ t.Fatalf("read due: %v", err)
+ }
+
+ parts := strings.Split(strings.TrimSpace(string(data)), " ")
+ if len(parts) != 3 {
+ t.Fatalf("unexpected command: %q", data)
+ }
+ dueStr := strings.TrimPrefix(parts[2], "due:")
+ dueTime, err := time.Parse("2006-01-02", dueStr)
+ if err != nil {
+ t.Fatalf("parse due: %v", err)
+ }
+ days := int(time.Until(dueTime).Hours() / 24)
+ if days < 7 || days > 37 {
+ t.Fatalf("due date out of range: %d", days)
+ }
+}