" Author: Eric Van Dewoestine " Original concept and versions up to 0.32 written by " Gergely Kontra " Version: 2.1 " GetLatestVimScripts: 1643 1 :AutoInstall: supertab.vim " " Description: {{{ " Use your tab key to do all your completion in insert mode! " You can cycle forward and backward with the and keys " Note: you must press once to be able to cycle back " " http://www.vim.org/scripts/script.php?script_id=1643 " }}} " " License: {{{ " Copyright (c) 2002 - 2014 " All rights reserved. " " Redistribution and use of this software in source and binary forms, with " or without modification, are permitted provided that the following " conditions are met: " " * Redistributions of source code must retain the above " copyright notice, this list of conditions and the " following disclaimer. " " * Redistributions in binary form must reproduce the above " copyright notice, this list of conditions and the " following disclaimer in the documentation and/or other " materials provided with the distribution. " " * Neither the name of Gergely Kontra or Eric Van Dewoestine nor the names " of its contributors may be used to endorse or promote products derived " from this software without specific prior written permission of Gergely " Kontra or Eric Van Dewoestine. " " THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS " IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, " THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR " PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR " CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, " EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, " PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR " PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF " LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING " NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS " SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. " }}} " " Testing Info: {{{ " Running vim + supertab with the absolute bare minimum settings: " $ vim -u NONE -U NONE -c "set nocp | runtime plugin/supertab.vim" " }}} if v:version < 700 finish endif if exists('complType') " Integration with other completion functions. finish endif if exists("loaded_supertab") finish endif let loaded_supertab = 1 let s:save_cpo=&cpo set cpo&vim " Global Variables {{{ if !exists("g:SuperTabDefaultCompletionType") let g:SuperTabDefaultCompletionType = "" endif if !exists("g:SuperTabContextDefaultCompletionType") let g:SuperTabContextDefaultCompletionType = "" endif if !exists("g:SuperTabCompletionContexts") let g:SuperTabCompletionContexts = ['s:ContextText'] endif if !exists("g:SuperTabRetainCompletionDuration") let g:SuperTabRetainCompletionDuration = 'insert' endif if !exists("g:SuperTabNoCompleteBefore") " retain backwards compatability if exists("g:SuperTabMidWordCompletion") && !g:SuperTabMidWordCompletion let g:SuperTabNoCompleteBefore = ['\k'] else let g:SuperTabNoCompleteBefore = [] endif endif if !exists("g:SuperTabNoCompleteAfter") " retain backwards compatability if exists("g:SuperTabLeadingSpaceCompletion") && g:SuperTabLeadingSpaceCompletion let g:SuperTabNoCompleteAfter = [] else let g:SuperTabNoCompleteAfter = ['^', '\s'] endif endif if !exists("g:SuperTabMappingForward") let g:SuperTabMappingForward = '' endif if !exists("g:SuperTabMappingBackward") let g:SuperTabMappingBackward = '' endif if !exists("g:SuperTabMappingTabLiteral") let g:SuperTabMappingTabLiteral = '' endif if !exists("g:SuperTabLongestEnhanced") let g:SuperTabLongestEnhanced = 0 endif if !exists("g:SuperTabLongestHighlight") let g:SuperTabLongestHighlight = 0 endif if !exists("g:SuperTabCrMapping") let g:SuperTabCrMapping = 1 endif if !exists("g:SuperTabClosePreviewOnPopupClose") let g:SuperTabClosePreviewOnPopupClose = 0 endif if !exists("g:SuperTabUndoBreak") let g:SuperTabUndoBreak = 0 endif " }}} " Script Variables {{{ " construct the help text. let s:tabHelp = \ "Hit or CTRL-] on the completion type you wish to switch to.\n" . \ "Use :help ins-completion for more information.\n" . \ "\n" . \ "|| - Keywords in 'complete' searching down.\n" . \ "|| - Keywords in 'complete' searching up (SuperTab default).\n" . \ "|| - Whole lines.\n" . \ "|| - Keywords in current file.\n" . \ "|| - Keywords in 'dictionary'.\n" . \ "|| - Keywords in 'thesaurus', thesaurus-style.\n" . \ "|| - Keywords in the current and included files.\n" . \ "|| - Tags.\n" . \ "|| - File names.\n" . \ "|| - Definitions or macros.\n" . \ "|| - Vim command-line.\n" . \ "|| - User defined completion.\n" . \ "|| - Omni completion.\n" . \ "|s| - Spelling suggestions." " set the available completion types and modes. let s:types = \ "\\\\\\\\" . \ "\\\\\\\\\s" let s:modes = '/^E/^Y/^L/^N/^K/^T/^I/^]/^F/^D/^V/^P/^U/^O/s' let s:types = s:types . "np" let s:modes = s:modes . '/n/p' " }}} function! SuperTabSetDefaultCompletionType(type) " {{{ " Globally available function that users can use to set the default " completion type for the current buffer, like in an ftplugin. " init hack for workaround. let b:complCommandLine = 0 let b:SuperTabDefaultCompletionType = a:type " set the current completion type to the default call SuperTabSetCompletionType(b:SuperTabDefaultCompletionType) endfunction " }}} function! SuperTabSetCompletionType(type) " {{{ " Globally available function that users can use to create mappings to quickly " switch completion modes. Useful when a user wants to restore the default or " switch to another mode without having to kick off a completion of that type " or use SuperTabHelp. Note, this function only changes the current " completion type, not the default, meaning that the default will still be " restored once the configured retension duration has been met (see " g:SuperTabRetainCompletionDuration). To change the default for the current " buffer, use SuperTabDefaultCompletionType(type) instead. Example mapping to " restore SuperTab default: " nmap :call SetSuperTabCompletionType("") call s:InitBuffer() exec "let b:complType = \"" . escape(a:type, '<') . "\"" endfunction " }}} function! SuperTabAlternateCompletion(type) " {{{ " Function which can be mapped to a key to kick off an alternate completion " other than the default. For instance, if you have 'context' as the default " and want to map ctrl+space to issue keyword completion. " Note: due to the way vim expands ctrl characters in mappings, you cannot " create the alternate mapping like so: " imap =SuperTabAlternateCompletion("") " instead, you have to use \ to prevent vim from expanding the key " when creating the mapping. " gvim: " imap =SuperTabAlternateCompletion("\c-p>") " console: " imap =SuperTabAlternateCompletion("\c-p>") call SuperTabSetCompletionType(a:type) " end any current completion before attempting to start the new one. " use feedkeys to prevent possible remapping of from causing issues. "call feedkeys("\", 'n') " ^ since we can't detect completion mode vs regular insert mode, we force " vim into keyword completion mode and end that mode to prevent the regular " insert behavior of from occurring. call feedkeys("\\\", 'n') call feedkeys(b:complType, 'n') return '' endfunction " }}} function! SuperTabLongestHighlight(dir) " {{{ " When longest highlight is enabled, this function is used to do the actual " selection of the completion popup entry. if !pumvisible() return '' endif return a:dir == -1 ? "\" : "\" endfunction " }}} function! s:Init() " {{{ " Setup mechanism to restore original completion type upon leaving insert " mode if configured to do so if g:SuperTabRetainCompletionDuration == 'insert' augroup supertab_retain autocmd! autocmd InsertLeave * call s:SetDefaultCompletionType() augroup END endif endfunction " }}} function! s:InitBuffer() " {{{ if exists('b:SuperTabNoCompleteBefore') return endif let b:complReset = 0 let b:complTypeManual = !exists('b:complTypeManual') ? '' : b:complTypeManual let b:complTypeContext = '' " init hack for workaround. let b:complCommandLine = 0 if !exists('b:SuperTabNoCompleteBefore') let b:SuperTabNoCompleteBefore = g:SuperTabNoCompleteBefore endif if !exists('b:SuperTabNoCompleteAfter') let b:SuperTabNoCompleteAfter = g:SuperTabNoCompleteAfter endif if !exists('b:SuperTabDefaultCompletionType') let b:SuperTabDefaultCompletionType = g:SuperTabDefaultCompletionType endif " set the current completion type to the default call SuperTabSetCompletionType(b:SuperTabDefaultCompletionType) " hack to programatically revert a change to snipmate that breaks supertab " but which the new maintainers don't care about: " http://github.com/garbas/vim-snipmate/issues/37 let snipmate = maparg('', 'i') if snipmate =~ 'u' && g:SuperTabMappingForward =~? '' let snipmate = substitute(snipmate, 'u', '', '') iunmap exec "inoremap " . snipmate endif endfunction " }}} function! s:ManualCompletionEnter() " {{{ " Handles manual entrance into completion mode. if &smd echo '' | echohl ModeMsg | echo '-- ^X++ mode (' . s:modes . ')' | echohl None endif let complType = nr2char(getchar()) if stridx(s:types, complType) != -1 if !exists('b:supertab_close_preview') let b:supertab_close_preview = !s:IsPreviewOpen() endif if stridx("\\", complType) != -1 " no memory, just scroll... return "\" . complType elseif stridx('np', complType) != -1 let complType = nr2char(char2nr(complType) - 96) else let complType = "\" . complType endif let b:complTypeManual = complType if index(['insert', 'session'], g:SuperTabRetainCompletionDuration) != -1 let b:complType = complType endif " Hack to workaround bug when invoking command line completion via = if complType == "\\" return s:CommandLineCompletion() endif call s:InitBuffer() " optionally enable enhanced longest completion if g:SuperTabLongestEnhanced && &completeopt =~ 'longest' call s:EnableLongestEnhancement() " handle backspacing which triggers g:SuperTabNoCompleteAfter match elseif s:IsNoCompleteAfterReset() call s:EnableNoCompleteAfterReset() endif if g:SuperTabLongestHighlight && \ &completeopt =~ 'longest' && \ &completeopt =~ 'menu' && \ !pumvisible() let dir = (complType == "\\") ? -1 : 1 call feedkeys("\=SuperTabLongestHighlight(" . dir . ")\", 'n') endif return complType endif echohl "Unknown mode" return complType endfunction " }}} function! s:SetCompletionType() " {{{ " Sets the completion type based on what the user has chosen from the help " buffer. let chosen = substitute(getline('.'), '.*|\(.*\)|.*', '\1', '') if chosen != getline('.') let winnr = b:winnr close exec winnr . 'winc w' call SuperTabSetCompletionType(chosen) endif endfunction " }}} function! s:SetDefaultCompletionType() " {{{ if exists('b:SuperTabDefaultCompletionType') && \ (!exists('b:complCommandLine') || !b:complCommandLine) call SuperTabSetCompletionType(b:SuperTabDefaultCompletionType) endif endfunction " }}} function! SuperTab(command) " {{{ " Used to perform proper cycle navigation as the user requests the next or " previous entry in a completion list, and determines whether or not to simply " retain the normal usage of based on the cursor position. if exists('b:SuperTabDisabled') && b:SuperTabDisabled if exists('s:Tab') return s:Tab() endif return ( \ g:SuperTabMappingForward ==? '' || \ g:SuperTabMappingBackward ==? '' \ ) ? "\" : '' endif call s:InitBuffer() if s:WillComplete() if !exists('b:supertab_close_preview') let b:supertab_close_preview = !s:IsPreviewOpen() endif " optionally enable enhanced longest completion if g:SuperTabLongestEnhanced && &completeopt =~ 'longest' call s:EnableLongestEnhancement() " handle backspacing which triggers g:SuperTabNoCompleteAfter match elseif s:IsNoCompleteAfterReset() call s:EnableNoCompleteAfterReset() endif if !pumvisible() let b:complTypeManual = '' endif " exception: if in mode, then should move up the list, and " down the list. if a:command == 'p' && !b:complReset && \ (b:complType == "\" || \ (b:complType == 'context' && \ b:complTypeManual == '' && \ b:complTypeContext == "\")) return "\" elseif a:command == 'p' && !b:complReset && \ (b:complType == "\" || \ (b:complType == 'context' && \ b:complTypeManual == '' && \ b:complTypeContext == "\")) return "\" " already in completion mode and not resetting for longest enhancement, so " just scroll to next/previous elseif pumvisible() && !b:complReset let type = b:complType == 'context' ? b:complTypeContext : b:complType if a:command == 'n' return type == "\" || type == "\\" ? "\" : "\" endif return type == "\" || type == "\\" ? "\" : "\" endif " handle 'context' completion. if b:complType == 'context' let complType = s:ContextCompletion() if complType == '' exec "let complType = \"" . \ escape(g:SuperTabContextDefaultCompletionType, '<') . "\"" endif let b:complTypeContext = complType " Hack to workaround bug when invoking command line completion via = elseif b:complType == "\\" let complType = s:CommandLineCompletion() else let complType = b:complType endif " switch / completion in mode if a:command == 'p' if complType == "\\" let complType = "\\" elseif complType == "\\" let complType = "\\" endif endif " highlight first result if longest enabled if g:SuperTabLongestHighlight && \ &completeopt =~ 'longest' && \ &completeopt =~ 'menu' && \ (!pumvisible() || b:complReset) let dir = (complType == "\") ? -1 : 1 call feedkeys("\=SuperTabLongestHighlight(" . dir . ")\", 'n') endif if b:complReset let b:complReset = 0 " not an accurate condition for everyone, but better than sending " at the wrong time. if pumvisible() return "\" . complType endif endif if g:SuperTabUndoBreak && !pumvisible() return "\u" . complType endif return complType endif if exists('s:Tab') return s:Tab() endif return ( \ g:SuperTabMappingForward ==? '' || \ g:SuperTabMappingBackward ==? '' \ ) ? "\" : '' endfunction " }}} function! s:SuperTabHelp() " {{{ " Opens a help window where the user can choose a completion type to enter. let winnr = winnr() if bufwinnr("SuperTabHelp") == -1 botright split SuperTabHelp setlocal noswapfile setlocal buftype=nowrite setlocal bufhidden=delete silent put =s:tabHelp call cursor(1, 1) silent 1,delete _ call cursor(4, 1) exec "resize " . line('$') syntax match Special "|.\{-}|" setlocal readonly setlocal nomodifiable nmap :call SetCompletionType() nmap :call SetCompletionType() else exec bufwinnr("SuperTabHelp") . "winc w" endif let b:winnr = winnr endfunction " }}} function! s:WillComplete(...) " {{{ " Determines if completion should be kicked off at the current location. " Optional arg: " col: The column to check at, otherwise use the current column. " if an arg was supplied, then we will re-check even if already in " completion mode. if pumvisible() && !a:0 return 1 endif let line = getline('.') let cnum = a:0 ? a:1 : col('.') " honor SuperTabNoCompleteAfter let pre = cnum >= 2 ? line[:cnum - 2] : '' let complAfterType = type(b:SuperTabNoCompleteAfter) if complAfterType == 3 " the option was provided as a list of patterns for pattern in b:SuperTabNoCompleteAfter if pre =~ pattern . '$' return 0 endif endfor elseif complAfterType == 2 " the option was provided as a funcref return !b:SuperTabNoCompleteAfter(pre) endif " honor SuperTabNoCompleteBefore " Within a word, but user does not have mid word completion enabled. let post = line[cnum - 1:] let complBeforeType = type(b:SuperTabNoCompleteBefore) if complBeforeType == 3 " a list of patterns for pattern in b:SuperTabNoCompleteBefore if post =~ '^' . pattern return 0 endif endfor elseif complBeforeType == 2 " the option was provided as a funcref return !b:SuperTabNoCompleteBefore(post) endif return 1 endfunction " }}} function! s:EnableLongestEnhancement() " {{{ augroup supertab_reset autocmd! autocmd InsertLeave,CursorMovedI \ call s:ReleaseKeyPresses() | autocmd! supertab_reset augroup END call s:CaptureKeyPresses() endfunction " }}} function! s:IsNoCompleteAfterReset() " {{{ " if the user has g:SuperTabNoCompleteAfter set, then re-map so that " backspacing to a point where one of the g:SuperTabNoCompleteAfter " entries matches will cause completion mode to exit. let complAfterType = type(b:SuperTabNoCompleteAfter) if complAfterType == 2 return 1 endif return len(g:SuperTabNoCompleteAfter) && g:SuperTabNoCompleteAfter != ['^', '\s'] endfunction " }}} function! s:EnableNoCompleteAfterReset() " {{{ augroup supertab_reset autocmd! autocmd InsertLeave,CursorMovedI \ call s:ReleaseKeyPresses() | autocmd! supertab_reset augroup END " short version of s:CaptureKeyPresses if !exists('b:capturing') || !b:capturing let b:capturing = 1 let b:capturing_start = col('.') let b:captured = { \ '': maparg('', 'i'), \ '': maparg('', 'i'), \ } imap =CompletionReset("\bs>") imap =CompletionReset("\c-h>") endif endfunction " }}} function! s:CompletionReset(char) " {{{ let b:complReset = 1 " handle exiting completion mode if user has g:SuperTabNoCompleteAfter set " and they are about to backspace to a point where that maches one of the " entries in that var. if (a:char == "\" || a:char == "\") && s:IsNoCompleteAfterReset() if !s:WillComplete(col('.') - 1) " exit from completion mode then issue the currently requested backspace call feedkeys("\\\", 'n') return '' endif endif return a:char endfunction " }}} function! s:CaptureKeyPresses() " {{{ if !exists('b:capturing') || !b:capturing let b:capturing = 1 let b:capturing_start = col('.') " save any previous mappings " TODO: capture additional info provided by vim 7.3.032 and up. let b:captured = { \ '': maparg('', 'i'), \ '': maparg('', 'i'), \ } " TODO: use &keyword to get an accurate list of chars to map for c in split('abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890_', '.\zs') let existing = maparg(c, 'i') let b:captured[c] = existing exec 'imap ' . c . ' =CompletionReset("' . c . '")' endfor imap =CompletionReset("\bs>") imap =CompletionReset("\c-h>") endif endfunction " }}} function! s:IsPreviewOpen() " {{{ let wins = tabpagewinnr(tabpagenr(), '$') let winnr = 1 while winnr <= wins if getwinvar(winnr, '&previewwindow') == 1 return 1 endif let winnr += 1 endwhile return 0 endfunction " }}} function! s:ClosePreview() " {{{ if exists('b:supertab_close_preview') && b:supertab_close_preview let preview = 0 for bufnum in tabpagebuflist() if getwinvar(bufwinnr(bufnum), '&previewwindow') let preview = 1 break endif endfor if preview pclose try doautocmd supertab_preview_closed User catch /E216/ " ignore: no autocmds defined endtry endif endif silent! unlet b:supertab_close_preview endfunction " }}} function! s:ReleaseKeyPresses() " {{{ if exists('b:capturing') && b:capturing let b:capturing = 0 for c in keys(b:captured) exec 'iunmap ' . c endfor " restore any previous mappings for [key, rhs] in items(b:captured) if rhs != '' let args = substitute(rhs, '.*\(".\{-}"\).*', '\1', '') if args != rhs let args = substitute(args, '<', '', 'g') let expr = substitute(rhs, '\(.*\)".\{-}"\(.*\)', '\1%s\2', '') let rhs = printf(expr, args) endif exec printf("imap %s %s", key, rhs) endif endfor unlet b:captured if mode() == 'i' && &completeopt =~ 'menu' && b:capturing_start != col('.') " force full exit from completion mode (don't exit insert mode since " that will break repeating with '.') call feedkeys("\\", 'n') endif unlet b:capturing_start endif endfunction " }}} function! s:CommandLineCompletion() " {{{ " Hack needed to account for apparent bug in vim command line mode completion " when invoked via = " This hack will trigger InsertLeave which will then invoke " s:SetDefaultCompletionType. To prevent default completion from being " restored prematurely, set an internal flag for s:SetDefaultCompletionType " to check for. let b:complCommandLine = 1 return "\\:call feedkeys('\\\', 'n') | " . \ "let b:complCommandLine = 0\" endfunction " }}} function! s:ContextCompletion() " {{{ let contexts = exists('b:SuperTabCompletionContexts') ? \ b:SuperTabCompletionContexts : g:SuperTabCompletionContexts for context in contexts try let Context = function(context) let complType = Context() unlet Context if type(complType) == 1 && complType != '' return complType endif catch /E700/ echohl Error echom 'supertab: no context function "' . context . '" found.' echohl None endtry endfor return '' endfunction " }}} function! s:ContextDiscover() " {{{ let discovery = exists('g:SuperTabContextDiscoverDiscovery') ? \ g:SuperTabContextDiscoverDiscovery : [] " loop through discovery list to find the default if !empty(discovery) for pair in discovery let var = substitute(pair, '\(.*\):.*', '\1', '') let type = substitute(pair, '.*:\(.*\)', '\1', '') exec 'let value = ' . var if value !~ '^\s*$' && value != '0' exec "let complType = \"" . escape(type, '<') . "\"" return complType endif endfor endif endfunction " }}} function! s:ContextText() " {{{ let exclusions = exists('g:SuperTabContextTextFileTypeExclusions') ? \ g:SuperTabContextTextFileTypeExclusions : [] if index(exclusions, &ft) == -1 let curline = getline('.') let cnum = col('.') let synname = synIDattr(synID(line('.'), cnum - 1, 1), 'name') if curline =~ '.*/\w*\%' . cnum . 'c' || \ ((has('win32') || has('win64')) && curline =~ '.*\\\w*\%' . cnum . 'c') return "\\" elseif curline =~ '.*\(\w\|[\])]\)\(\.\|>\?::\|->\)\w*\%' . cnum . 'c' && \ synname !~ '\(String\|Comment\)' let omniPrecedence = exists('g:SuperTabContextTextOmniPrecedence') ? \ g:SuperTabContextTextOmniPrecedence : ['&completefunc', '&omnifunc'] for omniFunc in omniPrecedence if omniFunc !~ '^&' let omniFunc = '&' . omniFunc endif if getbufvar(bufnr('%'), omniFunc) != '' return omniFunc == '&omnifunc' ? "\\" : "\\" endif endfor endif endif endfunction " }}} function! s:ExpandMap(map) " {{{ let map = a:map if map =~ '' let plug = substitute(map, '.\{-}\(\w\+\).*', '\1', '') let plug_map = maparg(plug, 'i') let map = substitute(map, '.\{-}\(\w\+\).*', plug_map, '') endif return map endfunction " }}} function! SuperTabChain(completefunc, completekeys) " {{{ if a:completefunc != 'SuperTabCodeComplete' let b:SuperTabChain = [a:completefunc, a:completekeys] setlocal completefunc=SuperTabCodeComplete endif endfunction " }}} function! SuperTabCodeComplete(findstart, base) " {{{ if !exists('b:SuperTabChain') echoe 'No completion chain has been set.' return -2 endif if len(b:SuperTabChain) != 2 echoe 'Completion chain can only be used with 1 completion function ' . \ 'and 1 fallback completion key binding.' return -2 endif let Func = function(b:SuperTabChain[0]) if a:findstart let start = Func(a:findstart, a:base) if start >= 0 return start endif return col('.') - 1 endif let results = Func(a:findstart, a:base) if len(results) return results endif exec 'let keys = "' . escape(b:SuperTabChain[1], '<') . '"' call feedkeys("\" . keys, 'nt') return [] endfunction " }}} " Autocmds {{{ if g:SuperTabClosePreviewOnPopupClose augroup supertab_close_preview autocmd! autocmd InsertLeave,CursorMovedI * call s:ClosePreview() augroup END endif " }}} " Key Mappings {{{ " map a regular tab to ctrl-tab (note: doesn't work in console vim) exec 'inoremap ' . g:SuperTabMappingTabLiteral . ' ' imap =ManualCompletionEnter() imap