395 lines
9.7 KiB
VimL
Raw Normal View History

2014-07-23 00:08:57 +02:00
" LaTeX plugin for Vim
"
" Maintainer: Karl Yngve Lervåg
" Email: karl.yngve@gmail.com
"
function! latex#toc#init(initialized) " {{{1
if !g:latex_toc_enabled | return | endif
" Define commands
command! -buffer VimLatexTocOpen call latex#toc#open()
command! -buffer VimLatexTocToggle call latex#toc#toggle()
" Define mappings
if g:latex_mappings_enabled
nnoremap <buffer> <LocalLeader>lt :call latex#toc#open()<cr>
nnoremap <buffer> <LocalLeader>lT :call latex#toc#toggle()<cr>
2013-10-05 13:53:42 +02:00
endif
endfunction
function! latex#toc#open() " {{{1
2014-07-04 20:15:28 +02:00
" Go to TOC if it already exists
2013-10-05 13:53:42 +02:00
let winnr = bufwinnr(bufnr('LaTeX TOC'))
if winnr >= 0
silent execute winnr . 'wincmd w'
return
endif
2014-07-04 20:15:28 +02:00
" Store current buffer number and position
2013-10-05 13:53:42 +02:00
let calling_buf = bufnr('%')
let calling_file = expand('%:p')
2014-07-04 20:15:28 +02:00
let calling_line = line('.')
2014-07-01 22:34:26 +02:00
2014-07-27 20:41:23 +02:00
" Parse TOC data
let toc = s:parse_toc()
2014-07-04 20:15:28 +02:00
" Resize vim session if wanted, then create TOC window
2013-10-05 13:53:42 +02:00
if g:latex_toc_resize
silent exe "set columns+=" . g:latex_toc_width
endif
2014-07-01 22:34:26 +02:00
silent exe g:latex_toc_split_side g:latex_toc_width . 'vnew LaTeX\ TOC'
2014-07-04 20:15:28 +02:00
" Set buffer local variables
2014-07-01 22:34:26 +02:00
let b:toc = toc
let b:toc_numbers = 1
let b:calling_win = bufwinnr(calling_buf)
2014-07-15 13:41:12 +02:00
" Add TOC entries (and keep track of closest index)
2014-07-04 20:15:28 +02:00
let index = 0
let closest_index = 0
2014-07-01 22:34:26 +02:00
for entry in toc
call append('$',
\ printf('%-10s%-140s%s',
\ entry.number,
\ entry.title,
\ s:max_level - entry.level))
2014-07-15 13:41:12 +02:00
2014-07-04 20:15:28 +02:00
let index += 1
2014-07-15 13:41:12 +02:00
if entry.file == calling_file && entry.line <= calling_line
let closest_index = index
2014-07-04 20:15:28 +02:00
endif
2014-07-01 22:34:26 +02:00
endfor
2013-10-05 13:53:42 +02:00
2014-07-04 20:15:28 +02:00
" Add help info (if desired)
2013-10-05 13:53:42 +02:00
if !g:latex_toc_hide_help
call append('$', "")
call append('$', "<Esc>/q: close")
call append('$', "<Space>: jump")
call append('$', "<Enter>: jump and close")
call append('$', "s: hide numbering")
endif
2014-07-04 20:15:28 +02:00
" Delete empty first line and jump to the closest section
0delete _
call setpos('.', [0, closest_index, 0, 0])
2013-10-23 21:00:20 +02:00
2013-10-05 13:53:42 +02:00
" Set filetype and lock buffer
setlocal filetype=latextoc
setlocal nomodifiable
endfunction
function! latex#toc#toggle() " {{{1
2013-10-05 13:53:42 +02:00
if bufwinnr(bufnr('LaTeX TOC')) >= 0
if g:latex_toc_resize
silent exe "set columns-=" . g:latex_toc_width
endif
silent execute 'bwipeout' . bufnr('LaTeX TOC')
else
call latex#toc#open()
silent execute 'wincmd w'
endif
endfunction
2014-07-27 20:41:23 +02:00
2013-10-05 13:53:42 +02:00
" }}}1
2014-07-27 20:41:23 +02:00
" {{{1 TOC variables
let s:max_level = 0
let s:count_matters = 0
" Define dictionary to keep track of TOC numbers
2014-07-19 00:12:37 +02:00
let s:number = {
\ 'part' : 0,
\ 'chapter' : 0,
\ 'section' : 0,
\ 'subsection' : 0,
\ 'subsubsection' : 0,
\ 'subsubsubsection' : 0,
\ 'current_level' : 0,
2014-07-19 00:12:37 +02:00
\ 'preamble' : 0,
\ 'frontmatter' : 0,
\ 'mainmatter' : 0,
\ 'appendix' : 0,
\ 'backmatter' : 0,
\ }
" Map for section hierarchy
let s:sec_to_value = {
\ '_' : 0,
\ 'subsubsubsection' : 1,
\ 'subsubsection' : 2,
\ 'subsection' : 3,
\ 'section' : 4,
\ 'chapter' : 5,
\ 'part' : 6,
\ }
2014-07-16 23:20:11 +02:00
" Define regular expressions to match document parts
let s:re_input = '\v^\s*\\%(input|include)\s*\{'
let s:re_input_file = s:re_input . '\zs[^\}]+\ze}'
let s:re_sec = '\v^\s*\\%(part|chapter|%(sub)*section)\*?\s*\{'
let s:re_sec_starred = '\v^\s*\\%(part|chapter|%(sub)*section)\*'
let s:re_sec_level = '\v^\s*\\\zs%(part|chapter|%(sub)*section)'
let s:re_sec_title = s:re_sec . '\zs.{-}\ze\}?$'
let s:re_matters = '\v^\s*\\%(front|main|back)matter>'
2014-07-19 00:12:37 +02:00
let s:re_structure = '\v^\s*\\((front|main|back)matter|appendix)>'
let s:re_structure_match = '\v((front|main|back)matter|appendix)'
let s:re_other = {
\ 'toc' : {
\ 'title' : 'Table of contents',
\ 're' : '\v^\s*\\tableofcontents',
\ },
\ 'index' : {
\ 'title' : 'Alphabetical index',
\ 're' : '\v^\s*\\printindex\[?',
\ },
\ 'bib' : {
\ 'title' : 'Bibliography',
\ 're' : '\v^\s*\\%('
\ . 'printbib%(liography|heading)\s*(\{|\[)?'
\ . '|begin\s*\{\s*thebibliography\s*\}'
\ . '|bibliography\s*\{)',
\ },
\ }
" }}}1
2014-07-27 20:41:23 +02:00
function! s:parse_toc() " {{{1
let file = g:latex#data[b:latex.id].tex
" Reset TOC numbering
call s:number_reset('preamble')
" Find max level and number of \*matter commands
let s:max_level = 0
let s:count_matters = 0
call s:parse_limits(file)
" Parse TOC data
return s:parse_file(file)
endfunction
" }}}1
function! s:parse_limits(file) " {{{1
if !filereadable(a:file)
2014-07-29 09:27:48 +02:00
echoerr "Error in latex#toc s:parse_limits"
2014-07-27 20:41:23 +02:00
echoerr "File not readable: " . a:file
return ''
endif
for line in readfile(a:file)
if line =~# s:re_input
2014-07-30 18:10:42 +02:00
call s:parse_limits(s:parse_line_input(line, a:file))
2014-07-27 20:41:23 +02:00
elseif line =~# s:re_sec
let s:max_level = max([s:max_level,
\ s:sec_to_value[matchstr(line, s:re_sec_level)]])
elseif line =~# s:re_matters
let s:count_matters += 1
endif
endfor
endfunction
" }}}1
function! s:parse_file(file) " {{{1
2014-07-04 20:15:28 +02:00
" Parses tex file for TOC entries
"
" The function returns a list of entries. Each entry is a dictionary:
"
" entry = {
" title : "Some title",
" number : "3.1.2",
" file : /path/to/file.tex,
" line : 142,
" level : 2,
2014-07-04 20:15:28 +02:00
" }
if a:file == ''
return []
elseif !filereadable(a:file)
2014-07-27 20:41:23 +02:00
echoerr "Error in latex#toc s:parse_file"
2014-07-04 20:15:28 +02:00
echoerr "File not readable: " . a:file
return []
2013-10-05 13:53:42 +02:00
endif
2014-07-04 20:15:28 +02:00
let toc = []
let lnum = 0
for line in readfile(a:file)
2014-07-01 22:34:26 +02:00
let lnum += 1
2014-07-04 20:15:28 +02:00
2014-07-08 12:32:41 +02:00
" 1. Parse inputs or includes
if line =~# s:re_input
2014-07-30 18:10:42 +02:00
call extend(toc, s:parse_file(s:parse_line_input(line, a:file)))
2014-07-04 20:15:28 +02:00
continue
endif
" 2. Parse preamble
if s:number.preamble
if line =~# '\v^\s*\\documentclass'
call add(toc, {
\ 'title' : 'Preamble',
\ 'number' : '',
\ 'file' : a:file,
\ 'line' : lnum,
\ 'level' : s:max_level,
\ })
continue
endif
if line =~# '\v^\s*\\begin\{document\}'
let s:number.preamble = 0
endif
continue
endif
2014-07-19 00:12:37 +02:00
" 3. Parse document structure (front-/main-/backmatter, appendix)
if line =~# s:re_structure
call s:number_reset(matchstr(line, s:re_structure_match))
2014-07-04 20:15:28 +02:00
continue
endif
2014-07-09 12:11:55 +02:00
2014-07-19 00:12:37 +02:00
" 4. Parse \parts, \chapters, \sections, and \subsections
if line =~# s:re_sec
call add(toc, s:parse_line_sec(a:file, lnum, line))
continue
endif
2014-07-19 00:12:37 +02:00
" 5. Parse other stuff
for other in values(s:re_other)
if line =~# other.re
call add(toc, {
\ 'title' : other.title,
\ 'number' : '',
\ 'file' : a:file,
\ 'line' : lnum,
\ 'level' : s:max_level,
2014-07-19 00:12:37 +02:00
\ })
continue
endif
endfor
2014-07-01 22:34:26 +02:00
endfor
2013-10-05 13:53:42 +02:00
2014-07-01 22:34:26 +02:00
return toc
endfunction
2013-10-05 13:53:42 +02:00
2014-07-30 18:10:42 +02:00
function! s:parse_line_input(line, file) " {{{1
2014-07-08 12:32:41 +02:00
let l:file = matchstr(a:line, s:re_input_file)
2014-07-30 18:10:42 +02:00
" Ensure file has extension
2014-07-04 20:15:28 +02:00
if l:file !~# '.tex$'
let l:file .= '.tex'
2013-10-05 13:53:42 +02:00
endif
" Only return full path names
2014-07-31 23:29:05 +02:00
if l:file !~# '\v^(\/|[A-Z]:)'
2014-07-30 18:10:42 +02:00
let l:file = fnamemodify(a:file, ':p:h') . '/' . l:file
endif
2014-07-29 09:27:48 +02:00
" Only return filename if it is readable
if filereadable(l:file)
return l:file
else
return ''
endif
2014-07-04 20:15:28 +02:00
endfunction
function! s:parse_line_sec(file, lnum, line) " {{{1
2014-07-08 12:32:41 +02:00
let title = matchstr(a:line, s:re_sec_title)
let level = matchstr(a:line, s:re_sec_level)
let starred = a:line =~# s:re_sec_starred ? 1 : 0
let number = s:number_increment(level, starred)
2014-07-04 20:15:28 +02:00
return {
2014-07-08 12:32:41 +02:00
\ 'title' : title,
2014-07-09 12:11:55 +02:00
\ 'number' : number,
2014-07-04 20:15:28 +02:00
\ 'file' : a:file,
\ 'line' : a:lnum,
\ 'level' : s:number.current_level,
2014-07-04 20:15:28 +02:00
\ }
2013-10-05 13:53:42 +02:00
endfunction
" }}}1
2014-07-19 00:12:37 +02:00
function! s:number_reset(part) " {{{1
for key in keys(s:number)
let s:number[key] = 0
endfor
let s:number[a:part] = 1
2014-07-09 12:11:55 +02:00
endfunction
function! s:number_increment(level, starred) " {{{1
" Store current level
let s:number.current_level = s:sec_to_value[a:level]
2014-07-09 12:11:55 +02:00
" Check if level should be incremented
if a:starred
2014-07-08 12:32:41 +02:00
return ''
endif
2013-10-05 13:53:42 +02:00
2014-07-09 12:11:55 +02:00
" Increment numbers
if a:level == 'part'
let s:number.part += 1
let s:number.chapter = 0
let s:number.section = 0
let s:number.subsection = 0
let s:number.subsubsection = 0
let s:number.subsubsubsection = 0
2014-07-09 12:11:55 +02:00
elseif a:level == 'chapter'
let s:number.chapter += 1
let s:number.section = 0
let s:number.subsection = 0
let s:number.subsubsection = 0
let s:number.subsubsubsection = 0
2014-07-09 12:11:55 +02:00
elseif a:level == 'section'
let s:number.section += 1
let s:number.subsection = 0
let s:number.subsubsection = 0
let s:number.subsubsubsection = 0
2014-07-09 12:11:55 +02:00
elseif a:level == 'subsection'
let s:number.subsection += 1
let s:number.subsubsection = 0
let s:number.subsubsubsection = 0
2014-07-09 12:11:55 +02:00
elseif a:level == 'subsubsection'
let s:number.subsubsection += 1
let s:number.subsubsubsection = 0
elseif a:level == 'subsubsubsection'
let s:number.subsubsubsection += 1
2014-07-09 12:11:55 +02:00
endif
2014-07-09 12:43:10 +02:00
return s:number_print()
2014-07-09 12:11:55 +02:00
endfunction
function! s:number_print() " {{{1
2014-07-09 12:11:55 +02:00
let number = [
2014-07-09 12:43:10 +02:00
\ s:number.part,
\ s:number.chapter,
\ s:number.section,
\ s:number.subsection,
\ s:number.subsubsection,
\ s:number.subsubsubsection,
2014-07-09 12:11:55 +02:00
\ ]
2014-07-19 00:12:37 +02:00
" Remove unused parts
2014-07-09 12:43:10 +02:00
while number[0] == 0
call remove(number, 0)
endwhile
while number[-1] == 0
call remove(number, -1)
endwhile
2014-07-19 00:12:37 +02:00
" Change numbering in frontmatter, appendix, and backmatter
if s:count_matters > 1
\ && (s:number.frontmatter || s:number.backmatter)
return ""
2014-07-19 00:12:37 +02:00
elseif s:number.appendix
2014-07-09 12:43:10 +02:00
let number[0] = nr2char(number[0] + 64)
endif
return join(number, '.')
2014-07-04 20:15:28 +02:00
endfunction
2014-07-08 12:32:41 +02:00
2014-07-04 20:15:28 +02:00
" }}}1
2014-02-10 14:21:43 +01:00
" vim: fdm=marker