123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313 |
- let s:go_stack = []
- let s:go_stack_level = 0
- function! go#def#Jump(mode) abort
- let fname = fnamemodify(expand("%"), ':p:gs?\\?/?')
- " so guru right now is slow for some people. previously we were using
- " godef which also has it's own quirks. But this issue come up so many
- " times I've decided to support both. By default we still use guru as it
- " covers all edge cases, but now anyone can switch to godef if they wish
- let bin_name = get(g:, 'go_def_mode', 'guru')
- if bin_name == 'godef'
- if &modified
- " Write current unsaved buffer to a temp file and use the modified content
- let l:tmpname = tempname()
- call writefile(go#util#GetLines(), l:tmpname)
- let fname = l:tmpname
- endif
- let bin_path = go#path#CheckBinPath("godef")
- if empty(bin_path)
- return
- endif
- let command = printf("%s -f=%s -o=%s -t", go#util#Shellescape(bin_path),
- \ go#util#Shellescape(fname), go#util#OffsetCursor())
- let out = go#util#System(command)
- if exists("l:tmpname")
- call delete(l:tmpname)
- endif
- elseif bin_name == 'guru'
- let bin_path = go#path#CheckBinPath("guru")
- if empty(bin_path)
- return
- endif
- let cmd = [bin_path]
- let stdin_content = ""
- if &modified
- let content = join(go#util#GetLines(), "\n")
- let stdin_content = fname . "\n" . strlen(content) . "\n" . content
- call add(cmd, "-modified")
- endif
- if exists('g:go_build_tags')
- let tags = get(g:, 'go_build_tags')
- call extend(cmd, ["-tags", tags])
- endif
- let fname = fname.':#'.go#util#OffsetCursor()
- call extend(cmd, ["definition", fname])
- if go#util#has_job()
- let l:spawn_args = {
- \ 'cmd': cmd,
- \ 'complete': function('s:jump_to_declaration_cb', [a:mode, bin_name]),
- \ }
- if &modified
- let l:spawn_args.input = stdin_content
- endif
- call go#util#EchoProgress("searching declaration ...")
- call s:def_job(spawn_args)
- return
- endif
- let command = join(cmd, " ")
- if &modified
- let out = go#util#System(command, stdin_content)
- else
- let out = go#util#System(command)
- endif
- else
- call go#util#EchoError('go_def_mode value: '. bin_name .' is not valid. Valid values are: [godef, guru]')
- return
- endif
- if go#util#ShellError() != 0
- call go#util#EchoError(out)
- return
- endif
- call go#def#jump_to_declaration(out, a:mode, bin_name)
- endfunction
- function! s:jump_to_declaration_cb(mode, bin_name, job, exit_status, data) abort
- if a:exit_status != 0
- return
- endif
- call go#def#jump_to_declaration(a:data[0], a:mode, a:bin_name)
- call go#util#EchoSuccess(fnamemodify(a:data[0], ":t"))
- endfunction
- function! go#def#jump_to_declaration(out, mode, bin_name) abort
- let final_out = a:out
- if a:bin_name == "godef"
- " append the type information to the same line so our we can parse it.
- " This makes it compatible with guru output.
- let final_out = join(split(a:out, '\n'), ':')
- endif
- " strip line ending
- let out = split(final_out, go#util#LineEnding())[0]
- if go#util#IsWin()
- let parts = split(out, '\(^[a-zA-Z]\)\@<!:')
- else
- let parts = split(out, ':')
- endif
- let filename = parts[0]
- let line = parts[1]
- let col = parts[2]
- let ident = parts[3]
- " Remove anything newer than the current position, just like basic
- " vim tag support
- if s:go_stack_level == 0
- let s:go_stack = []
- else
- let s:go_stack = s:go_stack[0:s:go_stack_level-1]
- endif
- " increment the stack counter
- let s:go_stack_level += 1
- " push it on to the jumpstack
- let stack_entry = {'line': line("."), 'col': col("."), 'file': expand('%:p'), 'ident': ident}
- call add(s:go_stack, stack_entry)
- " needed for restoring back user setting this is because there are two
- " modes of switchbuf which we need based on the split mode
- let old_switchbuf = &switchbuf
- normal! m'
- if filename != fnamemodify(expand("%"), ':p:gs?\\?/?')
- " jump to existing buffer if, 1. we have enabled it, 2. the buffer is loaded
- " and 3. there is buffer window number we switch to
- if get(g:, 'go_def_reuse_buffer', 0) && bufloaded(filename) != 0 && bufwinnr(filename) != -1
- " jumpt to existing buffer if it exists
- execute bufwinnr(filename) . 'wincmd w'
- else
- if &modified
- let cmd = 'hide edit'
- else
- let cmd = 'edit'
- endif
- if a:mode == "tab"
- let &switchbuf = "useopen,usetab,newtab"
- if bufloaded(filename) == 0
- tab split
- else
- let cmd = 'sbuf'
- endif
- elseif a:mode == "split"
- split
- elseif a:mode == "vsplit"
- vsplit
- endif
- " open the file and jump to line and column
- exec cmd fnameescape(fnamemodify(filename, ':.'))
- endif
- endif
- call cursor(line, col)
- " also align the line to middle of the view
- normal! zz
- let &switchbuf = old_switchbuf
- endfunction
- function! go#def#SelectStackEntry() abort
- let target_window = go#ui#GetReturnWindow()
- if empty(target_window)
- let target_window = winnr()
- endif
- let highlighted_stack_entry = matchstr(getline("."), '^..\zs\(\d\+\)')
- if !empty(highlighted_stack_entry)
- execute target_window . "wincmd w"
- call go#def#Stack(str2nr(highlighted_stack_entry))
- endif
- call go#ui#CloseWindow()
- endfunction
- function! go#def#StackUI() abort
- if len(s:go_stack) == 0
- call go#util#EchoError("godef stack empty")
- return
- endif
- let stackOut = ['" <Up>,<Down>:navigate <Enter>:jump <Esc>,q:exit']
- let i = 0
- while i < len(s:go_stack)
- let entry = s:go_stack[i]
- let prefix = ""
- if i == s:go_stack_level
- let prefix = ">"
- else
- let prefix = " "
- endif
- call add(stackOut, printf("%s %d %s|%d col %d|%s",
- \ prefix, i+1, entry["file"], entry["line"], entry["col"], entry["ident"]))
- let i += 1
- endwhile
- if s:go_stack_level == i
- call add(stackOut, "> ")
- endif
- call go#ui#OpenWindow("GoDef Stack", stackOut, "godefstack")
- noremap <buffer> <silent> <CR> :<C-U>call go#def#SelectStackEntry()<CR>
- noremap <buffer> <silent> <Esc> :<C-U>call go#ui#CloseWindow()<CR>
- noremap <buffer> <silent> q :<C-U>call go#ui#CloseWindow()<CR>
- endfunction
- function! go#def#StackClear(...) abort
- let s:go_stack = []
- let s:go_stack_level = 0
- endfunction
- function! go#def#StackPop(...) abort
- if len(s:go_stack) == 0
- call go#util#EchoError("godef stack empty")
- return
- endif
- if s:go_stack_level == 0
- call go#util#EchoError("at bottom of the godef stack")
- return
- endif
- if !len(a:000)
- let numPop = 1
- else
- let numPop = a:1
- endif
- let newLevel = str2nr(s:go_stack_level) - str2nr(numPop)
- call go#def#Stack(newLevel + 1)
- endfunction
- function! go#def#Stack(...) abort
- if len(s:go_stack) == 0
- call go#util#EchoError("godef stack empty")
- return
- endif
- if !len(a:000)
- " Display interactive stack
- call go#def#StackUI()
- return
- else
- let jumpTarget = a:1
- endif
- if jumpTarget !~ '^\d\+$'
- if jumpTarget !~ '^\s*$'
- call go#util#EchoError("location must be a number")
- endif
- return
- endif
- let jumpTarget = str2nr(jumpTarget) - 1
- if jumpTarget >= 0 && jumpTarget < len(s:go_stack)
- let s:go_stack_level = jumpTarget
- let target = s:go_stack[s:go_stack_level]
- " jump
- if expand('%:p') != target["file"]
- if &modified
- exec 'hide edit' target["file"]
- else
- exec 'edit' target["file"]
- endif
- endif
- call cursor(target["line"], target["col"])
- normal! zz
- else
- call go#util#EchoError("invalid location. Try :GoDefStack to see the list of valid entries")
- endif
- endfunction
- function s:def_job(args) abort
- let callbacks = go#job#Spawn(a:args)
- let start_options = {
- \ 'callback': callbacks.callback,
- \ 'exit_cb': callbacks.exit_cb,
- \ 'close_cb': callbacks.close_cb,
- \ }
- if &modified
- let l:tmpname = tempname()
- call writefile(split(a:args.input, "\n"), l:tmpname, "b")
- let l:start_options.in_io = "file"
- let l:start_options.in_name = l:tmpname
- endif
- call job_start(a:args.cmd, start_options)
- endfunction
- " vim: sw=2 ts=2 et
|