package ui import ( "errors" "fmt" "log" "os" "strings" "github.com/charmbracelet/bubbles/spinner" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/charm" "github.com/charmbracelet/charm/ui/common" "github.com/muesli/gitcha" te "github.com/muesli/termenv" ) const ( noteCharacterLimit = 256 // should match server ) // UIConfig contains flags for debugging the TUI. type UIConfig struct { Logfile string `env:"GLOW_UI_LOGFILE"` HighPerformancePager bool `env:"GLOW_UI_HIGH_PERFORMANCE_PAGER" default:"true"` GlamourEnabled bool `env:"GLOW_UI_ENABLE_GLAMOUR" default:"true"` } var ( config UIConfig glowLogoTextColor = common.Color("#ECFD65") ) // NewProgram returns a new Tea program func NewProgram(style string, cfg UIConfig) *tea.Program { config = cfg if config.Logfile != "" { log.Println("-- Starting Glow ----------------") log.Printf("High performance pager: %v", cfg.HighPerformancePager) log.Printf("Glamour rendering: %v", cfg.GlamourEnabled) log.Println("Bubble Tea now initializing...") } return tea.NewProgram(initialize(style), update, view) } // MESSAGES type errMsg error type newCharmClientMsg *charm.Client type sshAuthErrMsg struct{} type keygenFailedMsg struct{} type keygenSuccessMsg struct{} type initLocalFileSearchMsg struct { cwd string ch chan gitcha.SearchResult } type foundLocalFileMsg gitcha.SearchResult type localFileSearchFinished struct{} type gotStashMsg []*charm.Markdown type stashLoadErrMsg struct{ err error } type gotNewsMsg []*charm.Markdown type newsLoadErrMsg struct{ err error } // MODEL type state int const ( stateShowStash state = iota stateShowDocument ) // String translates the staus to a human-readable string. This is just for // debugging. func (s state) String() string { return [...]string{ "showing stash", "showing document", }[s] } type keygenState int const ( keygenUnstarted keygenState = iota keygenRunning keygenFinished ) type model struct { cc *charm.Client user *charm.User keygenState keygenState state state fatalErr error stash stashModel pager pagerModel terminalWidth int terminalHeight int cwd string // directory from which we're running Glow // Channel that receives paths to local markdown files // (via the github.com/muesli/gitcha package) localFileFinder chan gitcha.SearchResult } // unloadDocument unloads a document from the pager. Note that while this // method alters the model we also need to send along any commands returned. func (m *model) unloadDocument() []tea.Cmd { m.state = stateShowStash m.stash.state = stashStateReady m.pager.unload() m.pager.showHelp = false var batch []tea.Cmd if m.pager.viewport.HighPerformanceRendering { batch = append(batch, tea.ClearScrollArea) } if !m.stash.loaded.done() || m.stash.loadingFromNetwork { batch = append(batch, spinner.Tick(m.stash.spinner)) } return batch } // INIT func initialize(style string) func() (tea.Model, tea.Cmd) { return func() (tea.Model, tea.Cmd) { if style == "auto" { dbg := te.HasDarkBackground() if dbg == true { style = "dark" } else { style = "light" } } m := model{ stash: newStashModel(), pager: newPagerModel(style), state: stateShowStash, keygenState: keygenUnstarted, } return m, tea.Batch( findLocalFiles, newCharmClient, spinner.Tick(m.stash.spinner), ) } } // UPDATE func update(msg tea.Msg, mdl tea.Model) (tea.Model, tea.Cmd) { m, ok := mdl.(model) if !ok { return model{ fatalErr: errors.New("could not perform assertion on model in update"), }, tea.Quit } // If there's been an error, any key exits if m.fatalErr != nil { if _, ok := msg.(tea.KeyMsg); ok { return m, tea.Quit } } var cmds []tea.Cmd switch msg := msg.(type) { case tea.KeyMsg: switch msg.String() { case "q", "esc": var cmd tea.Cmd // Send these keys through to stash switch m.state { case stateShowStash: switch m.stash.state { case stashStateSettingNote, stashStatePromptDelete, stashStateShowingError: m.stash, cmd = stashUpdate(msg, m.stash) return m, cmd } // Special cases for the pager case stateShowDocument: var batch []tea.Cmd if m.pager.state == pagerStateBrowse { // If the user is just browing a document, exit the pager. batch = m.unloadDocument() } else { // Otherwise send these key messages through to pager for // processing newPagerModel, cmd := pagerUpdate(msg, m.pager) m.pager = newPagerModel batch = append(batch, cmd) } return m, tea.Batch(batch...) } return m, tea.Quit case "left", "h", "delete": if m.state == stateShowDocument && m.pager.state == pagerStateBrowse { cmds = append(cmds, m.unloadDocument()...) } // Ctrl+C always quits no matter where in the application you are. case "ctrl+c": return m, tea.Quit // Repaint case "ctrl+l": // TODO return m, nil } // Window size is received when starting up and on every resize case tea.WindowSizeMsg: m.terminalWidth = msg.Width m.terminalHeight = msg.Height m.stash.setSize(msg.Width, msg.Height) m.pager.setSize(msg.Width, msg.Height) // TODO: load more stash pages if we've resized, are on the last page, // and haven't loaded more pages yet. // We've started looking for local files case initLocalFileSearchMsg: m.localFileFinder = msg.ch m.cwd = msg.cwd cmds = append(cmds, findNextLocalFile(m)) // We found a local file case foundLocalFileMsg: pathStr, err := localFileToMarkdown(m.cwd, gitcha.SearchResult(msg)) if err == nil { m.stash.addMarkdowns(pathStr) } cmds = append(cmds, findNextLocalFile(m)) case sshAuthErrMsg: // If we haven't run the keygen yet, do that if m.keygenState != keygenFinished { m.keygenState = keygenRunning cmds = append(cmds, generateSSHKeys) } else { // The keygen ran but things still didn't work and we can't auth m.stash.err = errors.New("SSH authentication failed; we tried ssh-agent, loading keys from disk, and generating SSH keys") // Even though it failed, news/stash loading is finished m.stash.loaded |= loadedStash | loadedNews m.stash.loadingFromNetwork = false } case keygenFailedMsg: // Keygen failed. That sucks. m.stash.err = errors.New("could not authenticate; could not generate SSH keys") m.keygenState = keygenFinished // Even though it failed, news/stash loading is finished m.stash.loaded |= loadedStash | loadedNews m.stash.loadingFromNetwork = false case keygenSuccessMsg: // The keygen's done, so let's try initializing the charm client again m.keygenState = keygenFinished cmds = append(cmds, newCharmClient) case newCharmClientMsg: m.cc = msg m.stash.cc = msg m.pager.cc = msg cmds = append(cmds, loadStash(m.stash), loadNews(m.stash)) case fetchedMarkdownMsg: m.pager.currentDocument = msg cmds = append(cmds, renderWithGlamour(m.pager, msg.Body)) case contentRenderedMsg: m.state = stateShowDocument case noteSavedMsg: // A note was saved to a document. This will have be done in the // pager, so we'll need to find the corresponding note in the stash. // So, pass the message to the stash for processing. stashModel, cmd := stashUpdate(msg, m.stash) m.stash = stashModel return m, cmd case localFileSearchFinished, gotStashMsg, gotNewsMsg: // Also pass these messages to the stash so we can keep it updated // about network activity. stashModel, cmd := stashUpdate(msg, m.stash) m.stash = stashModel return m, cmd } // Process children switch m.state { case stateShowStash: newStashModel, cmd := stashUpdate(msg, m.stash) m.stash = newStashModel cmds = append(cmds, cmd) case stateShowDocument: newPagerModel, cmd := pagerUpdate(msg, m.pager) m.pager = newPagerModel cmds = append(cmds, cmd) } return m, tea.Batch(cmds...) } // VIEW func view(mdl tea.Model) string { m, ok := mdl.(model) if !ok { return "could not perform assertion on model in view" } if m.fatalErr != nil { return errorView(m.fatalErr, true) } var s string switch m.state { case stateShowStash: return stashView(m.stash) case stateShowDocument: return pagerView(m.pager) } return "\n" + indent(s, 2) } func errorView(err error, fatal bool) string { exitMsg := "press any key to " if fatal { exitMsg += "exit" } else { exitMsg += "return" } s := fmt.Sprintf("%s\n\n%v\n\n%s", te.String(" ERROR "). Foreground(common.Cream.Color()). Background(common.Red.Color()). String(), err, common.Subtle(exitMsg), ) return "\n" + indent(s, 3) } // COMMANDS func findLocalFiles() tea.Msg { cwd, err := os.Getwd() if err != nil { return errMsg(err) } ch := gitcha.FindFileFromList(cwd, []string{"*.md"}) return initLocalFileSearchMsg{ch: ch, cwd: cwd} } func findNextLocalFile(m model) tea.Cmd { return func() tea.Msg { pathStr, ok := <-m.localFileFinder if ok { // Okay now find the next one return foundLocalFileMsg(pathStr) } // We're done return localFileSearchFinished{} } } func newCharmClient() tea.Msg { cfg, err := charm.ConfigFromEnv() if err != nil { return errMsg(err) } cc, err := charm.NewClient(cfg) if err == charm.ErrMissingSSHAuth { return sshAuthErrMsg{} } else if err != nil { return errMsg(err) } return newCharmClientMsg(cc) } func loadStash(m stashModel) tea.Cmd { return func() tea.Msg { stash, err := m.cc.GetStash(m.page) if err != nil { return stashLoadErrMsg{err: err} } return gotStashMsg(stash) } } func loadNews(m stashModel) tea.Cmd { return func() tea.Msg { news, err := m.cc.GetNews(1) // just fetch the first page if err != nil { return newsLoadErrMsg{err: err} } return gotNewsMsg(news) } } func generateSSHKeys() tea.Msg { _, err := charm.NewSSHKeyPair() if err != nil { return keygenFailedMsg{} } return keygenSuccessMsg{} } // ETC // Convert local file path to Markdown. Note that we could be doing things // like checking if the file is a directory, but we trust that gitcha has // already done that. func localFileToMarkdown(cwd string, res gitcha.SearchResult) (*markdown, error) { md := &markdown{ markdownType: localMarkdown, localPath: res.Path, Markdown: &charm.Markdown{}, } // Strip absolute path md.Markdown.Note = strings.Replace(res.Path, cwd+"/", "", -1) // Get last modified time t := res.Info.ModTime() md.CreatedAt = &t return md, nil } func indent(s string, n int) string { if n <= 0 || s == "" { return s } l := strings.Split(s, "\n") b := strings.Builder{} i := strings.Repeat(" ", n) for _, v := range l { fmt.Fprintf(&b, "%s%s\n", i, v) } return b.String() } func min(a, b int) int { if a < b { return a } return b } func max(a, b int) int { if a > b { return a } return b }