mirror of
https://github.com/Llewellynvdm/fzf.git
synced 2024-12-23 19:39:07 +00:00
parent
3f0e6a5806
commit
e82eb27787
@ -4,7 +4,6 @@ import (
|
|||||||
"regexp"
|
"regexp"
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"unicode"
|
|
||||||
|
|
||||||
"github.com/junegunn/fzf/src/algo"
|
"github.com/junegunn/fzf/src/algo"
|
||||||
)
|
)
|
||||||
@ -31,6 +30,7 @@ type term struct {
|
|||||||
typ termType
|
typ termType
|
||||||
inv bool
|
inv bool
|
||||||
text []rune
|
text []rune
|
||||||
|
caseSensitive bool
|
||||||
origText []rune
|
origText []rune
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -88,36 +88,27 @@ func BuildPattern(mode Mode, caseMode Case,
|
|||||||
caseSensitive, hasInvTerm := true, false
|
caseSensitive, hasInvTerm := true, false
|
||||||
terms := []term{}
|
terms := []term{}
|
||||||
|
|
||||||
switch caseMode {
|
|
||||||
case CaseSmart:
|
|
||||||
hasUppercase := false
|
|
||||||
for _, r := range runes {
|
|
||||||
if unicode.IsUpper(r) {
|
|
||||||
hasUppercase = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !hasUppercase {
|
|
||||||
runes, caseSensitive = []rune(strings.ToLower(asString)), false
|
|
||||||
}
|
|
||||||
case CaseIgnore:
|
|
||||||
runes, caseSensitive = []rune(strings.ToLower(asString)), false
|
|
||||||
}
|
|
||||||
|
|
||||||
switch mode {
|
switch mode {
|
||||||
case ModeExtended, ModeExtendedExact:
|
case ModeExtended, ModeExtendedExact:
|
||||||
terms = parseTerms(mode, string(runes))
|
terms = parseTerms(mode, caseMode, asString)
|
||||||
for _, term := range terms {
|
for _, term := range terms {
|
||||||
if term.inv {
|
if term.inv {
|
||||||
hasInvTerm = true
|
hasInvTerm = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
default:
|
||||||
|
lowerString := strings.ToLower(asString)
|
||||||
|
caseSensitive = caseMode == CaseRespect ||
|
||||||
|
caseMode == CaseSmart && lowerString != asString
|
||||||
|
if !caseSensitive {
|
||||||
|
asString = lowerString
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ptr := &Pattern{
|
ptr := &Pattern{
|
||||||
mode: mode,
|
mode: mode,
|
||||||
caseSensitive: caseSensitive,
|
caseSensitive: caseSensitive,
|
||||||
text: runes,
|
text: []rune(asString),
|
||||||
terms: terms,
|
terms: terms,
|
||||||
hasInvTerm: hasInvTerm,
|
hasInvTerm: hasInvTerm,
|
||||||
nth: nth,
|
nth: nth,
|
||||||
@ -133,11 +124,17 @@ func BuildPattern(mode Mode, caseMode Case,
|
|||||||
return ptr
|
return ptr
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseTerms(mode Mode, str string) []term {
|
func parseTerms(mode Mode, caseMode Case, str string) []term {
|
||||||
tokens := _splitRegex.Split(str, -1)
|
tokens := _splitRegex.Split(str, -1)
|
||||||
terms := []term{}
|
terms := []term{}
|
||||||
for _, token := range tokens {
|
for _, token := range tokens {
|
||||||
typ, inv, text := termFuzzy, false, token
|
typ, inv, text := termFuzzy, false, token
|
||||||
|
lowerText := strings.ToLower(text)
|
||||||
|
caseSensitive := caseMode == CaseRespect ||
|
||||||
|
caseMode == CaseSmart && text != lowerText
|
||||||
|
if !caseSensitive {
|
||||||
|
text = lowerText
|
||||||
|
}
|
||||||
origText := []rune(text)
|
origText := []rune(text)
|
||||||
if mode == ModeExtendedExact {
|
if mode == ModeExtendedExact {
|
||||||
typ = termExact
|
typ = termExact
|
||||||
@ -166,6 +163,7 @@ func parseTerms(mode Mode, str string) []term {
|
|||||||
typ: typ,
|
typ: typ,
|
||||||
inv: inv,
|
inv: inv,
|
||||||
text: []rune(text),
|
text: []rune(text),
|
||||||
|
caseSensitive: caseSensitive,
|
||||||
origText: origText})
|
origText: origText})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -280,7 +278,7 @@ func dupItem(item *Item, offsets []Offset) *Item {
|
|||||||
|
|
||||||
func (p *Pattern) fuzzyMatch(item *Item) (int, int) {
|
func (p *Pattern) fuzzyMatch(item *Item) (int, int) {
|
||||||
input := p.prepareInput(item)
|
input := p.prepareInput(item)
|
||||||
return p.iter(algo.FuzzyMatch, input, p.text)
|
return p.iter(algo.FuzzyMatch, input, p.caseSensitive, p.text)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Pattern) extendedMatch(item *Item) []Offset {
|
func (p *Pattern) extendedMatch(item *Item) []Offset {
|
||||||
@ -288,7 +286,7 @@ func (p *Pattern) extendedMatch(item *Item) []Offset {
|
|||||||
offsets := []Offset{}
|
offsets := []Offset{}
|
||||||
for _, term := range p.terms {
|
for _, term := range p.terms {
|
||||||
pfun := p.procFun[term.typ]
|
pfun := p.procFun[term.typ]
|
||||||
if sidx, eidx := p.iter(pfun, input, term.text); sidx >= 0 {
|
if sidx, eidx := p.iter(pfun, input, term.caseSensitive, term.text); sidx >= 0 {
|
||||||
if term.inv {
|
if term.inv {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@ -319,10 +317,10 @@ func (p *Pattern) prepareInput(item *Item) *[]Token {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *Pattern) iter(pfun func(bool, *[]rune, []rune) (int, int),
|
func (p *Pattern) iter(pfun func(bool, *[]rune, []rune) (int, int),
|
||||||
tokens *[]Token, pattern []rune) (int, int) {
|
tokens *[]Token, caseSensitive bool, pattern []rune) (int, int) {
|
||||||
for _, part := range *tokens {
|
for _, part := range *tokens {
|
||||||
prefixLength := part.prefixLength
|
prefixLength := part.prefixLength
|
||||||
if sidx, eidx := pfun(p.caseSensitive, part.text, pattern); sidx >= 0 {
|
if sidx, eidx := pfun(caseSensitive, part.text, pattern); sidx >= 0 {
|
||||||
return sidx + prefixLength, eidx + prefixLength
|
return sidx + prefixLength, eidx + prefixLength
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -535,6 +535,10 @@ class TestGoFZF < TestBase
|
|||||||
tmux.send_keys :Enter
|
tmux.send_keys :Enter
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def test_smart_case_for_each_term
|
||||||
|
assert_equal 1, `echo Foo bar | #{FZF} -x -f "foo Fbar" | wc -l`.to_i
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def writelines path, lines, timeout = 10
|
def writelines path, lines, timeout = 10
|
||||||
File.open(path, 'w') do |f|
|
File.open(path, 'w') do |f|
|
||||||
|
Loading…
Reference in New Issue
Block a user