From 7fc2737ce36039d57ee8f370c3358d99a1bb1bef Mon Sep 17 00:00:00 2001 From: IdanArye Date: Wed, 28 Aug 2013 22:47:21 +0300 Subject: [PATCH] Vim plugin completed and tested on Linux --- editors/vim/README.md | 25 ++++++- editors/vim/autoload/dcomplete.vim | 111 +++++++++++++++++++++++++++++ editors/vim/ftplugin/d.vim | 6 ++ 3 files changed, 141 insertions(+), 1 deletion(-) create mode 100644 editors/vim/autoload/dcomplete.vim create mode 100644 editors/vim/ftplugin/d.vim diff --git a/editors/vim/README.md b/editors/vim/README.md index 106acc3..315fe16 100644 --- a/editors/vim/README.md +++ b/editors/vim/README.md @@ -1 +1,24 @@ -#VIM Integration +A plugin for using DCD with vim. + +Tested on Linux + +Installation and Configuration +============================== +Put the autoload and ftplugin folders in your vim runtime path. + +Compile DCD and put both dcd-client and dcd-server in your path, or set the +global variable `g:dcd\_path` to where you put DCD. + +Usage +=================== +When the filetype is D, use the `DCDstartServer` command to start the server +and the `DCDstopServer` command to stop the server. + +Use the `DCDaddPath` command to add an import path to the server. Make sure you +escape spaces! + +Use the `DCD` command to send arbitary commands to the server via the client. +The syntax is the same as with `dcd-client`, so you can use it without +arguments to print the help message. + +When the server is running, use `CTRL`+`x` `CTRL`+`o` to use DCD completion. diff --git a/editors/vim/autoload/dcomplete.vim b/editors/vim/autoload/dcomplete.vim new file mode 100644 index 0000000..f8f5307 --- /dev/null +++ b/editors/vim/autoload/dcomplete.vim @@ -0,0 +1,111 @@ +"The completion function +function! dcomplete#Complete(findstart,base) + if a:findstart + let prePos=searchpos('\W',"bn") + let preChar=getline(prePos[0])[prePos[1]-1] + if '.'==preChar + let b:completionColumn=prePos[1]+1 + return prePos[1] + endif + "If we can't find a dot, we look for a paren. + let parenPos=searchpos("(","bn",line('.')) + if parenPos[0] + if getline('.')[parenPos[1]:col('.')-2]=~'^\s*\w*$' + let b:completionColumn=parenPos[1]+1 + return parenPos[1] + endif + endif + "If we can't find either, look for the beginning of the word + if line('.')==prePos[0] && getline(prePos[0])[prePos[1]]=~'\w' + return prePos[1] + endif + "If we can't find any of the above, DCD can't help us. + return -2 + else + "Run DCD + let scanResult=s:runDCDToGetAutocompletion() + "Split the result text to lines. + let resultLines=split(scanResult,"\n") + + "if we have less than one line - something wen wrong + if empty(resultLines) + return 'bad...' + endif + "identify completion type via the first line. + if resultLines[0]=='identifiers' + return s:parsePairs(a:base,resultLines[1:],'','') + elseif resultLines[0]=='calltips' + return s:parseCalltips(a:base,resultLines[1:]) + endif + return [] + endif +endfunction + +"Get the DCD server command path +function! dcomplete#DCDserver() + if exists('g:dcd_path') + return shellescape(g:dcd_path).(has('win32') ? '\' : '/').'dcd-server' + else + return 'dcd-server' + end +endfunction + +"Get the DCD client command path +function! dcomplete#DCDclient() + if exists('g:dcd_path') + return shellescape(g:dcd_path).(has('win32') ? '\' : '/').'dcd-client' + else + return 'dcd-client' + end +endfunction + +"Run DCD to get autocompletion results +function! s:runDCDToGetAutocompletion() + + let l:tmpFileName=tempname() + "Save the temp file in unix format for better reading of byte position. + let l:oldFileFormat=&fileformat + set fileformat=unix + let l:bytePosition=line2byte('.')+b:completionColumn-2 + exec "write ".l:tmpFileName + let &fileformat=l:oldFileFormat + let scanResult=system(dcomplete#DCDclient().' --cursorPos '.l:bytePosition.' <'.shellescape(l:tmpFileName)) + call delete(l:tmpFileName) + + return scanResult +endfunction + +"Parse simple pair results +function! s:parsePairs(base,resultLines,addBefore,addAfter) + let result=[] + for resultLine in a:resultLines + if len(resultLine) + let lineParts=split(resultLine) + if lineParts[0]=~'^'.a:base && 2==len(lineParts) && 1==len(lineParts[1]) + call add(result,{'word':a:addBefore.lineParts[0].a:addAfter,'kind':lineParts[1]}) + endif + end + endfor + return result +endfunction + +"Parse function calltips results +function! s:parseCalltips(base,resultLines) + let result=[a:base] + for resultLine in a:resultLines + if 0<=match(resultLine,".*(.*)") + let funcArgs=[] + for funcArg in split(resultLine[match(resultLine,'(')+1:-2],', ') + let argParts=split(funcArg) + if 1 /dev/null &' +command! -buffer -nargs=? DCD execute '!'.dcomplete#DCDclient().' '. +command! -buffer DCDstopServer DCD --shutdown +command! -buffer -nargs=1 -complete=dir DCDaddPath DCD -I