"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " fuzzyfinder.vim : The buffer/file/MRU/favorite/etc. explorer " with the fuzzy pattern """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " " Last Change: 09-Nov-2007. " Author: Takeshi Nishida " Version: 1.5, for Vim 7.0 " Licence: MIT Licence " "----------------------------------------------------------------------------- " Description: " Fuzzyfinder provides convenient ways to quickly reach the " buffer/file/command you want. Fuzzyfinder finds matching " files/buffers/commands with a fuzzy pattern to which it converted the " inputting pattern. " " Fuzzyfinder has 6 modes: " - Buffer mode " - File mode " - MRU-files mode (most recently used files) " - MRU-commands mode (most recently used commands) " - Favorite-files mode " - Directory mode (for :cd command) " " E.g.: the inputting pattern / the fuzzy pattern " abc -> *a*b*c* " a?c -> *a?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 the 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: " Starting The Explorer: " You can start the explorer by the following commands: " " :FuzzyFinderBuffer - launchs the buffer explorer. " :FuzzyFinderFile - launchs the file explorer. " :FuzzyFinderMruFile - launchs the MRU-files explorer. " :FuzzyFinderMruCmd - launchs the MRU-commands explorer. " :FuzzyFinderFavFile - launchs the favorite-files explorer. " :FuzzyFinderDir - launchs the directory explorer. " " It is recommended to map these. " " In The Explorer: " The inputting pattern you typed is converted to the fuzzy pattern and " buffers/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. By " default, it is shown at the beginning. " " If too many items (400, by default) were matched, the completion is " aborted to reduce nonresponse. " " If inputting pattern matched the item perfectly, the item is shown " first. Same applies to the item number in the buffer/MRU/favorite " mode. The item whose file name has longer prefix matching is placed " upper. The item which matched more sequentially is placed upper. It " lets the first item into selected in completion menu. " " Pressing opens selected item in previous window. If selected item " was a directory in the file mode, it just inserts text. Use to " open in new window which is made from split previous window, or " To open in new window which is made from split previous window " vertically. In MRU-commands mode, press and it executes selected " command, or press / and it just puts text into the " command-line. These key mappings are customizable. " " To cancel and return to previous window, leave the insert mode. " " To Switch the mode without leaving a insert mode, use or " . This key mapping is customizable. " " If you want to temporarily change whether or not to ignore case, use " . This key mapping is customizable. " " About Abbreviations And Multiple Search: " You can use abbreviations and multiple search in each mode. For " example, set as below: " " :let g:FuzzyFinder_FileModeVars = " \ { "abbrevMap" : " \ { "^WORK" : [ "~/project/**/src/", " \ ".vim/plugin/" " \ ] " \ } " \ } " " And input "WORKtxt" in the file-mode explorer, then it searches by " following patterns: " " "~/project/**/src/*t*x*t*" " ".vim/plugin/*t*x*t*" " " Adding Favorite Files: " You can add a favorite file by the following commands: " " :FuzzyFinderAddFavFile {filename} " " If you do not specify the filename, current file name is used. " " About Information File: " Fuzzyfinder writes information of the MRU, favorite, etc to the file " by default (~/.vimfuzzyfinder). If you don't want the file, set as " below: " " :let g:FuzzyFinder_InfoFile = "" " " About Migemo: " Migemo is a search method for Japanese language. " "----------------------------------------------------------------------------- " Options: " g:FuzzyFinder_InfoFile: " This is the file name to write information of the MRU, etc. If "" was " set, it does not write to the file. " " g:FuzzyFinder_KeyOpen: " This is a list. These are mapped to select completion item or to " finish input and open a buffer/file. " [0]: " This is mapped to open in previous window. " [1]: " This is mapped to open in new window which is made from split " previous window. " [2]: " This is mapped to open in new window which is made from split " previous window vertically. " " g:FuzzyFinder_KeySwitchMode: " This is a list. This is mapped to switch the mode in the insert mode. " [0]: " This is mapped to switch to the next mode. " [1]: " This is mapped to switch to the previous mode. " " g:FuzzyFinder_KeySwitchIgnoreCase: " This is mapped to temporarily switch whether or not to ignore case. " " g:FuzzyFinder_IgnoreCase: " It ignores case in search patterns if non-zero is set. " " g:FuzzyFinder_Migemo: " It uses Migemo if non-zero is set. " " g:FuzzyFinder_BufferModeVars: " This is a dictionary. This applies only to buffer mode. " "initialInput": " This is a text which is inserted at the beginning of the mode. " "abbrevMap" " This is a dictionary. Each value must be a list. All matchs of a " key in inputting text is expanded with a value. " "minLength": " It does not complete if a length of inputting is less than this. " "excludedIndicator": " Items whose indicators match this are excluded. " " g:FuzzyFinder_FileModeVars: " This is a dictionary. This applies only to mode. " "initialInput": " This is a text which is inserted at the beginning of the mode. " "abbrevMap" " This is a dictionary. Each value must be a list. All matchs of a " key in inputting text is expanded with a value. " "minLength": " It does not complete if a length of inputting is less than this. " "excludedPath": " Items whose paths match this are excluded. " "maxMatches": " If items were matched over this, processing completion is aborted. " If 0 was set, it does not limit. " " g:FuzzyFinder_MruFileModeVars: " This is a dictionary. This applies only to the MRU-files mode. " "initialInput": " This is a text which is inserted at the beginning of the mode. " "abbrevMap" " This is a dictionary. Each value must be a list. All matchs of a " key in inputting text is expanded with a value. " "minLength": " It does not complete if a length of inputting is less than this. " "excludedPath": " Items whose paths match this are excluded. " "ignoreSpecialBuffers" " It ignores special buffers if non-zero is set. " "timeFormat": " This is a string to format registered time. See :help strftime() " for details. " "maxItems": " This is an upper limit of MRU files to be stored. If 0 was set, it " does not limit. " " g:FuzzyFinder_MruCmdModeVars: " This is a dictionary. This applies only to the MRU-commands mode. " "initialInput": " This is a text which is inserted at the beginning of the mode. " "abbrevMap" " This is a dictionary. Each value must be a list. All matchs of a " key in inputting text is expanded with a value. " "minLength": " It does not complete if a length of inputting is less than this. " "excludedCommand": " Items whose commands match this are excluded. " "timeFormat": " This is a string to format registered time. See :help strftime() " for details. " "maxItems": " This is an upper limit of MRU commands to be stored. If 0 was set, " it does not limit. " " g:FuzzyFinder_FavFileModeVars: " This is a dictionary. This applies only to the favorite-files mode. " "initialInput": " This is a text which is inserted at the beginning of the mode. " "abbrevMap" " This is a dictionary. Each value must be a list. All matchs of a " key in inputting text is expanded with a value. " "minLength": " It does not complete if a length of inputting is less than this. " "timeFormat": " This is a string to format registered time. See :help strftime() " for details. " " g:FuzzyFinder_DirModeVars: " This is a dictionary. This applies only to the directory mode. " "initialInput": " This is a text which is inserted at the beginning of the mode. " "abbrevMap" " This is a dictionary. Each value must be a list. All matchs of a " key in inputting text is expanded with a value. " "minLength": " It does not complete if a length of inputting is less than this. " "excludedPath": " Items whose paths match this are excluded. " "----------------------------------------------------------------------------- " Settings Example: " My settings for gvim: " :let g:FuzzyFinder_KeyOpen = ['', '', ''] " :let g:FuzzyFinder_KeySwitchMode = ['', ''] " :let g:FuzzyFinder_KeySwitchIgnoreCase = '' " :let g:FuzzyFinder_FileModeVars = " \ { 'abbrevMap' : { '\C^VP' : ['$VIM/runtime/plugin/' , " \ '$VIM/vimfiles/plugin/', " \ '~/vim/plugin'], " \ '\C^VC' : ['$VIM/runtime/colors/' , " \ '$VIM/vimfiles/colors/', " \ '~/vim/colors']} " \ } " :let g:FuzzyFinder_MruFileModeVars = { 'maxItems' : 99 } " :let g:FuzzyFinder_MruCmdModeVars = { 'maxItems' : 99 } " :nnoremap :FuzzyFinderBuffer " :nnoremap :FuzzyFinderFile " :nnoremap :FuzzyFinderMruFile " :nnoremap :FuzzyFinderMruCmd " :nnoremap :FuzzyFinderFavFile " :nnoremap :FuzzyFinderDir " "----------------------------------------------------------------------------- " Thanks: " Vincent Wang " Ingo Karkat " Nikolay Golubev " Brian Doyle " "----------------------------------------------------------------------------- " ChangeLog: " 1.5: " - Added the directory mode. " - Fixed the bug that it caused an error when switch a mode in the " insert mode. " - Changed g:FuzzyFinder_KeySwitchMode type to a list. " " 1.4: " - Changed the specification of the information file. " - Added the MRU-commands mode. " - Renamed :FuzzyFinderAddFavorite command to :FuzzyFinderAddFavFile. " - Renamed g:FuzzyFinder_MruModeVars option to " g:FuzzyFinder_MruFileModeVars. " - Renamed g:FuzzyFinder_FavoriteModeVars option to " g:FuzzyFinder_FavFileModeVars. " - Changed to show registered time of each item in MRU/favorite mode. " - Added 'timeFormat' option for MRU/favorite modes. " " 1.3: " - Fixed a handling of multi-byte characters. " " 1.2: " - Added support for Migemo. (Migemo is Japanese search method.) " " 1.1: " - Added the favorite mode. " - Added new features, which are abbreviations and multiple search. " - Added 'abbrevMap' option for each mode. " - Added g:FuzzyFinder_MruModeVars['ignoreSpecialBuffers'] option. " - Fixed the bug that it did not work correctly when a user have mapped " or . " " 1.0: " - Added the MRU mode. " - Added commands to add and use original mode. " - Improved the sorting algorithm for completion items. " - Added 'initialInput' option to automatically insert a text at the " beginning of a mode. " - Changed that 'excludedPath' option works for the entire path. " - Renamed some options. " - Changed default values of some options. " - Packed the mode-specific options to dictionaries. " - Removed some options. " " 0.6: " - Fixed some bugs. " 0.5: " - Improved response by aborting processing too many items. " - Changed to be able to open a buffer/file not only in previous window " but also in new window. " - Fixed a bug that recursive searching with '**' does not work. " - Added g:FuzzyFinder_CompletionItemLimit option. " - Added g:FuzzyFinder_KeyOpen option. " " 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. " " "----------------------------------------------------------------------------- """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " INCLUDE GUARD: """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" if exists('loaded_fuzzyfinder') || v:version < 700 finish endif let loaded_fuzzyfinder = 1 """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " INITIALIZATION FUNCTION: """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" function! Initialize() "------------------------------------------------------------------------- " CONSTANTS let s:info_version = 104 let s:path_separator = has('win32') ? '\' : '/' let s:prompt = '>' let s:aborted_result = [{'word': ' ', 'abbr': 'ABORT: Too many matches (> g:FuzzyFinder_FileModeVars.maxMatches)'}] let s:matching_rate_base = 10000000 let s:sid_prefix = matchstr(expand(''), '\d\+_') let s:msg_rm_info = "==================================================\n" . \ " Your fuzzyfinder information file is no longer \n" . \ " supported. Please remove\n" . \ " \"%s\".\n" . \ "==================================================" "------------------------------------------------------------------------- " OPTIONS if !exists('g:FuzzyFinder_InfoFile') let g:FuzzyFinder_InfoFile = '~/.vimfuzzyfinder' endif "......................................................................... if !exists('g:FuzzyFinder_KeyOpen') let g:FuzzyFinder_KeyOpen = ['', '', ''] endif "......................................................................... if !exists('g:FuzzyFinder_KeySwitchMode') let g:FuzzyFinder_KeySwitchMode = ['', ''] endif "......................................................................... if !exists('g:FuzzyFinder_KeySwitchIgnoreCase') let g:FuzzyFinder_KeySwitchIgnoreCase = '' endif "......................................................................... if !exists('g:FuzzyFinder_IgnoreCase') let g:FuzzyFinder_IgnoreCase = &ignorecase endif "......................................................................... if !exists('g:FuzzyFinder_Migemo') let g:FuzzyFinder_Migemo = has('migemo') endif "......................................................................... if !exists('g:FuzzyFinder_BufferModeVars') let g:FuzzyFinder_BufferModeVars = {} endif call extend(g:FuzzyFinder_BufferModeVars, \ { 'name' : 'buffer', \ 'bufferName' : '[FuzzyFinder - Buffer]', \ 'complete' : function('CompleteBuffer'), \ 'open' : function('OpenBuffer'), \ 'initialInput' : '', \ 'abbrevMap' : {}, \ 'minLength' : 0, \ 'excludedIndicator' : '[u\-]', \ } ,'keep') "......................................................................... if !exists('g:FuzzyFinder_FileModeVars') let g:FuzzyFinder_FileModeVars = {} endif call extend(g:FuzzyFinder_FileModeVars, \ { 'name' : 'file', \ 'bufferName' : '[FuzzyFinder - File]', \ 'complete' : function('CompleteFile'), \ 'open' : function('OpenFile_MruFile_FavFile'), \ 'initialInput' : '', \ 'abbrevMap' : {}, \ 'minLength' : 0, \ 'excludedPath' : '\v\~$|\.bak$|\.swp$|((^|[/\\])\.[/\\]$)', \ 'maxMatches' : 400, \ } ,'keep') "......................................................................... if !exists('g:FuzzyFinder_MruFileModeVars') let g:FuzzyFinder_MruFileModeVars = {} endif call extend(g:FuzzyFinder_MruFileModeVars, \ { 'name' : 'mru_file', \ 'bufferName' : '[FuzzyFinder - MRU Files]', \ 'complete' : function('CompleteMruFile'), \ 'open' : function('OpenFile_MruFile_FavFile'), \ 'initialInput' : '', \ 'abbrevMap' : {}, \ 'minLength' : 0, \ 'excludedPath' : '\v\~$|\.bak$|\.swp$', \ 'ignoreSpecialBuffers' : 1, \ 'timeFormat' : '(%x %H:%M:%S)', \ 'maxItems' : 20, \ } ,'keep') "......................................................................... if !exists('g:FuzzyFinder_MruCmdModeVars') let g:FuzzyFinder_MruCmdModeVars = {} endif call extend(g:FuzzyFinder_MruCmdModeVars, \ { 'name' : 'mru_cmd', \ 'commandFormat' : ["%s\", "%s", "%s"], \ 'bufferName' : '[FuzzyFinder - MRU Commands]', \ 'complete' : function('CompleteMruCmd'), \ 'open' : function('OpenMruCmd'), \ 'initialInput' : '', \ 'abbrevMap' : {}, \ 'minLength' : 0, \ 'excludedCommand' : '^.\{0,4}$', \ 'timeFormat' : '(%x %H:%M:%S)', \ 'maxItems' : 20, \ } ,'keep') "......................................................................... if !exists('g:FuzzyFinder_FavFileModeVars') let g:FuzzyFinder_FavFileModeVars = {} endif call extend(g:FuzzyFinder_FavFileModeVars, \ { 'name' : 'fav_file', \ 'bufferName' : '[FuzzyFinder - Favorite Files]', \ 'complete' : function('CompleteFavFile'), \ 'open' : function('OpenFile_MruFile_FavFile'), \ 'initialInput' : '', \ 'abbrevMap' : {}, \ 'minLength' : 0, \ 'timeFormat' : '(%x %H:%M:%S)', \ } ,'keep') "......................................................................... if !exists('g:FuzzyFinder_DirModeVars') let g:FuzzyFinder_DirModeVars = {} endif call extend(g:FuzzyFinder_DirModeVars, \ { 'name' : 'directory', \ 'bufferName' : '[FuzzyFinder - Directory]', \ 'complete' : function('CompleteDir'), \ 'open' : function('OpenDir'), \ 'initialInput' : "\=fnamemodify('.', ':p')\", \ 'abbrevMap' : {}, \ 'minLength' : 0, \ 'excludedPath' : '\v(^|[/\\])\.{1,2}[/\\]$', \ } ,'keep') "......................................................................... "------------------------------------------------------------------------- " COMMANDS command! -bar -narg=0 FuzzyFinderBuffer call OpenFuzzyFinder(g:FuzzyFinder_BufferModeVars.name) command! -bar -narg=0 FuzzyFinderFile call OpenFuzzyFinder(g:FuzzyFinder_FileModeVars.name) command! -bar -narg=0 FuzzyFinderMruFile call OpenFuzzyFinder(g:FuzzyFinder_MruFileModeVars.name) command! -bar -narg=0 FuzzyFinderMruCmd call OpenFuzzyFinder(g:FuzzyFinder_MruCmdModeVars.name) command! -bar -narg=0 FuzzyFinderFavFile call OpenFuzzyFinder(g:FuzzyFinder_FavFileModeVars.name) command! -bar -narg=0 FuzzyFinderDir call OpenFuzzyFinder(g:FuzzyFinder_DirModeVars.name) command! -bar -narg=1 FuzzyFinder call OpenFuzzyFinder() command! -bar -narg=1 FuzzyFinderAddMode call AddMode() command! -bar -narg=? -complete=file FuzzyFinderAddFavFile call UpdateFavFileInfo(, 1) "------------------------------------------------------------------------- " AUTOCOMMANDS augroup FuzzyFinder_GlobalAutoCommand autocmd! autocmd BufEnter * call OnBufEnter() autocmd BufWritePost * call OnBufWritePost() augroup END "------------------------------------------------------------------------- " MAPPING cnoremap OnCmdLineCR() "------------------------------------------------------------------------- " ETC let s:info = ReadInfoFile() call AddMode(g:FuzzyFinder_BufferModeVars ) call AddMode(g:FuzzyFinder_FileModeVars ) call AddMode(g:FuzzyFinder_MruFileModeVars) call AddMode(g:FuzzyFinder_MruCmdModeVars ) call AddMode(g:FuzzyFinder_FavFileModeVars) call AddMode(g:FuzzyFinder_DirModeVars ) endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " FUNCTIONS: """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" "----------------------------------------------------------------------------- function! OpenFuzzyFinder(mode) let s:info = ReadInfoFile() let s:vars_current = s:vars[a:mode] let s:last_col = -1 if exists('s:buf_nr') " a buffer for fuzzyfinder was already created execute 'buffer ' . s:buf_nr delete _ else 1new let s:buf_nr = bufnr('%') " suspend autocomplpop.vim if exists(':AutoComplPopLock') :AutoComplPopLock endif "..................................................................... " global setting let s:_completeopt = &completeopt let &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 let &l:completefunc = s:sid_prefix . 'Complete' "..................................................................... " autocommands augroup FuzzyFinder_LocalAutoCommand autocmd! autocmd CursorMovedI call feedkeys(OnCursorMovedI(), 'n') autocmd InsertLeave nested call feedkeys(OnInsertLeave() , 'n') autocmd BufLeave call feedkeys(OnBufLeave() , 'n') augroup END "..................................................................... " mapping let i = 0 | while i < len(g:FuzzyFinder_KeyOpen) execute "inoremap " . g:FuzzyFinder_KeyOpen[i] . " OnCR(". i. ", 0)" let i += 1 endwhile execute "inoremap " . "" . " OnBS()" execute "inoremap " . g:FuzzyFinder_KeySwitchMode[0] . " OnSwitchMode(+1)" execute "inoremap " . g:FuzzyFinder_KeySwitchMode[1] . " OnSwitchMode(-1)" execute "inoremap " . g:FuzzyFinder_KeySwitchIgnoreCase . " OnSwitchIgnoreCase()" endif execute 'file ' . s:vars_current.bufferName " Starts insert mode and makes CursorMovedI event now. Command prompt is " needed to forces a completion menu to update every typing. call feedkeys('i' . s:prompt . s:vars_current.initialInput, 'n') endfunction "----------------------------------------------------------------------------- function! AddMode(new_vars) if !exists('s:vars') let s:vars = {} let s:vars[a:new_vars.name] = a:new_vars let s:vars[a:new_vars.name].nextMode = a:new_vars.name let s:vars[a:new_vars.name].prevMode = a:new_vars.name let s:first_mode_name = a:new_vars.name elseif !has_key(s:vars, a:new_vars.name) let s:vars[a:new_vars.name] = a:new_vars let s:vars[a:new_vars.name].nextMode = s:first_mode_name let s:vars[a:new_vars.name].prevMode = s:vars[s:first_mode_name].prevMode let s:vars[s:vars[a:new_vars.name].nextMode].prevMode = a:new_vars.name let s:vars[s:vars[a:new_vars.name].prevMode].nextMode = a:new_vars.name else let s:vars[a:new_vars.name] = a:new_vars endif endfunction "----------------------------------------------------------------------------- function! Complete(findstart, base) if a:findstart return 0 endif let result = [] if ExistsPrompt(a:base) && strlen(a:base) >= s:vars_current.minLength for expanded_base in ExpandAbbrevMap(a:base[strlen(s:prompt):], s:vars_current.abbrevMap) let result += s:vars_current.complete(expanded_base) endfor endif if empty(result) return [] endif if has_key(result[0], 'order') call sort(result, 'SortByMultipleOrder') endif call feedkeys("\\", 'n') return result endfunction "----------------------------------------------------------------------------- function! CompleteBuffer(base) let patterns = MakeFuzzyPattern(a:base) redir => buffers | silent buffers! | redir END echo 'pattern:' . patterns.wi . (g:FuzzyFinder_Migemo ? ' + migemo' : '') let matchlist_pattern = '^\s*\(\d*\)\([^"]*\)"\([^"]*\)".*$' return map(filter(map(split(buffers, "\n"), \ 'matchlist(v:val, matchlist_pattern)'), \ 'v:val[1] != s:buf_nr && v:val[2] !~ s:vars_current.excludedIndicator && ' . \ '(v:val[1] == patterns.base || v:val[3] =~ patterns.re)'), \ 'MakeCompletionItem(v:val[3], v:val[1], v:val[2] . v:val[3], "", a:base, 1)' \ ) endfunction "----------------------------------------------------------------------------- function! CompleteFile(base) let patterns = map(SplitPath(a:base), 'MakeFuzzyPattern(v:val)') echo 'Making file list...' let result = GlobEx(patterns[0].base, patterns[1].re, s:vars_current.excludedPath) echo 'Evaluating...' if s:vars_current.maxMatches > 0 && len(result) > s:vars_current.maxMatches let result = s:aborted_result else call map(result, 'MakeCompletionItem(v:val, -1, v:val, "", a:base, 1)') endif echo 'pattern:' . patterns[0].base . patterns[1].wi . (g:FuzzyFinder_Migemo ? ' + migemo' : '') return result endfunction "----------------------------------------------------------------------------- function! CompleteMruFile(base) let patterns = MakeFuzzyPattern(a:base) if !has_key(s:info, s:vars_current.name) return [] endif echo 'pattern:' . patterns.wi . (g:FuzzyFinder_Migemo ? ' + migemo' : '') return map(filter(MakeNumberedList(copy(s:info[s:vars_current.name]), 1), \ 'v:val[0] == patterns.base || v:val[1].path =~ patterns.re'), \ 'MakeCompletionItem(v:val[1].path, v:val[0], v:val[1].path, \ strftime(s:vars_current.timeFormat, \ v:val[1].time), \ a:base, 1)' \ ) endfunction "----------------------------------------------------------------------------- function! CompleteMruCmd(base) let patterns = MakeFuzzyPattern(a:base) if !has_key(s:info, s:vars_current.name) return [] endif echo 'pattern:' . patterns.wi . (g:FuzzyFinder_Migemo ? ' + migemo' : '') return map(filter(MakeNumberedList(copy(s:info[s:vars_current.name]), 1), \ 'v:val[0] == patterns.base || v:val[1].command =~ patterns.re'), \ 'MakeCompletionItem(v:val[1].command, v:val[0], v:val[1].command, \ strftime(s:vars_current.timeFormat, \ v:val[1].time), \ a:base, 0)' \ ) endfunction "----------------------------------------------------------------------------- function! CompleteFavFile(base) let patterns = MakeFuzzyPattern(a:base) if !has_key(s:info, s:vars_current.name) return [] endif echo 'pattern:' . patterns.wi . (g:FuzzyFinder_Migemo ? ' + migemo' : '') return map(filter(MakeNumberedList(copy(s:info[s:vars_current.name]), 1), \ 'v:val[0] == patterns.base || v:val[1].path =~ patterns.re'), \ 'MakeCompletionItem(v:val[1].path, v:val[0], v:val[1].path, \ strftime(s:vars_current.timeFormat, \ v:val[1].time), \ a:base, 1)' \ ) endfunction "----------------------------------------------------------------------------- function! CompleteDir(base) let patterns = map(SplitPath(a:base), 'MakeFuzzyPattern(v:val)') echo 'Making directory list...' let result = filter(GlobEx(patterns[0].base, patterns[1].re, s:vars_current.excludedPath), \ 'v:val =~ ''[/\\]$''') if len(patterns[1].base) == 0 call insert(result, patterns[0].base) endif echo 'Evaluating...' call map(result, 'MakeCompletionItem(v:val, -1, v:val, "", a:base, 1)') if len(patterns[1].base) == 0 let result[0].word = matchstr(result[0].word, '^.*[^/\\]') endif echo 'pattern:' . patterns[0].base . patterns[1].wi . (g:FuzzyFinder_Migemo ? ' + migemo' : '') return result endfunction "----------------------------------------------------------------------------- function! OpenBuffer(expr, mode) if a:mode == 0 return ':buffer ' . a:expr . "\" elseif a:mode == 1 return ':sbuffer ' . a:expr . "\" elseif a:mode == 2 return ':vertical :sbuffer ' . a:expr . "\" endif return "" endfunction "----------------------------------------------------------------------------- function! OpenFile_MruFile_FavFile(expr, mode) if a:mode == 0 return ':edit ' . a:expr . "\" elseif a:mode == 1 return ':split ' . a:expr . "\" elseif a:mode == 2 return ':vsplit ' . a:expr . "\" endif return "" endfunction "----------------------------------------------------------------------------- function! OpenMruCmd(expr, mode) if a:mode == 0 call UpdateMruCmdInfo(a:expr) return a:expr . "\" elseif a:mode == 1 || a:mode == 2 return a:expr endif return "" endfunction "----------------------------------------------------------------------------- function! OpenDir(expr, mode) if a:mode == 0 return ':cd ' . a:expr . "\" elseif a:mode == 1 || a:mode == 2 return ':cd ' . a:expr endif return "" endfunction "----------------------------------------------------------------------------- " 'str' -> {'base':'str', 'wi':'*s*t*r*', 're':'\V\.\*s\.\*t\.\*r\.\*'} function! MakeFuzzyPattern(base) let wi = '' for char in split(a:base, '\zs') if wi !~ '[*?]$' && char !~ '[*?]' let wi .= '*'. char else let wi .= char endif endfor if wi !~ '[*?]$' let wi .= '*' endif let re = '\V' . substitute(substitute(substitute(escape(wi, '\'), \ '*', '\\.\\*', 'g'), \ '?', '\\.', 'g'), \ '[', '\\[', 'g') if g:FuzzyFinder_Migemo && a:base !~ '[^\x01-\x7e]' let re .= '\|\m.*' . substitute(migemo(a:base), '\\_s\*', '.*', 'g') . '.*' endif return { 'base': a:base, 'wi':wi, 're': re } endfunction "----------------------------------------------------------------------------- function! ExpandAbbrevMap(base, abbrev_map) let result = [a:base] " expand for [pattern, sub_list] in items(a:abbrev_map) let exprs = result let result = [] for expr in exprs let result += map(copy(sub_list), 'substitute(expr, pattern, v:val, "g")') endfor endfor return Unique(result) endfunction "----------------------------------------------------------------------------- function Unique(in) let result = [] " unique let i_in = 0 | while i_in < len(a:in) let i_result = 0 | while i_result < len(result) if a:in[i_in] == result[i_result] break endif let i_result += 1 endwhile if i_result == len(result) call add(result, a:in[i_in]) endif let i_in += 1 endwhile return result endfunction "----------------------------------------------------------------------------- function! MakeNumberedList(in, first) let i = 0 while i < len(a:in) let a:in[i] = [i + a:first, a:in[i]] let i += 1 endwhile return a:in endfunction "----------------------------------------------------------------------------- function! ExistsPrompt(in) return strlen(a:in) >= strlen(s:prompt) && a:in[:strlen(s:prompt) -1] ==# s:prompt endfunction "----------------------------------------------------------------------------- function! SplitPath(path) let dir = matchstr(a:path, '^.*[/\\]') return [dir, a:path[strlen(dir):]] endfunction "----------------------------------------------------------------------------- function! GlobEx(dir, file, excluded) if !exists('s:glob_ex_caches') let s:glob_ex_caches = {} endif if a:dir =~ '\S' let key = a:dir else let key = ' ' endif if !has_key(s:glob_ex_caches, key) if key =~ '\S' let dir_list = split(expand(a:dir), "\n") else let dir_list = [""] endif let s:glob_ex_caches[key] = [] for d in dir_list for f in ['.*', '*'] let s:glob_ex_caches[key] += split(glob(d . f), "\n") endfor endfor if len(dir_list) <= 1 call map(s:glob_ex_caches[key], '[a:dir, SplitPath(v:val)[1], GetPathSeparatorIfDirectory(v:val)]') else call map(s:glob_ex_caches[key], 'SplitPath(v:val) + [GetPathSeparatorIfDirectory(v:val)]') endif if len(a:excluded) call filter(s:glob_ex_caches[key], 'v:val[0] . v:val[1] . v:val[2] !~ a:excluded') endif endif return map(filter(copy(s:glob_ex_caches[key]), \ 'v:val[1] =~ a:file'), \ 'v:val[0] . v:val[1] . v:val[2]') endfunction "----------------------------------------------------------------------------- function! ClearGlobExCache() unlet! s:glob_ex_caches endfunction "----------------------------------------------------------------------------- function! MakeCompletionItem(expr, number_str, abbr, time, input, evals_path_tail) let number = str2nr(a:number_str) if (number >= 0 && str2nr(number) == str2nr(a:input)) || (a:expr == a:input) let rate = s:matching_rate_base elseif a:evals_path_tail let rate = EvaluateMatchingRate(SplitPath(matchstr(a:expr, '^.*[^/\\]'))[1], \ SplitPath(a:input)[1]) else let rate = EvaluateMatchingRate(a:expr, a:input) endif return { 'word' : a:expr, \ 'abbr' : (number >= 0 ? printf('%2d: ', number) : '') . a:abbr, \ 'menu' : (len(a:time) ? a:time . ' ' : '') . '[' . MakeRateStars(rate, 5) . ']', \ 'order' : [-rate, (number >= 0 ? number : a:expr)]} endfunction "----------------------------------------------------------------------------- function! EvaluateMatchingRate(expr, pattern) if a:expr == a:pattern return s:matching_rate_base endif let rate = 0.0 let rate_increment = (s:matching_rate_base * 9) / (len(a:pattern) * 10) " zero divide ok let matched = 1 let i0 = 0 | let i1 = 0 | while i0 < len(a:expr) && i1 < len(a:pattern) if a:expr[i0] == a:pattern[i1] let i1 += 1 let rate += rate_increment let matched = 1 elseif matched let rate_increment = rate_increment / 2 let matched = 0 endif let i0 += 1 endwhile return rate endfunction "----------------------------------------------------------------------------- function! MakeRateStars(rate, base) let len = (a:base * a:rate) / s:matching_rate_base return repeat('*', len) . repeat('.', a:base - len) endfunction "----------------------------------------------------------------------------- function! SortByMultipleOrder(i1, i2) if has_key(a:i1, 'order') && has_key(a:i2, 'order') let i = 0 | while i < len(a:i1.order) && i < len(a:i2.order) if a:i1.order[i] > a:i2.order[i] return +1 elseif a:i1.order[i] < a:i2.order[i] return -1 endif let i += 1 endwhile endif return 0 endfunction "----------------------------------------------------------------------------- function! GetPathSeparatorIfDirectory(path) if isdirectory(a:path) return s:path_separator else return '' endif endfunction "----------------------------------------------------------------------------- function! ReadInfoFile() try let lines = filter(map(readfile(expand(g:FuzzyFinder_InfoFile)), \ 'matchlist(v:val, "^\\([^\\t]\\+\\)\\t\\(.*\\)$")'), \ '!empty(v:val)') catch /.*/ return {} endtry let info = {} for line in lines if !has_key(info, line[1]) if line[1] == 'version' && line[2] != s:info_version echo printf(s:msg_rm_info, expand(g:FuzzyFinder_InfoFile)) let g:FuzzyFinder_InfoFile = '' return {} endif execute('let info[line[1]] = [' . line[2] . ']') else execute('call add(info[line[1]], ' . line[2] . ')') endif endfor return info endfunction "----------------------------------------------------------------------------- function! WriteInfoFile(info) let a:info.version = [s:info_version] let lines = [] for [key, value] in items(a:info) let lines += map(copy(value), 'key . "\t" . string(v:val)') endfor try call writefile(lines, expand(g:FuzzyFinder_InfoFile)) catch /.*/ endtry endfunction "----------------------------------------------------------------------------- function! UpdateMruFileInfo(path) let s:info = ReadInfoFile() if !has_key(s:info, 'mru_file') let s:info.mru_file = [] endif let s:info.mru_file = filter(insert(filter(s:info.mru_file,'v:val.path != a:path'), \ { 'path' : a:path, 'time' : localtime() }), \ 'v:val.path !~ g:FuzzyFinder_MruFileModeVars.excludedPath && filereadable(v:val.path)' \ )[0 : g:FuzzyFinder_MruFileModeVars.maxItems - 1] call WriteInfoFile(s:info) endfunction "----------------------------------------------------------------------------- function! UpdateMruCmdInfo(command) let s:info = ReadInfoFile() if !has_key(s:info, 'mru_cmd') let s:info.mru_cmd = [] endif let s:info.mru_cmd = filter(insert(filter(s:info.mru_cmd,'v:val.command != a:command'), \ { 'command' : a:command, 'time' : localtime() }), \ 'v:val.command !~ g:FuzzyFinder_MruCmdModeVars.excludedCommand' \ )[0 : g:FuzzyFinder_MruCmdModeVars.maxItems - 1] call WriteInfoFile(s:info) endfunction "----------------------------------------------------------------------------- function! UpdateFavFileInfo(in_file, adds) if empty(a:in_file) let file = expand('%:p') else let file = fnamemodify(a:in_file, ':p') endif let s:info = ReadInfoFile() if !has_key(s:info, 'fav_file') let s:info.fav_file = [] endif let s:info.fav_file = filter(s:info.fav_file, 'v:val.path != file') if a:adds let s:info.fav_file = add(s:info.fav_file, \ { 'path' : file, 'time' : localtime() }) endif call WriteInfoFile(s:info) endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " EVENT HANDLER: """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" "----------------------------------------------------------------------------- function! OnBufEnter() if !g:FuzzyFinder_MruFileModeVars.ignoreSpecialBuffers || empty(&buftype) call UpdateMruFileInfo(expand('%:p')) " == ':p' endif return "" endfunction "----------------------------------------------------------------------------- function! OnBufWritePost() if !g:FuzzyFinder_MruFileModeVars.ignoreSpecialBuffers || empty(&buftype) call UpdateMruFileInfo(expand('%:p')) " == ':p' endif return "" endfunction "----------------------------------------------------------------------------- function! OnCursorMovedI() " Command prompt is removed? if !ExistsPrompt(getline('.')) call setline('.', s:prompt . getline('.')) return repeat("\", len(s:prompt)) elseif col('.') <= len(s:prompt) return repeat("\", len(s:prompt) - col('.') + 1) elseif col('.') > strlen(getline('.')) && col('.') != s:last_col " if the cursor is placed on the end of the line and has been actually moved. let s:last_col = col('.') return "\\" endif return "" endfunction "----------------------------------------------------------------------------- function! OnInsertLeave() if exists('s:reserved_switch_mode') call OpenFuzzyFinder(s:reserved_switch_mode > 0 ? s:vars_current.nextMode \ : s:vars_current.prevMode) unlet s:reserved_switch_mode else quit endif return "" endfunction "----------------------------------------------------------------------------- function! OnBufLeave() " resume autocomplpop.vim if exists(':AutoComplPopUnlock') :AutoComplPopUnlock endif call ClearGlobExCache() let &completeopt = s:_completeopt let &ignorecase = s:_ignorecase unlet s:buf_nr quit " Quit when other window clicked without leaving a insert mode. if exists('s:reserved_command') let command = s:vars_current.open(s:reserved_command[0], s:reserved_command[1]) unlet s:reserved_command else let command = "" endif call garbagecollect() return command endfunction "----------------------------------------------------------------------------- function! OnCR(index, check_dir) if pumvisible() return "\\=" . s:sid_prefix . "OnCR('" . a:index . "'," . 1 . ")\" endif if a:check_dir && getline('.') =~ '[/\\]$' return "" endif if ExistsPrompt(getline('.')) let s:reserved_command = [getline('.')[strlen(s:prompt):], a:index] else let s:reserved_command = [getline('.') , a:index] endif "let s:reserved_command = printf(s:vars_current.commandFormat[a:index], s:reserved_command) return "\" endfunction "----------------------------------------------------------------------------- function! OnBS() if pumvisible() return "\\" else return "\" endif endfunction "----------------------------------------------------------------------------- function! OnSwitchMode(next_prev) let s:reserved_switch_mode = a:next_prev return "\" endfunction "----------------------------------------------------------------------------- function! OnSwitchIgnoreCase() let &ignorecase = !&ignorecase echo "ignorecase = " . &ignorecase let s:last_col = -1 return OnCursorMovedI() endfunction "----------------------------------------------------------------------------- function! OnCmdLineCR() let type = getcmdtype() if type == ':' || type == '/' call UpdateMruCmdInfo(type . getcmdline()) endif return "\" endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " INITIALIZE: """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" call Initialize() """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""