Filtering on tags, Rounding on 'time'

This commit is contained in:
2025-12-04 14:35:16 -06:00
parent b30d663eb1
commit 729084ae74
5 changed files with 74 additions and 18 deletions

4
.tmuxrc Normal file
View File

@@ -0,0 +1,4 @@
rename-window dev
neww -n build
neww -n run
select-window -t 0:0

View File

@@ -1,6 +1,5 @@
/* /*
Copyright © 2022 Brian Buller <brian@bullercodeworks.com> Copyright © 2022 Brian Buller <brian@bullercodeworks.com>
*/ */
package cmd package cmd
@@ -74,6 +73,7 @@ func opListTimers(cmd *cobra.Command, args []string) error {
} }
filter := util.BuildFilterFromArgs(args) filter := util.BuildFilterFromArgs(args)
list = list.Filter(filter) list = list.Filter(filter)
list.Sort(timertxt.SortFinishDateAsc)
dayTotals := make(map[string]time.Duration) dayTotals := make(map[string]time.Duration)
for _, v := range list.GetTimerSlice() { for _, v := range list.GetTimerSlice() {

View File

@@ -1,6 +1,5 @@
/* /*
Copyright © 2022 Brian Buller <brian@bullercodeworks.com> Copyright © 2022 Brian Buller <brian@bullercodeworks.com>
*/ */
package cmd package cmd
@@ -10,6 +9,7 @@ import (
"git.bullercodeworks.com/brian/gime/cli" "git.bullercodeworks.com/brian/gime/cli"
"git.bullercodeworks.com/brian/gime/util" "git.bullercodeworks.com/brian/gime/util"
"git.bullercodeworks.com/brian/go-timertxt"
"github.com/spf13/cobra" "github.com/spf13/cobra"
) )
@@ -22,25 +22,58 @@ var timeCmd = &cobra.Command{
func init() { func init() {
rootCmd.AddCommand(timeCmd) rootCmd.AddCommand(timeCmd)
// Local Flags
timeCmd.Flags().BoolP("all", "a", false, "Include done.txt timers")
timeCmd.Flags().BoolP("done", "d", false, "Only done.txt timers")
} }
func opShowTimers(cmd *cobra.Command, args []string) error { func opShowTimers(cmd *cobra.Command, args []string) error {
var err error var err error
var includeArchive bool
var onlyArchive bool
if onlyArchive, err = cmd.Flags().GetBool("done"); err != nil {
return err
} else if includeArchive, err = cmd.Flags().GetBool("all"); err != nil {
return err
}
p := cli.Program{} p := cli.Program{}
err = p.Initialize() err = p.Initialize()
if err != nil { if err != nil {
return err return err
} }
if !onlyArchive {
if err := p.LoadTimerList(); err != nil { if err := p.LoadTimerList(); err != nil {
return err return err
} }
list := p.GetFilteredTimerList(args) }
if includeArchive || onlyArchive {
if err := p.LoadDoneList(); err != nil {
return err
}
}
list := timertxt.NewTimerList()
if !onlyArchive {
list = p.TimerList
if includeArchive {
for _, tmr := range p.DoneList.GetTimerSlice() {
list.AddTimer(tmr)
}
}
} else {
list = p.DoneList
}
filter := util.BuildFilterFromArgs(args)
list = list.Filter(filter)
list.Sort(timertxt.SortFinishDateAsc)
var isActive bool var isActive bool
var total time.Duration var total time.Duration
for _, v := range list.GetTimerSlice() { for _, v := range list.GetTimerSlice() {
dur := v.FinishDate.Sub(v.StartDate) dur := util.Round(v.FinishDate.Sub(v.StartDate))
if v.FinishDate.IsZero() { if v.FinishDate.IsZero() {
dur = time.Now().Sub(v.StartDate) dur = util.Round(time.Now().Sub(v.StartDate))
isActive = true isActive = true
} }
total += dur total += dur

View File

@@ -11,6 +11,7 @@ import (
*/ */
type PromptForTagWiddle struct { type PromptForTagWiddle struct {
active bool active bool
visible bool
x, y, w, h int x, y, w, h int
origKey, origVal string origKey, origVal string
@@ -36,8 +37,8 @@ func NewPromptForTagWiddle(x, y, w, h int, key, val string) *PromptForTagWiddle
origKey: key, origVal: val, origKey: key, origVal: val,
keyInput: keyInp, keyInput: keyInp,
valInput: widdles.NewToggleField("Value", val, 0, 0, 0, 0), valInput: widdles.NewToggleField("Value", val, 0, 0, 0, 0),
cancelButton: widdles.NewButton("Cancel", 0, 0, 0, 0), cancelButton: widdles.NewButton("Cancel"),
doneButton: widdles.NewButton("Done", 0, 0, 0, 0), doneButton: widdles.NewButton("Done"),
} }
} }
@@ -86,6 +87,8 @@ func (w *PromptForTagWiddle) View(style wandle.Style) {
wandle.Print(w.x+1, w.y+w.h-2, style, w.msg) wandle.Print(w.x+1, w.y+w.h-2, style, w.msg)
} }
func (w *PromptForTagWiddle) SetVisible(v bool) { w.visible = v }
func (w *PromptForTagWiddle) IsVisible() bool { return w.visible }
func (w *PromptForTagWiddle) IsActive() bool { return w.active } func (w *PromptForTagWiddle) IsActive() bool { return w.active }
func (w *PromptForTagWiddle) SetActive(b bool) { w.active = b } func (w *PromptForTagWiddle) SetActive(b bool) { w.active = b }
func (w *PromptForTagWiddle) Focusable() bool { return true } func (w *PromptForTagWiddle) Focusable() bool { return true }

View File

@@ -102,9 +102,11 @@ func DurationToDecimal(dur time.Duration) float64 {
mins := dur.Minutes() - (dur.Hours() * 60) mins := dur.Minutes() - (dur.Hours() * 60)
return dur.Hours() + (mins / 60) return dur.Hours() + (mins / 60)
} }
func AddDurations(dur1, dur2 time.Duration) time.Duration { func AddDurations(dur1, dur2 time.Duration) time.Duration {
return time.Duration(int64(dur1) + int64(dur2)) return time.Duration(int64(dur1) + int64(dur2))
} }
func SubDurations(dur1, dur2 time.Duration) time.Duration { func SubDurations(dur1, dur2 time.Duration) time.Duration {
return time.Duration(int64(dur1) - int64(dur2)) return time.Duration(int64(dur1) - int64(dur2))
} }
@@ -359,10 +361,12 @@ func BuildFilterFromArgs(args []string) func(*timertxt.Timer) bool {
end := time.Now() end := time.Now()
var contextFilters []string var contextFilters []string
var projectFilters []string var projectFilters []string
var tagFilters []string
var allFilters []func(timertxt.Timer) bool var allFilters []func(timertxt.Timer) bool
if len(args) > 0 { if len(args) > 0 {
contextFilters, args = GetContextsFromSlice(args) contextFilters, args = GetContextsFromSlice(args)
projectFilters, args = GetProjectsFromSlice(args) projectFilters, args = GetProjectsFromSlice(args)
tagFilters, args = GetAdditionalTagsFromSlice(args)
} }
if len(args) > 0 { if len(args) > 0 {
var err error var err error
@@ -394,22 +398,33 @@ func BuildFilterFromArgs(args []string) func(*timertxt.Timer) bool {
allFilters = append(allFilters, func(t timertxt.Timer) bool { allFilters = append(allFilters, func(t timertxt.Timer) bool {
for _, v := range contextFilters { for _, v := range contextFilters {
v = strings.TrimPrefix(v, "@") v = strings.TrimPrefix(v, "@")
if !t.HasContext(v) { if t.HasContext(v) {
return false
}
}
return true return true
}
}
return false
}) })
} }
if len(projectFilters) > 0 { if len(projectFilters) > 0 {
allFilters = append(allFilters, func(t timertxt.Timer) bool { allFilters = append(allFilters, func(t timertxt.Timer) bool {
for _, v := range projectFilters { for _, v := range projectFilters {
v = strings.TrimPrefix(v, "+") v = strings.TrimPrefix(v, "+")
if !t.HasProject(v) { if t.HasProject(v) {
return false
}
}
return true return true
}
}
return false
})
}
if len(tagFilters) > 0 {
allFilters = append(allFilters, func(t timertxt.Timer) bool {
for _, v := range tagFilters {
pts := strings.Split(v, ":")
if len(pts) == 2 && t.HasTag(pts[0]) && t.GetTag(pts[0]) == pts[1] {
return true
}
}
return false
}) })
} }
doFilters := func(t *timertxt.Timer) bool { doFilters := func(t *timertxt.Timer) bool {
@@ -441,6 +456,7 @@ func StringSliceContains(sl []string, val string) bool {
} }
return false return false
} }
func AppendStringIfDistinct(sl []string, val string) []string { func AppendStringIfDistinct(sl []string, val string) []string {
if !StringSliceContains(sl, val) { if !StringSliceContains(sl, val) {
return append(sl, val) return append(sl, val)