let g:vebugger_breakpoints=[] "Initialize the default pipe bufferers function! vebugger#std#setStandardBufferers(debugger) for l:pipe in values(a:debugger.pipes) let l:pipe.bufferer = function('vebugger#std#readNewLinesFromPipe') endfor endfunction "Initialize the std part of the debugger's state function! vebugger#std#setStandardState(debugger) let a:debugger.state.std={ \'config':{ \ 'externalFileStop_flowCommand':''}, \'location':{}, \'callstack':[], \'evaluateExpressions':[]} endfunction "Initialize the std part of the debugger's read result template function! vebugger#std#setStandardReadResultTemplate(debugger) let a:debugger.readResultTemplate.std={ \'programOutput':{}, \'location':{}, \'callstack':{}, \'evaluatedExpression':{}, \'programFinish':{}, \'exception':{}} endfunction "Initialize the std part of the debugger's write actions template function! vebugger#std#setStandardWriteactionsTemplate(debugger) let a:debugger.writeActionsTemplate.std={ \'flow':'', \'breakpoints':[], \'evaluateExpressions':[], \'executeStatements':[], \'removeAfterDisplayed':[], \'closeDebugger':''} endfunction "Adds the std_ functions to the debugger object function! vebugger#std#addStandardFunctions(debugger) for l:k in keys(s:standardFunctions) let a:debugger['std_'.l:k]=s:standardFunctions[l:k] endfor endfunction "Add the standard think handlers to the debugger function! vebugger#std#addStandardThinkHandlers(debugger) for l:ThinkHandler in values(s:standardThinkHandlers) call a:debugger.addThinkHandler(l:ThinkHandler) endfor endfunction "Add the standard close handlers to the debugger function! vebugger#std#addStandardCloseHandlers(debugger) for l:CloseHandler in values(s:standardCloseHandlers) call a:debugger.addCloseHandler(l:CloseHandler) endfor endfunction "Performs the standard initialization of the debugger object function! vebugger#std#standardInit(debugger) call vebugger#std#setStandardBufferers(a:debugger) call vebugger#std#setStandardState(a:debugger) call vebugger#std#setStandardReadResultTemplate(a:debugger) call vebugger#std#setStandardWriteactionsTemplate(a:debugger) call vebugger#std#addStandardFunctions(a:debugger) call vebugger#std#addStandardThinkHandlers(a:debugger) call vebugger#std#addStandardCloseHandlers(a:debugger) endfunction "Start a debugger with the std settings function! vebugger#std#startDebugger(command) let l:debugger=vebugger#startDebugger(a:command) call vebugger#std#standardInit(l:debugger) return l:debugger endfunction "Read and return all new lines from a Vebugger pipe object. function! vebugger#std#readNewLinesFromPipe() dict let l:lastNewline = strridx(self.buffer, "\n") if 0 <= l:lastNewline let l:outLines = split(strpart(self.buffer, 0, l:lastNewline), '\r\n\|\n\|\r') let self.buffer = strpart(self.buffer, l:lastNewline + 1) return l:outLines endif return [] endfunction "Opens the shell buffer for a debugger. The shell buffer displays the output "of the debugged program, and when it's closed the debugger gets terminated. "Shell buffers should not be used when attaching a debugger to a running "process. function! vebugger#std#openShellBuffer(debugger) if has_key(a:debugger,'shellBuffer') if -1 if exists('b:debugger') | call b:debugger.kill() | endif setlocal buftype=nofile setlocal bufhidden=wipe let a:debugger.shellBuffer=bufnr('') silent file Vebugger:Shell wincmd p endfunction "Closes the shell buffer function! vebugger#std#closeShellBuffer(debugger) if has_key(a:debugger,'shellBuffer') if -1" if mode() == 'n' " Call echo normaly call feedkeys(l:echoKeys) elseif mode() == 'i' " Execute command in insert mode call feedkeys("\".l:echoKeys) endif " NOTE: Other modes are not supported else " Without timer support, feedkeys won't work and we have " to echo directly echo join(l:echoLines, "\n") endif endfunction "Print an expression that it's evaluation was previously requested function! s:standardThinkHandlers.printEvaluatedExpression(readResult,debugger) dict let l:evaluatedExpression=a:readResult.std.evaluatedExpression if !empty(l:evaluatedExpression) if !empty(get(l:evaluatedExpression,'expression')) let l:index=index(a:debugger.state.std.evaluateExpressions,l:evaluatedExpression.expression) if 0<=l:index call remove(a:debugger.state.std.evaluateExpressions,l:index) endif endif call s:printWithFeedKeys(l:evaluatedExpression) call a:debugger.addWriteAction('std','removeAfterDisplayed',a:readResult) endif endfunction "Close the debugger when the program is finished but the debugger wasn't "closed automatically function! s:standardThinkHandlers.closeDebuggerWhenProgramFinishes(readResult,debugger) dict if !empty(a:readResult.std.programFinish) call a:debugger.setWriteAction('std','closeDebugger','close') endif endfunction "Print an exception message function! s:standardThinkHandlers.printException(readResult,debugger) dict if !empty(a:readResult.std.exception) echohl WarningMsg echo a:readResult.std.exception.message."\n" echohl None endif endfunction let s:standardCloseHandlers={} "Remove the currently executed line when a debugger is closed function! s:standardCloseHandlers.removeCurrentMarker(debugger) dict let a:debugger.state.std.location={} sign unplace 1 endfunction sign define vebugger_current text=-> if hlexists('BreakPoint') sign define vebugger_breakpoint text=** linehl=BreakPoint texthl=BreakPoint else sign define vebugger_breakpoint text=** linehl=ColorColumn texthl=ColorColumn endif "Update all the marks(currently executed line and breakpoints) for a file function! vebugger#std#updateMarksForFile(state,filename) let l:filename=fnamemodify(a:filename,":p") if -1 < bufnr(l:filename) exe 'sign unplace 1 file='.fnameescape(fnamemodify(l:filename,':p')) exe 'sign unplace 2 file='.fnameescape(fnamemodify(l:filename,':p')) for l:breakpoint in g:vebugger_breakpoints if fnamemodify(l:breakpoint.file,':p')==fnamemodify(a:filename,':p') exe 'sign place 2 name=vebugger_breakpoint line='.l:breakpoint.line.' file='.fnameescape(fnamemodify(l:breakpoint.file,':p')) endif endfor if !empty(a:state) if !empty(a:state.std.location) if fnamemodify(a:state.std.location.file,':p')==fnamemodify(a:filename,':p') exe 'sign place 1 name=vebugger_current line='.a:state.std.location.line.' file='.fnameescape(fnamemodify(l:filename,':p')) endif endif endif endif endfunction "Toggle a breakpoint on and off function! vebugger#std#toggleBreakpoint(file,line) let l:debugger=vebugger#getActiveDebugger() let l:debuggerState=empty(l:debugger) \? {} \: l:debugger.state for l:i in range(len(g:vebugger_breakpoints)) let l:breakpoint=g:vebugger_breakpoints[l:i] if l:breakpoint.file==a:file && l:breakpoint.line==a:line call remove(g:vebugger_breakpoints,l:i) if !empty(l:debugger) call l:debugger.addWriteActionAndPerform('std','breakpoints',{ \'action':'remove', \'file':(a:file), \'line':(a:line)}) endif call vebugger#std#updateMarksForFile(l:debuggerState,a:file) return endif endfor call add(g:vebugger_breakpoints,{'file':(a:file),'line':(a:line)}) if !empty(l:debugger) call l:debugger.addWriteActionAndPerform('std','breakpoints',{ \'action':'add', \'file':(a:file), \'line':(a:line)}) endif call vebugger#std#updateMarksForFile(l:debuggerState,a:file) endfunction "Clear all breakpoints function! vebugger#std#clearBreakpoints() let l:debugger=vebugger#getActiveDebugger() let l:debuggerState=empty(l:debugger) ? {} : l:debugger.state let l:files=[] for l:breakpoint in g:vebugger_breakpoints if index(l:files,l:breakpoint.file)<0 call add(l:files,l:breakpoint.file) endif if !empty(l:debugger) call l:debugger.addWriteAction('std','breakpoints',extend({'action':'remove'},l:breakpoint)) endif endfor if !empty(l:debugger) call l:debugger.performWriteActions() endif let g:vebugger_breakpoints=[] for l:file in l:files call vebugger#std#updateMarksForFile(l:debuggerState,l:file) endfor endfunction