Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How can I script vim to run perltidy on a buffer?

Tags:

vim

perl

At my current job, we have coding-style standards that are different from the ones I normally follow. Fortunately, we have a canned RC file for perltidy that I can apply to reformat files before I submit them to our review process.

I have code for emacs that I use to run a command over a buffer and replace the buffer with the output, which I have adapted for this. But I sometimes alternate between emacs and vim, and would like to have the same capabilities there. I'm sure that this or something similar is simple and had been done and re-done many times over. But I've not had much luck finding any examples of vim-script that seem to do what I need. Which is, in essence, to be able to hit a key combo (like Ctrl-F6, what I use in emacs) and have the buffer be reformatted in-place by perltidy. While I'm a comfortable vim-user, I'm completely clueless at writing this sort of thing for vim.

like image 505
rjray Avatar asked Feb 26 '10 23:02

rjray


4 Answers

After trying @hobbs answer I noticed that when filtering the entire buffer through perltidy the cursor returned to byte 1, and I had to make a mental note of the original line number so I could go back after :Tidy completed.

So building on @hobbs' and @Ignacio's answers, I added the following to my .vimrc:

"define :Tidy command to run perltidy on visual selection || entire buffer"
command -range=% -nargs=* Tidy <line1>,<line2>!perltidy

"run :Tidy on entire buffer and return cursor to (approximate) original position"
fun DoTidy()
    let l = line(".")
    let c = col(".")
    :Tidy
    call cursor(l, c)
endfun

"shortcut for normal mode to run on entire buffer then return to current line"
au Filetype perl nmap <F2> :call DoTidy()<CR>

"shortcut for visual mode to run on the current visual selection"
au Filetype perl vmap <F2> :Tidy<CR>

(closing " added to comments for SO syntax highlighting purposes (not required, but valid vim syntax))

DoTidy() will return the cursor to its original position plus or minus at most X bytes, where X is the number of bytes added/removed by perltidy relative to the original cursor position. But this is fairly trivial as long as you keep things tidy :).

[Vim version: 7.2]

EDIT: Updated DoTidy() to incorporate @mikew's comment for readability and for compatibility with Vim 7.0

like image 148
MisterEd Avatar answered Nov 02 '22 23:11

MisterEd


My tidy command:

command -range=% -nargs=* Tidy <line1>,<line2>!
  \perltidy (your default options go here) <args>

If you use a visual selection or provide a range then it will tidy the selected range, otherwise it will use the whole file. You can put a set of default options (if you have any) at the point where I wrote (your default options go here), but any arguments that you provide to :Tidy will be appended to the perltidy commandline, overriding your defaults. (If you use a .perltidyrc you might not have default args -- that's fine -- but then again you might want to have a default like --profile=vim that sets up defaults only for when you're working in vim. Whatever works.)

like image 39
hobbs Avatar answered Nov 02 '22 23:11

hobbs


The command to filter the entire buffer through an external program is:

:%!command

Put the following in ~/.vimrc to bind it to Ctrl-F6 in normal mode:

:nmap <C-F6> :%!command<CR>

For added fun:

:au Filetype perl nmap <C-F6> :%!command<CR>

This will only map the filter if editing a Perl file.

like image 9
Ignacio Vazquez-Abrams Avatar answered Nov 02 '22 22:11

Ignacio Vazquez-Abrams


Taking hobbs' answer a step further, you can map that command to a shortcut key:

command -range=% -nargs=* Tidy <line1>,<line2>!perltidy -q
noremap <C-F6> :Tidy<CR>

And another step further: Only map the command when you're in a Perl buffer (since you probably wouldn't want to run perltidy on any other language):

autocmd BufRead,BufNewFile *.pl,*.plx,*.pm command! -range=% -nargs=* Tidy <line1>,<line2>!perltidy -q
autocmd BufRead,BufNewFile *.pl,*.plx,*.pm noremap <C-F6> :Tidy<CR>

Now you can press Ctrl-F6 without an active selection to format the whole file, or with an active selection to format just that section.

like image 4
mkerley Avatar answered Nov 02 '22 21:11

mkerley