bufexplorer.vim 57 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697
  1. "============================================================================
  2. " Copyright: Copyright (c) 2001-2025, Jeff Lanzarotta
  3. " All rights reserved.
  4. "
  5. " Redistribution and use in source and binary forms, with or
  6. " without modification, are permitted provided that the
  7. " following conditions are met:
  8. "
  9. " * Redistributions of source code must retain the above
  10. " copyright notice, this list of conditions and the following
  11. " disclaimer.
  12. "
  13. " * Redistributions in binary form must reproduce the above
  14. " copyright notice, this list of conditions and the following
  15. " disclaimer in the documentation and/or other materials
  16. " provided with the distribution.
  17. "
  18. " * Neither the name of the {organization} nor the names of its
  19. " contributors may be used to endorse or promote products
  20. " derived from this software without specific prior written
  21. " permission.
  22. "
  23. " THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
  24. " CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
  25. " INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
  26. " MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  27. " DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
  28. " CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  29. " SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
  30. " NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  31. " LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  32. " HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  33. " CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
  34. " OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
  35. " EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  36. " Name Of File: bufexplorer.vim
  37. " Description: Buffer Explorer Vim Plugin
  38. " Maintainer: Jeff Lanzarotta (my name at gmail dot com)
  39. " Last Changed: Thursday, 20 March 2025
  40. " Version: See g:bufexplorer_version for version number.
  41. " Usage: This file should reside in the plugin directory and be
  42. " automatically sourced.
  43. "
  44. " You may use the default keymappings of
  45. "
  46. " <Leader>be - Opens BufExplorer
  47. " <Leader>bt - Toggles BufExplorer open or closed
  48. " <Leader>bs - Opens horizontally split window BufExplorer
  49. " <Leader>bv - Opens vertically split window BufExplorer
  50. "
  51. " Or you can override the defaults and define your own mapping
  52. " in your vimrc file, for example:
  53. "
  54. " nnoremap <silent> <F11> :BufExplorer<CR>
  55. " nnoremap <silent> <s-F11> :ToggleBufExplorer<CR>
  56. " nnoremap <silent> <m-F11> :BufExplorerHorizontalSplit<CR>
  57. " nnoremap <silent> <c-F11> :BufExplorerVerticalSplit<CR>
  58. "
  59. " Or you can use
  60. "
  61. " ":BufExplorer" - Opens BufExplorer
  62. " ":ToggleBufExplorer" - Opens/Closes BufExplorer
  63. " ":BufExplorerHorizontalSplit" - Opens horizontally window BufExplorer
  64. " ":BufExplorerVerticalSplit" - Opens vertically split window BufExplorer
  65. "
  66. " For more help see supplied documentation.
  67. " History: See supplied documentation.
  68. "=============================================================================
  69. " Exit quickly if already running or when 'compatible' is set. {{{1
  70. if exists("g:bufexplorer_version") || &cp
  71. finish
  72. endif
  73. "1}}}
  74. " Version number.
  75. let g:bufexplorer_version = "7.9.0"
  76. " Plugin Code {{{1
  77. " Check for Vim version {{{2
  78. if !exists("g:bufExplorerVersionWarn")
  79. let g:bufExplorerVersionWarn = 1
  80. endif
  81. " Make sure we are using the correct version of Vim. If not, do not load the
  82. " plugin.
  83. if v:version < 704
  84. if g:bufExplorerVersionWarn
  85. echohl WarningMsg
  86. echo "Sorry, bufexplorer ".g:bufexplorer_version." required Vim 7.4 or greater."
  87. echohl None
  88. endif
  89. finish
  90. endif
  91. " Create commands {{{2
  92. command! BufExplorer :call BufExplorer()
  93. command! ToggleBufExplorer :call ToggleBufExplorer()
  94. command! BufExplorerHorizontalSplit :call BufExplorerHorizontalSplit()
  95. command! BufExplorerVerticalSplit :call BufExplorerVerticalSplit()
  96. " Set {{{2
  97. function! s:Set(var, default)
  98. if !exists(a:var)
  99. if type(a:default)
  100. execute "let" a:var "=" string(a:default)
  101. else
  102. execute "let" a:var "=" a:default
  103. endif
  104. return 1
  105. endif
  106. return 0
  107. endfunction
  108. " Script variables {{{2
  109. let s:MRU_Exclude_List = ["[BufExplorer]","__MRU_Files__","[Buf\ List]"]
  110. let s:name = '[BufExplorer]'
  111. let s:originBuffer = 0
  112. " Buffer number of the BufExplorer window.
  113. let s:bufExplorerBuffer = 0
  114. let s:running = 0
  115. let s:sort_by = ["number", "name", "fullpath", "mru", "extension"]
  116. let s:splitMode = ""
  117. let s:didSplit = 0
  118. let s:types = ["fullname", "homename", "path", "relativename", "relativepath", "shortname"]
  119. " Setup the autocommands that handle stuff. {{{2
  120. augroup BufExplorer
  121. autocmd!
  122. autocmd WinEnter * call s:DoWinEnter()
  123. autocmd BufEnter * call s:DoBufEnter()
  124. autocmd BufDelete * call s:DoBufDelete()
  125. if exists('##TabClosed')
  126. autocmd TabClosed * call s:DoTabClosed()
  127. endif
  128. autocmd BufWinEnter \[BufExplorer\] call s:Initialize()
  129. autocmd BufWinLeave \[BufExplorer\] call s:Cleanup()
  130. augroup END
  131. " AssignTabId {{{2
  132. " Assign a `tabId` to the given tab.
  133. function! s:AssignTabId(tabNbr)
  134. " Create a unique `tabId` based on the current time and an incrementing
  135. " counter value that helps ensure uniqueness.
  136. let tabId = reltimestr(reltime()) . ':' . s:tabIdCounter
  137. call settabvar(a:tabNbr, 'bufexp_tabId', tabId)
  138. let s:tabIdCounter = (s:tabIdCounter + 1) % 1000000000
  139. return tabId
  140. endfunction
  141. let s:tabIdCounter = 0
  142. " GetTabId {{{2
  143. " Retrieve the `tabId` for the given tab (or '' if the tab has no `tabId`).
  144. function! s:GetTabId(tabNbr)
  145. return gettabvar(a:tabNbr, 'bufexp_tabId', '')
  146. endfunction
  147. " MRU data structure {{{2
  148. " An MRU data structure is a dictionary that holds a circular doubly linked list
  149. " of `item` values. The dictionary contains three keys:
  150. " 'head': a sentinel `item` representing the head of the list.
  151. " 'next': a dictionary mapping an `item` to the next `item` in the list.
  152. " 'prev': a dictionary mapping an `item` to the previous `item` in the list.
  153. " E.g., an MRU holding buffer numbers will use `0` (an invalid buffer number) as
  154. " `head`. With the buffer numbers `1`, `2`, and `3`, an example MRU would be:
  155. "
  156. " +--<---------<---------<---------<---------<+
  157. " `next` | |
  158. " +--> +---+ --> +---+ --> +---+ --> +---+ -->+
  159. " `head` | 0 | | 1 | | 2 | | 3 |
  160. " +<-- +---+ <-- +---+ <-- +---+ <-- +---+ <--+
  161. " `prev` | |
  162. " +->-------->--------->--------->--------->--+
  163. "
  164. " `head` allows the chosen sentinel item to differ in value and type; for
  165. " example, `head` could be the string '.', allowing an MRU of strings (such as
  166. " for `TabId` values).
  167. "
  168. " Note that dictionary keys are always strings. Integers may be used, but they
  169. " are converted to strings when used (and `keys(theDictionary)` will be a
  170. " list of strings, not of integers).
  171. " MRUNew {{{2
  172. function! s:MRUNew(head)
  173. let [next, prev] = [{}, {}]
  174. let next[a:head] = a:head
  175. let prev[a:head] = a:head
  176. return { 'head': a:head, 'next': next, 'prev': prev }
  177. endfunction
  178. " MRULen {{{2
  179. function! s:MRULen(mru)
  180. " Do not include the always-present `mru.head` item.
  181. return len(a:mru.next) - 1
  182. endfunction
  183. " MRURemoveMustExist {{{2
  184. " `item` must exist in `mru`.
  185. function! s:MRURemoveMustExist(mru, item)
  186. let [next, prev] = [a:mru.next, a:mru.prev]
  187. let prevItem = prev[a:item]
  188. let nextItem = next[a:item]
  189. let next[prevItem] = nextItem
  190. let prev[nextItem] = prevItem
  191. unlet next[a:item]
  192. unlet prev[a:item]
  193. endfunction
  194. " MRURemove {{{2
  195. " `item` need not exist in `mru`.
  196. function! s:MRURemove(mru, item)
  197. if has_key(a:mru.next, a:item)
  198. call s:MRURemoveMustExist(a:mru, a:item)
  199. endif
  200. endfunction
  201. " MRUAdd {{{2
  202. function! s:MRUAdd(mru, item)
  203. let [next, prev] = [a:mru.next, a:mru.prev]
  204. let prevItem = a:mru.head
  205. let nextItem = next[prevItem]
  206. if a:item != nextItem
  207. call s:MRURemove(a:mru, a:item)
  208. let next[a:item] = nextItem
  209. let prev[a:item] = prevItem
  210. let next[prevItem] = a:item
  211. let prev[nextItem] = a:item
  212. endif
  213. endfunction
  214. " MRUGetItems {{{2
  215. " Return list of up to `maxItems` items in MRU order.
  216. " `maxItems == 0` => unlimited.
  217. function! s:MRUGetItems(mru, maxItems)
  218. let [head, next] = [a:mru.head, a:mru.next]
  219. let items = []
  220. let item = next[head]
  221. while item != head
  222. if a:maxItems > 0 && len(items) >= a:maxItems
  223. break
  224. endif
  225. call add(items, item)
  226. let item = next[item]
  227. endwhile
  228. return items
  229. endfunction
  230. " MRUGetOrdering {{{2
  231. " Return dictionary mapping up to `maxItems` from `item` to MRU order.
  232. " `maxItems == 0` => unlimited.
  233. function! s:MRUGetOrdering(mru, maxItems)
  234. let [head, next] = [a:mru.head, a:mru.next]
  235. let items = {}
  236. let order = 0
  237. let item = next[head]
  238. while item != head
  239. if a:maxItems > 0 && order >= a:maxItems
  240. break
  241. endif
  242. let items[item] = order
  243. let order = order + 1
  244. let item = next[item]
  245. endwhile
  246. return items
  247. endfunction
  248. " MRU trackers {{{2
  249. " `.head` value for tab MRU:
  250. let s:tabIdHead = '.'
  251. " Track MRU buffers globally (independent of tabs).
  252. let s:bufMru = s:MRUNew(0)
  253. " Track MRU buffers for each tab, indexed by `tabId`.
  254. " `s:bufMruByTab[tabId] -> MRU structure`.
  255. let s:bufMruByTab = {}
  256. " Track MRU tabs for each buffer, indexed by `bufNbr`.
  257. " `s:tabMruByBuf[burNbr] -> MRU structure`.
  258. let s:tabMruByBuf = {}
  259. " MRURemoveBuf {{{2
  260. function! s:MRURemoveBuf(bufNbr)
  261. call s:MRURemove(s:bufMru, a:bufNbr)
  262. if has_key(s:tabMruByBuf, a:bufNbr)
  263. let mru = s:tabMruByBuf[a:bufNbr]
  264. let [head, next] = [mru.head, mru.next]
  265. let tabId = next[head]
  266. while tabId != head
  267. call s:MRURemoveMustExist(s:bufMruByTab[tabId], a:bufNbr)
  268. let tabId = next[tabId]
  269. endwhile
  270. unlet s:tabMruByBuf[a:bufNbr]
  271. endif
  272. endfunction
  273. " MRURemoveTab {{{2
  274. function! s:MRURemoveTab(tabId)
  275. if has_key(s:bufMruByTab, a:tabId)
  276. let mru = s:bufMruByTab[a:tabId]
  277. let [head, next] = [mru.head, mru.next]
  278. let bufNbr = next[head]
  279. while bufNbr != head
  280. call s:MRURemoveMustExist(s:tabMruByBuf[bufNbr], a:tabId)
  281. let bufNbr = next[bufNbr]
  282. endwhile
  283. unlet s:bufMruByTab[a:tabId]
  284. endif
  285. endfunction
  286. " MRUAddBufTab {{{2
  287. function! s:MRUAddBufTab(bufNbr, tabId)
  288. if s:ShouldIgnore(a:bufNbr)
  289. return
  290. endif
  291. call s:MRUAdd(s:bufMru, a:bufNbr)
  292. if !has_key(s:bufMruByTab, a:tabId)
  293. let s:bufMruByTab[a:tabId] = s:MRUNew(0)
  294. endif
  295. let bufMru = s:bufMruByTab[a:tabId]
  296. call s:MRUAdd(bufMru, a:bufNbr)
  297. if !has_key(s:tabMruByBuf, a:bufNbr)
  298. let s:tabMruByBuf[a:bufNbr] = s:MRUNew(s:tabIdHead)
  299. endif
  300. let tabMru = s:tabMruByBuf[a:bufNbr]
  301. call s:MRUAdd(tabMru, a:tabId)
  302. endfunction
  303. " MRUTabForBuf {{{2
  304. " Return `tabId` most recently used by `bufNbr`.
  305. " If no `tabId` is found for `bufNbr`, return `s:tabIdHead`.
  306. function! s:MRUTabForBuf(bufNbr)
  307. let tabMru = get(s:tabMruByBuf, a:bufNbr, s:alwaysEmptyTabMru)
  308. return tabMru.next[tabMru.head]
  309. endfunction
  310. " An always-empty MRU for tabs as a default when looking up
  311. " `s:tabMruByBuf[bufNbr]` for an unknown `bufNbr`.
  312. let s:alwaysEmptyTabMru = s:MRUNew(s:tabIdHead)
  313. " MRUTabHasSeenBuf {{{2
  314. " Return true if `tabId` has ever seen `bufNbr`.
  315. function! s:MRUTabHasSeenBuf(tabId, bufNbr)
  316. let mru = get(s:bufMruByTab, a:tabId, s:alwaysEmptyBufMru)
  317. return has_key(mru.next, a:bufNbr)
  318. endfunction
  319. " MRUTabShouldShowBuf {{{2
  320. " Return true if `tabId` should show `bufNbr`.
  321. " This is a function of current display modes.
  322. function! s:MRUTabShouldShowBuf(tabId, bufNbr)
  323. if !g:bufExplorerShowTabBuffer
  324. " We are showing buffers from all tabs.
  325. return 1
  326. elseif g:bufExplorerOnlyOneTab
  327. " We are showing buffers that were most recently seen in this tab.
  328. return s:MRUTabForBuf(a:bufNbr) == a:tabId
  329. else
  330. " We are showing buffers that have ever been seen in this tab.
  331. return s:MRUTabHasSeenBuf(a:tabId, a:bufNbr)
  332. endif
  333. endfunction
  334. " MRUListedBuffersForTab {{{2
  335. " Return list of up to `maxBuffers` listed buffers in MRU order for the tab.
  336. " `maxBuffers == 0` => unlimited.
  337. function! s:MRUListedBuffersForTab(tabId, maxBuffers)
  338. let bufNbrs = []
  339. let mru = get(s:bufMruByTab, a:tabId, s:alwaysEmptyBufMru)
  340. let [head, next] = [mru.head, mru.next]
  341. let bufNbr = next[head]
  342. while bufNbr != head
  343. if a:maxBuffers > 0 && len(bufNbrs) >= a:maxBuffers
  344. break
  345. endif
  346. if buflisted(bufNbr) && s:MRUTabShouldShowBuf(a:tabId, bufNbr)
  347. call add(bufNbrs, bufNbr)
  348. endif
  349. let bufNbr = next[bufNbr]
  350. endwhile
  351. return bufNbrs
  352. endfunction
  353. " An always-empty MRU for buffers as a default when looking up
  354. " `s:bufMruByTab[tabId]` for an unknown `tabId`.
  355. let s:alwaysEmptyBufMru = s:MRUNew(0)
  356. " MRUOrderForBuf {{{2
  357. " Return the position of `bufNbr` in the current MRU ordering.
  358. " This is a function of the current display mode. When showing buffers from all
  359. " tabs, it's the global MRU order; otherwise, it the MRU order for the tab at
  360. " BufExplorer launch. The latter includes all buffers seen in this tab, which
  361. " is sufficient whether `g:bufExplorerOnlyOneTab` is true or false.
  362. function! s:MRUOrderForBuf(bufNbr)
  363. if !exists('s:mruOrder')
  364. if g:bufExplorerShowTabBuffer
  365. let mru = get(s:bufMruByTab, s:tabIdAtLaunch, s:alwaysEmptyBufMru)
  366. else
  367. let mru = s:bufMru
  368. endif
  369. let s:mruOrder = s:MRUGetOrdering(mru, 0)
  370. endif
  371. return get(s:mruOrder, a:bufNbr, len(s:mruOrder))
  372. endfunction
  373. " MRUEnsureTabId {{{2
  374. function! s:MRUEnsureTabId(tabNbr)
  375. let tabId = s:GetTabId(a:tabNbr)
  376. if tabId == ''
  377. let tabId = s:AssignTabId(a:tabNbr)
  378. for bufNbr in tabpagebuflist(a:tabNbr)
  379. call s:MRUAddBufTab(bufNbr, tabId)
  380. endfor
  381. endif
  382. return tabId
  383. endfunction
  384. " MRUGarbageCollectBufs {{{2
  385. " Requires `s:raw_buffer_listing`.
  386. function! s:MRUGarbageCollectBufs()
  387. for bufNbr in values(s:bufMru.next)
  388. if bufNbr != 0 && !has_key(s:raw_buffer_listing, bufNbr)
  389. call s:MRURemoveBuf(bufNbr)
  390. endif
  391. endfor
  392. endfunction
  393. " MRUGarbageCollectTabs {{{2
  394. function! s:MRUGarbageCollectTabs()
  395. let numTabs = tabpagenr('$')
  396. let liveTabIds = {}
  397. for tabNbr in range(1, numTabs)
  398. let tabId = s:GetTabId(tabNbr)
  399. if tabId != ''
  400. let liveTabIds[tabId] = 1
  401. endif
  402. endfor
  403. for tabId in keys(s:bufMruByTab)
  404. if tabId != s:tabIdHead && !has_key(liveTabIds, tabId)
  405. call s:MRURemoveTab(tabId)
  406. endif
  407. endfor
  408. endfunction
  409. " DoWinEnter {{{2
  410. function! s:DoWinEnter()
  411. let bufNbr = str2nr(expand("<abuf>"))
  412. let tabNbr = tabpagenr()
  413. let tabId = s:GetTabId(tabNbr)
  414. " Ignore `WinEnter` for a newly created tab; this event comes when creating
  415. " a new tab, and the buffer at that moment is one that is about to be
  416. " replaced by the buffer to which we are switching; this latter buffer will
  417. " be handled by the forthcoming `BufEnter` event.
  418. if tabId != ''
  419. call s:MRUAddBufTab(bufNbr, tabId)
  420. endif
  421. endfunction
  422. " DoBufEnter {{{2
  423. function! s:DoBufEnter()
  424. let bufNbr = str2nr(expand("<abuf>"))
  425. let tabNbr = tabpagenr()
  426. let tabId = s:MRUEnsureTabId(tabNbr)
  427. call s:MRUAddBufTab(bufNbr, tabId)
  428. endfunction
  429. " DoBufDelete {{{2
  430. function! s:DoBufDelete()
  431. let bufNbr = str2nr(expand("<abuf>"))
  432. call s:MRURemoveBuf(bufNbr)
  433. endfunction
  434. " DoTabClosed {{{2
  435. function! s:DoTabClosed()
  436. call s:MRUGarbageCollectTabs()
  437. endfunction
  438. " ShouldIgnore {{{2
  439. function! s:ShouldIgnore(buf)
  440. " Ignore temporary buffers with buftype set.
  441. if empty(getbufvar(a:buf, "&buftype")) == 0
  442. return 1
  443. endif
  444. " Ignore buffers with no name.
  445. if empty(bufname(a:buf)) == 1
  446. return 1
  447. endif
  448. " Ignore the BufExplorer buffer.
  449. if fnamemodify(bufname(a:buf), ":t") == s:name
  450. return 1
  451. endif
  452. " Ignore any buffers in the exclude list.
  453. if index(s:MRU_Exclude_List, bufname(a:buf)) >= 0
  454. return 1
  455. endif
  456. " Else return 0 to indicate that the buffer was not ignored.
  457. return 0
  458. endfunction
  459. " Initialize {{{2
  460. function! s:Initialize()
  461. call s:SetLocalSettings()
  462. let s:running = 1
  463. endfunction
  464. " Cleanup {{{2
  465. function! s:Cleanup()
  466. if exists("s:_insertmode")
  467. let &insertmode = s:_insertmode
  468. endif
  469. if exists("s:_showcmd")
  470. let &showcmd = s:_showcmd
  471. endif
  472. if exists("s:_cpo")
  473. let &cpo = s:_cpo
  474. endif
  475. if exists("s:_report")
  476. let &report = s:_report
  477. endif
  478. let s:running = 0
  479. let s:splitMode = ""
  480. let s:didSplit = 0
  481. delmarks!
  482. endfunction
  483. " SetLocalSettings {{{2
  484. function! s:SetLocalSettings()
  485. let s:_insertmode = &insertmode
  486. set noinsertmode
  487. let s:_showcmd = &showcmd
  488. set noshowcmd
  489. let s:_cpo = &cpo
  490. set cpo&vim
  491. let s:_report = &report
  492. let &report = 10000
  493. setlocal nonumber
  494. setlocal foldcolumn=0
  495. setlocal nofoldenable
  496. setlocal cursorline
  497. setlocal nospell
  498. setlocal nobuflisted
  499. setlocal filetype=bufexplorer
  500. endfunction
  501. " BufExplorerHorizontalSplit {{{2
  502. function! BufExplorerHorizontalSplit()
  503. let s:splitMode = "sp"
  504. execute "BufExplorer"
  505. let s:splitMode = ""
  506. endfunction
  507. " BufExplorerVerticalSplit {{{2
  508. function! BufExplorerVerticalSplit()
  509. let s:splitMode = "vsp"
  510. execute "BufExplorer"
  511. let s:splitMode = ""
  512. endfunction
  513. " ToggleBufExplorer {{{2
  514. function! ToggleBufExplorer()
  515. if exists("s:running") && s:running == 1 && bufname(winbufnr(0)) == s:name
  516. call s:Close()
  517. else
  518. call BufExplorer()
  519. endif
  520. endfunction
  521. " BufExplorer {{{2
  522. function! BufExplorer()
  523. let name = s:name
  524. if !has("win32")
  525. " On non-Windows boxes, escape the name so that is shows up correctly.
  526. let name = escape(name, "[]")
  527. endif
  528. " Make sure there is only one explorer open at a time.
  529. if s:running == 1
  530. " Go to the open buffer.
  531. if has("gui")
  532. execute "drop" name
  533. endif
  534. return
  535. endif
  536. " Add zero to ensure the variable is treated as a number.
  537. let s:originBuffer = bufnr("%") + 0
  538. let s:tabIdAtLaunch = s:MRUEnsureTabId(tabpagenr())
  539. " Forget any cached MRU ordering from previous invocations.
  540. unlet! s:mruOrder
  541. let s:raw_buffer_listing = s:GetBufferInfo(0)
  542. call s:MRUGarbageCollectBufs()
  543. call s:MRUGarbageCollectTabs()
  544. " We may have to split the current window.
  545. if s:splitMode != ""
  546. " Save off the original settings.
  547. let [_splitbelow, _splitright] = [&splitbelow, &splitright]
  548. " Set the setting to ours.
  549. let [&splitbelow, &splitright] = [g:bufExplorerSplitBelow, g:bufExplorerSplitRight]
  550. let _size = (s:splitMode == "sp") ? g:bufExplorerSplitHorzSize : g:bufExplorerSplitVertSize
  551. " Split the window either horizontally or vertically.
  552. if _size <= 0
  553. execute 'keepalt ' . s:splitMode
  554. else
  555. execute 'keepalt ' . _size . s:splitMode
  556. endif
  557. " Restore the original settings.
  558. let [&splitbelow, &splitright] = [_splitbelow, _splitright]
  559. " Remember that a split was triggered
  560. let s:didSplit = 1
  561. endif
  562. if !exists("b:displayMode") || b:displayMode != "winmanager"
  563. " Do not use keepalt when opening bufexplorer to allow the buffer that
  564. " we are leaving to become the new alternate buffer
  565. execute "silent keepjumps hide edit".name
  566. endif
  567. " Record BufExplorer's buffer number.
  568. let s:bufExplorerBuffer = bufnr('%')
  569. call s:DisplayBufferList()
  570. " Position the cursor in the newly displayed list on the line representing
  571. " the active buffer. The active buffer is the line with the '%' character
  572. " in it.
  573. execute search("%")
  574. if exists('#User#BufExplorer_Started')
  575. " Notify that BufExplorer has started. This is an opportunity to make
  576. " custom buffer-local mappings and the like.
  577. doautocmd User BufExplorer_Started
  578. endif
  579. endfunction
  580. " Tracks `tabId` at BufExplorer launch.
  581. let s:tabIdAtLaunch = ''
  582. " DisplayBufferList {{{2
  583. function! s:DisplayBufferList()
  584. setlocal buftype=nofile
  585. setlocal modifiable
  586. setlocal noreadonly
  587. setlocal noswapfile
  588. setlocal nowrap
  589. setlocal bufhidden=wipe
  590. call s:MapKeys()
  591. " Wipe out any existing lines in case BufExplorer buffer exists and the
  592. " user had changed any global settings that might reduce the number of
  593. " lines needed in the buffer.
  594. silent keepjumps 1,$d _
  595. call setline(1, s:CreateHelp())
  596. call s:BuildBufferList()
  597. call cursor(s:firstBufferLine, 1)
  598. if !g:bufExplorerResize
  599. normal! zz
  600. endif
  601. setlocal nomodifiable
  602. endfunction
  603. " RedisplayBufferList {{{2
  604. function! s:RedisplayBufferList()
  605. call s:RebuildBufferList()
  606. call s:UpdateHelpStatus()
  607. endfunction
  608. " MapKeys {{{2
  609. function! s:MapKeys()
  610. nnoremap <silent> <buffer> <Plug>(BufExplorer_BufferDelete) :call <SID>RemoveBuffer("delete")<CR>
  611. nnoremap <silent> <buffer> <Plug>(BufExplorer_BufferDeleteForced) :call <SID>RemoveBuffer("force_delete")<CR>
  612. nnoremap <silent> <buffer> <Plug>(BufExplorer_BufferWipe) :call <SID>RemoveBuffer("wipe")<CR>
  613. nnoremap <silent> <buffer> <Plug>(BufExplorer_BufferWipeForced) :call <SID>RemoveBuffer("force_wipe")<CR>
  614. nnoremap <silent> <buffer> <Plug>(BufExplorer_Close) :call <SID>Close()<CR>
  615. nnoremap <silent> <buffer> <Plug>(BufExplorer_OpenBuffer) :call <SID>SelectBuffer()<CR>
  616. nnoremap <silent> <buffer> <Plug>(BufExplorer_OpenBufferAsk) :call <SID>SelectBuffer("ask")<CR>
  617. nnoremap <silent> <buffer> <Plug>(BufExplorer_OpenBufferSplitAbove) :call <SID>SelectBuffer("split", "st")<CR>
  618. nnoremap <silent> <buffer> <Plug>(BufExplorer_OpenBufferSplitBelow) :call <SID>SelectBuffer("split", "sb")<CR>
  619. nnoremap <silent> <buffer> <Plug>(BufExplorer_OpenBufferSplitLeft) :call <SID>SelectBuffer("split", "vl")<CR>
  620. nnoremap <silent> <buffer> <Plug>(BufExplorer_OpenBufferSplitRight) :call <SID>SelectBuffer("split", "vr")<CR>
  621. nnoremap <silent> <buffer> <Plug>(BufExplorer_OpenBufferTab) :call <SID>SelectBuffer("tab")<CR>
  622. nnoremap <silent> <buffer> <Plug>(BufExplorer_SortByNext) :call <SID>SortSelect()<CR>
  623. nnoremap <silent> <buffer> <Plug>(BufExplorer_SortByPrev) :call <SID>ReverseSortSelect()<CR>
  624. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleFindActive) :call <SID>ToggleFindActive()<CR>
  625. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleHelp) :call <SID>ToggleHelp()<CR>
  626. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleOnlyOneTab) :call <SID>ToggleOnlyOneTab()<CR>
  627. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleReverseSort) :call <SID>SortReverse()<CR>
  628. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleShowRelativePath) :call <SID>ToggleShowRelativePath()<CR>
  629. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleShowTabBuffer) :call <SID>ToggleShowTabBuffer()<CR>
  630. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleShowTerminal) :call <SID>ToggleShowTerminal()<CR>
  631. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleShowUnlisted) :call <SID>ToggleShowUnlisted()<CR>
  632. nnoremap <silent> <buffer> <Plug>(BufExplorer_ToggleSplitOutPathName) :call <SID>ToggleSplitOutPathName()<CR>
  633. if exists("b:displayMode") && b:displayMode == "winmanager"
  634. nnoremap <buffer> <silent> <tab> :call <SID>SelectBuffer()<CR>
  635. endif
  636. nmap <nowait> <buffer> <2-leftmouse> <Plug>(BufExplorer_OpenBuffer)
  637. nmap <nowait> <buffer> <CR> <Plug>(BufExplorer_OpenBuffer)
  638. nmap <nowait> <buffer> <F1> <Plug>(BufExplorer_ToggleHelp)
  639. nmap <nowait> <buffer> <s-cr> <Plug>(BufExplorer_OpenBufferTab)
  640. nmap <nowait> <buffer> a <Plug>(BufExplorer_ToggleFindActive)
  641. nmap <nowait> <buffer> b <Plug>(BufExplorer_OpenBufferAsk)
  642. nmap <nowait> <buffer> B <Plug>(BufExplorer_ToggleOnlyOneTab)
  643. nmap <nowait> <buffer> d <Plug>(BufExplorer_BufferDelete)
  644. nmap <nowait> <buffer> D <Plug>(BufExplorer_BufferWipe)
  645. nmap <nowait> <buffer> f <Plug>(BufExplorer_OpenBufferSplitBelow)
  646. nmap <nowait> <buffer> F <Plug>(BufExplorer_OpenBufferSplitAbove)
  647. nmap <nowait> <buffer> o <Plug>(BufExplorer_OpenBuffer)
  648. nmap <nowait> <buffer> p <Plug>(BufExplorer_ToggleSplitOutPathName)
  649. nmap <nowait> <buffer> q <Plug>(BufExplorer_Close)
  650. nmap <nowait> <buffer> r <Plug>(BufExplorer_ToggleReverseSort)
  651. nmap <nowait> <buffer> R <Plug>(BufExplorer_ToggleShowRelativePath)
  652. nmap <nowait> <buffer> s <Plug>(BufExplorer_SortByNext)
  653. nmap <nowait> <buffer> S <Plug>(BufExplorer_SortByPrev)
  654. nmap <nowait> <buffer> t <Plug>(BufExplorer_OpenBufferTab)
  655. nmap <nowait> <buffer> T <Plug>(BufExplorer_ToggleShowTabBuffer)
  656. nmap <nowait> <buffer> u <Plug>(BufExplorer_ToggleShowUnlisted)
  657. nmap <nowait> <buffer> v <Plug>(BufExplorer_OpenBufferSplitRight)
  658. nmap <nowait> <buffer> V <Plug>(BufExplorer_OpenBufferSplitLeft)
  659. nmap <nowait> <buffer> X <Plug>(BufExplorer_ToggleShowTerminal)
  660. for k in ["G", "n", "N", "L", "M", "H"]
  661. execute "nnoremap <buffer> <silent>" k ":keepjumps normal!" k."<CR>"
  662. endfor
  663. endfunction
  664. " ToggleHelp {{{2
  665. function! s:ToggleHelp()
  666. let g:bufExplorerDetailedHelp = !g:bufExplorerDetailedHelp
  667. setlocal modifiable
  668. " Save position.
  669. normal! ma
  670. " Remove old header.
  671. if s:firstBufferLine > 1
  672. execute "keepjumps 1,".(s:firstBufferLine - 1) "d _"
  673. endif
  674. call append(0, s:CreateHelp())
  675. silent! normal! g`a
  676. delmarks a
  677. setlocal nomodifiable
  678. if exists("b:displayMode") && b:displayMode == "winmanager"
  679. call WinManagerForceReSize("BufExplorer")
  680. endif
  681. endfunction
  682. " GetHelpStatus {{{2
  683. function! s:GetHelpStatus()
  684. let ret = '" Sorted by '.((g:bufExplorerReverseSort == 1) ? "reverse " : "").g:bufExplorerSortBy
  685. let ret .= ' | '.((g:bufExplorerFindActive == 0) ? "Don't " : "")."Locate buffer"
  686. let ret .= ((g:bufExplorerShowUnlisted == 0) ? "" : " | Show unlisted")
  687. let ret .= ((g:bufExplorerShowTabBuffer == 0) ? "" : " | Show buffers/tab")
  688. let ret .= ((g:bufExplorerOnlyOneTab == 0) ? "" : " | One tab/buffer")
  689. let ret .= ' | '.((g:bufExplorerShowRelativePath == 0) ? "Absolute" : "Relative")
  690. let ret .= ' '.((g:bufExplorerSplitOutPathName == 0) ? "Full" : "Split")." path"
  691. let ret .= ((g:bufExplorerShowTerminal == 0) ? "" : " | Show terminal")
  692. return ret
  693. endfunction
  694. " CreateHelp {{{2
  695. function! s:CreateHelp()
  696. if g:bufExplorerDefaultHelp == 0 && g:bufExplorerDetailedHelp == 0
  697. let s:firstBufferLine = 1
  698. return []
  699. endif
  700. let header = []
  701. if g:bufExplorerDetailedHelp == 1
  702. call add(header, '" Buffer Explorer ('.g:bufexplorer_version.')')
  703. call add(header, '" --------------------------')
  704. call add(header, '" <F1> : toggle this help')
  705. call add(header, '" <enter> or o or Mouse-Double-Click : open buffer under cursor')
  706. call add(header, '" <shift-enter> or t : open buffer in another tab')
  707. call add(header, '" a : toggle find active buffer')
  708. call add(header, '" b : Fast buffer switching with b<any bufnum>')
  709. call add(header, '" B : toggle showing buffers only on their MRU tabs')
  710. call add(header, '" d : delete buffer')
  711. call add(header, '" D : wipe buffer')
  712. call add(header, '" F : open buffer in another window above the current')
  713. call add(header, '" f : open buffer in another window below the current')
  714. call add(header, '" p : toggle splitting of file and path name')
  715. call add(header, '" q : quit')
  716. call add(header, '" r : reverse sort')
  717. call add(header, '" R : toggle showing relative or full paths')
  718. call add(header, '" s : cycle thru "sort by" fields '.string(s:sort_by).'')
  719. call add(header, '" S : reverse cycle thru "sort by" fields')
  720. call add(header, '" T : toggle showing all buffers/only buffers used on this tab')
  721. call add(header, '" u : toggle showing unlisted buffers')
  722. call add(header, '" V : open buffer in another window on the left of the current')
  723. call add(header, '" v : open buffer in another window on the right of the current')
  724. call add(header, '" X : toggle showing terminal buffers')
  725. else
  726. call add(header, '" Press <F1> for Help')
  727. endif
  728. if (!exists("b:displayMode") || b:displayMode != "winmanager") || (b:displayMode == "winmanager" && g:bufExplorerDetailedHelp == 1)
  729. call add(header, s:GetHelpStatus())
  730. call add(header, '"=')
  731. endif
  732. let s:firstBufferLine = len(header) + 1
  733. return header
  734. endfunction
  735. " CalculateBufferDetails {{{2
  736. " Calculate `buf`-related details.
  737. " Only these fields of `buf` must be defined on entry:
  738. " - `._bufnr`
  739. " - `.attributes`
  740. " - `.line`
  741. function! s:CalculateBufferDetails(buf)
  742. let buf = a:buf
  743. let name = bufname(buf._bufnr)
  744. let buf["hasNoName"] = empty(name)
  745. if buf.hasNoName
  746. let name = "[No Name]"
  747. endif
  748. let buf.isterminal = getbufvar(buf._bufnr, '&buftype') == 'terminal'
  749. if buf.isterminal
  750. " Neovim uses paths with `term://` prefix, where the provided path
  751. " is the current working directory when the terminal was launched, e.g.:
  752. " - Unix:
  753. " term://~/tmp/sort//1464953:/bin/bash
  754. " - Windows:
  755. " term://C:\apps\nvim-win64\bin//6408:C:\Windows\system32\cmd.exe
  756. " Vim uses paths starting with `!`, with no provided path, e.g.:
  757. " - Unix:
  758. " !/bin/bash
  759. " - Windows:
  760. " !C:\Windows\system32\cmd.exe
  761. " Use the terminal's current working directory as the `path`.
  762. " For `shortname`, use `!PID:shellName`, prefixed with `!` as Vim does,
  763. " and without the shell's path for brevity, e.g.:
  764. " `/bin/bash` -> `!bash`
  765. " `1464953:/bin/bash` -> `!1464953:bash`
  766. " `C:\Windows\system32\cmd.exe` -> `!cmd.exe`
  767. " `6408:C:\Windows\system32\cmd.exe` -> `!6408:cmd.exe`
  768. " Neovim-style name format:
  769. " term://(cwd)//(pid):(shellPath)
  770. " e.g.:
  771. " term://~/tmp/sort//1464953:/bin/bash
  772. " `cwd` is the directory at terminal launch.
  773. let termNameParts = matchlist(name, '\v\c^term://(.*)//(\d+):(.*)$')
  774. if len(termNameParts) > 0
  775. let [cwd, pidStr, shellPath] = termNameParts[1:3]
  776. let pid = str2nr(pidStr)
  777. let shellName = fnamemodify(shellPath, ':t')
  778. else
  779. " Default to Vim's current working directory.
  780. let cwd = '.'
  781. let shellName = fnamemodify(name, ':t')
  782. let pid = -1
  783. if exists('*term_getjob') && exists('*job_info')
  784. let job = term_getjob(buf._bufnr)
  785. if job != v:null
  786. let pid = job_info(job).process
  787. endif
  788. endif
  789. endif
  790. if pid < 0
  791. let shortname = '!' . shellName
  792. else
  793. let shortname = '!' . pid . ':' . shellName
  794. " On some systems having a `/proc` filesystem (e.g., Linux, *BSD,
  795. " Solaris), each process has a `cwd` symlink for the current working
  796. " directory. `resolve()` will return the actual current working
  797. " directory if possible; otherwise, it will return the symlink path
  798. " unchanged.
  799. let cwd_symlink = '/proc/' . pid . '/cwd'
  800. let resolved_cwd = resolve(cwd_symlink)
  801. if resolved_cwd != cwd_symlink
  802. let cwd = resolved_cwd
  803. endif
  804. endif
  805. let slashed_path = fnamemodify(cwd, ':p')
  806. let buf.fullname = slashed_path . shortname
  807. let buf.shortname = shortname
  808. let homepath = fnamemodify(slashed_path, ':~:h')
  809. let buf.path = homepath
  810. let buf.homename = fnamemodify(buf.fullname, ':~')
  811. let buf.relativepath = fnamemodify(slashed_path, ':~:.:h')
  812. let buf.relativename = fnamemodify(buf.fullname, ':~:.')
  813. return
  814. endif
  815. let buf.fullname = simplify(fnamemodify(name, ':p'))
  816. let buf.isdir = getftype(buf.fullname) == "dir"
  817. if buf.isdir
  818. " `buf.fullname` ends with a path separator; this will be
  819. " removed via the first `:h` applied to `buf.fullname` (except
  820. " for the root directory, where the path separator will remain).
  821. let parent = fnamemodify(buf.fullname, ':h:h')
  822. let buf.shortname = fnamemodify(buf.fullname, ':h:t')
  823. " Special case for root directory: fnamemodify('/', ':h:t') == ''
  824. if buf.shortname == ''
  825. let buf.shortname = '.'
  826. endif
  827. " Must perform shortening (`:~`, `:.`) before `:h`.
  828. let buf.homename = fnamemodify(buf.fullname, ':~:h')
  829. let buf.relativename = fnamemodify(buf.fullname, ':~:.:h')
  830. else
  831. let parent = fnamemodify(buf.fullname, ':h')
  832. let buf.shortname = fnamemodify(buf.fullname, ':t')
  833. let buf.homename = fnamemodify(buf.fullname, ':~')
  834. let buf.relativename = fnamemodify(buf.fullname, ':~:.')
  835. endif
  836. " `:p` on `parent` adds back the path separator which permits more
  837. " effective shortening (`:~`, `:.`), but `:h` is required afterward
  838. " to trim this separator.
  839. let buf.path = fnamemodify(parent, ':p:~:h')
  840. let buf.relativepath = fnamemodify(parent, ':p:~:.:h')
  841. endfunction
  842. " GetBufferInfo {{{2
  843. " Return dictionary `{ bufNbr : buf }`.
  844. " - If `onlyBufNbr > 0`, dictionary will contain at most that buffer.
  845. " On return, only these fields are set for each `buf`:
  846. " - `._bufnr`
  847. " - `.attributes`
  848. " - `.line`
  849. " Other fields will be populated by `s:CalculateBufferDetails()`.
  850. function! s:GetBufferInfo(onlyBufNbr)
  851. redir => bufoutput
  852. " Below, `:silent buffers` allows capturing the output via `:redir` but
  853. " prevents display to the user.
  854. if a:onlyBufNbr > 0 && buflisted(a:onlyBufNbr)
  855. " We care only about the listed buffer `a:onlyBufNbr`, so no need to
  856. " enumerate unlisted buffers.
  857. silent buffers
  858. else
  859. " Use `!` to show all buffers including the unlisted ones.
  860. silent buffers!
  861. endif
  862. redir END
  863. if a:onlyBufNbr > 0
  864. " Since we are only interested in this specified buffer remove the
  865. " other buffers listed.
  866. " Use a very-magic pattern starting with a newline and a run of zero or
  867. " more spaces/tabs:
  868. let onlyLinePattern = '\v\n\s*'
  869. " Continue with the buffer number followed by a non-digit character
  870. " (which will be a buffer attribute character such as `u` or ` `).
  871. let onlyLinePattern .= a:onlyBufNbr . '\D'
  872. " Finish with a run of zero or more non-newline characters plus newline:
  873. let onlyLinePattern .= '[^\n]*\n'
  874. let bufoutput = matchstr("\n" . bufoutput . "\n", onlyLinePattern)
  875. endif
  876. let all = {}
  877. " Loop over each line in the buffer.
  878. for line in split(bufoutput, '\n')
  879. let bits = split(line, '"')
  880. " Use first and last components after the split on '"', in case a
  881. " filename with an embedded '"' is present.
  882. let buf = {"attributes": bits[0], "line": substitute(bits[-1], '\s*', '', '')}
  883. let buf._bufnr = str2nr(buf.attributes)
  884. let all[buf._bufnr] = buf
  885. endfor
  886. return all
  887. endfunction
  888. " BuildBufferList {{{2
  889. function! s:BuildBufferList()
  890. let table = []
  891. " Loop through every buffer.
  892. for buf in values(s:raw_buffer_listing)
  893. " `buf.attributes` must exist, but we defer the expensive work of
  894. " calculating other buffer details (e.g., `buf.fullname`) until we know
  895. " the user wants to view this buffer.
  896. " Skip BufExplorer's buffer.
  897. if buf._bufnr == s:bufExplorerBuffer
  898. continue
  899. endif
  900. " Skip unlisted buffers if we are not to show them.
  901. if !g:bufExplorerShowUnlisted && buf.attributes =~ "u"
  902. " Skip unlisted buffers if we are not to show them.
  903. continue
  904. endif
  905. " Ensure buffer details are computed for this buffer.
  906. if !has_key(buf, 'fullname')
  907. call s:CalculateBufferDetails(buf)
  908. endif
  909. " Skip 'No Name' buffers if we are not to show them.
  910. if g:bufExplorerShowNoName == 0 && buf.hasNoName
  911. continue
  912. endif
  913. " Should we show this buffer in this tab?
  914. if !s:MRUTabShouldShowBuf(s:tabIdAtLaunch, buf._bufnr)
  915. continue
  916. endif
  917. " Skip terminal buffers if we are not to show them.
  918. if !g:bufExplorerShowTerminal && buf.isterminal
  919. continue
  920. endif
  921. " Skip directory buffers if we are not to show them.
  922. if !g:bufExplorerShowDirectories && buf.isdir
  923. continue
  924. endif
  925. let row = [buf.attributes]
  926. if exists("g:loaded_webdevicons")
  927. let row += [WebDevIconsGetFileTypeSymbol(buf.fullname, buf.isdir)]
  928. endif
  929. " Are we to split the path and file name?
  930. if g:bufExplorerSplitOutPathName
  931. let type = (g:bufExplorerShowRelativePath) ? "relativepath" : "path"
  932. let row += [buf.shortname, buf[type]]
  933. else
  934. let type = (g:bufExplorerShowRelativePath) ? "relativename" : "homename"
  935. let row += [buf[type]]
  936. endif
  937. let row += [buf.line]
  938. call add(table, row)
  939. endfor
  940. let lines = s:MakeLines(table)
  941. call setline(s:firstBufferLine, lines)
  942. let firstMissingLine = s:firstBufferLine + len(lines)
  943. if line('$') >= firstMissingLine
  944. " Clear excess lines starting with `firstMissingLine`.
  945. execute "silent keepjumps ".firstMissingLine.',$d _'
  946. endif
  947. call s:SortListing()
  948. endfunction
  949. " MakeLines {{{2
  950. function! s:MakeLines(table)
  951. if len(a:table) == 0
  952. return []
  953. endif
  954. let lines = []
  955. " To avoid trailing whitespace, do not pad the final column.
  956. let numColumnsToPad = len(a:table[0]) - 1
  957. let maxWidths = repeat([0], numColumnsToPad)
  958. for row in a:table
  959. let i = 0
  960. while i < numColumnsToPad
  961. let maxWidths[i] = max([maxWidths[i], s:StringWidth(row[i])])
  962. let i = i + 1
  963. endwhile
  964. endfor
  965. let pads = []
  966. for w in maxWidths
  967. call add(pads, repeat(' ', w))
  968. endfor
  969. for row in a:table
  970. let i = 0
  971. while i < numColumnsToPad
  972. let row[i] .= strpart(pads[i], s:StringWidth(row[i]))
  973. let i = i + 1
  974. endwhile
  975. call add(lines, join(row, ' '))
  976. endfor
  977. return lines
  978. endfunction
  979. " SelectBuffer {{{2
  980. function! s:SelectBuffer(...)
  981. " Sometimes messages are not cleared when we get here so it looks like an
  982. " error has occurred when it really has not.
  983. "echo ""
  984. let _bufNbr = -1
  985. if (a:0 == 1) && (a:1 == "ask")
  986. " Ask the user for input.
  987. call inputsave()
  988. let cmd = input("Enter buffer number to switch to: ")
  989. call inputrestore()
  990. " Clear the message area from the previous prompt.
  991. redraw | echo
  992. if strlen(cmd) > 0
  993. let _bufNbr = str2nr(cmd)
  994. else
  995. call s:Error("Invalid buffer number, try again.")
  996. return
  997. endif
  998. else
  999. " Are we on a line with a file name?
  1000. if line('.') < s:firstBufferLine
  1001. execute "normal! \<CR>"
  1002. return
  1003. endif
  1004. let _bufNbr = str2nr(getline('.'))
  1005. " Check and see if we are running BufferExplorer via WinManager.
  1006. if exists("b:displayMode") && b:displayMode == "winmanager"
  1007. let _bufName = expand("#"._bufNbr.":p")
  1008. if (a:0 == 1) && (a:1 == "tab")
  1009. call WinManagerFileEdit(_bufName, 1)
  1010. else
  1011. call WinManagerFileEdit(_bufName, 0)
  1012. endif
  1013. return
  1014. endif
  1015. endif
  1016. if bufexists(_bufNbr)
  1017. " Get the tab number where this buffer is located in.
  1018. let tabNbr = s:GetTabNbr(_bufNbr)
  1019. if exists("g:bufExplorerChgWin") && g:bufExplorerChgWin <=winnr("$")
  1020. execute g:bufExplorerChgWin."wincmd w"
  1021. execute "keepjumps keepalt silent b!" _bufNbr
  1022. " Are we supposed to open the selected buffer in a tab?
  1023. elseif (a:0 == 1) && (a:1 == "tab")
  1024. call s:Close()
  1025. " Open a new tab with the selected buffer in it.
  1026. if v:version > 704 || ( v:version == 704 && has('patch2237') )
  1027. " new syntax for last tab as of 7.4.2237
  1028. execute "$tab split +buffer" . _bufNbr
  1029. else
  1030. execute "999tab split +buffer" . _bufNbr
  1031. endif
  1032. " Are we supposed to open the selected buffer in a split?
  1033. elseif (a:0 == 2) && (a:1 == "split")
  1034. call s:Close()
  1035. if (a:2 == "vl")
  1036. execute "vert topleft sb "._bufNbr
  1037. elseif (a:2 == "vr")
  1038. execute "vert belowright sb "._bufNbr
  1039. elseif (a:2 == "st")
  1040. execute "topleft sb "._bufNbr
  1041. else " = sb
  1042. execute "belowright sb "._bufNbr
  1043. endif
  1044. else
  1045. " Request to open in current (BufExplorer) window.
  1046. if g:bufExplorerFindActive && tabNbr > 0
  1047. " Close BufExplorer window and switch to existing tab/window.
  1048. call s:Close()
  1049. execute tabNbr . "tabnext"
  1050. execute bufwinnr(_bufNbr) . "wincmd w"
  1051. else
  1052. " Use BufExplorer window for the buffer.
  1053. execute "keepjumps keepalt silent b!" _bufNbr
  1054. endif
  1055. endif
  1056. " Make the buffer 'listed' again.
  1057. call setbufvar(_bufNbr, "&buflisted", "1")
  1058. " Call any associated function references. g:bufExplorerFuncRef may be
  1059. " an individual function reference or it may be a list containing
  1060. " function references. It will ignore anything that's not a function
  1061. " reference.
  1062. "
  1063. " See :help FuncRef for more on function references.
  1064. if exists("g:BufExplorerFuncRef")
  1065. if type(g:BufExplorerFuncRef) == 2
  1066. keepj call g:BufExplorerFuncRef()
  1067. elseif type(g:BufExplorerFuncRef) == 3
  1068. for FncRef in g:BufExplorerFuncRef
  1069. if type(FncRef) == 2
  1070. keepj call FncRef()
  1071. endif
  1072. endfor
  1073. endif
  1074. endif
  1075. else
  1076. call s:Error("Sorry, that buffer no longer exists, please select another")
  1077. call s:DeleteBuffer(_bufNbr, "wipe")
  1078. endif
  1079. endfunction
  1080. " RemoveBuffer {{{2
  1081. " Valid `mode` values:
  1082. " - "delete"
  1083. " - "force_delete"
  1084. " - "wipe"
  1085. " - "force_wipe"
  1086. function! s:RemoveBuffer(mode)
  1087. " Are we on a line with a file name?
  1088. if line('.') < s:firstBufferLine
  1089. return
  1090. endif
  1091. let mode = a:mode
  1092. let forced = mode =~# '^force_'
  1093. " These commands are to temporarily suspend the activity of winmanager.
  1094. if exists("b:displayMode") && b:displayMode == "winmanager"
  1095. call WinManagerSuspendAUs()
  1096. end
  1097. let bufNbr = str2nr(getline('.'))
  1098. let buf = s:raw_buffer_listing[bufNbr]
  1099. if !forced && (buf.isterminal || getbufvar(bufNbr, '&modified'))
  1100. if buf.isterminal
  1101. let msg = "Buffer " . bufNbr . " is a terminal"
  1102. else
  1103. let msg = "No write since last change for buffer " . bufNbr
  1104. endif
  1105. " Calling confirm() requires Vim built with dialog option.
  1106. if !has("dialog_con") && !has("dialog_gui")
  1107. call s:Error(msg . "; cannot remove without 'force'")
  1108. return
  1109. endif
  1110. let answer = confirm(msg . "; Remove anyway?", "&Yes\n&No", 2)
  1111. if answer == 1
  1112. let mode = 'force_' . mode
  1113. else
  1114. return
  1115. endif
  1116. endif
  1117. " Okay, everything is good, delete or wipe the buffer.
  1118. call s:DeleteBuffer(bufNbr, mode)
  1119. " Reactivate winmanager autocommand activity.
  1120. if exists("b:displayMode") && b:displayMode == "winmanager"
  1121. call WinManagerForceReSize("BufExplorer")
  1122. call WinManagerResumeAUs()
  1123. end
  1124. endfunction
  1125. " DeleteBuffer {{{2
  1126. " Valid `mode` values:
  1127. " - "delete"
  1128. " - "force_delete"
  1129. " - "wipe"
  1130. " - "force_wipe"
  1131. function! s:DeleteBuffer(bufNbr, mode)
  1132. " This routine assumes that the buffer to be removed is on the current line.
  1133. if a:mode =~# 'delete$' && bufexists(a:bufNbr) && !buflisted(a:bufNbr)
  1134. call s:Error('Buffer ' . a:bufNbr
  1135. \ . ' is unlisted; must `wipe` to remove')
  1136. return
  1137. endif
  1138. try
  1139. " Wipe/Delete buffer from Vim.
  1140. if a:mode == "wipe"
  1141. execute "silent bwipe" a:bufNbr
  1142. elseif a:mode == "force_wipe"
  1143. execute "silent bwipe!" a:bufNbr
  1144. elseif a:mode == "force_delete"
  1145. execute "silent bdelete!" a:bufNbr
  1146. else
  1147. execute "silent bdelete" a:bufNbr
  1148. endif
  1149. catch
  1150. call s:Error(v:exception)
  1151. endtry
  1152. if bufexists(a:bufNbr)
  1153. " Buffer is still present. We may have failed to wipe it, or it may
  1154. " have changed attributes (as `:bd` only makes a buffer unlisted).
  1155. " Regather information on this buffer, update the buffer list, and
  1156. " redisplay.
  1157. let info = s:GetBufferInfo(a:bufNbr)
  1158. let s:raw_buffer_listing[a:bufNbr] = info[a:bufNbr]
  1159. call s:RedisplayBufferList()
  1160. else
  1161. " Delete the buffer from the list on screen.
  1162. setlocal modifiable
  1163. normal! "_dd
  1164. setlocal nomodifiable
  1165. " Delete the buffer from the raw buffer list.
  1166. unlet s:raw_buffer_listing[a:bufNbr]
  1167. endif
  1168. endfunction
  1169. " Close {{{2
  1170. function! s:Close()
  1171. " Get only the listed buffers associated with the current tab (up to 2).
  1172. let listed = s:MRUListedBuffersForTab(s:tabIdAtLaunch, 2)
  1173. " If we needed to split the main window, close the split one.
  1174. if s:didSplit
  1175. execute "wincmd c"
  1176. endif
  1177. " Check to see if there are anymore buffers listed.
  1178. if len(listed) == 0
  1179. " Since there are no buffers left to switch to, open a new empty
  1180. " buffers.
  1181. execute "enew"
  1182. else
  1183. " Since there are buffers left to switch to, switch to the previous and
  1184. " then the current.
  1185. for b in reverse(listed[0:1])
  1186. execute "keepjumps silent b ".b
  1187. endfor
  1188. endif
  1189. " Clear any messages.
  1190. echo
  1191. endfunction
  1192. " ToggleShowTerminal {{{2
  1193. function! s:ToggleShowTerminal()
  1194. let g:bufExplorerShowTerminal = !g:bufExplorerShowTerminal
  1195. call s:RedisplayBufferList()
  1196. endfunction
  1197. " ToggleSplitOutPathName {{{2
  1198. function! s:ToggleSplitOutPathName()
  1199. let g:bufExplorerSplitOutPathName = !g:bufExplorerSplitOutPathName
  1200. call s:RedisplayBufferList()
  1201. endfunction
  1202. " ToggleShowRelativePath {{{2
  1203. function! s:ToggleShowRelativePath()
  1204. let g:bufExplorerShowRelativePath = !g:bufExplorerShowRelativePath
  1205. call s:RedisplayBufferList()
  1206. endfunction
  1207. " ToggleShowTabBuffer {{{2
  1208. function! s:ToggleShowTabBuffer()
  1209. " Forget any cached MRU ordering, as it depends on
  1210. " `g:bufExplorerShowTabBuffer`.
  1211. unlet! s:mruOrder
  1212. let g:bufExplorerShowTabBuffer = !g:bufExplorerShowTabBuffer
  1213. call s:RedisplayBufferList()
  1214. endfunction
  1215. " ToggleOnlyOneTab {{{2
  1216. function! s:ToggleOnlyOneTab()
  1217. let g:bufExplorerOnlyOneTab = !g:bufExplorerOnlyOneTab
  1218. call s:RedisplayBufferList()
  1219. endfunction
  1220. " ToggleShowUnlisted {{{2
  1221. function! s:ToggleShowUnlisted()
  1222. let g:bufExplorerShowUnlisted = !g:bufExplorerShowUnlisted
  1223. call s:RedisplayBufferList()
  1224. endfunction
  1225. " ToggleFindActive {{{2
  1226. function! s:ToggleFindActive()
  1227. let g:bufExplorerFindActive = !g:bufExplorerFindActive
  1228. call s:UpdateHelpStatus()
  1229. endfunction
  1230. " RebuildBufferList {{{2
  1231. function! s:RebuildBufferList()
  1232. setlocal modifiable
  1233. let curPos = getpos('.')
  1234. let num_bufs = s:BuildBufferList()
  1235. call setpos('.', curPos)
  1236. setlocal nomodifiable
  1237. return num_bufs
  1238. endfunction
  1239. " UpdateHelpStatus {{{2
  1240. function! s:UpdateHelpStatus()
  1241. setlocal modifiable
  1242. let text = s:GetHelpStatus()
  1243. call setline(s:firstBufferLine - 2, text)
  1244. setlocal nomodifiable
  1245. endfunction
  1246. " Key_number {{{2
  1247. function! s:Key_number(line)
  1248. let _bufnr = str2nr(a:line)
  1249. let key = [printf('%9d', _bufnr)]
  1250. return key
  1251. endfunction
  1252. " Key_name {{{2
  1253. function! s:Key_name(line)
  1254. let _bufnr = str2nr(a:line)
  1255. let buf = s:raw_buffer_listing[_bufnr]
  1256. let key = [buf.shortname, buf.fullname]
  1257. return key
  1258. endfunction
  1259. " Key_fullpath {{{2
  1260. function! s:Key_fullpath(line)
  1261. let _bufnr = str2nr(a:line)
  1262. let buf = s:raw_buffer_listing[_bufnr]
  1263. let key = [buf.fullname]
  1264. return key
  1265. endfunction
  1266. " Key_extension {{{2
  1267. function! s:Key_extension(line)
  1268. let _bufnr = str2nr(a:line)
  1269. let buf = s:raw_buffer_listing[_bufnr]
  1270. let extension = fnamemodify(buf.shortname, ':e')
  1271. let key = [extension, buf.shortname, buf.fullname]
  1272. return key
  1273. endfunction
  1274. " Key_mru {{{2
  1275. function! s:Key_mru(line)
  1276. let _bufnr = str2nr(a:line)
  1277. let buf = s:raw_buffer_listing[_bufnr]
  1278. let pos = s:MRUOrderForBuf(_bufnr)
  1279. return [printf('%9d', pos), buf.fullname]
  1280. endfunction
  1281. " SortByKeyFunc {{{2
  1282. function! s:SortByKeyFunc(keyFunc)
  1283. let keyedLines = []
  1284. for line in getline(s:firstBufferLine, "$")
  1285. let key = eval(a:keyFunc . '(line)')
  1286. call add(keyedLines, join(key + [line], "\1"))
  1287. endfor
  1288. " Ignore case when sorting by passing `1`:
  1289. call sort(keyedLines, 1)
  1290. if g:bufExplorerReverseSort
  1291. call reverse(keyedLines)
  1292. endif
  1293. let lines = []
  1294. for keyedLine in keyedLines
  1295. call add(lines, split(keyedLine, "\1")[-1])
  1296. endfor
  1297. call setline(s:firstBufferLine, lines)
  1298. endfunction
  1299. " SortReverse {{{2
  1300. function! s:SortReverse()
  1301. let g:bufExplorerReverseSort = !g:bufExplorerReverseSort
  1302. call s:ReSortListing()
  1303. endfunction
  1304. " SortSelect {{{2
  1305. function! s:SortSelect()
  1306. let g:bufExplorerSortBy = get(s:sort_by, index(s:sort_by, g:bufExplorerSortBy) + 1, s:sort_by[0])
  1307. call s:ReSortListing()
  1308. endfunction
  1309. " ReverseSortSelect {{{2
  1310. function! s:ReverseSortSelect()
  1311. let g:bufExplorerSortBy = get(s:sort_by, index(s:sort_by, g:bufExplorerSortBy) - 1, s:sort_by[-1])
  1312. call s:ReSortListing()
  1313. endfunction
  1314. " ReSortListing {{{2
  1315. function! s:ReSortListing()
  1316. setlocal modifiable
  1317. let curPos = getpos('.')
  1318. call s:SortListing()
  1319. call s:UpdateHelpStatus()
  1320. call setpos('.', curPos)
  1321. setlocal nomodifiable
  1322. endfunction
  1323. " SortListing {{{2
  1324. function! s:SortListing()
  1325. call s:SortByKeyFunc("<SID>Key_" . g:bufExplorerSortBy)
  1326. endfunction
  1327. " Error {{{2
  1328. " Display a message using ErrorMsg highlight group.
  1329. function! s:Error(msg)
  1330. echohl ErrorMsg
  1331. echomsg a:msg
  1332. echohl None
  1333. endfunction
  1334. " Warning {{{2
  1335. " Display a message using WarningMsg highlight group.
  1336. function! s:Warning(msg)
  1337. echohl WarningMsg
  1338. echomsg a:msg
  1339. echohl None
  1340. endfunction
  1341. " GetTabNbr {{{2
  1342. function! s:GetTabNbr(bufNbr)
  1343. " Prefer current tab.
  1344. if bufwinnr(a:bufNbr) > 0
  1345. return tabpagenr()
  1346. endif
  1347. " Searching buffer bufno, in tabs.
  1348. for i in range(tabpagenr("$"))
  1349. if index(tabpagebuflist(i + 1), a:bufNbr) != -1
  1350. return i + 1
  1351. endif
  1352. endfor
  1353. return 0
  1354. endfunction
  1355. " GetWinNbr" {{{2
  1356. function! s:GetWinNbr(tabNbr, bufNbr)
  1357. " window number in tabpage.
  1358. let tablist = tabpagebuflist(a:tabNbr)
  1359. " Number: 0
  1360. " String: 1
  1361. " Funcref: 2
  1362. " List: 3
  1363. " Dictionary: 4
  1364. " Float: 5
  1365. if type(tablist) == 3
  1366. return index(tabpagebuflist(a:tabNbr), a:bufNbr) + 1
  1367. else
  1368. return 1
  1369. endif
  1370. endfunction
  1371. " StringWidth" {{{2
  1372. if exists('*strwidth')
  1373. function s:StringWidth(s)
  1374. return strwidth(a:s)
  1375. endfunction
  1376. else
  1377. function s:StringWidth(s)
  1378. return len(a:s)
  1379. endfunction
  1380. endif
  1381. " Winmanager Integration {{{2
  1382. let g:BufExplorer_title = "\[Buf\ List\]"
  1383. call s:Set("g:bufExplorerResize", 1)
  1384. call s:Set("g:bufExplorerMaxHeight", 25) " Handles dynamic resizing of the window.
  1385. " Evaluate a Vimscript expression in the context of this file.
  1386. " This enables debugging of script-local variables and functions from outside
  1387. " the plugin, e.g.:
  1388. " :echo BufExplorer_eval('s:bufMru')
  1389. function! BufExplorer_eval(expr)
  1390. return eval(a:expr)
  1391. endfunction
  1392. " Execute a Vimscript statement in the context of this file.
  1393. " This enables setting script-local variables from outside the plugin, e.g.:
  1394. " :call BufExplorer_execute('let s:bufMru = s:MRUNew(0)')
  1395. function! BufExplorer_execute(statement)
  1396. execute a:statement
  1397. endfunction
  1398. " function! to start display. Set the mode to 'winmanager' for this buffer.
  1399. " This is to figure out how this plugin was called. In a standalone fashion
  1400. " or by winmanager.
  1401. function! BufExplorer_Start()
  1402. let b:displayMode = "winmanager"
  1403. call s:SetLocalSettings()
  1404. call BufExplorer()
  1405. endfunction
  1406. " Returns whether the display is okay or not.
  1407. function! BufExplorer_IsValid()
  1408. return 0
  1409. endfunction
  1410. " Handles dynamic refreshing of the window.
  1411. function! BufExplorer_Refresh()
  1412. let b:displayMode = "winmanager"
  1413. call s:SetLocalSettings()
  1414. call BufExplorer()
  1415. endfunction
  1416. function! BufExplorer_ReSize()
  1417. if !g:bufExplorerResize
  1418. return
  1419. end
  1420. let nlines = min([line("$"), g:bufExplorerMaxHeight])
  1421. execute nlines." wincmd _"
  1422. " The following lines restore the layout so that the last file line is also
  1423. " the last window line. Sometimes, when a line is deleted, although the
  1424. " window size is exactly equal to the number of lines in the file, some of
  1425. " the lines are pushed up and we see some lagging '~'s.
  1426. let pres = getpos(".")
  1427. normal! $
  1428. let _scr = &scrolloff
  1429. let &scrolloff = 0
  1430. normal! z-
  1431. let &scrolloff = _scr
  1432. call setpos(".", pres)
  1433. endfunction
  1434. " Default values {{{2
  1435. call s:Set("g:bufExplorerDisableDefaultKeyMapping", 0) " Do not disable default key mappings.
  1436. call s:Set("g:bufExplorerDefaultHelp", 1) " Show default help?
  1437. call s:Set("g:bufExplorerDetailedHelp", 0) " Show detailed help?
  1438. call s:Set("g:bufExplorerFindActive", 1) " When selecting an active buffer, take you to the window where it is active?
  1439. call s:Set("g:bufExplorerOnlyOneTab", 1) " Show buffer only on MRU tab? (Applies when `g:bufExplorerShowTabBuffer` is true.)
  1440. call s:Set("g:bufExplorerReverseSort", 0) " Sort in reverse order by default?
  1441. call s:Set("g:bufExplorerShowDirectories", 1) " (Dir's are added by commands like ':e .')
  1442. call s:Set("g:bufExplorerShowRelativePath", 0) " Show listings with relative or absolute paths?
  1443. call s:Set("g:bufExplorerShowTabBuffer", 0) " Show only buffer(s) for this tab?
  1444. call s:Set("g:bufExplorerShowUnlisted", 0) " Show unlisted buffers?
  1445. call s:Set("g:bufExplorerShowNoName", 0) " Show 'No Name' buffers?
  1446. call s:Set("g:bufExplorerSortBy", "mru") " Sorting methods are in s:sort_by:
  1447. call s:Set("g:bufExplorerSplitBelow", &splitbelow) " Should horizontal splits be below or above current window?
  1448. call s:Set("g:bufExplorerSplitOutPathName", 1) " Split out path and file name?
  1449. call s:Set("g:bufExplorerSplitRight", &splitright) " Should vertical splits be on the right or left of current window?
  1450. call s:Set("g:bufExplorerSplitVertSize", 0) " Height for a vertical split. If <=0, default Vim size is used.
  1451. call s:Set("g:bufExplorerSplitHorzSize", 0) " Height for a horizontal split. If <=0, default Vim size is used.
  1452. call s:Set("g:bufExplorerShowTerminal", 1) " Show terminal buffers?
  1453. " Default key mapping {{{2
  1454. if !hasmapto('BufExplorer') && g:bufExplorerDisableDefaultKeyMapping == 0
  1455. nnoremap <script> <silent> <unique> <Leader>be :BufExplorer<CR>
  1456. endif
  1457. if !hasmapto('ToggleBufExplorer') && g:bufExplorerDisableDefaultKeyMapping == 0
  1458. nnoremap <script> <silent> <unique> <Leader>bt :ToggleBufExplorer<CR>
  1459. endif
  1460. if !hasmapto('BufExplorerHorizontalSplit') && g:bufExplorerDisableDefaultKeyMapping == 0
  1461. nnoremap <script> <silent> <unique> <Leader>bs :BufExplorerHorizontalSplit<CR>
  1462. endif
  1463. if !hasmapto('BufExplorerVerticalSplit') && g:bufExplorerDisableDefaultKeyMapping == 0
  1464. nnoremap <script> <silent> <unique> <Leader>bv :BufExplorerVerticalSplit<CR>
  1465. endif
  1466. " vim:ft=vim foldmethod=marker sw=4