.dotfiles/vim/bundle/autoproto/plugin/autoproto.vim
2014-05-29 23:58:08 +02:00

753 lines
14 KiB
VimL

" GPL Notice:
"
" This program is free software; you can redistribute it and/or modify
" it under the terms of the GNU General Public License as published by
" the Free Software Foundation; either version 2 of the License, or
" (at your option) any later version.
"
" This program is distributed in the hope that it will be useful,
" but WITHOUT ANY WARRANTY; without even the implied warranty of
" MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
" GNU Library General Public License for more details.
"
" You should have received a copy of the GNU General Public License
" along with this program; if not, write to the Free Software
" Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
"
"
" Name:
"
" autoproto.vim
"
"
" Copyright:
"
" Jochen Baier, 2006 (email@Jochen-Baier.de)
"
"
" Based on:
"
" cream-pop.vim
" CursorHold example
"
" Version: 0.06
" Last Modified: Jun 20, 2006
"
"
" The "autoproto" plugin displays the function parameter/prototypes
" in the preview window if you type a left parenthesis behind
" a function name.
" With this plugin it is not necessary to memorize large parameter lists.
" You do not need to leave insert mode. Typing is almost not affected.
"
"
" Supported Languages: C
"
" Installation:
"
" * Drop autoproto.vim into your plugin directory
"
" * The script need a tag file to work:
" - for your project: ":!ctags -R ." (or use the gvim button)
" - for library functions:
" run: "ctags -R -f ~/.vim/systags --c-kinds=+p /usr/include /usr/local/include"
" put "set tags+=~/.vim/systags" in your .vimrc file.
"
" * Open some *.c file and write for example: "XCreateWindow ("
" (or "XCreateWindow(").
" If the tag exist it will be shown in the preview window.
" Note: only *.c files are supported ! The file must exist.
"
"
function! Debug (...)
"remove for debug
return
let i = 1
let msg= ""
while i <= a:0
exe "let msg=msg.a:".i
if i < a:0
let msg=msg." "
endif
let i=i+1
endwhile
call Decho (msg)
endfunction
function! Rm_leading_spaces (string)
let first_letter=0
while 1==1
if a:string[first_letter] != " "
break
else
let first_letter=first_letter + 1
endif
if first_letter >= strlen(a:string)
call Debug ("leading spaces problem -> skip")
return 0
endif
endwhile
return strpart(a:string, first_letter)
endfunction
function Rm_trailing_spaces (string)
let line=a:string
let string_len = strlen (line)
let last_letter=string_len - 1
let space_count = 0
while 1==1
if line[last_letter] != " "
break
else
let last_letter=last_letter - 1
let space_count = space_count + 1
call Debug ("space_count: " . space_count)
if space_count > 1
call Debug ("too many trailing spaces -> skip")
return 0
endif
endif
if last_letter <= 1
call Debug ("trailing spaces problem -> skip")
return 0
endif
endwhile
return strpart(line, 0, last_letter + 1)
endfunction
function! Chop_it (da_line)
let line=a:da_line
let line=Rm_trailing_spaces (line)
if line == "0"
return 0
endif
call Debug ("after remove trailing spaces: <" . line. ">" )
let line=Rm_leading_spaces(line)
if line == "0"
return 0
endif
"could be inside a comment
if strpart(line, 0, 2) == "/*"
call Debug ("large comment found")
return 0
endif
if strpart(line, 0, 2) == "//"
call Debug ("line comment found")
return 0
endif
"semicolon ?
let semi=strridx (line, ";")
if semi != -1
let line=strpart(line, semi+1)
call Debug("after last semi: <" . line . ">")
endif
let line=Rm_leading_spaces(line)
call Debug("line after leading_spaces after semi: <" . line . ">")
if line == "0"
return 0
endif
call Debug ("cleand line: <" . line . ">")
"could be a new function definition...
if strridx (line, "=") == -1
if strpart(line, 0, 6) == "static"
call Debug ("static found")
return 0
endif
if strpart(line, 0, 4) == "void"
call Debug ("void found")
return 0
endif
if strpart(line, 0, 6) == "extern"
call Debug ("extern found")
return 0
endif
if strpart(line, 0, 8) == "volatile"
call Debug ("volatile found")
return 0
endif
if strpart(line, 0, 6) == "inline"
call Debug ("inline found")
return 0
endif
if strpart(line, 0, 6) == "struct"
call Debug ("inline found")
return 0
endif
if strpart(line, 0, 8) == "unsigned"
call Debug ("unsigned found")
return 0
endif
if strpart(line, 0, 3) == "int"
call Debug ("int found")
return 0
endif
if strpart(line, 0, 4) == "long"
call Debug ("long found")
return 0
endif
if strpart(line, 0, 5) == "fload"
call Debug ("fload found")
return 0
endif
if strpart(line, 0, 4) == "char"
call Debug ("char found")
return 0
endif
endif
"comma ?
let comma=strridx (line, ",")
if comma != -1
let line=strpart(line, comma+1)
call Debug("after comma: <" . line . ">")
endif
let line=Rm_leading_spaces(line)
if line == "0"
return 0
endif
"could be inside a string
if stridx (line, "\"") != -1
call Debug("probably inside a string")
return 0
endif
"last space
let last_blank= strridx(line, " ")
if last_blank != -1
call Debug ("last_blank found")
let line = strpart(line, last_blank + 1)
endif
call Debug ("last_string:<" . line . ">")
if strlen (line) == 0
call Debug ("last_string zero length -> skip")
return 0
endif
"signs..
let start_pos = -1
let tmp= strridx(line, "!")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "=")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "{")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "+")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "-")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, ">")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "<")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, ":")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "?")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, ")")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "(")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "&")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "|")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "/")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "%")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "~")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "\"")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
let tmp= strridx(line, "*")
if tmp != -1
if tmp > start_pos
let start_pos=tmp
endif
endif
if start_pos >= (strlen (line)-1)
call Debug("start_pos >= strlen")
return 0
endif
call Debug ("start_pos: " . start_pos)
let line = strpart(line, start_pos+1)
call Debug("last word:<". line . ">")
let word_len =strlen (line)
call Debug ("word len:" . word_len)
if word_len == 0
call Debug ("zero length word ->skip")
return 0
endif
while 1==1
if line == "if"
return 0
endif
if line == "for"
return 0
endif
if line == "while"
return 0
endif
if line == "switch"
return 0
endif
if line == "sizeof"
return 0
endif
if line == "int"
return 0
endif
if line == "long"
return 0
endif
if line == "float"
return 0
endif
if line == "char"
return 0
endif
if line == "return"
return 0
endif
break
endwhile
return line
endfunction
function! Display_tag (tag)
silent! wincmd P
if &previewwindow
match none
wincmd p
endif
let ignorec=&ignorecase
if ignorec==1
call Debug ("ignorecase is set")
set noignorecase
endif
try
exe "ptag " . a:tag
catch
call Debug ("ptag failed !")
if ignorec == 1
set ignorecase
endif
return 0
endtry
if ignorec == 1
set ignorecase
endif
silent! wincmd P
if &previewwindow
if has("folding")
silent! .foldopen
endif
let cur_pos=winline()
if cur_pos <= 1
let cur_pos = 0
else
let cur_pos=cur_pos -1
endif
call search("$", "b")
let w = substitute(a:tag, '\\', '\\\\', "")
call search('\<\V' . a:tag . '\>')
hi previewWord term=bold ctermbg=green guibg=green
exe 'match previewWord "\%' . line(".") . 'l\%' . col(".") . 'c\k*"'
if cur_pos != 0
execute "normal " . cur_pos . "\<C-e>"
endif
wincmd p
endif
return 1
endfunction
function! Semi_typed()
call Debug ("Semi_typed, reset all")
call Unmap_semi()
call Unmap_close_bracket()
if exists ("b:recent_braces")
unlet b:recent_braces
endif
endfunction
function! Unmap_close_bracket()
try
iunmap <silent> <buffer> )
catch
call Debug ("umap close bracket failed but catched!")
endtry
endfunction
function! Map_close_bracket()
try
inoremap <silent> <buffer> ) )<ESC>:call Close_bracket()<CR>li
catch
call Debug ("map close bracket failed")
endtry
endfunction
function! Map_semi()
try
inoremap <silent> <buffer>; ;<C-o>:call Semi_typed()<CR>
catch
call Debug ("mapping semi failed")
endtry
endfunction
function! Unmap_semi()
try
iunmap <silent> <buffer> ;
catch
call Debug ("unmap semi failed")
endtry
endfunction
function! Close_bracket()
call Debug ("close bracket")
if !exists ("b:recent_braces")
call Debug ("no recent_braces in close_bracket")
call Unmap_close_bracket ()
return
endif
execute "normal %"
let brace_pos=getpos(".")
execute "normal %"
let brace_line=brace_pos[1]
let brace_column=brace_pos[2]
call Debug ("found matching brace at: brace_line: " . brace_line . " brace_column: " . brace_column )
let tag_iter = 0
let tag_index = -1
for list_line in b:recent_braces
if list_line[1] == brace_line
if list_line[2] == brace_column
let tag_index=tag_iter
break
endif
endif
let tag_iter = tag_iter +1
endfor
if tag_index != -1
call Debug ("matchin tag: <" . b:recent_braces[tag_index][0] . ">" )
else
call Debug ("no matching tag found")
return
endif
if tag_index == 0
call Debug ("last recent tag: delete recent_braces list, unmap")
unlet b:recent_braces
call Unmap_close_bracket()
silent! wincmd P
if &previewwindow
match none
q
normal a
endif
return
endif
let tag_to_display = b:recent_braces[tag_index-1][0]
call Debug ("tag_to_display: " . tag_to_display)
call remove (b:recent_braces, tag_index)
call Display_tag (tag_to_display)
endfunction
function! Open_bracket (...)
let brace_pos=getpos(".")
call Debug ("function: Open_bracket")
let line=getline (".")
let line_len=strlen (line)
call Debug ("line: <" . line . ">")
call Debug ("line_len: " . line_len)
call Debug ("brace_pos[2]:" . brace_pos[2])
let cursor_column=brace_pos[2]
if line_len < 1
call Debug ("line too short -> skip")
return 0
endif
if line_len == 1
if line == " "
call Debug ("line is only one blank --> skip")
return
endif
endif
let eol=1
if line_len > cursor_column
let line=strpart (line, 0, cursor_column-1)
"call Semi_typed ()
let eol=0
call Debug ("line before cursor: <" . line . ">" )
endif
let line=Chop_it (line)
if line == "0"
call Debug ("tag is 0 -> skip")
return
endif
if strlen (line) < 1
call Debug ("tag has zero length -> skip")
return
endif
call Debug ("tag is:<" . line . ">")
if line =~ '\a'
call Debug ("letters...")
else
call Debug ("no letters found -> skip")
return
endif
if Display_tag (line) != "0"
let brace_line=brace_pos[1]
let brace_column=brace_pos[2] + eol
call Debug ("brace_line: " . brace_line . " brace_column: " . brace_column )
if !exists ("b:recent_braces")
call Debug ("no recent_braces")
let b:recent_braces=[[line, brace_line, brace_column]]
call Map_close_bracket()
call Map_semi()
else
call Debug ("resent_braces exist")
call add (b:recent_braces, [line, brace_line, brace_column])
endif
endif
endfunction
function! Map_comma (...)
inoremap <silent> <buffer> ( <C-o>:call Open_bracket()<CR>(
endfunction
autocmd BufRead *.c call Map_comma()