mirror of
https://github.com/walles/moar.git
synced 2024-11-27 01:05:23 +03:00
696 lines
16 KiB
Go
696 lines
16 KiB
Go
package m
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"time"
|
|
"unicode"
|
|
"unicode/utf8"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
"github.com/walles/moar/twin"
|
|
)
|
|
|
|
type _PagerMode int
|
|
|
|
const (
|
|
_Viewing _PagerMode = 0
|
|
_Searching _PagerMode = 1
|
|
_NotFound _PagerMode = 2
|
|
)
|
|
|
|
type StatusBarStyle int
|
|
|
|
const (
|
|
STATUSBAR_STYLE_INVERSE StatusBarStyle = iota
|
|
STATUSBAR_STYLE_PLAIN
|
|
STATUSBAR_STYLE_BOLD
|
|
)
|
|
|
|
// How do we render unprintable characters?
|
|
type UnprintableStyle int
|
|
|
|
const (
|
|
UNPRINTABLE_STYLE_HIGHLIGHT UnprintableStyle = iota
|
|
UNPRINTABLE_STYLE_WHITESPACE
|
|
)
|
|
|
|
type eventSpinnerUpdate struct {
|
|
spinner string
|
|
}
|
|
|
|
type eventMoreLinesAvailable struct{}
|
|
|
|
// Styling of line numbers
|
|
var _numberStyle = twin.StyleDefault.WithAttr(twin.AttrDim)
|
|
|
|
// Pager is the main on-screen pager
|
|
type Pager struct {
|
|
reader *Reader
|
|
screen twin.Screen
|
|
quit bool
|
|
scrollPosition scrollPosition
|
|
leftColumnZeroBased int
|
|
|
|
mode _PagerMode
|
|
searchString string
|
|
searchPattern *regexp.Regexp
|
|
|
|
isShowingHelp bool
|
|
preHelpState *_PreHelpState
|
|
|
|
// NewPager shows lines by default, this field can hide them
|
|
ShowLineNumbers bool
|
|
|
|
StatusBarStyle StatusBarStyle
|
|
UnprintableStyle UnprintableStyle
|
|
|
|
WrapLongLines bool
|
|
|
|
// If true, pager will clear the screen on return. If false, pager will
|
|
// clear the last line, and show the cursor.
|
|
DeInit bool
|
|
}
|
|
|
|
type _PreHelpState struct {
|
|
reader *Reader
|
|
scrollPosition scrollPosition
|
|
leftColumnZeroBased int
|
|
}
|
|
|
|
const _EofMarkerFormat = "\x1b[7m" // Reverse video
|
|
|
|
var _HelpReader = NewReaderFromText("Help", `
|
|
Welcome to Moar, the nice pager!
|
|
|
|
Miscellaneous
|
|
-------------
|
|
* Press 'q' or ESC to quit
|
|
* Press 'w' to toggle wrapping of long lines
|
|
|
|
Moving around
|
|
-------------
|
|
* Arrow keys
|
|
* 'h', 'l' for left and right (as in vim)
|
|
* Left / right can be used to hide / show line numbers
|
|
* PageUp / 'b' and PageDown / 'f'
|
|
* Half page 'u'p / 'd'own
|
|
* Home and End for start / end of the document
|
|
* < to go to the start of the document
|
|
* > to go to the end of the document
|
|
* RETURN moves down one line
|
|
* SPACE moves down a page
|
|
|
|
Searching
|
|
---------
|
|
* Type / to start searching, then type what you want to find
|
|
* Type RETURN to stop searching
|
|
* Find next by typing 'n' (for "next")
|
|
* Find previous by typing SHIFT-N or 'p' (for "previous")
|
|
* Search is case sensitive if it contains any UPPER CASE CHARACTERS
|
|
* Search is interpreted as a regexp if it is a valid one
|
|
|
|
Reporting bugs
|
|
--------------
|
|
File issues at https://github.com/walles/moar/issues, or post
|
|
questions to johan.walles@gmail.com.
|
|
|
|
Installing Moar as your default pager
|
|
-------------------------------------
|
|
Put the following line in your .bashrc or .bash_profile:
|
|
export PAGER=/usr/local/bin/moar.rb
|
|
|
|
Source Code
|
|
-----------
|
|
Available at https://github.com/walles/moar/.
|
|
`)
|
|
|
|
// NewPager creates a new Pager
|
|
func NewPager(r *Reader) *Pager {
|
|
return &Pager{
|
|
reader: r,
|
|
quit: false,
|
|
ShowLineNumbers: true,
|
|
DeInit: true,
|
|
}
|
|
}
|
|
|
|
func (p *Pager) addSearchFooter() {
|
|
_, height := p.screen.Size()
|
|
|
|
pos := 0
|
|
for _, token := range "Search: " + p.searchString {
|
|
p.screen.SetCell(pos, height-1, twin.NewCell(token, twin.StyleDefault))
|
|
pos++
|
|
}
|
|
|
|
// Add a cursor
|
|
p.screen.SetCell(pos, height-1, twin.NewCell(' ', twin.StyleDefault.WithAttr(twin.AttrReverse)))
|
|
}
|
|
|
|
func (p *Pager) setFooter(footer string) {
|
|
width, height := p.screen.Size()
|
|
|
|
pos := 0
|
|
var footerStyle twin.Style
|
|
if p.StatusBarStyle == STATUSBAR_STYLE_INVERSE {
|
|
footerStyle = twin.StyleDefault.WithAttr(twin.AttrReverse)
|
|
} else if p.StatusBarStyle == STATUSBAR_STYLE_PLAIN {
|
|
footerStyle = twin.StyleDefault
|
|
} else if p.StatusBarStyle == STATUSBAR_STYLE_BOLD {
|
|
footerStyle = twin.StyleDefault.WithAttr(twin.AttrBold)
|
|
} else {
|
|
panic(fmt.Sprint("Unrecognized footer style: ", footerStyle))
|
|
}
|
|
for _, token := range footer {
|
|
p.screen.SetCell(pos, height-1, twin.NewCell(token, footerStyle))
|
|
pos++
|
|
}
|
|
|
|
for ; pos < width; pos++ {
|
|
p.screen.SetCell(pos, height-1, twin.NewCell(' ', footerStyle))
|
|
}
|
|
}
|
|
|
|
// Quit leaves the help screen or quits the pager
|
|
func (p *Pager) Quit() {
|
|
if !p.isShowingHelp {
|
|
p.quit = true
|
|
return
|
|
}
|
|
|
|
// Reset help
|
|
p.isShowingHelp = false
|
|
p.reader = p.preHelpState.reader
|
|
p.scrollPosition = p.preHelpState.scrollPosition
|
|
p.leftColumnZeroBased = p.preHelpState.leftColumnZeroBased
|
|
p.preHelpState = nil
|
|
}
|
|
|
|
func (p *Pager) scrollToSearchHits() {
|
|
if p.searchPattern == nil {
|
|
// This is not a search
|
|
return
|
|
}
|
|
|
|
firstHitPosition := p.findFirstHit(p.scrollPosition, false)
|
|
if firstHitPosition == nil {
|
|
// No match, give up
|
|
return
|
|
}
|
|
|
|
if p.isVisible(*firstHitPosition) {
|
|
// Already on-screen, never mind
|
|
return
|
|
}
|
|
|
|
p.scrollPosition = *firstHitPosition
|
|
}
|
|
|
|
func (p *Pager) findFirstHit(startPosition scrollPosition, backwards bool) *scrollPosition {
|
|
// FIXME: We should take startPosition.deltaScreenLines into account as well!
|
|
searchPosition := startPosition
|
|
for {
|
|
line := p.reader.GetLine(searchPosition.lineNumberOneBased)
|
|
if line == nil {
|
|
// No match, give up
|
|
return nil
|
|
}
|
|
|
|
lineText := line.Plain()
|
|
if p.searchPattern.MatchString(lineText) {
|
|
return &scrollPosition{
|
|
lineNumberOneBased: searchPosition.lineNumberOneBased,
|
|
}
|
|
}
|
|
|
|
if backwards {
|
|
searchPosition.PreviousLine(1)
|
|
} else {
|
|
searchPosition.NextLine(1)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p *Pager) scrollToNextSearchHit() {
|
|
if p.searchPattern == nil {
|
|
// Nothing to search for, never mind
|
|
return
|
|
}
|
|
|
|
if p.reader.GetLineCount() == 0 {
|
|
// Nothing to search in, never mind
|
|
return
|
|
}
|
|
|
|
var firstSearchPosition scrollPosition
|
|
|
|
switch p.mode {
|
|
case _Viewing:
|
|
// Start searching on the first line below the bottom of the screen
|
|
firstSearchPosition = p.getLastVisiblePosition().NextLine(1)
|
|
|
|
case _NotFound:
|
|
// Restart searching from the top
|
|
p.mode = _Viewing
|
|
firstSearchPosition = scrollPosition{
|
|
lineNumberOneBased: 1,
|
|
}
|
|
|
|
default:
|
|
panic(fmt.Sprint("Unknown search mode when finding next: ", p.mode))
|
|
}
|
|
|
|
firstHitPosition := p.findFirstHit(firstSearchPosition, false)
|
|
if firstHitPosition == nil {
|
|
p.mode = _NotFound
|
|
return
|
|
}
|
|
p.scrollPosition = *firstHitPosition
|
|
}
|
|
|
|
func (p *Pager) scrollToPreviousSearchHit() {
|
|
if p.searchPattern == nil {
|
|
// Nothing to search for, never mind
|
|
return
|
|
}
|
|
|
|
if p.reader.GetLineCount() == 0 {
|
|
// Nothing to search in, never mind
|
|
return
|
|
}
|
|
|
|
var firstSearchPosition scrollPosition
|
|
|
|
switch p.mode {
|
|
case _Viewing:
|
|
// Start searching on the first line above the top of the screen
|
|
firstSearchPosition = p.scrollPosition.PreviousLine(1)
|
|
|
|
case _NotFound:
|
|
// Restart searching from the bottom
|
|
p.mode = _Viewing
|
|
p.scrollToEnd()
|
|
|
|
default:
|
|
panic(fmt.Sprint("Unknown search mode when finding previous: ", p.mode))
|
|
}
|
|
|
|
firstHitPosition := p.findFirstHit(firstSearchPosition, true)
|
|
if firstHitPosition == nil {
|
|
p.mode = _NotFound
|
|
return
|
|
}
|
|
p.scrollPosition = *firstHitPosition
|
|
}
|
|
|
|
func (p *Pager) updateSearchPattern() {
|
|
p.searchPattern = toPattern(p.searchString)
|
|
|
|
p.scrollToSearchHits()
|
|
|
|
// FIXME: If the user is typing, indicate to user if we didn't find anything
|
|
}
|
|
|
|
// toPattern compiles a search string into a pattern.
|
|
//
|
|
// If the string contains only lower-case letter the pattern will be case insensitive.
|
|
//
|
|
// If the string is empty the pattern will be nil.
|
|
//
|
|
// If the string does not compile into a regexp the pattern will match the string verbatim
|
|
func toPattern(compileMe string) *regexp.Regexp {
|
|
if len(compileMe) == 0 {
|
|
return nil
|
|
}
|
|
|
|
hasUppercase := false
|
|
for _, char := range compileMe {
|
|
if unicode.IsUpper(char) {
|
|
hasUppercase = true
|
|
}
|
|
}
|
|
|
|
// Smart case; be case insensitive unless there are upper case chars
|
|
// in the search string
|
|
prefix := "(?i)"
|
|
if hasUppercase {
|
|
prefix = ""
|
|
}
|
|
|
|
pattern, err := regexp.Compile(prefix + compileMe)
|
|
if err == nil {
|
|
// Search string is a regexp
|
|
return pattern
|
|
}
|
|
|
|
pattern, err = regexp.Compile(prefix + regexp.QuoteMeta(compileMe))
|
|
if err == nil {
|
|
// Pattern matching the string exactly
|
|
return pattern
|
|
}
|
|
|
|
// Unable to create a match-string-verbatim pattern
|
|
panic(err)
|
|
}
|
|
|
|
// From: https://stackoverflow.com/a/57005674/473672
|
|
func removeLastChar(s string) string {
|
|
r, size := utf8.DecodeLastRuneInString(s)
|
|
if r == utf8.RuneError && (size == 0 || size == 1) {
|
|
size = 0
|
|
}
|
|
return s[:len(s)-size]
|
|
}
|
|
|
|
func (p *Pager) scrollToEnd() {
|
|
p.scrollPosition = scrollPosition{
|
|
lineNumberOneBased: p.reader.GetLineCount(),
|
|
// FIXME: Set deltaScreenLines as well so we're *really* at the end
|
|
}
|
|
}
|
|
|
|
func (p *Pager) onSearchKey(key twin.KeyCode) {
|
|
switch key {
|
|
case twin.KeyEscape, twin.KeyEnter:
|
|
p.mode = _Viewing
|
|
|
|
case twin.KeyBackspace, twin.KeyDelete:
|
|
if len(p.searchString) == 0 {
|
|
return
|
|
}
|
|
|
|
p.searchString = removeLastChar(p.searchString)
|
|
p.updateSearchPattern()
|
|
|
|
case twin.KeyUp:
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(1)
|
|
p.mode = _Viewing
|
|
|
|
case twin.KeyDown:
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.NextLine(1)
|
|
p.mode = _Viewing
|
|
|
|
case twin.KeyPgUp:
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(height - 1)
|
|
p.mode = _Viewing
|
|
|
|
case twin.KeyPgDown:
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.NextLine(height - 1)
|
|
p.mode = _Viewing
|
|
|
|
default:
|
|
log.Debugf("Unhandled search key event %v", key)
|
|
}
|
|
}
|
|
|
|
func (p *Pager) moveRight(delta int) {
|
|
if p.ShowLineNumbers && delta > 0 {
|
|
p.ShowLineNumbers = false
|
|
return
|
|
}
|
|
|
|
if p.leftColumnZeroBased == 0 && delta < 0 {
|
|
p.ShowLineNumbers = true
|
|
return
|
|
}
|
|
|
|
result := p.leftColumnZeroBased + delta
|
|
if result < 0 {
|
|
p.leftColumnZeroBased = 0
|
|
} else {
|
|
p.leftColumnZeroBased = result
|
|
}
|
|
}
|
|
|
|
func (p *Pager) onKey(keyCode twin.KeyCode) {
|
|
if p.mode == _Searching {
|
|
p.onSearchKey(keyCode)
|
|
return
|
|
}
|
|
if p.mode != _Viewing && p.mode != _NotFound {
|
|
panic(fmt.Sprint("Unhandled mode: ", p.mode))
|
|
}
|
|
|
|
// Reset the not-found marker on non-search keypresses
|
|
p.mode = _Viewing
|
|
|
|
switch keyCode {
|
|
case twin.KeyEscape:
|
|
p.Quit()
|
|
|
|
case twin.KeyUp:
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(1)
|
|
|
|
case twin.KeyDown, twin.KeyEnter:
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.NextLine(1)
|
|
|
|
case twin.KeyRight:
|
|
p.moveRight(16)
|
|
|
|
case twin.KeyLeft:
|
|
p.moveRight(-16)
|
|
|
|
case twin.KeyHome:
|
|
p.scrollPosition = scrollPosition{}
|
|
|
|
case twin.KeyEnd:
|
|
p.scrollToEnd()
|
|
|
|
case twin.KeyPgDown:
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.NextLine(height - 1)
|
|
|
|
case twin.KeyPgUp:
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(height - 1)
|
|
|
|
default:
|
|
log.Debugf("Unhandled key event %v", keyCode)
|
|
}
|
|
}
|
|
|
|
func (p *Pager) onSearchRune(char rune) {
|
|
p.searchString = p.searchString + string(char)
|
|
p.updateSearchPattern()
|
|
}
|
|
|
|
func (p *Pager) onRune(char rune) {
|
|
if p.mode == _Searching {
|
|
p.onSearchRune(char)
|
|
return
|
|
}
|
|
if p.mode != _Viewing && p.mode != _NotFound {
|
|
panic(fmt.Sprint("Unhandled mode: ", p.mode))
|
|
}
|
|
|
|
switch char {
|
|
case 'q':
|
|
p.Quit()
|
|
|
|
case '?':
|
|
if !p.isShowingHelp {
|
|
p.preHelpState = &_PreHelpState{
|
|
reader: p.reader,
|
|
scrollPosition: p.scrollPosition,
|
|
leftColumnZeroBased: p.leftColumnZeroBased,
|
|
}
|
|
p.reader = _HelpReader
|
|
p.scrollPosition = scrollPosition{}
|
|
p.leftColumnZeroBased = 0
|
|
p.isShowingHelp = true
|
|
}
|
|
|
|
case 'k', 'y':
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(1)
|
|
|
|
case 'j', 'e':
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.NextLine(1)
|
|
|
|
case 'l':
|
|
// vim right
|
|
p.moveRight(16)
|
|
|
|
case 'h':
|
|
// vim left
|
|
p.moveRight(-16)
|
|
|
|
case '<', 'g':
|
|
p.scrollPosition = scrollPosition{}
|
|
|
|
case '>', 'G':
|
|
p.scrollToEnd()
|
|
|
|
case 'f', ' ':
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.NextLine(height - 1)
|
|
|
|
case 'b':
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(height - 1)
|
|
|
|
case 'u':
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(height / 2)
|
|
|
|
case 'd':
|
|
_, height := p.screen.Size()
|
|
p.scrollPosition = p.scrollPosition.NextLine(height / 2)
|
|
|
|
case '/':
|
|
p.mode = _Searching
|
|
p.searchString = ""
|
|
p.searchPattern = nil
|
|
|
|
case 'n':
|
|
p.scrollToNextSearchHit()
|
|
|
|
case 'p', 'N':
|
|
p.scrollToPreviousSearchHit()
|
|
|
|
case 'w':
|
|
p.WrapLongLines = !p.WrapLongLines
|
|
|
|
default:
|
|
log.Debugf("Unhandled rune keypress '%s'", string(char))
|
|
}
|
|
}
|
|
|
|
// StartPaging brings up the pager on screen
|
|
func (p *Pager) StartPaging(screen twin.Screen) {
|
|
unprintableStyle = p.UnprintableStyle
|
|
SetManPageFormatFromEnv()
|
|
|
|
p.screen = screen
|
|
|
|
go func() {
|
|
for {
|
|
// Wait for new lines to appear...
|
|
<-p.reader.moreLinesAdded
|
|
|
|
// ... and notify the main loop so it can show them:
|
|
screen.Events() <- eventMoreLinesAvailable{}
|
|
|
|
// Delay updates a bit so that we don't waste time refreshing
|
|
// the screen too often.
|
|
//
|
|
// Note that the delay is *after* reacting, this way single-line
|
|
// updates are reacted to immediately, and the first output line
|
|
// read will appear on screen without delay.
|
|
time.Sleep(200 * time.Millisecond)
|
|
}
|
|
}()
|
|
|
|
go func() {
|
|
// Spin the spinner as long as contents is still loading
|
|
done := false
|
|
spinnerFrames := [...]string{"/.\\", "-o-", "\\O/", "| |"}
|
|
spinnerIndex := 0
|
|
for {
|
|
// Break this loop on the reader.done signal...
|
|
select {
|
|
case <-p.reader.done:
|
|
done = true
|
|
default:
|
|
// This default case makes this an async read
|
|
}
|
|
|
|
if done {
|
|
break
|
|
}
|
|
|
|
screen.Events() <- eventSpinnerUpdate{spinnerFrames[spinnerIndex]}
|
|
spinnerIndex++
|
|
if spinnerIndex >= len(spinnerFrames) {
|
|
spinnerIndex = 0
|
|
}
|
|
|
|
time.Sleep(200 * time.Millisecond)
|
|
}
|
|
|
|
// Empty our spinner, loading done!
|
|
screen.Events() <- eventSpinnerUpdate{""}
|
|
}()
|
|
|
|
// Main loop
|
|
spinner := ""
|
|
for !p.quit {
|
|
if len(screen.Events()) == 0 {
|
|
// Nothing more to process for now, redraw the screen!
|
|
p.redraw(spinner)
|
|
}
|
|
|
|
event := <-screen.Events()
|
|
switch event := event.(type) {
|
|
case twin.EventKeyCode:
|
|
log.Tracef("Handling key event %d...", event.KeyCode())
|
|
p.onKey(event.KeyCode())
|
|
|
|
case twin.EventRune:
|
|
log.Tracef("Handling rune event '%c'/0x%04x...", event.Rune(), event.Rune())
|
|
p.onRune(event.Rune())
|
|
|
|
case twin.EventMouse:
|
|
log.Tracef("Handling mouse event %d...", event.Buttons())
|
|
switch event.Buttons() {
|
|
case twin.MouseWheelUp:
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.PreviousLine(1)
|
|
|
|
case twin.MouseWheelDown:
|
|
// Clipping is done in _Redraw()
|
|
p.scrollPosition = p.scrollPosition.NextLine(1)
|
|
|
|
case twin.MouseWheelLeft:
|
|
p.moveRight(-16)
|
|
|
|
case twin.MouseWheelRight:
|
|
p.moveRight(16)
|
|
}
|
|
|
|
case twin.EventResize:
|
|
// We'll be implicitly redrawn just by taking another lap in the loop
|
|
|
|
case eventMoreLinesAvailable:
|
|
// Doing nothing here is fine; screen will be refreshed on the next
|
|
// iteration of the main loop.
|
|
|
|
case eventSpinnerUpdate:
|
|
spinner = event.spinner
|
|
|
|
default:
|
|
log.Warnf("Unhandled event type: %v", event)
|
|
}
|
|
}
|
|
|
|
if p.reader.err != nil {
|
|
log.Warnf("Reader reported an error: %s", p.reader.err.Error())
|
|
}
|
|
}
|
|
|
|
// After the pager has exited and the normal screen has been restored, you can
|
|
// call this method to print the pager contents to screen again, faking
|
|
// "leaving" pager contents on screen after exit.
|
|
func (p *Pager) ReprintAfterExit() error {
|
|
// Figure out how many screen lines are used by pager contents
|
|
|
|
// FIXME: Make sure this method works with non-0 scrollPosition deltaScreenLines
|
|
|
|
_, height := p.screen.Size()
|
|
heightWithoutFooter := height - 1
|
|
lineCount := len(p.reader.GetLines(p.scrollPosition.lineNumberOneBased, heightWithoutFooter).lines)
|
|
|
|
if lineCount > 0 {
|
|
p.screen.ShowNLines(lineCount)
|
|
}
|
|
return nil
|
|
}
|