vimrc/sources_non_forked/vim-go/autoload/go/impl.vim

127 lines
3.2 KiB
VimL
Raw Normal View History

2016-12-27 14:46:49 +00:00
function! go#impl#Impl(...) abort
2016-06-26 11:12:36 +00:00
let binpath = go#path#CheckBinPath('impl')
if empty(binpath)
return
endif
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
let recv = ""
let iface = ""
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
if a:0 == 0
" user didn't passed anything, just called ':GoImpl'
let receiveType = expand("<cword>")
let recv = printf("%s *%s", tolower(receiveType)[0], receiveType)
let iface = input("vim-go: generating method stubs for interface: ")
redraw!
if empty(iface)
call go#util#EchoError('usage: interface type is not provided')
return
2016-05-14 11:57:54 +00:00
endif
2016-06-26 11:12:36 +00:00
elseif a:0 == 1
" we assume the user only passed the interface type,
" i.e: ':GoImpl io.Writer'
let receiveType = expand("<cword>")
let recv = printf("%s *%s", tolower(receiveType)[0], receiveType)
let iface = a:1
elseif a:0 > 2
" user passed receiver and interface type both,
" i.e: 'GoImpl f *Foo io.Writer'
let recv = join(a:000[:-2], ' ')
let iface = a:000[-1]
else
call go#util#EchoError('usage: GoImpl {receiver} {interface}')
return
endif
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
let result = go#util#System(printf("%s '%s' '%s'", binpath, recv, iface))
if go#util#ShellError() != 0
call go#util#EchoError(result)
return
endif
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
if result ==# ''
return
end
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
let pos = getpos('.')
put =''
put =result
call setpos('.', pos)
2016-05-14 11:57:54 +00:00
endfunction
if exists('*uniq')
2016-06-26 11:12:36 +00:00
function! s:uniq(list)
return uniq(a:list)
endfunction
2016-05-14 11:57:54 +00:00
else
2016-06-26 11:12:36 +00:00
" Note: Believe that the list is sorted
function! s:uniq(list)
let i = len(a:list) - 1
while 0 < i
if a:list[i-1] ==# a:list[i]
call remove(a:list, i)
let i -= 2
else
let i -= 1
endif
endwhile
return a:list
endfunction
2016-05-14 11:57:54 +00:00
endif
2016-12-27 14:46:49 +00:00
function! s:root_dirs() abort
2016-06-26 11:12:36 +00:00
let dirs = []
2016-08-02 12:48:32 +00:00
let root = go#util#goroot()
2016-06-26 11:12:36 +00:00
if root !=# '' && isdirectory(root)
call add(dirs, root)
endif
2016-05-14 11:57:54 +00:00
2016-08-02 12:48:32 +00:00
let paths = map(split(go#util#gopath(), go#util#PathListSep()), "substitute(v:val, '\\\\', '/', 'g')")
2016-06-26 11:12:36 +00:00
if go#util#ShellError()
return []
endif
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
if !empty(filter(paths, 'isdirectory(v:val)'))
call extend(dirs, paths)
endif
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
return dirs
2016-05-14 11:57:54 +00:00
endfunction
2016-12-27 14:46:49 +00:00
function! s:go_packages(dirs) abort
2016-06-26 11:12:36 +00:00
let pkgs = []
for d in a:dirs
2016-08-02 12:48:32 +00:00
let pkg_root = expand(d . '/pkg/' . go#util#osarch())
2016-06-26 11:12:36 +00:00
call extend(pkgs, split(globpath(pkg_root, '**/*.a', 1), "\n"))
endfor
return map(pkgs, "fnamemodify(v:val, ':t:r')")
2016-05-14 11:57:54 +00:00
endfunction
2016-12-27 14:46:49 +00:00
function! s:interface_list(pkg) abort
2016-06-26 11:12:36 +00:00
let contents = split(go#util#System('go doc ' . a:pkg), "\n")
if go#util#ShellError()
return []
endif
2016-05-14 11:57:54 +00:00
2016-06-26 11:12:36 +00:00
call filter(contents, 'v:val =~# ''^type\s\+\h\w*\s\+interface''')
return map(contents, 'a:pkg . "." . matchstr(v:val, ''^type\s\+\zs\h\w*\ze\s\+interface'')')
2016-05-14 11:57:54 +00:00
endfunction
" Complete package and interface for {interface}
2016-12-27 14:46:49 +00:00
function! go#impl#Complete(arglead, cmdline, cursorpos) abort
2016-06-26 11:12:36 +00:00
let words = split(a:cmdline, '\s\+', 1)
if words[-1] ==# ''
return s:uniq(sort(s:go_packages(s:root_dirs())))
elseif words[-1] =~# '^\h\w*$'
return s:uniq(sort(filter(s:go_packages(s:root_dirs()), 'stridx(v:val, words[-1]) == 0')))
elseif words[-1] =~# '^\h\w*\.\%(\h\w*\)\=$'
let [pkg, interface] = split(words[-1], '\.', 1)
echomsg pkg
return s:uniq(sort(filter(s:interface_list(pkg), 'v:val =~? words[-1]')))
else
return []
endif
2016-05-14 11:57:54 +00:00
endfunction
2016-06-26 11:12:36 +00:00
" vim: sw=2 ts=2 et