"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " fuzzyfinder.vim : The buffer/file/MRU explorer with the fuzzy pattern """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " " Last Change: 09-Oct-2007. " Author: Takeshi Nishida " Version: 1.0, for Vim 7.0 " Licence: MIT Licence " "----------------------------------------------------------------------------- " Description: " Fuzzyfinder provides convenient ways to quickly reach the buffer/file you " want. Fuzzyfinder finds matching files/buffers with a fuzzy pattern to " which it 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 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: " :FuzzyFinderBuffer command launches the buffer explorer. :FuzzyFinderFile " command launches the file explorer. :FuzzyFinderMru command launches the " MRU explorer. " " It is recommended to map these. Personally, I map to " :FuzzyFinderBuffer and to :FuzzyFinderFile. " " To Switch the mode without leaving a insert mode, use . This key " mapping is customizable. " " If you want to temporarily change whether or not to ignore case, use " . This key mapping is customizable. " " 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 buffer number in the buffer mode and the number in the " MRU 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. These key " mappings are customizable. " " To cancel and return to previous window, leave the insert mode. " " About information file: " Fuzzyfinder writes information of the MRU, etc to file by default " (~/.vimfuzzyfinder). If you don't want the file, set as below: " :let g:FuzzyFinder_InfoFile = '' " " About adding original mode: " This feature is UNDERSPECIFIED. You can add original mode by " :FuzzyFinderAddMode command. To start the explorer with original mode, " use :FuzzyFinder command. " " "----------------------------------------------------------------------------- " 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 mapped to switch the mode in insert 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_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. " '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 file mode. " 'initialInput': " This is a text which is inserted at the beginning of the mode. " '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_MruModeVars: " This is a dictionary. This applies only to the MRU mode. " 'initialInput': " This is a text which is inserted at the beginning of the mode. " 'minLength': " It does not complete if a length of inputting is less than this. " 'excludedPath': " Items whose paths match this are excluded. " 'maxItems': " This is an upper limit of MRU items to be stored. If 0 was set, it " does not limit. " "----------------------------------------------------------------------------- " Thanks: " Vincent Wang " Ingo Karkat " Nikolay Golubev " Brian Doyle " "----------------------------------------------------------------------------- " ChangeLog: " 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:version = '1.0' 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:mode_buffer = 'buffer' let s:mode_file = 'file' let s:mode_mru = 'mru' "------------------------------------------------------------------------- " 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_BufferModeVars') let g:FuzzyFinder_BufferModeVars = {} endif call extend(g:FuzzyFinder_BufferModeVars, \ { 'name' : 'buffer', \ 'command' : [':buffer ', ':sbuffer ', ':vertical :sbuffer '], \ 'bufferName' : '[FuzzyFinder - Buffer]', \ 'initialInput' : '', \ 'minLength' : 0, \ 'excludedIndicator' : '[u\-]', \ 'complete' : function('CompleteBuffer') \ } ,'keep') "......................................................................... if !exists('g:FuzzyFinder_FileModeVars') let g:FuzzyFinder_FileModeVars = {} endif call extend(g:FuzzyFinder_FileModeVars, \ { 'name' : 'file', \ 'command' : [':edit ', ':split ', ':vsplit '], \ 'bufferName' : '[FuzzyFinder - File]', \ 'initialInput' : '', \ 'minLength' : 0, \ 'excludedPath' : '\v\~$|\.bak$|\.swp$|((^|[/\\])\.[/\\]$)', \ 'maxMatches' : 400, \ 'complete' : function('CompleteFile') \ } ,'keep') "......................................................................... if !exists('g:FuzzyFinder_MruModeVars') let g:FuzzyFinder_MruModeVars = {} endif call extend(g:FuzzyFinder_MruModeVars, \ { 'name' : 'mru', \ 'command' : [':edit ', ':split ', ':vsplit '], \ 'bufferName' : '[FuzzyFinder - MRU]', \ 'initialInput' : '', \ 'minLength' : 0, \ 'excludedPath' : '\v\~$|\.bak$|\.swp$', \ 'maxItems' : 20, \ 'complete' : function('CompleteMru') \ } ,'keep') "......................................................................... "------------------------------------------------------------------------- " COMMANDS command! -narg=0 -bar FuzzyFinderBuffer call OpenFuzzyFinder(g:FuzzyFinder_BufferModeVars.name) command! -narg=0 -bar FuzzyFinderFile call OpenFuzzyFinder(g:FuzzyFinder_FileModeVars.name) command! -narg=0 -bar FuzzyFinderMru call OpenFuzzyFinder(g:FuzzyFinder_MruModeVars.name) command! -narg=1 -bar FuzzyFinder call OpenFuzzyFinder() command! -narg=1 -bar FuzzyFinderAddMode call AddMode() "------------------------------------------------------------------------- " AUTOCOMMANDS augroup FuzzyFinder_GlobalAutoCommand autocmd! autocmd BufEnter * call OnBufEnter() "autocmd BufReadPre * call OnBufReadPre() autocmd BufWritePost * call OnBufWritePost() augroup END "------------------------------------------------------------------------- " ETC let s:info = ReadInfoFile() call AddMode(g:FuzzyFinder_BufferModeVars) call AddMode(g:FuzzyFinder_FileModeVars ) call AddMode(g:FuzzyFinder_MruModeVars ) endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " FUNCTIONS """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" "----------------------------------------------------------------------------- function! OpenFuzzyFinder(mode) let s:info = ReadInfoFile() let s:mode_current = a:mode let s:last_col = -1 if exists('s:buf_nr') " a buffer for fuzzyfinder 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 = GetSIDPrefix() . '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 . " OnSwitchMode()" execute "inoremap " . g:FuzzyFinder_KeySwitchIgnoreCase . " OnSwitchIgnoreCase()" endif execute 'file ' . s:vars[s:mode_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[s:mode_current].initialInput, 'n') endfunction "----------------------------------------------------------------------------- function! AddMode(new_vars) if !exists('s:vars') let s:vars = {} endif if has_key(s:vars, a:new_vars.name) let s:vars[a:new_vars.name] = a:new_vars else let s:vars[a:new_vars.name] = a:new_vars if exists('s:last_added_key') let s:vars[a:new_vars.name].nextMode = s:vars[s:last_added_key].nextMode let s:vars[s:last_added_key].nextMode = a:new_vars.name let s:last_added_key = a:new_vars.name else let s:vars[a:new_vars.name].nextMode = a:new_vars.name let s:last_added_key = a:new_vars.name endif endif endfunction "----------------------------------------------------------------------------- function! Complete(findstart, base) if a:findstart return 0 endif if !ExistsPrompt(a:base) return [] endif let result = s:vars[s:mode_current].complete(a:base[strlen(s:prompt):]) if empty(result) return [] endif if has_key(result[0], 'order') call sort(result, 'SortByMultipleOrder') endif call feedkeys("\\") return result endfunction "----------------------------------------------------------------------------- function! CompleteBuffer(base) dict let patterns = MakeFuzzyPattern(a:base) if strlen(a:base) < self.minLength return [] endif echo "Making buffer list..." redir => buffers | silent buffers! | redir END let matchlist_pattern = '^\s*\(\d*\)\s*\([^"]*\)\s*"\([^"]*\)".*$' let result = filter(map(split(buffers, "\n"), \ 'matchlist(v:val, matchlist_pattern)'), \ 'v:val[1] != s:buf_nr && v:val[2] !~ self.excludedIndicator && ' . \ '(v:val[1] == patterns.base || v:val[3] =~ patterns.re)') echo 'Evaluating...' call map(result, 'MakeCompletionItemWithRating(v:val[3], v:val[1], v:val[0], a:base)') echo 'pattern:' . patterns.wi return result endfunction "----------------------------------------------------------------------------- function! CompleteFile(base) dict let patterns = map(SplitPath(a:base), 'MakeFuzzyPattern(v:val)') if strlen(patterns[1].base) < self.minLength return [] endif echo 'Making file list...' let result = GlobEx(patterns[0].base, patterns[1].re, self.excludedPath) echo 'Evaluating...' if g:FuzzyFinder_FileModeVars.maxMatches > 0 && len(result) > g:FuzzyFinder_FileModeVars.maxMatches let result = s:aborted_result else call map(result, 'MakeCompletionItemWithRating(v:val, -1, v:val, a:base)') endif echo 'pattern:' . patterns[0].base . patterns[1].wi return result endfunction "----------------------------------------------------------------------------- function! CompleteMru(base) dict let patterns = MakeFuzzyPattern(a:base) if strlen(a:base) < self.minLength || !has_key(s:info, s:mode_mru) return [] endif let result = copy(s:info[s:mode_mru]) let i = 0 | while i < len(result) let n = i + 1 let result[i] = MakeCompletionItemWithRating(result[i], n, n . ": " . result[i], a:base) let result[i].order[1] = n let result[i].number = n let i += 1 endwhile echo 'pattern:' . patterns.wi return filter(result, 'v:val.number == patterns.base || v:val.word =~ patterns.re') 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') return { 'base': a:base, 'wi':wi, 're': re } 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_cache_dir') || s:glob_ex_cache_dir != a:dir let s:glob_ex_cache_dir = a:dir if a:dir =~ '\S' let dir_list = split(expand(a:dir), "\n") else let dir_list = [""] endif let s:glob_ex_cache = [] for d in dir_list for f in ['.*', '*'] let s:glob_ex_cache += split(glob(d . f), "\n") endfor endfor if len(dir_list) <= 1 call map(s:glob_ex_cache, '[a:dir, SplitPath(v:val)[1], GetPathSeparatorIfDirectory(v:val)]') else call map(s:glob_ex_cache, 'SplitPath(v:val) + [GetPathSeparatorIfDirectory(v:val)]') endif call filter(s:glob_ex_cache, 'v:val[0] . v:val[1] . v:val[2] !~ a:excluded') endif return map(filter(copy(s:glob_ex_cache), \ 'v:val[1] =~ a:file'), \ 'v:val[0] . v:val[1] . v:val[2]') endfunction "----------------------------------------------------------------------------- function! ClearGlobExCache() unlet! s:glob_ex_cache_dir unlet! s:glob_ex_cache endfunction "----------------------------------------------------------------------------- function! MakeCompletionItemWithRating(path, buf_nr, abbr, input) let rate = EvaluateMatchingRate(a:path, a:buf_nr, a:input) let rate_based_5 = (5 * rate) / s:matching_rate_base return { 'word' : a:path, \ 'abbr' : a:abbr, \ 'menu' : '|' . repeat('*', rate_based_5) . repeat('.', 5 - rate_based_5), \ 'order' : [-rate, a:path]} endfunction "----------------------------------------------------------------------------- function! EvaluateMatchingRate(path, buf_nr, input) let path_tail = SplitPath(matchstr(a:path , '^.*[^/\\]'))[1] let input_tail = SplitPath(a:input)[1] if (a:buf_nr >= 0 && str2nr(a:buf_nr) == str2nr(a:input)) || a:path == a:input || path_tail == input_tail return s:matching_rate_base endif let rate = 0.0 let rate_increment = (s:matching_rate_base * 9) / (len(input_tail) * 10) " zero divide ok let matched = 1 let i0 = 0 | let i1 = 0 | while i0 < len(path_tail) && i1 < len(input_tail) if path_tail[i0] == input_tail[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! 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() let info = {} try let lines = filter(map(readfile(expand(g:FuzzyFinder_InfoFile)), \ 'matchlist(v:val, "^\\([^\\t]\\+\\)\\t\\(.*\\)$")'), \ '!empty(v:val)') catch /.*/ return info endtry for line in lines if !has_key(info, line[1]) let info[line[1]] = [] endif call add(info[line[1]], line[2]) endfor if has_key(info, 'version') && info.version[0] != s:version let info = {} endif return info endfunction "----------------------------------------------------------------------------- function! WriteInfoFile(info) let a:info.version = [s:version] let lines = [] for [key, value] in items(a:info) let lines += map(copy(value), 'key . "\t" . v:val') endfor try call writefile(lines, expand(g:FuzzyFinder_InfoFile)) catch /.*/ endtry endfunction "----------------------------------------------------------------------------- function! UpdateMruInfo(file) if !has_key(s:info, 'mru') let s:info.mru = [] endif let s:info.mru = filter(insert(filter(s:info.mru,'v:val != a:file'), a:file), \ 'v:val !~ s:vars.mru.excludedPath && filereadable(v:val)')[0 : s:vars.mru.maxItems - 1] call WriteInfoFile(s:info) endfunction "----------------------------------------------------------------------------- function! GetSIDPrefix() return matchstr(expand(''), '\d\+_') endfunction "----------------------------------------------------------------------------- function! OnBufEnter() call UpdateMruInfo(expand(':p')) return "" endfunction "----------------------------------------------------------------------------- function! OnBufWritePost() call UpdateMruInfo(expand(':p')) 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:vars[s:mode_current].nextMode) 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:reserved_command unlet s:reserved_command else let command = "" endif return command endfunction "----------------------------------------------------------------------------- function! OnCR(index, check_dir) if pumvisible() return "\\=" . GetSIDPrefix() . "OnCR('" . a:index . "'," . 1 . ")\" endif if a:check_dir && getline('.') =~ '[/\\]$' return "" endif if ExistsPrompt(getline('.')) let s:reserved_command = s:vars[s:mode_current].command[a:index] . getline('.')[strlen(s:prompt):] . "\" else let s:reserved_command = s:vars[s:mode_current].command[a:index] . getline('.') . "\" endif return "\" endfunction "----------------------------------------------------------------------------- function! OnBS() if pumvisible() return "\\" else return "\" endif endfunction "----------------------------------------------------------------------------- function! OnSwitchMode() let s:reserved_switch_mode = 1 return "\" endfunction "----------------------------------------------------------------------------- function! OnSwitchIgnoreCase() let &ignorecase = !&ignorecase echo "ignorecase = " . &ignorecase let s:last_col = -1 return OnCursorMovedI() endfunction """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" " INITIALIZE """""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" call Initialize()