"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " fuzzyfinder.vim : Buffer and file explorer with the fuzzy pattern " Last Change: 07-Sep-2007. " Author: Takeshi Nishida " Version: 0.4, for Vim 7.0 " Licence: MIT Licence " """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " Description: " With fuzzyfinder, you can quickly reach the buffer or file you want with " less typing. Fuzzyfinder find matching files or buffers with the fuzzy " pattern to which fuzzyfinder converted the inputting pattern. " " E.g.: the inputting pattern / the fuzzy pattern " abc -> *a*b*c* " a*b*c* -> *a*b*c* " a?b*c? -> *a?b*c? (? matches one character.) " dir/file -> dir/*f*i*l*e* " d*r/file -> d*r/*f*i*l*e* " ../**/s -> ../**/*s* (** allows searching a directory tree.) " " You will be happy when: " "./OhLongLongLongLongLongFile.txt" " "./AhLongLongLongLongLongName.txt" " "./AhLongLongLongLongLongFile.txt" <- you want :O " Type "AF" and "AhLongLongLongLongLongFile.txt" will be select. :D " " Fuzzyfinder supports multibyte. " """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " Installation: " Drop this file in your plugin directory. If you have installed " autocomplpop.vim (vimscript #1879), please update to the latest version " to prevent interference. " """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " Usage: " :FuzzyFinderBuffer opens a small window for a buffer explorer. " :FuzzyFinderFile opens a small window for a file explorer. " " It is recommended to map these. Personally, I map these to and " . " " To switch between a buffer mode and a file mode without leaving a insert " mode, use by default. " " If you want to temporarily change whether or not to ignore case, use " by default. " " The inputting pattern you typed is converted to the fuzzy pattern " and buffers or files which match the pattern is shown in a completion " menu. " " A completion menu is shown when you type at the end of the line and the " length of inputting pattern is more than setting value. At the start of " a buffer mode, all buffers list is shown by default. In a file mode, a " completion menu is not shown while the length of inputting filename is " less than 1 by default. " " If inputting pattern matches the item perfectly, the item is shown first " and selected in completion menu. Same applies to the buffer number in a " buffer mode. " " Pressing opens selected item in the previous window. If selected " item is directory in a file mode, it inserts but does not open. " " To cancel and return the previous window, leave a insert mode. " " """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " Options: " See a section setting global value below. " """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " Thanks: " Vincent Wang " Ingo Karkat " Nikolay Golubev " """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " ChangeLog: " 0.4: " - Improved response of the input. " - Improved the sorting algorithm for completion items. It is based " on the matching level. 1st is perfect matching, 2nd is prefix " matching, and 3rd is fuzzy matching. " - Added g:FuzzyFinder_ExcludePattern option. " - Removed g:FuzzyFinder_WildIgnore option. " - Removed g:FuzzyFinder_EchoPattern option. " - Removed g:FuzzyFinder_PathSeparator option. " - Changed the default value of g:FuzzyFinder_MinLengthFile from 1 to " 0. " " 0.3: " - Added g:FuzzyFinder_IgnoreCase option. " - Added g:FuzzyFinder_KeyToggleIgnoreCase option. " - Added g:FuzzyFinder_EchoPattern option. " - Changed the open command in a buffer mode from ":edit" to " ":buffer" to avoid being reset cursor position. " - Changed the default value of g:FuzzyFinder_KeyToggleMode from " to because does not work on some CUI " environments. " - Changed to avoid being loaded by Vim before 7.0. " - Fixed a bug with making a fuzzy pattern which has '\'. " " 0.2: " - A bug it does not work on Linux is fixed. " " 0.1: " - First release. " """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" if exists("loaded_fuzzyfinder") || v:version < 700 finish endif let loaded_fuzzyfinder = 1 """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " Map this to toggle buffer mode and file mode in insert mode. if !exists('g:FuzzyFinder_KeyToggleMode') let g:FuzzyFinder_KeyToggleMode = '' endif " Map this to temporarily toggle whether or not to ignore case. if !exists('g:FuzzyFinder_KeyToggleIgnoreCase') let g:FuzzyFinder_KeyToggleIgnoreCase = '' endif " In buffer mode, do not complete if a length of inputting is less than this. if !exists('g:FuzzyFinder_MinLengthBuffer') let g:FuzzyFinder_MinLengthBuffer = 0 endif " In file mode, do not complete if a length of inputting is less than this. if !exists('g:FuzzyFinder_MinLengthFile') let g:FuzzyFinder_MinLengthFile = 0 endif " In buffer and file mode, ignore case in search patterns. if !exists('g:FuzzyFinder_IgnoreCase') let g:FuzzyFinder_IgnoreCase = &ignorecase endif " In file mode, files which matchs this are ignored. if !exists('g:FuzzyFinder_ExcludePattern') let g:FuzzyFinder_ExcludePattern = '^\.$\|\.bak$\|\~$\|\.swp$' endif " In buffer mode, buffers with an indicator matching this are ignored. if !exists('g:FuzzyFinder_ExcludeIndicator') let g:FuzzyFinder_ExcludeIndicator = '[u\-]' endif """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" command! -narg=0 -bar FuzzyFinderBuffer call StartBufferMode() command! -narg=0 -bar FuzzyFinderFile call StartFileMode() """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" let s:pathSeparator = has('win32') ? '\' : '/' let s:cmdPrompt = '>' let s:bufNr = -1 let s:isLastModeBuffer = 0 let s:openCommand = 0 let s:reserveToggleMode = 0 """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! StartBufferMode() call OpenInputWindow(1) endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! StartFileMode() call OpenInputWindow(0) endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! ToggleMode() let s:reserveToggleMode = 1 return "\" endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! ToggleIgnoreCase() let &ignorecase = !&ignorecase echo "ignorecase=" . &ignorecase return "" endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! OpenInputWindow(isBufferMode) " differences between modes if a:isBufferMode let s:openCommand = ':buffer ' let bufferName = '[FuzzyFinder - Buffer]' let completeFunc = 'FuzzyFinder_CompleteBuffer' else let s:openCommand = ':edit ' let bufferName = '[FuzzyFinder - File]' let completeFunc = 'FuzzyFinder_CompleteFile' endif let s:isLastModeBuffer = a:isBufferMode let s:lastInputLength = -1 if s:bufNr != -1 " a buffer already created execute 'buffer ' . s:bufNr delete _ else 1new let s:bufNr = bufnr('%') " suspend autocomplpop.vim if exists(':AutoComplPopLock') :AutoComplPopLock endif " global setting let s:_completeopt = &completeopt set completeopt=menuone let s:_ignorecase = &ignorecase let &ignorecase = g:FuzzyFinder_IgnoreCase " local setting setlocal bufhidden=wipe setlocal buftype=nofile setlocal noswapfile setlocal nobuflisted setlocal modifiable " mapping inoremap OnCR() inoremap pumvisible() ? "\\" : "\" execute "inoremap " . g:FuzzyFinder_KeyToggleMode . \ " ToggleMode()" execute "inoremap " . g:FuzzyFinder_KeyToggleIgnoreCase . \ " ToggleIgnoreCase()" " auto command augroup FuzzyFinder_AutoCommand autocmd! autocmd InsertLeave nested call OnInsertLeave() autocmd CursorMovedI call OnCursorMovedI() autocmd BufLeave call OnBufLeave() augroup END endif let &l:completefunc = completeFunc execute 'file ' . bufferName " start insert mode. s:cmdPrompt makes CursorMovedI event now " and forces a completion menu to update every typing. call feedkeys('i' . s:cmdPrompt, 'n') endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! FuzzyFinder_CompleteBuffer(findstart, base) if a:findstart return 0 endif let input = ExtractPromptedInput(a:base) if !input[0] || strlen(input[1]) < g:FuzzyFinder_MinLengthBuffer echo "" return [] endif let patternW = MakeFuzzyPattern(input[1]) let patternR = ConvertWildcardToRegexp(patternW) " make a list of buffers redir => l:lines silent buffers! redir END let result = [] for line in split(lines, "\n") " bufInfo[1]: number, bufInfo[2]: indicator, bufInfo[3]: filename let bufInfo = matchlist(line, '^\s*\(\d*\)\s*\([^"]*\)\s*"\([^"]*\)"') if bufInfo[1] == s:bufNr || bufInfo[2] =~ g:FuzzyFinder_ExcludeIndicator || \ (bufInfo[1] != input[1] && bufInfo[3] !~ patternR) continue endif call add(result, MakeCompletionItemWithMatchingLevel(bufInfo[3], bufInfo[1], line, input[1])) endfor call sort(result, "SortByMatchingLevel") call feedkeys(!empty(result) ? "\\" : "\", 'n') echo "pattern:" . patternW return result endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! FuzzyFinder_CompleteFile(findstart, base) if a:findstart return 0 endif let input = ExtractPromptedInput(a:base) let pathPair = SplitPath(input[1]) if !input[0] || strlen(pathPair[1]) < g:FuzzyFinder_MinLengthFile echo "" return [] endif let patternW = pathPair[0] . MakeFuzzyPattern(pathPair[1]) echo 'Making file list...' let result = GlobEx(patternW) echo 'Evaluating...' call map (result, 'MakeCompletionItemWithMatchingLevel(v:val, -1, v:val, input[1])') call sort(result, 'SortByMatchingLevel') call feedkeys(!empty(result) ? "\\" : "\", 'n') echo 'pattern:' . patternW return result endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! OnCR() return pumvisible() ? "\\=col('.') > strlen(getline('.')) && getline('.') !~ '[/\\\\]$' ? \"\\\" : \"\"\" : "\" endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! OnCursorMovedI() let input = ExtractPromptedInput(getline('1')) if line('.') != 1 " Fix call feedkeys("\" . s:openCommand . input[1] . "\", 'n') "call feedkeys("\:e " . input[1] . "\", 'n') return elseif !input[0] " a command prompt is removed call feedkeys("\" . s:cmdPrompt . "\", 'n') return endif let deltaInputLength = strlen(getline('.')) - s:lastInputLength let s:lastInputLength = strlen(getline('.')) " if the line was changed and cursor is placed on the end of the line if deltaInputLength != 0 && col('.') > s:lastInputLength call feedkeys("\\", 'n') endif endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! OnInsertLeave() if s:reserveToggleMode let s:reserveToggleMode = 0 call OpenInputWindow(!s:isLastModeBuffer) else quit endif endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! OnBufLeave() " resume autocomplpop.vim if exists(':AutoComplPopUnlock') :AutoComplPopUnlock endif call ClearGlobExCache() let &completeopt = s:_completeopt let &ignorecase = s:_ignorecase let s:bufNr = -1 quit " Quit when other window clicked without leaving a insert mode. endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " "str" -> "*s*t*r*" function! MakeFuzzyPattern(in) let out = '' for char in split(a:in,'\zs') if out !~ '[*?]$' && char !~ '[*?]' let out .= '*'. char else let out .= char endif endfor if out !~ '[*?]$' return out . '*' endif return out endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! ConvertWildcardToRegexp(in) let out = escape(a:in, '\') let out = substitute(out, '*', '\\.\\*', 'g') let out = substitute(out, '?', '\\.' , 'g') let out = substitute(out, '[', '\\[' , 'g') let out = '\V' . out return out endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! ExpandCase(in) let out = '' for char in split(a:in,'\zs') if char =~ '\a' let out .= '[' . toupper(char) . tolower(char) . ']' else let out .= char endif endfor return out endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! ExtractPromptedInput(in) if strlen(a:in) < strlen(s:cmdPrompt) || \ a:in[:strlen(s:cmdPrompt) -1] !=# s:cmdPrompt return [0, a:in] endif return [1, a:in[strlen(s:cmdPrompt):]] endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! SplitPath(path) let dir = matchstr(a:path, '^.*[/\\]') return [dir, a:path[strlen(dir):]] endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! GlobEx(patternW) let patternWHead = SplitPath(a:patternW)[0] let patternR = ConvertWildcardToRegexp(a:patternW) if !exists('s:globExCacheDir') || s:globExCacheDir != patternWHead let s:globExCacheDir = patternWHead let dirList = split(expand(patternWHead), "\n") if len(dirList) <= 1 " Do not expand here let dirList = [patternWHead] endif for dir in dirList let expDir = expand(dir) let s:globExCache = split(glob(expDir . '.*') . "\n" . glob(expDir . '*' ), "\n") call map (s:globExCache, 'SplitPath(v:val)[1]') call filter(s:globExCache, 'v:val !~ g:FuzzyFinder_ExcludePattern') call map (s:globExCache, 'dir . v:val . (isdirectory(expDir . v:val) ? s:pathSeparator : "")') endfor endif return filter(copy(s:globExCache), 'v:val =~ patternR') endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! ClearGlobExCache() unlet! s:globExCacheDir unlet! s:globExCache endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! ShellEscapeEx(path) return a:path =~ '\s' ? shellescape(a:path) : a:path endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! MakeCompletionItemWithMatchingLevel(path, bufNr, abbr, input) let pathTail = SplitPath(matchstr(a:path , '^.*[^/\\]'))[1] let inputTail = SplitPath(matchstr(a:input, '^.*[^/\\]'))[1] " Evaluates matching level if a:bufNr >= 0 && a:bufNr == a:input let mlevel = strlen(inputTail) + 3 elseif a:path == a:input let mlevel = strlen(inputTail) + 2 elseif pathTail == inputTail let mlevel = strlen(inputTail) + 1 else let mlevel = 0 while mlevel < strlen(inputTail) && pathTail[mlevel] == inputTail[mlevel] let mlevel += 1 endwhile endif return { 'word' : a:path, \ 'abbr' : a:abbr, \ 'menu' : '| ' . repeat('*', mlevel), \ 'mlevel': mlevel} endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! SortByMatchingLevel(i1, i2) if a:i1['mlevel'] < a:i2['mlevel'] return +1 elseif a:i1['mlevel'] > a:i2['mlevel'] return -1 elseif a:i1['word'] > a:i2['word'] return +1 elseif a:i1['word'] < a:i2['word'] return -1 endif return 0 endfunction """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""