mirror of
https://github.com/go-sylixos/elvish.git
synced 2024-12-13 18:07:51 +08:00
256 lines
5.3 KiB
Go
256 lines
5.3 KiB
Go
package edit
|
|
|
|
import (
|
|
"errors"
|
|
|
|
"src.elv.sh/pkg/cli"
|
|
"src.elv.sh/pkg/cli/modes"
|
|
"src.elv.sh/pkg/cli/term"
|
|
"src.elv.sh/pkg/cli/tk"
|
|
"src.elv.sh/pkg/eval"
|
|
"src.elv.sh/pkg/parse"
|
|
"src.elv.sh/pkg/parse/parseutil"
|
|
"src.elv.sh/pkg/ui"
|
|
)
|
|
|
|
//elvdoc:fn binding-table
|
|
//
|
|
// Converts a normal map into a binding map.
|
|
|
|
//elvdoc:fn -dump-buf
|
|
//
|
|
// Dumps the current UI buffer as HTML. This command is used to generate
|
|
// "ttyshots" on the [website](https://elv.sh).
|
|
//
|
|
// Example:
|
|
//
|
|
// ```elvish
|
|
// ttyshot = ~/a.html
|
|
// edit:insert:binding[Ctrl-X] = { edit:-dump-buf > $tty }
|
|
// ```
|
|
|
|
func dumpBuf(tty cli.TTY) string {
|
|
return bufToHTML(tty.Buffer())
|
|
}
|
|
|
|
//elvdoc:fn close-mode
|
|
//
|
|
// Closes the current active mode.
|
|
|
|
func closeMode(app cli.App) {
|
|
app.PopAddon()
|
|
}
|
|
|
|
//elvdoc:fn end-of-history
|
|
//
|
|
// Adds a notification saying "End of history".
|
|
|
|
func endOfHistory(app cli.App) {
|
|
app.Notify("End of history")
|
|
}
|
|
|
|
//elvdoc:fn redraw
|
|
//
|
|
// ```elvish
|
|
// edit:redraw &full=$false
|
|
// ```
|
|
//
|
|
// Triggers a redraw.
|
|
//
|
|
// The `&full` option controls whether to do a full redraw. By default, all
|
|
// redraws performed by the line editor are incremental redraws, updating only
|
|
// the part of the screen that has changed from the last redraw. A full redraw
|
|
// updates the entire command line.
|
|
|
|
type redrawOpts struct{ Full bool }
|
|
|
|
func (redrawOpts) SetDefaultOptions() {}
|
|
|
|
func redraw(app cli.App, opts redrawOpts) {
|
|
if opts.Full {
|
|
app.RedrawFull()
|
|
} else {
|
|
app.Redraw()
|
|
}
|
|
}
|
|
|
|
//elvdoc:fn clear
|
|
//
|
|
// ```elvish
|
|
// edit:clear
|
|
// ```
|
|
//
|
|
// Clears the screen.
|
|
//
|
|
// This command should be used in place of the external `clear` command to clear
|
|
// the screen.
|
|
|
|
func clear(app cli.App, tty cli.TTY) {
|
|
tty.HideCursor()
|
|
tty.ClearScreen()
|
|
app.RedrawFull()
|
|
tty.ShowCursor()
|
|
}
|
|
|
|
//elvdoc:fn insert-raw
|
|
//
|
|
// Requests the next terminal input to be inserted uninterpreted.
|
|
|
|
func insertRaw(app cli.App, tty cli.TTY) {
|
|
codeArea, ok := focusedCodeArea(app)
|
|
if !ok {
|
|
return
|
|
}
|
|
tty.SetRawInput(1)
|
|
w := modes.NewStub(modes.StubSpec{
|
|
Bindings: tk.FuncBindings(func(w tk.Widget, event term.Event) bool {
|
|
switch event := event.(type) {
|
|
case term.KeyEvent:
|
|
codeArea.MutateState(func(s *tk.CodeAreaState) {
|
|
s.Buffer.InsertAtDot(string(event.Rune))
|
|
})
|
|
app.PopAddon()
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}),
|
|
Name: " RAW ",
|
|
})
|
|
app.PushAddon(w)
|
|
}
|
|
|
|
//elvdoc:fn key
|
|
//
|
|
// ```elvish
|
|
// edit:key $string
|
|
// ```
|
|
//
|
|
// Parses a string into a key.
|
|
|
|
var errMustBeKeyOrString = errors.New("must be key or string")
|
|
|
|
func toKey(v interface{}) (ui.Key, error) {
|
|
switch v := v.(type) {
|
|
case ui.Key:
|
|
return v, nil
|
|
case string:
|
|
return ui.ParseKey(v)
|
|
default:
|
|
return ui.Key{}, errMustBeKeyOrString
|
|
}
|
|
}
|
|
|
|
//elvdoc:fn notify
|
|
//
|
|
// ```elvish
|
|
// edit:notify $message
|
|
// ```
|
|
//
|
|
// Prints a notification message.
|
|
//
|
|
// If called while the editor is active, this will print the message above the
|
|
// editor, and redraw the editor.
|
|
//
|
|
// If called while the editor is inactive, the message will be queued, and shown
|
|
// once the editor becomes active.
|
|
|
|
//elvdoc:fn return-line
|
|
//
|
|
// Causes the Elvish REPL to end the current read iteration and evaluate the
|
|
// code it just read. If called from a key binding, takes effect after the key
|
|
// binding returns.
|
|
|
|
//elvdoc:fn return-eof
|
|
//
|
|
// Causes the Elvish REPL to terminate. If called from a key binding, takes
|
|
// effect after the key binding returns.
|
|
|
|
//elvdoc:fn smart-enter
|
|
//
|
|
// Inserts a literal newline if the current code is not syntactically complete
|
|
// Elvish code. Accepts the current line otherwise.
|
|
|
|
func smartEnter(app cli.App) {
|
|
codeArea, ok := focusedCodeArea(app)
|
|
if !ok {
|
|
return
|
|
}
|
|
commit := false
|
|
codeArea.MutateState(func(s *tk.CodeAreaState) {
|
|
buf := &s.Buffer
|
|
if isSyntaxComplete(buf.Content) {
|
|
commit = true
|
|
} else {
|
|
buf.InsertAtDot("\n")
|
|
}
|
|
})
|
|
if commit {
|
|
app.CommitCode()
|
|
}
|
|
}
|
|
|
|
func isSyntaxComplete(code string) bool {
|
|
_, err := parse.Parse(parse.Source{Code: code}, parse.Config{})
|
|
if err != nil {
|
|
for _, e := range err.(*parse.Error).Entries {
|
|
if e.Context.From == len(code) {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
//elvdoc:fn wordify
|
|
//
|
|
//
|
|
// ```elvish
|
|
// edit:wordify $code
|
|
// ```
|
|
// Breaks Elvish code into words.
|
|
|
|
func wordify(fm *eval.Frame, code string) error {
|
|
out := fm.ValueOutput()
|
|
for _, s := range parseutil.Wordify(code) {
|
|
err := out.Put(s)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func initTTYBuiltins(app cli.App, tty cli.TTY, nb eval.NsBuilder) {
|
|
nb.AddGoFns(map[string]interface{}{
|
|
"-dump-buf": func() string { return dumpBuf(tty) },
|
|
"insert-raw": func() { insertRaw(app, tty) },
|
|
"clear": func() { clear(app, tty) },
|
|
})
|
|
}
|
|
|
|
func initMiscBuiltins(app cli.App, nb eval.NsBuilder) {
|
|
nb.AddGoFns(map[string]interface{}{
|
|
"binding-table": makeBindingMap,
|
|
"close-mode": func() { closeMode(app) },
|
|
"end-of-history": func() { endOfHistory(app) },
|
|
"key": toKey,
|
|
"notify": app.Notify,
|
|
"redraw": func(opts redrawOpts) { redraw(app, opts) },
|
|
"return-line": app.CommitCode,
|
|
"return-eof": app.CommitEOF,
|
|
"smart-enter": func() { smartEnter(app) },
|
|
"wordify": wordify,
|
|
})
|
|
}
|
|
|
|
// Like mode.FocusedCodeArea, but handles the error by writing a notification.
|
|
func focusedCodeArea(app cli.App) (tk.CodeArea, bool) {
|
|
codeArea, err := modes.FocusedCodeArea(app)
|
|
if err != nil {
|
|
app.Notify(err.Error())
|
|
return nil, false
|
|
}
|
|
return codeArea, true
|
|
}
|