fzf/src/options.go

682 lines
17 KiB
Go
Raw Normal View History

2015-01-02 04:49:30 +09:00
package fzf
import (
"fmt"
"os"
"regexp"
2015-05-31 16:46:54 +09:00
"strconv"
2015-01-02 04:49:30 +09:00
"strings"
"unicode/utf8"
"github.com/junegunn/fzf/src/curses"
2015-01-12 12:56:17 +09:00
"github.com/junegunn/go-shellwords"
2015-01-02 04:49:30 +09:00
)
2015-01-12 03:01:24 +09:00
const usage = `usage: fzf [options]
2015-01-02 04:49:30 +09:00
2015-06-08 23:28:41 +09:00
Search
2015-01-02 04:49:30 +09:00
-x, --extended Extended-search mode
-e, --extended-exact Extended-search mode (exact match)
-i Case-insensitive match (default: smart-case match)
+i Case-sensitive match
-n, --nth=N[,..] Comma-separated list of field index expressions
for limiting search scope. Each can be a non-zero
integer or a range expression ([BEGIN]..[END])
--with-nth=N[,..] Transform the item using index expressions for search
-d, --delimiter=STR Field delimiter regex for --nth (default: AWK-style)
+s, --no-sort Do not sort the result
--tac Reverse the order of the input
--tiebreak=CRI Sort criterion when the scores are tied;
[length|begin|end|index] (default: length)
2015-01-02 04:49:30 +09:00
Interface
-m, --multi Enable multi-select with tab/shift-tab
2015-03-22 21:25:46 +09:00
--ansi Enable processing of ANSI color codes
2015-01-02 04:49:30 +09:00
--no-mouse Disable mouse
2015-05-31 16:46:54 +09:00
--color=COLSPEC Base scheme (dark|light|16|bw) and/or custom colors
2015-01-02 04:49:30 +09:00
--black Use black background
--reverse Reverse orientation
--no-hscroll Disable horizontal scroll
2015-04-21 23:50:53 +09:00
--inline-info Display finder info inline with the query
2015-01-02 04:49:30 +09:00
--prompt=STR Input prompt (default: '> ')
2015-05-21 01:51:24 +09:00
--toggle-sort=KEY Key to toggle sort
--bind=KEYBINDS Custom key bindings. Refer to the man page.
--history=FILE History file
--history-max=N Maximum number of history entries (default: 1000)
2015-01-02 04:49:30 +09:00
Scripting
-q, --query=STR Start the finder with the given query
-1, --select-1 Automatically select the only match
-0, --exit-0 Exit immediately when there's no match
-f, --filter=STR Filter mode. Do not start interactive finder.
--null Read null-byte separated strings from input
2015-01-02 04:49:30 +09:00
--print-query Print query as the first line
--expect=KEYS Comma-separated list of keys to complete fzf
2015-02-13 12:25:19 +09:00
--sync Synchronous search for multi-staged filtering
2015-01-02 04:49:30 +09:00
Environment variables
FZF_DEFAULT_COMMAND Default command to use when input is tty
2015-02-13 12:25:19 +09:00
FZF_DEFAULT_OPTS Defaults options. (e.g. '-x -m')
2015-01-02 04:49:30 +09:00
`
2015-01-12 03:01:24 +09:00
// Mode denotes the current search mode
2015-01-02 04:49:30 +09:00
type Mode int
2015-01-12 03:01:24 +09:00
// Search modes
2015-01-02 04:49:30 +09:00
const (
2015-01-12 03:01:24 +09:00
ModeFuzzy Mode = iota
ModeExtended
ModeExtendedExact
2015-01-02 04:49:30 +09:00
)
2015-01-12 03:01:24 +09:00
// Case denotes case-sensitivity of search
2015-01-02 04:49:30 +09:00
type Case int
2015-01-12 03:01:24 +09:00
// Case-sensitivities
2015-01-02 04:49:30 +09:00
const (
2015-01-12 03:01:24 +09:00
CaseSmart Case = iota
CaseIgnore
CaseRespect
2015-01-02 04:49:30 +09:00
)
// Sort criteria
type tiebreak int
const (
byLength tiebreak = iota
byBegin
byEnd
byIndex
)
2015-01-12 03:01:24 +09:00
// Options stores the values of command-line options
2015-01-02 04:49:30 +09:00
type Options struct {
Mode Mode
Case Case
Nth []Range
WithNth []Range
Delimiter *regexp.Regexp
Sort int
Tac bool
Tiebreak tiebreak
2015-01-02 04:49:30 +09:00
Multi bool
2015-03-19 01:59:14 +09:00
Ansi bool
2015-01-02 04:49:30 +09:00
Mouse bool
Theme *curses.ColorTheme
2015-01-02 04:49:30 +09:00
Black bool
Reverse bool
Hscroll bool
2015-04-21 23:50:53 +09:00
InlineInfo bool
2015-01-02 04:49:30 +09:00
Prompt string
Query string
Select1 bool
Exit0 bool
Filter *string
2015-05-20 21:25:15 +09:00
ToggleSort bool
Expect []int
2015-05-20 21:25:15 +09:00
Keymap map[int]actionType
2015-01-02 04:49:30 +09:00
PrintQuery bool
ReadZero bool
2015-02-13 12:25:19 +09:00
Sync bool
History *History
2015-01-02 04:49:30 +09:00
Version bool
}
2015-05-31 16:46:54 +09:00
func defaultTheme() *curses.ColorTheme {
if strings.Contains(os.Getenv("TERM"), "256") {
2015-05-31 16:46:54 +09:00
return curses.Dark256
}
2015-05-31 16:46:54 +09:00
return curses.Default16
}
2015-05-31 16:46:54 +09:00
func defaultOptions() *Options {
2015-01-02 04:49:30 +09:00
return &Options{
2015-01-12 03:01:24 +09:00
Mode: ModeFuzzy,
Case: CaseSmart,
2015-01-02 04:49:30 +09:00
Nth: make([]Range, 0),
WithNth: make([]Range, 0),
Delimiter: nil,
Sort: 1000,
Tac: false,
Tiebreak: byLength,
2015-01-02 04:49:30 +09:00
Multi: false,
2015-03-19 01:59:14 +09:00
Ansi: false,
2015-01-02 04:49:30 +09:00
Mouse: true,
2015-05-31 16:46:54 +09:00
Theme: defaultTheme(),
2015-01-02 04:49:30 +09:00
Black: false,
Reverse: false,
Hscroll: true,
2015-04-21 23:50:53 +09:00
InlineInfo: false,
2015-01-02 04:49:30 +09:00
Prompt: "> ",
Query: "",
Select1: false,
Exit0: false,
Filter: nil,
2015-05-20 21:25:15 +09:00
ToggleSort: false,
Expect: []int{},
2015-05-20 21:25:15 +09:00
Keymap: defaultKeymap(),
2015-01-02 04:49:30 +09:00
PrintQuery: false,
ReadZero: false,
2015-02-13 12:25:19 +09:00
Sync: false,
History: nil,
2015-01-02 04:49:30 +09:00
Version: false}
}
func help(ok int) {
2015-01-12 03:01:24 +09:00
os.Stderr.WriteString(usage)
2015-01-02 04:49:30 +09:00
os.Exit(ok)
}
func errorExit(msg string) {
os.Stderr.WriteString(msg + "\n")
help(1)
}
func optString(arg string, prefixes ...string) (bool, string) {
for _, prefix := range prefixes {
if strings.HasPrefix(arg, prefix) {
return true, arg[len(prefix):]
}
2015-01-02 04:49:30 +09:00
}
2015-01-12 03:01:24 +09:00
return false, ""
2015-01-02 04:49:30 +09:00
}
func nextString(args []string, i *int, message string) string {
if len(args) > *i+1 {
*i++
} else {
errorExit(message)
}
return args[*i]
}
2015-05-31 16:46:54 +09:00
func optionalNextString(args []string, i *int) string {
if len(args) > *i+1 {
*i++
return args[*i]
}
return ""
}
func atoi(str string) int {
num, err := strconv.Atoi(str)
if err != nil {
errorExit("not a valid integer: " + str)
}
return num
}
func nextInt(args []string, i *int, message string) int {
if len(args) > *i+1 {
*i++
} else {
errorExit(message)
}
return atoi(args[*i])
}
2015-01-02 04:49:30 +09:00
func optionalNumeric(args []string, i *int) int {
if len(args) > *i+1 {
if strings.IndexAny(args[*i+1], "0123456789") == 0 {
*i++
}
}
return 1 // Don't care
}
func splitNth(str string) []Range {
if match, _ := regexp.MatchString("^[0-9,-.]+$", str); !match {
errorExit("invalid format: " + str)
}
tokens := strings.Split(str, ",")
ranges := make([]Range, len(tokens))
for idx, s := range tokens {
r, ok := ParseRange(&s)
if !ok {
errorExit("invalid format: " + str)
}
ranges[idx] = r
}
return ranges
}
func delimiterRegexp(str string) *regexp.Regexp {
rx, e := regexp.Compile(str)
if e != nil {
str = regexp.QuoteMeta(str)
}
rx, e = regexp.Compile(fmt.Sprintf("(?:.*?%s)|(?:.+?$)", str))
if e != nil {
errorExit("invalid regular expression: " + e.Error())
}
return rx
}
func isAlphabet(char uint8) bool {
return char >= 'a' && char <= 'z'
}
2015-03-31 22:05:02 +09:00
func parseKeyChords(str string, message string) []int {
if len(str) == 0 {
errorExit(message)
}
tokens := strings.Split(str, ",")
if str == "," || strings.HasPrefix(str, ",,") || strings.HasSuffix(str, ",,") || strings.Index(str, ",,,") >= 0 {
tokens = append(tokens, ",")
}
var chords []int
2015-03-31 22:05:02 +09:00
for _, key := range tokens {
if len(key) == 0 {
continue // ignore
}
lkey := strings.ToLower(key)
if len(key) == 6 && strings.HasPrefix(lkey, "ctrl-") && isAlphabet(lkey[5]) {
chords = append(chords, curses.CtrlA+int(lkey[5])-'a')
} else if len(key) == 5 && strings.HasPrefix(lkey, "alt-") && isAlphabet(lkey[4]) {
chords = append(chords, curses.AltA+int(lkey[4])-'a')
} else if len(key) == 2 && strings.HasPrefix(lkey, "f") && key[1] >= '1' && key[1] <= '4' {
chords = append(chords, curses.F1+int(key[1])-'1')
} else if utf8.RuneCountInString(key) == 1 {
chords = append(chords, curses.AltZ+int([]rune(key)[0]))
} else {
errorExit("unsupported key: " + key)
}
}
return chords
}
func parseTiebreak(str string) tiebreak {
switch strings.ToLower(str) {
case "length":
return byLength
case "index":
return byIndex
case "begin":
return byBegin
case "end":
return byEnd
default:
errorExit("invalid sort criterion: " + str)
}
return byLength
}
2015-05-31 16:46:54 +09:00
func dupeTheme(theme *curses.ColorTheme) *curses.ColorTheme {
dupe := *theme
return &dupe
}
func parseTheme(defaultTheme *curses.ColorTheme, str string) *curses.ColorTheme {
theme := dupeTheme(defaultTheme)
for _, str := range strings.Split(strings.ToLower(str), ",") {
switch str {
case "dark":
theme = dupeTheme(curses.Dark256)
case "light":
theme = dupeTheme(curses.Light256)
case "16":
theme = dupeTheme(curses.Default16)
case "bw", "no":
theme = nil
default:
fail := func() {
errorExit("invalid color specification: " + str)
}
// Color is disabled
if theme == nil {
errorExit("colors disabled; cannot customize colors")
}
pair := strings.Split(str, ":")
if len(pair) != 2 {
fail()
}
ansi32, err := strconv.Atoi(pair[1])
if err != nil || ansi32 < -1 || ansi32 > 255 {
fail()
}
ansi := int16(ansi32)
switch pair[0] {
case "fg":
theme.Fg = ansi
theme.UseDefault = theme.UseDefault && ansi < 0
case "bg":
theme.Bg = ansi
theme.UseDefault = theme.UseDefault && ansi < 0
case "fg+":
theme.Current = ansi
case "bg+":
theme.DarkBg = ansi
case "hl":
theme.Match = ansi
case "hl+":
theme.CurrentMatch = ansi
case "prompt":
theme.Prompt = ansi
case "spinner":
theme.Spinner = ansi
case "info":
theme.Info = ansi
case "pointer":
theme.Cursor = ansi
case "marker":
theme.Selected = ansi
default:
fail()
}
}
}
2015-05-31 16:46:54 +09:00
return theme
}
2015-05-20 21:25:15 +09:00
func parseKeymap(keymap map[int]actionType, toggleSort bool, str string) (map[int]actionType, bool) {
for _, pairStr := range strings.Split(str, ",") {
2015-05-22 00:02:14 +09:00
fail := func() {
errorExit("invalid key binding: " + pairStr)
}
2015-05-20 21:25:15 +09:00
pair := strings.Split(pairStr, ":")
if len(pair) != 2 {
2015-05-22 00:02:14 +09:00
fail()
2015-05-20 21:25:15 +09:00
}
keys := parseKeyChords(pair[0], "key name required")
if len(keys) != 1 {
2015-05-22 00:02:14 +09:00
fail()
2015-05-20 21:25:15 +09:00
}
key := keys[0]
act := strings.ToLower(pair[1])
2015-05-22 00:02:14 +09:00
switch act {
2015-06-14 01:54:56 +09:00
case "ignore":
keymap[key] = actIgnore
2015-05-20 21:25:15 +09:00
case "beginning-of-line":
keymap[key] = actBeginningOfLine
case "abort":
keymap[key] = actAbort
case "accept":
keymap[key] = actAccept
case "backward-char":
keymap[key] = actBackwardChar
case "backward-delete-char":
keymap[key] = actBackwardDeleteChar
case "backward-word":
keymap[key] = actBackwardWord
case "clear-screen":
keymap[key] = actClearScreen
case "delete-char":
keymap[key] = actDeleteChar
case "end-of-line":
keymap[key] = actEndOfLine
case "forward-char":
keymap[key] = actForwardChar
case "forward-word":
keymap[key] = actForwardWord
case "kill-line":
keymap[key] = actKillLine
case "kill-word":
keymap[key] = actKillWord
case "unix-line-discard", "line-discard":
keymap[key] = actUnixLineDiscard
case "unix-word-rubout", "word-rubout":
keymap[key] = actUnixWordRubout
case "yank":
keymap[key] = actYank
case "backward-kill-word":
keymap[key] = actBackwardKillWord
case "toggle-down":
keymap[key] = actToggleDown
case "toggle-up":
keymap[key] = actToggleUp
case "toggle-all":
keymap[key] = actToggleAll
case "select-all":
keymap[key] = actSelectAll
case "deselect-all":
keymap[key] = actDeselectAll
case "toggle":
keymap[key] = actToggle
2015-05-20 21:25:15 +09:00
case "down":
keymap[key] = actDown
case "up":
keymap[key] = actUp
case "page-up":
keymap[key] = actPageUp
case "page-down":
keymap[key] = actPageDown
case "previous-history":
keymap[key] = actPreviousHistory
case "next-history":
keymap[key] = actNextHistory
2015-05-20 21:25:15 +09:00
case "toggle-sort":
keymap[key] = actToggleSort
toggleSort = true
default:
errorExit("unknown action: " + act)
}
}
return keymap, toggleSort
}
func checkToggleSort(keymap map[int]actionType, str string) map[int]actionType {
2015-03-31 22:05:02 +09:00
keys := parseKeyChords(str, "key name required")
if len(keys) != 1 {
errorExit("multiple keys specified")
}
2015-05-20 21:25:15 +09:00
keymap[keys[0]] = actToggleSort
return keymap
2015-03-31 22:05:02 +09:00
}
2015-01-02 04:49:30 +09:00
func parseOptions(opts *Options, allArgs []string) {
keymap := make(map[int]actionType)
var historyMax int
if opts.History == nil {
historyMax = defaultHistoryMax
} else {
historyMax = opts.History.maxSize
}
setHistory := func(path string) {
h, e := NewHistory(path, historyMax)
if e != nil {
errorExit(e.Error())
}
opts.History = h
}
setHistoryMax := func(max int) {
historyMax = max
if historyMax < 1 {
errorExit("history max must be a positive integer")
}
if opts.History != nil {
opts.History.maxSize = historyMax
}
}
2015-01-02 04:49:30 +09:00
for i := 0; i < len(allArgs); i++ {
arg := allArgs[i]
switch arg {
case "-h", "--help":
help(0)
case "-x", "--extended":
2015-01-12 03:01:24 +09:00
opts.Mode = ModeExtended
2015-01-02 04:49:30 +09:00
case "-e", "--extended-exact":
2015-01-12 03:01:24 +09:00
opts.Mode = ModeExtendedExact
2015-01-02 04:49:30 +09:00
case "+x", "--no-extended", "+e", "--no-extended-exact":
2015-01-12 03:01:24 +09:00
opts.Mode = ModeFuzzy
2015-01-02 04:49:30 +09:00
case "-q", "--query":
opts.Query = nextString(allArgs, &i, "query string required")
case "-f", "--filter":
filter := nextString(allArgs, &i, "query string required")
opts.Filter = &filter
case "--expect":
2015-03-31 22:05:02 +09:00
opts.Expect = parseKeyChords(nextString(allArgs, &i, "key names required"), "key names required")
case "--tiebreak":
opts.Tiebreak = parseTiebreak(nextString(allArgs, &i, "sort criterion required"))
2015-05-20 21:25:15 +09:00
case "--bind":
keymap, opts.ToggleSort = parseKeymap(keymap, opts.ToggleSort, nextString(allArgs, &i, "bind expression required"))
case "--color":
2015-05-31 16:46:54 +09:00
spec := optionalNextString(allArgs, &i)
if len(spec) == 0 {
opts.Theme = defaultTheme()
} else {
opts.Theme = parseTheme(opts.Theme, spec)
}
2015-03-31 22:05:02 +09:00
case "--toggle-sort":
keymap = checkToggleSort(keymap, nextString(allArgs, &i, "key name required"))
2015-05-20 21:25:15 +09:00
opts.ToggleSort = true
2015-01-02 04:49:30 +09:00
case "-d", "--delimiter":
opts.Delimiter = delimiterRegexp(nextString(allArgs, &i, "delimiter required"))
case "-n", "--nth":
opts.Nth = splitNth(nextString(allArgs, &i, "nth expression required"))
case "--with-nth":
opts.WithNth = splitNth(nextString(allArgs, &i, "nth expression required"))
case "-s", "--sort":
opts.Sort = optionalNumeric(allArgs, &i)
case "+s", "--no-sort":
opts.Sort = 0
case "--tac":
opts.Tac = true
case "--no-tac":
opts.Tac = false
2015-01-02 04:49:30 +09:00
case "-i":
2015-01-12 03:01:24 +09:00
opts.Case = CaseIgnore
2015-01-02 04:49:30 +09:00
case "+i":
2015-01-12 03:01:24 +09:00
opts.Case = CaseRespect
2015-01-02 04:49:30 +09:00
case "-m", "--multi":
opts.Multi = true
case "+m", "--no-multi":
opts.Multi = false
2015-03-19 01:59:14 +09:00
case "--ansi":
opts.Ansi = true
case "--no-ansi":
opts.Ansi = false
2015-01-02 04:49:30 +09:00
case "--no-mouse":
opts.Mouse = false
case "+c", "--no-color":
opts.Theme = nil
2015-01-02 04:49:30 +09:00
case "+2", "--no-256":
opts.Theme = curses.Default16
2015-01-02 04:49:30 +09:00
case "--black":
opts.Black = true
case "--no-black":
opts.Black = false
case "--reverse":
opts.Reverse = true
case "--no-reverse":
opts.Reverse = false
case "--hscroll":
opts.Hscroll = true
case "--no-hscroll":
opts.Hscroll = false
2015-04-21 23:50:53 +09:00
case "--inline-info":
opts.InlineInfo = true
case "--no-inline-info":
opts.InlineInfo = false
2015-01-02 04:49:30 +09:00
case "-1", "--select-1":
opts.Select1 = true
case "+1", "--no-select-1":
opts.Select1 = false
case "-0", "--exit-0":
opts.Exit0 = true
case "+0", "--no-exit-0":
opts.Exit0 = false
case "--null":
opts.ReadZero = true
2015-01-02 04:49:30 +09:00
case "--print-query":
opts.PrintQuery = true
case "--no-print-query":
opts.PrintQuery = false
case "--prompt":
opts.Prompt = nextString(allArgs, &i, "prompt string required")
2015-02-13 12:25:19 +09:00
case "--sync":
opts.Sync = true
case "--no-sync":
opts.Sync = false
case "--async":
opts.Sync = false
case "--no-history":
opts.History = nil
case "--history":
setHistory(nextString(allArgs, &i, "history file path required"))
case "--history-max":
setHistoryMax(nextInt(allArgs, &i, "history max size required"))
2015-01-02 04:49:30 +09:00
case "--version":
opts.Version = true
default:
if match, value := optString(arg, "-q", "--query="); match {
2015-01-02 04:49:30 +09:00
opts.Query = value
} else if match, value := optString(arg, "-f", "--filter="); match {
2015-01-02 04:49:30 +09:00
opts.Filter = &value
} else if match, value := optString(arg, "-d", "--delimiter="); match {
2015-01-02 04:49:30 +09:00
opts.Delimiter = delimiterRegexp(value)
} else if match, value := optString(arg, "--prompt="); match {
opts.Prompt = value
} else if match, value := optString(arg, "-n", "--nth="); match {
2015-01-02 04:49:30 +09:00
opts.Nth = splitNth(value)
} else if match, value := optString(arg, "--with-nth="); match {
opts.WithNth = splitNth(value)
} else if match, _ := optString(arg, "-s", "--sort="); match {
2015-01-02 04:49:30 +09:00
opts.Sort = 1 // Don't care
2015-03-31 22:05:02 +09:00
} else if match, value := optString(arg, "--toggle-sort="); match {
keymap = checkToggleSort(keymap, value)
2015-05-20 21:25:15 +09:00
opts.ToggleSort = true
} else if match, value := optString(arg, "--expect="); match {
2015-03-31 22:05:02 +09:00
opts.Expect = parseKeyChords(value, "key names required")
} else if match, value := optString(arg, "--tiebreak="); match {
opts.Tiebreak = parseTiebreak(value)
} else if match, value := optString(arg, "--color="); match {
2015-05-31 16:46:54 +09:00
opts.Theme = parseTheme(opts.Theme, value)
2015-05-20 21:25:15 +09:00
} else if match, value := optString(arg, "--bind="); match {
keymap, opts.ToggleSort = parseKeymap(keymap, opts.ToggleSort, value)
} else if match, value := optString(arg, "--history="); match {
setHistory(value)
} else if match, value := optString(arg, "--history-max="); match {
setHistoryMax(atoi(value))
2015-01-02 04:49:30 +09:00
} else {
errorExit("unknown option: " + arg)
}
}
}
// Change default actions for CTRL-N / CTRL-P when --history is used
if opts.History != nil {
if _, prs := keymap[curses.CtrlP]; !prs {
keymap[curses.CtrlP] = actPreviousHistory
}
if _, prs := keymap[curses.CtrlN]; !prs {
keymap[curses.CtrlN] = actNextHistory
}
}
// Override default key bindings
for key, act := range keymap {
opts.Keymap[key] = act
}
// If we're not using extended search mode, --nth option becomes irrelevant
// if it contains the whole range
if opts.Mode == ModeFuzzy || len(opts.Nth) == 1 {
for _, r := range opts.Nth {
if r.begin == rangeEllipsis && r.end == rangeEllipsis {
opts.Nth = make([]Range, 0)
return
}
}
}
2015-01-02 04:49:30 +09:00
}
2015-01-12 03:01:24 +09:00
// ParseOptions parses command-line options
2015-01-02 04:49:30 +09:00
func ParseOptions() *Options {
2015-01-12 03:01:24 +09:00
opts := defaultOptions()
2015-01-02 04:49:30 +09:00
// Options from Env var
words, _ := shellwords.Parse(os.Getenv("FZF_DEFAULT_OPTS"))
parseOptions(opts, words)
// Options from command-line arguments
parseOptions(opts, os.Args[1:])
return opts
}