gime/ui/list_timers.go

402 lines
10 KiB
Go
Raw Normal View History

2023-01-12 12:53:26 +00:00
package ui
import (
"fmt"
"strings"
"time"
"git.bullercodeworks.com/brian/gime/util"
"git.bullercodeworks.com/brian/go-timertxt"
"git.bullercodeworks.com/brian/wandle"
"git.bullercodeworks.com/brian/widdles"
"github.com/nsf/termbox-go"
)
2023-01-12 17:09:49 +00:00
const (
activeToggleActive = iota
activeToggleInactive
activeToggleAll
activeToggleErr
)
2023-01-12 12:53:26 +00:00
type listTimersScreen struct {
ui *Ui
initialized bool
menu *widdles.TopMenu
scrollbar *widdles.Scrollbar
cursor int
2023-01-12 17:09:49 +00:00
activeToggle int
fullList *timertxt.TimerList
timerList *timertxt.TimerList
doneList *timertxt.TimerList
2023-01-12 12:53:26 +00:00
2023-01-12 17:09:49 +00:00
fullFilterList *timertxt.TimerList
2023-01-12 12:53:26 +00:00
timerFilterList *timertxt.TimerList
doneFilterList *timertxt.TimerList
selected map[int]bool
inputDialog *widdles.InputDialog
filter string
2023-01-12 17:09:49 +00:00
choiceMenu *widdles.MenuV
tagEditor *PromptForTagWiddle
//partManager *PartManager
2023-01-12 12:53:26 +00:00
msg string
err error
}
type ListTimersMsg ScreenMsg
func NewListTimersScreen(u *Ui) *listTimersScreen {
w, h := termbox.Size()
s := listTimersScreen{
ui: u,
menu: widdles.NewTopMenu(0, 0, 0),
scrollbar: widdles.NewScrollbar(w-2, 2, 1, h-2),
selected: make(map[int]bool),
inputDialog: widdles.NewInputDialog("", ""),
2023-01-12 17:09:49 +00:00
choiceMenu: widdles.NewMenuV(0, 0, 0, 0),
tagEditor: NewPromptForTagWiddle(0, 0, widdles.AUTO_SIZE, widdles.AUTO_SIZE, "", ""),
2023-01-12 12:53:26 +00:00
}
return &s
}
func (s *listTimersScreen) Init() wandle.Cmd {
if s.initialized {
return nil
}
s.initialized = true
// Set up the top menu
fileMenu := s.menu.NewSubMenu("File")
settingsOption := widdles.NewMenuItem("Settings")
settingsOption.SetCommand(s.ui.GotoScreen(SettingsId))
fileMenu.AddOption(settingsOption)
quitOption := widdles.NewMenuItem("Quit")
quitOption.Hotkey = termbox.KeyCtrlC
quitOption.SetCommand(func() wandle.Msg { return wandle.Quit() })
fileMenu.AddOption(quitOption)
s.menu.Measure()
// Timer Lists
s.timerList, s.doneList = s.ui.program.TimerList, s.ui.program.DoneList
s.fullList = timertxt.NewTimerList()
s.fullList.AddTimers(s.timerList.GetTimerSlice())
s.fullList.AddTimers(s.doneList.GetTimerSlice())
2023-01-12 17:09:49 +00:00
s.fullFilterList = s.fullList
2023-01-12 12:53:26 +00:00
s.timerFilterList, s.doneFilterList = s.timerList, s.doneList
2023-01-12 17:09:49 +00:00
s.fullFilterList.Sort(timertxt.SORT_START_DATE_DESC)
2023-01-12 12:53:26 +00:00
s.timerFilterList.Sort(timertxt.SORT_START_DATE_DESC)
s.doneFilterList.Sort(timertxt.SORT_START_DATE_DESC)
w, h := termbox.Size()
2023-01-12 17:09:49 +00:00
s.choiceMenu.SetBorder(wandle.BRD_CSIMPLE)
s.choiceMenu.SetX((w / 2) - 7)
s.choiceMenu.SetY((h / 2) - 7)
s.choiceMenu.SetWidth(widdles.AUTO_SIZE)
s.choiceMenu.SetHeight(widdles.AUTO_SIZE)
s.choiceMenu.SetPadding(0, 1, 0, 1)
s.tagEditor.SetX(w / 4)
s.tagEditor.SetY(h / 4)
s.tagEditor.SetWidth(w / 2)
s.tagEditor.SetHeight(h / 2)
s.updateFullFilterList()
2023-01-12 12:53:26 +00:00
return nil
}
func (s *listTimersScreen) Update(msg wandle.Msg) wandle.Cmd {
switch msg := msg.(type) {
case ScreenMsg:
case termbox.Event:
return s.handleTermboxEvent(msg)
}
return nil
}
func (s *listTimersScreen) View(style wandle.Style) {
_, h := termbox.Size()
y := 2
2023-01-12 17:09:49 +00:00
printedTimers := 0
if s.activeToggle == activeToggleAll || s.activeToggle == activeToggleActive {
wandle.Print(1, y, style.Bold(true), "Active Timers")
y++
for idx, tmr := range s.timerFilterList.GetTimerSlice() {
if y > h-2 {
break
}
st := style
if s.cursor == idx {
st = st.Invert()
}
if s.selected[idx] {
wandle.Print(1, y, st, "[✔] ")
} else {
wandle.Print(1, y, st, "[ ] ")
}
s.ViewTimer(5, y, st, tmr)
y++
2023-01-12 12:53:26 +00:00
}
y++
2023-01-12 17:09:49 +00:00
printedTimers = s.timerFilterList.Size()
2023-01-12 12:53:26 +00:00
}
2023-01-12 17:09:49 +00:00
if s.activeToggle == activeToggleAll || s.activeToggle == activeToggleInactive {
wandle.Print(1, y, style.Bold(true), "Done Timers")
2023-01-12 12:53:26 +00:00
y++
2023-01-12 17:09:49 +00:00
for idx, tmr := range s.doneFilterList.GetTimerSlice() {
if y > h-3 {
break
}
st := style
if s.cursor == printedTimers+idx {
st = st.Invert()
}
if s.selected[printedTimers+idx] {
wandle.Print(1, y, st, "[✔] ")
} else {
wandle.Print(1, y, st, "[ ] ")
}
s.ViewTimer(5, y, st, tmr)
y++
}
2023-01-12 12:53:26 +00:00
}
2023-01-12 17:09:49 +00:00
wandle.Print(1, h-1, style, "[A]ctive toggle, [p]roject(+), [c]ontext(@), [t]ags(:)")
2023-01-12 12:53:26 +00:00
if len(s.selected) > 0 {
wandle.Print(39, h-1, style, fmt.Sprintf("(%d selected)", len(s.selected)))
}
s.scrollbar.View(style)
if s.menu.IsActive() {
s.menu.View(style)
}
2023-01-12 17:09:49 +00:00
if s.choiceMenu.IsActive() {
s.choiceMenu.View(style)
}
if s.tagEditor.IsActive() {
s.tagEditor.View(style)
}
wandle.Print(1, h-2, style, s.msg)
2023-01-12 12:53:26 +00:00
}
func (s *listTimersScreen) ViewTimer(x, y int, style wandle.Style, tmr *timertxt.Timer) {
var tags []string
for _, k := range util.SortedTagKeyList(tmr.AdditionalTags) {
tags = append(tags, fmt.Sprintf("%s:%s", k, tmr.AdditionalTags[k]))
}
wandle.Print(x, y, style, fmt.Sprintf("%s %s %s %s %s", tmr.StartDate.Format(time.Stamp), tmr.Duration(), tmr.Contexts, tmr.Projects, strings.Join(tags, "; ")))
}
func (s *listTimersScreen) handleTermboxEvent(msg termbox.Event) wandle.Cmd {
2023-01-12 17:09:49 +00:00
if s.choiceMenu.IsActive() {
return s.choiceMenu.Update(msg)
}
if s.tagEditor.IsActive() {
return s.tagEditor.Update(msg)
2023-01-12 12:53:26 +00:00
}
if (msg.Type == termbox.EventKey && msg.Key == termbox.KeyEsc) || s.menu.IsActive() {
return s.menu.Update(msg)
}
switch msg.Type {
case termbox.EventKey:
if msg.Key == termbox.KeyEnter {
2023-01-12 17:09:49 +00:00
// TODO: Edit the entry
2023-01-12 12:53:26 +00:00
/*
if s.cursor >= 0 && s.cursor < s.timerFilterList.Size()+s.doneFilterList.Size() {
} else {
}
*/
} else if msg.Key == termbox.KeySpace {
// (un)Select the entry
if v := s.selected[s.cursor]; v {
delete(s.selected, s.cursor)
} else {
s.selected[s.cursor] = true
}
2023-01-12 17:09:49 +00:00
if s.cursor < s.fullFilterList.Size()-1 {
s.cursor++
}
} else if msg.Ch == 'A' {
s.activeToggle = (s.activeToggle + 1) % activeToggleErr
s.updateFullFilterList()
2023-01-12 12:53:26 +00:00
} else if msg.Key == termbox.KeyArrowUp || msg.Ch == 'k' {
if s.cursor > 0 {
s.cursor--
} else {
s.cursor = 0
}
return nil
} else if msg.Key == termbox.KeyArrowDown || msg.Ch == 'j' {
2023-01-12 17:09:49 +00:00
if s.cursor < s.fullFilterList.Size()-1 {
2023-01-12 12:53:26 +00:00
s.cursor++
} else {
2023-01-12 17:09:49 +00:00
s.cursor = s.fullFilterList.Size() - 1
2023-01-12 12:53:26 +00:00
}
return nil
} else if msg.Ch == 'G' {
2023-01-12 17:09:49 +00:00
s.cursor = s.fullFilterList.Size() - 1
2023-01-12 12:53:26 +00:00
} else if msg.Ch == 't' {
// Edit tag(s)
2023-01-12 17:09:49 +00:00
return s.showEditTagsChoice()
2023-01-12 12:53:26 +00:00
} else if msg.Ch == 'p' {
// Edit project(s)
2023-01-12 17:09:49 +00:00
// TODO: Prompt for Choice: Add/Edit/Remove
2023-01-12 12:53:26 +00:00
projs := s.fullList.GetProjects()
2023-01-12 17:09:49 +00:00
_ = projs
2023-01-12 12:53:26 +00:00
} else if msg.Ch == 'c' {
// Edit context(s)
2023-01-12 17:09:49 +00:00
// TODO: Prompt for choice: Add/Edit/Remove
2023-01-12 12:53:26 +00:00
ctxts := s.fullList.GetContexts()
2023-01-12 17:09:49 +00:00
_ = ctxts
}
}
return nil
}
func (s *listTimersScreen) showEditTagsChoice() wandle.Cmd {
tags := s.getSelectedTimerTags()
var showTagEditor = func(key, val string) wandle.Cmd {
return func() wandle.Msg {
s.tagEditor.SetTag(key, val)
// TODO: Set Done & Cancel Commands
s.choiceMenu.SetActive(false)
s.tagEditor.SetActive(true)
return wandle.EmptyCmd
}
}
s.choiceMenu.SetTitle("")
s.choiceMenu.ClearOptions()
addTag := widdles.NewMenuItem("Add New Tag")
addTag.SetCommand(showTagEditor("", ""))
s.choiceMenu.AddOption(addTag)
editTag := widdles.NewMenuItem("Edit Tag")
editTag.SetEnabled(len(tags) > 0)
editTag.SetCommand(func() wandle.Msg {
s.choiceMenu.ClearOptions()
s.choiceMenu.SetTitle("Choose Tag to Edit")
for k, v := range tags {
var vals string
if len(v) == 1 {
vals = v[0]
2023-01-12 12:53:26 +00:00
} else {
2023-01-12 17:09:49 +00:00
vals = "..."
2023-01-12 12:53:26 +00:00
}
2023-01-12 17:09:49 +00:00
opt := widdles.NewMenuItem(fmt.Sprintf("%s (%s)", k, vals))
opt.SetCommand(showTagEditor(k, vals))
s.choiceMenu.AddOption(opt)
2023-01-12 12:53:26 +00:00
}
2023-01-12 17:09:49 +00:00
return wandle.EmptyCmd
})
s.choiceMenu.AddOption(editTag)
removeTag := widdles.NewMenuItem("Remove Tag")
removeTag.SetCommand(func() wandle.Msg {
s.choiceMenu.ClearOptions()
s.choiceMenu.SetTitle("Choose Tag to Remove")
return wandle.EmptyCmd
})
s.choiceMenu.AddOption(removeTag)
s.choiceMenu.SetActive(true)
//tags := s.fullList.GetTagKeys()
//_ = tags
return wandle.EmptyCmd
}
func (s *listTimersScreen) updateFullFilterList() {
s.fullFilterList = timertxt.NewTimerList()
switch s.activeToggle {
case activeToggleAll:
s.fullFilterList.Combine(s.timerFilterList)
s.fullFilterList.Combine(s.doneFilterList)
case activeToggleActive:
s.fullFilterList.Combine(s.timerFilterList)
case activeToggleInactive:
s.fullFilterList.Combine(s.doneFilterList)
}
if s.cursor >= s.fullFilterList.Size() {
s.cursor = s.fullFilterList.Size() - 1
2023-01-12 12:53:26 +00:00
}
}
func (s *listTimersScreen) gotoSettingsScreen() wandle.Msg {
return ScreenMsg{
source: ListTimersId,
command: CmdGotoSettings,
}
}
2023-01-12 17:09:49 +00:00
func (s *listTimersScreen) getSelectedTimerTags() map[string][]string {
ret := make(map[string][]string)
selected := len(s.selected)
var selTimer *timertxt.Timer
if selected == 0 {
if s.cursor < s.fullFilterList.Size() {
selTimer, s.err = s.fullFilterList.GetTimer(s.cursor)
for k, v := range selTimer.AdditionalTags {
ret[k] = []string{v}
}
}
} else {
for i := range s.selected {
if tmr, err := s.fullFilterList.GetTimer(i); err == nil {
for k, v := range tmr.AdditionalTags {
if !util.StringSliceContains(ret[k], v) {
ret[k] = append(ret[k], v)
}
}
}
}
}
return ret
}
func (s *listTimersScreen) getSelectedTimerProjects() []string {
var ret []string
selected := len(s.selected)
var selTimer *timertxt.Timer
if selected == 0 {
if s.cursor < s.fullFilterList.Size() {
selTimer, s.err = s.fullFilterList.GetTimer(s.cursor)
for _, v := range selTimer.Projects {
ret = append(ret, v)
}
}
} else {
for i := range s.selected {
if tmr, err := s.fullFilterList.GetTimer(i); err == nil {
for _, v := range tmr.Projects {
if !util.StringSliceContains(ret, v) {
ret = append(ret, v)
}
}
}
}
}
return ret
}
func (s *listTimersScreen) getSelectedTimerContexts() []string {
var ret []string
selected := len(s.selected)
var selTimer *timertxt.Timer
if selected == 0 {
if s.cursor < s.fullFilterList.Size() {
selTimer, s.err = s.fullFilterList.GetTimer(s.cursor)
for _, v := range selTimer.Contexts {
ret = append(ret, v)
}
}
} else {
for i := range s.selected {
if tmr, err := s.fullFilterList.GetTimer(i); err == nil {
for _, v := range tmr.Contexts {
if !util.StringSliceContains(ret, v) {
ret = append(ret, v)
}
}
}
}
}
return ret
}