mirror of
1
0
Fork 0
ultimate-vim/sources_non_forked/vim-go/autoload/go/complete.vim

157 lines
4.6 KiB
VimL
Raw Normal View History

2016-06-11 09:56:50 -04:00
function! s:gocodeCurrentBuffer()
2014-10-31 17:30:24 -04:00
let buf = getline(1, '$')
if &encoding != 'utf-8'
let buf = map(buf, 'iconv(v:val, &encoding, "utf-8")')
endif
if &l:fileformat == 'dos'
" XXX: line2byte() depend on 'fileformat' option.
" so if fileformat is 'dos', 'buf' must include '\r'.
let buf = map(buf, 'v:val."\r"')
endif
let file = tempname()
call writefile(buf, file)
return file
2016-06-11 09:56:50 -04:00
endfunction
2014-10-31 17:30:24 -04:00
2016-06-11 09:56:50 -04:00
function! s:gocodeCommand(cmd, preargs, args)
2014-10-31 17:30:24 -04:00
for i in range(0, len(a:args) - 1)
2016-05-14 07:57:54 -04:00
let a:args[i] = go#util#Shellescape(a:args[i])
2014-10-31 17:30:24 -04:00
endfor
for i in range(0, len(a:preargs) - 1)
2016-05-14 07:57:54 -04:00
let a:preargs[i] = go#util#Shellescape(a:preargs[i])
2014-10-31 17:30:24 -04:00
endfor
2016-06-11 09:56:50 -04:00
let bin_path = go#path#CheckBinPath("gocode")
2015-01-18 07:58:28 -05:00
if empty(bin_path)
return
2014-10-31 17:30:24 -04:00
endif
2015-07-13 06:22:46 -04:00
" we might hit cache problems, as gocode doesn't handle well different
" GOPATHS: https://github.com/nsf/gocode/issues/239
let old_gopath = $GOPATH
let $GOPATH = go#path#Detect()
2016-05-14 07:57:54 -04:00
let result = go#util#System(printf('%s %s %s %s', go#util#Shellescape(bin_path), join(a:preargs), go#util#Shellescape(a:cmd), join(a:args)))
2015-07-13 06:22:46 -04:00
let $GOPATH = old_gopath
2016-05-14 07:57:54 -04:00
if go#util#ShellError() != 0
2014-10-31 17:30:24 -04:00
return "[\"0\", []]"
else
if &encoding != 'utf-8'
let result = iconv(result, 'utf-8', &encoding)
endif
return result
endif
2016-06-11 09:56:50 -04:00
endfunction
2014-10-31 17:30:24 -04:00
2016-06-11 09:56:50 -04:00
function! s:gocodeCurrentBufferOpt(filename)
2014-10-31 17:30:24 -04:00
return '-in=' . a:filename
2016-06-11 09:56:50 -04:00
endfunction
let s:optionsEnabled = 0
function! s:gocodeEnableOptions()
if s:optionsEnabled
return
endif
let bin_path = go#path#CheckBinPath("gocode")
if empty(bin_path)
return
endif
let s:optionsEnabled = 1
call go#util#System(printf('%s set propose-builtins %s', go#util#Shellescape(bin_path), s:toBool(get(g:, 'go_gocode_propose_builtins', 1))))
call go#util#System(printf('%s set autobuild %s', go#util#Shellescape(bin_path), s:toBool(get(g:, 'go_gocode_autobuild', 1))))
endfunction
function! s:toBool(val)
if a:val | return 'true ' | else | return 'false' | endif
endfunction
function! s:gocodeAutocomplete()
call s:gocodeEnableOptions()
2014-10-31 17:30:24 -04:00
let filename = s:gocodeCurrentBuffer()
let result = s:gocodeCommand('autocomplete',
\ [s:gocodeCurrentBufferOpt(filename), '-f=vim'],
2016-03-20 14:01:44 -04:00
\ [expand('%:p'), go#util#OffsetCursor()])
2014-10-31 17:30:24 -04:00
call delete(filename)
return result
2016-06-11 09:56:50 -04:00
endfunction
2014-10-31 17:30:24 -04:00
2016-03-20 14:01:44 -04:00
function! go#complete#GetInfo()
let offset = go#util#OffsetCursor()+1
2014-10-31 17:30:24 -04:00
let filename = s:gocodeCurrentBuffer()
let result = s:gocodeCommand('autocomplete',
\ [s:gocodeCurrentBufferOpt(filename), '-f=godit'],
2016-03-20 14:01:44 -04:00
\ [expand('%:p'), offset])
2014-10-31 17:30:24 -04:00
call delete(filename)
" first line is: Charcount,,NumberOfCandidates, i.e: 8,,1
" following lines are candiates, i.e: func foo(name string),,foo(
let out = split(result, '\n')
" no candidates are found
if len(out) == 1
2015-01-18 07:58:28 -05:00
return ""
2014-10-31 17:30:24 -04:00
endif
" only one candiate is found
if len(out) == 2
return split(out[1], ',,')[0]
endif
" to many candidates are available, pick one that maches the word under the
" cursor
let infos = []
for info in out[1:]
call add(infos, split(info, ',,')[0])
endfor
let wordMatch = '\<' . expand("<cword>") . '\>'
" escape single quotes in wordMatch before passing it to filter
let wordMatch = substitute(wordMatch, "'", "''", "g")
let filtered = filter(infos, "v:val =~ '".wordMatch."'")
if len(filtered) == 1
return filtered[0]
endif
2015-01-18 07:58:28 -05:00
return ""
2014-10-31 17:30:24 -04:00
endfunction
2016-03-14 06:04:57 -04:00
function! go#complete#Info(auto)
" auto is true if we were called by g:go_auto_type_info's autocmd
2014-10-31 17:30:24 -04:00
let result = go#complete#GetInfo()
2015-01-18 07:58:28 -05:00
if !empty(result)
2016-03-14 06:04:57 -04:00
" if auto, and the result is a PANIC by gocode, hide it
if a:auto && result ==# 'PANIC PANIC PANIC' | return | endif
2014-10-31 17:30:24 -04:00
echo "vim-go: " | echohl Function | echon result | echohl None
endif
2015-01-18 07:58:28 -05:00
endfunction
2014-10-31 17:30:24 -04:00
2015-03-14 16:02:10 -04:00
function! s:trim_bracket(val)
let a:val.word = substitute(a:val.word, '[(){}\[\]]\+$', '', '')
return a:val
endfunction
2016-06-11 09:56:50 -04:00
function! go#complete#Complete(findstart, base)
2014-10-31 17:30:24 -04:00
"findstart = 1 when we need to get the text length
if a:findstart == 1
execute "silent let g:gocomplete_completions = " . s:gocodeAutocomplete()
return col('.') - g:gocomplete_completions[0] - 1
"findstart = 0 when we need to return the list of completions
else
2015-03-14 16:02:10 -04:00
let s = getline(".")[col('.') - 1]
if s =~ '[(){}\{\}]'
return map(copy(g:gocomplete_completions[1]), 's:trim_bracket(v:val)')
endif
2014-10-31 17:30:24 -04:00
return g:gocomplete_completions[1]
endif
endf
" vim:ts=4:sw=4:et