Working on UI
This commit is contained in:
		
							
								
								
									
										5
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| # This how we want to name the binary output | ||||
| # This is what we want to name the binary output | ||||
| BINARY=gime | ||||
|  | ||||
| # These are the values we want to pass for VERSION and BUILD | ||||
| @@ -14,5 +14,8 @@ LDFLAGS=-ldflags "-w -s -X cmd.Version=${VERSION} -X cmd.Build=${BUILD}" | ||||
| gime: | ||||
| 	go build ${LDFLAGS} -o build/${BINARY} | ||||
|  | ||||
| runui: | ||||
| 	cd build && ./gime ui | ||||
|  | ||||
| clean: | ||||
| 	rm build/* | ||||
|   | ||||
| @@ -4,8 +4,8 @@ Copyright © 2022 Brian Buller <brian@bullercodeworks.com> | ||||
| package cmd | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
|  | ||||
| 	"git.bullercodeworks.com/brian/gime/cli" | ||||
| 	"git.bullercodeworks.com/brian/gime/ui" | ||||
| 	"github.com/spf13/cobra" | ||||
| ) | ||||
|  | ||||
| @@ -21,6 +21,6 @@ func init() { | ||||
| } | ||||
|  | ||||
| func opUi(cmd *cobra.Command, args []string) error { | ||||
| 	fmt.Println("ui called") | ||||
| 	return nil | ||||
| 	p := &cli.Program{} | ||||
| 	return ui.RunTUI(p) | ||||
| } | ||||
|   | ||||
							
								
								
									
										8
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,11 +1,17 @@ | ||||
| module git.bullercodeworks.com/brian/gime | ||||
|  | ||||
| go 1.17 | ||||
| go 1.18 | ||||
|  | ||||
| replace git.bullercodeworks.com/brian/go-timertxt => /home/brbuller/Development/go/src/git.bullercodeworks.com/brian/go-timertxt | ||||
|  | ||||
| replace git.bullercodeworks.com/brian/wandle => /home/brbuller/Development/go/src/git.bullercodeworks.com/brian/wandle | ||||
|  | ||||
| replace git.bullercodeworks.com/brian/widdles => /home/brbuller/Development/go/src/git.bullercodeworks.com/brian/widdles | ||||
|  | ||||
| require ( | ||||
| 	git.bullercodeworks.com/brian/go-timertxt v0.0.0-20210302170637-d35b67037e23 | ||||
| 	git.bullercodeworks.com/brian/wandle v1.0.3 | ||||
| 	git.bullercodeworks.com/brian/widdles v0.0.0-00010101000000-000000000000 | ||||
| 	github.com/br0xen/termbox-util v0.0.0-20200220160819-dc6d6950ba00 | ||||
| 	github.com/br0xen/user-config v0.0.0-20170914134719-16e743ec93a2 | ||||
| 	github.com/muesli/go-app-paths v0.2.2 | ||||
|   | ||||
							
								
								
									
										168
									
								
								ui/list_timers.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								ui/list_timers.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,168 @@ | ||||
| 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" | ||||
| ) | ||||
|  | ||||
| type listTimersScreen struct { | ||||
| 	ui *Ui | ||||
|  | ||||
| 	initialized bool | ||||
| 	menu        *widdles.TopMenu | ||||
| 	scrollbar   *widdles.Scrollbar | ||||
|  | ||||
| 	cursor int | ||||
|  | ||||
| 	timerList *timertxt.TimerList | ||||
| 	doneList  *timertxt.TimerList | ||||
|  | ||||
| 	timerFilterList *timertxt.TimerList | ||||
| 	doneFilterList  *timertxt.TimerList | ||||
|  | ||||
| 	filter string | ||||
|  | ||||
| 	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), | ||||
| 	} | ||||
| 	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.timerFilterList, s.doneFilterList = s.timerList, s.doneList | ||||
| 	s.timerFilterList.Sort(timertxt.SORT_START_DATE_DESC) | ||||
| 	s.doneFilterList.Sort(timertxt.SORT_START_DATE_DESC) | ||||
|  | ||||
| 	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() | ||||
| 	if s.menu.IsActive() { | ||||
| 		s.menu.View(style) | ||||
| 	} | ||||
| 	y := 2 | ||||
| 	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() | ||||
| 		} | ||||
| 		wandle.Print(1, y, st, "[ ]") | ||||
| 		s.ViewTimer(5, y, st, tmr) | ||||
| 		y++ | ||||
| 	} | ||||
| 	y++ | ||||
| 	wandle.Print(1, y, style.Bold(true), "Done Timers") | ||||
| 	y++ | ||||
| 	for idx, tmr := range s.doneFilterList.GetTimerSlice() { | ||||
| 		if y > h-2 { | ||||
| 			break | ||||
| 		} | ||||
| 		st := style | ||||
| 		if s.cursor == s.timerFilterList.Size()+idx { | ||||
| 			st = st.Invert() | ||||
| 		} | ||||
| 		wandle.Print(1, y, st, "[ ]") | ||||
| 		s.ViewTimer(5, y, st, tmr) | ||||
| 		y++ | ||||
| 	} | ||||
|  | ||||
| 	s.scrollbar.View(style) | ||||
| } | ||||
|  | ||||
| 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 { | ||||
| 	if (msg.Type == termbox.EventKey && msg.Key == termbox.KeyEsc) || s.menu.IsActive() { | ||||
| 		return s.menu.Update(msg) | ||||
| 	} | ||||
| 	switch msg.Type { | ||||
| 	case termbox.EventKey: | ||||
| 		top := s.timerFilterList.Size() + s.doneFilterList.Size() - 2 | ||||
| 		if msg.Key == termbox.KeyEnter { | ||||
| 			/* | ||||
| 				if s.cursor >= 0 && s.cursor < s.timerFilterList.Size()+s.doneFilterList.Size() { | ||||
| 				} else { | ||||
| 				} | ||||
| 			*/ | ||||
| 		} 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' { | ||||
| 			if s.cursor < top { | ||||
| 				s.cursor++ | ||||
| 			} else { | ||||
| 				s.cursor = top | ||||
| 			} | ||||
| 			return nil | ||||
| 		} else if msg.Ch == 'G' { | ||||
| 			s.cursor = top | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *listTimersScreen) gotoSettingsScreen() wandle.Msg { | ||||
| 	return ScreenMsg{ | ||||
| 		source:  ListTimersId, | ||||
| 		command: CmdGotoSettings, | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										71
									
								
								ui/screen_settings.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								ui/screen_settings.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| package ui | ||||
|  | ||||
| import ( | ||||
| 	"git.bullercodeworks.com/brian/wandle" | ||||
| 	"git.bullercodeworks.com/brian/widdles" | ||||
| 	"github.com/nsf/termbox-go" | ||||
| ) | ||||
|  | ||||
| type settingsScreen struct { | ||||
| 	ui          *Ui | ||||
| 	initialized bool | ||||
| 	menu        *widdles.TopMenu | ||||
| } | ||||
|  | ||||
| type SettingsMsg ScreenMsg | ||||
|  | ||||
| func NewSettingsScreen(u *Ui) *settingsScreen { | ||||
| 	return &settingsScreen{ | ||||
| 		ui:   u, | ||||
| 		menu: widdles.NewTopMenu(0, 0, 0), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *settingsScreen) Init() wandle.Cmd { | ||||
| 	if s.initialized { | ||||
| 		return nil | ||||
| 	} | ||||
| 	s.initialized = true | ||||
| 	// Set up the top menu | ||||
| 	fileMenu := s.menu.NewSubMenu("File") | ||||
| 	quitOption := widdles.NewMenuItem("Quit") | ||||
| 	quitOption.Hotkey = termbox.KeyCtrlC | ||||
| 	quitOption.SetCommand(func() wandle.Msg { return wandle.Quit() }) | ||||
| 	fileMenu.AddOption(quitOption) | ||||
| 	gotoMenu := s.menu.NewSubMenu("Goto") | ||||
| 	timerListOption := widdles.NewMenuItem("Timer List") | ||||
| 	timerListOption.SetCommand(s.ui.GotoScreen(ListTimersId)) | ||||
| 	gotoMenu.AddOption(timerListOption) | ||||
| 	s.menu.Measure() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *settingsScreen) Update(msg wandle.Msg) wandle.Cmd { | ||||
| 	switch msg := msg.(type) { | ||||
| 	case ScreenMsg: | ||||
| 	case termbox.Event: | ||||
| 		return s.handleTermboxEvent(msg) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *settingsScreen) View(style wandle.Style) { | ||||
| 	wandle.Print(1, 1, style, "Settings") | ||||
| 	if s.menu.IsActive() { | ||||
| 		s.menu.View(style) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *settingsScreen) handleTermboxEvent(msg termbox.Event) wandle.Cmd { | ||||
| 	if (msg.Type == termbox.EventKey && msg.Key == termbox.KeyEsc) || s.menu.IsActive() { | ||||
| 		return s.menu.Update(msg) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *settingsScreen) gotoTimerList() wandle.Msg { | ||||
| 	return ScreenMsg{ | ||||
| 		source:  SettingsId, | ||||
| 		command: CmdGotoTimerList, | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										128
									
								
								ui/ui.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								ui/ui.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | ||||
| package ui | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"time" | ||||
|  | ||||
| 	"git.bullercodeworks.com/brian/gime/cli" | ||||
| 	"git.bullercodeworks.com/brian/wandle" | ||||
| 	"github.com/nsf/termbox-go" | ||||
| ) | ||||
|  | ||||
| // Screen/Source Ids | ||||
| const ( | ||||
| 	ListTimersId = ScreenMsgSource(iota << 8) | ||||
| 	SettingsId | ||||
| 	ErrorId | ||||
| ) | ||||
|  | ||||
| // Commands | ||||
| const ( | ||||
| 	CmdCanceled = ScreenMsgCommand(iota) | ||||
| 	CmdSaved | ||||
|  | ||||
| 	// Goto Screen Commands | ||||
| 	CmdGotoSettings | ||||
| 	CmdGotoTimerList | ||||
| ) | ||||
|  | ||||
| func RunTUI(p *cli.Program) error { | ||||
| 	ui := NewUi(p) | ||||
| 	ui.debug = true | ||||
| 	if err := ui.Start(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	// Exiting | ||||
| 	fmt.Printf("Done\n") | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| type Ui struct { | ||||
| 	debug   bool | ||||
| 	wandle  *wandle.Program | ||||
| 	program *cli.Program | ||||
| 	err     error | ||||
|  | ||||
| 	screens map[ScreenMsgSource]wandle.Screen | ||||
|  | ||||
| 	prevScreen, currScreen ScreenMsgSource | ||||
| } | ||||
|  | ||||
| func NewUi(p *cli.Program) *Ui { | ||||
| 	ui := &Ui{ | ||||
| 		screens: make(map[ScreenMsgSource]wandle.Screen), | ||||
| 		program: p, | ||||
| 		err:     p.Initialize(), | ||||
| 	} | ||||
| 	var s wandle.Screen | ||||
| 	var sId ScreenMsgSource | ||||
| 	if ui.err != nil { | ||||
| 		s, sId = NewSettingsScreen(ui), SettingsId | ||||
| 	} else { | ||||
| 		if ui.err = ui.program.LoadTimerList(); ui.err != nil { | ||||
| 			s, sId = NewSettingsScreen(ui), SettingsId | ||||
| 		} else if ui.err = ui.program.LoadDoneList(); ui.err != nil { | ||||
| 			s, sId = NewSettingsScreen(ui), SettingsId | ||||
| 		} else { | ||||
| 			s, sId = NewListTimersScreen(ui), ListTimersId | ||||
| 		} | ||||
| 	} | ||||
| 	ui.screens[sId] = s | ||||
| 	ui.currScreen = sId | ||||
| 	ui.wandle = wandle.NewProgram(s) | ||||
| 	ui.wandle.Style(wandle.NewStyle( | ||||
| 		termbox.RGBToAttribute(uint8(0), uint8(255), uint8(0)), | ||||
| 		termbox.RGBToAttribute(uint8(0), uint8(0), uint8(0)), | ||||
| 	)) | ||||
| 	return ui | ||||
| } | ||||
|  | ||||
| func (u *Ui) GotoScreen(id ScreenMsgSource) func() wandle.Msg { | ||||
| 	u.prevScreen, u.currScreen = u.currScreen, id | ||||
| 	if s, ok := u.screens[id]; ok { | ||||
| 		return wandle.SwitchScreenCmd(s) | ||||
| 	} | ||||
| 	var s wandle.Screen | ||||
| 	switch id { | ||||
| 	case ListTimersId: | ||||
| 		s = NewListTimersScreen(u) | ||||
| 	case SettingsId: | ||||
| 		s = NewSettingsScreen(u) | ||||
| 	} | ||||
| 	u.screens[id] = s | ||||
| 	return wandle.SwitchScreenCmd(s) | ||||
| } | ||||
| func (u *Ui) CanGoBack() bool { | ||||
| 	return u.prevScreen > 0 | ||||
| } | ||||
| func (u *Ui) GoBack() func() wandle.Msg { | ||||
| 	if !u.CanGoBack() { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return u.GotoScreen(u.prevScreen) | ||||
| } | ||||
|  | ||||
| func (u *Ui) Log(v string) error { | ||||
| 	f, err := os.OpenFile("gime.log", os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| 	defer f.Close() | ||||
| 	_, err = f.WriteString(fmt.Sprintf("%s: %s\n", time.Now().Format(time.RFC3339), v)) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (u *Ui) Start() error { | ||||
| 	return u.wandle.Start() | ||||
| } | ||||
|  | ||||
| type ScreenMsg struct { | ||||
| 	source  ScreenMsgSource | ||||
| 	command ScreenMsgCommand | ||||
| 	data    interface{} | ||||
| 	err     error | ||||
| } | ||||
|  | ||||
| type ScreenMsgSource int | ||||
| type ScreenMsgCommand int | ||||
| @@ -5,6 +5,7 @@ import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| @@ -416,3 +417,12 @@ func BuildFilterFromArgs(args []string) func(*timertxt.Timer) bool { | ||||
| 	} | ||||
| 	return doFilters | ||||
| } | ||||
|  | ||||
| func SortedTagKeyList(m map[string]string) []string { | ||||
| 	var ret []string | ||||
| 	for k := range m { | ||||
| 		ret = append(ret, k) | ||||
| 	} | ||||
| 	sort.Strings(ret) | ||||
| 	return ret | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user