2015-03-07 23:02:15 +01:00
|
|
|
" vimtex - LaTeX plugin for Vim
|
2014-07-23 00:08:57 +02:00
|
|
|
"
|
|
|
|
" Maintainer: Karl Yngve Lervåg
|
|
|
|
" Email: karl.yngve@gmail.com
|
|
|
|
"
|
|
|
|
|
2015-03-19 15:12:12 +01:00
|
|
|
if exists('b:did_indent')
|
2013-10-10 14:47:08 +02:00
|
|
|
finish
|
|
|
|
endif
|
|
|
|
let b:did_indent = 1
|
2016-02-23 21:03:53 +01:00
|
|
|
let b:did_vimtex_indent = 1
|
2015-01-26 15:41:40 +01:00
|
|
|
|
2015-03-07 23:02:15 +01:00
|
|
|
call vimtex#util#set_default('g:vimtex_indent_enabled', 1)
|
|
|
|
if !g:vimtex_indent_enabled | finish | endif
|
2015-01-26 15:41:40 +01:00
|
|
|
|
2013-10-10 14:47:08 +02:00
|
|
|
let s:cpo_save = &cpo
|
|
|
|
set cpo&vim
|
|
|
|
|
|
|
|
setlocal autoindent
|
2016-08-01 22:57:43 +02:00
|
|
|
setlocal indentexpr=VimtexIndent(v:lnum)
|
2013-10-10 14:47:08 +02:00
|
|
|
setlocal indentkeys&
|
2015-10-26 22:43:36 +01:00
|
|
|
setlocal indentkeys+=[,(,{,),},],\&,=item
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-08-01 22:57:43 +02:00
|
|
|
function! VimtexIndent(lnum) " {{{1
|
2016-10-13 09:49:28 +02:00
|
|
|
let l:prev_lnum = s:get_prev_line(prevnonblank(a:lnum - 1))
|
|
|
|
if l:prev_lnum == 0 | return indent(a:lnum) | endif
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-03-24 12:52:45 +01:00
|
|
|
" Get current and previous line and remove comments
|
2016-10-13 09:49:28 +02:00
|
|
|
let l:line = substitute(getline(a:lnum), '\\\@<!%.*', '', '')
|
|
|
|
let l:prev_line = substitute(getline(l:prev_lnum), '\\\@<!%.*', '', '')
|
2013-10-10 14:47:08 +02:00
|
|
|
|
|
|
|
" Check for verbatim modes
|
2016-10-13 09:49:28 +02:00
|
|
|
if s:is_verbatim(l:line, a:lnum)
|
|
|
|
return empty(l:line) ? indent(l:prev_lnum) : indent(a:lnum)
|
2013-10-10 14:47:08 +02:00
|
|
|
endif
|
|
|
|
|
|
|
|
" Align on ampersands
|
2016-10-13 09:49:28 +02:00
|
|
|
if l:line =~# '^\s*&' && l:prev_line =~# '\\\@<!&.*'
|
|
|
|
return indent(a:lnum) + match(l:prev_line, '\\\@<!&') - stridx(l:line, '&')
|
2016-03-24 12:52:45 +01:00
|
|
|
endif
|
|
|
|
|
|
|
|
" Use previous indentation for comments
|
2016-10-13 09:49:28 +02:00
|
|
|
if l:line =~# '^\s*%'
|
2016-08-01 22:57:43 +02:00
|
|
|
return indent(a:lnum)
|
2013-10-10 14:47:08 +02:00
|
|
|
endif
|
|
|
|
|
2016-10-13 09:49:28 +02:00
|
|
|
" Ensure previous line does not start with ampersand
|
|
|
|
let l:prev_lnum = s:get_prev_line(l:prev_lnum, 'ignore-ampersands')
|
|
|
|
if l:prev_lnum == 0 | return 0 | endif
|
|
|
|
let l:prev_line = substitute(getline(l:prev_lnum), '\\\@<!%.*', '', '')
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-10-13 09:49:28 +02:00
|
|
|
" Indent environments, delimiters, and tikz
|
|
|
|
let l:ind = indent(l:prev_lnum)
|
|
|
|
let l:ind += s:indent_envs(l:line, l:prev_line)
|
|
|
|
let l:ind += s:indent_delims(l:line, a:lnum, l:prev_line, l:prev_lnum)
|
|
|
|
let l:ind += s:indent_tikz(l:prev_lnum, l:prev_line)
|
2016-03-24 22:25:16 +01:00
|
|
|
return l:ind
|
|
|
|
endfunction
|
|
|
|
"}}}
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-09-17 22:24:20 +02:00
|
|
|
function! s:get_prev_line(lnum, ...) " {{{1
|
|
|
|
let l:ignore_amps = a:0 > 0
|
2016-03-28 00:22:10 +02:00
|
|
|
let l:lnum = a:lnum
|
|
|
|
let l:prev = getline(l:lnum)
|
|
|
|
|
|
|
|
while l:lnum != 0
|
2016-09-17 22:24:20 +02:00
|
|
|
\ && (l:prev =~# '^\s*%'
|
|
|
|
\ || s:is_verbatim(l:prev, l:lnum)
|
|
|
|
\ || !l:ignore_amps && match(l:prev, '^\s*&') >= 0)
|
2016-03-28 00:22:10 +02:00
|
|
|
let l:lnum = prevnonblank(l:lnum - 1)
|
|
|
|
let l:prev = getline(l:lnum)
|
|
|
|
endwhile
|
|
|
|
|
|
|
|
return l:lnum
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" }}}1
|
|
|
|
function! s:is_verbatim(line, lnum) " {{{1
|
|
|
|
let l:env = a:line !~# '\v\\%(begin|end)\{%(verbatim|lstlisting|minted)'
|
|
|
|
let l:syn = synIDattr(synID(a:lnum, 1, 1), 'name') ==# 'texZone'
|
|
|
|
return l:env && l:syn
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" }}}1
|
|
|
|
|
2016-03-24 22:25:16 +01:00
|
|
|
function! s:indent_envs(cur, prev) " {{{1
|
|
|
|
let l:ind = 0
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-03-24 23:13:32 +01:00
|
|
|
" First for general environments
|
2016-03-28 00:22:10 +02:00
|
|
|
let l:ind += &sw*((a:prev =~# '\\begin{.*}') && (a:prev !~# 'document'))
|
|
|
|
let l:ind -= &sw*((a:cur =~# '\\end{.*}') && (a:cur !~# 'document'))
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-03-24 23:13:32 +01:00
|
|
|
" Indentation for prolonged items in lists
|
|
|
|
let l:ind += &sw*((a:prev =~# s:envs_item) && (a:cur !~# s:envs_enditem))
|
|
|
|
let l:ind -= &sw*((a:cur =~# s:envs_item) && (a:prev !~# s:envs_begitem))
|
|
|
|
let l:ind -= &sw*((a:cur =~# s:envs_endlist) && (a:prev !~# s:envs_begitem))
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-03-24 22:25:16 +01:00
|
|
|
return l:ind
|
|
|
|
endfunction
|
2015-08-25 21:38:05 +02:00
|
|
|
|
2016-03-24 22:25:16 +01:00
|
|
|
let s:envs_lists = 'itemize\|description\|enumerate\|thebibliography'
|
2016-03-24 23:13:32 +01:00
|
|
|
let s:envs_item = '^\s*\\item'
|
|
|
|
let s:envs_beglist = '\\begin{\%(' . s:envs_lists . '\)'
|
|
|
|
let s:envs_endlist = '\\end{\%(' . s:envs_lists . '\)'
|
|
|
|
let s:envs_begitem = s:envs_item . '\|' . s:envs_beglist
|
|
|
|
let s:envs_enditem = s:envs_item . '\|' . s:envs_endlist
|
2016-03-24 22:25:16 +01:00
|
|
|
|
|
|
|
" }}}1
|
2016-10-13 09:49:28 +02:00
|
|
|
function! s:indent_delims(line, lnum, prev_line, prev_lnum) " {{{1
|
|
|
|
let [l:text, l:math] = s:split(a:line, a:lnum)
|
|
|
|
let [l:prev_tex, l:prev_math] = s:split(a:prev_line, a:prev_lnum, 'prev')
|
|
|
|
|
|
|
|
return &sw*( max([ s:count(l:prev_math, s:re_delims[0])
|
|
|
|
\ - s:count(l:prev_math, s:re_delims[1])
|
|
|
|
\ + s:count(l:prev_tex, s:re_delims[2])
|
|
|
|
\ - s:count(l:prev_tex, s:re_delims[3]), 0])
|
|
|
|
\ - max([ s:count(l:math, s:re_delims[1])
|
|
|
|
\ - s:count(l:math, s:re_delims[0])
|
|
|
|
\ + s:count(l:text, s:re_delims[3])
|
|
|
|
\ - s:count(l:text, s:re_delims[2]), 0]))
|
2016-03-24 22:25:16 +01:00
|
|
|
endfunction
|
2013-10-10 14:47:08 +02:00
|
|
|
|
2016-10-13 09:49:28 +02:00
|
|
|
"
|
|
|
|
" This fetches the regexes for delimiters in text and math mode:
|
|
|
|
" s:re_delims[0] == math open
|
|
|
|
" s:re_delims[1] == math close
|
|
|
|
" s:re_delims[2] == text open
|
|
|
|
" s:re_delims[3] == text close
|
|
|
|
"
|
|
|
|
let s:re_delims = vimtex#delim#get_delim_regexes()
|
2016-10-11 22:56:50 +02:00
|
|
|
|
|
|
|
function! s:count(line, pattern) " {{{2
|
2016-10-13 09:49:28 +02:00
|
|
|
let l:sum = 0
|
|
|
|
let l:indx = match(a:line, a:pattern)
|
|
|
|
while l:indx >= 0
|
|
|
|
let l:sum += 1
|
|
|
|
let l:match = matchstr(a:line, a:pattern, l:indx)
|
|
|
|
let l:indx += len(l:match)
|
|
|
|
let l:indx = match(a:line, a:pattern, l:indx)
|
2016-10-11 22:56:50 +02:00
|
|
|
endwhile
|
2016-10-13 09:49:28 +02:00
|
|
|
return l:sum
|
2016-10-11 22:56:50 +02:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
" }}}2
|
2016-10-13 09:49:28 +02:00
|
|
|
function! s:split(line, lnum, ...) " {{{2
|
2016-10-11 22:56:50 +02:00
|
|
|
let l:map = map(range(1,col([a:lnum, strlen(a:line)])),
|
|
|
|
\ '[v:val, vimtex#util#in_mathzone(a:lnum, v:val)]')
|
|
|
|
|
|
|
|
" Adjust math mode limits (currently handle only $'s)
|
|
|
|
let l:prev = 1
|
|
|
|
for l:i in range(len(l:map))
|
|
|
|
if l:map[l:i][1] == 1 && l:prev == 0
|
|
|
|
let l:prev = l:map[l:i][1]
|
|
|
|
let l:map[l:i][1] = 0
|
|
|
|
else
|
|
|
|
let l:prev = l:map[l:i][1]
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
if l:map[0][1] == 1 && a:line[0] ==# '$'
|
|
|
|
let l:map[0][1] = 0
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Extract normal text
|
|
|
|
let l:normal = ''
|
|
|
|
for [l:i, l:val] in l:map
|
|
|
|
if l:val == 0
|
|
|
|
let l:normal .= a:line[l:i - 1]
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
let l:normal = substitute(l:normal, '\\verb\(.\).\{}\1', '', 'g')
|
|
|
|
|
2016-10-13 09:49:28 +02:00
|
|
|
"
|
|
|
|
" Extract math text (either at beginning or end of line, depending on if we
|
|
|
|
" are looking at the current line or a previous line)
|
|
|
|
"
|
|
|
|
if a:0 == 0
|
|
|
|
" Extract math text from beginning of line
|
|
|
|
let l:math = ''
|
|
|
|
let l:indx = 0
|
|
|
|
while l:map[l:indx][1] == 1
|
|
|
|
let l:math .= a:line[l:map[l:indx][0] - 1]
|
|
|
|
let l:indx += 1
|
|
|
|
endwhile
|
|
|
|
else
|
|
|
|
" Extract math text from end of line
|
|
|
|
let l:math = ''
|
|
|
|
let l:indx = -1
|
|
|
|
while l:map[l:indx][1] == 1
|
|
|
|
let l:math = a:line[l:map[l:indx][0] - 1] . l:math
|
|
|
|
let l:indx -= 1
|
|
|
|
endwhile
|
|
|
|
endif
|
2016-10-11 22:56:50 +02:00
|
|
|
|
2016-10-13 09:49:28 +02:00
|
|
|
return [l:normal, l:math]
|
2016-10-11 22:56:50 +02:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
" }}}2
|
|
|
|
|
2016-03-24 22:25:16 +01:00
|
|
|
" }}}1
|
2016-03-24 23:50:36 +01:00
|
|
|
function! s:indent_tikz(lnum, prev) " {{{1
|
2016-04-28 23:27:34 +02:00
|
|
|
if vimtex#env#is_inside('tikzpicture')
|
2016-03-24 23:50:36 +01:00
|
|
|
let l:prev_starts = a:prev =~# s:tikz_commands
|
|
|
|
let l:prev_stops = a:prev =~# ';\s*$'
|
|
|
|
|
|
|
|
" Increase indent on tikz command start
|
|
|
|
if l:prev_starts && ! l:prev_stops
|
|
|
|
return &sw
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Decrease indent on tikz command end, i.e. on semicolon
|
|
|
|
if ! l:prev_starts && l:prev_stops
|
|
|
|
let l:context = join(getline(max([1,a:lnum-4]), a:lnum-1), '')
|
|
|
|
return -&sw*(l:context =~# s:tikz_commands)
|
|
|
|
endif
|
2014-04-14 08:48:33 +02:00
|
|
|
endif
|
|
|
|
|
2016-03-24 23:50:36 +01:00
|
|
|
return 0
|
2013-10-10 14:47:08 +02:00
|
|
|
endfunction
|
2016-03-24 22:25:16 +01:00
|
|
|
|
|
|
|
let s:tikz_commands = '\v\\%(' . join([
|
|
|
|
\ 'draw',
|
|
|
|
\ 'fill',
|
|
|
|
\ 'path',
|
|
|
|
\ 'node',
|
|
|
|
\ 'coordinate',
|
|
|
|
\ 'add%(legendentry|plot)',
|
|
|
|
\ ], '|') . ')'
|
|
|
|
|
|
|
|
" }}}1
|
|
|
|
|
2013-10-10 14:47:08 +02:00
|
|
|
let &cpo = s:cpo_save
|
|
|
|
unlet s:cpo_save
|
|
|
|
|
2014-12-08 20:44:17 +01:00
|
|
|
" vim: fdm=marker sw=2
|