mru.vim 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039
  1. " File: mru.vim
  2. " Author: Yegappan Lakshmanan (yegappan AT yahoo DOT com)
  3. " Version: 3.9
  4. " Last Modified: Feb 3, 2015
  5. " Copyright: Copyright (C) 2003-2015 Yegappan Lakshmanan
  6. " License: Permission is hereby granted to use and distribute this code,
  7. " with or without modifications, provided that this copyright
  8. " notice is copied with it. Like anything else that's free,
  9. " mru.vim is provided *as is* and comes with no warranty of any
  10. " kind, either expressed or implied. In no event will the copyright
  11. " holder be liable for any damages resulting from the use of this
  12. " software.
  13. "
  14. " Overview
  15. " --------
  16. " The Most Recently Used (MRU) plugin provides an easy access to a list of
  17. " recently opened/edited files in Vim. This plugin automatically stores the
  18. " file names as you open/edit them in Vim.
  19. "
  20. " This plugin will work on all the platforms where Vim is supported. This
  21. " plugin will work in both console and GUI Vim. This version of the MRU
  22. " plugin needs Vim 7.0 and above. If you are using an earlier version of
  23. " Vim, then you should use an older version of the MRU plugin.
  24. "
  25. " The recently used filenames are stored in a file specified by the Vim
  26. " MRU_File variable.
  27. "
  28. " The Github repository for the MRU plugin is available at:
  29. "
  30. " http://github.com/yegappan/mru
  31. "
  32. " Installation
  33. " ------------
  34. " 1. Copy the mru.vim file to one of the following directories:
  35. " $HOME/.vim/plugin - Unix like systems
  36. " $HOME/vimfiles/plugin - MS-Windows
  37. " $VIM:vimfiles:plugin - Macintosh
  38. " $VIM/vimfiles/plugin - All
  39. " 2. Restart Vim.
  40. " 3. You can use the ":MRU" command to list and edit the recently used files.
  41. " In GUI Vim, you can use the 'File->Recent Files' menu to access the
  42. " recently used files.
  43. "
  44. " To uninstall this plugin, remove this file (mru.vim) from the
  45. " $HOME/.vim/plugin or $HOME/vimfiles/plugin or the $VIM/vimfile/plugin
  46. " directory.
  47. "
  48. " Usage
  49. " -----
  50. " To list and edit files from the MRU list, you can use the ":MRU" command.
  51. " The ":MRU" command displays the MRU file list in a temporary Vim window. If
  52. " the MRU window is already opened, then the MRU list displayed in the window
  53. " is refreshed.
  54. "
  55. " If you are using GUI Vim, then the names of the recently edited files are
  56. " added to the "File->Recent Files" menu. You can select the name of a file
  57. " from this sub-menu to edit the file.
  58. "
  59. " You can use the normal Vim commands to move around in the MRU window. You
  60. " cannot make changes in the MRU window.
  61. "
  62. " You can select a file name to edit by pressing the <Enter> key or by double
  63. " clicking the left mouse button on a file name. The selected file will be
  64. " opened. If the file is already opened in a window, the cursor will be moved
  65. " to that window. Otherwise, the file is opened in the previous window. If the
  66. " previous window has a modified buffer or is the preview window or is used by
  67. " some other plugin, then the file is opened in a new window.
  68. "
  69. " You can press the 'o' key to open the file name under the cursor in the
  70. " MRU window in a new window. You can also press <Shift-Enter> instead of 'o'
  71. " to open the file in a new window.
  72. "
  73. " To open a file from the MRU window in read-only mode (view), press the 'v'
  74. " key.
  75. "
  76. " To open a file from the MRU window in a new tab, press the 't' key. If the
  77. " file is already opened in a window in the current or in another tab, then
  78. " the cursor is moved to that tab. Otherwise, a new tab is opened.
  79. "
  80. " You can open multiple files from the MRU window by specifying a count before
  81. " pressing '<Enter>' or 'v' or 'o' or 't'. You can also visually (using
  82. " linewise visual mode) select multiple filenames and invoke the commands to
  83. " open the files. Each selected file will be opened in a separate window or
  84. " tab.
  85. "
  86. " You can press the 'u' key in the MRU window to update the file list. This is
  87. " useful if you keep the MRU window open always.
  88. "
  89. " You can close the MRU window by pressing the 'q' key or the <Esc> key or
  90. " using one of the Vim window commands.
  91. "
  92. " To display only files matching a pattern from the MRU list in the MRU
  93. " window, you can specify a pattern to the ":MRU" command. For example, to
  94. " display only file names matching "vim" in them, you can use the following
  95. " command ":MRU vim". When you specify a partial file name and only one
  96. " matching filename is found, then the ":MRU" command will edit that file.
  97. "
  98. " The ":MRU" command supports command-line completion of file names from
  99. " the MRU list. You can enter a partial file name and then press <Tab>
  100. " or <Ctrl-D> to complete or list all the matching file names. Note that
  101. " after typing the ":MRU" command, you have to enter a space before completing
  102. " the file names with <Tab>.
  103. "
  104. " When a file supplied to the ":MRU" command is not present in the MRU list,
  105. " but it is a readable file, then the file will be opened (even though it is
  106. " not present in the MRU list). This is useful if you want to open a file
  107. " present in the same directory as a file in the MRU list. You can use the
  108. " command-line completion of the ":MRU" command to complete the full path of a
  109. " file and then modify the path to open another file present in the same path.
  110. "
  111. " Whenever the MRU list changes, the MRU file is updated with the latest MRU
  112. " list. When you have multiple instances of Vim running at the same time, the
  113. " latest MRU list will show up in all the instances of Vim.
  114. "
  115. " The MRUFilename syntax group is used to highlight the file names in the MRU
  116. " window. By default, this syntax group is linked to the Identifier highlight
  117. " group. You can change the highlight group by adding the following line in
  118. " your .vimrc:
  119. "
  120. " highlight link MRUFileName LineNr
  121. "
  122. " The MRU buffer uses the 'mru file type. You can use this file type to add
  123. " custom auto commands, syntax highlighting, etc.
  124. "
  125. " Configuration
  126. " -------------
  127. " By changing the following variables you can configure the behavior of this
  128. " plugin. Set the following variables in your .vimrc file using the 'let'
  129. " command.
  130. "
  131. " The list of recently edited file names is stored in the file specified by the
  132. " MRU_File variable. The default setting for this variable is
  133. " $HOME/.vim_mru_files for Unix-like systems and $USERPROFILE/_vim_mru_files
  134. " for MS-Windows systems. You can change this variable to point to a file by
  135. " adding the following line to the .vimrc file:
  136. "
  137. " let MRU_File = 'd:\myhome\_vim_mru_files'
  138. "
  139. " By default, the plugin will remember the names of the last 100 used files.
  140. " As you edit more files, old file names will be removed from the MRU list.
  141. " You can set the 'MRU_Max_Entries' variable to remember more file names. For
  142. " example, to remember 1000 most recently used file names, you can use
  143. "
  144. " let MRU_Max_Entries = 1000
  145. "
  146. " By default, all the edited file names will be added to the MRU list. If you
  147. " want to exclude file names matching a list of patterns, you can set the
  148. " MRU_Exclude_Files variable to a list of Vim regular expressions. By default,
  149. " this variable is set to an empty string. For example, to not include files
  150. " in the temporary (/tmp, /var/tmp and d:\temp) directories, you can set the
  151. " MRU_Exclude_Files variable to
  152. "
  153. " let MRU_Exclude_Files = '^/tmp/.*\|^/var/tmp/.*' " For Unix
  154. " let MRU_Exclude_Files = '^c:\\temp\\.*' " For MS-Windows
  155. "
  156. " The specified pattern should be a Vim regular expression pattern.
  157. "
  158. " If you want to add only file names matching a set of patterns to the MRU
  159. " list, then you can set the MRU_Include_Files variable. This variable should
  160. " be set to a Vim regular expression pattern. For example, to add only .c and
  161. " .h files to the MRU list, you can set this variable as below:
  162. "
  163. " let MRU_Include_Files = '\.c$\|\.h$'
  164. "
  165. " By default, MRU_Include_Files is set to an empty string and all the edited
  166. " filenames are added to the MRU list.
  167. "
  168. " The default height of the MRU window is 8. You can set the MRU_Window_Height
  169. " variable to change the window height.
  170. "
  171. " let MRU_Window_Height = 15
  172. "
  173. " By default, when the :MRU command is invoked, the MRU list will be displayed
  174. " in a new window. Instead, if you want the MRU plugin to reuse the current
  175. " window, then you can set the 'MRU_Use_Current_Window' variable to one.
  176. "
  177. " let MRU_Use_Current_Window = 1
  178. "
  179. " The MRU plugin will reuse the current window. When a file name is selected,
  180. " the file is also opened in the current window.
  181. "
  182. " When you select a file from the MRU window, the MRU window will be
  183. " automatically closed and the selected file will be opened in the previous
  184. " window. You can set the 'MRU_Auto_Close' variable to zero to keep the MRU
  185. " window open.
  186. "
  187. " let MRU_Auto_Close = 0
  188. "
  189. " If you don't use the "File->Recent Files" menu and want to disable it,
  190. " then you can set the 'MRU_Add_Menu' variable to zero. By default, the
  191. " menu is enabled.
  192. "
  193. " let MRU_Add_Menu = 0
  194. "
  195. " If too many file names are present in the MRU list, then updating the MRU
  196. " menu to list all the file names makes Vim slow. To avoid this, the
  197. " MRU_Max_Menu_Entries variable controls the number of file names to show in
  198. " the MRU menu. By default, this is set to 10. You can change this to show
  199. " more entries in the menu.
  200. "
  201. " let MRU_Max_Menu_Entries = 20
  202. "
  203. " If many file names are present in the MRU list, then the MRU menu is split
  204. " into sub-menus. Each sub-menu contains MRU_Max_Submenu_Entries file names.
  205. " The default setting for this is 10. You can change this to increase the
  206. " number of file names displayed in a single sub-menu:
  207. "
  208. " let MRU_Max_Submenu_Entries = 15
  209. "
  210. " In the MRU window, the filenames are displayed in two parts. The first part
  211. " contains the file name without the path and the second part contains the
  212. " full path to the file in parenthesis. This format is controlled by the
  213. " MRU_Filename_Format variable. If you prefer to change this to some other
  214. " format, then you can modify the MRU_Filename_Format variable. For example,
  215. " to display the full path without splitting it, you can set this variable
  216. " as shown below:
  217. "
  218. " let MRU_Filename_Format =
  219. " \ {'formatter':'v:val', 'parser':'.*', 'syntax': '[^/\\]\+$'}
  220. "
  221. " ****************** Do not modify after this line ************************
  222. if exists('loaded_mru')
  223. finish
  224. endif
  225. let loaded_mru=1
  226. if v:version < 700
  227. finish
  228. endif
  229. " Line continuation used here
  230. let s:cpo_save = &cpo
  231. set cpo&vim
  232. " MRU configuration variables {{{1
  233. " Maximum number of entries allowed in the MRU list
  234. if !exists('MRU_Max_Entries')
  235. let MRU_Max_Entries = 100
  236. endif
  237. " Files to exclude from the MRU list
  238. if !exists('MRU_Exclude_Files')
  239. let MRU_Exclude_Files = ''
  240. endif
  241. " Files to include in the MRU list
  242. if !exists('MRU_Include_Files')
  243. let MRU_Include_Files = ''
  244. endif
  245. " Height of the MRU window
  246. " Default height is 8
  247. if !exists('MRU_Window_Height')
  248. let MRU_Window_Height = 8
  249. endif
  250. if !exists('MRU_Use_Current_Window')
  251. let MRU_Use_Current_Window = 0
  252. endif
  253. if !exists('MRU_Auto_Close')
  254. let MRU_Auto_Close = 1
  255. endif
  256. if !exists('MRU_File')
  257. if has('unix') || has('macunix')
  258. let MRU_File = $HOME . '/.vim_mru_files'
  259. else
  260. let MRU_File = $VIM . '/_vim_mru_files'
  261. if has('win32')
  262. " MS-Windows
  263. if $USERPROFILE != ''
  264. let MRU_File = $USERPROFILE . '\_vim_mru_files'
  265. endif
  266. endif
  267. endif
  268. endif
  269. " Option for enabling or disabling the MRU menu
  270. if !exists('MRU_Add_Menu')
  271. let MRU_Add_Menu = 1
  272. endif
  273. " Maximum number of file names to show in the MRU menu. If too many files are
  274. " listed in the menu, then Vim becomes slow when updating the menu. So set
  275. " this to a low value.
  276. if !exists('MRU_Max_Menu_Entries')
  277. let MRU_Max_Menu_Entries = 10
  278. endif
  279. " Maximum number of file names to show in a MRU sub-menu. If the MRU list
  280. " contains more file names than this setting, then the MRU menu is split into
  281. " one or more sub-menus.
  282. if !exists('MRU_Max_Submenu_Entries')
  283. let MRU_Max_Submenu_Entries = 10
  284. endif
  285. " When only a single matching filename is found in the MRU list, the following
  286. " option controls whether the file name is displayed in the MRU window or the
  287. " file is directly opened. When this variable is set to 0 and a single
  288. " matching file name is found, then the file is directly opened.
  289. if !exists('MRU_Window_Open_Always')
  290. let MRU_Window_Open_Always = 0
  291. endif
  292. " When opening a file from the MRU list, the file is opened in the current
  293. " tab. If the selected file has to be opened in a tab always, then set the
  294. " following variable to 1. If the file is already opened in a tab, then the
  295. " cursor will be moved to that tab.
  296. if !exists('MRU_Open_File_Use_Tabs')
  297. let MRU_Open_File_Use_Tabs = 0
  298. endif
  299. " Format of the file names displayed in the MRU window.
  300. " The default is to display the filename followed by the complete path to the
  301. " file in parenthesis. This variable controls the expressions used to format
  302. " and parse the path. This can be changed to display the filenames in a
  303. " different format. The 'formatter' specifies how to split/format the filename
  304. " and 'parser' specifies how to read the filename back; 'syntax' matches the
  305. " part to be highlighted.
  306. if !exists('MRU_Filename_Format')
  307. let MRU_Filename_Format = {
  308. \ 'formatter': 'fnamemodify(v:val, ":t") . " (" . v:val . ")"',
  309. \ 'parser': '(\zs.*\ze)',
  310. \ 'syntax': '^.\{-}\ze('
  311. \}
  312. endif
  313. " Control to temporarily lock the MRU list. Used to prevent files from
  314. " getting added to the MRU list when the ':vimgrep' command is executed.
  315. let s:mru_list_locked = 0
  316. " MRU_LoadList {{{1
  317. " Loads the latest list of file names from the MRU file
  318. function! s:MRU_LoadList()
  319. " If the MRU file is present, then load the list of filenames. Otherwise
  320. " start with an empty list.
  321. if filereadable(g:MRU_File)
  322. let s:MRU_files = readfile(g:MRU_File)
  323. if s:MRU_files[0] =~# '^\s*" Most recently edited files in Vim'
  324. " Generated by the previous version of the MRU plugin.
  325. " Discard the list.
  326. let s:MRU_files = []
  327. elseif s:MRU_files[0] =~# '^#'
  328. " Remove the comment line
  329. call remove(s:MRU_files, 0)
  330. else
  331. " Unsupported format
  332. let s:MRU_files = []
  333. endif
  334. else
  335. let s:MRU_files = []
  336. endif
  337. " Refresh the MRU menu with the latest list of filenames
  338. call s:MRU_Refresh_Menu()
  339. endfunction
  340. " MRU_SaveList {{{1
  341. " Saves the MRU file names to the MRU file
  342. function! s:MRU_SaveList()
  343. let l = []
  344. call add(l, '# Most recently edited files in Vim (version 3.0)')
  345. call extend(l, s:MRU_files)
  346. call writefile(l, g:MRU_File)
  347. endfunction
  348. " MRU_AddFile {{{1
  349. " Adds a file to the MRU file list
  350. " acmd_bufnr - Buffer number of the file to add
  351. function! s:MRU_AddFile(acmd_bufnr)
  352. if s:mru_list_locked
  353. " MRU list is currently locked
  354. return
  355. endif
  356. " Get the full path to the filename
  357. let fname = fnamemodify(bufname(a:acmd_bufnr + 0), ':p')
  358. if fname == ''
  359. return
  360. endif
  361. " Skip temporary buffers with buftype set. The buftype is set for buffers
  362. " used by plugins.
  363. if &buftype != ''
  364. return
  365. endif
  366. if g:MRU_Include_Files != ''
  367. " If MRU_Include_Files is set, include only files matching the
  368. " specified pattern
  369. if fname !~# g:MRU_Include_Files
  370. return
  371. endif
  372. endif
  373. if g:MRU_Exclude_Files != ''
  374. " Do not add files matching the pattern specified in the
  375. " MRU_Exclude_Files to the MRU list
  376. if fname =~# g:MRU_Exclude_Files
  377. return
  378. endif
  379. endif
  380. " If the filename is not already present in the MRU list and is not
  381. " readable then ignore it
  382. let idx = index(s:MRU_files, fname)
  383. if idx == -1
  384. if !filereadable(fname)
  385. " File is not readable and is not in the MRU list
  386. return
  387. endif
  388. endif
  389. " Load the latest MRU file list
  390. call s:MRU_LoadList()
  391. " Remove the new file name from the existing MRU list (if already present)
  392. call filter(s:MRU_files, 'v:val !=# fname')
  393. " Add the new file list to the beginning of the updated old file list
  394. call insert(s:MRU_files, fname, 0)
  395. " Trim the list
  396. if len(s:MRU_files) > g:MRU_Max_Entries
  397. call remove(s:MRU_files, g:MRU_Max_Entries, -1)
  398. endif
  399. " Save the updated MRU list
  400. call s:MRU_SaveList()
  401. " Refresh the MRU menu
  402. call s:MRU_Refresh_Menu()
  403. " If the MRU window is open, update the displayed MRU list
  404. let bname = '__MRU_Files__'
  405. let winnum = bufwinnr(bname)
  406. if winnum != -1
  407. let cur_winnr = winnr()
  408. call s:MRU_Open_Window()
  409. if winnr() != cur_winnr
  410. exe cur_winnr . 'wincmd w'
  411. endif
  412. endif
  413. endfunction
  414. " MRU_escape_filename {{{1
  415. " Escape special characters in a filename. Special characters in file names
  416. " that should be escaped (for security reasons)
  417. let s:esc_filename_chars = ' *?[{`$%#"|!<>();&' . "'\t\n"
  418. function! s:MRU_escape_filename(fname)
  419. if exists("*fnameescape")
  420. return fnameescape(a:fname)
  421. else
  422. return escape(a:fname, s:esc_filename_chars)
  423. endif
  424. endfunction
  425. " MRU_Edit_File {{{1
  426. " Edit the specified file
  427. " filename - Name of the file to edit
  428. " sanitized - Specifies whether the filename is already escaped for special
  429. " characters or not.
  430. function! s:MRU_Edit_File(filename, sanitized)
  431. if !a:sanitized
  432. let esc_fname = s:MRU_escape_filename(a:filename)
  433. else
  434. let esc_fname = a:filename
  435. endif
  436. " If the user wants to always open the file in a tab, then open the file
  437. " in a tab. If it is already opened in a tab, then the cursor will be
  438. " moved to that tab.
  439. if g:MRU_Open_File_Use_Tabs
  440. call s:MRU_Open_File_In_Tab(a:filename, esc_fname)
  441. return
  442. endif
  443. " If the file is already open in one of the windows, jump to it
  444. let winnum = bufwinnr('^' . a:filename . '$')
  445. if winnum != -1
  446. if winnum != winnr()
  447. exe winnum . 'wincmd w'
  448. endif
  449. else
  450. if !&hidden && (&modified || &buftype != '' || &previewwindow)
  451. " Current buffer has unsaved changes or is a special buffer or is
  452. " the preview window. The 'hidden' option is also not set.
  453. " So open the file in a new window.
  454. exe 'split ' . esc_fname
  455. else
  456. " The current file can be replaced with the selected file.
  457. exe 'edit ' . esc_fname
  458. endif
  459. endif
  460. endfunction
  461. " MRU_Open_File_In_Tab
  462. " Open a file in a tab. If the file is already opened in a tab, jump to the
  463. " tab. Otherwise, create a new tab and open the file.
  464. " fname : Name of the file to open
  465. " esc_fname : File name with special characters escaped
  466. function! s:MRU_Open_File_In_Tab(fname, esc_fname)
  467. " If the selected file is already open in the current tab or in
  468. " another tab, jump to it. Otherwise open it in a new tab
  469. if bufwinnr('^' . a:fname . '$') == -1
  470. let tabnum = -1
  471. let i = 1
  472. let bnum = bufnr('^' . a:fname . '$')
  473. while i <= tabpagenr('$')
  474. if index(tabpagebuflist(i), bnum) != -1
  475. let tabnum = i
  476. break
  477. endif
  478. let i += 1
  479. endwhile
  480. if tabnum != -1
  481. " Goto the tab containing the file
  482. exe 'tabnext ' . i
  483. else
  484. " Open a new tab as the last tab page
  485. exe '$tabnew ' . a:esc_fname
  486. endif
  487. endif
  488. " Jump to the window containing the file
  489. let winnum = bufwinnr('^' . a:fname . '$')
  490. if winnum != winnr()
  491. exe winnum . 'wincmd w'
  492. endif
  493. endfunction
  494. " MRU_Window_Edit_File {{{1
  495. " fname : Name of the file to edit. May specify single or multiple
  496. " files.
  497. " edit_type : Specifies how to edit the file. Can be one of 'edit' or 'view'.
  498. " 'view' - Open the file as a read-only file
  499. " 'edit' - Edit the file as a regular file
  500. " multi : Specifies whether a single file or multiple files need to be
  501. " opened.
  502. " open_type : Specifies where to open the file.
  503. " useopen - If the file is already present in a window, then
  504. " jump to that window. Otherwise, open the file in
  505. " the previous window.
  506. " newwin_horiz - Open the file in a new horizontal window.
  507. " newwin_vert - Open the file in a new vertical window.
  508. " newtab - Open the file in a new tab. If the file is already
  509. " opened in a tab, then jump to that tab.
  510. " preview - Open the file in the preview window
  511. function! s:MRU_Window_Edit_File(fname, multi, edit_type, open_type)
  512. let esc_fname = s:MRU_escape_filename(a:fname)
  513. if a:open_type ==# 'newwin_horiz'
  514. " Edit the file in a new horizontally split window above the previous
  515. " window
  516. wincmd p
  517. exe 'belowright new ' . esc_fname
  518. elseif a:open_type ==# 'newwin_vert'
  519. " Edit the file in a new vertically split window above the previous
  520. " window
  521. wincmd p
  522. exe 'belowright vnew ' . esc_fname
  523. elseif a:open_type ==# 'newtab' || g:MRU_Open_File_Use_Tabs
  524. call s:MRU_Open_File_In_Tab(a:fname, esc_fname)
  525. elseif a:open_type ==# 'preview'
  526. " Edit the file in the preview window
  527. exe 'topleft pedit ' . esc_fname
  528. else
  529. " If the selected file is already open in one of the windows,
  530. " jump to it
  531. let winnum = bufwinnr('^' . a:fname . '$')
  532. if winnum != -1
  533. exe winnum . 'wincmd w'
  534. else
  535. if g:MRU_Auto_Close == 1 && g:MRU_Use_Current_Window == 0
  536. " Jump to the window from which the MRU window was opened
  537. if exists('s:MRU_last_buffer')
  538. let last_winnr = bufwinnr(s:MRU_last_buffer)
  539. if last_winnr != -1 && last_winnr != winnr()
  540. exe last_winnr . 'wincmd w'
  541. endif
  542. endif
  543. else
  544. if g:MRU_Use_Current_Window == 0
  545. " Goto the previous window
  546. " If MRU_Use_Current_Window is set to one, then the
  547. " current window is used to open the file
  548. wincmd p
  549. endif
  550. endif
  551. let split_window = 0
  552. if (!&hidden && (&modified || &previewwindow)) || a:multi
  553. " Current buffer has unsaved changes or is the preview window
  554. " or the user is opening multiple files
  555. " So open the file in a new window
  556. let split_window = 1
  557. endif
  558. if &buftype != ''
  559. " Current buffer is a special buffer (maybe used by a plugin)
  560. if g:MRU_Use_Current_Window == 0 ||
  561. \ bufnr('%') != bufnr('__MRU_Files__')
  562. let split_window = 1
  563. endif
  564. endif
  565. " Edit the file
  566. if split_window
  567. " Current buffer has unsaved changes or is a special buffer or
  568. " is the preview window. So open the file in a new window
  569. if a:edit_type ==# 'edit'
  570. exe 'split ' . esc_fname
  571. else
  572. exe 'sview ' . esc_fname
  573. endif
  574. else
  575. if a:edit_type ==# 'edit'
  576. exe 'edit ' . esc_fname
  577. else
  578. exe 'view ' . esc_fname
  579. endif
  580. endif
  581. endif
  582. endif
  583. endfunction
  584. " MRU_Select_File_Cmd {{{1
  585. " Open a file selected from the MRU window
  586. "
  587. " 'opt' has two values separated by comma. The first value specifies how to
  588. " edit the file and can be either 'edit' or 'view'. The second value
  589. " specifies where to open the file. It can take one of the following values:
  590. " 'useopen' to open file in the previous window
  591. " 'newwin_horiz' to open the file in a new horizontal split window
  592. " 'newwin_vert' to open the file in a new vertical split window.
  593. " 'newtab' to open the file in a new tab.
  594. " If multiple file names are selected using visual mode, then open multiple
  595. " files (either in split windows or tabs)
  596. function! s:MRU_Select_File_Cmd(opt) range
  597. let [edit_type, open_type] = split(a:opt, ',')
  598. let fnames = getline(a:firstline, a:lastline)
  599. if g:MRU_Auto_Close == 1 && g:MRU_Use_Current_Window == 0
  600. " Automatically close the window if the file window is
  601. " not used to display the MRU list.
  602. silent! close
  603. endif
  604. let multi = 0
  605. for f in fnames
  606. if f == ''
  607. continue
  608. endif
  609. " The text in the MRU window contains the filename in parenthesis
  610. let file = matchstr(f, g:MRU_Filename_Format.parser)
  611. call s:MRU_Window_Edit_File(file, multi, edit_type, open_type)
  612. if a:firstline != a:lastline
  613. " Opening multiple files
  614. let multi = 1
  615. endif
  616. endfor
  617. endfunction
  618. " MRU_Warn_Msg {{{1
  619. " Display a warning message
  620. function! s:MRU_Warn_Msg(msg)
  621. echohl WarningMsg
  622. echo a:msg
  623. echohl None
  624. endfunction
  625. " MRU_Open_Window {{{1
  626. " Display the Most Recently Used file list in a temporary window.
  627. " If the optional argument is supplied, then it specifies the pattern of files
  628. " to selectively display in the MRU window.
  629. function! s:MRU_Open_Window(...)
  630. " Load the latest MRU file list
  631. call s:MRU_LoadList()
  632. " Check for empty MRU list
  633. if empty(s:MRU_files)
  634. call s:MRU_Warn_Msg('MRU file list is empty')
  635. return
  636. endif
  637. " Save the current buffer number. This is used later to open a file when a
  638. " entry is selected from the MRU window. The window number is not saved,
  639. " as the window number will change when new windows are opened.
  640. let s:MRU_last_buffer = bufnr('%')
  641. let bname = '__MRU_Files__'
  642. " If the window is already open, jump to it
  643. let winnum = bufwinnr(bname)
  644. if winnum != -1
  645. if winnr() != winnum
  646. " If not already in the window, jump to it
  647. exe winnum . 'wincmd w'
  648. endif
  649. setlocal modifiable
  650. " Delete the contents of the buffer to the black-hole register
  651. silent! %delete _
  652. else
  653. if g:MRU_Use_Current_Window
  654. " Reuse the current window
  655. "
  656. " If the __MRU_Files__ buffer exists, then reuse it. Otherwise open
  657. " a new buffer
  658. let bufnum = bufnr(bname)
  659. if bufnum == -1
  660. let cmd = 'edit ' . bname
  661. else
  662. let cmd = 'buffer ' . bufnum
  663. endif
  664. exe cmd
  665. if bufnr('%') != bufnr(bname)
  666. " Failed to edit the MRU buffer
  667. return
  668. endif
  669. else
  670. " Open a new window at the bottom
  671. " If the __MRU_Files__ buffer exists, then reuse it. Otherwise open
  672. " a new buffer
  673. let bufnum = bufnr(bname)
  674. if bufnum == -1
  675. let wcmd = bname
  676. else
  677. let wcmd = '+buffer' . bufnum
  678. endif
  679. exe 'silent! botright ' . g:MRU_Window_Height . 'split ' . wcmd
  680. endif
  681. endif
  682. setlocal modifiable
  683. " Mark the buffer as scratch
  684. setlocal buftype=nofile
  685. setlocal bufhidden=delete
  686. setlocal noswapfile
  687. setlocal nowrap
  688. setlocal nobuflisted
  689. " Set the 'filetype' to 'mru'. This allows the user to apply custom
  690. " syntax highlighting or other changes to the MRU bufer.
  691. setlocal filetype=mru
  692. " Use fixed height for the MRU window
  693. setlocal winfixheight
  694. " Setup the cpoptions properly for the maps to work
  695. let old_cpoptions = &cpoptions
  696. set cpoptions&vim
  697. " Create mappings to select and edit a file from the MRU list
  698. nnoremap <buffer> <silent> <CR>
  699. \ :call <SID>MRU_Select_File_Cmd('edit,useopen')<CR>
  700. vnoremap <buffer> <silent> <CR>
  701. \ :call <SID>MRU_Select_File_Cmd('edit,useopen')<CR>
  702. nnoremap <buffer> <silent> o
  703. \ :call <SID>MRU_Select_File_Cmd('edit,newwin_horiz')<CR>
  704. vnoremap <buffer> <silent> o
  705. \ :call <SID>MRU_Select_File_Cmd('edit,newwin_horiz')<CR>
  706. nnoremap <buffer> <silent> <S-CR>
  707. \ :call <SID>MRU_Select_File_Cmd('edit,newwin_horiz')<CR>
  708. vnoremap <buffer> <silent> <S-CR>
  709. \ :call <SID>MRU_Select_File_Cmd('edit,newwin_horiz')<CR>
  710. nnoremap <buffer> <silent> O
  711. \ :call <SID>MRU_Select_File_Cmd('edit,newwin_vert')<CR>
  712. vnoremap <buffer> <silent> O
  713. \ :call <SID>MRU_Select_File_Cmd('edit,newwin_vert')<CR>
  714. nnoremap <buffer> <silent> t
  715. \ :call <SID>MRU_Select_File_Cmd('edit,newtab')<CR>
  716. vnoremap <buffer> <silent> t
  717. \ :call <SID>MRU_Select_File_Cmd('edit,newtab')<CR>
  718. nnoremap <buffer> <silent> v
  719. \ :call <SID>MRU_Select_File_Cmd('view,useopen')<CR>
  720. nnoremap <buffer> <silent> p
  721. \ :call <SID>MRU_Select_File_Cmd('view,preview')<CR>
  722. vnoremap <buffer> <silent> p
  723. \ :<C-u>if line("'<") == line("'>")<Bar>
  724. \ call <SID>MRU_Select_File_Cmd('open,preview')<Bar>
  725. \ else<Bar>
  726. \ echoerr "Only a single file can be previewed"<Bar>
  727. \ endif<CR>
  728. nnoremap <buffer> <silent> u :MRU<CR>
  729. nnoremap <buffer> <silent> <2-LeftMouse>
  730. \ :call <SID>MRU_Select_File_Cmd('edit,useopen')<CR>
  731. nnoremap <buffer> <silent> q :close<CR>
  732. " Restore the previous cpoptions settings
  733. let &cpoptions = old_cpoptions
  734. " Display the MRU list
  735. if a:0 == 0
  736. " No search pattern specified. Display the complete list
  737. let m = copy(s:MRU_files)
  738. else
  739. " Display only the entries matching the specified pattern
  740. " First try using it as a literal pattern
  741. let m = filter(copy(s:MRU_files), 'stridx(v:val, a:1) != -1')
  742. if len(m) == 0
  743. " No match. Try using it as a regular expression
  744. let m = filter(copy(s:MRU_files), 'v:val =~# a:1')
  745. endif
  746. endif
  747. " Get the tail part of the file name (without the directory) and display
  748. " it along with the full path in parenthesis.
  749. let output = map(m, g:MRU_Filename_Format.formatter)
  750. silent! 0put =output
  751. " Delete the empty line at the end of the buffer
  752. silent! $delete _
  753. " Move the cursor to the beginning of the file
  754. normal! gg
  755. " Add syntax highlighting for the file names
  756. if has_key(g:MRU_Filename_Format, 'syntax')
  757. exe "syntax match MRUFileName '" . g:MRU_Filename_Format.syntax . "'"
  758. highlight default link MRUFileName Identifier
  759. endif
  760. setlocal nomodifiable
  761. endfunction
  762. " MRU_Complete {{{1
  763. " Command-line completion function used by :MRU command
  764. function! s:MRU_Complete(ArgLead, CmdLine, CursorPos)
  765. if a:ArgLead == ''
  766. " Return the complete list of MRU files
  767. return s:MRU_files
  768. else
  769. " Return only the files matching the specified pattern
  770. return filter(copy(s:MRU_files), 'v:val =~? a:ArgLead')
  771. endif
  772. endfunction
  773. " MRU_Cmd {{{1
  774. " Function to handle the MRU command
  775. " pat - File name pattern passed to the MRU command
  776. function! s:MRU_Cmd(pat)
  777. if a:pat == ''
  778. " No arguments specified. Open the MRU window
  779. call s:MRU_Open_Window()
  780. return
  781. endif
  782. " Load the latest MRU file
  783. call s:MRU_LoadList()
  784. " Empty MRU list
  785. if empty(s:MRU_files)
  786. call s:MRU_Warn_Msg('MRU file list is empty')
  787. return
  788. endif
  789. " First use the specified string as a literal string and search for
  790. " filenames containing the string. If only one filename is found,
  791. " then edit it (unless the user wants to open the MRU window always)
  792. let m = filter(copy(s:MRU_files), 'stridx(v:val, a:pat) != -1')
  793. if len(m) > 0
  794. if len(m) == 1 && !g:MRU_Window_Open_Always
  795. call s:MRU_Edit_File(m[0], 0)
  796. return
  797. endif
  798. " More than one file matches. Try find an accurate match
  799. let new_m = filter(m, 'v:val ==# a:pat')
  800. if len(new_m) == 1 && !g:MRU_Window_Open_Always
  801. call s:MRU_Edit_File(new_m[0], 0)
  802. return
  803. endif
  804. " Couldn't find an exact match, open the MRU window with all the
  805. " files matching the pattern.
  806. call s:MRU_Open_Window(a:pat)
  807. return
  808. endif
  809. " Use the specified string as a regular expression pattern and search
  810. " for filenames matching the pattern
  811. let m = filter(copy(s:MRU_files), 'v:val =~? a:pat')
  812. if len(m) == 0
  813. " If an existing file (not present in the MRU list) is specified,
  814. " then open the file.
  815. if filereadable(a:pat)
  816. call s:MRU_Edit_File(a:pat, 0)
  817. return
  818. endif
  819. " No filenames matching the specified pattern are found
  820. call s:MRU_Warn_Msg("MRU file list doesn't contain " .
  821. \ "files matching " . a:pat)
  822. return
  823. endif
  824. if len(m) == 1 && !g:MRU_Window_Open_Always
  825. call s:MRU_Edit_File(m[0], 0)
  826. return
  827. endif
  828. call s:MRU_Open_Window(a:pat)
  829. endfunction
  830. " MRU_add_files_to_menu {{{1
  831. " Adds a list of files to the "Recent Files" sub menu under the "File" menu.
  832. " prefix - Prefix to use for each of the menu entries
  833. " file_list - List of file names to add to the menu
  834. function! s:MRU_add_files_to_menu(prefix, file_list)
  835. for fname in a:file_list
  836. " Escape special characters in the filename
  837. let esc_fname = escape(fnamemodify(fname, ':t'), ".\\" .
  838. \ s:esc_filename_chars)
  839. let esc_fname = substitute(esc_fname, '&', '&&', 'g')
  840. " Truncate the directory name if it is long
  841. let dir_name = fnamemodify(fname, ':h')
  842. let len = strlen(dir_name)
  843. " Shorten long file names by adding only few characters from
  844. " the beginning and end.
  845. if len > 30
  846. let dir_name = strpart(dir_name, 0, 10) .
  847. \ '...' .
  848. \ strpart(dir_name, len - 20)
  849. endif
  850. let esc_dir_name = escape(dir_name, ".\\" . s:esc_filename_chars)
  851. let esc_dir_name = substitute(esc_dir_name, '&', '&&', 'g')
  852. let menu_path = '&File.&Recent\ Files.' . a:prefix . esc_fname .
  853. \ '\ (' . esc_dir_name . ')'
  854. let esc_mfname = s:MRU_escape_filename(fname)
  855. exe 'anoremenu <silent> ' . menu_path .
  856. \ " :call <SID>MRU_Edit_File('" . esc_mfname . "', 1)<CR>"
  857. exe 'tmenu ' . menu_path . ' Edit file ' . esc_mfname
  858. endfor
  859. endfunction
  860. " MRU_Refresh_Menu {{{1
  861. " Refresh the MRU menu
  862. function! s:MRU_Refresh_Menu()
  863. if !has('menu') || !g:MRU_Add_Menu
  864. " No support for menus
  865. return
  866. endif
  867. " Setup the cpoptions properly for the maps to work
  868. let old_cpoptions = &cpoptions
  869. set cpoptions&vim
  870. " Remove the MRU menu
  871. " To retain the teared-off MRU menu, we need to add a dummy entry
  872. silent! unmenu &File.&Recent\ Files
  873. " The menu priority of the File menu is 10. If the MRU plugin runs
  874. " first before menu.vim, the File menu order may not be correct.
  875. " So specify the priority of the File menu here.
  876. 10noremenu &File.&Recent\ Files.Dummy <Nop>
  877. silent! unmenu! &File.&Recent\ Files
  878. anoremenu <silent> &File.&Recent\ Files.Refresh\ list
  879. \ :call <SID>MRU_LoadList()<CR>
  880. exe 'tmenu File.&Recent\ Files.Refresh\ list Reload the MRU file list from '
  881. \ . s:MRU_escape_filename(g:MRU_File)
  882. anoremenu File.&Recent\ Files.-SEP1- :
  883. " Add the filenames in the MRU list to the menu
  884. let entry_cnt = len(s:MRU_files)
  885. if entry_cnt > g:MRU_Max_Menu_Entries
  886. " Show only MRU_Max_Menu_Entries file names in the menu
  887. let mru_list = s:MRU_files[0 : g:MRU_Max_Menu_Entries - 1]
  888. let entry_cnt = g:MRU_Max_Menu_Entries
  889. else
  890. let mru_list = s:MRU_files
  891. endif
  892. if entry_cnt > g:MRU_Max_Submenu_Entries
  893. " Split the MRU menu into sub-menus
  894. for start_idx in range(0, entry_cnt, g:MRU_Max_Submenu_Entries)
  895. let last_idx = start_idx + g:MRU_Max_Submenu_Entries - 1
  896. if last_idx >= entry_cnt
  897. let last_idx = entry_cnt - 1
  898. endif
  899. let prefix = 'Files\ (' . (start_idx + 1) . '\.\.\.' .
  900. \ (last_idx + 1) . ').'
  901. call s:MRU_add_files_to_menu(prefix,
  902. \ mru_list[start_idx : last_idx])
  903. endfor
  904. else
  905. call s:MRU_add_files_to_menu('', mru_list)
  906. endif
  907. " Remove the dummy menu entry
  908. unmenu &File.&Recent\ Files.Dummy
  909. " Restore the previous cpoptions settings
  910. let &cpoptions = old_cpoptions
  911. endfunction
  912. " Load the MRU list on plugin startup
  913. call s:MRU_LoadList()
  914. " MRU autocommands {{{1
  915. " Autocommands to detect the most recently used files
  916. autocmd BufRead * call s:MRU_AddFile(expand('<abuf>'))
  917. autocmd BufNewFile * call s:MRU_AddFile(expand('<abuf>'))
  918. autocmd BufWritePost * call s:MRU_AddFile(expand('<abuf>'))
  919. " The ':vimgrep' command adds all the files searched to the buffer list.
  920. " This also modifies the MRU list, even though the user didn't edit the
  921. " files. Use the following autocmds to prevent this.
  922. autocmd QuickFixCmdPre *vimgrep* let s:mru_list_locked = 1
  923. autocmd QuickFixCmdPost *vimgrep* let s:mru_list_locked = 0
  924. " Command to open the MRU window
  925. command! -nargs=? -complete=customlist,s:MRU_Complete MRU
  926. \ call s:MRU_Cmd(<q-args>)
  927. command! -nargs=? -complete=customlist,s:MRU_Complete Mru
  928. \ call s:MRU_Cmd(<q-args>)
  929. " }}}
  930. " restore 'cpo'
  931. let &cpo = s:cpo_save
  932. unlet s:cpo_save
  933. " vim:set foldenable foldmethod=marker: