added vim stuff
This commit is contained in:
parent
09e2e500ce
commit
8730d7bd9f
7 changed files with 1092 additions and 0 deletions
35
vim/.vim/after/plugin/snipMate.vim
Normal file
35
vim/.vim/after/plugin/snipMate.vim
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
" These are the mappings for snipMate.vim. Putting it here ensures that it
|
||||||
|
" will be mapped after other plugins such as supertab.vim.
|
||||||
|
if !exists('loaded_snips') || exists('s:did_snips_mappings')
|
||||||
|
finish
|
||||||
|
endif
|
||||||
|
let s:did_snips_mappings = 1
|
||||||
|
|
||||||
|
ino <silent> <tab> <c-r>=TriggerSnippet()<cr>
|
||||||
|
snor <silent> <tab> <esc>i<right><c-r>=TriggerSnippet()<cr>
|
||||||
|
ino <silent> <s-tab> <c-r>=BackwardsSnippet()<cr>
|
||||||
|
snor <silent> <s-tab> <esc>i<right><c-r>=BackwardsSnippet()<cr>
|
||||||
|
ino <silent> <c-r><tab> <c-r>=ShowAvailableSnips()<cr>
|
||||||
|
|
||||||
|
" The default mappings for these are annoying & sometimes break snipMate.
|
||||||
|
" You can change them back if you want, I've put them here for convenience.
|
||||||
|
snor <bs> b<bs>
|
||||||
|
snor <right> <esc>a
|
||||||
|
snor <left> <esc>bi
|
||||||
|
snor ' b<bs>'
|
||||||
|
snor ` b<bs>`
|
||||||
|
snor % b<bs>%
|
||||||
|
snor U b<bs>U
|
||||||
|
snor ^ b<bs>^
|
||||||
|
snor \ b<bs>\
|
||||||
|
snor <c-x> b<bs><c-x>
|
||||||
|
|
||||||
|
" By default load snippets in snippets_dir
|
||||||
|
if empty(snippets_dir)
|
||||||
|
finish
|
||||||
|
endif
|
||||||
|
|
||||||
|
call GetSnippets(snippets_dir, '_') " Get global snippets
|
||||||
|
|
||||||
|
au FileType * if &ft != 'help' | call GetSnippets(snippets_dir, &ft) | endif
|
||||||
|
" vim:noet:sw=4:ts=4:ft=vim
|
433
vim/.vim/autoload/snipMate.vim
Normal file
433
vim/.vim/autoload/snipMate.vim
Normal file
|
@ -0,0 +1,433 @@
|
||||||
|
fun! Filename(...)
|
||||||
|
let filename = expand('%:t:r')
|
||||||
|
if filename == '' | return a:0 == 2 ? a:2 : '' | endif
|
||||||
|
return !a:0 || a:1 == '' ? filename : substitute(a:1, '$1', filename, 'g')
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun s:RemoveSnippet()
|
||||||
|
unl! g:snipPos s:curPos s:snipLen s:endCol s:endLine s:prevLen
|
||||||
|
\ s:lastBuf s:oldWord
|
||||||
|
if exists('s:update')
|
||||||
|
unl s:startCol s:origWordLen s:update
|
||||||
|
if exists('s:oldVars') | unl s:oldVars s:oldEndCol | endif
|
||||||
|
endif
|
||||||
|
aug! snipMateAutocmds
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun snipMate#expandSnip(snip, col)
|
||||||
|
let lnum = line('.') | let col = a:col
|
||||||
|
|
||||||
|
let snippet = s:ProcessSnippet(a:snip)
|
||||||
|
" Avoid error if eval evaluates to nothing
|
||||||
|
if snippet == '' | return '' | endif
|
||||||
|
|
||||||
|
" Expand snippet onto current position with the tab stops removed
|
||||||
|
let snipLines = split(substitute(snippet, '$\d\+\|${\d\+.\{-}}', '', 'g'), "\n", 1)
|
||||||
|
|
||||||
|
let line = getline(lnum)
|
||||||
|
let afterCursor = strpart(line, col - 1)
|
||||||
|
" Keep text after the cursor
|
||||||
|
if afterCursor != "\t" && afterCursor != ' '
|
||||||
|
let line = strpart(line, 0, col - 1)
|
||||||
|
let snipLines[-1] .= afterCursor
|
||||||
|
else
|
||||||
|
let afterCursor = ''
|
||||||
|
" For some reason the cursor needs to move one right after this
|
||||||
|
if line != '' && col == 1 && &ve != 'all' && &ve != 'onemore'
|
||||||
|
let col += 1
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
call setline(lnum, line.snipLines[0])
|
||||||
|
|
||||||
|
" Autoindent snippet according to previous indentation
|
||||||
|
let indent = matchend(line, '^.\{-}\ze\(\S\|$\)') + 1
|
||||||
|
call append(lnum, map(snipLines[1:], "'".strpart(line, 0, indent - 1)."'.v:val"))
|
||||||
|
|
||||||
|
" Open any folds snippet expands into
|
||||||
|
if &fen | sil! exe lnum.','.(lnum + len(snipLines) - 1).'foldopen' | endif
|
||||||
|
|
||||||
|
let [g:snipPos, s:snipLen] = s:BuildTabStops(snippet, lnum, col - indent, indent)
|
||||||
|
|
||||||
|
if s:snipLen
|
||||||
|
aug snipMateAutocmds
|
||||||
|
au CursorMovedI * call s:UpdateChangedSnip(0)
|
||||||
|
au InsertEnter * call s:UpdateChangedSnip(1)
|
||||||
|
aug END
|
||||||
|
let s:lastBuf = bufnr(0) " Only expand snippet while in current buffer
|
||||||
|
let s:curPos = 0
|
||||||
|
let s:endCol = g:snipPos[s:curPos][1]
|
||||||
|
let s:endLine = g:snipPos[s:curPos][0]
|
||||||
|
|
||||||
|
call cursor(g:snipPos[s:curPos][0], g:snipPos[s:curPos][1])
|
||||||
|
let s:prevLen = [line('$'), col('$')]
|
||||||
|
if g:snipPos[s:curPos][2] != -1 | return s:SelectWord() | endif
|
||||||
|
else
|
||||||
|
unl g:snipPos s:snipLen
|
||||||
|
" Place cursor at end of snippet if no tab stop is given
|
||||||
|
let newlines = len(snipLines) - 1
|
||||||
|
call cursor(lnum + newlines, indent + len(snipLines[-1]) - len(afterCursor)
|
||||||
|
\ + (newlines ? 0: col - 1))
|
||||||
|
endif
|
||||||
|
return ''
|
||||||
|
endf
|
||||||
|
|
||||||
|
" Prepare snippet to be processed by s:BuildTabStops
|
||||||
|
fun s:ProcessSnippet(snip)
|
||||||
|
let snippet = a:snip
|
||||||
|
" Evaluate eval (`...`) expressions.
|
||||||
|
" Using a loop here instead of a regex fixes a bug with nested "\=".
|
||||||
|
if stridx(snippet, '`') != -1
|
||||||
|
while match(snippet, '`.\{-}`') != -1
|
||||||
|
let snippet = substitute(snippet, '`.\{-}`',
|
||||||
|
\ substitute(eval(matchstr(snippet, '`\zs.\{-}\ze`')),
|
||||||
|
\ "\n\\%$", '', ''), '')
|
||||||
|
endw
|
||||||
|
let snippet = substitute(snippet, "\r", "\n", 'g')
|
||||||
|
endif
|
||||||
|
|
||||||
|
" Place all text after a colon in a tab stop after the tab stop
|
||||||
|
" (e.g. "${#:foo}" becomes "${:foo}foo").
|
||||||
|
" This helps tell the position of the tab stops later.
|
||||||
|
let snippet = substitute(snippet, '${\d\+:\(.\{-}\)}', '&\1', 'g')
|
||||||
|
|
||||||
|
" Update the a:snip so that all the $# become the text after
|
||||||
|
" the colon in their associated ${#}.
|
||||||
|
" (e.g. "${1:foo}" turns all "$1"'s into "foo")
|
||||||
|
let i = 1
|
||||||
|
while stridx(snippet, '${'.i) != -1
|
||||||
|
let s = matchstr(snippet, '${'.i.':\zs.\{-}\ze}')
|
||||||
|
if s != ''
|
||||||
|
let snippet = substitute(snippet, '$'.i, s.'&', 'g')
|
||||||
|
endif
|
||||||
|
let i += 1
|
||||||
|
endw
|
||||||
|
|
||||||
|
if &et " Expand tabs to spaces if 'expandtab' is set.
|
||||||
|
return substitute(snippet, '\t', repeat(' ', &sts ? &sts : &sw), 'g')
|
||||||
|
endif
|
||||||
|
return snippet
|
||||||
|
endf
|
||||||
|
|
||||||
|
" Counts occurences of haystack in needle
|
||||||
|
fun s:Count(haystack, needle)
|
||||||
|
let counter = 0
|
||||||
|
let index = stridx(a:haystack, a:needle)
|
||||||
|
while index != -1
|
||||||
|
let index = stridx(a:haystack, a:needle, index+1)
|
||||||
|
let counter += 1
|
||||||
|
endw
|
||||||
|
return counter
|
||||||
|
endf
|
||||||
|
|
||||||
|
" Builds a list of a list of each tab stop in the snippet containing:
|
||||||
|
" 1.) The tab stop's line number.
|
||||||
|
" 2.) The tab stop's column number
|
||||||
|
" (by getting the length of the string between the last "\n" and the
|
||||||
|
" tab stop).
|
||||||
|
" 3.) The length of the text after the colon for the current tab stop
|
||||||
|
" (e.g. "${1:foo}" would return 3). If there is no text, -1 is returned.
|
||||||
|
" 4.) If the "${#:}" construct is given, another list containing all
|
||||||
|
" the matches of "$#", to be replaced with the placeholder. This list is
|
||||||
|
" composed the same way as the parent; the first item is the line number,
|
||||||
|
" and the second is the column.
|
||||||
|
fun s:BuildTabStops(snip, lnum, col, indent)
|
||||||
|
let snipPos = []
|
||||||
|
let i = 1
|
||||||
|
let withoutVars = substitute(a:snip, '$\d\+', '', 'g')
|
||||||
|
while stridx(a:snip, '${'.i) != -1
|
||||||
|
let beforeTabStop = matchstr(withoutVars, '^.*\ze${'.i.'\D')
|
||||||
|
let withoutOthers = substitute(withoutVars, '${\('.i.'\D\)\@!\d\+.\{-}}', '', 'g')
|
||||||
|
|
||||||
|
let j = i - 1
|
||||||
|
call add(snipPos, [0, 0, -1])
|
||||||
|
let snipPos[j][0] = a:lnum + s:Count(beforeTabStop, "\n")
|
||||||
|
let snipPos[j][1] = a:indent + len(matchstr(withoutOthers, '.*\(\n\|^\)\zs.*\ze${'.i.'\D'))
|
||||||
|
if snipPos[j][0] == a:lnum | let snipPos[j][1] += a:col | endif
|
||||||
|
|
||||||
|
" Get all $# matches in another list, if ${#:name} is given
|
||||||
|
if stridx(withoutVars, '${'.i.':') != -1
|
||||||
|
let snipPos[j][2] = len(matchstr(withoutVars, '${'.i.':\zs.\{-}\ze}'))
|
||||||
|
let dots = repeat('.', snipPos[j][2])
|
||||||
|
call add(snipPos[j], [])
|
||||||
|
let withoutOthers = substitute(a:snip, '${\d\+.\{-}}\|$'.i.'\@!\d\+', '', 'g')
|
||||||
|
while match(withoutOthers, '$'.i.'\(\D\|$\)') != -1
|
||||||
|
let beforeMark = matchstr(withoutOthers, '^.\{-}\ze'.dots.'$'.i.'\(\D\|$\)')
|
||||||
|
call add(snipPos[j][3], [0, 0])
|
||||||
|
let snipPos[j][3][-1][0] = a:lnum + s:Count(beforeMark, "\n")
|
||||||
|
let snipPos[j][3][-1][1] = a:indent + (snipPos[j][3][-1][0] > a:lnum
|
||||||
|
\ ? len(matchstr(beforeMark, '.*\n\zs.*'))
|
||||||
|
\ : a:col + len(beforeMark))
|
||||||
|
let withoutOthers = substitute(withoutOthers, '$'.i.'\ze\(\D\|$\)', '', '')
|
||||||
|
endw
|
||||||
|
endif
|
||||||
|
let i += 1
|
||||||
|
endw
|
||||||
|
return [snipPos, i - 1]
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun snipMate#jumpTabStop(backwards)
|
||||||
|
let leftPlaceholder = exists('s:origWordLen')
|
||||||
|
\ && s:origWordLen != g:snipPos[s:curPos][2]
|
||||||
|
if leftPlaceholder && exists('s:oldEndCol')
|
||||||
|
let startPlaceholder = s:oldEndCol + 1
|
||||||
|
endif
|
||||||
|
|
||||||
|
if exists('s:update')
|
||||||
|
call s:UpdatePlaceholderTabStops()
|
||||||
|
else
|
||||||
|
call s:UpdateTabStops()
|
||||||
|
endif
|
||||||
|
|
||||||
|
" Don't reselect placeholder if it has been modified
|
||||||
|
if leftPlaceholder && g:snipPos[s:curPos][2] != -1
|
||||||
|
if exists('startPlaceholder')
|
||||||
|
let g:snipPos[s:curPos][1] = startPlaceholder
|
||||||
|
else
|
||||||
|
let g:snipPos[s:curPos][1] = col('.')
|
||||||
|
let g:snipPos[s:curPos][2] = 0
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
let s:curPos += a:backwards ? -1 : 1
|
||||||
|
" Loop over the snippet when going backwards from the beginning
|
||||||
|
if s:curPos < 0 | let s:curPos = s:snipLen - 1 | endif
|
||||||
|
|
||||||
|
if s:curPos == s:snipLen
|
||||||
|
let sMode = s:endCol == g:snipPos[s:curPos-1][1]+g:snipPos[s:curPos-1][2]
|
||||||
|
call s:RemoveSnippet()
|
||||||
|
return sMode ? "\<tab>" : TriggerSnippet()
|
||||||
|
endif
|
||||||
|
|
||||||
|
call cursor(g:snipPos[s:curPos][0], g:snipPos[s:curPos][1])
|
||||||
|
|
||||||
|
let s:endLine = g:snipPos[s:curPos][0]
|
||||||
|
let s:endCol = g:snipPos[s:curPos][1]
|
||||||
|
let s:prevLen = [line('$'), col('$')]
|
||||||
|
|
||||||
|
return g:snipPos[s:curPos][2] == -1 ? '' : s:SelectWord()
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun s:UpdatePlaceholderTabStops()
|
||||||
|
let changeLen = s:origWordLen - g:snipPos[s:curPos][2]
|
||||||
|
unl s:startCol s:origWordLen s:update
|
||||||
|
if !exists('s:oldVars') | return | endif
|
||||||
|
" Update tab stops in snippet if text has been added via "$#"
|
||||||
|
" (e.g., in "${1:foo}bar$1${2}").
|
||||||
|
if changeLen != 0
|
||||||
|
let curLine = line('.')
|
||||||
|
|
||||||
|
for pos in g:snipPos
|
||||||
|
if pos == g:snipPos[s:curPos] | continue | endif
|
||||||
|
let changed = pos[0] == curLine && pos[1] > s:oldEndCol
|
||||||
|
let changedVars = 0
|
||||||
|
let endPlaceholder = pos[2] - 1 + pos[1]
|
||||||
|
" Subtract changeLen from each tab stop that was after any of
|
||||||
|
" the current tab stop's placeholders.
|
||||||
|
for [lnum, col] in s:oldVars
|
||||||
|
if lnum > pos[0] | break | endif
|
||||||
|
if pos[0] == lnum
|
||||||
|
if pos[1] > col || (pos[2] == -1 && pos[1] == col)
|
||||||
|
let changed += 1
|
||||||
|
elseif col < endPlaceholder
|
||||||
|
let changedVars += 1
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
let pos[1] -= changeLen * changed
|
||||||
|
let pos[2] -= changeLen * changedVars " Parse variables within placeholders
|
||||||
|
" e.g., "${1:foo} ${2:$1bar}"
|
||||||
|
|
||||||
|
if pos[2] == -1 | continue | endif
|
||||||
|
" Do the same to any placeholders in the other tab stops.
|
||||||
|
for nPos in pos[3]
|
||||||
|
let changed = nPos[0] == curLine && nPos[1] > s:oldEndCol
|
||||||
|
for [lnum, col] in s:oldVars
|
||||||
|
if lnum > nPos[0] | break | endif
|
||||||
|
if nPos[0] == lnum && nPos[1] > col
|
||||||
|
let changed += 1
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
let nPos[1] -= changeLen * changed
|
||||||
|
endfor
|
||||||
|
endfor
|
||||||
|
endif
|
||||||
|
unl s:endCol s:oldVars s:oldEndCol
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun s:UpdateTabStops()
|
||||||
|
let changeLine = s:endLine - g:snipPos[s:curPos][0]
|
||||||
|
let changeCol = s:endCol - g:snipPos[s:curPos][1]
|
||||||
|
if exists('s:origWordLen')
|
||||||
|
let changeCol -= s:origWordLen
|
||||||
|
unl s:origWordLen
|
||||||
|
endif
|
||||||
|
let lnum = g:snipPos[s:curPos][0]
|
||||||
|
let col = g:snipPos[s:curPos][1]
|
||||||
|
" Update the line number of all proceeding tab stops if <cr> has
|
||||||
|
" been inserted.
|
||||||
|
if changeLine != 0
|
||||||
|
let changeLine -= 1
|
||||||
|
for pos in g:snipPos
|
||||||
|
if pos[0] >= lnum
|
||||||
|
if pos[0] == lnum | let pos[1] += changeCol | endif
|
||||||
|
let pos[0] += changeLine
|
||||||
|
endif
|
||||||
|
if pos[2] == -1 | continue | endif
|
||||||
|
for nPos in pos[3]
|
||||||
|
if nPos[0] >= lnum
|
||||||
|
if nPos[0] == lnum | let nPos[1] += changeCol | endif
|
||||||
|
let nPos[0] += changeLine
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
endfor
|
||||||
|
elseif changeCol != 0
|
||||||
|
" Update the column of all proceeding tab stops if text has
|
||||||
|
" been inserted/deleted in the current line.
|
||||||
|
for pos in g:snipPos
|
||||||
|
if pos[1] >= col && pos[0] == lnum
|
||||||
|
let pos[1] += changeCol
|
||||||
|
endif
|
||||||
|
if pos[2] == -1 | continue | endif
|
||||||
|
for nPos in pos[3]
|
||||||
|
if nPos[0] > lnum | break | endif
|
||||||
|
if nPos[0] == lnum && nPos[1] >= col
|
||||||
|
let nPos[1] += changeCol
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
endfor
|
||||||
|
endif
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun s:SelectWord()
|
||||||
|
let s:origWordLen = g:snipPos[s:curPos][2]
|
||||||
|
let s:oldWord = strpart(getline('.'), g:snipPos[s:curPos][1] - 1,
|
||||||
|
\ s:origWordLen)
|
||||||
|
let s:prevLen[1] -= s:origWordLen
|
||||||
|
if !empty(g:snipPos[s:curPos][3])
|
||||||
|
let s:update = 1
|
||||||
|
let s:endCol = -1
|
||||||
|
let s:startCol = g:snipPos[s:curPos][1] - 1
|
||||||
|
endif
|
||||||
|
if !s:origWordLen | return '' | endif
|
||||||
|
let l = col('.') != 1 ? 'l' : ''
|
||||||
|
if &sel == 'exclusive'
|
||||||
|
return "\<esc>".l.'v'.s:origWordLen."l\<c-g>"
|
||||||
|
endif
|
||||||
|
return s:origWordLen == 1 ? "\<esc>".l.'gh'
|
||||||
|
\ : "\<esc>".l.'v'.(s:origWordLen - 1)."l\<c-g>"
|
||||||
|
endf
|
||||||
|
|
||||||
|
" This updates the snippet as you type when text needs to be inserted
|
||||||
|
" into multiple places (e.g. in "${1:default text}foo$1bar$1",
|
||||||
|
" "default text" would be highlighted, and if the user types something,
|
||||||
|
" UpdateChangedSnip() would be called so that the text after "foo" & "bar"
|
||||||
|
" are updated accordingly)
|
||||||
|
"
|
||||||
|
" It also automatically quits the snippet if the cursor is moved out of it
|
||||||
|
" while in insert mode.
|
||||||
|
fun s:UpdateChangedSnip(entering)
|
||||||
|
if exists('g:snipPos') && bufnr(0) != s:lastBuf
|
||||||
|
call s:RemoveSnippet()
|
||||||
|
elseif exists('s:update') " If modifying a placeholder
|
||||||
|
if !exists('s:oldVars') && s:curPos + 1 < s:snipLen
|
||||||
|
" Save the old snippet & word length before it's updated
|
||||||
|
" s:startCol must be saved too, in case text is added
|
||||||
|
" before the snippet (e.g. in "foo$1${2}bar${1:foo}").
|
||||||
|
let s:oldEndCol = s:startCol
|
||||||
|
let s:oldVars = deepcopy(g:snipPos[s:curPos][3])
|
||||||
|
endif
|
||||||
|
let col = col('.') - 1
|
||||||
|
|
||||||
|
if s:endCol != -1
|
||||||
|
let changeLen = col('$') - s:prevLen[1]
|
||||||
|
let s:endCol += changeLen
|
||||||
|
else " When being updated the first time, after leaving select mode
|
||||||
|
if a:entering | return | endif
|
||||||
|
let s:endCol = col - 1
|
||||||
|
endif
|
||||||
|
|
||||||
|
" If the cursor moves outside the snippet, quit it
|
||||||
|
if line('.') != g:snipPos[s:curPos][0] || col < s:startCol ||
|
||||||
|
\ col - 1 > s:endCol
|
||||||
|
unl! s:startCol s:origWordLen s:oldVars s:update
|
||||||
|
return s:RemoveSnippet()
|
||||||
|
endif
|
||||||
|
|
||||||
|
call s:UpdateVars()
|
||||||
|
let s:prevLen[1] = col('$')
|
||||||
|
elseif exists('g:snipPos')
|
||||||
|
if !a:entering && g:snipPos[s:curPos][2] != -1
|
||||||
|
let g:snipPos[s:curPos][2] = -2
|
||||||
|
endif
|
||||||
|
|
||||||
|
let col = col('.')
|
||||||
|
let lnum = line('.')
|
||||||
|
let changeLine = line('$') - s:prevLen[0]
|
||||||
|
|
||||||
|
if lnum == s:endLine
|
||||||
|
let s:endCol += col('$') - s:prevLen[1]
|
||||||
|
let s:prevLen = [line('$'), col('$')]
|
||||||
|
endif
|
||||||
|
if changeLine != 0
|
||||||
|
let s:endLine += changeLine
|
||||||
|
let s:endCol = col
|
||||||
|
endif
|
||||||
|
|
||||||
|
" Delete snippet if cursor moves out of it in insert mode
|
||||||
|
if (lnum == s:endLine && (col > s:endCol || col < g:snipPos[s:curPos][1]))
|
||||||
|
\ || lnum > s:endLine || lnum < g:snipPos[s:curPos][0]
|
||||||
|
call s:RemoveSnippet()
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endf
|
||||||
|
|
||||||
|
" This updates the variables in a snippet when a placeholder has been edited.
|
||||||
|
" (e.g., each "$1" in "${1:foo} $1bar $1bar")
|
||||||
|
fun s:UpdateVars()
|
||||||
|
let newWordLen = s:endCol - s:startCol + 1
|
||||||
|
let newWord = strpart(getline('.'), s:startCol, newWordLen)
|
||||||
|
if newWord == s:oldWord || empty(g:snipPos[s:curPos][3])
|
||||||
|
return
|
||||||
|
endif
|
||||||
|
|
||||||
|
let changeLen = g:snipPos[s:curPos][2] - newWordLen
|
||||||
|
let curLine = line('.')
|
||||||
|
let startCol = col('.')
|
||||||
|
let oldStartSnip = s:startCol
|
||||||
|
let updateTabStops = changeLen != 0
|
||||||
|
let i = 0
|
||||||
|
|
||||||
|
for [lnum, col] in g:snipPos[s:curPos][3]
|
||||||
|
if updateTabStops
|
||||||
|
let start = s:startCol
|
||||||
|
if lnum == curLine && col <= start
|
||||||
|
let s:startCol -= changeLen
|
||||||
|
let s:endCol -= changeLen
|
||||||
|
endif
|
||||||
|
for nPos in g:snipPos[s:curPos][3][(i):]
|
||||||
|
" This list is in ascending order, so quit if we've gone too far.
|
||||||
|
if nPos[0] > lnum | break | endif
|
||||||
|
if nPos[0] == lnum && nPos[1] > col
|
||||||
|
let nPos[1] -= changeLen
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
if lnum == curLine && col > start
|
||||||
|
let col -= changeLen
|
||||||
|
let g:snipPos[s:curPos][3][i][1] = col
|
||||||
|
endif
|
||||||
|
let i += 1
|
||||||
|
endif
|
||||||
|
|
||||||
|
" "Very nomagic" is used here to allow special characters.
|
||||||
|
call setline(lnum, substitute(getline(lnum), '\%'.col.'c\V'.
|
||||||
|
\ escape(s:oldWord, '\'), escape(newWord, '\&'), ''))
|
||||||
|
endfor
|
||||||
|
if oldStartSnip != s:startCol
|
||||||
|
call cursor(0, startCol + s:startCol - oldStartSnip)
|
||||||
|
endif
|
||||||
|
|
||||||
|
let s:oldWord = newWord
|
||||||
|
let g:snipPos[s:curPos][2] = newWordLen
|
||||||
|
endf
|
||||||
|
" vim:noet:sw=4:ts=4:ft=vim
|
286
vim/.vim/doc/snipMate.txt
Normal file
286
vim/.vim/doc/snipMate.txt
Normal file
|
@ -0,0 +1,286 @@
|
||||||
|
*snipMate.txt* Plugin for using TextMate-style snippets in Vim.
|
||||||
|
|
||||||
|
snipMate *snippet* *snippets* *snipMate*
|
||||||
|
Last Change: July 13, 2009
|
||||||
|
|
||||||
|
|snipMate-description| Description
|
||||||
|
|snipMate-syntax| Snippet syntax
|
||||||
|
|snipMate-usage| Usage
|
||||||
|
|snipMate-settings| Settings
|
||||||
|
|snipMate-features| Features
|
||||||
|
|snipMate-disadvantages| Disadvantages to TextMate
|
||||||
|
|snipMate-contact| Contact
|
||||||
|
|
||||||
|
For Vim version 7.0 or later.
|
||||||
|
This plugin only works if 'compatible' is not set.
|
||||||
|
{Vi does not have any of these features.}
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
DESCRIPTION *snipMate-description*
|
||||||
|
|
||||||
|
snipMate.vim implements some of TextMate's snippets features in Vim. A
|
||||||
|
snippet is a piece of often-typed text that you can insert into your
|
||||||
|
document using a trigger word followed by a <tab>.
|
||||||
|
|
||||||
|
For instance, in a C file using the default installation of snipMate.vim, if
|
||||||
|
you type "for<tab>" in insert mode, it will expand a typical for loop in C: >
|
||||||
|
|
||||||
|
for (i = 0; i < count; i++) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
To go to the next item in the loop, simply <tab> over to it; if there is
|
||||||
|
repeated code, such as the "i" variable in this example, you can simply
|
||||||
|
start typing once it's highlighted and all the matches specified in the
|
||||||
|
snippet will be updated. To go in reverse, use <shift-tab>.
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
SYNTAX *snippet-syntax*
|
||||||
|
|
||||||
|
Snippets can be defined in two ways. They can be in their own file, named
|
||||||
|
after their trigger in 'snippets/<filetype>/<trigger>.snippet', or they can be
|
||||||
|
defined together in a 'snippets/<filetype>.snippets' file. Note that dotted
|
||||||
|
'filetype' syntax is supported -- e.g., you can use >
|
||||||
|
|
||||||
|
:set ft=html.eruby
|
||||||
|
|
||||||
|
to activate snippets for both HTML and eRuby for the current file.
|
||||||
|
|
||||||
|
The syntax for snippets in *.snippets files is the following: >
|
||||||
|
|
||||||
|
snippet trigger
|
||||||
|
expanded text
|
||||||
|
more expanded text
|
||||||
|
|
||||||
|
Note that the first hard tab after the snippet trigger is required, and not
|
||||||
|
expanded in the actual snippet. The syntax for *.snippet files is the same,
|
||||||
|
only without the trigger declaration and starting indentation.
|
||||||
|
|
||||||
|
Also note that snippets must be defined using hard tabs. They can be expanded
|
||||||
|
to spaces later if desired (see |snipMate-indenting|).
|
||||||
|
|
||||||
|
"#" is used as a line-comment character in *.snippets files; however, they can
|
||||||
|
only be used outside of a snippet declaration. E.g.: >
|
||||||
|
|
||||||
|
# this is a correct comment
|
||||||
|
snippet trigger
|
||||||
|
expanded text
|
||||||
|
snippet another_trigger
|
||||||
|
# this isn't a comment!
|
||||||
|
expanded text
|
||||||
|
<
|
||||||
|
This should hopefully be obvious with the included syntax highlighting.
|
||||||
|
|
||||||
|
*snipMate-${#}*
|
||||||
|
Tab stops ~
|
||||||
|
|
||||||
|
By default, the cursor is placed at the end of a snippet. To specify where the
|
||||||
|
cursor is to be placed next, use "${#}", where the # is the number of the tab
|
||||||
|
stop. E.g., to place the cursor first on the id of a <div> tag, and then allow
|
||||||
|
the user to press <tab> to go to the middle of it:
|
||||||
|
>
|
||||||
|
snippet div
|
||||||
|
<div id="${1}">
|
||||||
|
${2}
|
||||||
|
</div>
|
||||||
|
<
|
||||||
|
*snipMate-placeholders* *snipMate-${#:}* *snipMate-$#*
|
||||||
|
Placeholders ~
|
||||||
|
|
||||||
|
Placeholder text can be supplied using "${#:text}", where # is the number of
|
||||||
|
the tab stop. This text then can be copied throughout the snippet using "$#",
|
||||||
|
given # is the same number as used before. So, to make a C for loop: >
|
||||||
|
|
||||||
|
snippet for
|
||||||
|
for (${2:i}; $2 < ${1:count}; $1++) {
|
||||||
|
${4}
|
||||||
|
}
|
||||||
|
|
||||||
|
This will cause "count" to first be selected and change if the user starts
|
||||||
|
typing. When <tab> is pressed, the "i" in ${2}'s position will be selected;
|
||||||
|
all $2 variables will default to "i" and automatically be updated if the user
|
||||||
|
starts typing.
|
||||||
|
NOTE: "$#" syntax is used only for variables, not for tab stops as in TextMate.
|
||||||
|
|
||||||
|
Variables within variables are also possible. For instance: >
|
||||||
|
|
||||||
|
snippet opt
|
||||||
|
<option value="${1:option}">${2:$1}</option>
|
||||||
|
|
||||||
|
Will, as usual, cause "option" to first be selected and update all the $1
|
||||||
|
variables if the user starts typing. Since one of these variables is inside of
|
||||||
|
${2}, this text will then be used as a placeholder for the next tab stop,
|
||||||
|
allowing the user to change it if he wishes.
|
||||||
|
|
||||||
|
To copy a value throughout a snippet without supplying default text, simply
|
||||||
|
use the "${#:}" construct without the text; e.g.: >
|
||||||
|
|
||||||
|
snippet foo
|
||||||
|
${1:}bar$1
|
||||||
|
< *snipMate-commands*
|
||||||
|
Interpolated Vim Script ~
|
||||||
|
|
||||||
|
Snippets can also contain Vim script commands that are executed (via |eval()|)
|
||||||
|
when the snippet is inserted. Commands are given inside backticks (`...`); for
|
||||||
|
TextMates's functionality, use the |system()| function. E.g.: >
|
||||||
|
|
||||||
|
snippet date
|
||||||
|
`system("date +%Y-%m-%d")`
|
||||||
|
|
||||||
|
will insert the current date, assuming you are on a Unix system. Note that you
|
||||||
|
can also (and should) use |strftime()| for this example.
|
||||||
|
|
||||||
|
Filename([{expr}] [, {defaultText}]) *snipMate-filename* *Filename()*
|
||||||
|
|
||||||
|
Since the current filename is used often in snippets, a default function
|
||||||
|
has been defined for it in snipMate.vim, appropriately called Filename().
|
||||||
|
|
||||||
|
With no arguments, the default filename without an extension is returned;
|
||||||
|
the first argument specifies what to place before or after the filename,
|
||||||
|
and the second argument supplies the default text to be used if the file
|
||||||
|
has not been named. "$1" in the first argument is replaced with the filename;
|
||||||
|
if you only want the filename to be returned, the first argument can be left
|
||||||
|
blank. Examples: >
|
||||||
|
|
||||||
|
snippet filename
|
||||||
|
`Filename()`
|
||||||
|
snippet filename_with_default
|
||||||
|
`Filename('', 'name')`
|
||||||
|
snippet filename_foo
|
||||||
|
`filename('$1_foo')`
|
||||||
|
|
||||||
|
The first example returns the filename if it the file has been named, and an
|
||||||
|
empty string if it hasn't. The second returns the filename if it's been named,
|
||||||
|
and "name" if it hasn't. The third returns the filename followed by "_foo" if
|
||||||
|
it has been named, and an empty string if it hasn't.
|
||||||
|
|
||||||
|
*multi_snip*
|
||||||
|
To specify that a snippet can have multiple matches in a *.snippets file, use
|
||||||
|
this syntax: >
|
||||||
|
|
||||||
|
snippet trigger A description of snippet #1
|
||||||
|
expand this text
|
||||||
|
snippet trigger A description of snippet #2
|
||||||
|
expand THIS text!
|
||||||
|
|
||||||
|
In this example, when "trigger<tab>" is typed, a numbered menu containing all
|
||||||
|
of the descriptions of the "trigger" will be shown; when the user presses the
|
||||||
|
corresponding number, that snippet will then be expanded.
|
||||||
|
|
||||||
|
To create a snippet with multiple matches using *.snippet files,
|
||||||
|
simply place all the snippets in a subdirectory with the trigger name:
|
||||||
|
'snippets/<filetype>/<trigger>/<name>.snippet'.
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
USAGE *snipMate-usage*
|
||||||
|
|
||||||
|
*'snippets'* *g:snippets_dir*
|
||||||
|
Snippets are by default looked for any 'snippets' directory in your
|
||||||
|
'runtimepath'. Typically, it is located at '~/.vim/snippets/' on *nix or
|
||||||
|
'$HOME\vimfiles\snippets\' on Windows. To change that location or add another
|
||||||
|
one, change the g:snippets_dir variable in your |.vimrc| to your preferred
|
||||||
|
directory, or use the |ExtractSnips()|function. This will be used by the
|
||||||
|
|globpath()| function, and so accepts the same syntax as it (e.g.,
|
||||||
|
comma-separated paths).
|
||||||
|
|
||||||
|
ExtractSnipsFile({directory}, {filetype}) *ExtractSnipsFile()* *.snippets*
|
||||||
|
|
||||||
|
ExtractSnipsFile() extracts the specified *.snippets file for the given
|
||||||
|
filetype. A .snippets file contains multiple snippet declarations for the
|
||||||
|
filetype. It is further explained above, in |snippet-syntax|.
|
||||||
|
|
||||||
|
ExtractSnips({directory}, {filetype}) *ExtractSnips()* *.snippet*
|
||||||
|
|
||||||
|
ExtractSnips() extracts *.snippet files from the specified directory and
|
||||||
|
defines them as snippets for the given filetype. The directory tree should
|
||||||
|
look like this: 'snippets/<filetype>/<trigger>.snippet'. If the snippet has
|
||||||
|
multiple matches, it should look like this:
|
||||||
|
'snippets/<filetype>/<trigger>/<name>.snippet' (see |multi_snip|).
|
||||||
|
|
||||||
|
*ResetSnippets()*
|
||||||
|
The ResetSnippets() function removes all snippets from memory. This is useful
|
||||||
|
to put at the top of a snippet setup file for if you would like to |:source|
|
||||||
|
it multiple times.
|
||||||
|
|
||||||
|
*list-snippets* *i_CTRL-R_<Tab>*
|
||||||
|
If you would like to see what snippets are available, simply type <c-r><tab>
|
||||||
|
in the current buffer to show a list via |popupmenu-completion|.
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
SETTINGS *snipMate-settings* *g:snips_author*
|
||||||
|
|
||||||
|
The g:snips_author string (similar to $TM_FULLNAME in TextMate) should be set
|
||||||
|
to your name; it can then be used in snippets to automatically add it. E.g.: >
|
||||||
|
|
||||||
|
let g:snips_author = 'Hubert Farnsworth'
|
||||||
|
snippet name
|
||||||
|
`g:snips_author`
|
||||||
|
<
|
||||||
|
*snipMate-expandtab* *snipMate-indenting*
|
||||||
|
If you would like your snippets to be expanded using spaces instead of tabs,
|
||||||
|
just enable 'expandtab' and set 'softtabstop' to your preferred amount of
|
||||||
|
spaces. If 'softtabstop' is not set, 'shiftwidth' is used instead.
|
||||||
|
|
||||||
|
*snipMate-remap*
|
||||||
|
snipMate does not come with a setting to customize the trigger key, but you
|
||||||
|
can remap it easily in the two lines it's defined in the 'after' directory
|
||||||
|
under 'plugin/snipMate.vim'. For instance, to change the trigger key
|
||||||
|
to CTRL-J, just change this: >
|
||||||
|
|
||||||
|
ino <tab> <c-r>=TriggerSnippet()<cr>
|
||||||
|
snor <tab> <esc>i<right><c-r>=TriggerSnippet()<cr>
|
||||||
|
|
||||||
|
to this: >
|
||||||
|
ino <c-j> <c-r>=TriggerSnippet()<cr>
|
||||||
|
snor <c-j> <esc>i<right><c-r>=TriggerSnippet()<cr>
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
FEATURES *snipMate-features*
|
||||||
|
|
||||||
|
snipMate.vim has the following features among others:
|
||||||
|
- The syntax of snippets is very similar to TextMate's, allowing
|
||||||
|
easy conversion.
|
||||||
|
- The position of the snippet is kept transparently (i.e. it does not use
|
||||||
|
markers/placeholders written to the buffer), which allows you to escape
|
||||||
|
out of an incomplete snippet, something particularly useful in Vim.
|
||||||
|
- Variables in snippets are updated as-you-type.
|
||||||
|
- Snippets can have multiple matches.
|
||||||
|
- Snippets can be out of order. For instance, in a do...while loop, the
|
||||||
|
condition can be added before the code.
|
||||||
|
- [New] File-based snippets are supported.
|
||||||
|
- [New] Triggers after non-word delimiters are expanded, e.g. "foo"
|
||||||
|
in "bar.foo".
|
||||||
|
- [New] <shift-tab> can now be used to jump tab stops in reverse order.
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
DISADVANTAGES *snipMate-disadvantages*
|
||||||
|
|
||||||
|
snipMate.vim currently has the following disadvantages to TextMate's snippets:
|
||||||
|
- There is no $0; the order of tab stops must be explicitly stated.
|
||||||
|
- Placeholders within placeholders are not possible. E.g.: >
|
||||||
|
|
||||||
|
'<div${1: id="${2:some_id}}">${3}</div>'
|
||||||
|
<
|
||||||
|
In TextMate this would first highlight ' id="some_id"', and if
|
||||||
|
you hit delete it would automatically skip ${2} and go to ${3}
|
||||||
|
on the next <tab>, but if you didn't delete it it would highlight
|
||||||
|
"some_id" first. You cannot do this in snipMate.vim.
|
||||||
|
- Regex cannot be performed on variables, such as "${1/.*/\U&}"
|
||||||
|
- Placeholders cannot span multiple lines.
|
||||||
|
- Activating snippets in different scopes of the same file is
|
||||||
|
not possible.
|
||||||
|
|
||||||
|
Perhaps some of these features will be added in a later release.
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
CONTACT *snipMate-contact* *snipMate-author*
|
||||||
|
|
||||||
|
To contact the author (Michael Sanders), please email:
|
||||||
|
msanders42+snipmate <at> gmail <dot> com
|
||||||
|
|
||||||
|
I greatly appreciate any suggestions or improvements offered for the script.
|
||||||
|
|
||||||
|
==============================================================================
|
||||||
|
|
||||||
|
vim:tw=78:ts=8:ft=help:norl:
|
10
vim/.vim/ftplugin/html_snip_helper.vim
Normal file
10
vim/.vim/ftplugin/html_snip_helper.vim
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
" Helper function for (x)html snippets
|
||||||
|
if exists('s:did_snip_helper') || &cp || !exists('loaded_snips')
|
||||||
|
finish
|
||||||
|
endif
|
||||||
|
let s:did_snip_helper = 1
|
||||||
|
|
||||||
|
" Automatically closes tag if in xhtml
|
||||||
|
fun! Close()
|
||||||
|
return stridx(&ft, 'xhtml') == -1 ? '' : ' /'
|
||||||
|
endf
|
247
vim/.vim/plugin/snipMate.vim
Normal file
247
vim/.vim/plugin/snipMate.vim
Normal file
|
@ -0,0 +1,247 @@
|
||||||
|
" File: snipMate.vim
|
||||||
|
" Author: Michael Sanders
|
||||||
|
" Last Updated: July 13, 2009
|
||||||
|
" Version: 0.83
|
||||||
|
" Description: snipMate.vim implements some of TextMate's snippets features in
|
||||||
|
" Vim. A snippet is a piece of often-typed text that you can
|
||||||
|
" insert into your document using a trigger word followed by a "<tab>".
|
||||||
|
"
|
||||||
|
" For more help see snipMate.txt; you can do this by using:
|
||||||
|
" :helptags ~/.vim/doc
|
||||||
|
" :h snipMate.txt
|
||||||
|
|
||||||
|
if exists('loaded_snips') || &cp || version < 700
|
||||||
|
finish
|
||||||
|
endif
|
||||||
|
let loaded_snips = 1
|
||||||
|
if !exists('snips_author') | let snips_author = 'Me' | endif
|
||||||
|
|
||||||
|
au BufRead,BufNewFile *.snippets\= set ft=snippet
|
||||||
|
au FileType snippet setl noet fdm=indent
|
||||||
|
|
||||||
|
let s:snippets = {} | let s:multi_snips = {}
|
||||||
|
|
||||||
|
if !exists('snippets_dir')
|
||||||
|
let snippets_dir = substitute(globpath(&rtp, 'snippets/'), "\n", ',', 'g')
|
||||||
|
endif
|
||||||
|
|
||||||
|
fun! MakeSnip(scope, trigger, content, ...)
|
||||||
|
let multisnip = a:0 && a:1 != ''
|
||||||
|
let var = multisnip ? 's:multi_snips' : 's:snippets'
|
||||||
|
if !has_key({var}, a:scope) | let {var}[a:scope] = {} | endif
|
||||||
|
if !has_key({var}[a:scope], a:trigger)
|
||||||
|
let {var}[a:scope][a:trigger] = multisnip ? [[a:1, a:content]] : a:content
|
||||||
|
elseif multisnip | let {var}[a:scope][a:trigger] += [[a:1, a:content]]
|
||||||
|
else
|
||||||
|
echom 'Warning in snipMate.vim: Snippet '.a:trigger.' is already defined.'
|
||||||
|
\ .' See :h multi_snip for help on snippets with multiple matches.'
|
||||||
|
endif
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun! ExtractSnips(dir, ft)
|
||||||
|
for path in split(globpath(a:dir, '*'), "\n")
|
||||||
|
if isdirectory(path)
|
||||||
|
let pathname = fnamemodify(path, ':t')
|
||||||
|
for snipFile in split(globpath(path, '*.snippet'), "\n")
|
||||||
|
call s:ProcessFile(snipFile, a:ft, pathname)
|
||||||
|
endfor
|
||||||
|
elseif fnamemodify(path, ':e') == 'snippet'
|
||||||
|
call s:ProcessFile(path, a:ft)
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
endf
|
||||||
|
|
||||||
|
" Processes a single-snippet file; optionally add the name of the parent
|
||||||
|
" directory for a snippet with multiple matches.
|
||||||
|
fun s:ProcessFile(file, ft, ...)
|
||||||
|
let keyword = fnamemodify(a:file, ':t:r')
|
||||||
|
if keyword == '' | return | endif
|
||||||
|
try
|
||||||
|
let text = join(readfile(a:file), "\n")
|
||||||
|
catch /E484/
|
||||||
|
echom "Error in snipMate.vim: couldn't read file: ".a:file
|
||||||
|
endtry
|
||||||
|
return a:0 ? MakeSnip(a:ft, a:1, text, keyword)
|
||||||
|
\ : MakeSnip(a:ft, keyword, text)
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun! ExtractSnipsFile(file, ft)
|
||||||
|
if !filereadable(a:file) | return | endif
|
||||||
|
let text = readfile(a:file)
|
||||||
|
let inSnip = 0
|
||||||
|
for line in text + ["\n"]
|
||||||
|
if inSnip && (line[0] == "\t" || line == '')
|
||||||
|
let content .= strpart(line, 1)."\n"
|
||||||
|
continue
|
||||||
|
elseif inSnip
|
||||||
|
call MakeSnip(a:ft, trigger, content[:-2], name)
|
||||||
|
let inSnip = 0
|
||||||
|
endif
|
||||||
|
|
||||||
|
if line[:6] == 'snippet'
|
||||||
|
let inSnip = 1
|
||||||
|
let trigger = strpart(line, 8)
|
||||||
|
let name = ''
|
||||||
|
let space = stridx(trigger, ' ') + 1
|
||||||
|
if space " Process multi snip
|
||||||
|
let name = strpart(trigger, space)
|
||||||
|
let trigger = strpart(trigger, 0, space - 1)
|
||||||
|
endif
|
||||||
|
let content = ''
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun! ResetSnippets()
|
||||||
|
let s:snippets = {} | let s:multi_snips = {} | let g:did_ft = {}
|
||||||
|
endf
|
||||||
|
|
||||||
|
let g:did_ft = {}
|
||||||
|
fun! GetSnippets(dir, filetypes)
|
||||||
|
for ft in split(a:filetypes, '\.')
|
||||||
|
if has_key(g:did_ft, ft) | continue | endif
|
||||||
|
call s:DefineSnips(a:dir, ft, ft)
|
||||||
|
if ft == 'objc' || ft == 'cpp' || ft == 'cs'
|
||||||
|
call s:DefineSnips(a:dir, 'c', ft)
|
||||||
|
elseif ft == 'xhtml'
|
||||||
|
call s:DefineSnips(a:dir, 'html', 'xhtml')
|
||||||
|
endif
|
||||||
|
let g:did_ft[ft] = 1
|
||||||
|
endfor
|
||||||
|
endf
|
||||||
|
|
||||||
|
" Define "aliasft" snippets for the filetype "realft".
|
||||||
|
fun s:DefineSnips(dir, aliasft, realft)
|
||||||
|
for path in split(globpath(a:dir, a:aliasft.'/')."\n".
|
||||||
|
\ globpath(a:dir, a:aliasft.'-*/'), "\n")
|
||||||
|
call ExtractSnips(path, a:realft)
|
||||||
|
endfor
|
||||||
|
for path in split(globpath(a:dir, a:aliasft.'.snippets')."\n".
|
||||||
|
\ globpath(a:dir, a:aliasft.'-*.snippets'), "\n")
|
||||||
|
call ExtractSnipsFile(path, a:realft)
|
||||||
|
endfor
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun! TriggerSnippet()
|
||||||
|
if exists('g:SuperTabMappingForward')
|
||||||
|
if g:SuperTabMappingForward == "<tab>"
|
||||||
|
let SuperTabKey = "\<c-n>"
|
||||||
|
elseif g:SuperTabMappingBackward == "<tab>"
|
||||||
|
let SuperTabKey = "\<c-p>"
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
if pumvisible() " Update snippet if completion is used, or deal with supertab
|
||||||
|
if exists('SuperTabKey')
|
||||||
|
call feedkeys(SuperTabKey) | return ''
|
||||||
|
endif
|
||||||
|
call feedkeys("\<esc>a", 'n') " Close completion menu
|
||||||
|
call feedkeys("\<tab>") | return ''
|
||||||
|
endif
|
||||||
|
|
||||||
|
if exists('g:snipPos') | return snipMate#jumpTabStop(0) | endif
|
||||||
|
|
||||||
|
let word = matchstr(getline('.'), '\S\+\%'.col('.').'c')
|
||||||
|
for scope in [bufnr('%')] + split(&ft, '\.') + ['_']
|
||||||
|
let [trigger, snippet] = s:GetSnippet(word, scope)
|
||||||
|
" If word is a trigger for a snippet, delete the trigger & expand
|
||||||
|
" the snippet.
|
||||||
|
if snippet != ''
|
||||||
|
let col = col('.') - len(trigger)
|
||||||
|
sil exe 's/\V'.escape(trigger, '/.').'\%#//'
|
||||||
|
return snipMate#expandSnip(snippet, col)
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
|
||||||
|
if exists('SuperTabKey')
|
||||||
|
call feedkeys(SuperTabKey)
|
||||||
|
return ''
|
||||||
|
endif
|
||||||
|
return "\<tab>"
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun! BackwardsSnippet()
|
||||||
|
if exists('g:snipPos') | return snipMate#jumpTabStop(1) | endif
|
||||||
|
|
||||||
|
if exists('g:SuperTabMappingForward')
|
||||||
|
if g:SuperTabMappingBackward == "<s-tab>"
|
||||||
|
let SuperTabKey = "\<c-p>"
|
||||||
|
elseif g:SuperTabMappingForward == "<s-tab>"
|
||||||
|
let SuperTabKey = "\<c-n>"
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
if exists('SuperTabKey')
|
||||||
|
call feedkeys(SuperTabKey)
|
||||||
|
return ''
|
||||||
|
endif
|
||||||
|
return "\<s-tab>"
|
||||||
|
endf
|
||||||
|
|
||||||
|
" Check if word under cursor is snippet trigger; if it isn't, try checking if
|
||||||
|
" the text after non-word characters is (e.g. check for "foo" in "bar.foo")
|
||||||
|
fun s:GetSnippet(word, scope)
|
||||||
|
let word = a:word | let snippet = ''
|
||||||
|
while snippet == ''
|
||||||
|
if exists('s:snippets["'.a:scope.'"]["'.escape(word, '\"').'"]')
|
||||||
|
let snippet = s:snippets[a:scope][word]
|
||||||
|
elseif exists('s:multi_snips["'.a:scope.'"]["'.escape(word, '\"').'"]')
|
||||||
|
let snippet = s:ChooseSnippet(a:scope, word)
|
||||||
|
if snippet == '' | break | endif
|
||||||
|
else
|
||||||
|
if match(word, '\W') == -1 | break | endif
|
||||||
|
let word = substitute(word, '.\{-}\W', '', '')
|
||||||
|
endif
|
||||||
|
endw
|
||||||
|
if word == '' && a:word != '.' && stridx(a:word, '.') != -1
|
||||||
|
let [word, snippet] = s:GetSnippet('.', a:scope)
|
||||||
|
endif
|
||||||
|
return [word, snippet]
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun s:ChooseSnippet(scope, trigger)
|
||||||
|
let snippet = []
|
||||||
|
let i = 1
|
||||||
|
for snip in s:multi_snips[a:scope][a:trigger]
|
||||||
|
let snippet += [i.'. '.snip[0]]
|
||||||
|
let i += 1
|
||||||
|
endfor
|
||||||
|
if i == 2 | return s:multi_snips[a:scope][a:trigger][0][1] | endif
|
||||||
|
let num = inputlist(snippet) - 1
|
||||||
|
return num == -1 ? '' : s:multi_snips[a:scope][a:trigger][num][1]
|
||||||
|
endf
|
||||||
|
|
||||||
|
fun! ShowAvailableSnips()
|
||||||
|
let line = getline('.')
|
||||||
|
let col = col('.')
|
||||||
|
let word = matchstr(getline('.'), '\S\+\%'.col.'c')
|
||||||
|
let words = [word]
|
||||||
|
if stridx(word, '.')
|
||||||
|
let words += split(word, '\.', 1)
|
||||||
|
endif
|
||||||
|
let matchlen = 0
|
||||||
|
let matches = []
|
||||||
|
for scope in [bufnr('%')] + split(&ft, '\.') + ['_']
|
||||||
|
let triggers = has_key(s:snippets, scope) ? keys(s:snippets[scope]) : []
|
||||||
|
if has_key(s:multi_snips, scope)
|
||||||
|
let triggers += keys(s:multi_snips[scope])
|
||||||
|
endif
|
||||||
|
for trigger in triggers
|
||||||
|
for word in words
|
||||||
|
if word == ''
|
||||||
|
let matches += [trigger] " Show all matches if word is empty
|
||||||
|
elseif trigger =~ '^'.word
|
||||||
|
let matches += [trigger]
|
||||||
|
let len = len(word)
|
||||||
|
if len > matchlen | let matchlen = len | endif
|
||||||
|
endif
|
||||||
|
endfor
|
||||||
|
endfor
|
||||||
|
endfor
|
||||||
|
|
||||||
|
" This is to avoid a bug with Vim when using complete(col - matchlen, matches)
|
||||||
|
" (Issue#46 on the Google Code snipMate issue tracker).
|
||||||
|
call setline(line('.'), substitute(line, repeat('.', matchlen).'\%'.col.'c', '', ''))
|
||||||
|
call complete(col, matches)
|
||||||
|
return ''
|
||||||
|
endf
|
||||||
|
" vim:noet:sw=4:ts=4:ft=vim
|
62
vim/.vim/plugin/templates.vim
Normal file
62
vim/.vim/plugin/templates.vim
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
" Vim global plugin for providing templates for new files
|
||||||
|
" Licence: The MIT License (MIT)
|
||||||
|
" Commit: d3682469538340454d1e11d143735683cc75a13a
|
||||||
|
" {{{ Copyright (c) 2015 Aristotle Pagaltzis <pagaltzis@gmx.de>
|
||||||
|
"
|
||||||
|
" Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
" of this software and associated documentation files (the "Software"), to deal
|
||||||
|
" in the Software without restriction, including without limitation the rights
|
||||||
|
" to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
" copies of the Software, and to permit persons to whom the Software is
|
||||||
|
" furnished to do so, subject to the following conditions:
|
||||||
|
"
|
||||||
|
" The above copyright notice and this permission notice shall be included in
|
||||||
|
" all copies or substantial portions of the Software.
|
||||||
|
"
|
||||||
|
" THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
" IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
" FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
" AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
" LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
" OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
" THE SOFTWARE.
|
||||||
|
" }}}
|
||||||
|
|
||||||
|
if exists( 'g:loaded_template' ) | finish | endif
|
||||||
|
let g:loaded_template = 1
|
||||||
|
|
||||||
|
augroup template
|
||||||
|
autocmd!
|
||||||
|
autocmd FileType * if line2byte( line( '$' ) + 1 ) == -1 | call s:loadtemplate( &filetype ) | endif
|
||||||
|
augroup END
|
||||||
|
|
||||||
|
function! s:globpathlist( path, ... )
|
||||||
|
let i = 1
|
||||||
|
let result = a:path
|
||||||
|
while i <= a:0
|
||||||
|
let result = substitute( escape( globpath( result, a:{i} ), ' ,\' ), "\n", ',', 'g' )
|
||||||
|
if strlen( result ) == 0 | return '' | endif
|
||||||
|
let i = i + 1
|
||||||
|
endwhile
|
||||||
|
return result
|
||||||
|
endfunction
|
||||||
|
|
||||||
|
function! s:loadtemplate( filetype )
|
||||||
|
let templatefile = matchstr( s:globpathlist( &runtimepath, 'templates', a:filetype ), '\(\\.\|[^,]\)*', 0 )
|
||||||
|
if strlen( templatefile ) == 0 | return | endif
|
||||||
|
silent execute 1 'read' templatefile
|
||||||
|
1 delete _
|
||||||
|
if search( 'cursor:', 'W' )
|
||||||
|
let cursorline = strpart( getline( '.' ), col( '.' ) - 1 )
|
||||||
|
let y = matchstr( cursorline, '^cursor:\s*\zs\d\+\ze' )
|
||||||
|
let x = matchstr( cursorline, '^cursor:\s*\d\+\s\+\zs\d\+\ze' )
|
||||||
|
let d = matchstr( cursorline, '^cursor:\s*\d\+\s\+\(\d\+\s\+\)\?\zsdel\>\ze' )
|
||||||
|
if ! strlen( x ) | let x = 0 | endif
|
||||||
|
if ! strlen( y ) | let y = 0 | endif
|
||||||
|
if d == 'del' | delete _ | endif
|
||||||
|
call cursor( y, x )
|
||||||
|
endif
|
||||||
|
set nomodified
|
||||||
|
endfunction
|
||||||
|
|
||||||
|
command -nargs=1 New new | set ft=<args>
|
19
vim/.vim/syntax/snippet.vim
Normal file
19
vim/.vim/syntax/snippet.vim
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
" Syntax highlighting for snippet files (used for snipMate.vim)
|
||||||
|
" Hopefully this should make snippets a bit nicer to write!
|
||||||
|
syn match snipComment '^#.*'
|
||||||
|
syn match placeHolder '\${\d\+\(:.\{-}\)\=}' contains=snipCommand
|
||||||
|
syn match tabStop '\$\d\+'
|
||||||
|
syn match snipCommand '`.\{-}`'
|
||||||
|
syn match snippet '^snippet.*' transparent contains=multiSnipText,snipKeyword
|
||||||
|
syn match multiSnipText '\S\+ \zs.*' contained
|
||||||
|
syn match snipKeyword '^snippet'me=s+8 contained
|
||||||
|
syn match snipError "^[^#s\t].*$"
|
||||||
|
|
||||||
|
hi link snipComment Comment
|
||||||
|
hi link multiSnipText String
|
||||||
|
hi link snipKeyword Keyword
|
||||||
|
hi link snipComment Comment
|
||||||
|
hi link placeHolder Special
|
||||||
|
hi link tabStop Special
|
||||||
|
hi link snipCommand String
|
||||||
|
hi link snipError Error
|
Loading…
Reference in a new issue