undo.vim 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. " =============================================================================
  2. " File: autoload/ctrlp/undo.vim
  3. " Description: Undo extension
  4. " Author: Kien Nguyen <github.com/kien>
  5. " =============================================================================
  6. " Init {{{1
  7. if ( exists('g:loaded_ctrlp_undo') && g:loaded_ctrlp_undo )
  8. fini
  9. en
  10. let g:loaded_ctrlp_undo = 1
  11. cal add(g:ctrlp_ext_vars, {
  12. \ 'init': 'ctrlp#undo#init()',
  13. \ 'accept': 'ctrlp#undo#accept',
  14. \ 'lname': 'undo',
  15. \ 'sname': 'udo',
  16. \ 'enter': 'ctrlp#undo#enter()',
  17. \ 'exit': 'ctrlp#undo#exit()',
  18. \ 'type': 'line',
  19. \ 'sort': 0,
  20. \ 'nolim': 1,
  21. \ })
  22. let s:id = g:ctrlp_builtins + len(g:ctrlp_ext_vars)
  23. let s:text = map(['second', 'seconds', 'minutes', 'hours', 'days', 'weeks',
  24. \ 'months', 'years'], '" ".v:val." ago"')
  25. " Utilities {{{1
  26. fu! s:getundo()
  27. if exists('*undotree')
  28. \ && ( v:version > 703 || ( v:version == 703 && has('patch005') ) )
  29. retu [1, undotree()]
  30. el
  31. redi => result
  32. sil! undol
  33. redi END
  34. retu [0, split(result, "\n")[1:]]
  35. en
  36. endf
  37. fu! s:flatten(tree, cur)
  38. let flatdict = {}
  39. for each in a:tree
  40. let saved = has_key(each, 'save') ? 'saved' : ''
  41. let current = each['seq'] == a:cur ? 'current' : ''
  42. cal extend(flatdict, { each['seq'] : [each['time'], saved, current] })
  43. if has_key(each, 'alt')
  44. cal extend(flatdict, s:flatten(each['alt'], a:cur))
  45. en
  46. endfo
  47. retu flatdict
  48. endf
  49. fu! s:elapsed(nr)
  50. let [text, time] = [s:text, localtime() - a:nr]
  51. let mins = time / 60
  52. let hrs = time / 3600
  53. let days = time / 86400
  54. let wks = time / 604800
  55. let mons = time / 2592000
  56. let yrs = time / 31536000
  57. if yrs > 1
  58. retu yrs.text[7]
  59. elsei mons > 1
  60. retu mons.text[6]
  61. elsei wks > 1
  62. retu wks.text[5]
  63. elsei days > 1
  64. retu days.text[4]
  65. elsei hrs > 1
  66. retu hrs.text[3]
  67. elsei mins > 1
  68. retu mins.text[2]
  69. elsei time == 1
  70. retu time.text[0]
  71. elsei time < 120
  72. retu time.text[1]
  73. en
  74. endf
  75. fu! s:syntax()
  76. if ctrlp#nosy() | retu | en
  77. for [ke, va] in items({'T': 'Directory', 'Br': 'Comment', 'Nr': 'String',
  78. \ 'Sv': 'Comment', 'Po': 'Title'})
  79. cal ctrlp#hicheck('CtrlPUndo'.ke, va)
  80. endfo
  81. sy match CtrlPUndoT '\v\d+ \zs[^ ]+\ze|\d+:\d+:\d+'
  82. sy match CtrlPUndoBr '\[\|\]'
  83. sy match CtrlPUndoNr '\[\d\+\]' contains=CtrlPUndoBr
  84. sy match CtrlPUndoSv 'saved'
  85. sy match CtrlPUndoPo 'current'
  86. endf
  87. fu! s:dict2list(dict)
  88. for ke in keys(a:dict)
  89. let a:dict[ke][0] = s:elapsed(a:dict[ke][0])
  90. endfo
  91. retu map(keys(a:dict), 'eval(''[v:val, a:dict[v:val]]'')')
  92. endf
  93. fu! s:compval(...)
  94. retu a:2[0] - a:1[0]
  95. endf
  96. fu! s:format(...)
  97. let saved = !empty(a:1[1][1]) ? ' '.a:1[1][1] : ''
  98. let current = !empty(a:1[1][2]) ? ' '.a:1[1][2] : ''
  99. retu a:1[1][0].' ['.a:1[0].']'.saved.current
  100. endf
  101. fu! s:formatul(...)
  102. let parts = matchlist(a:1,
  103. \ '\v^\s+(\d+)\s+\d+\s+([^ ]+\s?[^ ]+|\d+\s\w+\s\w+)(\s*\d*)$')
  104. retu parts == [] ? '----'
  105. \ : parts[2].' ['.parts[1].']'.( parts[3] != '' ? ' saved' : '' )
  106. endf
  107. " Public {{{1
  108. fu! ctrlp#undo#init()
  109. let entries = s:undos[0] ? s:undos[1]['entries'] : s:undos[1]
  110. if empty(entries) | retu [] | en
  111. if !exists('s:lines')
  112. if s:undos[0]
  113. let entries = s:dict2list(s:flatten(entries, s:undos[1]['seq_cur']))
  114. let s:lines = map(sort(entries, 's:compval'), 's:format(v:val)')
  115. el
  116. let s:lines = map(reverse(entries), 's:formatul(v:val)')
  117. en
  118. en
  119. cal s:syntax()
  120. retu s:lines
  121. endf
  122. fu! ctrlp#undo#accept(mode, str)
  123. let undon = matchstr(a:str, '\[\zs\d\+\ze\]')
  124. if empty(undon) | retu | en
  125. cal ctrlp#exit()
  126. exe 'u' undon
  127. endf
  128. fu! ctrlp#undo#id()
  129. retu s:id
  130. endf
  131. fu! ctrlp#undo#enter()
  132. let s:undos = s:getundo()
  133. endf
  134. fu! ctrlp#undo#exit()
  135. unl! s:lines
  136. endf
  137. "}}}
  138. " vim:fen:fdm=marker:fmr={{{,}}}:fdl=0:fdc=1:ts=2:sw=2:sts=2