auto-pairs/plugin/auto-pairs.vim

343 lines
8.5 KiB
VimL
Raw Permalink Normal View History

2011-05-24 19:31:55 +08:00
" Insert or delete brackets, parens, quotes in pairs.
2011-05-23 01:11:23 +08:00
" Maintainer: JiangMiao <jiangfriend@gmail.com>
2011-12-29 21:42:43 +08:00
" Contributor: camthompson
2012-05-14 09:54:25 +08:00
" Last Change: 2012-05-14
" Version: 1.2.0
" Homepage: http://www.vim.org/scripts/script.php?script_id=3599
2011-05-23 01:11:23 +08:00
" Repository: https://github.com/jiangmiao/auto-pairs
if exists('g:AutoPairsLoaded') || &cp
finish
end
let g:AutoPairsLoaded = 1
if !exists('g:AutoPairs')
2012-01-17 13:14:56 +08:00
let g:AutoPairs = {'(':')', '[':']', '{':'}',"'":"'",'"':'"', '`':'`'}
2011-05-23 01:11:23 +08:00
end
if !exists('g:AutoPairsParens')
let g:AutoPairsParens = {'(':')', '[':']', '{':'}'}
end
2011-06-10 02:23:47 +08:00
let g:AutoExtraPairs = copy(g:AutoPairs)
let g:AutoExtraPairs['<'] = '>'
2011-05-23 01:11:23 +08:00
if !exists('g:AutoPairsMapBS')
let g:AutoPairsMapBS = 1
end
if !exists('g:AutoPairsMapCR')
let g:AutoPairsMapCR = 1
end
if !exists('g:AutoPairsMapSpace')
let g:AutoPairsMapSpace = 1
end
if !exists('g:AutoPairsCenterLine')
let g:AutoPairsCenterLine = 1
end
if !exists('g:AutoPairsShortcutToggle')
let g:AutoPairsShortcutToggle = '<M-p>'
end
2011-06-10 02:36:51 +08:00
if !exists('g:AutoPairsShortcutFastWrap')
let g:AutoPairsShortcutFastWrap = '<M-e>'
end
if !exists('g:AutoPairsShortcutJump')
let g:AutoPairsShortcutJump = '<M-n>'
endif
2012-05-14 09:54:25 +08:00
" Fly mode will for closed pair to jump to closed pair instead of insert.
" also support AutoPairsBackInsert to insert pairs where jumped.
if !exists('g:AutoPairsFlyMode')
let g:AutoPairsFlyMode = 1
endif
2011-05-24 19:07:26 +08:00
2012-05-14 09:54:25 +08:00
" Work with Fly Mode, insert pair where jumped
if !exists('g:AutoPairsShortcutBackInsert')
let g:AutoPairsShortcutBackInsert = '<M-b>'
endif
" Will auto generated {']' => 1, ..., '}' => 1}in initialize.
let g:AutoPairsClosedPairs = {}
2011-05-24 19:07:26 +08:00
2011-05-23 01:11:23 +08:00
function! AutoPairsInsert(key)
if !b:autopairs_enabled
return a:key
end
2011-05-23 01:11:23 +08:00
let line = getline('.')
let prev_char = line[col('.')-2]
let current_char = line[col('.')-1]
let next_char = line[col('.')]
2011-05-23 01:11:23 +08:00
let eol = 0
if col('$') - col('.') <= 1
let eol = 1
end
2011-05-23 01:11:23 +08:00
" Ignore auto close if prev character is \
if prev_char == '\'
return a:key
end
2012-05-14 09:54:25 +08:00
" The key is difference open-pair, then it means only for ) ] } by default
2011-05-23 01:11:23 +08:00
if !has_key(g:AutoPairs, a:key)
" Skip the character if next character is space
if current_char == ' ' && next_char == a:key
return "\<Right>\<Right>"
end
2012-01-17 13:14:56 +08:00
" Skip the character if closed pair is next character
2011-12-22 12:30:04 +08:00
if current_char == ''
let next_lineno = line('.')+1
let next_line = getline(nextnonblank(next_lineno))
let next_char = matchstr(next_line, '\s*\zs.')
if next_char == a:key
return "\<ESC>e^a"
endif
endif
" Skip the character if current character is the same as input
if current_char == a:key
return "\<Right>"
end
2012-05-14 09:54:25 +08:00
" Fly Mode, and the key is closed-pairs, search closed-pair and jump
if g:AutoPairsFlyMode && has_key(g:AutoPairsClosedPairs, a:key)
let b:autopairs_saved_pair = [a:key, getpos('.')]
" Use 's' flag to overwritee '' or not is a question
if(search(a:key, 'W'))
return "\<Right>"
end
endif
" Input directly if the key is not an open key
2011-05-23 01:11:23 +08:00
return a:key
end
let open = a:key
let close = g:AutoPairs[open]
if current_char == close && open == close
return "\<Right>"
end
2011-12-30 16:05:37 +08:00
" Ignore auto close ' if follows a word
" MUST after closed check. 'hello|'
if a:key == "'" && prev_char =~ '\v\w'
return a:key
end
2012-01-17 13:17:21 +08:00
" support for ''' ``` and """
if open == close
" The key must be ' " `
let pprev_char = line[col('.')-3]
if pprev_char == open && prev_char == open
" Double pair found
return a:key
end
end
2011-12-29 21:42:43 +08:00
return open.close."\<Left>"
2011-05-23 01:11:23 +08:00
endfunction
function! AutoPairsDelete()
let line = getline('.')
let current_char = line[col('.')-1]
2011-05-23 01:11:23 +08:00
let prev_char = line[col('.')-2]
let pprev_char = line[col('.')-3]
if pprev_char == '\'
return "\<BS>"
end
" Delete last two spaces in parens, work with MapSpace
if has_key(g:AutoPairs, pprev_char) && prev_char == ' ' && current_char == ' '
return "\<BS>\<DEL>"
endif
if has_key(g:AutoPairs, prev_char)
2011-05-23 01:11:23 +08:00
let close = g:AutoPairs[prev_char]
if match(line,'^\s*'.close, col('.')-1) != -1
2011-11-13 16:47:49 +08:00
let space = matchstr(line, '^\s*', col('.')-1)
return "\<BS>". repeat("\<DEL>", len(space)+1)
2012-03-04 00:59:38 +08:00
else
let nline = getline(line('.')+1)
if nline =~ '^\s*'.close
let space = matchstr(nline, '^\s*')
return "\<BS>\<DEL>". repeat("\<DEL>", len(space)+1)
end
2011-05-23 01:11:23 +08:00
end
end
return "\<BS>"
endfunction
function! AutoPairsJump()
call search('["\]'')}]','W')
2011-05-23 01:11:23 +08:00
endfunction
" Fast wrap the word in brackets
function! AutoPairsFastWrap()
2011-05-24 19:07:26 +08:00
let line = getline('.')
let current_char = line[col('.')-1]
let next_char = line[col('.')]
" Ignore EOL
if col('.') == col('$')
return ''
end
2011-06-10 02:23:47 +08:00
normal! x
2012-05-14 09:54:25 +08:00
if next_char =~ '\s'
2011-06-10 02:23:47 +08:00
call search('\S', 'W')
let next_char = getline('.')[col('.')-1]
2011-05-24 19:07:26 +08:00
end
2011-06-10 02:23:47 +08:00
if has_key(g:AutoExtraPairs, next_char)
let close = g:AutoExtraPairs[next_char]
call search(close, 'W')
return "\<RIGHT>".current_char."\<LEFT>"
else
2012-05-14 09:54:25 +08:00
if next_char =~ '\w'
execute "normal! he"
end
execute "normal! a".current_char
2011-06-10 02:23:47 +08:00
return ""
end
2011-05-24 19:07:26 +08:00
endfunction
2011-05-23 01:11:23 +08:00
function! AutoPairsMap(key)
let escaped_key = substitute(a:key, "'", "''", 'g')
execute 'inoremap <buffer> <silent> '.a:key." <C-R>=AutoPairsInsert('".escaped_key."')<CR>"
2011-05-23 01:11:23 +08:00
endfunction
function! AutoPairsToggle()
if b:autopairs_enabled
let b:autopairs_enabled = 0
echo 'AutoPairs Disabled.'
else
let b:autopairs_enabled = 1
echo 'AutoPairs Enabled.'
end
return ''
endfunction
function! AutoPairsReturn()
let line = getline('.')
let pline = getline(line('.')-1)
let prev_char = pline[strlen(pline)-1]
let cmd = ''
let cur_char = line[col('.')-1]
if has_key(g:AutoPairs, prev_char) && g:AutoPairs[prev_char] == cur_char
if g:AutoPairsCenterLine && winline() * 1.5 >= winheight(0)
2011-06-13 18:13:01 +08:00
let cmd = " \<C-O>zz\<ESC>cl"
end
" conflict with javascript and coffee
" javascript need indent new line
" coffeescript forbid indent new line
2012-05-14 09:54:25 +08:00
if &filetype == 'coffeescript' || &filetype == 'coffee'
return "\<ESC>k==o".cmd
else
return "\<ESC>=ko".cmd
endif
end
return ''
endfunction
function! AutoPairsSpace()
let line = getline('.')
let prev_char = line[col('.')-2]
let cmd = ''
let cur_char =line[col('.')-1]
if has_key(g:AutoPairsParens, prev_char) && g:AutoPairsParens[prev_char] == cur_char
let cmd = "\<SPACE>\<LEFT>"
endif
return "\<SPACE>".cmd
endfunction
2012-05-14 09:54:25 +08:00
function! AutoPairsBackInsert()
if exists('b:autopairs_saved_pair')
let pair = b:autopairs_saved_pair[0]
let pos = b:autopairs_saved_pair[1]
call setpos('.', pos)
return pair
endif
return ''
endfunction
2011-05-23 01:11:23 +08:00
function! AutoPairsInit()
let b:autopairs_loaded = 1
let b:autopairs_enabled = 1
" buffer level map pairs keys
2011-05-23 01:11:23 +08:00
for [open, close] in items(g:AutoPairs)
call AutoPairsMap(open)
if open != close
call AutoPairsMap(close)
end
2011-05-24 19:07:26 +08:00
let g:AutoPairsClosedPairs[close] = 1
2011-05-23 01:11:23 +08:00
endfor
" Still use <buffer> level mapping for <BS> <SPACE>
if g:AutoPairsMapBS
execute 'inoremap <buffer> <silent> <expr> <BS> AutoPairsDelete()'
end
if g:AutoPairsMapSpace
execute 'inoremap <buffer> <silent> <expr> <SPACE> AutoPairsSpace()'
end
if g:AutoPairsShortcutFastWrap != ''
execute 'inoremap <buffer> <silent> '.g:AutoPairsShortcutFastWrap.' <C-R>=AutoPairsFastWrap()<CR>'
end
2012-05-14 09:54:25 +08:00
if g:AutoPairsShortcutBackInsert != ''
execute 'inoremap <buffer> <silent> '.g:AutoPairsShortcutBackInsert.' <C-R>=AutoPairsBackInsert()<CR>'
end
if g:AutoPairsShortcutToggle != ''
" use <expr> to ensure showing the status when toggle
execute 'inoremap <buffer> <silent> <expr> '.g:AutoPairsShortcutToggle.' AutoPairsToggle()'
execute 'noremap <buffer> <silent> '.g:AutoPairsShortcutToggle.' :call AutoPairsToggle()<CR>'
2011-05-23 01:11:23 +08:00
end
if g:AutoPairsShortcutJump != ''
execute 'inoremap <buffer> <silent> ' . g:AutoPairsShortcutJump. ' <ESC>:call AutoPairsJump()<CR>a'
execute 'noremap <buffer> <silent> ' . g:AutoPairsShortcutJump. ' :call AutoPairsJump()<CR>'
end
2011-05-23 01:11:23 +08:00
endfunction
function! AutoPairsForceInit()
if exists('b:autopairs_loaded')
return
else
call AutoPairsInit()
endif
endfunction
2012-03-06 23:12:55 +08:00
" Always silent the command
inoremap <silent> <SID>AutoPairsReturn <C-R>=AutoPairsReturn()<CR>
" Global keys mapping
" comptible with other plugin
if g:AutoPairsMapCR
let old_cr = maparg('<CR>', 'i')
if old_cr == ''
let old_cr = '<CR>'
endif
if old_cr !~ 'AutoPairsReturn'
2012-03-06 23:24:17 +08:00
execute 'imap <CR> '.old_cr.'<SID>AutoPairsReturn'
end
endif
au BufEnter * :call AutoPairsForceInit()