1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
|
package editor
import (
"fmt"
"os"
"os/exec"
"path/filepath"
"strings"
)
const defaultEditor = "nano"
// Option defines an editor option.
//
// An Option may act differently in some editors, or not be supported in
// some of them.
type Option func(editor, filename string) (args []string, pathInArgs bool)
// OpenAtLine opens the file at the given line number in supported editors.
func OpenAtLine(number uint) Option {
plusLineEditors := []string{"vi", "vim", "nvim", "nano", "emacs", "kak", "gedit"}
return func(editor, filename string) ([]string, bool) {
for _, e := range plusLineEditors {
if editor == e {
return []string{fmt.Sprintf("+%d", number)}, false
}
}
if editor == "code" {
return []string{
"--goto",
fmt.Sprintf("%s:%d", filename, number),
}, true
}
return nil, false
}
}
// Cmd returns a *exec.Cmd editing the given path with $EDITOR or nano if no
// $EDITOR is set.
func Cmd(app, path string, options ...Option) (*exec.Cmd, error) {
if os.Getenv("SNAP_REVISION") != "" {
return nil, fmt.Errorf("Did you install with Snap? %[1]s is sandboxed and unable to open an editor. Please install %[1]s with Go or another package manager to enable editing.", app)
}
editor, args := getEditor()
editorName := filepath.Base(editor)
needsToAppendPath := true
for _, opt := range options {
optArgs, pathInArgs := opt(editorName, path)
if pathInArgs {
needsToAppendPath = false
}
args = append(args, optArgs...)
}
if needsToAppendPath {
args = append(args, path)
}
return exec.Command(editor, args...), nil
}
func getEditor() (string, []string) {
editor := strings.Fields(os.Getenv("EDITOR"))
if len(editor) > 1 {
return editor[0], editor[1:]
}
if len(editor) == 1 {
return editor[0], []string{}
}
return defaultEditor, []string{}
}
|