let s:man_tag_depth = 0 let s:man_sect_arg = '' let s:man_find_arg = '-w' try if !has('win32') && $OSTYPE !~? 'cygwin\|linux' && system('uname -s') =~? 'SunOS' && system('uname -r') =~? '^5' let s:man_sect_arg = '-s' let s:man_find_arg = '-l' endif catch /E145:/ " Ignore the error in restricted mode endtry function man#get_page(...) abort let invoked_from_man = (&filetype ==# 'man') if a:0 == 0 echoerr 'argument required' return elseif a:0 > 2 echoerr 'too many arguments' return elseif a:0 == 2 let [page, sect] = [a:2, 0 + a:1] elseif type(1) == type(a:1) let [page, sect] = ['', a:1] else let [page, sect] = [a:1, ''] endif if page == '' let page = expand('') endif let [page, sect] = s:parse_page_and_section(sect, page) if 0 + sect > 0 && s:find_page(sect, page) == 0 let sect = '' endif if s:find_page(sect, page) == 0 echo 'No manual entry for '.page return endif exec 'let s:man_tag_buf_'.s:man_tag_depth.' = '.bufnr('%') exec 'let s:man_tag_lin_'.s:man_tag_depth.' = '.line('.') exec 'let s:man_tag_col_'.s:man_tag_depth.' = '.col('.') let s:man_tag_depth = s:man_tag_depth + 1 let editcmd = 'edit' " Use an existing 'man' window, else open a new one. if &filetype !=# 'man' let thiswin = winnr() wincmd b if winnr() > 1 exe "norm! " . thiswin . "\w" while 1 if &filetype == 'man' break endif wincmd w if thiswin == winnr() break endif endwhile endif if &filetype !=# 'man' let editcmd = 'tabnew' endif endif silent exec editcmd.' man://'.page.(empty(sect)?'':'('.sect.')') setlocal modifiable silent keepjumps norm! 1G"_dG if empty($MANWIDTH) let $MANWIDTH = winwidth(0) endif silent exec 'r!/usr/bin/man '.s:cmd(sect, page).' | col -b' " Remove blank lines from top and bottom. while getline(1) =~ '^\s*$' silent keepjumps norm! gg"_dd endwhile while getline('$') =~ '^\s*$' silent keepjumps norm! G"_dd endwhile setlocal nomodified setlocal filetype=man if invoked_from_man || editcmd ==# 'tabnew' call s:set_window_local_options() endif endfunction function s:set_window_local_options() abort setlocal colorcolumn=0 foldcolumn=0 nonumber setlocal nolist norelativenumber nofoldenable endfunction function man#pop_page() abort if s:man_tag_depth > 0 let s:man_tag_depth = s:man_tag_depth - 1 exec "let s:man_tag_buf=s:man_tag_buf_".s:man_tag_depth exec "let s:man_tag_lin=s:man_tag_lin_".s:man_tag_depth exec "let s:man_tag_col=s:man_tag_col_".s:man_tag_depth exec s:man_tag_buf."b" exec s:man_tag_lin exec "norm! ".s:man_tag_col."|" exec "unlet s:man_tag_buf_".s:man_tag_depth exec "unlet s:man_tag_lin_".s:man_tag_depth exec "unlet s:man_tag_col_".s:man_tag_depth unlet s:man_tag_buf s:man_tag_lin s:man_tag_col endif endfunction " Expects a string like 'access' or 'access(2)'. function s:parse_page_and_section(sect, str) abort try let save_isk = &iskeyword setlocal iskeyword-=(,) let page = substitute(a:str, '(*\(\k\+\).*', '\1', '') let sect = substitute(a:str, '\(\k\+\)(\([^()]*\)).*', '\2', '') if sect == page || -1 == match(sect, '^[0-9 ]\+$') let sect = a:sect endif catch let &l:iskeyword = save_isk echoerr 'man.vim: failed to parse: "'.a:str.'"' endtry return [page, sect] endfunction function s:cmd(sect, page) abort if 0 + a:sect > 0 return s:man_sect_arg.' '.a:sect.' '.a:page endif return a:page endfunction function s:find_page(sect, page) abort let where = system('/usr/bin/man '.s:man_find_arg.' '.s:cmd(a:sect, a:page)) if where !~ "^/" if matchstr(where, " [^ ]*$") !~ "^ /" return 0 endif endif return 1 endfunction