174 lines
3.3 KiB
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)
|
|
}
|
|
}
|