" LaTeX plugin for Vim " " Maintainer: Karl Yngve LervÄg " Email: karl.yngve@gmail.com " function! latex#latexmk#init(initialized) " {{{1 if !g:latex_latexmk_enabled | return | endif " " Check system compatibility " if s:system_incompatible() | return | endif " " Initialize pid for current tex file " if !has_key(g:latex#data[b:latex.id], 'pid') let g:latex#data[b:latex.id].pid = 0 endif " " Define commands " com! -buffer VimLatexCompile call latex#latexmk#compile() com! -buffer VimLatexStop call latex#latexmk#stop() com! -buffer VimLatexStopAll call latex#latexmk#stop_all() com! -buffer VimLatexErrors call latex#latexmk#errors(1) com! -buffer VimLatexOutput call latex#latexmk#output() com! -buffer -bang VimLatexClean call latex#latexmk#clean( == "!") com! -buffer -bang VimLatexStatus call latex#latexmk#status( == "!") " " Set default mappings " if g:latex_mappings_enabled nnoremap ll :call latex#latexmk#compile() nnoremap lc :call latex#latexmk#clean(0) nnoremap lC :call latex#latexmk#clean(1) nnoremap lg :call latex#latexmk#status(0) nnoremap lG :call latex#latexmk#status(1) nnoremap lk :call latex#latexmk#stop() nnoremap lK :call latex#latexmk#stop_all() nnoremap le :call latex#latexmk#errors(1) nnoremap lo :call latex#latexmk#output() endif " " Ensure that all latexmk processes are stopped when vim exits " Note: Only need to define this once, globally. " if !a:initialized augroup latex_latexmk autocmd! autocmd VimLeave * call latex#latexmk#stop_all() augroup END endif " " If all buffers for a given latex project are closed, kill latexmk " Note: This must come after the above so that the autocmd group is properly " refreshed if necessary " augroup latex_latexmk autocmd BufUnload call s:stop_buffer() augroup END endfunction " }}}1 function! latex#latexmk#clean(full) " {{{1 let data = g:latex#data[b:latex.id] if data.pid echomsg "latexmk is already running" return endif " " Run latexmk clean process " if has('win32') let cmd = 'cd /D ' . shellescape(data.root) . ' & ' else let cmd = 'cd ' . shellescape(data.root) . '; ' endif if a:full let cmd .= 'latexmk -C ' else let cmd .= 'latexmk -c ' endif let cmd .= shellescape(data.base) let g:latex#data[b:latex.id].cmds.clean = cmd let exe = { \ 'cmd' : cmd, \ 'bg' : 0, \ } call latex#util#execute(exe) if a:full echomsg "latexmk full clean finished" else echomsg "latexmk clean finished" endif endfunction " }}}1 function! latex#latexmk#compile() " {{{1 let data = g:latex#data[b:latex.id] if data.pid echomsg "latexmk is already running for `" . data.base . "'" return endif call s:latexmk_set_cmd(data) " Start latexmk " Define execute dictionary and latexmk command let exe = {} let exe.null = 0 let exe.cmd = data.cmds.compile call latex#util#execute(exe) " Save PID call s:latexmk_set_pid(data) " Finished echomsg 'latexmk started successfully' endfunction " }}}1 function! latex#latexmk#errors(force) " {{{1 cclose let log = g:latex#data[b:latex.id].log() if empty(log) if a:force echo "No log file found!" endif return endif if g:latex_quickfix_autojump execute 'cfile ' . fnameescape(log) else execute 'cgetfile ' . fnameescape(log) endif " " There are two options that determine when to open the quickfix window. If " forced, the quickfix window is always opened when there are errors or " warnings (forced typically imply that the functions is called from the " normal mode mapping). Else the behaviour is based on the settings. " let open_quickfix_window = a:force \ || (g:latex_quickfix_mode > 0 \ && (g:latex_quickfix_open_on_warning \ || s:log_contains_error(log))) if open_quickfix_window botright cwindow if g:latex_quickfix_mode == 2 wincmd p endif redraw! endif endfunction " }}}1 function! latex#latexmk#output() " {{{1 let tmp = g:latex#data[b:latex.id].tmp if empty(tmp) | return | endif " Create latexmk output window if bufnr(tmp) >= 0 silent exe 'bwipeout' . bufnr(tmp) endif silent exe 'split ' . tmp " Better automatic update augroup tmp_update autocmd! autocmd BufEnter * silent! checktime autocmd CursorHold * silent! checktime autocmd CursorHoldI * silent! checktime autocmd CursorMoved * silent! checktime autocmd CursorMovedI * silent! checktime augroup END silent exe 'autocmd! BufDelete ' . tmp . ' augroup! tmp_update' " Set some mappings nnoremap q :bwipeout " Set some buffer options setlocal autoread setlocal nomodifiable endfunction " }}}1 function! latex#latexmk#status(detailed) " {{{1 if a:detailed let running = 0 for data in g:latex#data if data.pid if !running echo "latexmk is running" let running = 1 endif let name = data.tex if len(name) >= winwidth('.') - 20 let name = "..." . name[-winwidth('.')+23:] endif echom printf('pid: %6s, file: %-s', data.pid, name) endif endfor if !running echo "latexmk is not running" endif else if g:latex#data[b:latex.id].pid echo "latexmk is running" else echo "latexmk is not running" endif endif endfunction " }}}1 function! latex#latexmk#stop() " {{{1 let pid = g:latex#data[b:latex.id].pid let base = g:latex#data[b:latex.id].base if pid call s:latexmk_kill_pid(pid) let g:latex#data[b:latex.id].pid = 0 echo "latexmk stopped for `" . base . "'" else echo "latexmk is not running for `" . base . "'" endif endfunction " }}}1 function! latex#latexmk#stop_all() " {{{1 for data in g:latex#data if data.pid call s:latexmk_kill_pid(data.pid) let data.pid = 0 endif endfor endfunction " }}}1 " Helper functions for latexmk command function! s:latexmk_set_cmd(data) " {{{1 " Note: We don't send output to /dev/null, but rather to a temporary file, " which allows inspection of latexmk output let tmp = tempname() if has('win32') let cmd = 'cd /D ' . shellescape(a:data.root) let cmd .= ' && set max_print_line=2000 & latexmk' else let cmd = 'cd ' . shellescape(a:data.root) let cmd .= ' && max_print_line=2000 latexmk' endif let cmd .= ' -' . g:latex_latexmk_output let cmd .= ' -verbose ' let cmd .= ' -pvc' let cmd .= ' ' . g:latex_latexmk_options let cmd .= ' -e ' . shellescape('$pdflatex =~ s/ / -file-line-error /') if g:latex_latexmk_callback && has('clientserver') let callback = 'vim --servername ' . v:servername \ . ' --remote-expr \"latex\#latexmk\#errors(0)\"' if has('win32') let cmd .= ' -e "$success_cmd .= ''' . callback . '''"' \ . ' -e "$failure_cmd .= ''' . callback . '''"' else let cmd .= ' -e ''$success_cmd .= "' . callback . '"''' \ . ' -e ''$failure_cmd .= "' . callback . '"''' endif endif let cmd .= ' ' . shellescape(a:data.base) if has('win32') let cmd .= ' >' . tmp let cmd = 'cmd /s /c "' . cmd . '"' else let cmd .= ' &>' . tmp endif let a:data.cmds.compile = cmd let a:data.tmp = tmp endfunction " }}}1 function! s:latexmk_set_pid(data) " {{{1 if has('win32') let tmpfile = tempname() silent execute '!cmd /c "wmic process where ' \ . '(CommandLine LIKE "latexmk\%' . a:data.base . '\%") ' \ . 'get ProcessId /value' \ . '| \%systemroot\%\system32\find "ProcessId" ' \ . '>' . tmpfile . ' "' let pids = readfile(tmpfile) let a:data.pid = strpart(pids[0], 10) else let a:data.pid = system('pgrep -nf "^perl.*latexmk"')[:-2] endif endfunction " }}}1 function! s:latexmk_kill_pid(pid) " {{{1 let exe = {} let exe.bg = 0 let exe.null = 0 if has('win32') let exe.cmd = 'taskkill /PID ' . a:pid . ' /T /F' else let exe.cmd = 'kill ' . a:pid endif call latex#util#execute(exe) endfunction " }}}1 function! s:log_contains_error(logfile) " {{{1 let lines = readfile(a:logfile) let lines = filter(lines, 'v:val =~ ''^.*:\d\+: ''') let lines = uniq(map(lines, 'matchstr(v:val, ''^.*\ze:\d\+:'')')) let lines = map(lines, 'fnameescape(fnamemodify(v:val, '':p''))') let lines = filter(lines, 'filereadable(v:val)') return len(lines) > 0 endfunction function! s:stop_buffer() " {{{1 " " Only run if latex variables are set " if !exists('b:latex') | return | endif let id = b:latex.id let pid = g:latex#data[id].pid " " Only stop if latexmk is running " if pid " " Count the number of buffers that point to current latex blob " let n = 0 for b in filter(range(1, bufnr("$")), 'buflisted(v:val)') if id == getbufvar(b, 'latex', {'id' : -1}).id let n += 1 endif endfor " " Only stop if current buffer is the last for current latex blob " if n == 1 silent call latex#latexmk#stop() endif endif endfunction function! s:system_incompatible() " {{{1 if has('win32') let required = ['latexmk'] else let required = ['latexmk', 'pgrep'] endif " " Check for required executables " for cmd in required if !executable(cmd) echom "Warning: Could not initialize latex#latexmk" echom " Missing executable: " . cmd return 1 endif endfor endfunction " }}}1 " vim: fdm=marker