Files

149 lines
5.8 KiB
VimL

" ingo/text/frompattern.vim: Functions to get matches from the current buffer.
"
" DEPENDENCIES:
"
" Copyright: (C) 2013-2015 Ingo Karkat
" The VIM LICENSE applies to this script; see ':help copyright'.
"
" Maintainer: Ingo Karkat <ingo@karkat.de>
"
" REVISION DATE REMARKS
" 1.025.004 06-May-2015 Add ingo#text#frompattern#GetAroundHere(),
" inspired by
" http://stackoverflow.com/questions/30073662/vim-copy-match-with-cursor-position-atom-to-local-variable
" 1.024.003 17-Apr-2015 ingo#text#frompattern#GetHere(): Do not move the
" cursor (to the end of the matched pattern); this
" is unexpected and can be easily avoided.
" 1.014.002 27-Sep-2013 Add ingo#text#frompattern#GetHere().
" 1.012.001 03-Sep-2013 file creation
function! ingo#text#frompattern#GetHere( pattern, ... )
"******************************************************************************
"* PURPOSE:
" Extract the match of a:pattern starting from the current cursor position.
"* ASSUMPTIONS / PRECONDITIONS:
" None.
"* EFFECTS / POSTCONDITIONS:
" None.
"* INPUTS:
" a:pattern Regular expression to search. 'ignorecase', 'smartcase' and
" 'magic' applies. When empty, the last search pattern |"/| is
" used.
" a:lastLine End line number to search for the start of the pattern.
" Optional; defaults to the current line.
"* RETURN VALUES:
" Matched text, or empty string.
"******************************************************************************
let l:startPos = getpos('.')[1:2]
let l:endPos = searchpos(a:pattern, 'cenW', (a:0 ? a:1 : line('.')))
if l:endPos == [0, 0]
return ''
endif
return ingo#text#Get(l:startPos, l:endPos)
endfunction
function! ingo#text#frompattern#GetAroundHere( pattern, ... )
"******************************************************************************
"* PURPOSE:
" Extract the match of a:pattern starting the match from the current cursor
" position, but (unlike ingo#text#frompattern#GetHere()), also include matched
" characters _before_ the current position.
"* ASSUMPTIONS / PRECONDITIONS:
" None.
"* EFFECTS / POSTCONDITIONS:
" None.
"* INPUTS:
" a:pattern Regular expression to search. 'ignorecase', 'smartcase' and
" 'magic' applies. When empty, the last search pattern |"/| is
" used.
" a:lastLine End line number to search for the start of the pattern.
" Optional; defaults to the current line.
" a:firstLine First line number to search for the start of the pattern.
" Optional; defaults to the current line.
"* RETURN VALUES:
" Matched text, or empty string.
"******************************************************************************
let l:startPos = searchpos(a:pattern, 'bcnW', (a:0 >= 2 ? a:2 : line('.')))
if l:startPos == [0, 0]
return ''
endif
let l:endPos = searchpos(a:pattern, 'cenW', (a:0 ? a:1 : line('.')))
if l:endPos == [0, 0]
return ''
endif
return ingo#text#Get(l:startPos, l:endPos)
endfunction
function! s:UniqueAdd( list, expr )
if index(a:list, a:expr) == -1
call add(a:list, a:expr)
endif
endfunction
function! ingo#text#frompattern#Get( firstLine, lastLine, pattern, replacement, isOnlyFirstMatch, isUnique )
"******************************************************************************
"* PURPOSE:
" Extract all non-overlapping matches of a:pattern in the a:firstLine,
" a:lastLine range and return them (optionally a submatch / replacement, or
" only first or unique matches) as a List.
"* SEE ALSO:
" - ingo#str#frompattern#Get() extracts matches from a string / List of lines
" instead of the current buffer.
"* ASSUMPTIONS / PRECONDITIONS:
" None.
"* EFFECTS / POSTCONDITIONS:
" None.
"* INPUTS:
" a:firstLine Start line number to search.
" a:lastLine End line number to search.
" a:pattern Regular expression to search. 'ignorecase', 'smartcase' and
" 'magic' applies. When empty, the last search pattern |"/| is
" used.
" a:replacement Optional replacement substitute(). When not empty, each
" match is processed through substitute() with a:pattern.
" When a:pattern cannot be used (e.g. because it references
" cursor or buffer position via special atoms like \%# and
" therefore doesn't work standalone), you can also pass a
" [replPattern, replacement] tuple, which will then be
" globally applied to the match.
" a:isOnlyFirstMatch Flag whether to include only the first match in every
" line.
" a:isUnique Flag whether duplicate matches are omitted from the
" result. When set, the result will consist of unique
" matches.
"* RETURN VALUES:
" List of (optionally replaced) matches, or empty List when no matches.
"******************************************************************************
let l:save_view = winsaveview()
let l:matches = []
call cursor(a:firstLine, 1)
let l:isFirst = 1
while 1
let l:startPos = searchpos(a:pattern, (l:isFirst ? 'c' : '') . 'W', a:lastLine)
let l:isFirst = 0
if l:startPos == [0, 0] | break | endif
let l:endPos = searchpos(a:pattern, 'ceW', a:lastLine)
if l:endPos == [0, 0] | break | endif
let l:match = ingo#text#Get(l:startPos, l:endPos)
if ! empty(a:replacement)
if type(a:replacement) == type([])
let l:match = substitute(l:match, a:replacement[0], a:replacement[1], 'g')
else
let l:match = substitute(l:match, (empty(a:pattern) ? @/ : a:pattern), a:replacement, '')
endif
endif
if a:isUnique
call s:UniqueAdd(l:matches, l:match)
else
call add(l:matches, l:match)
endif
"****D echomsg '****' string(l:startPos) string(l:endPos) string(l:match)
if a:isOnlyFirstMatch
normal! $
endif
endwhile
call winrestview(l:save_view)
return l:matches
endfunction
" vim: set ts=8 sts=4 sw=4 noexpandtab ff=unix fdm=syntax :