package ui import ( "errors" "fmt" "log" "os" "strings" tea "github.com/charmbracelet/bubbletea" "github.com/charmbracelet/charm" "github.com/charmbracelet/charm/ui/common" "github.com/charmbracelet/charm/ui/keygen" "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("Render with Glamour: %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 initLocalFileSearchMsg struct { cwd string ch chan string } type foundLocalFileMsg string type localFileSearchFinished struct{} type gotStashMsg []*charm.Markdown type gotNewsMsg []*charm.Markdown // MODEL type loadedState byte const ( loadedStash loadedState = 1 << iota loadedNews loadedLocalFiles ) func (s loadedState) done() bool { return s&loadedStash != 0 && s&loadedNews != 0 && s&loadedLocalFiles != 0 } type state int const ( stateShowStash state = iota stateShowDocument ) type keygenState int const ( keygenUnstarted keygenState = iota keygenRunning keygenFinished ) // String translates the staus to a human-readable string. This is just for // debugging. func (s state) String() string { return [...]string{ "initializing", "running keygen", "keygen finished", "showing stash", "showing document", }[s] } type model struct { cc *charm.Client user *charm.User keygen keygen.Model keygenState keygenState state state err error stash stashModel pager pagerModel terminalWidth int terminalHeight int cwd string // directory from which we're running Glow loaded loadedState // what's loaded? we find out with bitmasking // Channel that receives paths to local markdown files // (via the github.com/muesli/gitcha package) localFileFinder chan string } func (m *model) unloadDocument() { m.state = stateShowStash m.stash.state = stashStateReady m.pager.unload() m.pager.showHelp = false } // 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" } } return model{ stash: newStashModel(), pager: newPagerModel(style), state: stateShowStash, keygenState: keygenUnstarted, }, tea.Batch( findLocalFiles, newCharmClient, ) } } // UPDATE func update(msg tea.Msg, mdl tea.Model) (tea.Model, tea.Cmd) { m, ok := mdl.(model) if !ok { return model{ err: errors.New("could not perform assertion on model in update"), }, tea.Quit } // If there's been an error, any key exits if m.err != nil { if _, ok := msg.(tea.KeyMsg); ok { return m, tea.Quit } } var ( cmd tea.Cmd cmds []tea.Cmd ) switch msg := msg.(type) { case tea.KeyMsg: switch msg.String() { case "q": fallthrough case "esc": var cmd tea.Cmd // Send these keys through to stash switch m.state { case stateShowStash: switch m.stash.state { case stashStateSettingNote: fallthrough case stashStatePromptDelete: m.stash, cmd = stashUpdate(msg, m.stash) return m, cmd } // Special cases for the pager case stateShowDocument: if m.pager.state == pagerStateBrowse { // If the user is just browing a document, exit the pager. m.unloadDocument() if m.pager.viewport.HighPerformanceRendering { cmd = tea.ClearScrollArea } } else { // Otherwise send keys through to pager for processing m.pager, cmd = pagerUpdate(msg, m.pager) } return m, cmd } return m, tea.Quit // 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 } case errMsg: m.err = msg 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, string(msg)) if err == nil { m.stash.hasLocalFiles = true m.stash.addMarkdowns(pathStr) } cmds = append(cmds, findNextLocalFile(m)) // We're finished searching for local files case localFileSearchFinished: m.loaded |= loadedLocalFiles case sshAuthErrMsg: // If we haven't run the keygen yet, do that if m.keygenState != keygenFinished { m.keygenState = keygenRunning m.keygen = keygen.NewModel() cmds = append(cmds, keygen.GenerateKeys) } else { // The keygen didn't work and we can't auth m.err = errors.New("SSH authentication failed") return m, tea.Quit } case keygen.DoneMsg: // 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.state = stateShowStash m.stash.cc = msg m.pager.cc = msg cmds = append(cmds, loadStash(m.stash), loadNews(m.stash)) case gotStashMsg: m.loaded |= loadedStash case gotNewsMsg: m.loaded |= loadedNews 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. m.stash, cmd = stashUpdate(msg, m.stash) cmds = append(cmds, cmd) case fetchedMarkdownMsg: m.pager.currentDocument = msg cmds = append(cmds, renderWithGlamour(m.pager, msg.Body)) case contentRenderedMsg: m.state = stateShowDocument } // Process keygen if m.keygenState == keygenRunning { mdl, cmd := keygen.Update(msg, tea.Model(m.keygen)) keygenModel, ok := mdl.(keygen.Model) if !ok { m.err = errors.New("could not perform assertion on keygen model in main update") return m, tea.Quit } m.keygen = keygenModel cmds = append(cmds, cmd) } // Process children switch m.state { case stateShowStash: m.stash, cmd = stashUpdate(msg, m.stash) cmds = append(cmds, cmd) case stateShowDocument: m.pager, cmd = pagerUpdate(msg, m.pager) 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.err != nil { return "\n" + indent(errorView(m.err), 2) } 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) string { return fmt.Sprintf("%s\n\n%v\n\n%s", te.String(" ERROR "). Foreground(common.Cream.Color()). Background(common.Red.Color()). String(), err, common.Subtle("Press any key to exit"), ) } // 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 errMsg(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 errMsg(err) } return gotNewsMsg(news) } } // 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, path string) (*markdown, error) { md := &markdown{ markdownType: localFile, localPath: path, Markdown: &charm.Markdown{}, } // Strip absolute path md.Markdown.Note = strings.Replace(path, cwd+"/", "", -1) // Get last modified time info, err := os.Stat(path) if err != nil { return nil, err } t := 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 }