pelican/internal/ui/app.go

149 lines
2.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"
)
type shiftDirection int
const (
shiftLeft shiftDirection = iota
shiftRight
)
const (
mainPage string = "main"
quitPage string = "quit"
addPage 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
}
// 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 17:50:27 +01:00
func (u *UI) shiftColumnFocus(s shiftDirection) {
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
//}