mirror of
https://github.com/charmbracelet/glow
synced 2025-01-20 16:03:57 +00:00
481 lines
10 KiB
Go
481 lines
10 KiB
Go
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("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 keygenFailedMsg struct{}
|
|
type keygenSuccessMsg 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 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
|
|
keygenState keygenState
|
|
state state
|
|
err 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 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"
|
|
}
|
|
}
|
|
|
|
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{
|
|
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:
|
|
var cmds []tea.Cmd
|
|
if m.pager.state == pagerStateBrowse {
|
|
// If the user is just browing a document, exit the pager.
|
|
m.unloadDocument()
|
|
if m.pager.viewport.HighPerformanceRendering {
|
|
cmds = append(cmds, tea.ClearScrollArea)
|
|
}
|
|
if !m.stash.loaded.done() || m.stash.loadingFromNetwork {
|
|
cmds = append(cmds, spinner.Tick(m.stash.spinner))
|
|
}
|
|
} else {
|
|
// Otherwise send these key messages through to pager for
|
|
// processing
|
|
newPagerModel, cmd := pagerUpdate(msg, m.pager)
|
|
m.pager = newPagerModel
|
|
cmds = append(cmds, cmd)
|
|
}
|
|
return m, tea.Batch(cmds...)
|
|
}
|
|
|
|
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))
|
|
|
|
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.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.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.state = stateShowStash
|
|
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:
|
|
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)
|
|
}
|
|
}
|
|
|
|
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, 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
|
|
}
|