Files
Ingo Karkat 6df3caee51 ingo#range#lines#Get(): If the range is a backwards-looking ?{pattern}?, we need to attempt the match on any line with :global/^/...
Else, the border behavior is inconsistent: ranges that extend the passed range at the bottom are (partially) included, but ranges that extend at the front would not be.
2017-05-26 17:53:09 +02:00

135 lines
5.3 KiB
VimL

" ingo/range/Lines.vim: Functions for retrieving line numbers of ranges.
"
" DEPENDENCIES:
" - ingo/cmdsargs/pattern.vim autoload script
" - ingo/range.vim autoload script
"
" Copyright: (C) 2014-2016 Ingo Karkat
" The VIM LICENSE applies to this script; see ':help copyright'.
"
" Maintainer: Ingo Karkat <ingo@karkat.de>
"
" REVISION DATE REMARKS
" 1.029.005 23-Dec-2016 ingo#range#lines#Get(): If the range is a
" backwards-looking ?{pattern}?, we need to
" attempt the match on any line with :global/^/...
" Else, the border behavior is inconsistent:
" ranges that extend the passed range at the
" bottom are (partially) included, but ranges that
" extend at the front would not be.
" 1.029.004 07-Dec-2016 ingo#range#lines#Get(): A single
" (a:isGetAllRanges = 0) /.../ range already
" clobbers the last search pattern. Save and
" restore if necessary, and base
" didClobberSearchHistory on that check.
" ingo#range#lines#Get(): Drop the ^ anchor for
" the range check to also detect /.../ as the
" end of the range.
" 1.023.003 26-Dec-2014 ENH: Add a:isGetAllRanges optional argument to
" ingo#range#lines#Get().
" 1.022.002 23-Sep-2014 ingo#range#lines#Get() needs to consider and
" temporarily disable closed folds when resolving
" /{pattern}/ ranges.
" 1.020.001 10-Jun-2014 file creation from
" autoload/PatternsOnText/Ranges.vim
function! s:RecordLine( records, startLnum, endLnum )
let l:lnum = line('.')
if l:lnum < a:startLnum || l:lnum > a:endLnum
let s:didRecord = 0
return
endif
let a:records[l:lnum] = 1
let s:didRecord = 1
endfunction
function! s:RecordLines( records, startLines, endLines, startLnum, endLnum ) range
execute printf('%d,%dcall s:RecordLine(a:records, a:startLnum, a:endLnum)', a:firstline, a:lastline)
if s:didRecord
call add(a:startLines, max([a:firstline, a:startLnum]))
call add(a:endLines, min([a:lastline, a:endLnum]))
endif
endfunction
function! ingo#range#lines#Get( startLnum, endLnum, range, ... )
"******************************************************************************
"* PURPOSE:
" Determine the line numbers and start and end lines of a:range that fall
" inside a:startLnum and a:endLnum. Closed folds do not affect the recorded
" lines; only the actually matched lines are considered.
"* ASSUMPTIONS / PRECONDITIONS:
" None.
"* EFFECTS / POSTCONDITIONS:
" Changes the cursor position in the buffer (to the beginning of the last line
" within the range).
"* INPUTS:
" a:startLnum First line number to be considered.
" a:endLnum Last line number to be considered.
" a:range Range in any format supported by Vim, e.g. 'a,'b or
" /^fun/,/^endfun/
" a:isGetAllRanges Optional flag whether (for pattern ranges like /.../),
" all (vs. only the next matching) ranges are determined.
" Defaults to 1; pass 0 to only get the next one.
"* RETURN VALUES:
" [recordedLnums, startLnums, endLnums, didClobberSearchHistory]
" recordedLnums Dictionary with all line numbers that fall into the range(s)
" as keys.
" startLnums List of line numbers where a range starts. Can contain
" multiple elements if a /pattern/ range is used.
" endLnums List of line numbers where a range ends.
" didClobberSearchHistory Flag whether a command was used that has added a
" temporary pattern to the search history. If true,
" call histdel('search', -1) at the end of the client
" function once.
"******************************************************************************
let l:isGetAllRanges = (! a:0 || a:1)
let [l:startLnum, l:endLnum] = [ingo#range#NetStart(a:startLnum), ingo#range#NetEnd(a:endLnum)]
let l:recordedLines = {}
let l:startLines = []
let l:endLines = []
let l:save_search = @/
let l:didClobberSearchHistory = 0
if l:isGetAllRanges && a:range =~# '[/?]'
" For patterns, we need :global to find _all_ (not just the first)
" matching ranges. For that, folds must be open / disabled. And because
" of that, the actual ranges must be determined first.
let l:save_foldenable = &l:foldenable
setlocal nofoldenable
let l:searchRange = a:range
if ingo#cmdargs#pattern#RawParse(a:range, [''], '\s*[,;]\s*\S.*')[0] ==# '?'
" If this is a simple /{pattern}/, we can just match that with
" :global. But for actual ranges, these should extend both upwards
" (?foo?,/bar/) as well as downwards (/foo/,/bar/). To handle the
" former, we must make :global attempt a match at any line.
let l:searchRange = '/^/' . a:range
endif
try
execute printf('silent! %d,%dglobal %s call <SID>RecordLines(l:recordedLines, l:startLines, l:endLines, %d, %d)',
\ l:startLnum, l:endLnum,
\ l:searchRange,
\ l:startLnum, l:endLnum
\)
finally
let &l:foldenable = l:save_foldenable
endtry
else
" For line number, marks, etc., we can just record them (limited to
" those that fall into the command's range).
execute printf('silent! %s call <SID>RecordLines(l:recordedLines, l:startLines, l:endLines, %d, %d)',
\ a:range,
\ l:startLnum, l:endLnum
\)
endif
if @/ !=# l:save_search
let @/ = l:save_search
let l:didClobberSearchHistory = 1
endif
return [l:recordedLines, l:startLines, l:endLines, l:didClobberSearchHistory]
endfunction
" vim: set ts=8 sts=4 sw=4 noexpandtab ff=unix fdm=syntax :