1495 lines
45 KiB
VimL
1495 lines
45 KiB
VimL
|
"=================================================
|
||
|
" File: autoload/undotree.vim
|
||
|
" Description: Manage your undo history in a graph.
|
||
|
" Author: David Knoble <ben.knoble@gmail.com>
|
||
|
" License: BSD
|
||
|
|
||
|
" Avoid installing twice.
|
||
|
if exists('g:autoloaded_undotree')
|
||
|
finish
|
||
|
endif
|
||
|
let g:autoloaded_undotree = 0
|
||
|
|
||
|
" At least version 7.3 with 005 patch is needed for undo branches.
|
||
|
" Refer to https://github.com/mbbill/undotree/issues/4 for details.
|
||
|
" Thanks kien
|
||
|
if v:version < 703
|
||
|
finish
|
||
|
endif
|
||
|
if (v:version == 703 && !has("patch005"))
|
||
|
finish
|
||
|
endif
|
||
|
let g:loaded_undotree = 1 " Signal plugin availability with a value of 1.
|
||
|
|
||
|
" Short time indicators
|
||
|
if g:undotree_ShortIndicators == 1
|
||
|
let s:timeSecond = '1 s'
|
||
|
let s:timeSeconds = ' s'
|
||
|
|
||
|
let s:timeMinute = '1 m'
|
||
|
let s:timeMinutes = ' m'
|
||
|
|
||
|
let s:timeHour = '1 h'
|
||
|
let s:timeHours = ' h'
|
||
|
|
||
|
let s:timeDay = '1 d'
|
||
|
let s:timeDays = ' d'
|
||
|
|
||
|
let s:timeOriginal = 'Orig'
|
||
|
else
|
||
|
let s:timeSecond = '1 second ago'
|
||
|
let s:timeSeconds = ' seconds ago'
|
||
|
|
||
|
let s:timeMinute = '1 minute ago'
|
||
|
let s:timeMinutes = ' minutes ago'
|
||
|
|
||
|
let s:timeHour = '1 hour ago'
|
||
|
let s:timeHours = ' hours ago'
|
||
|
|
||
|
let s:timeDay = '1 day ago'
|
||
|
let s:timeDays = ' days ago'
|
||
|
|
||
|
let s:timeOriginal = 'Original'
|
||
|
endif
|
||
|
|
||
|
"=================================================
|
||
|
" Help text
|
||
|
let s:helpmore = ['" ===== Marks ===== ',
|
||
|
\'" >num< : The current state',
|
||
|
\'" {num} : The next redo state',
|
||
|
\'" [num] : The latest state',
|
||
|
\'" s : Saved states',
|
||
|
\'" S : The last saved state',
|
||
|
\'" ===== Hotkeys =====']
|
||
|
if !g:undotree_HelpLine
|
||
|
let s:helpless = []
|
||
|
else
|
||
|
let s:helpless = ['" Press ? for help.']
|
||
|
endif
|
||
|
|
||
|
"Custom key mappings: add this function to your vimrc.
|
||
|
"You can define whatever mapping as you like, this is a hook function which
|
||
|
"will be called after undotree window initialized.
|
||
|
"
|
||
|
"function g:Undotree_CustomMap()
|
||
|
" map <buffer> <c-n> J
|
||
|
" map <buffer> <c-p> K
|
||
|
"endfunction
|
||
|
|
||
|
" Keymap
|
||
|
let s:keymap = []
|
||
|
" action, key, help.
|
||
|
let s:keymap += [['Help','?','Toggle quick help']]
|
||
|
let s:keymap += [['Close','q','Close undotree panel']]
|
||
|
let s:keymap += [['FocusTarget','<tab>','Set Focus back to the editor']]
|
||
|
let s:keymap += [['ClearHistory','C','Clear undo history (with confirmation)']]
|
||
|
let s:keymap += [['TimestampToggle','T','Toggle relative timestamp']]
|
||
|
let s:keymap += [['DiffToggle','D','Toggle the diff panel']]
|
||
|
let s:keymap += [['NextState','K','Move to the next undo state']]
|
||
|
let s:keymap += [['PreviousState','J','Move to the previous undo state']]
|
||
|
let s:keymap += [['NextSavedState','>','Move to the next saved state']]
|
||
|
let s:keymap += [['PreviousSavedState','<','Move to the previous saved state']]
|
||
|
let s:keymap += [['Redo','<c-r>','Redo']]
|
||
|
let s:keymap += [['Undo','u','Undo']]
|
||
|
let s:keymap += [['Enter','<2-LeftMouse>','Move to the current state']]
|
||
|
let s:keymap += [['Enter','<cr>','Move to the current state']]
|
||
|
|
||
|
" 'Diff' sign definitions. There are two 'delete' signs; a 'normal' one and one
|
||
|
" that is used if the very end of the buffer has been deleted (in which case the
|
||
|
" deleted text is actually bejond the end of the current buffer version and therefore
|
||
|
" it is not possible to place a sign on the exact line - because it doesn't exist.
|
||
|
" Instead, a 'special' delete sign is placed on the (existing) last line of the
|
||
|
" buffer)
|
||
|
exe 'sign define UndotreeAdd text=++ texthl='.undotree_HighlightSyntaxAdd
|
||
|
exe 'sign define UndotreeChg text=~~ texthl='.undotree_HighlightSyntaxChange
|
||
|
exe 'sign define UndotreeDel text=-- texthl='.undotree_HighlightSyntaxDel
|
||
|
exe 'sign define UndotreeDelEnd text=-v texthl='.undotree_HighlightSyntaxDel
|
||
|
|
||
|
" Id to use for all signs. This is an arbirary number that is hoped to be unique
|
||
|
" within the instance of vim. There is no way of guaranteeing it IS unique, which
|
||
|
" is a shame because it needs to be!
|
||
|
"
|
||
|
" Note that all signs are placed with the same Id - as long as we keep a count of
|
||
|
" how many we have placed (so we can remove them all again), this is ok
|
||
|
let s:signId = 2123654789
|
||
|
|
||
|
"=================================================
|
||
|
function! s:new(obj) abort
|
||
|
let newobj = deepcopy(a:obj)
|
||
|
call newobj.Init()
|
||
|
return newobj
|
||
|
endfunction
|
||
|
|
||
|
" Get formatted time
|
||
|
function! s:gettime(time) abort
|
||
|
if a:time == 0
|
||
|
return s:timeOriginal
|
||
|
endif
|
||
|
if !g:undotree_RelativeTimestamp
|
||
|
let today = substitute(strftime("%c",localtime())," .*$",'','g')
|
||
|
if today == substitute(strftime("%c",a:time)," .*$",'','g')
|
||
|
return strftime("%H:%M:%S",a:time)
|
||
|
else
|
||
|
return strftime("%H:%M:%S %b%d %Y",a:time)
|
||
|
endif
|
||
|
else
|
||
|
let sec = localtime() - a:time
|
||
|
if sec < 0
|
||
|
let sec = 0
|
||
|
endif
|
||
|
if sec < 60
|
||
|
if sec == 1
|
||
|
return s:timeSecond
|
||
|
else
|
||
|
return sec.s:timeSeconds
|
||
|
endif
|
||
|
endif
|
||
|
if sec < 3600
|
||
|
if (sec/60) == 1
|
||
|
return s:timeMinute
|
||
|
else
|
||
|
return (sec/60).s:timeMinutes
|
||
|
endif
|
||
|
endif
|
||
|
if sec < 86400 "3600*24
|
||
|
if (sec/3600) == 1
|
||
|
return s:timeHour
|
||
|
else
|
||
|
return (sec/3600).s:timeHours
|
||
|
endif
|
||
|
endif
|
||
|
if (sec/86400) == 1
|
||
|
return s:timeDay
|
||
|
else
|
||
|
return (sec/86400).s:timeDays
|
||
|
endif
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:exec(cmd) abort
|
||
|
call s:log("s:exec() ".a:cmd)
|
||
|
silent exe a:cmd
|
||
|
endfunction
|
||
|
|
||
|
" Don't trigger any events(like BufEnter which could cause redundant refresh)
|
||
|
function! s:exec_silent(cmd) abort
|
||
|
call s:log("s:exec_silent() ".a:cmd)
|
||
|
let ei_bak= &eventignore
|
||
|
set eventignore=BufEnter,BufLeave,BufWinLeave,InsertLeave,CursorMoved,BufWritePost
|
||
|
silent exe a:cmd
|
||
|
let &eventignore = ei_bak
|
||
|
endfunction
|
||
|
|
||
|
" Return a unique id each time.
|
||
|
let s:cntr = 0
|
||
|
function! s:getUniqueID() abort
|
||
|
let s:cntr = s:cntr + 1
|
||
|
return s:cntr
|
||
|
endfunction
|
||
|
|
||
|
" Set to 1 to enable debug log
|
||
|
let s:debug = 0
|
||
|
let s:debugfile = $HOME.'/undotree_debug.log'
|
||
|
" If debug file exists, enable debug output.
|
||
|
if filewritable(s:debugfile)
|
||
|
let s:debug = 1
|
||
|
exec 'redir >> '. s:debugfile
|
||
|
silent echo "=======================================\n"
|
||
|
redir END
|
||
|
endif
|
||
|
|
||
|
function! s:log(msg) abort
|
||
|
if s:debug
|
||
|
exec 'redir >> ' . s:debugfile
|
||
|
silent echon strftime('%H:%M:%S') . ': ' . string(a:msg) . "\n"
|
||
|
redir END
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:ObserveOptions()
|
||
|
augroup Undotree_OptionsObserver
|
||
|
try
|
||
|
autocmd!
|
||
|
if exists('+fdo')
|
||
|
let s:open_folds = &fdo =~# 'undo'
|
||
|
if exists('##OptionSet')
|
||
|
autocmd OptionSet foldopen let s:open_folds = v:option_new =~# 'undo'
|
||
|
endif
|
||
|
endif
|
||
|
finally
|
||
|
augroup END
|
||
|
endtry
|
||
|
endfunction
|
||
|
|
||
|
" Whether to open folds on undo/redo.
|
||
|
" Is 1 when 'undo' is in &fdo (see :help 'foldopen').
|
||
|
" default: 1
|
||
|
let s:open_folds = 1
|
||
|
|
||
|
if exists('v:vim_did_enter')
|
||
|
if !v:vim_did_enter
|
||
|
autocmd VimEnter * call s:ObserveOptions()
|
||
|
else
|
||
|
call s:ObserveOptions()
|
||
|
endif
|
||
|
else
|
||
|
autocmd VimEnter * call s:ObserveOptions()
|
||
|
call s:ObserveOptions()
|
||
|
endif
|
||
|
|
||
|
"=================================================
|
||
|
"Base class for panels.
|
||
|
let s:panel = {}
|
||
|
|
||
|
function! s:panel.Init() abort
|
||
|
let self.bufname = "invalid"
|
||
|
endfunction
|
||
|
|
||
|
function! s:panel.SetFocus() abort
|
||
|
let winnr = bufwinnr(self.bufname)
|
||
|
" already focused.
|
||
|
if winnr == winnr()
|
||
|
return
|
||
|
endif
|
||
|
if winnr == -1
|
||
|
echoerr "Fatal: window does not exist!"
|
||
|
return
|
||
|
endif
|
||
|
call s:log("SetFocus() winnr:".winnr." bufname:".self.bufname)
|
||
|
" wincmd would cause cursor outside window.
|
||
|
call s:exec_silent("norm! ".winnr."\<c-w>\<c-w>")
|
||
|
endfunction
|
||
|
|
||
|
function! s:panel.IsVisible() abort
|
||
|
if bufwinnr(self.bufname) != -1
|
||
|
return 1
|
||
|
else
|
||
|
return 0
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:panel.Hide() abort
|
||
|
call s:log(self.bufname." Hide()")
|
||
|
if !self.IsVisible()
|
||
|
return
|
||
|
endif
|
||
|
call self.SetFocus()
|
||
|
call s:exec("quit")
|
||
|
endfunction
|
||
|
|
||
|
"=================================================
|
||
|
" undotree panel class.
|
||
|
" extended from panel.
|
||
|
"
|
||
|
|
||
|
" {rawtree}
|
||
|
" |
|
||
|
" | ConvertInput() {seq2index}--> [seq1:index1]
|
||
|
" v [seq2:index2] ---+
|
||
|
" {tree} ... |
|
||
|
" | [asciimeta] |
|
||
|
" | Render() | |
|
||
|
" v v |
|
||
|
" [asciitree] --> [" * | SEQ DDMMYY "] <==> [node1{seq,time,..}] |
|
||
|
" [" |/ "] [node2{seq,time,..}] <---+
|
||
|
" ... ...
|
||
|
|
||
|
let s:undotree = s:new(s:panel)
|
||
|
|
||
|
function! s:undotree.Init() abort
|
||
|
let self.bufname = "undotree_".s:getUniqueID()
|
||
|
" Increase to make it unique.
|
||
|
let self.width = g:undotree_SplitWidth
|
||
|
let self.opendiff = g:undotree_DiffAutoOpen
|
||
|
let self.targetid = -1
|
||
|
let self.targetBufnr = -1
|
||
|
let self.rawtree = {} "data passed from undotree()
|
||
|
let self.tree = {} "data converted to internal format.
|
||
|
let self.seq_last = -1
|
||
|
let self.save_last = -1
|
||
|
let self.save_last_bak = -1
|
||
|
|
||
|
" seqs
|
||
|
let self.seq_cur = -1
|
||
|
let self.seq_curhead = -1
|
||
|
let self.seq_newhead = -1
|
||
|
let self.seq_saved = {} "{saved value -> seq} pair
|
||
|
|
||
|
"backup, for mark
|
||
|
let self.seq_cur_bak = -1
|
||
|
let self.seq_curhead_bak = -1
|
||
|
let self.seq_newhead_bak = -1
|
||
|
|
||
|
let self.asciitree = [] "output data.
|
||
|
let self.asciimeta = [] "meta data behind ascii tree.
|
||
|
let self.seq2index = {} "table used to convert seq to index.
|
||
|
let self.showHelp = 0
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.BindKey() abort
|
||
|
if v:version > 703 || (v:version == 703 && has("patch1261"))
|
||
|
let map_options = ' <nowait> '
|
||
|
else
|
||
|
let map_options = ''
|
||
|
endif
|
||
|
let map_options = map_options.' <silent> <buffer> '
|
||
|
for i in s:keymap
|
||
|
silent exec 'nmap '.map_options.i[1].' <plug>Undotree'.i[0]
|
||
|
silent exec 'nnoremap '.map_options.'<plug>Undotree'.i[0]
|
||
|
\ .' :call <sid>undotreeAction("'.i[0].'")<cr>'
|
||
|
endfor
|
||
|
if exists('*g:Undotree_CustomMap')
|
||
|
call g:Undotree_CustomMap()
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.BindAu() abort
|
||
|
" Auto exit if it's the last window
|
||
|
augroup Undotree_Main
|
||
|
au!
|
||
|
au BufEnter <buffer> call s:exitIfLast()
|
||
|
au BufEnter,BufLeave <buffer> if exists('t:undotree') |
|
||
|
\let t:undotree.width = winwidth(winnr()) | endif
|
||
|
au BufWinLeave <buffer> if exists('t:diffpanel') |
|
||
|
\call t:diffpanel.Hide() | endif
|
||
|
augroup end
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.Action(action) abort
|
||
|
call s:log("undotree.Action() ".a:action)
|
||
|
if !self.IsVisible() || !exists('b:isUndotreeBuffer')
|
||
|
echoerr "Fatal: window does not exist."
|
||
|
return
|
||
|
endif
|
||
|
if !has_key(self,'Action'.a:action)
|
||
|
echoerr "Fatal: Action does not exist!"
|
||
|
return
|
||
|
endif
|
||
|
silent exec 'call self.Action'.a:action.'()'
|
||
|
endfunction
|
||
|
|
||
|
" Helper function, do action in target window, and then update itself.
|
||
|
function! s:undotree.ActionInTarget(cmd) abort
|
||
|
if !self.SetTargetFocus()
|
||
|
return
|
||
|
endif
|
||
|
" Target should be a normal buffer.
|
||
|
if (&bt == '' || &bt == 'acwrite') && (&modifiable == 1) && (mode() == 'n')
|
||
|
call s:exec(a:cmd)
|
||
|
" Open folds so that the change being undone/redone is visible.
|
||
|
if s:open_folds
|
||
|
call s:exec('normal! zv')
|
||
|
endif
|
||
|
call self.Update()
|
||
|
endif
|
||
|
" Update not always set current focus.
|
||
|
call self.SetFocus()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionHelp() abort
|
||
|
let self.showHelp = !self.showHelp
|
||
|
call self.Draw()
|
||
|
call self.MarkSeqs()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionFocusTarget() abort
|
||
|
call self.SetTargetFocus()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionEnter() abort
|
||
|
let index = self.Screen2Index(line('.'))
|
||
|
if index < 0
|
||
|
return
|
||
|
endif
|
||
|
let seq = self.asciimeta[index].seq
|
||
|
if seq == -1
|
||
|
return
|
||
|
endif
|
||
|
if seq == 0
|
||
|
call self.ActionInTarget('norm 9999u')
|
||
|
return
|
||
|
endif
|
||
|
call self.ActionInTarget('u '.self.asciimeta[index].seq)
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionUndo() abort
|
||
|
call self.ActionInTarget('undo')
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionRedo() abort
|
||
|
call self.ActionInTarget("redo")
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionPreviousState() abort
|
||
|
call self.ActionInTarget('earlier')
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionNextState() abort
|
||
|
call self.ActionInTarget('later')
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionPreviousSavedState() abort
|
||
|
call self.ActionInTarget('earlier 1f')
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionNextSavedState() abort
|
||
|
call self.ActionInTarget('later 1f')
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionDiffToggle() abort
|
||
|
let self.opendiff = !self.opendiff
|
||
|
call t:diffpanel.Toggle()
|
||
|
call self.UpdateDiff()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionTimestampToggle() abort
|
||
|
if !self.SetTargetFocus()
|
||
|
return
|
||
|
endif
|
||
|
let g:undotree_RelativeTimestamp = !g:undotree_RelativeTimestamp
|
||
|
let self.targetBufnr = -1 "force update
|
||
|
call self.Update()
|
||
|
" Update not always set current focus.
|
||
|
call self.SetFocus()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionClearHistory() abort
|
||
|
if input("Clear ALL undo history? Type \"YES\" to continue: ") != "YES"
|
||
|
return
|
||
|
endif
|
||
|
if !self.SetTargetFocus()
|
||
|
return
|
||
|
endif
|
||
|
let ul_bak = &undolevels
|
||
|
let mod_bak = &modified
|
||
|
let &undolevels = -1
|
||
|
call s:exec("norm! a \<BS>\<Esc>")
|
||
|
let &undolevels = ul_bak
|
||
|
let &modified = mod_bak
|
||
|
unlet ul_bak mod_bak
|
||
|
let self.targetBufnr = -1 "force update
|
||
|
call self.Update()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.ActionClose() abort
|
||
|
call self.Toggle()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.UpdateDiff() abort
|
||
|
call s:log("undotree.UpdateDiff()")
|
||
|
if !t:diffpanel.IsVisible()
|
||
|
return
|
||
|
endif
|
||
|
call t:diffpanel.Update(self.seq_cur,self.targetBufnr,self.targetid)
|
||
|
endfunction
|
||
|
|
||
|
" May fail due to target window closed.
|
||
|
function! s:undotree.SetTargetFocus() abort
|
||
|
for winnr in range(1, winnr('$')) "winnr starts from 1
|
||
|
if getwinvar(winnr,'undotree_id') == self.targetid
|
||
|
if winnr() != winnr
|
||
|
call s:exec("norm! ".winnr."\<c-w>\<c-w>")
|
||
|
return 1
|
||
|
endif
|
||
|
endif
|
||
|
endfor
|
||
|
return 0
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.Toggle() abort
|
||
|
"Global auto commands to keep undotree up to date.
|
||
|
let auEvents = "BufEnter,InsertLeave,CursorMoved,BufWritePost"
|
||
|
|
||
|
call s:log(self.bufname." Toggle()")
|
||
|
if self.IsVisible()
|
||
|
call self.Hide()
|
||
|
call t:diffpanel.Hide()
|
||
|
call self.SetTargetFocus()
|
||
|
augroup Undotree
|
||
|
autocmd!
|
||
|
augroup END
|
||
|
else
|
||
|
call self.Show()
|
||
|
if !g:undotree_SetFocusWhenToggle
|
||
|
call self.SetTargetFocus()
|
||
|
endif
|
||
|
augroup Undotree
|
||
|
au!
|
||
|
exec "au! ".auEvents." * call undotree#UndotreeUpdate()"
|
||
|
augroup END
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.GetStatusLine() abort
|
||
|
if self.seq_cur != -1
|
||
|
let seq_cur = self.seq_cur
|
||
|
else
|
||
|
let seq_cur = 'None'
|
||
|
endif
|
||
|
if self.seq_curhead != -1
|
||
|
let seq_curhead = self.seq_curhead
|
||
|
else
|
||
|
let seq_curhead = 'None'
|
||
|
endif
|
||
|
return 'current: '.seq_cur.' redo: '.seq_curhead
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.Show() abort
|
||
|
call s:log("undotree.Show()")
|
||
|
if self.IsVisible()
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
let self.targetid = w:undotree_id
|
||
|
|
||
|
" Create undotree window.
|
||
|
if exists("g:undotree_CustomUndotreeCmd")
|
||
|
let cmd = g:undotree_CustomUndotreeCmd . ' ' .
|
||
|
\self.bufname
|
||
|
elseif g:undotree_WindowLayout == 1 || g:undotree_WindowLayout == 2
|
||
|
let cmd = "topleft vertical" .
|
||
|
\self.width . ' new ' . self.bufname
|
||
|
else
|
||
|
let cmd = "botright vertical" .
|
||
|
\self.width . ' new ' . self.bufname
|
||
|
endif
|
||
|
call s:exec("silent keepalt ".cmd)
|
||
|
call self.SetFocus()
|
||
|
|
||
|
" We need a way to tell if the buffer is belong to undotree,
|
||
|
" bufname() is not always reliable.
|
||
|
let b:isUndotreeBuffer = 1
|
||
|
|
||
|
setlocal winfixwidth
|
||
|
setlocal noswapfile
|
||
|
setlocal buftype=nowrite
|
||
|
setlocal bufhidden=delete
|
||
|
setlocal nowrap
|
||
|
setlocal nolist
|
||
|
setlocal foldcolumn=0
|
||
|
setlocal nobuflisted
|
||
|
setlocal nospell
|
||
|
setlocal nonumber
|
||
|
setlocal norelativenumber
|
||
|
if g:undotree_CursorLine
|
||
|
setlocal cursorline
|
||
|
else
|
||
|
setlocal nocursorline
|
||
|
endif
|
||
|
setlocal nomodifiable
|
||
|
setlocal statusline=%!t:undotree.GetStatusLine()
|
||
|
setfiletype undotree
|
||
|
|
||
|
call self.BindKey()
|
||
|
call self.BindAu()
|
||
|
|
||
|
let ei_bak= &eventignore
|
||
|
set eventignore=all
|
||
|
|
||
|
call self.SetTargetFocus()
|
||
|
let self.targetBufnr = -1 "force update
|
||
|
call self.Update()
|
||
|
|
||
|
let &eventignore = ei_bak
|
||
|
|
||
|
if self.opendiff
|
||
|
call t:diffpanel.Show()
|
||
|
call self.UpdateDiff()
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
" called outside undotree window
|
||
|
function! s:undotree.Update() abort
|
||
|
if !self.IsVisible()
|
||
|
return
|
||
|
endif
|
||
|
" do nothing if we're in the undotree or diff panel
|
||
|
if exists('b:isUndotreeBuffer')
|
||
|
return
|
||
|
endif
|
||
|
if (&bt != '' && &bt != 'acwrite') || (&modifiable == 0) || (mode() != 'n')
|
||
|
if &bt == 'quickfix' || &bt == 'nofile'
|
||
|
"Do nothing for quickfix and q:
|
||
|
call s:log("undotree.Update() ignore quickfix")
|
||
|
return
|
||
|
endif
|
||
|
if self.targetBufnr == bufnr('%') && self.targetid == w:undotree_id
|
||
|
call s:log("undotree.Update() invalid buffer NOupdate")
|
||
|
return
|
||
|
endif
|
||
|
let emptybuf = 1 "This is not a valid buffer, could be help or something.
|
||
|
call s:log("undotree.Update() invalid buffer update")
|
||
|
else
|
||
|
let emptybuf = 0
|
||
|
"update undotree,set focus
|
||
|
if self.targetBufnr == bufnr('%')
|
||
|
let self.targetid = w:undotree_id
|
||
|
let newrawtree = undotree()
|
||
|
if self.rawtree == newrawtree
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
" same buffer, but seq changed.
|
||
|
if newrawtree.seq_last == self.seq_last
|
||
|
call s:log("undotree.Update() update seqs")
|
||
|
let self.rawtree = newrawtree
|
||
|
call self.ConvertInput(0) "only update seqs.
|
||
|
if (self.seq_cur == self.seq_cur_bak) &&
|
||
|
\(self.seq_curhead == self.seq_curhead_bak)&&
|
||
|
\(self.seq_newhead == self.seq_newhead_bak)&&
|
||
|
\(self.save_last == self.save_last_bak)
|
||
|
return
|
||
|
endif
|
||
|
call self.SetFocus()
|
||
|
call self.MarkSeqs()
|
||
|
call self.UpdateDiff()
|
||
|
return
|
||
|
endif
|
||
|
endif
|
||
|
endif
|
||
|
call s:log("undotree.Update() update whole tree")
|
||
|
|
||
|
let self.targetBufnr = bufnr('%')
|
||
|
let self.targetid = w:undotree_id
|
||
|
if emptybuf " Show an empty undo tree instead of do nothing.
|
||
|
let self.rawtree = {'seq_last':0,'entries':[],'time_cur':0,'save_last':0,'synced':1,'save_cur':0,'seq_cur':0}
|
||
|
else
|
||
|
let self.rawtree = undotree()
|
||
|
endif
|
||
|
let self.seq_last = self.rawtree.seq_last
|
||
|
let self.seq_cur = -1
|
||
|
let self.seq_curhead = -1
|
||
|
let self.seq_newhead = -1
|
||
|
call self.ConvertInput(1) "update all.
|
||
|
call self.Render()
|
||
|
call self.SetFocus()
|
||
|
call self.Draw()
|
||
|
call self.MarkSeqs()
|
||
|
call self.UpdateDiff()
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.AppendHelp() abort
|
||
|
if self.showHelp
|
||
|
call append(0,'') "empty line
|
||
|
for i in s:keymap
|
||
|
call append(0,'" '.i[1].' : '.i[2])
|
||
|
endfor
|
||
|
call append(0,s:helpmore)
|
||
|
else
|
||
|
if g:undotree_HelpLine
|
||
|
call append(0,'')
|
||
|
endif
|
||
|
call append(0,s:helpless)
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.Index2Screen(index) abort
|
||
|
" index starts from zero
|
||
|
let index_padding = 1
|
||
|
let empty_line = 1
|
||
|
let lineNr = a:index + index_padding + empty_line
|
||
|
" calculate line number according to the help text.
|
||
|
" index starts from zero and lineNr starts from 1
|
||
|
if self.showHelp
|
||
|
let lineNr += len(s:keymap) + len(s:helpmore)
|
||
|
else
|
||
|
let lineNr += len(s:helpless)
|
||
|
if !g:undotree_HelpLine
|
||
|
let lineNr -= empty_line
|
||
|
endif
|
||
|
endif
|
||
|
return lineNr
|
||
|
endfunction
|
||
|
|
||
|
" <0 if index is invalid. e.g. current line is in help text.
|
||
|
function! s:undotree.Screen2Index(line) abort
|
||
|
let index_padding = 1
|
||
|
let empty_line = 1
|
||
|
let index = a:line - index_padding - empty_line
|
||
|
|
||
|
if self.showHelp
|
||
|
let index -= len(s:keymap) + len(s:helpmore)
|
||
|
else
|
||
|
let index -= len(s:helpless)
|
||
|
if !g:undotree_HelpLine
|
||
|
let index += empty_line
|
||
|
endif
|
||
|
endif
|
||
|
return index
|
||
|
endfunction
|
||
|
|
||
|
" Current window must be undotree.
|
||
|
function! s:undotree.Draw() abort
|
||
|
" remember the current cursor position.
|
||
|
let savedview = winsaveview()
|
||
|
|
||
|
setlocal modifiable
|
||
|
" Delete text into blackhole register.
|
||
|
call s:exec('1,$ d _')
|
||
|
call append(0,self.asciitree)
|
||
|
|
||
|
call self.AppendHelp()
|
||
|
|
||
|
"remove the last empty line
|
||
|
call s:exec('$d _')
|
||
|
|
||
|
" restore previous cursor position.
|
||
|
call winrestview(savedview)
|
||
|
|
||
|
setlocal nomodifiable
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree.MarkSeqs() abort
|
||
|
call s:log("bak(cur,curhead,newhead): ".
|
||
|
\self.seq_cur_bak.' '.
|
||
|
\self.seq_curhead_bak.' '.
|
||
|
\self.seq_newhead_bak)
|
||
|
call s:log("(cur,curhead,newhead): ".
|
||
|
\self.seq_cur.' '.
|
||
|
\self.seq_curhead.' '.
|
||
|
\self.seq_newhead)
|
||
|
setlocal modifiable
|
||
|
" reset bak seq lines.
|
||
|
if self.seq_cur_bak != -1
|
||
|
let index = self.seq2index[self.seq_cur_bak]
|
||
|
call setline(self.Index2Screen(index),self.asciitree[index])
|
||
|
endif
|
||
|
if self.seq_curhead_bak != -1
|
||
|
let index = self.seq2index[self.seq_curhead_bak]
|
||
|
call setline(self.Index2Screen(index),self.asciitree[index])
|
||
|
endif
|
||
|
if self.seq_newhead_bak != -1
|
||
|
let index = self.seq2index[self.seq_newhead_bak]
|
||
|
call setline(self.Index2Screen(index),self.asciitree[index])
|
||
|
endif
|
||
|
" mark save seqs
|
||
|
for i in keys(self.seq_saved)
|
||
|
let index = self.seq2index[self.seq_saved[i]]
|
||
|
let lineNr = self.Index2Screen(index)
|
||
|
call setline(lineNr,substitute(self.asciitree[index],
|
||
|
\' \d\+ \zs \ze','s',''))
|
||
|
endfor
|
||
|
let max_saved_num = max(keys(self.seq_saved))
|
||
|
if max_saved_num > 0
|
||
|
let lineNr = self.Index2Screen(self.seq2index[self.seq_saved[max_saved_num]])
|
||
|
call setline(lineNr,substitute(getline(lineNr),'s','S',''))
|
||
|
endif
|
||
|
" mark new seqs.
|
||
|
if self.seq_cur != -1
|
||
|
let index = self.seq2index[self.seq_cur]
|
||
|
let lineNr = self.Index2Screen(index)
|
||
|
call setline(lineNr,substitute(getline(lineNr),
|
||
|
\'\zs \(\d\+\) \ze [sS ] ','>\1<',''))
|
||
|
" move cursor to that line.
|
||
|
call s:exec("normal! " . lineNr . "G")
|
||
|
endif
|
||
|
if self.seq_curhead != -1
|
||
|
let index = self.seq2index[self.seq_curhead]
|
||
|
let lineNr = self.Index2Screen(index)
|
||
|
call setline(lineNr,substitute(getline(lineNr),
|
||
|
\'\zs \(\d\+\) \ze [sS ] ','{\1}',''))
|
||
|
endif
|
||
|
if self.seq_newhead != -1
|
||
|
let index = self.seq2index[self.seq_newhead]
|
||
|
let lineNr = self.Index2Screen(index)
|
||
|
call setline(lineNr,substitute(getline(lineNr),
|
||
|
\'\zs \(\d\+\) \ze [sS ] ','[\1]',''))
|
||
|
endif
|
||
|
setlocal nomodifiable
|
||
|
endfunction
|
||
|
|
||
|
" tree node class
|
||
|
let s:node = {}
|
||
|
|
||
|
function! s:node.Init() abort
|
||
|
let self.seq = -1
|
||
|
let self.p = []
|
||
|
let self.time = -1
|
||
|
endfunction
|
||
|
|
||
|
function! s:undotree._parseNode(in,out) abort
|
||
|
" type(in) == type([]) && type(out) == type({})
|
||
|
if empty(a:in) "empty
|
||
|
return
|
||
|
endif
|
||
|
let curnode = a:out
|
||
|
for i in a:in
|
||
|
if has_key(i,'alt')
|
||
|
call self._parseNode(i.alt,curnode)
|
||
|
endif
|
||
|
let newnode = s:new(s:node)
|
||
|
let newnode.seq = i.seq
|
||
|
let newnode.time = i.time
|
||
|
if has_key(i,'newhead')
|
||
|
let self.seq_newhead = i.seq
|
||
|
endif
|
||
|
if has_key(i,'curhead')
|
||
|
let self.seq_curhead = i.seq
|
||
|
let self.seq_cur = curnode.seq
|
||
|
endif
|
||
|
if has_key(i,'save')
|
||
|
let self.seq_saved[i.save] = i.seq
|
||
|
endif
|
||
|
call extend(curnode.p,[newnode])
|
||
|
let curnode = newnode
|
||
|
endfor
|
||
|
endfunction
|
||
|
|
||
|
"Sample:
|
||
|
"let s:test={'seq_last': 4, 'entries': [{'seq': 3, 'alt': [{'seq': 1, 'time': 1345131443}, {'seq': 2, 'time': 1345131445}], 'time': 1345131490}, {'seq': 4, 'time': 1345131492, 'newhead': 1}], 'time_cur': 1345131493, 'save_last': 0, 'synced': 0, 'save_cur': 0, 'seq_cur': 4}
|
||
|
|
||
|
" updatetree: 0: no update, just assign seqs; 1: update and assign seqs.
|
||
|
function! s:undotree.ConvertInput(updatetree) abort
|
||
|
"reset seqs
|
||
|
let self.seq_cur_bak = self.seq_cur
|
||
|
let self.seq_curhead_bak = self.seq_curhead
|
||
|
let self.seq_newhead_bak = self.seq_newhead
|
||
|
let self.save_last_bak = self.save_last
|
||
|
|
||
|
let self.seq_cur = -1
|
||
|
let self.seq_curhead = -1
|
||
|
let self.seq_newhead = -1
|
||
|
let self.seq_saved = {}
|
||
|
|
||
|
"Generate root node
|
||
|
let root = s:new(s:node)
|
||
|
let root.seq = 0
|
||
|
let root.time = 0
|
||
|
|
||
|
call self._parseNode(self.rawtree.entries,root)
|
||
|
|
||
|
let self.save_last = self.rawtree.save_last
|
||
|
" Note: Normally, the current node should be the one that seq_cur points to,
|
||
|
" but in fact it's not. May be bug, bug anyway I found a workaround:
|
||
|
" first try to find the parent node of 'curhead', if not found, then use
|
||
|
" seq_cur.
|
||
|
if self.seq_cur == -1
|
||
|
let self.seq_cur = self.rawtree.seq_cur
|
||
|
endif
|
||
|
" undo history is cleared
|
||
|
if empty(self.rawtree.entries)
|
||
|
let self.seq_cur = 0
|
||
|
endif
|
||
|
if a:updatetree
|
||
|
let self.tree = root
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
"=================================================
|
||
|
" Ascii undo tree generator
|
||
|
"
|
||
|
" Example:
|
||
|
" 6 8 7
|
||
|
" |/ |
|
||
|
" 2 4
|
||
|
" \ |
|
||
|
" 1 3 5
|
||
|
" \ | /
|
||
|
" 0
|
||
|
|
||
|
" Tree sieve, p:fork, x:none
|
||
|
"
|
||
|
" x 8
|
||
|
" 8x | 7
|
||
|
" 87 \ \
|
||
|
" x87 6 | |
|
||
|
" 687 |/ /
|
||
|
" p7x | | 5
|
||
|
" p75 | 4 |
|
||
|
" p45 | 3 |
|
||
|
" p35 | |/
|
||
|
" pp 2 |
|
||
|
" 2p 1 |
|
||
|
" 1p |/
|
||
|
" p 0
|
||
|
" 0
|
||
|
"
|
||
|
" Data sample:
|
||
|
"let example = {'seq':0,'p':[{'seq':1,'p':[{'seq':2,'p':[{'seq':6,'p':[]},{'seq':8,'p':[]}]}]},{'seq':3,'p':[{'seq':4,'p':[{'seq':7,'p':[]}]}]},{'seq':5,'p':[]}]}
|
||
|
"
|
||
|
" Convert self.tree -> self.asciitree
|
||
|
function! s:undotree.Render() abort
|
||
|
" We gonna modify self.tree so we'd better make a copy first.
|
||
|
" Cannot make a copy because variable nested too deep, gosh.. okay,
|
||
|
" fine..
|
||
|
" let tree = deepcopy(self.tree)
|
||
|
let tree = self.tree
|
||
|
let slots = [tree]
|
||
|
let out = []
|
||
|
let outmeta = []
|
||
|
let seq2index = {}
|
||
|
let TYPE_E = type({})
|
||
|
let TYPE_P = type([])
|
||
|
let TYPE_X = type('x')
|
||
|
while slots != []
|
||
|
"find next node
|
||
|
let foundx = 0 " 1 if x element is found.
|
||
|
let index = 0 " Next element to be print.
|
||
|
|
||
|
" Find x element first.
|
||
|
for i in range(len(slots))
|
||
|
if type(slots[i]) == TYPE_X
|
||
|
let foundx = 1
|
||
|
let index = i
|
||
|
break
|
||
|
endif
|
||
|
endfor
|
||
|
|
||
|
" Then, find the element with minimun seq.
|
||
|
let minseq = 99999999
|
||
|
let minnode = {}
|
||
|
if foundx == 0
|
||
|
"assume undo level isn't more than this... of course
|
||
|
for i in range(len(slots))
|
||
|
if type(slots[i]) == TYPE_E
|
||
|
if slots[i].seq < minseq
|
||
|
let minseq = slots[i].seq
|
||
|
let index = i
|
||
|
let minnode = slots[i]
|
||
|
continue
|
||
|
endif
|
||
|
endif
|
||
|
if type(slots[i]) == TYPE_P
|
||
|
for j in slots[i]
|
||
|
if j.seq < minseq
|
||
|
let minseq = j.seq
|
||
|
let index = i
|
||
|
let minnode = j
|
||
|
continue
|
||
|
endif
|
||
|
endfor
|
||
|
endif
|
||
|
endfor
|
||
|
endif
|
||
|
|
||
|
" output.
|
||
|
let onespace = " "
|
||
|
let newline = onespace
|
||
|
let newmeta = {}
|
||
|
let node = slots[index]
|
||
|
if type(node) == TYPE_X
|
||
|
let newmeta = s:new(s:node) "invalid node.
|
||
|
if index+1 != len(slots) " not the last one, append '\'
|
||
|
for i in range(len(slots))
|
||
|
if i < index
|
||
|
let newline = newline.g:undotree_TreeVertShape.' '
|
||
|
endif
|
||
|
if i > index
|
||
|
let newline = newline.' '.g:undotree_TreeReturnShape
|
||
|
endif
|
||
|
endfor
|
||
|
endif
|
||
|
call remove(slots,index)
|
||
|
endif
|
||
|
if type(node) == TYPE_E
|
||
|
let newmeta = node
|
||
|
let seq2index[node.seq]=len(out)
|
||
|
for i in range(len(slots))
|
||
|
if index == i
|
||
|
let newline = newline.g:undotree_TreeNodeShape.' '
|
||
|
else
|
||
|
let newline = newline.g:undotree_TreeVertShape.' '
|
||
|
endif
|
||
|
endfor
|
||
|
let newline = newline.' '.(node.seq).' '.
|
||
|
\'('.s:gettime(node.time).')'
|
||
|
" update the printed slot to its child.
|
||
|
if empty(node.p)
|
||
|
let slots[index] = 'x'
|
||
|
endif
|
||
|
if len(node.p) == 1 "only one child.
|
||
|
let slots[index] = node.p[0]
|
||
|
endif
|
||
|
if len(node.p) > 1 "insert p node
|
||
|
let slots[index] = node.p
|
||
|
endif
|
||
|
let node.p = [] "cut reference.
|
||
|
endif
|
||
|
if type(node) == TYPE_P
|
||
|
let newmeta = s:new(s:node) "invalid node.
|
||
|
for k in range(len(slots))
|
||
|
if k < index
|
||
|
let newline = newline.g:undotree_TreeVertShape." "
|
||
|
endif
|
||
|
if k == index
|
||
|
let newline = newline.g:undotree_TreeVertShape.g:undotree_TreeSplitShape." "
|
||
|
endif
|
||
|
if k > index
|
||
|
let newline = newline.g:undotree_TreeSplitShape." "
|
||
|
endif
|
||
|
endfor
|
||
|
call remove(slots,index)
|
||
|
if len(node) == 2
|
||
|
if node[0].seq > node[1].seq
|
||
|
call insert(slots,node[1],index)
|
||
|
call insert(slots,node[0],index)
|
||
|
else
|
||
|
call insert(slots,node[0],index)
|
||
|
call insert(slots,node[1],index)
|
||
|
endif
|
||
|
endif
|
||
|
" split P to E+P if elements in p > 2
|
||
|
if len(node) > 2
|
||
|
call remove(node,index(node,minnode))
|
||
|
call insert(slots,minnode,index)
|
||
|
call insert(slots,node,index)
|
||
|
endif
|
||
|
endif
|
||
|
unlet node
|
||
|
if newline != onespace
|
||
|
let newline = substitute(newline,'\s*$','','g') "remove trailing space.
|
||
|
call insert(out,newline,0)
|
||
|
call insert(outmeta,newmeta,0)
|
||
|
endif
|
||
|
endwhile
|
||
|
let self.asciitree = out
|
||
|
let self.asciimeta = outmeta
|
||
|
" revert index.
|
||
|
let totallen = len(out)
|
||
|
for i in keys(seq2index)
|
||
|
let seq2index[i] = totallen - 1 - seq2index[i]
|
||
|
endfor
|
||
|
let self.seq2index = seq2index
|
||
|
endfunction
|
||
|
|
||
|
"=================================================
|
||
|
"diff panel
|
||
|
let s:diffpanel = s:new(s:panel)
|
||
|
|
||
|
function! s:diffpanel.Update(seq,targetBufnr,targetid) abort
|
||
|
call s:log('diffpanel.Update(),seq:'.a:seq.' bufname:'.bufname(a:targetBufnr))
|
||
|
if !self.diffexecutable
|
||
|
return
|
||
|
endif
|
||
|
let diffresult = []
|
||
|
let self.changes.add = 0
|
||
|
let self.changes.del = 0
|
||
|
|
||
|
if a:seq == 0
|
||
|
let diffresult = []
|
||
|
else
|
||
|
if has_key(self.cache,a:targetBufnr.'_'.a:seq)
|
||
|
call s:log("diff cache hit.")
|
||
|
let diffresult = self.cache[a:targetBufnr.'_'.a:seq]
|
||
|
else
|
||
|
" Double check the target winnr and bufnr
|
||
|
let targetWinnr = -1
|
||
|
for winnr in range(1, winnr('$')) "winnr starts from 1
|
||
|
if (getwinvar(winnr,'undotree_id') == a:targetid)
|
||
|
\&& winbufnr(winnr) == a:targetBufnr
|
||
|
let targetWinnr = winnr
|
||
|
endif
|
||
|
endfor
|
||
|
if targetWinnr == -1
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
let ei_bak = &eventignore
|
||
|
set eventignore=all
|
||
|
|
||
|
call s:exec_silent(targetWinnr." wincmd w")
|
||
|
|
||
|
" remember and restore cursor and window position.
|
||
|
let savedview = winsaveview()
|
||
|
|
||
|
let new = getbufline(a:targetBufnr,'^','$')
|
||
|
silent undo
|
||
|
let old = getbufline(a:targetBufnr,'^','$')
|
||
|
silent redo
|
||
|
|
||
|
call winrestview(savedview)
|
||
|
|
||
|
" diff files.
|
||
|
let tempfile1 = tempname()
|
||
|
let tempfile2 = tempname()
|
||
|
if writefile(old,tempfile1) == -1
|
||
|
echoerr "Can not write to temp file:".tempfile1
|
||
|
endif
|
||
|
if writefile(new,tempfile2) == -1
|
||
|
echoerr "Can not write to temp file:".tempfile2
|
||
|
endif
|
||
|
let diffresult = split(system(g:undotree_DiffCommand.' '.tempfile1.' '.tempfile2),"\n")
|
||
|
call s:log("diffresult: ".string(diffresult))
|
||
|
if delete(tempfile1) != 0
|
||
|
echoerr "Can not delete temp file:".tempfile1
|
||
|
endif
|
||
|
if delete(tempfile2) != 0
|
||
|
echoerr "Can not delete temp file:".tempfile2
|
||
|
endif
|
||
|
let &eventignore = ei_bak
|
||
|
"Update cache
|
||
|
let self.cache[a:targetBufnr.'_'.a:seq] = diffresult
|
||
|
endif
|
||
|
endif
|
||
|
|
||
|
call self.ParseDiff(diffresult, a:targetBufnr)
|
||
|
|
||
|
call self.SetFocus()
|
||
|
|
||
|
setlocal modifiable
|
||
|
call s:exec('1,$ d _')
|
||
|
|
||
|
call append(0,diffresult)
|
||
|
call append(0,'- seq: '.a:seq.' -')
|
||
|
|
||
|
"remove the last empty line
|
||
|
if getline("$") == ""
|
||
|
call s:exec('$d _')
|
||
|
endif
|
||
|
call s:exec('norm! gg') "move cursor to line 1.
|
||
|
setlocal nomodifiable
|
||
|
call t:undotree.SetFocus()
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.ParseDiff(diffresult, targetBufnr) abort
|
||
|
" set target focus first.
|
||
|
call t:undotree.SetTargetFocus()
|
||
|
|
||
|
" If 'a:diffresult' is empty then there are no new signs to place. However,
|
||
|
" we need to ensure any old signs are removed. This is especially important
|
||
|
" if we are at the very first sequence, otherwise signs get left
|
||
|
if (exists("w:undotree_diffsigns"))
|
||
|
while w:undotree_diffsigns > 0
|
||
|
exe 'sign unplace '.s:signId
|
||
|
let w:undotree_diffsigns -= 1
|
||
|
endwhile
|
||
|
endif
|
||
|
|
||
|
if empty(a:diffresult)
|
||
|
return
|
||
|
endif
|
||
|
|
||
|
" clear previous highlighted syntax
|
||
|
" matchadd associates with windows.
|
||
|
if exists("w:undotree_diffmatches")
|
||
|
for i in w:undotree_diffmatches
|
||
|
silent! call matchdelete(i)
|
||
|
endfor
|
||
|
endif
|
||
|
|
||
|
let w:undotree_diffmatches = []
|
||
|
let w:undotree_diffsigns = 0
|
||
|
let lineNr = 0
|
||
|
let l:lastLine = line('$')
|
||
|
for line in a:diffresult
|
||
|
let matchnum = matchstr(line,'^[0-9,\,]*[acd]\zs\d*\ze')
|
||
|
if !empty(matchnum)
|
||
|
let lineNr = str2nr(matchnum)
|
||
|
let matchwhat = matchstr(line,'^[0-9,\,]*\zs[acd]\ze\d*')
|
||
|
if matchwhat ==# 'd'
|
||
|
if g:undotree_HighlightChangedWithSign
|
||
|
" Normally, for a 'delete' change, the line number we have is always 1 less than the line we
|
||
|
" need to place the sign at, hence '+ 1'
|
||
|
" However, if the very end of the buffer has been deleted then this is not possible (because
|
||
|
" that bit of the buffer no longer exists), so we place a 'special' version of the 'delete'
|
||
|
" sign on what is the last available line)
|
||
|
exe 'sign place '.s:signId.' line='.((lineNr < l:lastLine) ? lineNr + 1 : l:lastLine).' name='.((lineNr < l:lastLine) ? 'UndotreeDel' : 'UndotreeDelEnd').' buffer='.a:targetBufnr
|
||
|
let w:undotree_diffsigns += 1
|
||
|
endif
|
||
|
|
||
|
let matchnum = 0
|
||
|
let matchwhat = ''
|
||
|
endif
|
||
|
continue
|
||
|
endif
|
||
|
if matchstr(line,'^<.*$') != ''
|
||
|
let self.changes.del += 1
|
||
|
endif
|
||
|
|
||
|
let matchtext = matchstr(line,'^>\zs .*$')
|
||
|
if empty(matchtext)
|
||
|
continue
|
||
|
endif
|
||
|
|
||
|
let self.changes.add += 1
|
||
|
if g:undotree_HighlightChangedText
|
||
|
if matchtext != ' '
|
||
|
let matchtext = '\%'.lineNr.'l\V'.escape(matchtext[1:],'"\') "remove beginning space.
|
||
|
call s:log("matchadd(".matchwhat.") -> ".matchtext)
|
||
|
call add(w:undotree_diffmatches,matchadd((matchwhat ==# 'a' ? g:undotree_HighlightSyntaxAdd : g:undotree_HighlightSyntaxChange),matchtext))
|
||
|
endif
|
||
|
endif
|
||
|
|
||
|
if g:undotree_HighlightChangedWithSign
|
||
|
exe 'sign place '.s:signId.' line='.lineNr.' name='.(matchwhat ==# 'a' ? 'UndotreeAdd' : 'UndotreeChg').' buffer='.a:targetBufnr
|
||
|
let w:undotree_diffsigns += 1
|
||
|
endif
|
||
|
|
||
|
let lineNr = lineNr+1
|
||
|
endfor
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.GetStatusLine() abort
|
||
|
let max = winwidth(0) - 4
|
||
|
let sum = self.changes.add + self.changes.del
|
||
|
if sum > max
|
||
|
let add = self.changes.add * max / sum + 1
|
||
|
let del = self.changes.del * max / sum + 1
|
||
|
else
|
||
|
let add = self.changes.add
|
||
|
let del = self.changes.del
|
||
|
endif
|
||
|
return string(sum).' '.repeat('+',add).repeat('-',del)
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.Init() abort
|
||
|
let self.bufname = "diffpanel_".s:getUniqueID()
|
||
|
let self.cache = {}
|
||
|
let self.changes = {'add':0, 'del':0}
|
||
|
let self.diffexecutable = executable(g:undotree_DiffCommand)
|
||
|
if !self.diffexecutable
|
||
|
" If the command contains parameters, strip out the executable itself
|
||
|
let cmd = matchstr(g:undotree_DiffCommand.' ', '.\{-}\ze\s.*')
|
||
|
let self.diffexecutable = executable(cmd)
|
||
|
if !self.diffexecutable
|
||
|
echoerr '"'.cmd.'" is not executable.'
|
||
|
endif
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.Toggle() abort
|
||
|
call s:log(self.bufname." Toggle()")
|
||
|
if self.IsVisible()
|
||
|
call self.Hide()
|
||
|
else
|
||
|
call self.Show()
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.Show() abort
|
||
|
call s:log("diffpanel.Show()")
|
||
|
if self.IsVisible()
|
||
|
return
|
||
|
endif
|
||
|
" Create diffpanel window.
|
||
|
call t:undotree.SetFocus() "can not exist without undotree
|
||
|
" remember and restore cursor and window position.
|
||
|
let savedview = winsaveview()
|
||
|
|
||
|
let ei_bak= &eventignore
|
||
|
set eventignore=all
|
||
|
|
||
|
if exists("g:undotree_CustomDiffpanelCmd")
|
||
|
let cmd = g:undotree_CustomDiffpanelCmd.' '.self.bufname
|
||
|
elseif g:undotree_WindowLayout == 1 || g:undotree_WindowLayout == 3
|
||
|
let cmd = 'belowright '.g:undotree_DiffpanelHeight.'new '.self.bufname
|
||
|
else
|
||
|
let cmd = 'botright '.g:undotree_DiffpanelHeight.'new '.self.bufname
|
||
|
endif
|
||
|
call s:exec_silent(cmd)
|
||
|
|
||
|
let b:isUndotreeBuffer = 1
|
||
|
|
||
|
setlocal winfixwidth
|
||
|
setlocal winfixheight
|
||
|
setlocal noswapfile
|
||
|
setlocal buftype=nowrite
|
||
|
setlocal bufhidden=delete
|
||
|
setlocal nowrap
|
||
|
setlocal nolist
|
||
|
setlocal nobuflisted
|
||
|
setlocal nospell
|
||
|
setlocal nonumber
|
||
|
setlocal norelativenumber
|
||
|
setlocal nocursorline
|
||
|
setlocal nomodifiable
|
||
|
setlocal statusline=%!t:diffpanel.GetStatusLine()
|
||
|
|
||
|
let &eventignore = ei_bak
|
||
|
|
||
|
" syntax need filetype autocommand
|
||
|
setfiletype diff
|
||
|
setlocal foldcolumn=0
|
||
|
setlocal nofoldenable
|
||
|
|
||
|
call self.BindAu()
|
||
|
call t:undotree.SetFocus()
|
||
|
call winrestview(savedview)
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.BindAu() abort
|
||
|
" Auto exit if it's the last window or undotree closed.
|
||
|
augroup Undotree_Diff
|
||
|
au!
|
||
|
au BufEnter <buffer> call s:exitIfLast()
|
||
|
au BufEnter <buffer> if !t:undotree.IsVisible()
|
||
|
\|call t:diffpanel.Hide() |endif
|
||
|
augroup end
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.CleanUpHighlight() abort
|
||
|
call s:log("CleanUpHighlight()")
|
||
|
" save current position
|
||
|
let curwinnr = winnr()
|
||
|
let savedview = winsaveview()
|
||
|
|
||
|
" clear w:undotree_diffmatches in all windows.
|
||
|
let winnum = winnr('$')
|
||
|
for i in range(1,winnum)
|
||
|
call s:exec_silent("norm! ".i."\<c-w>\<c-w>")
|
||
|
if exists("w:undotree_diffmatches")
|
||
|
for j in w:undotree_diffmatches
|
||
|
silent! call matchdelete(j)
|
||
|
endfor
|
||
|
let w:undotree_diffmatches = []
|
||
|
endif
|
||
|
if (exists("w:undotree_diffsigns"))
|
||
|
while w:undotree_diffsigns > 0
|
||
|
exe 'sign unplace '.s:signId
|
||
|
let w:undotree_diffsigns -= 1
|
||
|
endwhile
|
||
|
endif
|
||
|
endfor
|
||
|
|
||
|
"restore position
|
||
|
call s:exec_silent("norm! ".curwinnr."\<c-w>\<c-w>")
|
||
|
call winrestview(savedview)
|
||
|
endfunction
|
||
|
|
||
|
function! s:diffpanel.Hide() abort
|
||
|
call s:log(self.bufname." Hide()")
|
||
|
if !self.IsVisible()
|
||
|
return
|
||
|
endif
|
||
|
call self.SetFocus()
|
||
|
call s:exec("quit")
|
||
|
call self.CleanUpHighlight()
|
||
|
endfunction
|
||
|
|
||
|
"=================================================
|
||
|
" It will set the target of undotree window to the current editing buffer.
|
||
|
function! s:undotreeAction(action) abort
|
||
|
call s:log("undotreeAction()")
|
||
|
if !exists('t:undotree')
|
||
|
echoerr "Fatal: t:undotree does not exist!"
|
||
|
return
|
||
|
endif
|
||
|
call t:undotree.Action(a:action)
|
||
|
endfunction
|
||
|
|
||
|
function! s:exitIfLast() abort
|
||
|
let num = 0
|
||
|
if exists('t:undotree') && t:undotree.IsVisible()
|
||
|
let num = num + 1
|
||
|
endif
|
||
|
if exists('t:diffpanel') && t:diffpanel.IsVisible()
|
||
|
let num = num + 1
|
||
|
endif
|
||
|
if winnr('$') == num
|
||
|
if exists('t:undotree')
|
||
|
call t:undotree.Hide()
|
||
|
endif
|
||
|
if exists('t:diffpanel')
|
||
|
call t:diffpanel.Hide()
|
||
|
endif
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
"=================================================
|
||
|
" User command functions
|
||
|
"called outside undotree window
|
||
|
function! undotree#UndotreeUpdate() abort
|
||
|
if !exists('t:undotree')
|
||
|
return
|
||
|
endif
|
||
|
if !exists('w:undotree_id')
|
||
|
let w:undotree_id = 'id_'.s:getUniqueID()
|
||
|
call s:log("Unique window id assigned: ".w:undotree_id)
|
||
|
endif
|
||
|
" assume window layout won't change during updating.
|
||
|
let thiswinnr = winnr()
|
||
|
call t:undotree.Update()
|
||
|
" focus moved
|
||
|
if winnr() != thiswinnr
|
||
|
call s:exec("norm! ".thiswinnr."\<c-w>\<c-w>")
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! undotree#UndotreeToggle() abort
|
||
|
try
|
||
|
call s:log(">>> UndotreeToggle()")
|
||
|
if !exists('w:undotree_id')
|
||
|
let w:undotree_id = 'id_'.s:getUniqueID()
|
||
|
call s:log("Unique window id assigned: ".w:undotree_id)
|
||
|
endif
|
||
|
if !exists('t:undotree')
|
||
|
let t:undotree = s:new(s:undotree)
|
||
|
endif
|
||
|
if !exists('t:diffpanel')
|
||
|
let t:diffpanel = s:new(s:diffpanel)
|
||
|
endif
|
||
|
call t:undotree.Toggle()
|
||
|
call s:log("<<< UndotreeToggle() leave")
|
||
|
catch /^Vim\%((\a\+)\)\?:E11/
|
||
|
echohl ErrorMsg
|
||
|
echom v:exception
|
||
|
echohl NONE
|
||
|
endtry
|
||
|
endfunction
|
||
|
|
||
|
function! undotree#UndotreeIsVisible() abort
|
||
|
return (exists('t:undotree') && t:undotree.IsVisible())
|
||
|
endfunction
|
||
|
|
||
|
function! undotree#UndotreeHide() abort
|
||
|
if undotree#UndotreeIsVisible()
|
||
|
try
|
||
|
call undotree#UndotreeToggle()
|
||
|
catch /^Vim\%((\a\+)\)\?:E11/
|
||
|
echohl ErrorMsg
|
||
|
echom v:exception
|
||
|
echohl NONE
|
||
|
endtry
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! undotree#UndotreeShow() abort
|
||
|
try
|
||
|
if ! undotree#UndotreeIsVisible()
|
||
|
call undotree#UndotreeToggle()
|
||
|
else
|
||
|
call t:undotree.SetFocus()
|
||
|
endif
|
||
|
catch /^Vim\%((\a\+)\)\?:E11/
|
||
|
echohl ErrorMsg
|
||
|
echom v:exception
|
||
|
echohl NONE
|
||
|
endtry
|
||
|
endfunction
|
||
|
|
||
|
function! undotree#UndotreeFocus() abort
|
||
|
if undotree#UndotreeIsVisible()
|
||
|
try
|
||
|
call t:undotree.SetFocus()
|
||
|
catch /^Vim\%((\a\+)\)\?:E11/
|
||
|
echohl ErrorMsg
|
||
|
echom v:exception
|
||
|
echohl NONE
|
||
|
endtry
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
function! undotree#UndotreePersistUndo(goSetUndofile) abort
|
||
|
call s:log("undotree#UndotreePersistUndo(" . a:goSetUndofile . ")")
|
||
|
if ! &undofile
|
||
|
if !isdirectory(g:undotree_UndoDir)
|
||
|
call mkdir(g:undotree_UndoDir, 'p', 0700)
|
||
|
call s:log(" > [Dir " . g:undotree_UndoDir . "] created.")
|
||
|
endif
|
||
|
exe "set undodir=" . fnameescape(g:undotree_UndoDir)
|
||
|
call s:log(" > [set undodir=" . g:undotree_UndoDir . "] executed.")
|
||
|
if filereadable(undofile(expand('%'))) || a:goSetUndofile
|
||
|
setlocal undofile
|
||
|
call s:log(" > [setlocal undofile] executed")
|
||
|
endif
|
||
|
if a:goSetUndofile
|
||
|
silent! write
|
||
|
echo "A persistence undo file has been created."
|
||
|
endif
|
||
|
else
|
||
|
call s:log(" > Undofile has been set. Do nothing.")
|
||
|
endif
|
||
|
endfunction
|
||
|
|
||
|
" vim: set et fdm=marker sts=4 sw=4:
|