gemlog-cli/main.go

174 lines
3.3 KiB
Go

package main
import (
"fmt"
"gemini_site/gemlog"
"log/slog"
"os"
tea "github.com/charmbracelet/bubbletea"
)
type Action string
const (
Write Action = "write"
Read Action = "read"
Edit Action = "edit"
Delete Action = "delete"
)
var actions = []Action{Write, Read, Edit, Delete}
func TODOCmd() tea.Msg {
return gemlog.Notification("This action has not been implemented yet. Try another.")
}
type Page string
const (
ActionList Page = "actionList"
EntryList Page = "entryList"
Entry Page = "entry"
)
type entryListPageModel struct {
entries []gemlog.GemlogListEntry
action Action
}
type uiState struct {
notification string
cursor int
errorTxt string
page Page
entryListPage entryListPageModel
}
type context struct {
config *gemlog.Config
}
type model struct {
ui uiState
context *context
}
func initialModel(config *gemlog.Config) model {
return model{
ui: uiState{
page: ActionList,
entryListPage: entryListPageModel{
entries: []gemlog.GemlogListEntry{},
},
},
context: &context{
config: config,
},
}
}
func (m model) Init() tea.Cmd {
return tea.SetWindowTitle("Gemlog CLI")
}
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) {
case gemlog.ErrorMsg:
m.ui.errorTxt = fmt.Sprintf("%s\n\n", fmt.Errorf("%s", msg))
case gemlog.Notification:
m.ui.notification = fmt.Sprintf("%s\n\n", string(msg))
case gemlog.GemLogsLoaded:
m.ui.entryListPage.entries = msg.Logs
m.ui.cursor = 0
case tea.KeyMsg:
switch msg.String() {
case "ctrl+c", "q":
return m, tea.Quit
case "up", "k":
if m.ui.cursor > 0 {
m.ui.cursor--
}
case "down", "j":
if m.ui.cursor < len(actions)-1 {
m.ui.cursor++
}
case "enter", " ":
action := actions[m.ui.cursor]
switch action {
case Write:
return m, gemlog.WritePostCMD(m.context.config)
case Read:
m.ui.page = EntryList
return m, gemlog.LoadGemlogCMD(m.context.config)
case Edit:
return m, TODOCmd
case Delete:
return m, TODOCmd
}
}
}
return m, nil
}
func (m model) View() string {
if len(m.ui.errorTxt) > 0 {
return m.ui.errorTxt
}
s := ""
if m.ui.notification != "" {
s += m.ui.notification
} else {
s += "Welcome to gemlog cli!\n\nWhat post action would you like to take?\n\n"
}
if m.ui.page == ActionList {
for i, action := range actions {
cursor := " "
if m.ui.cursor == i {
cursor = ">"
}
s += fmt.Sprintf("%s %s\n", cursor, action)
}
}
if m.ui.page == EntryList {
slog.Info("rendering entry list", "count", len(m.ui.entryListPage.entries))
for i, entry := range m.ui.entryListPage.entries {
slog.Info("entry", "value", entry)
cursor := " "
if m.ui.cursor == i {
cursor = ">"
}
s += fmt.Sprintf("%s %s : %s\n", cursor, entry.Date, entry.Slug)
}
}
s += "\nPress q to quit.\n"
return s
}
func main() {
f, err := tea.LogToFile("debug.log", "debug")
if err != nil {
fmt.Println("fatal:", err)
os.Exit(1)
}
defer f.Close()
slog.Info("Starting gemlog cli")
config, err := gemlog.LoadConfig()
if err != nil {
fmt.Printf("Error loading config: %v", err)
os.Exit(1)
}
// TODO: check if we can reach db before starting program
p := tea.NewProgram(initialModel(config))
if _, err := p.Run(); err != nil {
fmt.Printf("Alas, there's been an error: %v", err)
os.Exit(1)
}
}