glow/ui/ui.go

539 lines
12 KiB
Go
Raw Normal View History

package ui
import (
"errors"
2020-05-22 18:03:21 +00:00
"fmt"
2020-06-20 23:54:01 +00:00
"log"
"os"
2020-05-27 15:55:00 +00:00
"strings"
"github.com/charmbracelet/bubbles/spinner"
tea "github.com/charmbracelet/bubbletea"
"github.com/charmbracelet/charm"
"github.com/charmbracelet/charm/keygen"
"github.com/charmbracelet/charm/ui/common"
"github.com/muesli/gitcha"
2020-05-14 19:06:13 +00:00
te "github.com/muesli/termenv"
)
2020-08-13 19:37:53 +00:00
const noteCharacterLimit = 256 // should match server
var (
config Config
glowLogoTextColor = common.Color("#ECFD65")
2020-08-18 19:41:57 +00:00
debug = false // true if we're logging to a file, in which case we'll log more stuff
2020-05-22 02:29:46 +00:00
)
2020-08-13 19:37:53 +00:00
// Config contains configuration specified to the TUI.
2020-08-07 16:34:48 +00:00
type Config struct {
IdentityFile string
// For debugging the UI
2020-06-20 23:54:01 +00:00
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"`
}
2020-08-13 19:37:53 +00:00
// NewProgram returns a new Tea program.
2020-08-07 16:34:48 +00:00
func NewProgram(style string, cfg Config) *tea.Program {
if cfg.Logfile != "" {
2020-06-20 23:54:01 +00:00
log.Println("-- Starting Glow ----------------")
log.Printf("High performance pager: %v", cfg.HighPerformancePager)
log.Printf("Glamour rendering: %v", cfg.GlamourEnabled)
2020-06-20 23:54:01 +00:00
log.Println("Bubble Tea now initializing...")
2020-08-18 19:41:57 +00:00
debug = true
2020-06-20 23:54:01 +00:00
}
config = cfg
2020-08-07 16:34:48 +00:00
return tea.NewProgram(initialize(cfg, style), update, view)
}
// MESSAGES
type errMsg error
type newCharmClientMsg *charm.Client
type sshAuthErrMsg struct{}
2020-08-18 19:41:57 +00:00
type keygenFailedMsg struct {
err error
}
func (k keygenFailedMsg) Error() string {
return k.err.Error()
}
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 }
2020-08-18 19:41:57 +00:00
func (s stashLoadErrMsg) Error() string {
return s.err.Error()
}
type gotNewsMsg []*charm.Markdown
type newsLoadErrMsg struct{ err error }
2020-05-14 19:06:13 +00:00
2020-08-18 19:41:57 +00:00
func (s newsLoadErrMsg) Error() string {
return s.err.Error()
}
// MODEL
type state int
const (
stateShowStash state = iota
2020-05-14 02:08:17 +00:00
stateShowDocument
)
2020-06-23 19:11:54 +00:00
// String translates the staus to a human-readable string. This is just for
2020-05-22 02:29:46 +00:00
// 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 {
2020-08-07 16:34:48 +00:00
cfg Config
2020-05-14 19:06:13 +00:00
cc *charm.Client
user *charm.User
keygenState keygenState
2020-05-14 19:06:13 +00:00
state state
fatalErr error
2020-05-14 19:06:13 +00:00
stash stashModel
pager pagerModel
2020-05-14 19:06:13 +00:00
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
2020-05-14 02:08:17 +00:00
}
// 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 {
2020-05-14 02:08:17 +00:00
m.state = stateShowStash
m.stash.state = stashStateReady
m.pager.unload()
2020-06-03 18:17:41 +00:00
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
2020-08-07 16:34:48 +00:00
func initialize(cfg Config, style string) func() (tea.Model, tea.Cmd) {
return func() (tea.Model, tea.Cmd) {
2020-05-16 18:45:32 +00:00
if style == "auto" {
dbg := te.HasDarkBackground()
if dbg == true {
style = "dark"
} else {
style = "light"
}
}
m := model{
2020-08-07 16:34:48 +00:00
cfg: cfg,
stash: newStashModel(),
pager: newPagerModel(style),
state: stateShowStash,
keygenState: keygenUnstarted,
}
return m, tea.Batch(
findLocalFiles,
2020-08-07 16:34:48 +00:00
newCharmClient(&cfg.IdentityFile),
spinner.Tick(m.stash.spinner),
)
}
}
// UPDATE
func update(msg tea.Msg, mdl tea.Model) (tea.Model, tea.Cmd) {
m, ok := mdl.(model)
if !ok {
2020-05-22 18:03:21 +00:00
return model{
fatalErr: errors.New("could not perform assertion on model in update"),
}, tea.Quit
}
2020-06-05 20:57:27 +00:00
// If there's been an error, any key exits
if m.fatalErr != nil {
2020-06-05 20:57:27 +00:00
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
2020-05-22 02:29:46 +00:00
2020-06-23 19:11:54 +00:00
// Send these keys through to stash
2020-05-22 02:29:46 +00:00
switch m.state {
case stateShowStash:
switch m.stash.state {
case stashStateSettingNote, stashStatePromptDelete, stashStateShowingError:
2020-05-22 02:29:46 +00:00
m.stash, cmd = stashUpdate(msg, m.stash)
return m, cmd
}
2020-06-23 19:11:54 +00:00
// Special cases for the pager
2020-05-22 02:29:46 +00:00
case stateShowDocument:
switch m.pager.state {
// If browsing, these keys have special cases
case pagerStateBrowse:
switch msg.String() {
case "q":
return m, tea.Quit
case "esc":
var batch []tea.Cmd
batch = m.unloadDocument()
return m, tea.Batch(batch...)
}
// If setting a note send all keys straight through
case pagerStateSetNote:
var batch []tea.Cmd
newPagerModel, cmd := pagerUpdate(msg, m.pager)
m.pager = newPagerModel
batch = append(batch, cmd)
return m, tea.Batch(batch...)
}
2020-05-14 02:08:17 +00:00
}
2020-05-22 02:29:46 +00:00
return m, tea.Quit
2020-05-14 02:08:17 +00:00
case "left", "h", "delete":
if m.state == stateShowDocument && m.pager.state == pagerStateBrowse {
cmds = append(cmds, m.unloadDocument()...)
return m, tea.Batch(cmds...)
}
2020-06-23 19:11:54 +00:00
// Ctrl+C always quits no matter where in the application you are.
case "ctrl+c":
return m, tea.Quit
2020-05-15 00:23:11 +00:00
// Repaint
2020-05-15 00:23:11 +00:00
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)
2020-05-18 22:58:19 +00:00
// TODO: load more stash pages if we've resized, are on the last page,
// and haven't loaded more pages yet.
2020-05-14 19:06:13 +00:00
// 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
2020-08-07 16:34:48 +00:00
cmds = append(cmds, newCharmClient(nil))
case newCharmClientMsg:
m.cc = msg
m.stash.cc = msg
m.pager.cc = msg
cmds = append(cmds, loadStash(m.stash), loadNews(m.stash))
2020-05-14 02:08:17 +00:00
2020-05-22 19:31:54 +00:00
case fetchedMarkdownMsg:
m.pager.currentDocument = msg
cmds = append(cmds, renderWithGlamour(m.pager, msg.Body))
2020-05-15 19:08:45 +00:00
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)
2020-05-14 02:08:17 +00:00
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)
}
switch m.state {
2020-05-14 02:08:17 +00:00
case stateShowDocument:
return pagerView(m.pager)
2020-08-13 19:26:34 +00:00
default:
return stashView(m.stash)
}
}
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",
2020-06-05 20:57:27 +00:00
te.String(" ERROR ").
Foreground(common.Cream.Color()).
Background(common.Red.Color()).
String(),
err,
common.Subtle(exitMsg),
2020-06-05 20:57:27 +00:00
)
2020-07-16 19:37:48 +00:00
return "\n" + indent(s, 3)
2020-06-05 20:57:27 +00:00
}
// COMMANDS
2020-05-18 21:53:46 +00:00
func findLocalFiles() tea.Msg {
cwd, err := os.Getwd()
if err != nil {
2020-08-18 19:41:57 +00:00
if debug {
log.Println("error finding local files:", err)
}
return errMsg(err)
}
2020-08-22 08:16:01 +00:00
ch, err := gitcha.FindFiles(cwd, []string{"*.md"})
if err != nil {
if debug {
log.Println("error finding local files:", err)
}
return errMsg(err)
}
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{}
}
}
2020-08-07 16:34:48 +00:00
func newCharmClient(identityFile *string) tea.Cmd {
return func() tea.Msg {
cfg, err := charm.ConfigFromEnv()
if err != nil {
return errMsg(err)
}
2020-08-07 16:34:48 +00:00
if identityFile != nil {
cfg.SSHKeyPath = *identityFile
}
2020-08-07 16:34:48 +00:00
cc, err := charm.NewClient(cfg)
if err == charm.ErrMissingSSHAuth {
2020-08-18 19:41:57 +00:00
if debug {
log.Println("missing SSH auth:", err)
}
2020-08-07 16:34:48 +00:00
return sshAuthErrMsg{}
} else if err != nil {
2020-08-18 19:41:57 +00:00
if debug {
log.Println("error creating new charm client:", err)
}
2020-08-07 16:34:48 +00:00
return errMsg(err)
}
return newCharmClientMsg(cc)
}
}
2020-05-14 00:21:37 +00:00
func loadStash(m stashModel) tea.Cmd {
return func() tea.Msg {
stash, err := m.cc.GetStash(m.page)
if err != nil {
2020-08-18 19:41:57 +00:00
if debug {
log.Println("error loading stash:", err)
}
2020-08-13 19:37:53 +00:00
return stashLoadErrMsg{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 {
2020-08-18 19:41:57 +00:00
if debug {
log.Println("error loading news:", err)
}
2020-08-13 19:37:53 +00:00
return newsLoadErrMsg{err}
}
return gotNewsMsg(news)
}
}
func generateSSHKeys() tea.Msg {
2020-08-18 15:08:39 +00:00
_, err := keygen.NewSSHKeyPair(nil)
if err != nil {
2020-08-18 19:41:57 +00:00
if debug {
log.Println("keygen failed:", err)
}
return keygenFailedMsg{err}
}
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
}
2020-05-27 15:55:00 +00:00
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()
}
2020-05-14 00:21:37 +00:00
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
}