2015-05-18 19:27:00 +02:00
" Function for finding the formatters for this filetype
2015-05-23 10:02:05 +02:00
" Result is stored in b:formatters
2015-05-18 19:27:00 +02:00
function ! s :find_formatters ( ...)
" Detect verbosity
2015-05-23 10:02:05 +02:00
let verbose = &verbose | | exists ( "g:autoformat_verbosemode" )
2015-05-18 19:27:00 +02:00
" Extract filetype to be used
2015-05-31 21:56:00 +02:00
let ftype = a :0 ? a :1 : &filetype
2015-04-09 10:59:54 +02:00
" Support composite filetypes by replacing dots with underscores
2015-05-31 21:56:00 +02:00
let compoundtype = substitute ( ftype , "[.]" , "_" , "g" )
2015-06-06 11:54:30 +02:00
if ftype = ~ "[.]"
" Try all super filetypes in search for formatters in a sane order
let ftypes = [compoundtype ] + split ( ftype , "[.]" )
else
let ftypes = [compoundtype ]
endif
2014-06-29 15:30:05 -04:00
2015-05-30 21:00:16 +02:00
" Warn for backward incompatible configuration
2015-05-31 21:56:00 +02:00
let old_formatprg_var = "g:formatprg_" .compoundtype
let old_formatprg_args_var = "g:formatprg_args_" .compoundtype
let old_formatprg_args_expr_var = "g:formatprg_args_expr_" .compoundtype
2015-05-30 21:00:16 +02:00
if exists ( old_formatprg_var ) | | exists ( old_formatprg_args_var ) | | exists ( old_formatprg_args_expr_var )
2015-06-06 11:54:30 +02:00
echohl WarningMsg |
\ echomsg "WARNING: the options g:formatprg_<filetype>, g:formatprg_args_<filetype> and g:formatprg_args_expr_<filetype> are no longer supported as of June 2015, due to major backward-incompatible improvements. Please check the README for help on how to configure your formatters." |
\ echohl None
2015-05-30 21:00:16 +02:00
endif
2015-06-06 11:54:30 +02:00
" Detect configuration for all possible ftypes
2015-05-18 19:27:00 +02:00
let b :formatters = []
2015-06-06 11:54:30 +02:00
for supertype in ftypes
2015-05-31 21:56:00 +02:00
let formatters_var = "g:formatters_" .supertype
if ! exists ( formatters_var )
" No formatters defined
if verbose
echoerr "No formatters defined for supertype '" .supertype
endif
else
let formatters = eval ( formatters_var )
if type ( formatters ) ! = 3
echoerr formatter_var ." is not a list"
else
let b :formatters = b :formatters + formatters
endif
2013-03-16 10:36:16 +01:00
endif
2015-05-31 21:56:00 +02:00
endfor
2013-03-14 18:50:31 +01:00
2015-05-31 21:56:00 +02:00
if len ( b :formatters ) = = 0
2015-05-18 19:27:00 +02:00
" No formatters defined
2015-05-23 10:02:05 +02:00
if verbose
2015-05-31 21:56:00 +02:00
echoerr "No formatters defined for filetype '" .ftype ."'."
2013-03-14 18:50:31 +01:00
endif
2013-12-20 20:34:38 +01:00
return 0
2013-03-14 18:50:31 +01:00
endif
2015-05-18 19:27:00 +02:00
return 1
endfunction
2015-05-23 10:02:05 +02:00
" Try all formatters, starting with the currently selected one, until one
" works. If none works, autoindent the buffer.
2015-05-30 11:02:07 +02:00
function ! s :TryAllFormatters ( ...) range
2015-05-18 19:27:00 +02:00
" Make sure formatters are defined and detected
if ! call ( '<SID>find_formatters' , a :000 )
2015-08-08 14:37:21 +02:00
" No formatters defined, so autoindent code
2015-08-08 14:01:26 +02:00
exe "normal gg=G"
2015-05-18 19:27:00 +02:00
return 0
endif
" Make sure index exist and is valid
if ! exists ( 'b:current_formatter_index' )
let b :current_formatter_index = 0
endif
if b :current_formatter_index > = len ( b :formatters )
let b :current_formatter_index = 0
endif
2015-05-23 10:02:05 +02:00
" Try all formatters, starting with selected one
2015-05-23 14:25:29 +02:00
let s :index = b :current_formatter_index
2015-05-23 10:02:05 +02:00
while 1
2015-05-23 14:25:29 +02:00
let formatdef_var = 'g:formatdef_' .b :formatters [s :index ]
2015-05-23 10:02:05 +02:00
" Formatter definition must be existent
2015-05-23 14:25:29 +02:00
if ! exists ( formatdef_var )
echoerr "No format definition found in '" .formatdef_var ."'."
2015-05-23 10:02:05 +02:00
return 0
endif
" Eval twice, once for getting definition content,
" once for getting the final expression
2015-05-23 14:25:29 +02:00
let &formatprg = eval ( eval ( formatdef_var ) )
2015-06-19 23:17:31 +02:00
2015-06-22 09:18:49 +02:00
" Detect if +python or +python3 is available, and call the corresponding function
2015-06-19 23:17:31 +02:00
if ! has ( "python" ) && ! has ( "python3" )
echohl WarningMsg |
\ echomsg "WARNING: vim has no support for python, but it is required to run the formatter!" |
\ echohl None
return 1
endif
if has ( "python" )
let success = s :TryFormatterPython ( )
else
let success = s :TryFormatterPython3 ( )
endif
if success
2015-05-23 10:02:05 +02:00
return 1
else
2015-05-23 14:25:29 +02:00
let s :index = ( s :index + 1 ) % len ( b :formatters )
2015-05-23 10:02:05 +02:00
endif
2015-05-23 14:25:29 +02:00
if s :index = = b :current_formatter_index
2015-08-08 14:37:21 +02:00
" Tried all formatters, none worked so autoindent code
2015-08-08 14:01:26 +02:00
exe "normal gg=G"
2015-05-23 10:02:05 +02:00
return 0
endif
endwhile
2013-03-15 23:12:09 +01:00
endfunction
2013-03-01 15:06:13 +01:00
2015-05-18 19:27:00 +02:00
2015-05-23 10:02:05 +02:00
" Call formatter
" If stderr is empty, apply result, return 1
" Otherwise, return 0
2015-06-22 09:18:49 +02:00
2015-06-19 23:17:31 +02:00
" +python version
function ! s :TryFormatterPython ( )
2015-05-23 10:02:05 +02:00
" Detect verbosity
let verbose = &verbose | | exists ( "g:autoformat_verbosemode" )
2013-03-14 18:50:31 +01:00
2015-05-18 19:27:00 +02:00
python << EOF
2015-05-31 20:58:14 +02:00
import vim , subprocess , os
2015-05-18 19:27:00 +02:00
from subprocess import Popen , PIPE
2015-05-31 21:56:00 +02:00
2015-08-31 21:11:15 +02:00
text = os .linesep .join ( vim .current .buffer [:])
2015-05-18 19:27:00 +02:00
formatprg = vim .eval ( '&formatprg' )
2015-05-30 16:18:30 +02:00
verbose = bool ( int ( vim .eval ( 'verbose' ) ) )
2015-05-31 20:58:14 +02:00
env = os .environ .copy ( )
if int ( vim .eval ( 'exists("g:formatterpath")' ) ) :
extra_path = vim .eval ( 'g:formatterpath' )
env ['PATH' ] = ':' .join ( extra_path ) + ':' + env ['PATH' ]
2015-05-31 21:56:00 +02:00
2015-05-31 20:58:14 +02:00
p = subprocess .Popen ( formatprg , env = env , shell = True , stdin = PIPE , stdout = PIPE , stderr = PIPE )
2015-05-18 19:27:00 +02:00
stdoutdata , stderrdata = p .communicate ( text )
if stderrdata :
2015-05-23 10:02:05 +02:00
if verbose :
2015-05-23 14:25:29 +02:00
formattername = vim .eval ( 'b:formatters[s:index]' )
2015-05-30 11:02:07 +02:00
print ( 'Formatter {} has errors: {}. Skipping.' .format ( formattername , stderrdata ) )
print ( 'Failing config: {} ' .format ( repr ( formatprg ) , stderrdata ) )
2015-05-23 10:02:05 +02:00
vim .command ( 'return 0' )
2015-05-18 19:27:00 +02:00
else :
2015-08-01 05:57:54 +02:00
# Often shell commands will append a newline at the end of their output .
# It is not entirely clear when and why that happens .
# However , extra newlines are almost never required , while there are linters that complain
# about superfluous newlines , so we remove one empty newline at the end of the file .
2015-08-31 21:11:15 +02:00
if stdoutdata [-1 ] = = os .linesep :
2015-08-01 05:57:54 +02:00
stdoutdata = stdoutdata [:-1 ]
2015-08-31 21:11:15 +02:00
vim .current .buffer [:] = stdoutdata .split ( os .linesep )
2015-05-18 19:27:00 +02:00
EOF
2015-05-23 10:02:05 +02:00
return 1
2013-03-14 18:50:31 +01:00
endfunction
2015-06-19 23:17:31 +02:00
" +python3 version
function ! s :TryFormatterPython3 ( )
" Detect verbosity
let verbose = &verbose | | exists ( "g:autoformat_verbosemode" )
python3 < < EOF
import vim , subprocess , os
from subprocess import Popen , PIPE
2015-08-31 21:11:15 +02:00
text = bytes ( os .linesep .join ( vim .current .buffer [:]) , 'utf-8' )
2015-06-19 23:17:31 +02:00
formatprg = vim .eval ( '&formatprg' )
verbose = bool ( int ( vim .eval ( 'verbose' ) ) )
env = os .environ .copy ( )
if int ( vim .eval ( 'exists("g:formatterpath")' ) ) :
extra_path = vim .eval ( 'g:formatterpath' )
env ['PATH' ] = ':' .join ( extra_path ) + ':' + env ['PATH' ]
p = subprocess .Popen ( formatprg , env = env , shell = True , stdin = PIPE , stdout = PIPE , stderr = PIPE )
stdoutdata , stderrdata = p .communicate ( text )
if stderrdata :
if verbose :
formattername = vim .eval ( 'b:formatters[s:index]' )
print ( 'Formatter {} has errors: {}. Skipping.' .format ( formattername , stderrdata ) )
print ( 'Failing config: {} ' .format ( repr ( formatprg ) , stderrdata ) )
vim .command ( 'return 0' )
else :
2015-08-01 05:57:54 +02:00
# Often shell commands will append a newline at the end of their output .
# It is not entirely clear when and why that happens .
# However , extra newlines are almost never required , while there are linters that complain
# about superfluous newlines , so we remove one empty newline at the end of the file .
2015-10-08 17:06:58 -04:00
stdoutdata_decoded = stdoutdata .decode ( 'utf-8' )
if stdoutdata_decoded [-1 ] = = os .linesep :
stdoutdata = stdoutdata_decoded [:-1 ]
2015-08-31 21:11:15 +02:00
vim .current .buffer [:] = stdoutdata .split ( os .linesep )
2015-06-19 23:17:31 +02:00
EOF
return 1
endfunction
2015-05-23 10:02:05 +02:00
2015-04-09 10:59:54 +02:00
" Create a command for formatting the entire buffer
2015-06-12 14:29:33 +02:00
" Save and recall window state to prevent vim from jumping to line 1
command ! - nargs = ? - range = % - complete = filetype Autoformat let winview = winsaveview ( ) | < line1 > , < line2 > call s :TryAllFormatters ( < f - args > ) | call winrestview ( winview )
2015-05-18 19:27:00 +02:00
" Functions for iterating through list of available formatters
function ! s :NextFormatter ( )
call s :find_formatters ( )
if ! exists ( 'b:current_formatter_index' )
let b :current_formatter_index = 0
endif
let b :current_formatter_index = ( b :current_formatter_index + 1 ) % len ( b :formatters )
2015-05-23 14:25:29 +02:00
echom 'Selected formatter: ' .b :formatters [b :current_formatter_index ]
2015-05-18 19:27:00 +02:00
endfunction
function ! s :PreviousFormatter ( )
call s :find_formatters ( )
if ! exists ( 'b:current_formatter_index' )
let b :current_formatter_index = 0
endif
2015-05-23 10:02:05 +02:00
let l = len ( b :formatters )
let b :current_formatter_index = ( b :current_formatter_index - 1 + l ) % l
2015-05-23 14:25:29 +02:00
echom 'Selected formatter: ' .b :formatters [b :current_formatter_index ]
2015-05-18 19:27:00 +02:00
endfunction
" Create commands for iterating through formatter list
command ! NextFormatter call s :NextFormatter ( )
command ! PreviousFormatter call s :PreviousFormatter ( )