pelican/internal/ui/ui.go

202 lines
3.8 KiB
Go
Raw Normal View History

package ui
import (
"fmt"
2023-02-14 07:54:10 +00:00
"codeflow.dananglin.me.uk/apollo/canal/internal/board"
"github.com/rivo/tview"
2023-04-22 19:18:47 +01:00
"github.com/gdamore/tcell/v2"
)
const (
2023-04-22 19:18:47 +01:00
shiftLeft int = iota
shiftRight
)
const (
2023-04-22 19:18:47 +01:00
mainPageName string = "main"
quitPageName string = "quit"
addPageName string = "add"
)
2023-04-22 17:50:27 +01:00
// UI does some magical stuff.
type UI struct {
*tview.Application
columns []column
flex *tview.Flex
pages *tview.Pages
focusedColumn int
board board.Board
2023-04-22 19:18:47 +01:00
quit *tview.Modal
}
// NewUI returns a new UI value.
func NewUI() UI {
u := UI{
Application: tview.NewApplication(),
pages: tview.NewPages(),
flex: tview.NewFlex(),
quit: tview.NewModal(),
focusedColumn: 0,
}
u.initialise()
return u
}
// initialise the UI.
func (u *UI) initialise() {
u.pages.AddPage(mainPageName, u.flex, true, true)
u.initQuitModal()
u.pages.AddPage(quitPageName, u.quit, false, false)
u.SetInputCapture(func(event *tcell.EventKey) *tcell.EventKey {
if event.Rune() == 'q' {
u.pages.ShowPage(quitPageName)
u.SetFocus(u.quit)
} else if event.Rune() == 'o' {
u.openBoard("")
}
return event
})
u.SetRoot(u.pages, true)
}
// initQuitModal initialises the quit modal.
func (u *UI) initQuitModal() {
quitDoneFunc := func(_ int, buttonLabel string) {
switch buttonLabel {
case "Quit":
u.shutdown()
default:
u.pages.SwitchToPage("main")
u.setColumnFocus()
}
}
u.quit.SetText("Do you want to quit the application?").
AddButtons([]string{"Quit", "Cancel"}).
SetDoneFunc(quitDoneFunc)
}
// shutdown shuts down the application.
2023-04-22 17:50:27 +01:00
func (u *UI) shutdown() {
u.closeBoard()
u.Stop()
}
// closeBoard closes the BoltDB database.
2023-04-22 17:50:27 +01:00
func (u *UI) closeBoard() {
_ = u.board.Close()
}
// openBoard opens the kanban project.
2023-04-22 17:50:27 +01:00
func (u *UI) openBoard(path string) error {
b, err := board.Open(path)
if err != nil {
return fmt.Errorf("unable to load board, %w", err)
}
2023-04-22 17:50:27 +01:00
u.board = b
2023-04-22 17:50:27 +01:00
if err = u.refresh(); err != nil {
return err
}
return nil
}
// refresh refreshes the UI.
2023-04-22 17:50:27 +01:00
func (u *UI) refresh() error {
statusList, err := u.board.StatusList()
if err != nil {
return fmt.Errorf("unable to get the status list, %w", err)
}
2023-04-22 17:50:27 +01:00
u.updateBoard(statusList)
2023-04-22 17:50:27 +01:00
u.setColumnFocus()
return nil
}
2023-04-22 17:50:27 +01:00
func (u *UI) updateBoard(statusList []board.Status) error {
u.flex.Clear()
columns := make([]column, len(statusList))
for i := range statusList {
2023-04-22 17:50:27 +01:00
columns[i] = u.newColumn(statusList[i].ID, statusList[i].Name)
if len(statusList[i].CardIds) > 0 {
2023-04-22 17:50:27 +01:00
cards, err := u.board.CardList(statusList[i].CardIds)
if err != nil {
return fmt.Errorf("unable to get the card list. %w", err)
}
for _, c := range cards {
columns[i].cards.AddItem(fmt.Sprintf("[%d] %s", c.Id(), c.Title), "", 0, nil)
}
}
2023-04-22 17:50:27 +01:00
u.flex.AddItem(columns[i].cards, 0, 1, false)
}
2023-04-22 17:50:27 +01:00
u.columns = columns
return nil
}
2023-04-22 19:18:47 +01:00
func (u *UI) shiftColumnFocus(s int) {
switch s {
case shiftRight:
2023-04-22 17:50:27 +01:00
if u.focusedColumn == len(u.columns)-1 {
u.focusedColumn = 0
} else {
2023-04-22 17:50:27 +01:00
u.focusedColumn++
}
case shiftLeft:
2023-04-22 17:50:27 +01:00
if u.focusedColumn == 0 {
u.focusedColumn = len(u.columns) - 1
} else {
2023-04-22 17:50:27 +01:00
u.focusedColumn--
}
}
2023-04-22 17:50:27 +01:00
u.setColumnFocus()
}
2023-04-22 17:50:27 +01:00
func (u *UI) setColumnFocus() {
u.SetFocus(u.columns[u.focusedColumn].cards)
}
// newCard creates a new card and saves it to the database.
2023-04-22 17:50:27 +01:00
func (u *UI) newCard(title, content string) error {
args := board.CardArgs{
NewTitle: title,
NewContent: content,
}
2023-04-22 17:50:27 +01:00
if err := u.board.CreateCard(args); err != nil {
return fmt.Errorf("unable to create card, %w", err)
}
2023-04-22 17:50:27 +01:00
u.refresh()
return nil
}
// TODO: Move 'Add' to the centre of the app
// TODO: Customize list primitive or create a new one
// TODO: If customizing existing list primitive, wrap list around a column type. Add statusID to it.
// TODO: Update card status (card ID, oldStatus, newStatus)
//func viewCard() error {
//return nil
//}