I'm trying to match the contents that belong between a certain (
and its matching )
as found by vim when using the motion %
.
More specifically, I'm looking for a regex that looks like this hypothetical /someKeyword (\{pair}\(.*\))\{pair}/
, if there were such modifiers as \{pair}
that when applied to two exactly two characters in a regex, makes the second one only match if it's the matching bracket to the first one (%
-wise).
The pattern I'm looking for should match the inner contents of the first bracket following someKeyword
(n.b. the code that it should work on is always correctly bracketed), as in the examples:
For someKeyword ("aaa")
the submatch will match "aaa"
.
Likewise someKeyword ("aaa)")
will match "aaa)"
and someKeyword(("double-nested stuff"))
will match ("double-nested stuff")
But also in cases like:
(
someKeyword("xyz"))
where it should match "xyz"
.
Is there any way to make use of vim's matching bracket functionality in regexes? And if not, what other solution might work to accomplish this?
Edit 1: the matched contents may span several lines.
In normal mode, press / to start a search, then type the pattern ( \<i\> ), then press Enter. If you have an example of the word you want to find on screen, you do not need to enter a search pattern. Simply move the cursor anywhere within the word, then press * to search for the next occurrence of that whole word.
To match a character having special meaning in regex, you need to use a escape sequence prefix with a backslash ( \ ). E.g., \. matches "." ; regex \+ matches "+" ; and regex \( matches "(" . You also need to use regex \\ to match "\" (back-slash).
Vim has several regex modes, one of which is very magic that's very similar to traditional regex. Just put \v in the front and you won't have to escape as much.
grep is one of the most useful and powerful commands in Linux for text processing. grep searches one or more input files for lines that match a regular expression and writes each matching line to standard output.
This is not possible with vim regular expressions (as language that allows such nested constructs is not regular), but is possible with 'regular' expressions provided by perl (as well as by other languages I do not know enough to be sure) and perl can be used from inside vim. I don't like vim-perl bindings (because it is very limited), but if you know all cases that should work, then you could use recursion feature of perl regular expressions (requires newer perl, I have 5.12*):
perl VIM::Msg($+{"outer"}) if $curbuf->Get(3) =~ /someKeyword\((?'outer'(?'inner'"(?:\\.|[^"])*"|'(?:[^']|'')*'|[^()]*|\((?P>inner)*\))*)\)/
Note that if can avoid such regular expressions, you should do it (because you depend on re compiler too much), so I suggest to use vim motions directly:
let s:reply=""
function! SetReplyToKeywordArgs(...)
let [sline, scol]=getpos("'[")[1:2]
let [eline, ecol]=getpos("']")[1:2]
let lchar=len(matchstr(getline(eline), '\%'.ecol.'c.'))
if lchar>1
let ecol+=lchar-1
endif
let text=[]
let ellcol=col([eline, '$'])
let slinestr=getline(sline)
if sline==eline
if ecol>=ellcol
call extend(text, [slinestr[(scol-1):], ""])
else
call add(text, slinestr[(scol-1):(ecol-1)])
endif
else
call add(text, slinestr[(scol-1):])
let elinestr=getline(eline)
if (eline-sline)>1
call extend(text, getline(sline+1, eline-1))
endif
if ecol<ellcol
call add(text, elinestr[:(ecol-1)])
else
call extend(text, [elinestr, ""])
endif
endif
let s:reply=join(text, "\n")
endfunction
function! GetKeywordArgs()
let winview=winsaveview()
keepjumps call search('someKeyword', 'e')
setlocal operatorfunc=SetReplyToKeywordArgs
keepjumps normal! f(g@i(
call winrestview(winview)
return s:reply
endfunction
You can use something like
let savedureg=@"
let saved0reg=@0
keepjumps normal! f(yi(
let s:reply=@"
let @"=savedureg
let @0=saved0reg
instead of operatorfunc to save and restore registers, but the above code leaves all registers and marks untouched, what I can't guarantee with saved* stuff. It also guarantees that if you remove join()
around text
, you will save information about the location of NULLs (if you care about them, of course). It is not possible with registers variant.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With