summaryrefslogtreecommitdiffstats
path: root/autoload/vimtex/view/mupdf.vim
diff options
context:
space:
mode:
Diffstat (limited to 'autoload/vimtex/view/mupdf.vim')
-rw-r--r--autoload/vimtex/view/mupdf.vim186
1 files changed, 186 insertions, 0 deletions
diff --git a/autoload/vimtex/view/mupdf.vim b/autoload/vimtex/view/mupdf.vim
new file mode 100644
index 00000000..95d3710c
--- /dev/null
+++ b/autoload/vimtex/view/mupdf.vim
@@ -0,0 +1,186 @@
+if !exists('g:polyglot_disabled') || index(g:polyglot_disabled, 'latex') == -1
+
+" vimtex - LaTeX plugin for Vim
+"
+" Maintainer: Karl Yngve LervÄg
+" Email: karl.yngve@gmail.com
+"
+
+function! vimtex#view#mupdf#new() abort " {{{1
+ " Check if the viewer is executable
+ if !executable('mupdf')
+ call vimtex#log#error(
+ \ 'MuPDF is not executable!',
+ \ '- vimtex viewer will not work!')
+ return {}
+ endif
+
+ " Use the xwin template
+ return vimtex#view#common#apply_xwin_template('MuPDF',
+ \ vimtex#view#common#apply_common_template(deepcopy(s:mupdf)))
+endfunction
+
+" }}}1
+
+let s:mupdf = {
+ \ 'name': 'MuPDF',
+ \}
+
+function! s:mupdf.start(outfile) dict abort " {{{1
+ let l:cmd = 'mupdf ' . g:vimtex_view_mupdf_options
+ \ . ' ' . vimtex#util#shellescape(a:outfile)
+ let self.process = vimtex#process#start(l:cmd)
+
+ call self.xwin_get_id()
+ call self.xwin_send_keys(g:vimtex_view_mupdf_send_keys)
+
+ if g:vimtex_view_forward_search_on_start
+ call self.forward_search(a:outfile)
+ endif
+endfunction
+
+" }}}1
+function! s:mupdf.forward_search(outfile) dict abort " {{{1
+ if !executable('xdotool') | return | endif
+ if !executable('synctex') | return | endif
+
+ let self.cmd_synctex_view = 'synctex view -i '
+ \ . (line('.') + 1) . ':'
+ \ . (col('.') + 1) . ':'
+ \ . vimtex#util#shellescape(expand('%:p'))
+ \ . ' -o ' . vimtex#util#shellescape(a:outfile)
+ \ . " | grep -m1 'Page:' | sed 's/Page://' | tr -d '\n'"
+ let self.page = system(self.cmd_synctex_view)
+
+ if self.page > 0
+ let l:cmd = 'xdotool'
+ \ . ' type --window ' . self.xwin_id
+ \ . ' "' . self.page . 'g"'
+ call vimtex#process#run(l:cmd)
+ let self.cmd_forward_search = l:cmd
+ endif
+
+ call self.focus_viewer()
+endfunction
+
+" }}}1
+function! s:mupdf.reverse_search() dict abort " {{{1
+ if !executable('xdotool') | return | endif
+ if !executable('synctex') | return | endif
+
+ let outfile = self.out()
+ if vimtex#view#common#not_readable(outfile) | return | endif
+
+ if !self.xwin_exists()
+ call vimtex#log#warning('Reverse search failed (is MuPDF open?)')
+ return
+ endif
+
+ " Get page number
+ let self.cmd_getpage = 'xdotool getwindowname ' . self.xwin_id
+ let self.cmd_getpage .= " | sed 's:.* - \\([0-9]*\\)/.*:\\1:'"
+ let self.cmd_getpage .= " | tr -d '\n'"
+ let self.page = system(self.cmd_getpage)
+ if self.page <= 0 | return | endif
+
+ " Get file
+ let self.cmd_getfile = 'synctex edit '
+ let self.cmd_getfile .= "-o \"" . self.page . ':288:108:' . outfile . "\""
+ let self.cmd_getfile .= "| grep 'Input:' | sed 's/Input://' "
+ let self.cmd_getfile .= "| head -n1 | tr -d '\n' 2>/dev/null"
+ let self.file = system(self.cmd_getfile)
+
+ " Get line
+ let self.cmd_getline = 'synctex edit '
+ let self.cmd_getline .= "-o \"" . self.page . ':288:108:' . outfile . "\""
+ let self.cmd_getline .= "| grep -m1 'Line:' | sed 's/Line://' "
+ let self.cmd_getline .= "| head -n1 | tr -d '\n'"
+ let self.line = system(self.cmd_getline)
+
+ " Go to file and line
+ silent exec 'edit ' . fnameescape(self.file)
+ if self.line > 0
+ silent exec ':' . self.line
+ " Unfold, move to top line to correspond to top pdf line, and go to end of
+ " line in case the corresponding pdf line begins on previous pdf page.
+ normal! zvztg_
+ endif
+endfunction
+
+" }}}1
+function! s:mupdf.compiler_callback(status) dict abort " {{{1
+ if !a:status && g:vimtex_view_use_temp_files < 2
+ return
+ endif
+
+ if g:vimtex_view_use_temp_files
+ call self.copy_files()
+ endif
+
+ if !filereadable(self.out()) | return | endif
+
+ if g:vimtex_view_automatic
+ "
+ " Search for existing window created by latexmk
+ " It may be necessary to wait some time before it is opened and
+ " recognized. Sometimes it is very quick, other times it may take
+ " a second. This way, we don't block longer than necessary.
+ "
+ if !has_key(self, 'started_through_callback')
+ for l:dummy in range(30)
+ sleep 50m
+ if self.xwin_exists() | break | endif
+ endfor
+ endif
+
+ if !self.xwin_exists() && !has_key(self, 'started_through_callback')
+ call self.start(self.out())
+ let self.started_through_callback = 1
+ endif
+ endif
+
+ if g:vimtex_view_use_temp_files || get(b:vimtex.compiler, 'callback')
+ call self.xwin_send_keys('r')
+ endif
+
+ if has_key(self, 'hook_callback')
+ call self.hook_callback()
+ endif
+endfunction
+
+" }}}1
+function! s:mupdf.latexmk_append_argument() dict abort " {{{1
+ if g:vimtex_view_use_temp_files
+ let cmd = ' -view=none'
+ else
+ let cmd = vimtex#compiler#latexmk#wrap_option('new_viewer_always', '0')
+ let cmd .= vimtex#compiler#latexmk#wrap_option('pdf_update_method', '2')
+ let cmd .= vimtex#compiler#latexmk#wrap_option('pdf_update_signal', 'SIGHUP')
+ let cmd .= vimtex#compiler#latexmk#wrap_option('pdf_previewer',
+ \ 'mupdf ' . g:vimtex_view_mupdf_options)
+ endif
+
+ return cmd
+endfunction
+
+" }}}1
+function! s:mupdf.focus_viewer() dict abort " {{{1
+ if !executable('xdotool') | return | endif
+
+ if self.xwin_id > 0
+ silent call system('xdotool windowactivate ' . self.xwin_id . ' --sync')
+ silent call system('xdotool windowraise ' . self.xwin_id)
+ endif
+endfunction
+
+" }}}1
+function! s:mupdf.focus_vim() dict abort " {{{1
+ if !executable('xdotool') | return | endif
+
+ silent call system('xdotool windowactivate ' . v:windowid . ' --sync')
+ silent call system('xdotool windowraise ' . v:windowid)
+endfunction
+
+" }}}1
+
+endif