git-bug/termui/termui.go

261 lines
4.9 KiB
Go
Raw Normal View History

2018-07-30 18:00:10 +03:00
package termui
import (
"github.com/MichaelMure/git-bug/cache"
"github.com/MichaelMure/git-bug/input"
2018-07-30 18:00:10 +03:00
"github.com/MichaelMure/git-bug/repository"
"github.com/jroimartin/gocui"
"github.com/pkg/errors"
2018-07-30 18:00:10 +03:00
)
var errTerminateMainloop = errors.New("terminate gocui mainloop")
2018-07-30 18:00:10 +03:00
type termUI struct {
2018-08-01 03:15:40 +03:00
g *gocui.Gui
gError chan error
cache cache.RepoCacher
activeWindow window
bugTable *bugTable
2018-08-01 03:15:40 +03:00
showBug *showBug
errorPopup *errorPopup
2018-08-12 03:42:03 +03:00
inputPopup *inputPopup
2018-07-30 18:00:10 +03:00
}
2018-08-01 03:15:40 +03:00
func (tui *termUI) activateWindow(window window) error {
if err := tui.activeWindow.disable(tui.g); err != nil {
return err
}
tui.activeWindow = window
return nil
}
2018-07-30 18:00:10 +03:00
var ui *termUI
type window interface {
keybindings(g *gocui.Gui) error
layout(g *gocui.Gui) error
2018-08-01 03:15:40 +03:00
disable(g *gocui.Gui) error
}
2018-07-30 18:00:10 +03:00
func Run(repo repository.Repo) error {
c := cache.NewRepoCache(repo)
ui = &termUI{
gError: make(chan error, 1),
cache: c,
bugTable: newBugTable(c),
2018-08-01 03:15:40 +03:00
showBug: newShowBug(c),
errorPopup: newErrorPopup(),
2018-08-12 03:42:03 +03:00
inputPopup: newInputPopup(),
2018-07-30 18:00:10 +03:00
}
ui.activeWindow = ui.bugTable
initGui()
err := <-ui.gError
if err != nil && err != gocui.ErrQuit {
return err
}
return nil
}
func initGui() {
2018-07-30 18:00:10 +03:00
g, err := gocui.NewGui(gocui.OutputNormal)
if err != nil {
ui.gError <- err
return
2018-07-30 18:00:10 +03:00
}
ui.g = g
2018-07-30 18:00:10 +03:00
ui.g.SetManagerFunc(layout)
2018-07-30 18:00:10 +03:00
err = keybindings(ui.g)
2018-07-30 18:00:10 +03:00
if err != nil {
ui.g.Close()
ui.g = nil
ui.gError <- err
return
2018-07-30 18:00:10 +03:00
}
err = g.MainLoop()
if err != nil && err != errTerminateMainloop {
if ui.g != nil {
ui.g.Close()
}
ui.gError <- err
2018-07-30 18:00:10 +03:00
}
return
2018-07-30 18:00:10 +03:00
}
func layout(g *gocui.Gui) error {
g.Cursor = false
2018-07-30 18:00:10 +03:00
if err := ui.activeWindow.layout(g); err != nil {
2018-07-30 18:00:10 +03:00
return err
}
if err := ui.errorPopup.layout(g); err != nil {
return err
}
2018-08-12 03:42:03 +03:00
if err := ui.inputPopup.layout(g); err != nil {
return err
}
2018-07-30 18:00:10 +03:00
return nil
}
func keybindings(g *gocui.Gui) error {
// Quit
if err := g.SetKeybinding("", gocui.KeyCtrlC, gocui.ModNone, quit); err != nil {
2018-07-30 18:00:10 +03:00
return err
}
if err := ui.bugTable.keybindings(g); err != nil {
return err
}
2018-07-30 18:00:10 +03:00
2018-08-01 03:15:40 +03:00
if err := ui.showBug.keybindings(g); err != nil {
return err
}
if err := ui.errorPopup.keybindings(g); err != nil {
return err
}
2018-08-12 03:42:03 +03:00
if err := ui.inputPopup.keybindings(g); err != nil {
return err
}
2018-07-30 18:00:10 +03:00
return nil
}
func quit(g *gocui.Gui, v *gocui.View) error {
return gocui.ErrQuit
}
func newBugWithEditor(repo cache.RepoCacher) error {
// This is somewhat hacky.
// As there is no way to pause gocui, run the editor and restart gocui,
// we have to stop it entirely and start a new one later.
//
// - an error channel is used to route the returned error of this new
// instance into the original launch function
// - a custom error (errTerminateMainloop) is used to terminate the original
// instance's mainLoop. This error is then filtered.
2018-07-30 18:00:10 +03:00
ui.g.Close()
ui.g = nil
2018-07-30 18:00:10 +03:00
title, message, err := input.BugCreateEditorInput(ui.cache.Repository(), "", "")
2018-07-30 18:00:10 +03:00
if err != nil && err != input.ErrEmptyTitle {
2018-07-30 18:00:10 +03:00
return err
}
if err == input.ErrEmptyTitle {
2018-08-12 03:42:03 +03:00
ui.errorPopup.Activate("Empty title, aborting.")
} else {
_, err := repo.NewBug(title, message)
if err != nil {
return err
}
}
initGui()
return errTerminateMainloop
}
func addCommentWithEditor(bug cache.BugCacher) error {
// This is somewhat hacky.
// As there is no way to pause gocui, run the editor and restart gocui,
// we have to stop it entirely and start a new one later.
//
// - an error channel is used to route the returned error of this new
// instance into the original launch function
// - a custom error (errTerminateMainloop) is used to terminate the original
// instance's mainLoop. This error is then filtered.
ui.g.Close()
ui.g = nil
message, err := input.BugCommentEditorInput(ui.cache.Repository())
if err != nil && err != input.ErrEmptyMessage {
return err
}
if err == input.ErrEmptyMessage {
2018-08-12 03:42:03 +03:00
ui.errorPopup.Activate("Empty message, aborting.")
} else {
err := bug.AddComment(message)
if err != nil {
return err
}
}
initGui()
return errTerminateMainloop
}
func setTitleWithEditor(bug cache.BugCacher) error {
// This is somewhat hacky.
// As there is no way to pause gocui, run the editor and restart gocui,
// we have to stop it entirely and start a new one later.
//
// - an error channel is used to route the returned error of this new
// instance into the original launch function
// - a custom error (errTerminateMainloop) is used to terminate the original
// instance's mainLoop. This error is then filtered.
ui.g.Close()
ui.g = nil
title, err := input.BugTitleEditorInput(ui.cache.Repository(), bug.Snapshot().Title)
if err != nil && err != input.ErrEmptyTitle {
return err
}
if err == input.ErrEmptyTitle {
2018-08-12 03:42:03 +03:00
ui.errorPopup.Activate("Empty title, aborting.")
} else {
err := bug.SetTitle(title)
if err != nil {
return err
}
2018-07-30 18:00:10 +03:00
}
initGui()
return errTerminateMainloop
2018-07-30 18:00:10 +03:00
}
func maxInt(a, b int) int {
if a > b {
return a
}
return b
}
func minInt(a, b int) int {
if a > b {
return b
}
return a
}