mirror of
				https://github.com/neovim/neovim.git
				synced 2025-10-26 12:27:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			152 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			VimL
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			VimL
		
	
	
	
	
	
| 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] = ['<cword>', a:1]
 | |
|   else
 | |
|     let [page, sect] = [a:1, '']
 | |
|   endif
 | |
| 
 | |
|   if page == '<cword>'
 | |
|     let page = expand('<cword>')
 | |
|   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 . "\<C-W>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
 | 
