/src/buffer.c

https://bitbucket.org/ultra_iter/vim-qt · C · 5722 lines · 4589 code · 367 blank · 766 comment · 1200 complexity · 8a2df500a792f7f65efd467f4a83acac MD5 · raw file

  1. /* vi:set ts=8 sts=4 sw=4:
  2. *
  3. * VIM - Vi IMproved by Bram Moolenaar
  4. *
  5. * Do ":help uganda" in Vim to read copying and usage conditions.
  6. * Do ":help credits" in Vim to see a list of people who contributed.
  7. * See README.txt for an overview of the Vim source code.
  8. */
  9. /*
  10. * buffer.c: functions for dealing with the buffer structure
  11. */
  12. /*
  13. * The buffer list is a double linked list of all buffers.
  14. * Each buffer can be in one of these states:
  15. * never loaded: BF_NEVERLOADED is set, only the file name is valid
  16. * not loaded: b_ml.ml_mfp == NULL, no memfile allocated
  17. * hidden: b_nwindows == 0, loaded but not displayed in a window
  18. * normal: loaded and displayed in a window
  19. *
  20. * Instead of storing file names all over the place, each file name is
  21. * stored in the buffer list. It can be referenced by a number.
  22. *
  23. * The current implementation remembers all file names ever used.
  24. */
  25. #include "vim.h"
  26. #if defined(FEAT_CMDL_COMPL) || defined(FEAT_LISTCMDS) || defined(FEAT_EVAL) || defined(FEAT_PERL)
  27. static char_u *buflist_match __ARGS((regprog_T *prog, buf_T *buf));
  28. # define HAVE_BUFLIST_MATCH
  29. static char_u *fname_match __ARGS((regprog_T *prog, char_u *name));
  30. #endif
  31. static void buflist_setfpos __ARGS((buf_T *buf, win_T *win, linenr_T lnum, colnr_T col, int copy_options));
  32. static wininfo_T *find_wininfo __ARGS((buf_T *buf, int skip_diff_buffer));
  33. #ifdef UNIX
  34. static buf_T *buflist_findname_stat __ARGS((char_u *ffname, struct stat *st));
  35. static int otherfile_buf __ARGS((buf_T *buf, char_u *ffname, struct stat *stp));
  36. static int buf_same_ino __ARGS((buf_T *buf, struct stat *stp));
  37. #else
  38. static int otherfile_buf __ARGS((buf_T *buf, char_u *ffname));
  39. #endif
  40. #ifdef FEAT_TITLE
  41. static int ti_change __ARGS((char_u *str, char_u **last));
  42. #endif
  43. static int append_arg_number __ARGS((win_T *wp, char_u *buf, int buflen, int add_file));
  44. static void free_buffer __ARGS((buf_T *));
  45. static void free_buffer_stuff __ARGS((buf_T *buf, int free_options));
  46. static void clear_wininfo __ARGS((buf_T *buf));
  47. #ifdef UNIX
  48. # define dev_T dev_t
  49. #else
  50. # define dev_T unsigned
  51. #endif
  52. #if defined(FEAT_SIGNS)
  53. static void insert_sign __ARGS((buf_T *buf, signlist_T *prev, signlist_T *next, int id, linenr_T lnum, int typenr));
  54. static void buf_delete_signs __ARGS((buf_T *buf));
  55. #endif
  56. #if defined(FEAT_WINDOWS) && defined(FEAT_QUICKFIX)
  57. static char *msg_loclist = N_("[Location List]");
  58. static char *msg_qflist = N_("[Quickfix List]");
  59. #endif
  60. #ifdef FEAT_AUTOCMD
  61. static char *e_auabort = N_("E855: Autocommands caused command to abort");
  62. #endif
  63. /*
  64. * Open current buffer, that is: open the memfile and read the file into
  65. * memory.
  66. * Return FAIL for failure, OK otherwise.
  67. */
  68. int
  69. open_buffer(read_stdin, eap, flags)
  70. int read_stdin; /* read file from stdin */
  71. exarg_T *eap; /* for forced 'ff' and 'fenc' or NULL */
  72. int flags; /* extra flags for readfile() */
  73. {
  74. int retval = OK;
  75. #ifdef FEAT_AUTOCMD
  76. buf_T *old_curbuf;
  77. #endif
  78. /*
  79. * The 'readonly' flag is only set when BF_NEVERLOADED is being reset.
  80. * When re-entering the same buffer, it should not change, because the
  81. * user may have reset the flag by hand.
  82. */
  83. if (readonlymode && curbuf->b_ffname != NULL
  84. && (curbuf->b_flags & BF_NEVERLOADED))
  85. curbuf->b_p_ro = TRUE;
  86. if (ml_open(curbuf) == FAIL)
  87. {
  88. /*
  89. * There MUST be a memfile, otherwise we can't do anything
  90. * If we can't create one for the current buffer, take another buffer
  91. */
  92. close_buffer(NULL, curbuf, 0, FALSE);
  93. for (curbuf = firstbuf; curbuf != NULL; curbuf = curbuf->b_next)
  94. if (curbuf->b_ml.ml_mfp != NULL)
  95. break;
  96. /*
  97. * if there is no memfile at all, exit
  98. * This is OK, since there are no changes to lose.
  99. */
  100. if (curbuf == NULL)
  101. {
  102. EMSG(_("E82: Cannot allocate any buffer, exiting..."));
  103. getout(2);
  104. }
  105. EMSG(_("E83: Cannot allocate buffer, using other one..."));
  106. enter_buffer(curbuf);
  107. return FAIL;
  108. }
  109. #ifdef FEAT_AUTOCMD
  110. /* The autocommands in readfile() may change the buffer, but only AFTER
  111. * reading the file. */
  112. old_curbuf = curbuf;
  113. modified_was_set = FALSE;
  114. #endif
  115. /* mark cursor position as being invalid */
  116. curwin->w_valid = 0;
  117. if (curbuf->b_ffname != NULL
  118. #ifdef FEAT_NETBEANS_INTG
  119. && netbeansReadFile
  120. #endif
  121. )
  122. {
  123. #ifdef FEAT_NETBEANS_INTG
  124. int oldFire = netbeansFireChanges;
  125. netbeansFireChanges = 0;
  126. #endif
  127. retval = readfile(curbuf->b_ffname, curbuf->b_fname,
  128. (linenr_T)0, (linenr_T)0, (linenr_T)MAXLNUM, eap,
  129. flags | READ_NEW);
  130. #ifdef FEAT_NETBEANS_INTG
  131. netbeansFireChanges = oldFire;
  132. #endif
  133. /* Help buffer is filtered. */
  134. if (curbuf->b_help)
  135. fix_help_buffer();
  136. }
  137. else if (read_stdin)
  138. {
  139. int save_bin = curbuf->b_p_bin;
  140. linenr_T line_count;
  141. /*
  142. * First read the text in binary mode into the buffer.
  143. * Then read from that same buffer and append at the end. This makes
  144. * it possible to retry when 'fileformat' or 'fileencoding' was
  145. * guessed wrong.
  146. */
  147. curbuf->b_p_bin = TRUE;
  148. retval = readfile(NULL, NULL, (linenr_T)0,
  149. (linenr_T)0, (linenr_T)MAXLNUM, NULL,
  150. flags | (READ_NEW + READ_STDIN));
  151. curbuf->b_p_bin = save_bin;
  152. if (retval == OK)
  153. {
  154. line_count = curbuf->b_ml.ml_line_count;
  155. retval = readfile(NULL, NULL, (linenr_T)line_count,
  156. (linenr_T)0, (linenr_T)MAXLNUM, eap,
  157. flags | READ_BUFFER);
  158. if (retval == OK)
  159. {
  160. /* Delete the binary lines. */
  161. while (--line_count >= 0)
  162. ml_delete((linenr_T)1, FALSE);
  163. }
  164. else
  165. {
  166. /* Delete the converted lines. */
  167. while (curbuf->b_ml.ml_line_count > line_count)
  168. ml_delete(line_count, FALSE);
  169. }
  170. /* Put the cursor on the first line. */
  171. curwin->w_cursor.lnum = 1;
  172. curwin->w_cursor.col = 0;
  173. /* Set or reset 'modified' before executing autocommands, so that
  174. * it can be changed there. */
  175. if (!readonlymode && !bufempty())
  176. changed();
  177. else if (retval != FAIL)
  178. unchanged(curbuf, FALSE);
  179. #ifdef FEAT_AUTOCMD
  180. # ifdef FEAT_EVAL
  181. apply_autocmds_retval(EVENT_STDINREADPOST, NULL, NULL, FALSE,
  182. curbuf, &retval);
  183. # else
  184. apply_autocmds(EVENT_STDINREADPOST, NULL, NULL, FALSE, curbuf);
  185. # endif
  186. #endif
  187. }
  188. }
  189. /* if first time loading this buffer, init b_chartab[] */
  190. if (curbuf->b_flags & BF_NEVERLOADED)
  191. (void)buf_init_chartab(curbuf, FALSE);
  192. /*
  193. * Set/reset the Changed flag first, autocmds may change the buffer.
  194. * Apply the automatic commands, before processing the modelines.
  195. * So the modelines have priority over auto commands.
  196. */
  197. /* When reading stdin, the buffer contents always needs writing, so set
  198. * the changed flag. Unless in readonly mode: "ls | gview -".
  199. * When interrupted and 'cpoptions' contains 'i' set changed flag. */
  200. if ((got_int && vim_strchr(p_cpo, CPO_INTMOD) != NULL)
  201. #ifdef FEAT_AUTOCMD
  202. || modified_was_set /* ":set modified" used in autocmd */
  203. # ifdef FEAT_EVAL
  204. || (aborting() && vim_strchr(p_cpo, CPO_INTMOD) != NULL)
  205. # endif
  206. #endif
  207. )
  208. changed();
  209. else if (retval != FAIL && !read_stdin)
  210. unchanged(curbuf, FALSE);
  211. save_file_ff(curbuf); /* keep this fileformat */
  212. /* require "!" to overwrite the file, because it wasn't read completely */
  213. #ifdef FEAT_EVAL
  214. if (aborting())
  215. #else
  216. if (got_int)
  217. #endif
  218. curbuf->b_flags |= BF_READERR;
  219. #ifdef FEAT_FOLDING
  220. /* Need to update automatic folding. Do this before the autocommands,
  221. * they may use the fold info. */
  222. foldUpdateAll(curwin);
  223. #endif
  224. #ifdef FEAT_AUTOCMD
  225. /* need to set w_topline, unless some autocommand already did that. */
  226. if (!(curwin->w_valid & VALID_TOPLINE))
  227. {
  228. curwin->w_topline = 1;
  229. # ifdef FEAT_DIFF
  230. curwin->w_topfill = 0;
  231. # endif
  232. }
  233. # ifdef FEAT_EVAL
  234. apply_autocmds_retval(EVENT_BUFENTER, NULL, NULL, FALSE, curbuf, &retval);
  235. # else
  236. apply_autocmds(EVENT_BUFENTER, NULL, NULL, FALSE, curbuf);
  237. # endif
  238. #endif
  239. if (retval != FAIL)
  240. {
  241. #ifdef FEAT_AUTOCMD
  242. /*
  243. * The autocommands may have changed the current buffer. Apply the
  244. * modelines to the correct buffer, if it still exists and is loaded.
  245. */
  246. if (buf_valid(old_curbuf) && old_curbuf->b_ml.ml_mfp != NULL)
  247. {
  248. aco_save_T aco;
  249. /* Go to the buffer that was opened. */
  250. aucmd_prepbuf(&aco, old_curbuf);
  251. #endif
  252. do_modelines(0);
  253. curbuf->b_flags &= ~(BF_CHECK_RO | BF_NEVERLOADED);
  254. #ifdef FEAT_AUTOCMD
  255. # ifdef FEAT_EVAL
  256. apply_autocmds_retval(EVENT_BUFWINENTER, NULL, NULL, FALSE, curbuf,
  257. &retval);
  258. # else
  259. apply_autocmds(EVENT_BUFWINENTER, NULL, NULL, FALSE, curbuf);
  260. # endif
  261. /* restore curwin/curbuf and a few other things */
  262. aucmd_restbuf(&aco);
  263. }
  264. #endif
  265. }
  266. return retval;
  267. }
  268. /*
  269. * Return TRUE if "buf" points to a valid buffer (in the buffer list).
  270. */
  271. int
  272. buf_valid(buf)
  273. buf_T *buf;
  274. {
  275. buf_T *bp;
  276. for (bp = firstbuf; bp != NULL; bp = bp->b_next)
  277. if (bp == buf)
  278. return TRUE;
  279. return FALSE;
  280. }
  281. /*
  282. * Close the link to a buffer.
  283. * "action" is used when there is no longer a window for the buffer.
  284. * It can be:
  285. * 0 buffer becomes hidden
  286. * DOBUF_UNLOAD buffer is unloaded
  287. * DOBUF_DELETE buffer is unloaded and removed from buffer list
  288. * DOBUF_WIPE buffer is unloaded and really deleted
  289. * When doing all but the first one on the current buffer, the caller should
  290. * get a new buffer very soon!
  291. *
  292. * The 'bufhidden' option can force freeing and deleting.
  293. *
  294. * When "abort_if_last" is TRUE then do not close the buffer if autocommands
  295. * cause there to be only one window with this buffer. e.g. when ":quit" is
  296. * supposed to close the window but autocommands close all other windows.
  297. */
  298. void
  299. close_buffer(win, buf, action, abort_if_last)
  300. win_T *win; /* if not NULL, set b_last_cursor */
  301. buf_T *buf;
  302. int action;
  303. int abort_if_last UNUSED;
  304. {
  305. #ifdef FEAT_AUTOCMD
  306. int is_curbuf;
  307. int nwindows;
  308. #endif
  309. int unload_buf = (action != 0);
  310. int del_buf = (action == DOBUF_DEL || action == DOBUF_WIPE);
  311. int wipe_buf = (action == DOBUF_WIPE);
  312. #ifdef FEAT_QUICKFIX
  313. /*
  314. * Force unloading or deleting when 'bufhidden' says so.
  315. * The caller must take care of NOT deleting/freeing when 'bufhidden' is
  316. * "hide" (otherwise we could never free or delete a buffer).
  317. */
  318. if (buf->b_p_bh[0] == 'd') /* 'bufhidden' == "delete" */
  319. {
  320. del_buf = TRUE;
  321. unload_buf = TRUE;
  322. }
  323. else if (buf->b_p_bh[0] == 'w') /* 'bufhidden' == "wipe" */
  324. {
  325. del_buf = TRUE;
  326. unload_buf = TRUE;
  327. wipe_buf = TRUE;
  328. }
  329. else if (buf->b_p_bh[0] == 'u') /* 'bufhidden' == "unload" */
  330. unload_buf = TRUE;
  331. #endif
  332. if (win != NULL)
  333. {
  334. /* Set b_last_cursor when closing the last window for the buffer.
  335. * Remember the last cursor position and window options of the buffer.
  336. * This used to be only for the current window, but then options like
  337. * 'foldmethod' may be lost with a ":only" command. */
  338. if (buf->b_nwindows == 1)
  339. set_last_cursor(win);
  340. buflist_setfpos(buf, win,
  341. win->w_cursor.lnum == 1 ? 0 : win->w_cursor.lnum,
  342. win->w_cursor.col, TRUE);
  343. }
  344. #ifdef FEAT_AUTOCMD
  345. /* When the buffer is no longer in a window, trigger BufWinLeave */
  346. if (buf->b_nwindows == 1)
  347. {
  348. buf->b_closing = TRUE;
  349. apply_autocmds(EVENT_BUFWINLEAVE, buf->b_fname, buf->b_fname,
  350. FALSE, buf);
  351. if (!buf_valid(buf))
  352. {
  353. /* Autocommands deleted the buffer. */
  354. aucmd_abort:
  355. EMSG(_(e_auabort));
  356. return;
  357. }
  358. buf->b_closing = FALSE;
  359. if (abort_if_last && one_window())
  360. /* Autocommands made this the only window. */
  361. goto aucmd_abort;
  362. /* When the buffer becomes hidden, but is not unloaded, trigger
  363. * BufHidden */
  364. if (!unload_buf)
  365. {
  366. buf->b_closing = TRUE;
  367. apply_autocmds(EVENT_BUFHIDDEN, buf->b_fname, buf->b_fname,
  368. FALSE, buf);
  369. if (!buf_valid(buf))
  370. /* Autocommands deleted the buffer. */
  371. goto aucmd_abort;
  372. buf->b_closing = FALSE;
  373. if (abort_if_last && one_window())
  374. /* Autocommands made this the only window. */
  375. goto aucmd_abort;
  376. }
  377. # ifdef FEAT_EVAL
  378. if (aborting()) /* autocmds may abort script processing */
  379. return;
  380. # endif
  381. }
  382. nwindows = buf->b_nwindows;
  383. #endif
  384. /* decrease the link count from windows (unless not in any window) */
  385. if (buf->b_nwindows > 0)
  386. --buf->b_nwindows;
  387. /* Return when a window is displaying the buffer or when it's not
  388. * unloaded. */
  389. if (buf->b_nwindows > 0 || !unload_buf)
  390. return;
  391. /* Always remove the buffer when there is no file name. */
  392. if (buf->b_ffname == NULL)
  393. del_buf = TRUE;
  394. /*
  395. * Free all things allocated for this buffer.
  396. * Also calls the "BufDelete" autocommands when del_buf is TRUE.
  397. */
  398. #ifdef FEAT_AUTOCMD
  399. /* Remember if we are closing the current buffer. Restore the number of
  400. * windows, so that autocommands in buf_freeall() don't get confused. */
  401. is_curbuf = (buf == curbuf);
  402. buf->b_nwindows = nwindows;
  403. #endif
  404. buf_freeall(buf, (del_buf ? BFA_DEL : 0) + (wipe_buf ? BFA_WIPE : 0));
  405. if (
  406. #ifdef FEAT_WINDOWS
  407. win_valid(win) &&
  408. #else
  409. win != NULL &&
  410. #endif
  411. win->w_buffer == buf)
  412. win->w_buffer = NULL; /* make sure we don't use the buffer now */
  413. #ifdef FEAT_AUTOCMD
  414. /* Autocommands may have deleted the buffer. */
  415. if (!buf_valid(buf))
  416. return;
  417. # ifdef FEAT_EVAL
  418. if (aborting()) /* autocmds may abort script processing */
  419. return;
  420. # endif
  421. /* Autocommands may have opened or closed windows for this buffer.
  422. * Decrement the count for the close we do here. */
  423. if (buf->b_nwindows > 0)
  424. --buf->b_nwindows;
  425. /*
  426. * It's possible that autocommands change curbuf to the one being deleted.
  427. * This might cause the previous curbuf to be deleted unexpectedly. But
  428. * in some cases it's OK to delete the curbuf, because a new one is
  429. * obtained anyway. Therefore only return if curbuf changed to the
  430. * deleted buffer.
  431. */
  432. if (buf == curbuf && !is_curbuf)
  433. return;
  434. #endif
  435. /* Change directories when the 'acd' option is set. */
  436. DO_AUTOCHDIR
  437. /*
  438. * Remove the buffer from the list.
  439. */
  440. if (wipe_buf)
  441. {
  442. #ifdef FEAT_SUN_WORKSHOP
  443. if (usingSunWorkShop)
  444. workshop_file_closed_lineno((char *)buf->b_ffname,
  445. (int)buf->b_last_cursor.lnum);
  446. #endif
  447. vim_free(buf->b_ffname);
  448. vim_free(buf->b_sfname);
  449. if (buf->b_prev == NULL)
  450. firstbuf = buf->b_next;
  451. else
  452. buf->b_prev->b_next = buf->b_next;
  453. if (buf->b_next == NULL)
  454. lastbuf = buf->b_prev;
  455. else
  456. buf->b_next->b_prev = buf->b_prev;
  457. free_buffer(buf);
  458. }
  459. else
  460. {
  461. if (del_buf)
  462. {
  463. /* Free all internal variables and reset option values, to make
  464. * ":bdel" compatible with Vim 5.7. */
  465. free_buffer_stuff(buf, TRUE);
  466. /* Make it look like a new buffer. */
  467. buf->b_flags = BF_CHECK_RO | BF_NEVERLOADED;
  468. /* Init the options when loaded again. */
  469. buf->b_p_initialized = FALSE;
  470. }
  471. buf_clear_file(buf);
  472. if (del_buf)
  473. buf->b_p_bl = FALSE;
  474. }
  475. }
  476. /*
  477. * Make buffer not contain a file.
  478. */
  479. void
  480. buf_clear_file(buf)
  481. buf_T *buf;
  482. {
  483. buf->b_ml.ml_line_count = 1;
  484. unchanged(buf, TRUE);
  485. #ifndef SHORT_FNAME
  486. buf->b_shortname = FALSE;
  487. #endif
  488. buf->b_p_eol = TRUE;
  489. buf->b_start_eol = TRUE;
  490. #ifdef FEAT_MBYTE
  491. buf->b_p_bomb = FALSE;
  492. buf->b_start_bomb = FALSE;
  493. #endif
  494. buf->b_ml.ml_mfp = NULL;
  495. buf->b_ml.ml_flags = ML_EMPTY; /* empty buffer */
  496. #ifdef FEAT_NETBEANS_INTG
  497. netbeans_deleted_all_lines(buf);
  498. #endif
  499. }
  500. /*
  501. * buf_freeall() - free all things allocated for a buffer that are related to
  502. * the file. flags:
  503. * BFA_DEL buffer is going to be deleted
  504. * BFA_WIPE buffer is going to be wiped out
  505. * BFA_KEEP_UNDO do not free undo information
  506. */
  507. void
  508. buf_freeall(buf, flags)
  509. buf_T *buf;
  510. int flags;
  511. {
  512. #ifdef FEAT_AUTOCMD
  513. int is_curbuf = (buf == curbuf);
  514. buf->b_closing = TRUE;
  515. apply_autocmds(EVENT_BUFUNLOAD, buf->b_fname, buf->b_fname, FALSE, buf);
  516. if (!buf_valid(buf)) /* autocommands may delete the buffer */
  517. return;
  518. if ((flags & BFA_DEL) && buf->b_p_bl)
  519. {
  520. apply_autocmds(EVENT_BUFDELETE, buf->b_fname, buf->b_fname, FALSE, buf);
  521. if (!buf_valid(buf)) /* autocommands may delete the buffer */
  522. return;
  523. }
  524. if (flags & BFA_WIPE)
  525. {
  526. apply_autocmds(EVENT_BUFWIPEOUT, buf->b_fname, buf->b_fname,
  527. FALSE, buf);
  528. if (!buf_valid(buf)) /* autocommands may delete the buffer */
  529. return;
  530. }
  531. buf->b_closing = FALSE;
  532. # ifdef FEAT_EVAL
  533. if (aborting()) /* autocmds may abort script processing */
  534. return;
  535. # endif
  536. /*
  537. * It's possible that autocommands change curbuf to the one being deleted.
  538. * This might cause curbuf to be deleted unexpectedly. But in some cases
  539. * it's OK to delete the curbuf, because a new one is obtained anyway.
  540. * Therefore only return if curbuf changed to the deleted buffer.
  541. */
  542. if (buf == curbuf && !is_curbuf)
  543. return;
  544. #endif
  545. #ifdef FEAT_DIFF
  546. diff_buf_delete(buf); /* Can't use 'diff' for unloaded buffer. */
  547. #endif
  548. #ifdef FEAT_SYN_HL
  549. /* Remove any ownsyntax, unless exiting. */
  550. if (firstwin != NULL && curwin->w_buffer == buf)
  551. reset_synblock(curwin);
  552. #endif
  553. #ifdef FEAT_FOLDING
  554. /* No folds in an empty buffer. */
  555. # ifdef FEAT_WINDOWS
  556. {
  557. win_T *win;
  558. tabpage_T *tp;
  559. FOR_ALL_TAB_WINDOWS(tp, win)
  560. if (win->w_buffer == buf)
  561. clearFolding(win);
  562. }
  563. # else
  564. if (curwin->w_buffer == buf)
  565. clearFolding(curwin);
  566. # endif
  567. #endif
  568. #ifdef FEAT_TCL
  569. tcl_buffer_free(buf);
  570. #endif
  571. ml_close(buf, TRUE); /* close and delete the memline/memfile */
  572. buf->b_ml.ml_line_count = 0; /* no lines in buffer */
  573. if ((flags & BFA_KEEP_UNDO) == 0)
  574. {
  575. u_blockfree(buf); /* free the memory allocated for undo */
  576. u_clearall(buf); /* reset all undo information */
  577. }
  578. #ifdef FEAT_SYN_HL
  579. syntax_clear(&buf->b_s); /* reset syntax info */
  580. #endif
  581. buf->b_flags &= ~BF_READERR; /* a read error is no longer relevant */
  582. }
  583. /*
  584. * Free a buffer structure and the things it contains related to the buffer
  585. * itself (not the file, that must have been done already).
  586. */
  587. static void
  588. free_buffer(buf)
  589. buf_T *buf;
  590. {
  591. free_buffer_stuff(buf, TRUE);
  592. #ifdef FEAT_LUA
  593. lua_buffer_free(buf);
  594. #endif
  595. #ifdef FEAT_MZSCHEME
  596. mzscheme_buffer_free(buf);
  597. #endif
  598. #ifdef FEAT_PERL
  599. perl_buf_free(buf);
  600. #endif
  601. #ifdef FEAT_PYTHON
  602. python_buffer_free(buf);
  603. #endif
  604. #ifdef FEAT_PYTHON3
  605. python3_buffer_free(buf);
  606. #endif
  607. #ifdef FEAT_RUBY
  608. ruby_buffer_free(buf);
  609. #endif
  610. #ifdef FEAT_AUTOCMD
  611. aubuflocal_remove(buf);
  612. #endif
  613. vim_free(buf);
  614. }
  615. /*
  616. * Free stuff in the buffer for ":bdel" and when wiping out the buffer.
  617. */
  618. static void
  619. free_buffer_stuff(buf, free_options)
  620. buf_T *buf;
  621. int free_options; /* free options as well */
  622. {
  623. if (free_options)
  624. {
  625. clear_wininfo(buf); /* including window-local options */
  626. free_buf_options(buf, TRUE);
  627. #ifdef FEAT_SPELL
  628. ga_clear(&buf->b_s.b_langp);
  629. #endif
  630. }
  631. #ifdef FEAT_EVAL
  632. vars_clear(&buf->b_vars.dv_hashtab); /* free all internal variables */
  633. hash_init(&buf->b_vars.dv_hashtab);
  634. #endif
  635. #ifdef FEAT_USR_CMDS
  636. uc_clear(&buf->b_ucmds); /* clear local user commands */
  637. #endif
  638. #ifdef FEAT_SIGNS
  639. buf_delete_signs(buf); /* delete any signs */
  640. #endif
  641. #ifdef FEAT_NETBEANS_INTG
  642. netbeans_file_killed(buf);
  643. #endif
  644. #ifdef FEAT_LOCALMAP
  645. map_clear_int(buf, MAP_ALL_MODES, TRUE, FALSE); /* clear local mappings */
  646. map_clear_int(buf, MAP_ALL_MODES, TRUE, TRUE); /* clear local abbrevs */
  647. #endif
  648. #ifdef FEAT_MBYTE
  649. vim_free(buf->b_start_fenc);
  650. buf->b_start_fenc = NULL;
  651. #endif
  652. }
  653. /*
  654. * Free the b_wininfo list for buffer "buf".
  655. */
  656. static void
  657. clear_wininfo(buf)
  658. buf_T *buf;
  659. {
  660. wininfo_T *wip;
  661. while (buf->b_wininfo != NULL)
  662. {
  663. wip = buf->b_wininfo;
  664. buf->b_wininfo = wip->wi_next;
  665. if (wip->wi_optset)
  666. {
  667. clear_winopt(&wip->wi_opt);
  668. #ifdef FEAT_FOLDING
  669. deleteFoldRecurse(&wip->wi_folds);
  670. #endif
  671. }
  672. vim_free(wip);
  673. }
  674. }
  675. #if defined(FEAT_LISTCMDS) || defined(PROTO)
  676. /*
  677. * Go to another buffer. Handles the result of the ATTENTION dialog.
  678. */
  679. void
  680. goto_buffer(eap, start, dir, count)
  681. exarg_T *eap;
  682. int start;
  683. int dir;
  684. int count;
  685. {
  686. # if defined(FEAT_WINDOWS) && defined(HAS_SWAP_EXISTS_ACTION)
  687. buf_T *old_curbuf = curbuf;
  688. swap_exists_action = SEA_DIALOG;
  689. # endif
  690. (void)do_buffer(*eap->cmd == 's' ? DOBUF_SPLIT : DOBUF_GOTO,
  691. start, dir, count, eap->forceit);
  692. # if defined(FEAT_WINDOWS) && defined(HAS_SWAP_EXISTS_ACTION)
  693. if (swap_exists_action == SEA_QUIT && *eap->cmd == 's')
  694. {
  695. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  696. cleanup_T cs;
  697. /* Reset the error/interrupt/exception state here so that
  698. * aborting() returns FALSE when closing a window. */
  699. enter_cleanup(&cs);
  700. # endif
  701. /* Quitting means closing the split window, nothing else. */
  702. win_close(curwin, TRUE);
  703. swap_exists_action = SEA_NONE;
  704. swap_exists_did_quit = TRUE;
  705. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  706. /* Restore the error/interrupt/exception state if not discarded by a
  707. * new aborting error, interrupt, or uncaught exception. */
  708. leave_cleanup(&cs);
  709. # endif
  710. }
  711. else
  712. handle_swap_exists(old_curbuf);
  713. # endif
  714. }
  715. #endif
  716. #if defined(HAS_SWAP_EXISTS_ACTION) || defined(PROTO)
  717. /*
  718. * Handle the situation of swap_exists_action being set.
  719. * It is allowed for "old_curbuf" to be NULL or invalid.
  720. */
  721. void
  722. handle_swap_exists(old_curbuf)
  723. buf_T *old_curbuf;
  724. {
  725. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  726. cleanup_T cs;
  727. # endif
  728. if (swap_exists_action == SEA_QUIT)
  729. {
  730. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  731. /* Reset the error/interrupt/exception state here so that
  732. * aborting() returns FALSE when closing a buffer. */
  733. enter_cleanup(&cs);
  734. # endif
  735. /* User selected Quit at ATTENTION prompt. Go back to previous
  736. * buffer. If that buffer is gone or the same as the current one,
  737. * open a new, empty buffer. */
  738. swap_exists_action = SEA_NONE; /* don't want it again */
  739. swap_exists_did_quit = TRUE;
  740. close_buffer(curwin, curbuf, DOBUF_UNLOAD, FALSE);
  741. if (!buf_valid(old_curbuf) || old_curbuf == curbuf)
  742. old_curbuf = buflist_new(NULL, NULL, 1L, BLN_CURBUF | BLN_LISTED);
  743. if (old_curbuf != NULL)
  744. enter_buffer(old_curbuf);
  745. /* If "old_curbuf" is NULL we are in big trouble here... */
  746. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  747. /* Restore the error/interrupt/exception state if not discarded by a
  748. * new aborting error, interrupt, or uncaught exception. */
  749. leave_cleanup(&cs);
  750. # endif
  751. }
  752. else if (swap_exists_action == SEA_RECOVER)
  753. {
  754. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  755. /* Reset the error/interrupt/exception state here so that
  756. * aborting() returns FALSE when closing a buffer. */
  757. enter_cleanup(&cs);
  758. # endif
  759. /* User selected Recover at ATTENTION prompt. */
  760. msg_scroll = TRUE;
  761. ml_recover();
  762. MSG_PUTS("\n"); /* don't overwrite the last message */
  763. cmdline_row = msg_row;
  764. do_modelines(0);
  765. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  766. /* Restore the error/interrupt/exception state if not discarded by a
  767. * new aborting error, interrupt, or uncaught exception. */
  768. leave_cleanup(&cs);
  769. # endif
  770. }
  771. swap_exists_action = SEA_NONE;
  772. }
  773. #endif
  774. #if defined(FEAT_LISTCMDS) || defined(PROTO)
  775. /*
  776. * do_bufdel() - delete or unload buffer(s)
  777. *
  778. * addr_count == 0: ":bdel" - delete current buffer
  779. * addr_count == 1: ":N bdel" or ":bdel N [N ..]" - first delete
  780. * buffer "end_bnr", then any other arguments.
  781. * addr_count == 2: ":N,N bdel" - delete buffers in range
  782. *
  783. * command can be DOBUF_UNLOAD (":bunload"), DOBUF_WIPE (":bwipeout") or
  784. * DOBUF_DEL (":bdel")
  785. *
  786. * Returns error message or NULL
  787. */
  788. char_u *
  789. do_bufdel(command, arg, addr_count, start_bnr, end_bnr, forceit)
  790. int command;
  791. char_u *arg; /* pointer to extra arguments */
  792. int addr_count;
  793. int start_bnr; /* first buffer number in a range */
  794. int end_bnr; /* buffer nr or last buffer nr in a range */
  795. int forceit;
  796. {
  797. int do_current = 0; /* delete current buffer? */
  798. int deleted = 0; /* number of buffers deleted */
  799. char_u *errormsg = NULL; /* return value */
  800. int bnr; /* buffer number */
  801. char_u *p;
  802. if (addr_count == 0)
  803. {
  804. (void)do_buffer(command, DOBUF_CURRENT, FORWARD, 0, forceit);
  805. }
  806. else
  807. {
  808. if (addr_count == 2)
  809. {
  810. if (*arg) /* both range and argument is not allowed */
  811. return (char_u *)_(e_trailing);
  812. bnr = start_bnr;
  813. }
  814. else /* addr_count == 1 */
  815. bnr = end_bnr;
  816. for ( ;!got_int; ui_breakcheck())
  817. {
  818. /*
  819. * delete the current buffer last, otherwise when the
  820. * current buffer is deleted, the next buffer becomes
  821. * the current one and will be loaded, which may then
  822. * also be deleted, etc.
  823. */
  824. if (bnr == curbuf->b_fnum)
  825. do_current = bnr;
  826. else if (do_buffer(command, DOBUF_FIRST, FORWARD, (int)bnr,
  827. forceit) == OK)
  828. ++deleted;
  829. /*
  830. * find next buffer number to delete/unload
  831. */
  832. if (addr_count == 2)
  833. {
  834. if (++bnr > end_bnr)
  835. break;
  836. }
  837. else /* addr_count == 1 */
  838. {
  839. arg = skipwhite(arg);
  840. if (*arg == NUL)
  841. break;
  842. if (!VIM_ISDIGIT(*arg))
  843. {
  844. p = skiptowhite_esc(arg);
  845. bnr = buflist_findpat(arg, p, command == DOBUF_WIPE, FALSE);
  846. if (bnr < 0) /* failed */
  847. break;
  848. arg = p;
  849. }
  850. else
  851. bnr = getdigits(&arg);
  852. }
  853. }
  854. if (!got_int && do_current && do_buffer(command, DOBUF_FIRST,
  855. FORWARD, do_current, forceit) == OK)
  856. ++deleted;
  857. if (deleted == 0)
  858. {
  859. if (command == DOBUF_UNLOAD)
  860. STRCPY(IObuff, _("E515: No buffers were unloaded"));
  861. else if (command == DOBUF_DEL)
  862. STRCPY(IObuff, _("E516: No buffers were deleted"));
  863. else
  864. STRCPY(IObuff, _("E517: No buffers were wiped out"));
  865. errormsg = IObuff;
  866. }
  867. else if (deleted >= p_report)
  868. {
  869. if (command == DOBUF_UNLOAD)
  870. {
  871. if (deleted == 1)
  872. MSG(_("1 buffer unloaded"));
  873. else
  874. smsg((char_u *)_("%d buffers unloaded"), deleted);
  875. }
  876. else if (command == DOBUF_DEL)
  877. {
  878. if (deleted == 1)
  879. MSG(_("1 buffer deleted"));
  880. else
  881. smsg((char_u *)_("%d buffers deleted"), deleted);
  882. }
  883. else
  884. {
  885. if (deleted == 1)
  886. MSG(_("1 buffer wiped out"));
  887. else
  888. smsg((char_u *)_("%d buffers wiped out"), deleted);
  889. }
  890. }
  891. }
  892. return errormsg;
  893. }
  894. /*
  895. * Implementation of the commands for the buffer list.
  896. *
  897. * action == DOBUF_GOTO go to specified buffer
  898. * action == DOBUF_SPLIT split window and go to specified buffer
  899. * action == DOBUF_UNLOAD unload specified buffer(s)
  900. * action == DOBUF_DEL delete specified buffer(s) from buffer list
  901. * action == DOBUF_WIPE delete specified buffer(s) really
  902. *
  903. * start == DOBUF_CURRENT go to "count" buffer from current buffer
  904. * start == DOBUF_FIRST go to "count" buffer from first buffer
  905. * start == DOBUF_LAST go to "count" buffer from last buffer
  906. * start == DOBUF_MOD go to "count" modified buffer from current buffer
  907. *
  908. * Return FAIL or OK.
  909. */
  910. int
  911. do_buffer(action, start, dir, count, forceit)
  912. int action;
  913. int start;
  914. int dir; /* FORWARD or BACKWARD */
  915. int count; /* buffer number or number of buffers */
  916. int forceit; /* TRUE for :...! */
  917. {
  918. buf_T *buf;
  919. buf_T *bp;
  920. int unload = (action == DOBUF_UNLOAD || action == DOBUF_DEL
  921. || action == DOBUF_WIPE);
  922. switch (start)
  923. {
  924. case DOBUF_FIRST: buf = firstbuf; break;
  925. case DOBUF_LAST: buf = lastbuf; break;
  926. default: buf = curbuf; break;
  927. }
  928. if (start == DOBUF_MOD) /* find next modified buffer */
  929. {
  930. while (count-- > 0)
  931. {
  932. do
  933. {
  934. buf = buf->b_next;
  935. if (buf == NULL)
  936. buf = firstbuf;
  937. }
  938. while (buf != curbuf && !bufIsChanged(buf));
  939. }
  940. if (!bufIsChanged(buf))
  941. {
  942. EMSG(_("E84: No modified buffer found"));
  943. return FAIL;
  944. }
  945. }
  946. else if (start == DOBUF_FIRST && count) /* find specified buffer number */
  947. {
  948. while (buf != NULL && buf->b_fnum != count)
  949. buf = buf->b_next;
  950. }
  951. else
  952. {
  953. bp = NULL;
  954. while (count > 0 || (!unload && !buf->b_p_bl && bp != buf))
  955. {
  956. /* remember the buffer where we start, we come back there when all
  957. * buffers are unlisted. */
  958. if (bp == NULL)
  959. bp = buf;
  960. if (dir == FORWARD)
  961. {
  962. buf = buf->b_next;
  963. if (buf == NULL)
  964. buf = firstbuf;
  965. }
  966. else
  967. {
  968. buf = buf->b_prev;
  969. if (buf == NULL)
  970. buf = lastbuf;
  971. }
  972. /* don't count unlisted buffers */
  973. if (unload || buf->b_p_bl)
  974. {
  975. --count;
  976. bp = NULL; /* use this buffer as new starting point */
  977. }
  978. if (bp == buf)
  979. {
  980. /* back where we started, didn't find anything. */
  981. EMSG(_("E85: There is no listed buffer"));
  982. return FAIL;
  983. }
  984. }
  985. }
  986. if (buf == NULL) /* could not find it */
  987. {
  988. if (start == DOBUF_FIRST)
  989. {
  990. /* don't warn when deleting */
  991. if (!unload)
  992. EMSGN(_("E86: Buffer %ld does not exist"), count);
  993. }
  994. else if (dir == FORWARD)
  995. EMSG(_("E87: Cannot go beyond last buffer"));
  996. else
  997. EMSG(_("E88: Cannot go before first buffer"));
  998. return FAIL;
  999. }
  1000. #ifdef FEAT_GUI
  1001. need_mouse_correct = TRUE;
  1002. #endif
  1003. #ifdef FEAT_LISTCMDS
  1004. /*
  1005. * delete buffer buf from memory and/or the list
  1006. */
  1007. if (unload)
  1008. {
  1009. int forward;
  1010. int retval;
  1011. /* When unloading or deleting a buffer that's already unloaded and
  1012. * unlisted: fail silently. */
  1013. if (action != DOBUF_WIPE && buf->b_ml.ml_mfp == NULL && !buf->b_p_bl)
  1014. return FAIL;
  1015. if (!forceit && bufIsChanged(buf))
  1016. {
  1017. #if defined(FEAT_GUI_DIALOG) || defined(FEAT_CON_DIALOG)
  1018. if ((p_confirm || cmdmod.confirm) && p_write)
  1019. {
  1020. dialog_changed(buf, FALSE);
  1021. # ifdef FEAT_AUTOCMD
  1022. if (!buf_valid(buf))
  1023. /* Autocommand deleted buffer, oops! It's not changed
  1024. * now. */
  1025. return FAIL;
  1026. # endif
  1027. /* If it's still changed fail silently, the dialog already
  1028. * mentioned why it fails. */
  1029. if (bufIsChanged(buf))
  1030. return FAIL;
  1031. }
  1032. else
  1033. #endif
  1034. {
  1035. EMSGN(_("E89: No write since last change for buffer %ld (add ! to override)"),
  1036. buf->b_fnum);
  1037. return FAIL;
  1038. }
  1039. }
  1040. /*
  1041. * If deleting the last (listed) buffer, make it empty.
  1042. * The last (listed) buffer cannot be unloaded.
  1043. */
  1044. for (bp = firstbuf; bp != NULL; bp = bp->b_next)
  1045. if (bp->b_p_bl && bp != buf)
  1046. break;
  1047. if (bp == NULL && buf == curbuf)
  1048. {
  1049. if (action == DOBUF_UNLOAD)
  1050. {
  1051. EMSG(_("E90: Cannot unload last buffer"));
  1052. return FAIL;
  1053. }
  1054. /* Close any other windows on this buffer, then make it empty. */
  1055. #ifdef FEAT_WINDOWS
  1056. close_windows(buf, TRUE);
  1057. #endif
  1058. setpcmark();
  1059. retval = do_ecmd(0, NULL, NULL, NULL, ECMD_ONE,
  1060. forceit ? ECMD_FORCEIT : 0, curwin);
  1061. /*
  1062. * do_ecmd() may create a new buffer, then we have to delete
  1063. * the old one. But do_ecmd() may have done that already, check
  1064. * if the buffer still exists.
  1065. */
  1066. if (buf != curbuf && buf_valid(buf) && buf->b_nwindows == 0)
  1067. close_buffer(NULL, buf, action, FALSE);
  1068. return retval;
  1069. }
  1070. #ifdef FEAT_WINDOWS
  1071. /*
  1072. * If the deleted buffer is the current one, close the current window
  1073. * (unless it's the only window). Repeat this so long as we end up in
  1074. * a window with this buffer.
  1075. */
  1076. while (buf == curbuf
  1077. # ifdef FEAT_AUTOCMD
  1078. && !(curwin->w_closing || curwin->w_buffer->b_closing)
  1079. # endif
  1080. && (firstwin != lastwin || first_tabpage->tp_next != NULL))
  1081. win_close(curwin, FALSE);
  1082. #endif
  1083. /*
  1084. * If the buffer to be deleted is not the current one, delete it here.
  1085. */
  1086. if (buf != curbuf)
  1087. {
  1088. #ifdef FEAT_WINDOWS
  1089. close_windows(buf, FALSE);
  1090. #endif
  1091. if (buf != curbuf && buf_valid(buf) && buf->b_nwindows <= 0)
  1092. close_buffer(NULL, buf, action, FALSE);
  1093. return OK;
  1094. }
  1095. /*
  1096. * Deleting the current buffer: Need to find another buffer to go to.
  1097. * There must be another, otherwise it would have been handled above.
  1098. * First use au_new_curbuf, if it is valid.
  1099. * Then prefer the buffer we most recently visited.
  1100. * Else try to find one that is loaded, after the current buffer,
  1101. * then before the current buffer.
  1102. * Finally use any buffer.
  1103. */
  1104. buf = NULL; /* selected buffer */
  1105. bp = NULL; /* used when no loaded buffer found */
  1106. #ifdef FEAT_AUTOCMD
  1107. if (au_new_curbuf != NULL && buf_valid(au_new_curbuf))
  1108. buf = au_new_curbuf;
  1109. # ifdef FEAT_JUMPLIST
  1110. else
  1111. # endif
  1112. #endif
  1113. #ifdef FEAT_JUMPLIST
  1114. if (curwin->w_jumplistlen > 0)
  1115. {
  1116. int jumpidx;
  1117. jumpidx = curwin->w_jumplistidx - 1;
  1118. if (jumpidx < 0)
  1119. jumpidx = curwin->w_jumplistlen - 1;
  1120. forward = jumpidx;
  1121. while (jumpidx != curwin->w_jumplistidx)
  1122. {
  1123. buf = buflist_findnr(curwin->w_jumplist[jumpidx].fmark.fnum);
  1124. if (buf != NULL)
  1125. {
  1126. if (buf == curbuf || !buf->b_p_bl)
  1127. buf = NULL; /* skip current and unlisted bufs */
  1128. else if (buf->b_ml.ml_mfp == NULL)
  1129. {
  1130. /* skip unloaded buf, but may keep it for later */
  1131. if (bp == NULL)
  1132. bp = buf;
  1133. buf = NULL;
  1134. }
  1135. }
  1136. if (buf != NULL) /* found a valid buffer: stop searching */
  1137. break;
  1138. /* advance to older entry in jump list */
  1139. if (!jumpidx && curwin->w_jumplistidx == curwin->w_jumplistlen)
  1140. break;
  1141. if (--jumpidx < 0)
  1142. jumpidx = curwin->w_jumplistlen - 1;
  1143. if (jumpidx == forward) /* List exhausted for sure */
  1144. break;
  1145. }
  1146. }
  1147. #endif
  1148. if (buf == NULL) /* No previous buffer, Try 2'nd approach */
  1149. {
  1150. forward = TRUE;
  1151. buf = curbuf->b_next;
  1152. for (;;)
  1153. {
  1154. if (buf == NULL)
  1155. {
  1156. if (!forward) /* tried both directions */
  1157. break;
  1158. buf = curbuf->b_prev;
  1159. forward = FALSE;
  1160. continue;
  1161. }
  1162. /* in non-help buffer, try to skip help buffers, and vv */
  1163. if (buf->b_help == curbuf->b_help && buf->b_p_bl)
  1164. {
  1165. if (buf->b_ml.ml_mfp != NULL) /* found loaded buffer */
  1166. break;
  1167. if (bp == NULL) /* remember unloaded buf for later */
  1168. bp = buf;
  1169. }
  1170. if (forward)
  1171. buf = buf->b_next;
  1172. else
  1173. buf = buf->b_prev;
  1174. }
  1175. }
  1176. if (buf == NULL) /* No loaded buffer, use unloaded one */
  1177. buf = bp;
  1178. if (buf == NULL) /* No loaded buffer, find listed one */
  1179. {
  1180. for (buf = firstbuf; buf != NULL; buf = buf->b_next)
  1181. if (buf->b_p_bl && buf != curbuf)
  1182. break;
  1183. }
  1184. if (buf == NULL) /* Still no buffer, just take one */
  1185. {
  1186. if (curbuf->b_next != NULL)
  1187. buf = curbuf->b_next;
  1188. else
  1189. buf = curbuf->b_prev;
  1190. }
  1191. }
  1192. /*
  1193. * make buf current buffer
  1194. */
  1195. if (action == DOBUF_SPLIT) /* split window first */
  1196. {
  1197. # ifdef FEAT_WINDOWS
  1198. /* If 'switchbuf' contains "useopen": jump to first window containing
  1199. * "buf" if one exists */
  1200. if ((swb_flags & SWB_USEOPEN) && buf_jump_open_win(buf))
  1201. return OK;
  1202. /* If 'switchbuf' contains "usetab": jump to first window in any tab
  1203. * page containing "buf" if one exists */
  1204. if ((swb_flags & SWB_USETAB) && buf_jump_open_tab(buf))
  1205. return OK;
  1206. if (win_split(0, 0) == FAIL)
  1207. # endif
  1208. return FAIL;
  1209. }
  1210. #endif
  1211. /* go to current buffer - nothing to do */
  1212. if (buf == curbuf)
  1213. return OK;
  1214. /*
  1215. * Check if the current buffer may be abandoned.
  1216. */
  1217. if (action == DOBUF_GOTO && !can_abandon(curbuf, forceit))
  1218. {
  1219. #if defined(FEAT_GUI_DIALOG) || defined(FEAT_CON_DIALOG)
  1220. if ((p_confirm || cmdmod.confirm) && p_write)
  1221. {
  1222. dialog_changed(curbuf, FALSE);
  1223. # ifdef FEAT_AUTOCMD
  1224. if (!buf_valid(buf))
  1225. /* Autocommand deleted buffer, oops! */
  1226. return FAIL;
  1227. # endif
  1228. }
  1229. if (bufIsChanged(curbuf))
  1230. #endif
  1231. {
  1232. EMSG(_(e_nowrtmsg));
  1233. return FAIL;
  1234. }
  1235. }
  1236. /* Go to the other buffer. */
  1237. set_curbuf(buf, action);
  1238. #if defined(FEAT_LISTCMDS) \
  1239. && (defined(FEAT_SCROLLBIND) || defined(FEAT_CURSORBIND))
  1240. if (action == DOBUF_SPLIT)
  1241. {
  1242. RESET_BINDING(curwin); /* reset 'scrollbind' and 'cursorbind' */
  1243. }
  1244. #endif
  1245. #if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  1246. if (aborting()) /* autocmds may abort script processing */
  1247. return FAIL;
  1248. #endif
  1249. return OK;
  1250. }
  1251. #endif /* FEAT_LISTCMDS */
  1252. /*
  1253. * Set current buffer to "buf". Executes autocommands and closes current
  1254. * buffer. "action" tells how to close the current buffer:
  1255. * DOBUF_GOTO free or hide it
  1256. * DOBUF_SPLIT nothing
  1257. * DOBUF_UNLOAD unload it
  1258. * DOBUF_DEL delete it
  1259. * DOBUF_WIPE wipe it out
  1260. */
  1261. void
  1262. set_curbuf(buf, action)
  1263. buf_T *buf;
  1264. int action;
  1265. {
  1266. buf_T *prevbuf;
  1267. int unload = (action == DOBUF_UNLOAD || action == DOBUF_DEL
  1268. || action == DOBUF_WIPE);
  1269. setpcmark();
  1270. if (!cmdmod.keepalt)
  1271. curwin->w_alt_fnum = curbuf->b_fnum; /* remember alternate file */
  1272. buflist_altfpos(curwin); /* remember curpos */
  1273. #ifdef FEAT_VISUAL
  1274. /* Don't restart Select mode after switching to another buffer. */
  1275. VIsual_reselect = FALSE;
  1276. #endif
  1277. /* close_windows() or apply_autocmds() may change curbuf */
  1278. prevbuf = curbuf;
  1279. #ifdef FEAT_AUTOCMD
  1280. apply_autocmds(EVENT_BUFLEAVE, NULL, NULL, FALSE, curbuf);
  1281. # ifdef FEAT_EVAL
  1282. if (buf_valid(prevbuf) && !aborting())
  1283. # else
  1284. if (buf_valid(prevbuf))
  1285. # endif
  1286. #endif
  1287. {
  1288. #ifdef FEAT_SYN_HL
  1289. if (prevbuf == curwin->w_buffer)
  1290. reset_synblock(curwin);
  1291. #endif
  1292. #ifdef FEAT_WINDOWS
  1293. if (unload)
  1294. close_windows(prevbuf, FALSE);
  1295. #endif
  1296. #if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  1297. if (buf_valid(prevbuf) && !aborting())
  1298. #else
  1299. if (buf_valid(prevbuf))
  1300. #endif
  1301. {
  1302. if (prevbuf == curbuf)
  1303. u_sync(FALSE);
  1304. close_buffer(prevbuf == curwin->w_buffer ? curwin : NULL, prevbuf,
  1305. unload ? action : (action == DOBUF_GOTO
  1306. && !P_HID(prevbuf)
  1307. && !bufIsChanged(prevbuf)) ? DOBUF_UNLOAD : 0, FALSE);
  1308. }
  1309. }
  1310. #ifdef FEAT_AUTOCMD
  1311. /* An autocommand may have deleted "buf", already entered it (e.g., when
  1312. * it did ":bunload") or aborted the script processing! */
  1313. # ifdef FEAT_EVAL
  1314. if (buf_valid(buf) && buf != curbuf && !aborting())
  1315. # else
  1316. if (buf_valid(buf) && buf != curbuf)
  1317. # endif
  1318. #endif
  1319. enter_buffer(buf);
  1320. }
  1321. /*
  1322. * Enter a new current buffer.
  1323. * Old curbuf must have been abandoned already!
  1324. */
  1325. void
  1326. enter_buffer(buf)
  1327. buf_T *buf;
  1328. {
  1329. /* Copy buffer and window local option values. Not for a help buffer. */
  1330. buf_copy_options(buf, BCO_ENTER | BCO_NOHELP);
  1331. if (!buf->b_help)
  1332. get_winopts(buf);
  1333. #ifdef FEAT_FOLDING
  1334. else
  1335. /* Remove all folds in the window. */
  1336. clearFolding(curwin);
  1337. foldUpdateAll(curwin); /* update folds (later). */
  1338. #endif
  1339. /* Get the buffer in the current window. */
  1340. curwin->w_buffer = buf;
  1341. curbuf = buf;
  1342. ++curbuf->b_nwindows;
  1343. #ifdef FEAT_DIFF
  1344. if (curwin->w_p_diff)
  1345. diff_buf_add(curbuf);
  1346. #endif
  1347. #ifdef FEAT_SYN_HL
  1348. curwin->w_s = &(buf->b_s);
  1349. #endif
  1350. /* Cursor on first line by default. */
  1351. curwin->w_cursor.lnum = 1;
  1352. curwin->w_cursor.col = 0;
  1353. #ifdef FEAT_VIRTUALEDIT
  1354. curwin->w_cursor.coladd = 0;
  1355. #endif
  1356. curwin->w_set_curswant = TRUE;
  1357. #ifdef FEAT_AUTOCMD
  1358. curwin->w_topline_was_set = FALSE;
  1359. #endif
  1360. /* mark cursor position as being invalid */
  1361. curwin->w_valid = 0;
  1362. /* Make sure the buffer is loaded. */
  1363. if (curbuf->b_ml.ml_mfp == NULL) /* need to load the file */
  1364. {
  1365. #ifdef FEAT_AUTOCMD
  1366. /* If there is no filetype, allow for detecting one. Esp. useful for
  1367. * ":ball" used in a autocommand. If there already is a filetype we
  1368. * might prefer to keep it. */
  1369. if (*curbuf->b_p_ft == NUL)
  1370. did_filetype = FALSE;
  1371. #endif
  1372. open_buffer(FALSE, NULL, 0);
  1373. }
  1374. else
  1375. {
  1376. if (!msg_silent)
  1377. need_fileinfo = TRUE; /* display file info after redraw */
  1378. (void)buf_check_timestamp(curbuf, FALSE); /* check if file changed */
  1379. #ifdef FEAT_AUTOCMD
  1380. curwin->w_topline = 1;
  1381. # ifdef FEAT_DIFF
  1382. curwin->w_topfill = 0;
  1383. # endif
  1384. apply_autocmds(EVENT_BUFENTER, NULL, NULL, FALSE, curbuf);
  1385. apply_autocmds(EVENT_BUFWINENTER, NULL, NULL, FALSE, curbuf);
  1386. #endif
  1387. }
  1388. /* If autocommands did not change the cursor position, restore cursor lnum
  1389. * and possibly cursor col. */
  1390. if (curwin->w_cursor.lnum == 1 && inindent(0))
  1391. buflist_getfpos();
  1392. check_arg_idx(curwin); /* check for valid arg_idx */
  1393. #ifdef FEAT_TITLE
  1394. maketitle();
  1395. #endif
  1396. #ifdef FEAT_AUTOCMD
  1397. /* when autocmds didn't change it */
  1398. if (curwin->w_topline == 1 && !curwin->w_topline_was_set)
  1399. #endif
  1400. scroll_cursor_halfway(FALSE); /* redisplay at correct position */
  1401. #ifdef FEAT_NETBEANS_INTG
  1402. /* Send fileOpened event because we've changed buffers. */
  1403. netbeans_file_activated(curbuf);
  1404. #endif
  1405. /* Change directories when the 'acd' option is set. */
  1406. DO_AUTOCHDIR
  1407. #ifdef FEAT_KEYMAP
  1408. if (curbuf->b_kmap_state & KEYMAP_INIT)
  1409. (void)keymap_init();
  1410. #endif
  1411. #ifdef FEAT_SPELL
  1412. /* May need to set the spell language. Can only do this after the buffer
  1413. * has been properly setup. */
  1414. if (!curbuf->b_help && curwin->w_p_spell && *curwin->w_s->b_p_spl != NUL)
  1415. (void)did_set_spelllang(curwin);
  1416. #endif
  1417. redraw_later(NOT_VALID);
  1418. }
  1419. #if defined(FEAT_AUTOCHDIR) || defined(PROTO)
  1420. /*
  1421. * Change to the directory of the current buffer.
  1422. */
  1423. void
  1424. do_autochdir()
  1425. {
  1426. if (curbuf->b_ffname != NULL && vim_chdirfile(curbuf->b_ffname) == OK)
  1427. shorten_fnames(TRUE);
  1428. }
  1429. #endif
  1430. /*
  1431. * functions for dealing with the buffer list
  1432. */
  1433. /*
  1434. * Add a file name to the buffer list. Return a pointer to the buffer.
  1435. * If the same file name already exists return a pointer to that buffer.
  1436. * If it does not exist, or if fname == NULL, a new entry is created.
  1437. * If (flags & BLN_CURBUF) is TRUE, may use current buffer.
  1438. * If (flags & BLN_LISTED) is TRUE, add new buffer to buffer list.
  1439. * If (flags & BLN_DUMMY) is TRUE, don't count it as a real buffer.
  1440. * This is the ONLY way to create a new buffer.
  1441. */
  1442. static int top_file_num = 1; /* highest file number */
  1443. buf_T *
  1444. buflist_new(ffname, sfname, lnum, flags)
  1445. char_u *ffname; /* full path of fname or relative */
  1446. char_u *sfname; /* short fname or NULL */
  1447. linenr_T lnum; /* preferred cursor line */
  1448. int flags; /* BLN_ defines */
  1449. {
  1450. buf_T *buf;
  1451. #ifdef UNIX
  1452. struct stat st;
  1453. #endif
  1454. fname_expand(curbuf, &ffname, &sfname); /* will allocate ffname */
  1455. /*
  1456. * If file name already exists in the list, update the entry.
  1457. */
  1458. #ifdef UNIX
  1459. /* On Unix we can use inode numbers when the file exists. Works better
  1460. * for hard links. */
  1461. if (sfname == NULL || mch_stat((char *)sfname, &st) < 0)
  1462. st.st_dev = (dev_T)-1;
  1463. #endif
  1464. if (ffname != NULL && !(flags & BLN_DUMMY) && (buf =
  1465. #ifdef UNIX
  1466. buflist_findname_stat(ffname, &st)
  1467. #else
  1468. buflist_findname(ffname)
  1469. #endif
  1470. ) != NULL)
  1471. {
  1472. vim_free(ffname);
  1473. if (lnum != 0)
  1474. buflist_setfpos(buf, curwin, lnum, (colnr_T)0, FALSE);
  1475. /* copy the options now, if 'cpo' doesn't have 's' and not done
  1476. * already */
  1477. buf_copy_options(buf, 0);
  1478. if ((flags & BLN_LISTED) && !buf->b_p_bl)
  1479. {
  1480. buf->b_p_bl = TRUE;
  1481. #ifdef FEAT_AUTOCMD
  1482. if (!(flags & BLN_DUMMY))
  1483. apply_autocmds(EVENT_BUFADD, NULL, NULL, FALSE, buf);
  1484. #endif
  1485. }
  1486. return buf;
  1487. }
  1488. /*
  1489. * If the current buffer has no name and no contents, use the current
  1490. * buffer. Otherwise: Need to allocate a new buffer structure.
  1491. *
  1492. * This is the ONLY place where a new buffer structure is allocated!
  1493. * (A spell file buffer is allocated in spell.c, but that's not a normal
  1494. * buffer.)
  1495. */
  1496. buf = NULL;
  1497. if ((flags & BLN_CURBUF)
  1498. && curbuf != NULL
  1499. && curbuf->b_ffname == NULL
  1500. && curbuf->b_nwindows <= 1
  1501. && (curbuf->b_ml.ml_mfp == NULL || bufempty()))
  1502. {
  1503. buf = curbuf;
  1504. #ifdef FEAT_AUTOCMD
  1505. /* It's like this buffer is deleted. Watch out for autocommands that
  1506. * change curbuf! If that happens, allocate a new buffer anyway. */
  1507. if (curbuf->b_p_bl)
  1508. apply_autocmds(EVENT_BUFDELETE, NULL, NULL, FALSE, curbuf);
  1509. if (buf == curbuf)
  1510. apply_autocmds(EVENT_BUFWIPEOUT, NULL, NULL, FALSE, curbuf);
  1511. # ifdef FEAT_EVAL
  1512. if (aborting()) /* autocmds may abort script processing */
  1513. return NULL;
  1514. # endif
  1515. #endif
  1516. #ifdef FEAT_QUICKFIX
  1517. # ifdef FEAT_AUTOCMD
  1518. if (buf == curbuf)
  1519. # endif
  1520. {
  1521. /* Make sure 'bufhidden' and 'buftype' are empty */
  1522. clear_string_option(&buf->b_p_bh);
  1523. clear_string_option(&buf->b_p_bt);
  1524. }
  1525. #endif
  1526. }
  1527. if (buf != curbuf || curbuf == NULL)
  1528. {
  1529. buf = (buf_T *)alloc_clear((unsigned)sizeof(buf_T));
  1530. if (buf == NULL)
  1531. {
  1532. vim_free(ffname);
  1533. return NULL;
  1534. }
  1535. }
  1536. if (ffname != NULL)
  1537. {
  1538. buf->b_ffname = ffname;
  1539. buf->b_sfname = vim_strsave(sfname);
  1540. }
  1541. clear_wininfo(buf);
  1542. buf->b_wininfo = (wininfo_T *)alloc_clear((unsigned)sizeof(wininfo_T));
  1543. if ((ffname != NULL && (buf->b_ffname == NULL || buf->b_sfname == NULL))
  1544. || buf->b_wininfo == NULL)
  1545. {
  1546. vim_free(buf->b_ffname);
  1547. buf->b_ffname = NULL;
  1548. vim_free(buf->b_sfname);
  1549. buf->b_sfname = NULL;
  1550. if (buf != curbuf)
  1551. free_buffer(buf);
  1552. return NULL;
  1553. }
  1554. if (buf == curbuf)
  1555. {
  1556. /* free all things allocated for this buffer */
  1557. buf_freeall(buf, 0);
  1558. if (buf != curbuf) /* autocommands deleted the buffer! */
  1559. return NULL;
  1560. #if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  1561. if (aborting()) /* autocmds may abort script processing */
  1562. return NULL;
  1563. #endif
  1564. /* buf->b_nwindows = 0; why was this here? */
  1565. free_buffer_stuff(buf, FALSE); /* delete local variables et al. */
  1566. #ifdef FEAT_KEYMAP
  1567. /* need to reload lmaps and set b:keymap_name */
  1568. curbuf->b_kmap_state |= KEYMAP_INIT;
  1569. #endif
  1570. }
  1571. else
  1572. {
  1573. /*
  1574. * put new buffer at the end of the buffer list
  1575. */
  1576. buf->b_next = NULL;
  1577. if (firstbuf == NULL) /* buffer list is empty */
  1578. {
  1579. buf->b_prev = NULL;
  1580. firstbuf = buf;
  1581. }
  1582. else /* append new buffer at end of list */
  1583. {
  1584. lastbuf->b_next = buf;
  1585. buf->b_prev = lastbuf;
  1586. }
  1587. lastbuf = buf;
  1588. buf->b_fnum = top_file_num++;
  1589. if (top_file_num < 0) /* wrap around (may cause duplicates) */
  1590. {
  1591. EMSG(_("W14: Warning: List of file names overflow"));
  1592. if (emsg_silent == 0)
  1593. {
  1594. out_flush();
  1595. ui_delay(3000L, TRUE); /* make sure it is noticed */
  1596. }
  1597. top_file_num = 1;
  1598. }
  1599. /*
  1600. * Always copy the options from the current buffer.
  1601. */
  1602. buf_copy_options(buf, BCO_ALWAYS);
  1603. }
  1604. buf->b_wininfo->wi_fpos.lnum = lnum;
  1605. buf->b_wininfo->wi_win = curwin;
  1606. #ifdef FEAT_EVAL
  1607. init_var_dict(&buf->b_vars, &buf->b_bufvar); /* init b: variables */
  1608. #endif
  1609. #ifdef FEAT_SYN_HL
  1610. hash_init(&buf->b_s.b_keywtab);
  1611. hash_init(&buf->b_s.b_keywtab_ic);
  1612. #endif
  1613. buf->b_fname = buf->b_sfname;
  1614. #ifdef UNIX
  1615. if (st.st_dev == (dev_T)-1)
  1616. buf->b_dev_valid = FALSE;
  1617. else
  1618. {
  1619. buf->b_dev_valid = TRUE;
  1620. buf->b_dev = st.st_dev;
  1621. buf->b_ino = st.st_ino;
  1622. }
  1623. #endif
  1624. buf->b_u_synced = TRUE;
  1625. buf->b_flags = BF_CHECK_RO | BF_NEVERLOADED;
  1626. if (flags & BLN_DUMMY)
  1627. buf->b_flags |= BF_DUMMY;
  1628. buf_clear_file(buf);
  1629. clrallmarks(buf); /* clear marks */
  1630. fmarks_check_names(buf); /* check file marks for this file */
  1631. buf->b_p_bl = (flags & BLN_LISTED) ? TRUE : FALSE; /* init 'buflisted' */
  1632. #ifdef FEAT_AUTOCMD
  1633. if (!(flags & BLN_DUMMY))
  1634. {
  1635. apply_autocmds(EVENT_BUFNEW, NULL, NULL, FALSE, buf);
  1636. if (flags & BLN_LISTED)
  1637. apply_autocmds(EVENT_BUFADD, NULL, NULL, FALSE, buf);
  1638. # ifdef FEAT_EVAL
  1639. if (aborting()) /* autocmds may abort script processing */
  1640. return NULL;
  1641. # endif
  1642. }
  1643. #endif
  1644. return buf;
  1645. }
  1646. /*
  1647. * Free the memory for the options of a buffer.
  1648. * If "free_p_ff" is TRUE also free 'fileformat', 'buftype' and
  1649. * 'fileencoding'.
  1650. */
  1651. void
  1652. free_buf_options(buf, free_p_ff)
  1653. buf_T *buf;
  1654. int free_p_ff;
  1655. {
  1656. if (free_p_ff)
  1657. {
  1658. #ifdef FEAT_MBYTE
  1659. clear_string_option(&buf->b_p_fenc);
  1660. #endif
  1661. clear_string_option(&buf->b_p_ff);
  1662. #ifdef FEAT_QUICKFIX
  1663. clear_string_option(&buf->b_p_bh);
  1664. clear_string_option(&buf->b_p_bt);
  1665. #endif
  1666. }
  1667. #ifdef FEAT_FIND_ID
  1668. clear_string_option(&buf->b_p_def);
  1669. clear_string_option(&buf->b_p_inc);
  1670. # ifdef FEAT_EVAL
  1671. clear_string_option(&buf->b_p_inex);
  1672. # endif
  1673. #endif
  1674. #if defined(FEAT_CINDENT) && defined(FEAT_EVAL)
  1675. clear_string_option(&buf->b_p_inde);
  1676. clear_string_option(&buf->b_p_indk);
  1677. #endif
  1678. #if defined(FEAT_BEVAL) && defined(FEAT_EVAL)
  1679. clear_string_option(&buf->b_p_bexpr);
  1680. #endif
  1681. #if defined(FEAT_CRYPT)
  1682. clear_string_option(&buf->b_p_cm);
  1683. #endif
  1684. #if defined(FEAT_EVAL)
  1685. clear_string_option(&buf->b_p_fex);
  1686. #endif
  1687. #ifdef FEAT_CRYPT
  1688. clear_string_option(&buf->b_p_key);
  1689. #endif
  1690. clear_string_option(&buf->b_p_kp);
  1691. clear_string_option(&buf->b_p_mps);
  1692. clear_string_option(&buf->b_p_fo);
  1693. clear_string_option(&buf->b_p_flp);
  1694. clear_string_option(&buf->b_p_isk);
  1695. #ifdef FEAT_KEYMAP
  1696. clear_string_option(&buf->b_p_keymap);
  1697. ga_clear(&buf->b_kmap_ga);
  1698. #endif
  1699. #ifdef FEAT_COMMENTS
  1700. clear_string_option(&buf->b_p_com);
  1701. #endif
  1702. #ifdef FEAT_FOLDING
  1703. clear_string_option(&buf->b_p_cms);
  1704. #endif
  1705. clear_string_option(&buf->b_p_nf);
  1706. #ifdef FEAT_SYN_HL
  1707. clear_string_option(&buf->b_p_syn);
  1708. #endif
  1709. #ifdef FEAT_SPELL
  1710. clear_string_option(&buf->b_s.b_p_spc);
  1711. clear_string_option(&buf->b_s.b_p_spf);
  1712. vim_free(buf->b_s.b_cap_prog);
  1713. buf->b_s.b_cap_prog = NULL;
  1714. clear_string_option(&buf->b_s.b_p_spl);
  1715. #endif
  1716. #ifdef FEAT_SEARCHPATH
  1717. clear_string_option(&buf->b_p_sua);
  1718. #endif
  1719. #ifdef FEAT_AUTOCMD
  1720. clear_string_option(&buf->b_p_ft);
  1721. #endif
  1722. #ifdef FEAT_CINDENT
  1723. clear_string_option(&buf->b_p_cink);
  1724. clear_string_option(&buf->b_p_cino);
  1725. #endif
  1726. #if defined(FEAT_CINDENT) || defined(FEAT_SMARTINDENT)
  1727. clear_string_option(&buf->b_p_cinw);
  1728. #endif
  1729. #ifdef FEAT_INS_EXPAND
  1730. clear_string_option(&buf->b_p_cpt);
  1731. #endif
  1732. #ifdef FEAT_COMPL_FUNC
  1733. clear_string_option(&buf->b_p_cfu);
  1734. clear_string_option(&buf->b_p_ofu);
  1735. #endif
  1736. #ifdef FEAT_QUICKFIX
  1737. clear_string_option(&buf->b_p_gp);
  1738. clear_string_option(&buf->b_p_mp);
  1739. clear_string_option(&buf->b_p_efm);
  1740. #endif
  1741. clear_string_option(&buf->b_p_ep);
  1742. clear_string_option(&buf->b_p_path);
  1743. clear_string_option(&buf->b_p_tags);
  1744. #ifdef FEAT_INS_EXPAND
  1745. clear_string_option(&buf->b_p_dict);
  1746. clear_string_option(&buf->b_p_tsr);
  1747. #endif
  1748. #ifdef FEAT_TEXTOBJ
  1749. clear_string_option(&buf->b_p_qe);
  1750. #endif
  1751. buf->b_p_ar = -1;
  1752. }
  1753. /*
  1754. * get alternate file n
  1755. * set linenr to lnum or altfpos.lnum if lnum == 0
  1756. * also set cursor column to altfpos.col if 'startofline' is not set.
  1757. * if (options & GETF_SETMARK) call setpcmark()
  1758. * if (options & GETF_ALT) we are jumping to an alternate file.
  1759. * if (options & GETF_SWITCH) respect 'switchbuf' settings when jumping
  1760. *
  1761. * return FAIL for failure, OK for success
  1762. */
  1763. int
  1764. buflist_getfile(n, lnum, options, forceit)
  1765. int n;
  1766. linenr_T lnum;
  1767. int options;
  1768. int forceit;
  1769. {
  1770. buf_T *buf;
  1771. #ifdef FEAT_WINDOWS
  1772. win_T *wp = NULL;
  1773. #endif
  1774. pos_T *fpos;
  1775. colnr_T col;
  1776. buf = buflist_findnr(n);
  1777. if (buf == NULL)
  1778. {
  1779. if ((options & GETF_ALT) && n == 0)
  1780. EMSG(_(e_noalt));
  1781. else
  1782. EMSGN(_("E92: Buffer %ld not found"), n);
  1783. return FAIL;
  1784. }
  1785. /* if alternate file is the current buffer, nothing to do */
  1786. if (buf == curbuf)
  1787. return OK;
  1788. if (text_locked())
  1789. {
  1790. text_locked_msg();
  1791. return FAIL;
  1792. }
  1793. #ifdef FEAT_AUTOCMD
  1794. if (curbuf_locked())
  1795. return FAIL;
  1796. #endif
  1797. /* altfpos may be changed by getfile(), get it now */
  1798. if (lnum == 0)
  1799. {
  1800. fpos = buflist_findfpos(buf);
  1801. lnum = fpos->lnum;
  1802. col = fpos->col;
  1803. }
  1804. else
  1805. col = 0;
  1806. #ifdef FEAT_WINDOWS
  1807. if (options & GETF_SWITCH)
  1808. {
  1809. /* If 'switchbuf' contains "useopen": jump to first window containing
  1810. * "buf" if one exists */
  1811. if (swb_flags & SWB_USEOPEN)
  1812. wp = buf_jump_open_win(buf);
  1813. /* If 'switchbuf' contians "usetab": jump to first window in any tab
  1814. * page containing "buf" if one exists */
  1815. if (wp == NULL && (swb_flags & SWB_USETAB))
  1816. wp = buf_jump_open_tab(buf);
  1817. /* If 'switchbuf' contains "split" or "newtab" and the current buffer
  1818. * isn't empty: open new window */
  1819. if (wp == NULL && (swb_flags & (SWB_SPLIT | SWB_NEWTAB)) && !bufempty())
  1820. {
  1821. if (swb_flags & SWB_NEWTAB) /* Open in a new tab */
  1822. tabpage_new();
  1823. else if (win_split(0, 0) == FAIL) /* Open in a new window */
  1824. return FAIL;
  1825. RESET_BINDING(curwin);
  1826. }
  1827. }
  1828. #endif
  1829. ++RedrawingDisabled;
  1830. if (getfile(buf->b_fnum, NULL, NULL, (options & GETF_SETMARK),
  1831. lnum, forceit) <= 0)
  1832. {
  1833. --RedrawingDisabled;
  1834. /* cursor is at to BOL and w_cursor.lnum is checked due to getfile() */
  1835. if (!p_sol && col != 0)
  1836. {
  1837. curwin->w_cursor.col = col;
  1838. check_cursor_col();
  1839. #ifdef FEAT_VIRTUALEDIT
  1840. curwin->w_cursor.coladd = 0;
  1841. #endif
  1842. curwin->w_set_curswant = TRUE;
  1843. }
  1844. return OK;
  1845. }
  1846. --RedrawingDisabled;
  1847. return FAIL;
  1848. }
  1849. /*
  1850. * go to the last know line number for the current buffer
  1851. */
  1852. void
  1853. buflist_getfpos()
  1854. {
  1855. pos_T *fpos;
  1856. fpos = buflist_findfpos(curbuf);
  1857. curwin->w_cursor.lnum = fpos->lnum;
  1858. check_cursor_lnum();
  1859. if (p_sol)
  1860. curwin->w_cursor.col = 0;
  1861. else
  1862. {
  1863. curwin->w_cursor.col = fpos->col;
  1864. check_cursor_col();
  1865. #ifdef FEAT_VIRTUALEDIT
  1866. curwin->w_cursor.coladd = 0;
  1867. #endif
  1868. curwin->w_set_curswant = TRUE;
  1869. }
  1870. }
  1871. #if defined(FEAT_QUICKFIX) || defined(FEAT_EVAL) || defined(PROTO)
  1872. /*
  1873. * Find file in buffer list by name (it has to be for the current window).
  1874. * Returns NULL if not found.
  1875. */
  1876. buf_T *
  1877. buflist_findname_exp(fname)
  1878. char_u *fname;
  1879. {
  1880. char_u *ffname;
  1881. buf_T *buf = NULL;
  1882. /* First make the name into a full path name */
  1883. ffname = FullName_save(fname,
  1884. #ifdef UNIX
  1885. TRUE /* force expansion, get rid of symbolic links */
  1886. #else
  1887. FALSE
  1888. #endif
  1889. );
  1890. if (ffname != NULL)
  1891. {
  1892. buf = buflist_findname(ffname);
  1893. vim_free(ffname);
  1894. }
  1895. return buf;
  1896. }
  1897. #endif
  1898. /*
  1899. * Find file in buffer list by name (it has to be for the current window).
  1900. * "ffname" must have a full path.
  1901. * Skips dummy buffers.
  1902. * Returns NULL if not found.
  1903. */
  1904. buf_T *
  1905. buflist_findname(ffname)
  1906. char_u *ffname;
  1907. {
  1908. #ifdef UNIX
  1909. struct stat st;
  1910. if (mch_stat((char *)ffname, &st) < 0)
  1911. st.st_dev = (dev_T)-1;
  1912. return buflist_findname_stat(ffname, &st);
  1913. }
  1914. /*
  1915. * Same as buflist_findname(), but pass the stat structure to avoid getting it
  1916. * twice for the same file.
  1917. * Returns NULL if not found.
  1918. */
  1919. static buf_T *
  1920. buflist_findname_stat(ffname, stp)
  1921. char_u *ffname;
  1922. struct stat *stp;
  1923. {
  1924. #endif
  1925. buf_T *buf;
  1926. for (buf = firstbuf; buf != NULL; buf = buf->b_next)
  1927. if ((buf->b_flags & BF_DUMMY) == 0 && !otherfile_buf(buf, ffname
  1928. #ifdef UNIX
  1929. , stp
  1930. #endif
  1931. ))
  1932. return buf;
  1933. return NULL;
  1934. }
  1935. #if defined(FEAT_LISTCMDS) || defined(FEAT_EVAL) || defined(FEAT_PERL) || defined(PROTO)
  1936. /*
  1937. * Find file in buffer list by a regexp pattern.
  1938. * Return fnum of the found buffer.
  1939. * Return < 0 for error.
  1940. */
  1941. int
  1942. buflist_findpat(pattern, pattern_end, unlisted, diffmode)
  1943. char_u *pattern;
  1944. char_u *pattern_end; /* pointer to first char after pattern */
  1945. int unlisted; /* find unlisted buffers */
  1946. int diffmode UNUSED; /* find diff-mode buffers only */
  1947. {
  1948. buf_T *buf;
  1949. regprog_T *prog;
  1950. int match = -1;
  1951. int find_listed;
  1952. char_u *pat;
  1953. char_u *patend;
  1954. int attempt;
  1955. char_u *p;
  1956. int toggledollar;
  1957. if (pattern_end == pattern + 1 && (*pattern == '%' || *pattern == '#'))
  1958. {
  1959. if (*pattern == '%')
  1960. match = curbuf->b_fnum;
  1961. else
  1962. match = curwin->w_alt_fnum;
  1963. #ifdef FEAT_DIFF
  1964. if (diffmode && !diff_mode_buf(buflist_findnr(match)))
  1965. match = -1;
  1966. #endif
  1967. }
  1968. /*
  1969. * Try four ways of matching a listed buffer:
  1970. * attempt == 0: without '^' or '$' (at any position)
  1971. * attempt == 1: with '^' at start (only at position 0)
  1972. * attempt == 2: with '$' at end (only match at end)
  1973. * attempt == 3: with '^' at start and '$' at end (only full match)
  1974. * Repeat this for finding an unlisted buffer if there was no matching
  1975. * listed buffer.
  1976. */
  1977. else
  1978. {
  1979. pat = file_pat_to_reg_pat(pattern, pattern_end, NULL, FALSE);
  1980. if (pat == NULL)
  1981. return -1;
  1982. patend = pat + STRLEN(pat) - 1;
  1983. toggledollar = (patend > pat && *patend == '$');
  1984. /* First try finding a listed buffer. If not found and "unlisted"
  1985. * is TRUE, try finding an unlisted buffer. */
  1986. find_listed = TRUE;
  1987. for (;;)
  1988. {
  1989. for (attempt = 0; attempt <= 3; ++attempt)
  1990. {
  1991. /* may add '^' and '$' */
  1992. if (toggledollar)
  1993. *patend = (attempt < 2) ? NUL : '$'; /* add/remove '$' */
  1994. p = pat;
  1995. if (*p == '^' && !(attempt & 1)) /* add/remove '^' */
  1996. ++p;
  1997. prog = vim_regcomp(p, p_magic ? RE_MAGIC : 0);
  1998. if (prog == NULL)
  1999. {
  2000. vim_free(pat);
  2001. return -1;
  2002. }
  2003. for (buf = firstbuf; buf != NULL; buf = buf->b_next)
  2004. if (buf->b_p_bl == find_listed
  2005. #ifdef FEAT_DIFF
  2006. && (!diffmode || diff_mode_buf(buf))
  2007. #endif
  2008. && buflist_match(prog, buf) != NULL)
  2009. {
  2010. if (match >= 0) /* already found a match */
  2011. {
  2012. match = -2;
  2013. break;
  2014. }
  2015. match = buf->b_fnum; /* remember first match */
  2016. }
  2017. vim_free(prog);
  2018. if (match >= 0) /* found one match */
  2019. break;
  2020. }
  2021. /* Only search for unlisted buffers if there was no match with
  2022. * a listed buffer. */
  2023. if (!unlisted || !find_listed || match != -1)
  2024. break;
  2025. find_listed = FALSE;
  2026. }
  2027. vim_free(pat);
  2028. }
  2029. if (match == -2)
  2030. EMSG2(_("E93: More than one match for %s"), pattern);
  2031. else if (match < 0)
  2032. EMSG2(_("E94: No matching buffer for %s"), pattern);
  2033. return match;
  2034. }
  2035. #endif
  2036. #if defined(FEAT_CMDL_COMPL) || defined(PROTO)
  2037. /*
  2038. * Find all buffer names that match.
  2039. * For command line expansion of ":buf" and ":sbuf".
  2040. * Return OK if matches found, FAIL otherwise.
  2041. */
  2042. int
  2043. ExpandBufnames(pat, num_file, file, options)
  2044. char_u *pat;
  2045. int *num_file;
  2046. char_u ***file;
  2047. int options;
  2048. {
  2049. int count = 0;
  2050. buf_T *buf;
  2051. int round;
  2052. char_u *p;
  2053. int attempt;
  2054. regprog_T *prog;
  2055. char_u *patc;
  2056. *num_file = 0; /* return values in case of FAIL */
  2057. *file = NULL;
  2058. /* Make a copy of "pat" and change "^" to "\(^\|[\/]\)". */
  2059. if (*pat == '^')
  2060. {
  2061. patc = alloc((unsigned)STRLEN(pat) + 11);
  2062. if (patc == NULL)
  2063. return FAIL;
  2064. STRCPY(patc, "\\(^\\|[\\/]\\)");
  2065. STRCPY(patc + 11, pat + 1);
  2066. }
  2067. else
  2068. patc = pat;
  2069. /*
  2070. * attempt == 0: try match with '\<', match at start of word
  2071. * attempt == 1: try match without '\<', match anywhere
  2072. */
  2073. for (attempt = 0; attempt <= 1; ++attempt)
  2074. {
  2075. if (attempt > 0 && patc == pat)
  2076. break; /* there was no anchor, no need to try again */
  2077. prog = vim_regcomp(patc + attempt * 11, RE_MAGIC);
  2078. if (prog == NULL)
  2079. {
  2080. if (patc != pat)
  2081. vim_free(patc);
  2082. return FAIL;
  2083. }
  2084. /*
  2085. * round == 1: Count the matches.
  2086. * round == 2: Build the array to keep the matches.
  2087. */
  2088. for (round = 1; round <= 2; ++round)
  2089. {
  2090. count = 0;
  2091. for (buf = firstbuf; buf != NULL; buf = buf->b_next)
  2092. {
  2093. if (!buf->b_p_bl) /* skip unlisted buffers */
  2094. continue;
  2095. p = buflist_match(prog, buf);
  2096. if (p != NULL)
  2097. {
  2098. if (round == 1)
  2099. ++count;
  2100. else
  2101. {
  2102. if (options & WILD_HOME_REPLACE)
  2103. p = home_replace_save(buf, p);
  2104. else
  2105. p = vim_strsave(p);
  2106. (*file)[count++] = p;
  2107. }
  2108. }
  2109. }
  2110. if (count == 0) /* no match found, break here */
  2111. break;
  2112. if (round == 1)
  2113. {
  2114. *file = (char_u **)alloc((unsigned)(count * sizeof(char_u *)));
  2115. if (*file == NULL)
  2116. {
  2117. vim_free(prog);
  2118. if (patc != pat)
  2119. vim_free(patc);
  2120. return FAIL;
  2121. }
  2122. }
  2123. }
  2124. vim_free(prog);
  2125. if (count) /* match(es) found, break here */
  2126. break;
  2127. }
  2128. if (patc != pat)
  2129. vim_free(patc);
  2130. *num_file = count;
  2131. return (count == 0 ? FAIL : OK);
  2132. }
  2133. #endif /* FEAT_CMDL_COMPL */
  2134. #ifdef HAVE_BUFLIST_MATCH
  2135. /*
  2136. * Check for a match on the file name for buffer "buf" with regprog "prog".
  2137. */
  2138. static char_u *
  2139. buflist_match(prog, buf)
  2140. regprog_T *prog;
  2141. buf_T *buf;
  2142. {
  2143. char_u *match;
  2144. /* First try the short file name, then the long file name. */
  2145. match = fname_match(prog, buf->b_sfname);
  2146. if (match == NULL)
  2147. match = fname_match(prog, buf->b_ffname);
  2148. return match;
  2149. }
  2150. /*
  2151. * Try matching the regexp in "prog" with file name "name".
  2152. * Return "name" when there is a match, NULL when not.
  2153. */
  2154. static char_u *
  2155. fname_match(prog, name)
  2156. regprog_T *prog;
  2157. char_u *name;
  2158. {
  2159. char_u *match = NULL;
  2160. char_u *p;
  2161. regmatch_T regmatch;
  2162. if (name != NULL)
  2163. {
  2164. regmatch.regprog = prog;
  2165. #ifdef CASE_INSENSITIVE_FILENAME
  2166. regmatch.rm_ic = TRUE; /* Always ignore case */
  2167. #else
  2168. regmatch.rm_ic = FALSE; /* Never ignore case */
  2169. #endif
  2170. if (vim_regexec(&regmatch, name, (colnr_T)0))
  2171. match = name;
  2172. else
  2173. {
  2174. /* Replace $(HOME) with '~' and try matching again. */
  2175. p = home_replace_save(NULL, name);
  2176. if (p != NULL && vim_regexec(&regmatch, p, (colnr_T)0))
  2177. match = name;
  2178. vim_free(p);
  2179. }
  2180. }
  2181. return match;
  2182. }
  2183. #endif
  2184. /*
  2185. * find file in buffer list by number
  2186. */
  2187. buf_T *
  2188. buflist_findnr(nr)
  2189. int nr;
  2190. {
  2191. buf_T *buf;
  2192. if (nr == 0)
  2193. nr = curwin->w_alt_fnum;
  2194. for (buf = firstbuf; buf != NULL; buf = buf->b_next)
  2195. if (buf->b_fnum == nr)
  2196. return (buf);
  2197. return NULL;
  2198. }
  2199. /*
  2200. * Get name of file 'n' in the buffer list.
  2201. * When the file has no name an empty string is returned.
  2202. * home_replace() is used to shorten the file name (used for marks).
  2203. * Returns a pointer to allocated memory, of NULL when failed.
  2204. */
  2205. char_u *
  2206. buflist_nr2name(n, fullname, helptail)
  2207. int n;
  2208. int fullname;
  2209. int helptail; /* for help buffers return tail only */
  2210. {
  2211. buf_T *buf;
  2212. buf = buflist_findnr(n);
  2213. if (buf == NULL)
  2214. return NULL;
  2215. return home_replace_save(helptail ? buf : NULL,
  2216. fullname ? buf->b_ffname : buf->b_fname);
  2217. }
  2218. /*
  2219. * Set the "lnum" and "col" for the buffer "buf" and the current window.
  2220. * When "copy_options" is TRUE save the local window option values.
  2221. * When "lnum" is 0 only do the options.
  2222. */
  2223. static void
  2224. buflist_setfpos(buf, win, lnum, col, copy_options)
  2225. buf_T *buf;
  2226. win_T *win;
  2227. linenr_T lnum;
  2228. colnr_T col;
  2229. int copy_options;
  2230. {
  2231. wininfo_T *wip;
  2232. for (wip = buf->b_wininfo; wip != NULL; wip = wip->wi_next)
  2233. if (wip->wi_win == win)
  2234. break;
  2235. if (wip == NULL)
  2236. {
  2237. /* allocate a new entry */
  2238. wip = (wininfo_T *)alloc_clear((unsigned)sizeof(wininfo_T));
  2239. if (wip == NULL)
  2240. return;
  2241. wip->wi_win = win;
  2242. if (lnum == 0) /* set lnum even when it's 0 */
  2243. lnum = 1;
  2244. }
  2245. else
  2246. {
  2247. /* remove the entry from the list */
  2248. if (wip->wi_prev)
  2249. wip->wi_prev->wi_next = wip->wi_next;
  2250. else
  2251. buf->b_wininfo = wip->wi_next;
  2252. if (wip->wi_next)
  2253. wip->wi_next->wi_prev = wip->wi_prev;
  2254. if (copy_options && wip->wi_optset)
  2255. {
  2256. clear_winopt(&wip->wi_opt);
  2257. #ifdef FEAT_FOLDING
  2258. deleteFoldRecurse(&wip->wi_folds);
  2259. #endif
  2260. }
  2261. }
  2262. if (lnum != 0)
  2263. {
  2264. wip->wi_fpos.lnum = lnum;
  2265. wip->wi_fpos.col = col;
  2266. }
  2267. if (copy_options)
  2268. {
  2269. /* Save the window-specific option values. */
  2270. copy_winopt(&win->w_onebuf_opt, &wip->wi_opt);
  2271. #ifdef FEAT_FOLDING
  2272. wip->wi_fold_manual = win->w_fold_manual;
  2273. cloneFoldGrowArray(&win->w_folds, &wip->wi_folds);
  2274. #endif
  2275. wip->wi_optset = TRUE;
  2276. }
  2277. /* insert the entry in front of the list */
  2278. wip->wi_next = buf->b_wininfo;
  2279. buf->b_wininfo = wip;
  2280. wip->wi_prev = NULL;
  2281. if (wip->wi_next)
  2282. wip->wi_next->wi_prev = wip;
  2283. return;
  2284. }
  2285. #ifdef FEAT_DIFF
  2286. static int wininfo_other_tab_diff __ARGS((wininfo_T *wip));
  2287. /*
  2288. * Return TRUE when "wip" has 'diff' set and the diff is only for another tab
  2289. * page. That's because a diff is local to a tab page.
  2290. */
  2291. static int
  2292. wininfo_other_tab_diff(wip)
  2293. wininfo_T *wip;
  2294. {
  2295. win_T *wp;
  2296. if (wip->wi_opt.wo_diff)
  2297. {
  2298. for (wp = firstwin; wp != NULL; wp = wp->w_next)
  2299. /* return FALSE when it's a window in the current tab page, thus
  2300. * the buffer was in diff mode here */
  2301. if (wip->wi_win == wp)
  2302. return FALSE;
  2303. return TRUE;
  2304. }
  2305. return FALSE;
  2306. }
  2307. #endif
  2308. /*
  2309. * Find info for the current window in buffer "buf".
  2310. * If not found, return the info for the most recently used window.
  2311. * When "skip_diff_buffer" is TRUE avoid windows with 'diff' set that is in
  2312. * another tab page.
  2313. * Returns NULL when there isn't any info.
  2314. */
  2315. static wininfo_T *
  2316. find_wininfo(buf, skip_diff_buffer)
  2317. buf_T *buf;
  2318. int skip_diff_buffer UNUSED;
  2319. {
  2320. wininfo_T *wip;
  2321. for (wip = buf->b_wininfo; wip != NULL; wip = wip->wi_next)
  2322. if (wip->wi_win == curwin
  2323. #ifdef FEAT_DIFF
  2324. && (!skip_diff_buffer || !wininfo_other_tab_diff(wip))
  2325. #endif
  2326. )
  2327. break;
  2328. /* If no wininfo for curwin, use the first in the list (that doesn't have
  2329. * 'diff' set and is in another tab page). */
  2330. if (wip == NULL)
  2331. {
  2332. #ifdef FEAT_DIFF
  2333. if (skip_diff_buffer)
  2334. {
  2335. for (wip = buf->b_wininfo; wip != NULL; wip = wip->wi_next)
  2336. if (!wininfo_other_tab_diff(wip))
  2337. break;
  2338. }
  2339. else
  2340. #endif
  2341. wip = buf->b_wininfo;
  2342. }
  2343. return wip;
  2344. }
  2345. /*
  2346. * Reset the local window options to the values last used in this window.
  2347. * If the buffer wasn't used in this window before, use the values from
  2348. * the most recently used window. If the values were never set, use the
  2349. * global values for the window.
  2350. */
  2351. void
  2352. get_winopts(buf)
  2353. buf_T *buf;
  2354. {
  2355. wininfo_T *wip;
  2356. clear_winopt(&curwin->w_onebuf_opt);
  2357. #ifdef FEAT_FOLDING
  2358. clearFolding(curwin);
  2359. #endif
  2360. wip = find_wininfo(buf, TRUE);
  2361. if (wip != NULL && wip->wi_optset)
  2362. {
  2363. copy_winopt(&wip->wi_opt, &curwin->w_onebuf_opt);
  2364. #ifdef FEAT_FOLDING
  2365. curwin->w_fold_manual = wip->wi_fold_manual;
  2366. curwin->w_foldinvalid = TRUE;
  2367. cloneFoldGrowArray(&wip->wi_folds, &curwin->w_folds);
  2368. #endif
  2369. }
  2370. else
  2371. copy_winopt(&curwin->w_allbuf_opt, &curwin->w_onebuf_opt);
  2372. #ifdef FEAT_FOLDING
  2373. /* Set 'foldlevel' to 'foldlevelstart' if it's not negative. */
  2374. if (p_fdls >= 0)
  2375. curwin->w_p_fdl = p_fdls;
  2376. #endif
  2377. #ifdef FEAT_SYN_HL
  2378. check_colorcolumn(curwin);
  2379. #endif
  2380. }
  2381. /*
  2382. * Find the position (lnum and col) for the buffer 'buf' for the current
  2383. * window.
  2384. * Returns a pointer to no_position if no position is found.
  2385. */
  2386. pos_T *
  2387. buflist_findfpos(buf)
  2388. buf_T *buf;
  2389. {
  2390. wininfo_T *wip;
  2391. static pos_T no_position = INIT_POS_T(1, 0, 0);
  2392. wip = find_wininfo(buf, FALSE);
  2393. if (wip != NULL)
  2394. return &(wip->wi_fpos);
  2395. else
  2396. return &no_position;
  2397. }
  2398. /*
  2399. * Find the lnum for the buffer 'buf' for the current window.
  2400. */
  2401. linenr_T
  2402. buflist_findlnum(buf)
  2403. buf_T *buf;
  2404. {
  2405. return buflist_findfpos(buf)->lnum;
  2406. }
  2407. #if defined(FEAT_LISTCMDS) || defined(PROTO)
  2408. /*
  2409. * List all know file names (for :files and :buffers command).
  2410. */
  2411. void
  2412. buflist_list(eap)
  2413. exarg_T *eap;
  2414. {
  2415. buf_T *buf;
  2416. int len;
  2417. int i;
  2418. for (buf = firstbuf; buf != NULL && !got_int; buf = buf->b_next)
  2419. {
  2420. /* skip unlisted buffers, unless ! was used */
  2421. if (!buf->b_p_bl && !eap->forceit)
  2422. continue;
  2423. msg_putchar('\n');
  2424. if (buf_spname(buf) != NULL)
  2425. STRCPY(NameBuff, buf_spname(buf));
  2426. else
  2427. home_replace(buf, buf->b_fname, NameBuff, MAXPATHL, TRUE);
  2428. len = vim_snprintf((char *)IObuff, IOSIZE - 20, "%3d%c%c%c%c%c \"%s\"",
  2429. buf->b_fnum,
  2430. buf->b_p_bl ? ' ' : 'u',
  2431. buf == curbuf ? '%' :
  2432. (curwin->w_alt_fnum == buf->b_fnum ? '#' : ' '),
  2433. buf->b_ml.ml_mfp == NULL ? ' ' :
  2434. (buf->b_nwindows == 0 ? 'h' : 'a'),
  2435. !buf->b_p_ma ? '-' : (buf->b_p_ro ? '=' : ' '),
  2436. (buf->b_flags & BF_READERR) ? 'x'
  2437. : (bufIsChanged(buf) ? '+' : ' '),
  2438. NameBuff);
  2439. /* put "line 999" in column 40 or after the file name */
  2440. i = 40 - vim_strsize(IObuff);
  2441. do
  2442. {
  2443. IObuff[len++] = ' ';
  2444. } while (--i > 0 && len < IOSIZE - 18);
  2445. vim_snprintf((char *)IObuff + len, (size_t)(IOSIZE - len),
  2446. _("line %ld"), buf == curbuf ? curwin->w_cursor.lnum
  2447. : (long)buflist_findlnum(buf));
  2448. msg_outtrans(IObuff);
  2449. out_flush(); /* output one line at a time */
  2450. ui_breakcheck();
  2451. }
  2452. }
  2453. #endif
  2454. /*
  2455. * Get file name and line number for file 'fnum'.
  2456. * Used by DoOneCmd() for translating '%' and '#'.
  2457. * Used by insert_reg() and cmdline_paste() for '#' register.
  2458. * Return FAIL if not found, OK for success.
  2459. */
  2460. int
  2461. buflist_name_nr(fnum, fname, lnum)
  2462. int fnum;
  2463. char_u **fname;
  2464. linenr_T *lnum;
  2465. {
  2466. buf_T *buf;
  2467. buf = buflist_findnr(fnum);
  2468. if (buf == NULL || buf->b_fname == NULL)
  2469. return FAIL;
  2470. *fname = buf->b_fname;
  2471. *lnum = buflist_findlnum(buf);
  2472. return OK;
  2473. }
  2474. /*
  2475. * Set the file name for "buf"' to 'ffname', short file name to 'sfname'.
  2476. * The file name with the full path is also remembered, for when :cd is used.
  2477. * Returns FAIL for failure (file name already in use by other buffer)
  2478. * OK otherwise.
  2479. */
  2480. int
  2481. setfname(buf, ffname, sfname, message)
  2482. buf_T *buf;
  2483. char_u *ffname, *sfname;
  2484. int message; /* give message when buffer already exists */
  2485. {
  2486. buf_T *obuf = NULL;
  2487. #ifdef UNIX
  2488. struct stat st;
  2489. #endif
  2490. if (ffname == NULL || *ffname == NUL)
  2491. {
  2492. /* Removing the name. */
  2493. vim_free(buf->b_ffname);
  2494. vim_free(buf->b_sfname);
  2495. buf->b_ffname = NULL;
  2496. buf->b_sfname = NULL;
  2497. #ifdef UNIX
  2498. st.st_dev = (dev_T)-1;
  2499. #endif
  2500. }
  2501. else
  2502. {
  2503. fname_expand(buf, &ffname, &sfname); /* will allocate ffname */
  2504. if (ffname == NULL) /* out of memory */
  2505. return FAIL;
  2506. /*
  2507. * if the file name is already used in another buffer:
  2508. * - if the buffer is loaded, fail
  2509. * - if the buffer is not loaded, delete it from the list
  2510. */
  2511. #ifdef UNIX
  2512. if (mch_stat((char *)ffname, &st) < 0)
  2513. st.st_dev = (dev_T)-1;
  2514. #endif
  2515. if (!(buf->b_flags & BF_DUMMY))
  2516. #ifdef UNIX
  2517. obuf = buflist_findname_stat(ffname, &st);
  2518. #else
  2519. obuf = buflist_findname(ffname);
  2520. #endif
  2521. if (obuf != NULL && obuf != buf)
  2522. {
  2523. if (obuf->b_ml.ml_mfp != NULL) /* it's loaded, fail */
  2524. {
  2525. if (message)
  2526. EMSG(_("E95: Buffer with this name already exists"));
  2527. vim_free(ffname);
  2528. return FAIL;
  2529. }
  2530. /* delete from the list */
  2531. close_buffer(NULL, obuf, DOBUF_WIPE, FALSE);
  2532. }
  2533. sfname = vim_strsave(sfname);
  2534. if (ffname == NULL || sfname == NULL)
  2535. {
  2536. vim_free(sfname);
  2537. vim_free(ffname);
  2538. return FAIL;
  2539. }
  2540. #ifdef USE_FNAME_CASE
  2541. # ifdef USE_LONG_FNAME
  2542. if (USE_LONG_FNAME)
  2543. # endif
  2544. fname_case(sfname, 0); /* set correct case for short file name */
  2545. #endif
  2546. vim_free(buf->b_ffname);
  2547. vim_free(buf->b_sfname);
  2548. buf->b_ffname = ffname;
  2549. buf->b_sfname = sfname;
  2550. }
  2551. buf->b_fname = buf->b_sfname;
  2552. #ifdef UNIX
  2553. if (st.st_dev == (dev_T)-1)
  2554. buf->b_dev_valid = FALSE;
  2555. else
  2556. {
  2557. buf->b_dev_valid = TRUE;
  2558. buf->b_dev = st.st_dev;
  2559. buf->b_ino = st.st_ino;
  2560. }
  2561. #endif
  2562. #ifndef SHORT_FNAME
  2563. buf->b_shortname = FALSE;
  2564. #endif
  2565. buf_name_changed(buf);
  2566. return OK;
  2567. }
  2568. /*
  2569. * Crude way of changing the name of a buffer. Use with care!
  2570. * The name should be relative to the current directory.
  2571. */
  2572. void
  2573. buf_set_name(fnum, name)
  2574. int fnum;
  2575. char_u *name;
  2576. {
  2577. buf_T *buf;
  2578. buf = buflist_findnr(fnum);
  2579. if (buf != NULL)
  2580. {
  2581. vim_free(buf->b_sfname);
  2582. vim_free(buf->b_ffname);
  2583. buf->b_ffname = vim_strsave(name);
  2584. buf->b_sfname = NULL;
  2585. /* Allocate ffname and expand into full path. Also resolves .lnk
  2586. * files on Win32. */
  2587. fname_expand(buf, &buf->b_ffname, &buf->b_sfname);
  2588. buf->b_fname = buf->b_sfname;
  2589. }
  2590. }
  2591. /*
  2592. * Take care of what needs to be done when the name of buffer "buf" has
  2593. * changed.
  2594. */
  2595. void
  2596. buf_name_changed(buf)
  2597. buf_T *buf;
  2598. {
  2599. /*
  2600. * If the file name changed, also change the name of the swapfile
  2601. */
  2602. if (buf->b_ml.ml_mfp != NULL)
  2603. ml_setname(buf);
  2604. if (curwin->w_buffer == buf)
  2605. check_arg_idx(curwin); /* check file name for arg list */
  2606. #ifdef FEAT_TITLE
  2607. maketitle(); /* set window title */
  2608. #endif
  2609. #ifdef FEAT_WINDOWS
  2610. status_redraw_all(); /* status lines need to be redrawn */
  2611. #endif
  2612. fmarks_check_names(buf); /* check named file marks */
  2613. ml_timestamp(buf); /* reset timestamp */
  2614. }
  2615. /*
  2616. * set alternate file name for current window
  2617. *
  2618. * Used by do_one_cmd(), do_write() and do_ecmd().
  2619. * Return the buffer.
  2620. */
  2621. buf_T *
  2622. setaltfname(ffname, sfname, lnum)
  2623. char_u *ffname;
  2624. char_u *sfname;
  2625. linenr_T lnum;
  2626. {
  2627. buf_T *buf;
  2628. /* Create a buffer. 'buflisted' is not set if it's a new buffer */
  2629. buf = buflist_new(ffname, sfname, lnum, 0);
  2630. if (buf != NULL && !cmdmod.keepalt)
  2631. curwin->w_alt_fnum = buf->b_fnum;
  2632. return buf;
  2633. }
  2634. /*
  2635. * Get alternate file name for current window.
  2636. * Return NULL if there isn't any, and give error message if requested.
  2637. */
  2638. char_u *
  2639. getaltfname(errmsg)
  2640. int errmsg; /* give error message */
  2641. {
  2642. char_u *fname;
  2643. linenr_T dummy;
  2644. if (buflist_name_nr(0, &fname, &dummy) == FAIL)
  2645. {
  2646. if (errmsg)
  2647. EMSG(_(e_noalt));
  2648. return NULL;
  2649. }
  2650. return fname;
  2651. }
  2652. /*
  2653. * Add a file name to the buflist and return its number.
  2654. * Uses same flags as buflist_new(), except BLN_DUMMY.
  2655. *
  2656. * used by qf_init(), main() and doarglist()
  2657. */
  2658. int
  2659. buflist_add(fname, flags)
  2660. char_u *fname;
  2661. int flags;
  2662. {
  2663. buf_T *buf;
  2664. buf = buflist_new(fname, NULL, (linenr_T)0, flags);
  2665. if (buf != NULL)
  2666. return buf->b_fnum;
  2667. return 0;
  2668. }
  2669. #if defined(BACKSLASH_IN_FILENAME) || defined(PROTO)
  2670. /*
  2671. * Adjust slashes in file names. Called after 'shellslash' was set.
  2672. */
  2673. void
  2674. buflist_slash_adjust()
  2675. {
  2676. buf_T *bp;
  2677. for (bp = firstbuf; bp != NULL; bp = bp->b_next)
  2678. {
  2679. if (bp->b_ffname != NULL)
  2680. slash_adjust(bp->b_ffname);
  2681. if (bp->b_sfname != NULL)
  2682. slash_adjust(bp->b_sfname);
  2683. }
  2684. }
  2685. #endif
  2686. /*
  2687. * Set alternate cursor position for the current buffer and window "win".
  2688. * Also save the local window option values.
  2689. */
  2690. void
  2691. buflist_altfpos(win)
  2692. win_T *win;
  2693. {
  2694. buflist_setfpos(curbuf, win, win->w_cursor.lnum, win->w_cursor.col, TRUE);
  2695. }
  2696. /*
  2697. * Return TRUE if 'ffname' is not the same file as current file.
  2698. * Fname must have a full path (expanded by mch_FullName()).
  2699. */
  2700. int
  2701. otherfile(ffname)
  2702. char_u *ffname;
  2703. {
  2704. return otherfile_buf(curbuf, ffname
  2705. #ifdef UNIX
  2706. , NULL
  2707. #endif
  2708. );
  2709. }
  2710. static int
  2711. otherfile_buf(buf, ffname
  2712. #ifdef UNIX
  2713. , stp
  2714. #endif
  2715. )
  2716. buf_T *buf;
  2717. char_u *ffname;
  2718. #ifdef UNIX
  2719. struct stat *stp;
  2720. #endif
  2721. {
  2722. /* no name is different */
  2723. if (ffname == NULL || *ffname == NUL || buf->b_ffname == NULL)
  2724. return TRUE;
  2725. if (fnamecmp(ffname, buf->b_ffname) == 0)
  2726. return FALSE;
  2727. #ifdef UNIX
  2728. {
  2729. struct stat st;
  2730. /* If no struct stat given, get it now */
  2731. if (stp == NULL)
  2732. {
  2733. if (!buf->b_dev_valid || mch_stat((char *)ffname, &st) < 0)
  2734. st.st_dev = (dev_T)-1;
  2735. stp = &st;
  2736. }
  2737. /* Use dev/ino to check if the files are the same, even when the names
  2738. * are different (possible with links). Still need to compare the
  2739. * name above, for when the file doesn't exist yet.
  2740. * Problem: The dev/ino changes when a file is deleted (and created
  2741. * again) and remains the same when renamed/moved. We don't want to
  2742. * mch_stat() each buffer each time, that would be too slow. Get the
  2743. * dev/ino again when they appear to match, but not when they appear
  2744. * to be different: Could skip a buffer when it's actually the same
  2745. * file. */
  2746. if (buf_same_ino(buf, stp))
  2747. {
  2748. buf_setino(buf);
  2749. if (buf_same_ino(buf, stp))
  2750. return FALSE;
  2751. }
  2752. }
  2753. #endif
  2754. return TRUE;
  2755. }
  2756. #if defined(UNIX) || defined(PROTO)
  2757. /*
  2758. * Set inode and device number for a buffer.
  2759. * Must always be called when b_fname is changed!.
  2760. */
  2761. void
  2762. buf_setino(buf)
  2763. buf_T *buf;
  2764. {
  2765. struct stat st;
  2766. if (buf->b_fname != NULL && mch_stat((char *)buf->b_fname, &st) >= 0)
  2767. {
  2768. buf->b_dev_valid = TRUE;
  2769. buf->b_dev = st.st_dev;
  2770. buf->b_ino = st.st_ino;
  2771. }
  2772. else
  2773. buf->b_dev_valid = FALSE;
  2774. }
  2775. /*
  2776. * Return TRUE if dev/ino in buffer "buf" matches with "stp".
  2777. */
  2778. static int
  2779. buf_same_ino(buf, stp)
  2780. buf_T *buf;
  2781. struct stat *stp;
  2782. {
  2783. return (buf->b_dev_valid
  2784. && stp->st_dev == buf->b_dev
  2785. && stp->st_ino == buf->b_ino);
  2786. }
  2787. #endif
  2788. /*
  2789. * Print info about the current buffer.
  2790. */
  2791. void
  2792. fileinfo(fullname, shorthelp, dont_truncate)
  2793. int fullname; /* when non-zero print full path */
  2794. int shorthelp;
  2795. int dont_truncate;
  2796. {
  2797. char_u *name;
  2798. int n;
  2799. char_u *p;
  2800. char_u *buffer;
  2801. size_t len;
  2802. buffer = alloc(IOSIZE);
  2803. if (buffer == NULL)
  2804. return;
  2805. if (fullname > 1) /* 2 CTRL-G: include buffer number */
  2806. {
  2807. vim_snprintf((char *)buffer, IOSIZE, "buf %d: ", curbuf->b_fnum);
  2808. p = buffer + STRLEN(buffer);
  2809. }
  2810. else
  2811. p = buffer;
  2812. *p++ = '"';
  2813. if (buf_spname(curbuf) != NULL)
  2814. STRCPY(p, buf_spname(curbuf));
  2815. else
  2816. {
  2817. if (!fullname && curbuf->b_fname != NULL)
  2818. name = curbuf->b_fname;
  2819. else
  2820. name = curbuf->b_ffname;
  2821. home_replace(shorthelp ? curbuf : NULL, name, p,
  2822. (int)(IOSIZE - (p - buffer)), TRUE);
  2823. }
  2824. vim_snprintf_add((char *)buffer, IOSIZE, "\"%s%s%s%s%s%s",
  2825. curbufIsChanged() ? (shortmess(SHM_MOD)
  2826. ? " [+]" : _(" [Modified]")) : " ",
  2827. (curbuf->b_flags & BF_NOTEDITED)
  2828. #ifdef FEAT_QUICKFIX
  2829. && !bt_dontwrite(curbuf)
  2830. #endif
  2831. ? _("[Not edited]") : "",
  2832. (curbuf->b_flags & BF_NEW)
  2833. #ifdef FEAT_QUICKFIX
  2834. && !bt_dontwrite(curbuf)
  2835. #endif
  2836. ? _("[New file]") : "",
  2837. (curbuf->b_flags & BF_READERR) ? _("[Read errors]") : "",
  2838. curbuf->b_p_ro ? (shortmess(SHM_RO) ? "[RO]"
  2839. : _("[readonly]")) : "",
  2840. (curbufIsChanged() || (curbuf->b_flags & BF_WRITE_MASK)
  2841. || curbuf->b_p_ro) ?
  2842. " " : "");
  2843. /* With 32 bit longs and more than 21,474,836 lines multiplying by 100
  2844. * causes an overflow, thus for large numbers divide instead. */
  2845. if (curwin->w_cursor.lnum > 1000000L)
  2846. n = (int)(((long)curwin->w_cursor.lnum) /
  2847. ((long)curbuf->b_ml.ml_line_count / 100L));
  2848. else
  2849. n = (int)(((long)curwin->w_cursor.lnum * 100L) /
  2850. (long)curbuf->b_ml.ml_line_count);
  2851. if (curbuf->b_ml.ml_flags & ML_EMPTY)
  2852. {
  2853. vim_snprintf_add((char *)buffer, IOSIZE, "%s", _(no_lines_msg));
  2854. }
  2855. #ifdef FEAT_CMDL_INFO
  2856. else if (p_ru)
  2857. {
  2858. /* Current line and column are already on the screen -- webb */
  2859. if (curbuf->b_ml.ml_line_count == 1)
  2860. vim_snprintf_add((char *)buffer, IOSIZE, _("1 line --%d%%--"), n);
  2861. else
  2862. vim_snprintf_add((char *)buffer, IOSIZE, _("%ld lines --%d%%--"),
  2863. (long)curbuf->b_ml.ml_line_count, n);
  2864. }
  2865. #endif
  2866. else
  2867. {
  2868. vim_snprintf_add((char *)buffer, IOSIZE,
  2869. _("line %ld of %ld --%d%%-- col "),
  2870. (long)curwin->w_cursor.lnum,
  2871. (long)curbuf->b_ml.ml_line_count,
  2872. n);
  2873. validate_virtcol();
  2874. len = STRLEN(buffer);
  2875. col_print(buffer + len, IOSIZE - len,
  2876. (int)curwin->w_cursor.col + 1, (int)curwin->w_virtcol + 1);
  2877. }
  2878. (void)append_arg_number(curwin, buffer, IOSIZE, !shortmess(SHM_FILE));
  2879. if (dont_truncate)
  2880. {
  2881. /* Temporarily set msg_scroll to avoid the message being truncated.
  2882. * First call msg_start() to get the message in the right place. */
  2883. msg_start();
  2884. n = msg_scroll;
  2885. msg_scroll = TRUE;
  2886. msg(buffer);
  2887. msg_scroll = n;
  2888. }
  2889. else
  2890. {
  2891. p = msg_trunc_attr(buffer, FALSE, 0);
  2892. if (restart_edit != 0 || (msg_scrolled && !need_wait_return))
  2893. /* Need to repeat the message after redrawing when:
  2894. * - When restart_edit is set (otherwise there will be a delay
  2895. * before redrawing).
  2896. * - When the screen was scrolled but there is no wait-return
  2897. * prompt. */
  2898. set_keep_msg(p, 0);
  2899. }
  2900. vim_free(buffer);
  2901. }
  2902. void
  2903. col_print(buf, buflen, col, vcol)
  2904. char_u *buf;
  2905. size_t buflen;
  2906. int col;
  2907. int vcol;
  2908. {
  2909. if (col == vcol)
  2910. vim_snprintf((char *)buf, buflen, "%d", col);
  2911. else
  2912. vim_snprintf((char *)buf, buflen, "%d-%d", col, vcol);
  2913. }
  2914. #if defined(FEAT_TITLE) || defined(PROTO)
  2915. /*
  2916. * put file name in title bar of window and in icon title
  2917. */
  2918. static char_u *lasttitle = NULL;
  2919. static char_u *lasticon = NULL;
  2920. void
  2921. maketitle()
  2922. {
  2923. char_u *p;
  2924. char_u *t_str = NULL;
  2925. char_u *i_name;
  2926. char_u *i_str = NULL;
  2927. int maxlen = 0;
  2928. int len;
  2929. int mustset;
  2930. char_u buf[IOSIZE];
  2931. int off;
  2932. if (!redrawing())
  2933. {
  2934. /* Postpone updating the title when 'lazyredraw' is set. */
  2935. need_maketitle = TRUE;
  2936. return;
  2937. }
  2938. need_maketitle = FALSE;
  2939. if (!p_title && !p_icon && lasttitle == NULL && lasticon == NULL)
  2940. return;
  2941. if (p_title)
  2942. {
  2943. if (p_titlelen > 0)
  2944. {
  2945. maxlen = p_titlelen * Columns / 100;
  2946. if (maxlen < 10)
  2947. maxlen = 10;
  2948. }
  2949. t_str = buf;
  2950. if (*p_titlestring != NUL)
  2951. {
  2952. #ifdef FEAT_STL_OPT
  2953. if (stl_syntax & STL_IN_TITLE)
  2954. {
  2955. int use_sandbox = FALSE;
  2956. int save_called_emsg = called_emsg;
  2957. # ifdef FEAT_EVAL
  2958. use_sandbox = was_set_insecurely((char_u *)"titlestring", 0);
  2959. # endif
  2960. called_emsg = FALSE;
  2961. build_stl_str_hl(curwin, t_str, sizeof(buf),
  2962. p_titlestring, use_sandbox,
  2963. 0, maxlen, NULL, NULL);
  2964. if (called_emsg)
  2965. set_string_option_direct((char_u *)"titlestring", -1,
  2966. (char_u *)"", OPT_FREE, SID_ERROR);
  2967. called_emsg |= save_called_emsg;
  2968. }
  2969. else
  2970. #endif
  2971. t_str = p_titlestring;
  2972. }
  2973. else
  2974. {
  2975. /* format: "fname + (path) (1 of 2) - VIM" */
  2976. if (curbuf->b_fname == NULL)
  2977. vim_strncpy(buf, (char_u *)_("[No Name]"), IOSIZE - 100);
  2978. else
  2979. {
  2980. p = transstr(gettail(curbuf->b_fname));
  2981. vim_strncpy(buf, p, IOSIZE - 100);
  2982. vim_free(p);
  2983. }
  2984. switch (bufIsChanged(curbuf)
  2985. + (curbuf->b_p_ro * 2)
  2986. + (!curbuf->b_p_ma * 4))
  2987. {
  2988. case 1: STRCAT(buf, " +"); break;
  2989. case 2: STRCAT(buf, " ="); break;
  2990. case 3: STRCAT(buf, " =+"); break;
  2991. case 4:
  2992. case 6: STRCAT(buf, " -"); break;
  2993. case 5:
  2994. case 7: STRCAT(buf, " -+"); break;
  2995. }
  2996. if (curbuf->b_fname != NULL)
  2997. {
  2998. /* Get path of file, replace home dir with ~ */
  2999. off = (int)STRLEN(buf);
  3000. buf[off++] = ' ';
  3001. buf[off++] = '(';
  3002. home_replace(curbuf, curbuf->b_ffname,
  3003. buf + off, IOSIZE - off, TRUE);
  3004. #ifdef BACKSLASH_IN_FILENAME
  3005. /* avoid "c:/name" to be reduced to "c" */
  3006. if (isalpha(buf[off]) && buf[off + 1] == ':')
  3007. off += 2;
  3008. #endif
  3009. /* remove the file name */
  3010. p = gettail_sep(buf + off);
  3011. if (p == buf + off)
  3012. /* must be a help buffer */
  3013. vim_strncpy(buf + off, (char_u *)_("help"),
  3014. (size_t)(IOSIZE - off - 1));
  3015. else
  3016. *p = NUL;
  3017. /* translate unprintable chars */
  3018. p = transstr(buf + off);
  3019. vim_strncpy(buf + off, p, (size_t)(IOSIZE - off - 1));
  3020. vim_free(p);
  3021. STRCAT(buf, ")");
  3022. }
  3023. append_arg_number(curwin, buf, IOSIZE, FALSE);
  3024. #if defined(FEAT_CLIENTSERVER)
  3025. if (serverName != NULL)
  3026. {
  3027. STRCAT(buf, " - ");
  3028. vim_strcat(buf, serverName, IOSIZE);
  3029. }
  3030. else
  3031. #endif
  3032. STRCAT(buf, " - VIM");
  3033. if (maxlen > 0)
  3034. {
  3035. /* make it shorter by removing a bit in the middle */
  3036. if (vim_strsize(buf) > maxlen)
  3037. trunc_string(buf, buf, maxlen, IOSIZE);
  3038. }
  3039. }
  3040. }
  3041. mustset = ti_change(t_str, &lasttitle);
  3042. if (p_icon)
  3043. {
  3044. i_str = buf;
  3045. if (*p_iconstring != NUL)
  3046. {
  3047. #ifdef FEAT_STL_OPT
  3048. if (stl_syntax & STL_IN_ICON)
  3049. {
  3050. int use_sandbox = FALSE;
  3051. int save_called_emsg = called_emsg;
  3052. # ifdef FEAT_EVAL
  3053. use_sandbox = was_set_insecurely((char_u *)"iconstring", 0);
  3054. # endif
  3055. called_emsg = FALSE;
  3056. build_stl_str_hl(curwin, i_str, sizeof(buf),
  3057. p_iconstring, use_sandbox,
  3058. 0, 0, NULL, NULL);
  3059. if (called_emsg)
  3060. set_string_option_direct((char_u *)"iconstring", -1,
  3061. (char_u *)"", OPT_FREE, SID_ERROR);
  3062. called_emsg |= save_called_emsg;
  3063. }
  3064. else
  3065. #endif
  3066. i_str = p_iconstring;
  3067. }
  3068. else
  3069. {
  3070. if (buf_spname(curbuf) != NULL)
  3071. i_name = (char_u *)buf_spname(curbuf);
  3072. else /* use file name only in icon */
  3073. i_name = gettail(curbuf->b_ffname);
  3074. *i_str = NUL;
  3075. /* Truncate name at 100 bytes. */
  3076. len = (int)STRLEN(i_name);
  3077. if (len > 100)
  3078. {
  3079. len -= 100;
  3080. #ifdef FEAT_MBYTE
  3081. if (has_mbyte)
  3082. len += (*mb_tail_off)(i_name, i_name + len) + 1;
  3083. #endif
  3084. i_name += len;
  3085. }
  3086. STRCPY(i_str, i_name);
  3087. trans_characters(i_str, IOSIZE);
  3088. }
  3089. }
  3090. mustset |= ti_change(i_str, &lasticon);
  3091. if (mustset)
  3092. resettitle();
  3093. }
  3094. /*
  3095. * Used for title and icon: Check if "str" differs from "*last". Set "*last"
  3096. * from "str" if it does.
  3097. * Return TRUE when "*last" changed.
  3098. */
  3099. static int
  3100. ti_change(str, last)
  3101. char_u *str;
  3102. char_u **last;
  3103. {
  3104. if ((str == NULL) != (*last == NULL)
  3105. || (str != NULL && *last != NULL && STRCMP(str, *last) != 0))
  3106. {
  3107. vim_free(*last);
  3108. if (str == NULL)
  3109. *last = NULL;
  3110. else
  3111. *last = vim_strsave(str);
  3112. return TRUE;
  3113. }
  3114. return FALSE;
  3115. }
  3116. /*
  3117. * Put current window title back (used after calling a shell)
  3118. */
  3119. void
  3120. resettitle()
  3121. {
  3122. mch_settitle(lasttitle, lasticon);
  3123. }
  3124. # if defined(EXITFREE) || defined(PROTO)
  3125. void
  3126. free_titles()
  3127. {
  3128. vim_free(lasttitle);
  3129. vim_free(lasticon);
  3130. }
  3131. # endif
  3132. #endif /* FEAT_TITLE */
  3133. #if defined(FEAT_STL_OPT) || defined(FEAT_GUI_TABLINE) || defined(PROTO)
  3134. /*
  3135. * Build a string from the status line items in "fmt".
  3136. * Return length of string in screen cells.
  3137. *
  3138. * Normally works for window "wp", except when working for 'tabline' then it
  3139. * is "curwin".
  3140. *
  3141. * Items are drawn interspersed with the text that surrounds it
  3142. * Specials: %-<wid>(xxx%) => group, %= => middle marker, %< => truncation
  3143. * Item: %-<minwid>.<maxwid><itemch> All but <itemch> are optional
  3144. *
  3145. * If maxwidth is not zero, the string will be filled at any middle marker
  3146. * or truncated if too long, fillchar is used for all whitespace.
  3147. */
  3148. int
  3149. build_stl_str_hl(wp, out, outlen, fmt, use_sandbox, fillchar,
  3150. maxwidth, hltab, tabtab)
  3151. win_T *wp;
  3152. char_u *out; /* buffer to write into != NameBuff */
  3153. size_t outlen; /* length of out[] */
  3154. char_u *fmt;
  3155. int use_sandbox UNUSED; /* "fmt" was set insecurely, use sandbox */
  3156. int fillchar;
  3157. int maxwidth;
  3158. struct stl_hlrec *hltab; /* return: HL attributes (can be NULL) */
  3159. struct stl_hlrec *tabtab; /* return: tab page nrs (can be NULL) */
  3160. {
  3161. char_u *p;
  3162. char_u *s;
  3163. char_u *t;
  3164. char_u *linecont;
  3165. #ifdef FEAT_EVAL
  3166. win_T *o_curwin;
  3167. buf_T *o_curbuf;
  3168. #endif
  3169. int empty_line;
  3170. colnr_T virtcol;
  3171. long l;
  3172. long n;
  3173. int prevchar_isflag;
  3174. int prevchar_isitem;
  3175. int itemisflag;
  3176. int fillable;
  3177. char_u *str;
  3178. long num;
  3179. int width;
  3180. int itemcnt;
  3181. int curitem;
  3182. int groupitem[STL_MAX_ITEM];
  3183. int groupdepth;
  3184. struct stl_item
  3185. {
  3186. char_u *start;
  3187. int minwid;
  3188. int maxwid;
  3189. enum
  3190. {
  3191. Normal,
  3192. Empty,
  3193. Group,
  3194. Middle,
  3195. Highlight,
  3196. TabPage,
  3197. Trunc
  3198. } type;
  3199. } item[STL_MAX_ITEM];
  3200. int minwid;
  3201. int maxwid;
  3202. int zeropad;
  3203. char_u base;
  3204. char_u opt;
  3205. #define TMPLEN 70
  3206. char_u tmp[TMPLEN];
  3207. char_u *usefmt = fmt;
  3208. struct stl_hlrec *sp;
  3209. #ifdef FEAT_EVAL
  3210. /*
  3211. * When the format starts with "%!" then evaluate it as an expression and
  3212. * use the result as the actual format string.
  3213. */
  3214. if (fmt[0] == '%' && fmt[1] == '!')
  3215. {
  3216. usefmt = eval_to_string_safe(fmt + 2, NULL, use_sandbox);
  3217. if (usefmt == NULL)
  3218. usefmt = fmt;
  3219. }
  3220. #endif
  3221. if (fillchar == 0)
  3222. fillchar = ' ';
  3223. #ifdef FEAT_MBYTE
  3224. /* Can't handle a multi-byte fill character yet. */
  3225. else if (mb_char2len(fillchar) > 1)
  3226. fillchar = '-';
  3227. #endif
  3228. /*
  3229. * Get line & check if empty (cursorpos will show "0-1").
  3230. * If inversion is possible we use it. Else '=' characters are used.
  3231. */
  3232. linecont = ml_get_buf(wp->w_buffer, wp->w_cursor.lnum, FALSE);
  3233. empty_line = (*linecont == NUL);
  3234. groupdepth = 0;
  3235. p = out;
  3236. curitem = 0;
  3237. prevchar_isflag = TRUE;
  3238. prevchar_isitem = FALSE;
  3239. for (s = usefmt; *s; )
  3240. {
  3241. if (curitem == STL_MAX_ITEM)
  3242. {
  3243. /* There are too many items. Add the error code to the statusline
  3244. * to give the user a hint about what went wrong. */
  3245. if (p + 6 < out + outlen)
  3246. {
  3247. mch_memmove(p, " E541", (size_t)5);
  3248. p += 5;
  3249. }
  3250. break;
  3251. }
  3252. if (*s != NUL && *s != '%')
  3253. prevchar_isflag = prevchar_isitem = FALSE;
  3254. /*
  3255. * Handle up to the next '%' or the end.
  3256. */
  3257. while (*s != NUL && *s != '%' && p + 1 < out + outlen)
  3258. *p++ = *s++;
  3259. if (*s == NUL || p + 1 >= out + outlen)
  3260. break;
  3261. /*
  3262. * Handle one '%' item.
  3263. */
  3264. s++;
  3265. if (*s == NUL) /* ignore trailing % */
  3266. break;
  3267. if (*s == '%')
  3268. {
  3269. if (p + 1 >= out + outlen)
  3270. break;
  3271. *p++ = *s++;
  3272. prevchar_isflag = prevchar_isitem = FALSE;
  3273. continue;
  3274. }
  3275. if (*s == STL_MIDDLEMARK)
  3276. {
  3277. s++;
  3278. if (groupdepth > 0)
  3279. continue;
  3280. item[curitem].type = Middle;
  3281. item[curitem++].start = p;
  3282. continue;
  3283. }
  3284. if (*s == STL_TRUNCMARK)
  3285. {
  3286. s++;
  3287. item[curitem].type = Trunc;
  3288. item[curitem++].start = p;
  3289. continue;
  3290. }
  3291. if (*s == ')')
  3292. {
  3293. s++;
  3294. if (groupdepth < 1)
  3295. continue;
  3296. groupdepth--;
  3297. t = item[groupitem[groupdepth]].start;
  3298. *p = NUL;
  3299. l = vim_strsize(t);
  3300. if (curitem > groupitem[groupdepth] + 1
  3301. && item[groupitem[groupdepth]].minwid == 0)
  3302. {
  3303. /* remove group if all items are empty */
  3304. for (n = groupitem[groupdepth] + 1; n < curitem; n++)
  3305. if (item[n].type == Normal)
  3306. break;
  3307. if (n == curitem)
  3308. {
  3309. p = t;
  3310. l = 0;
  3311. }
  3312. }
  3313. if (l > item[groupitem[groupdepth]].maxwid)
  3314. {
  3315. /* truncate, remove n bytes of text at the start */
  3316. #ifdef FEAT_MBYTE
  3317. if (has_mbyte)
  3318. {
  3319. /* Find the first character that should be included. */
  3320. n = 0;
  3321. while (l >= item[groupitem[groupdepth]].maxwid)
  3322. {
  3323. l -= ptr2cells(t + n);
  3324. n += (*mb_ptr2len)(t + n);
  3325. }
  3326. }
  3327. else
  3328. #endif
  3329. n = (long)(p - t) - item[groupitem[groupdepth]].maxwid + 1;
  3330. *t = '<';
  3331. mch_memmove(t + 1, t + n, (size_t)(p - (t + n)));
  3332. p = p - n + 1;
  3333. #ifdef FEAT_MBYTE
  3334. /* Fill up space left over by half a double-wide char. */
  3335. while (++l < item[groupitem[groupdepth]].minwid)
  3336. *p++ = fillchar;
  3337. #endif
  3338. /* correct the start of the items for the truncation */
  3339. for (l = groupitem[groupdepth] + 1; l < curitem; l++)
  3340. {
  3341. item[l].start -= n;
  3342. if (item[l].start < t)
  3343. item[l].start = t;
  3344. }
  3345. }
  3346. else if (abs(item[groupitem[groupdepth]].minwid) > l)
  3347. {
  3348. /* fill */
  3349. n = item[groupitem[groupdepth]].minwid;
  3350. if (n < 0)
  3351. {
  3352. /* fill by appending characters */
  3353. n = 0 - n;
  3354. while (l++ < n && p + 1 < out + outlen)
  3355. *p++ = fillchar;
  3356. }
  3357. else
  3358. {
  3359. /* fill by inserting characters */
  3360. mch_memmove(t + n - l, t, (size_t)(p - t));
  3361. l = n - l;
  3362. if (p + l >= out + outlen)
  3363. l = (long)((out + outlen) - p - 1);
  3364. p += l;
  3365. for (n = groupitem[groupdepth] + 1; n < curitem; n++)
  3366. item[n].start += l;
  3367. for ( ; l > 0; l--)
  3368. *t++ = fillchar;
  3369. }
  3370. }
  3371. continue;
  3372. }
  3373. minwid = 0;
  3374. maxwid = 9999;
  3375. zeropad = FALSE;
  3376. l = 1;
  3377. if (*s == '0')
  3378. {
  3379. s++;
  3380. zeropad = TRUE;
  3381. }
  3382. if (*s == '-')
  3383. {
  3384. s++;
  3385. l = -1;
  3386. }
  3387. if (VIM_ISDIGIT(*s))
  3388. {
  3389. minwid = (int)getdigits(&s);
  3390. if (minwid < 0) /* overflow */
  3391. minwid = 0;
  3392. }
  3393. if (*s == STL_USER_HL)
  3394. {
  3395. item[curitem].type = Highlight;
  3396. item[curitem].start = p;
  3397. item[curitem].minwid = minwid > 9 ? 1 : minwid;
  3398. s++;
  3399. curitem++;
  3400. continue;
  3401. }
  3402. if (*s == STL_TABPAGENR || *s == STL_TABCLOSENR)
  3403. {
  3404. if (*s == STL_TABCLOSENR)
  3405. {
  3406. if (minwid == 0)
  3407. {
  3408. /* %X ends the close label, go back to the previously
  3409. * define tab label nr. */
  3410. for (n = curitem - 1; n >= 0; --n)
  3411. if (item[n].type == TabPage && item[n].minwid >= 0)
  3412. {
  3413. minwid = item[n].minwid;
  3414. break;
  3415. }
  3416. }
  3417. else
  3418. /* close nrs are stored as negative values */
  3419. minwid = - minwid;
  3420. }
  3421. item[curitem].type = TabPage;
  3422. item[curitem].start = p;
  3423. item[curitem].minwid = minwid;
  3424. s++;
  3425. curitem++;
  3426. continue;
  3427. }
  3428. if (*s == '.')
  3429. {
  3430. s++;
  3431. if (VIM_ISDIGIT(*s))
  3432. {
  3433. maxwid = (int)getdigits(&s);
  3434. if (maxwid <= 0) /* overflow */
  3435. maxwid = 50;
  3436. }
  3437. }
  3438. minwid = (minwid > 50 ? 50 : minwid) * l;
  3439. if (*s == '(')
  3440. {
  3441. groupitem[groupdepth++] = curitem;
  3442. item[curitem].type = Group;
  3443. item[curitem].start = p;
  3444. item[curitem].minwid = minwid;
  3445. item[curitem].maxwid = maxwid;
  3446. s++;
  3447. curitem++;
  3448. continue;
  3449. }
  3450. if (vim_strchr(STL_ALL, *s) == NULL)
  3451. {
  3452. s++;
  3453. continue;
  3454. }
  3455. opt = *s++;
  3456. /* OK - now for the real work */
  3457. base = 'D';
  3458. itemisflag = FALSE;
  3459. fillable = TRUE;
  3460. num = -1;
  3461. str = NULL;
  3462. switch (opt)
  3463. {
  3464. case STL_FILEPATH:
  3465. case STL_FULLPATH:
  3466. case STL_FILENAME:
  3467. fillable = FALSE; /* don't change ' ' to fillchar */
  3468. if (buf_spname(wp->w_buffer) != NULL)
  3469. STRCPY(NameBuff, buf_spname(wp->w_buffer));
  3470. else
  3471. {
  3472. t = (opt == STL_FULLPATH) ? wp->w_buffer->b_ffname
  3473. : wp->w_buffer->b_fname;
  3474. home_replace(wp->w_buffer, t, NameBuff, MAXPATHL, TRUE);
  3475. }
  3476. trans_characters(NameBuff, MAXPATHL);
  3477. if (opt != STL_FILENAME)
  3478. str = NameBuff;
  3479. else
  3480. str = gettail(NameBuff);
  3481. break;
  3482. case STL_VIM_EXPR: /* '{' */
  3483. itemisflag = TRUE;
  3484. t = p;
  3485. while (*s != '}' && *s != NUL && p + 1 < out + outlen)
  3486. *p++ = *s++;
  3487. if (*s != '}') /* missing '}' or out of space */
  3488. break;
  3489. s++;
  3490. *p = 0;
  3491. p = t;
  3492. #ifdef FEAT_EVAL
  3493. vim_snprintf((char *)tmp, sizeof(tmp), "%d", curbuf->b_fnum);
  3494. set_internal_string_var((char_u *)"actual_curbuf", tmp);
  3495. o_curbuf = curbuf;
  3496. o_curwin = curwin;
  3497. curwin = wp;
  3498. curbuf = wp->w_buffer;
  3499. str = eval_to_string_safe(p, &t, use_sandbox);
  3500. curwin = o_curwin;
  3501. curbuf = o_curbuf;
  3502. do_unlet((char_u *)"g:actual_curbuf", TRUE);
  3503. if (str != NULL && *str != 0)
  3504. {
  3505. if (*skipdigits(str) == NUL)
  3506. {
  3507. num = atoi((char *)str);
  3508. vim_free(str);
  3509. str = NULL;
  3510. itemisflag = FALSE;
  3511. }
  3512. }
  3513. #endif
  3514. break;
  3515. case STL_LINE:
  3516. num = (wp->w_buffer->b_ml.ml_flags & ML_EMPTY)
  3517. ? 0L : (long)(wp->w_cursor.lnum);
  3518. break;
  3519. case STL_NUMLINES:
  3520. num = wp->w_buffer->b_ml.ml_line_count;
  3521. break;
  3522. case STL_COLUMN:
  3523. num = !(State & INSERT) && empty_line
  3524. ? 0 : (int)wp->w_cursor.col + 1;
  3525. break;
  3526. case STL_VIRTCOL:
  3527. case STL_VIRTCOL_ALT:
  3528. /* In list mode virtcol needs to be recomputed */
  3529. virtcol = wp->w_virtcol;
  3530. if (wp->w_p_list && lcs_tab1 == NUL)
  3531. {
  3532. wp->w_p_list = FALSE;
  3533. getvcol(wp, &wp->w_cursor, NULL, &virtcol, NULL);
  3534. wp->w_p_list = TRUE;
  3535. }
  3536. ++virtcol;
  3537. /* Don't display %V if it's the same as %c. */
  3538. if (opt == STL_VIRTCOL_ALT
  3539. && (virtcol == (colnr_T)(!(State & INSERT) && empty_line
  3540. ? 0 : (int)wp->w_cursor.col + 1)))
  3541. break;
  3542. num = (long)virtcol;
  3543. break;
  3544. case STL_PERCENTAGE:
  3545. num = (int)(((long)wp->w_cursor.lnum * 100L) /
  3546. (long)wp->w_buffer->b_ml.ml_line_count);
  3547. break;
  3548. case STL_ALTPERCENT:
  3549. str = tmp;
  3550. get_rel_pos(wp, str, TMPLEN);
  3551. break;
  3552. case STL_ARGLISTSTAT:
  3553. fillable = FALSE;
  3554. tmp[0] = 0;
  3555. if (append_arg_number(wp, tmp, (int)sizeof(tmp), FALSE))
  3556. str = tmp;
  3557. break;
  3558. case STL_KEYMAP:
  3559. fillable = FALSE;
  3560. if (get_keymap_str(wp, tmp, TMPLEN))
  3561. str = tmp;
  3562. break;
  3563. case STL_PAGENUM:
  3564. #if defined(FEAT_PRINTER) || defined(FEAT_GUI_TABLINE)
  3565. num = printer_page_num;
  3566. #else
  3567. num = 0;
  3568. #endif
  3569. break;
  3570. case STL_BUFNO:
  3571. num = wp->w_buffer->b_fnum;
  3572. break;
  3573. case STL_OFFSET_X:
  3574. base = 'X';
  3575. case STL_OFFSET:
  3576. #ifdef FEAT_BYTEOFF
  3577. l = ml_find_line_or_offset(wp->w_buffer, wp->w_cursor.lnum, NULL);
  3578. num = (wp->w_buffer->b_ml.ml_flags & ML_EMPTY) || l < 0 ?
  3579. 0L : l + 1 + (!(State & INSERT) && empty_line ?
  3580. 0 : (int)wp->w_cursor.col);
  3581. #endif
  3582. break;
  3583. case STL_BYTEVAL_X:
  3584. base = 'X';
  3585. case STL_BYTEVAL:
  3586. if (wp->w_cursor.col > (colnr_T)STRLEN(linecont))
  3587. num = 0;
  3588. else
  3589. {
  3590. #ifdef FEAT_MBYTE
  3591. num = (*mb_ptr2char)(linecont + wp->w_cursor.col);
  3592. #else
  3593. num = linecont[wp->w_cursor.col];
  3594. #endif
  3595. }
  3596. if (num == NL)
  3597. num = 0;
  3598. else if (num == CAR && get_fileformat(wp->w_buffer) == EOL_MAC)
  3599. num = NL;
  3600. break;
  3601. case STL_ROFLAG:
  3602. case STL_ROFLAG_ALT:
  3603. itemisflag = TRUE;
  3604. if (wp->w_buffer->b_p_ro)
  3605. str = (char_u *)((opt == STL_ROFLAG_ALT) ? ",RO" : "[RO]");
  3606. break;
  3607. case STL_HELPFLAG:
  3608. case STL_HELPFLAG_ALT:
  3609. itemisflag = TRUE;
  3610. if (wp->w_buffer->b_help)
  3611. str = (char_u *)((opt == STL_HELPFLAG_ALT) ? ",HLP"
  3612. : _("[Help]"));
  3613. break;
  3614. #ifdef FEAT_AUTOCMD
  3615. case STL_FILETYPE:
  3616. if (*wp->w_buffer->b_p_ft != NUL
  3617. && STRLEN(wp->w_buffer->b_p_ft) < TMPLEN - 3)
  3618. {
  3619. vim_snprintf((char *)tmp, sizeof(tmp), "[%s]",
  3620. wp->w_buffer->b_p_ft);
  3621. str = tmp;
  3622. }
  3623. break;
  3624. case STL_FILETYPE_ALT:
  3625. itemisflag = TRUE;
  3626. if (*wp->w_buffer->b_p_ft != NUL
  3627. && STRLEN(wp->w_buffer->b_p_ft) < TMPLEN - 2)
  3628. {
  3629. vim_snprintf((char *)tmp, sizeof(tmp), ",%s",
  3630. wp->w_buffer->b_p_ft);
  3631. for (t = tmp; *t != 0; t++)
  3632. *t = TOUPPER_LOC(*t);
  3633. str = tmp;
  3634. }
  3635. break;
  3636. #endif
  3637. #if defined(FEAT_WINDOWS) && defined(FEAT_QUICKFIX)
  3638. case STL_PREVIEWFLAG:
  3639. case STL_PREVIEWFLAG_ALT:
  3640. itemisflag = TRUE;
  3641. if (wp->w_p_pvw)
  3642. str = (char_u *)((opt == STL_PREVIEWFLAG_ALT) ? ",PRV"
  3643. : _("[Preview]"));
  3644. break;
  3645. case STL_QUICKFIX:
  3646. if (bt_quickfix(wp->w_buffer))
  3647. str = (char_u *)(wp->w_llist_ref
  3648. ? _(msg_loclist)
  3649. : _(msg_qflist));
  3650. break;
  3651. #endif
  3652. case STL_MODIFIED:
  3653. case STL_MODIFIED_ALT:
  3654. itemisflag = TRUE;
  3655. switch ((opt == STL_MODIFIED_ALT)
  3656. + bufIsChanged(wp->w_buffer) * 2
  3657. + (!wp->w_buffer->b_p_ma) * 4)
  3658. {
  3659. case 2: str = (char_u *)"[+]"; break;
  3660. case 3: str = (char_u *)",+"; break;
  3661. case 4: str = (char_u *)"[-]"; break;
  3662. case 5: str = (char_u *)",-"; break;
  3663. case 6: str = (char_u *)"[+-]"; break;
  3664. case 7: str = (char_u *)",+-"; break;
  3665. }
  3666. break;
  3667. case STL_HIGHLIGHT:
  3668. t = s;
  3669. while (*s != '#' && *s != NUL)
  3670. ++s;
  3671. if (*s == '#')
  3672. {
  3673. item[curitem].type = Highlight;
  3674. item[curitem].start = p;
  3675. item[curitem].minwid = -syn_namen2id(t, (int)(s - t));
  3676. curitem++;
  3677. }
  3678. ++s;
  3679. continue;
  3680. }
  3681. item[curitem].start = p;
  3682. item[curitem].type = Normal;
  3683. if (str != NULL && *str)
  3684. {
  3685. t = str;
  3686. if (itemisflag)
  3687. {
  3688. if ((t[0] && t[1])
  3689. && ((!prevchar_isitem && *t == ',')
  3690. || (prevchar_isflag && *t == ' ')))
  3691. t++;
  3692. prevchar_isflag = TRUE;
  3693. }
  3694. l = vim_strsize(t);
  3695. if (l > 0)
  3696. prevchar_isitem = TRUE;
  3697. if (l > maxwid)
  3698. {
  3699. while (l >= maxwid)
  3700. #ifdef FEAT_MBYTE
  3701. if (has_mbyte)
  3702. {
  3703. l -= ptr2cells(t);
  3704. t += (*mb_ptr2len)(t);
  3705. }
  3706. else
  3707. #endif
  3708. l -= byte2cells(*t++);
  3709. if (p + 1 >= out + outlen)
  3710. break;
  3711. *p++ = '<';
  3712. }
  3713. if (minwid > 0)
  3714. {
  3715. for (; l < minwid && p + 1 < out + outlen; l++)
  3716. {
  3717. /* Don't put a "-" in front of a digit. */
  3718. if (l + 1 == minwid && fillchar == '-' && VIM_ISDIGIT(*t))
  3719. *p++ = ' ';
  3720. else
  3721. *p++ = fillchar;
  3722. }
  3723. minwid = 0;
  3724. }
  3725. else
  3726. minwid *= -1;
  3727. while (*t && p + 1 < out + outlen)
  3728. {
  3729. *p++ = *t++;
  3730. /* Change a space by fillchar, unless fillchar is '-' and a
  3731. * digit follows. */
  3732. if (fillable && p[-1] == ' '
  3733. && (!VIM_ISDIGIT(*t) || fillchar != '-'))
  3734. p[-1] = fillchar;
  3735. }
  3736. for (; l < minwid && p + 1 < out + outlen; l++)
  3737. *p++ = fillchar;
  3738. }
  3739. else if (num >= 0)
  3740. {
  3741. int nbase = (base == 'D' ? 10 : (base == 'O' ? 8 : 16));
  3742. char_u nstr[20];
  3743. if (p + 20 >= out + outlen)
  3744. break; /* not sufficient space */
  3745. prevchar_isitem = TRUE;
  3746. t = nstr;
  3747. if (opt == STL_VIRTCOL_ALT)
  3748. {
  3749. *t++ = '-';
  3750. minwid--;
  3751. }
  3752. *t++ = '%';
  3753. if (zeropad)
  3754. *t++ = '0';
  3755. *t++ = '*';
  3756. *t++ = nbase == 16 ? base : (char_u)(nbase == 8 ? 'o' : 'd');
  3757. *t = 0;
  3758. for (n = num, l = 1; n >= nbase; n /= nbase)
  3759. l++;
  3760. if (opt == STL_VIRTCOL_ALT)
  3761. l++;
  3762. if (l > maxwid)
  3763. {
  3764. l += 2;
  3765. n = l - maxwid;
  3766. while (l-- > maxwid)
  3767. num /= nbase;
  3768. *t++ = '>';
  3769. *t++ = '%';
  3770. *t = t[-3];
  3771. *++t = 0;
  3772. vim_snprintf((char *)p, outlen - (p - out), (char *)nstr,
  3773. 0, num, n);
  3774. }
  3775. else
  3776. vim_snprintf((char *)p, outlen - (p - out), (char *)nstr,
  3777. minwid, num);
  3778. p += STRLEN(p);
  3779. }
  3780. else
  3781. item[curitem].type = Empty;
  3782. if (opt == STL_VIM_EXPR)
  3783. vim_free(str);
  3784. if (num >= 0 || (!itemisflag && str && *str))
  3785. prevchar_isflag = FALSE; /* Item not NULL, but not a flag */
  3786. curitem++;
  3787. }
  3788. *p = NUL;
  3789. itemcnt = curitem;
  3790. #ifdef FEAT_EVAL
  3791. if (usefmt != fmt)
  3792. vim_free(usefmt);
  3793. #endif
  3794. width = vim_strsize(out);
  3795. if (maxwidth > 0 && width > maxwidth)
  3796. {
  3797. /* Result is too long, must truncate somewhere. */
  3798. l = 0;
  3799. if (itemcnt == 0)
  3800. s = out;
  3801. else
  3802. {
  3803. for ( ; l < itemcnt; l++)
  3804. if (item[l].type == Trunc)
  3805. {
  3806. /* Truncate at %< item. */
  3807. s = item[l].start;
  3808. break;
  3809. }
  3810. if (l == itemcnt)
  3811. {
  3812. /* No %< item, truncate first item. */
  3813. s = item[0].start;
  3814. l = 0;
  3815. }
  3816. }
  3817. if (width - vim_strsize(s) >= maxwidth)
  3818. {
  3819. /* Truncation mark is beyond max length */
  3820. #ifdef FEAT_MBYTE
  3821. if (has_mbyte)
  3822. {
  3823. s = out;
  3824. width = 0;
  3825. for (;;)
  3826. {
  3827. width += ptr2cells(s);
  3828. if (width >= maxwidth)
  3829. break;
  3830. s += (*mb_ptr2len)(s);
  3831. }
  3832. /* Fill up for half a double-wide character. */
  3833. while (++width < maxwidth)
  3834. *s++ = fillchar;
  3835. }
  3836. else
  3837. #endif
  3838. s = out + maxwidth - 1;
  3839. for (l = 0; l < itemcnt; l++)
  3840. if (item[l].start > s)
  3841. break;
  3842. itemcnt = l;
  3843. *s++ = '>';
  3844. *s = 0;
  3845. }
  3846. else
  3847. {
  3848. #ifdef FEAT_MBYTE
  3849. if (has_mbyte)
  3850. {
  3851. n = 0;
  3852. while (width >= maxwidth)
  3853. {
  3854. width -= ptr2cells(s + n);
  3855. n += (*mb_ptr2len)(s + n);
  3856. }
  3857. }
  3858. else
  3859. #endif
  3860. n = width - maxwidth + 1;
  3861. p = s + n;
  3862. STRMOVE(s + 1, p);
  3863. *s = '<';
  3864. /* Fill up for half a double-wide character. */
  3865. while (++width < maxwidth)
  3866. {
  3867. s = s + STRLEN(s);
  3868. *s++ = fillchar;
  3869. *s = NUL;
  3870. }
  3871. --n; /* count the '<' */
  3872. for (; l < itemcnt; l++)
  3873. {
  3874. if (item[l].start - n >= s)
  3875. item[l].start -= n;
  3876. else
  3877. item[l].start = s;
  3878. }
  3879. }
  3880. width = maxwidth;
  3881. }
  3882. else if (width < maxwidth && STRLEN(out) + maxwidth - width + 1 < outlen)
  3883. {
  3884. /* Apply STL_MIDDLE if any */
  3885. for (l = 0; l < itemcnt; l++)
  3886. if (item[l].type == Middle)
  3887. break;
  3888. if (l < itemcnt)
  3889. {
  3890. p = item[l].start + maxwidth - width;
  3891. STRMOVE(p, item[l].start);
  3892. for (s = item[l].start; s < p; s++)
  3893. *s = fillchar;
  3894. for (l++; l < itemcnt; l++)
  3895. item[l].start += maxwidth - width;
  3896. width = maxwidth;
  3897. }
  3898. }
  3899. /* Store the info about highlighting. */
  3900. if (hltab != NULL)
  3901. {
  3902. sp = hltab;
  3903. for (l = 0; l < itemcnt; l++)
  3904. {
  3905. if (item[l].type == Highlight)
  3906. {
  3907. sp->start = item[l].start;
  3908. sp->userhl = item[l].minwid;
  3909. sp++;
  3910. }
  3911. }
  3912. sp->start = NULL;
  3913. sp->userhl = 0;
  3914. }
  3915. /* Store the info about tab pages labels. */
  3916. if (tabtab != NULL)
  3917. {
  3918. sp = tabtab;
  3919. for (l = 0; l < itemcnt; l++)
  3920. {
  3921. if (item[l].type == TabPage)
  3922. {
  3923. sp->start = item[l].start;
  3924. sp->userhl = item[l].minwid;
  3925. sp++;
  3926. }
  3927. }
  3928. sp->start = NULL;
  3929. sp->userhl = 0;
  3930. }
  3931. return width;
  3932. }
  3933. #endif /* FEAT_STL_OPT */
  3934. #if defined(FEAT_STL_OPT) || defined(FEAT_CMDL_INFO) \
  3935. || defined(FEAT_GUI_TABLINE) || defined(PROTO)
  3936. /*
  3937. * Get relative cursor position in window into "buf[buflen]", in the form 99%,
  3938. * using "Top", "Bot" or "All" when appropriate.
  3939. */
  3940. void
  3941. get_rel_pos(wp, buf, buflen)
  3942. win_T *wp;
  3943. char_u *buf;
  3944. int buflen;
  3945. {
  3946. long above; /* number of lines above window */
  3947. long below; /* number of lines below window */
  3948. above = wp->w_topline - 1;
  3949. #ifdef FEAT_DIFF
  3950. above += diff_check_fill(wp, wp->w_topline) - wp->w_topfill;
  3951. #endif
  3952. below = wp->w_buffer->b_ml.ml_line_count - wp->w_botline + 1;
  3953. if (below <= 0)
  3954. vim_strncpy(buf, (char_u *)(above == 0 ? _("All") : _("Bot")),
  3955. (size_t)(buflen - 1));
  3956. else if (above <= 0)
  3957. vim_strncpy(buf, (char_u *)_("Top"), (size_t)(buflen - 1));
  3958. else
  3959. vim_snprintf((char *)buf, (size_t)buflen, "%2d%%", above > 1000000L
  3960. ? (int)(above / ((above + below) / 100L))
  3961. : (int)(above * 100L / (above + below)));
  3962. }
  3963. #endif
  3964. /*
  3965. * Append (file 2 of 8) to "buf[buflen]", if editing more than one file.
  3966. * Return TRUE if it was appended.
  3967. */
  3968. static int
  3969. append_arg_number(wp, buf, buflen, add_file)
  3970. win_T *wp;
  3971. char_u *buf;
  3972. int buflen;
  3973. int add_file; /* Add "file" before the arg number */
  3974. {
  3975. char_u *p;
  3976. if (ARGCOUNT <= 1) /* nothing to do */
  3977. return FALSE;
  3978. p = buf + STRLEN(buf); /* go to the end of the buffer */
  3979. if (p - buf + 35 >= buflen) /* getting too long */
  3980. return FALSE;
  3981. *p++ = ' ';
  3982. *p++ = '(';
  3983. if (add_file)
  3984. {
  3985. STRCPY(p, "file ");
  3986. p += 5;
  3987. }
  3988. vim_snprintf((char *)p, (size_t)(buflen - (p - buf)),
  3989. wp->w_arg_idx_invalid ? "(%d) of %d)"
  3990. : "%d of %d)", wp->w_arg_idx + 1, ARGCOUNT);
  3991. return TRUE;
  3992. }
  3993. /*
  3994. * If fname is not a full path, make it a full path.
  3995. * Returns pointer to allocated memory (NULL for failure).
  3996. */
  3997. char_u *
  3998. fix_fname(fname)
  3999. char_u *fname;
  4000. {
  4001. /*
  4002. * Force expanding the path always for Unix, because symbolic links may
  4003. * mess up the full path name, even though it starts with a '/'.
  4004. * Also expand when there is ".." in the file name, try to remove it,
  4005. * because "c:/src/../README" is equal to "c:/README".
  4006. * Similarly "c:/src//file" is equal to "c:/src/file".
  4007. * For MS-Windows also expand names like "longna~1" to "longname".
  4008. */
  4009. #ifdef UNIX
  4010. return FullName_save(fname, TRUE);
  4011. #else
  4012. if (!vim_isAbsName(fname)
  4013. || strstr((char *)fname, "..") != NULL
  4014. || strstr((char *)fname, "//") != NULL
  4015. # ifdef BACKSLASH_IN_FILENAME
  4016. || strstr((char *)fname, "\\\\") != NULL
  4017. # endif
  4018. # if defined(MSWIN) || defined(DJGPP)
  4019. || vim_strchr(fname, '~') != NULL
  4020. # endif
  4021. )
  4022. return FullName_save(fname, FALSE);
  4023. fname = vim_strsave(fname);
  4024. # ifdef USE_FNAME_CASE
  4025. # ifdef USE_LONG_FNAME
  4026. if (USE_LONG_FNAME)
  4027. # endif
  4028. {
  4029. if (fname != NULL)
  4030. fname_case(fname, 0); /* set correct case for file name */
  4031. }
  4032. # endif
  4033. return fname;
  4034. #endif
  4035. }
  4036. /*
  4037. * Make "ffname" a full file name, set "sfname" to "ffname" if not NULL.
  4038. * "ffname" becomes a pointer to allocated memory (or NULL).
  4039. */
  4040. void
  4041. fname_expand(buf, ffname, sfname)
  4042. buf_T *buf UNUSED;
  4043. char_u **ffname;
  4044. char_u **sfname;
  4045. {
  4046. if (*ffname == NULL) /* if no file name given, nothing to do */
  4047. return;
  4048. if (*sfname == NULL) /* if no short file name given, use ffname */
  4049. *sfname = *ffname;
  4050. *ffname = fix_fname(*ffname); /* expand to full path */
  4051. #ifdef FEAT_SHORTCUT
  4052. if (!buf->b_p_bin)
  4053. {
  4054. char_u *rfname;
  4055. /* If the file name is a shortcut file, use the file it links to. */
  4056. rfname = mch_resolve_shortcut(*ffname);
  4057. if (rfname != NULL)
  4058. {
  4059. vim_free(*ffname);
  4060. *ffname = rfname;
  4061. *sfname = rfname;
  4062. }
  4063. }
  4064. #endif
  4065. }
  4066. /*
  4067. * Get the file name for an argument list entry.
  4068. */
  4069. char_u *
  4070. alist_name(aep)
  4071. aentry_T *aep;
  4072. {
  4073. buf_T *bp;
  4074. /* Use the name from the associated buffer if it exists. */
  4075. bp = buflist_findnr(aep->ae_fnum);
  4076. if (bp == NULL || bp->b_fname == NULL)
  4077. return aep->ae_fname;
  4078. return bp->b_fname;
  4079. }
  4080. #if defined(FEAT_WINDOWS) || defined(PROTO)
  4081. /*
  4082. * do_arg_all(): Open up to 'count' windows, one for each argument.
  4083. */
  4084. void
  4085. do_arg_all(count, forceit, keep_tabs)
  4086. int count;
  4087. int forceit; /* hide buffers in current windows */
  4088. int keep_tabs; /* keep current tabs, for ":tab drop file" */
  4089. {
  4090. int i;
  4091. win_T *wp, *wpnext;
  4092. char_u *opened; /* Array of weight for which args are open:
  4093. * 0: not opened
  4094. * 1: opened in other tab
  4095. * 2: opened in curtab
  4096. * 3: opened in curtab and curwin
  4097. */
  4098. int opened_len; /* length of opened[] */
  4099. int use_firstwin = FALSE; /* use first window for arglist */
  4100. int split_ret = OK;
  4101. int p_ea_save;
  4102. alist_T *alist; /* argument list to be used */
  4103. buf_T *buf;
  4104. tabpage_T *tpnext;
  4105. int had_tab = cmdmod.tab;
  4106. win_T *old_curwin, *last_curwin;
  4107. tabpage_T *old_curtab, *last_curtab;
  4108. win_T *new_curwin = NULL;
  4109. tabpage_T *new_curtab = NULL;
  4110. if (ARGCOUNT <= 0)
  4111. {
  4112. /* Don't give an error message. We don't want it when the ":all"
  4113. * command is in the .vimrc. */
  4114. return;
  4115. }
  4116. setpcmark();
  4117. opened_len = ARGCOUNT;
  4118. opened = alloc_clear((unsigned)opened_len);
  4119. if (opened == NULL)
  4120. return;
  4121. /* Autocommands may do anything to the argument list. Make sure it's not
  4122. * freed while we are working here by "locking" it. We still have to
  4123. * watch out for its size to be changed. */
  4124. alist = curwin->w_alist;
  4125. ++alist->al_refcount;
  4126. old_curwin = curwin;
  4127. old_curtab = curtab;
  4128. #ifdef FEAT_GUI
  4129. need_mouse_correct = TRUE;
  4130. #endif
  4131. /*
  4132. * Try closing all windows that are not in the argument list.
  4133. * Also close windows that are not full width;
  4134. * When 'hidden' or "forceit" set the buffer becomes hidden.
  4135. * Windows that have a changed buffer and can't be hidden won't be closed.
  4136. * When the ":tab" modifier was used do this for all tab pages.
  4137. */
  4138. if (had_tab > 0)
  4139. goto_tabpage_tp(first_tabpage);
  4140. for (;;)
  4141. {
  4142. tpnext = curtab->tp_next;
  4143. for (wp = firstwin; wp != NULL; wp = wpnext)
  4144. {
  4145. wpnext = wp->w_next;
  4146. buf = wp->w_buffer;
  4147. if (buf->b_ffname == NULL
  4148. || (!keep_tabs && buf->b_nwindows > 1)
  4149. #ifdef FEAT_VERTSPLIT
  4150. || wp->w_width != Columns
  4151. #endif
  4152. )
  4153. i = opened_len;
  4154. else
  4155. {
  4156. /* check if the buffer in this window is in the arglist */
  4157. for (i = 0; i < opened_len; ++i)
  4158. {
  4159. if (i < alist->al_ga.ga_len
  4160. && (AARGLIST(alist)[i].ae_fnum == buf->b_fnum
  4161. || fullpathcmp(alist_name(&AARGLIST(alist)[i]),
  4162. buf->b_ffname, TRUE) & FPC_SAME))
  4163. {
  4164. int weight = 1;
  4165. if (old_curtab == curtab)
  4166. {
  4167. ++weight;
  4168. if (old_curwin == wp)
  4169. ++weight;
  4170. }
  4171. if (weight > (int)opened[i])
  4172. {
  4173. opened[i] = (char_u)weight;
  4174. if (i == 0)
  4175. {
  4176. if (new_curwin != NULL)
  4177. new_curwin->w_arg_idx = opened_len;
  4178. new_curwin = wp;
  4179. new_curtab = curtab;
  4180. }
  4181. }
  4182. else if (keep_tabs)
  4183. i = opened_len;
  4184. if (wp->w_alist != alist)
  4185. {
  4186. /* Use the current argument list for all windows
  4187. * containing a file from it. */
  4188. alist_unlink(wp->w_alist);
  4189. wp->w_alist = alist;
  4190. ++wp->w_alist->al_refcount;
  4191. }
  4192. break;
  4193. }
  4194. }
  4195. }
  4196. wp->w_arg_idx = i;
  4197. if (i == opened_len && !keep_tabs)/* close this window */
  4198. {
  4199. if (P_HID(buf) || forceit || buf->b_nwindows > 1
  4200. || !bufIsChanged(buf))
  4201. {
  4202. /* If the buffer was changed, and we would like to hide it,
  4203. * try autowriting. */
  4204. if (!P_HID(buf) && buf->b_nwindows <= 1
  4205. && bufIsChanged(buf))
  4206. {
  4207. (void)autowrite(buf, FALSE);
  4208. #ifdef FEAT_AUTOCMD
  4209. /* check if autocommands removed the window */
  4210. if (!win_valid(wp) || !buf_valid(buf))
  4211. {
  4212. wpnext = firstwin; /* start all over... */
  4213. continue;
  4214. }
  4215. #endif
  4216. }
  4217. #ifdef FEAT_WINDOWS
  4218. /* don't close last window */
  4219. if (firstwin == lastwin
  4220. && (first_tabpage->tp_next == NULL || !had_tab))
  4221. #endif
  4222. use_firstwin = TRUE;
  4223. #ifdef FEAT_WINDOWS
  4224. else
  4225. {
  4226. win_close(wp, !P_HID(buf) && !bufIsChanged(buf));
  4227. # ifdef FEAT_AUTOCMD
  4228. /* check if autocommands removed the next window */
  4229. if (!win_valid(wpnext))
  4230. wpnext = firstwin; /* start all over... */
  4231. # endif
  4232. }
  4233. #endif
  4234. }
  4235. }
  4236. }
  4237. /* Without the ":tab" modifier only do the current tab page. */
  4238. if (had_tab == 0 || tpnext == NULL)
  4239. break;
  4240. # ifdef FEAT_AUTOCMD
  4241. /* check if autocommands removed the next tab page */
  4242. if (!valid_tabpage(tpnext))
  4243. tpnext = first_tabpage; /* start all over...*/
  4244. # endif
  4245. goto_tabpage_tp(tpnext);
  4246. }
  4247. /*
  4248. * Open a window for files in the argument list that don't have one.
  4249. * ARGCOUNT may change while doing this, because of autocommands.
  4250. */
  4251. if (count > opened_len || count <= 0)
  4252. count = opened_len;
  4253. #ifdef FEAT_AUTOCMD
  4254. /* Don't execute Win/Buf Enter/Leave autocommands here. */
  4255. ++autocmd_no_enter;
  4256. ++autocmd_no_leave;
  4257. #endif
  4258. last_curwin = curwin;
  4259. last_curtab = curtab;
  4260. win_enter(lastwin, FALSE);
  4261. #ifdef FEAT_WINDOWS
  4262. /* ":drop all" should re-use an empty window to avoid "--remote-tab"
  4263. * leaving an empty tab page when executed locally. */
  4264. if (keep_tabs && bufempty() && curbuf->b_nwindows == 1
  4265. && curbuf->b_ffname == NULL && !curbuf->b_changed)
  4266. use_firstwin = TRUE;
  4267. #endif
  4268. for (i = 0; i < count && i < opened_len && !got_int; ++i)
  4269. {
  4270. if (alist == &global_alist && i == global_alist.al_ga.ga_len - 1)
  4271. arg_had_last = TRUE;
  4272. if (opened[i] > 0)
  4273. {
  4274. /* Move the already present window to below the current window */
  4275. if (curwin->w_arg_idx != i)
  4276. {
  4277. for (wpnext = firstwin; wpnext != NULL; wpnext = wpnext->w_next)
  4278. {
  4279. if (wpnext->w_arg_idx == i)
  4280. {
  4281. if (keep_tabs)
  4282. {
  4283. new_curwin = wpnext;
  4284. new_curtab = curtab;
  4285. }
  4286. else
  4287. win_move_after(wpnext, curwin);
  4288. break;
  4289. }
  4290. }
  4291. }
  4292. }
  4293. else if (split_ret == OK)
  4294. {
  4295. if (!use_firstwin) /* split current window */
  4296. {
  4297. p_ea_save = p_ea;
  4298. p_ea = TRUE; /* use space from all windows */
  4299. split_ret = win_split(0, WSP_ROOM | WSP_BELOW);
  4300. p_ea = p_ea_save;
  4301. if (split_ret == FAIL)
  4302. continue;
  4303. }
  4304. #ifdef FEAT_AUTOCMD
  4305. else /* first window: do autocmd for leaving this buffer */
  4306. --autocmd_no_leave;
  4307. #endif
  4308. /*
  4309. * edit file "i"
  4310. */
  4311. curwin->w_arg_idx = i;
  4312. if (i == 0)
  4313. {
  4314. new_curwin = curwin;
  4315. new_curtab = curtab;
  4316. }
  4317. (void)do_ecmd(0, alist_name(&AARGLIST(alist)[i]), NULL, NULL,
  4318. ECMD_ONE,
  4319. ((P_HID(curwin->w_buffer)
  4320. || bufIsChanged(curwin->w_buffer)) ? ECMD_HIDE : 0)
  4321. + ECMD_OLDBUF, curwin);
  4322. #ifdef FEAT_AUTOCMD
  4323. if (use_firstwin)
  4324. ++autocmd_no_leave;
  4325. #endif
  4326. use_firstwin = FALSE;
  4327. }
  4328. ui_breakcheck();
  4329. /* When ":tab" was used open a new tab for a new window repeatedly. */
  4330. if (had_tab > 0 && tabpage_index(NULL) <= p_tpm)
  4331. cmdmod.tab = 9999;
  4332. }
  4333. /* Remove the "lock" on the argument list. */
  4334. alist_unlink(alist);
  4335. #ifdef FEAT_AUTOCMD
  4336. --autocmd_no_enter;
  4337. #endif
  4338. /* restore last referenced tabpage's curwin */
  4339. if (last_curtab != new_curtab)
  4340. {
  4341. if (valid_tabpage(last_curtab))
  4342. goto_tabpage_tp(last_curtab);
  4343. if (win_valid(last_curwin))
  4344. win_enter(last_curwin, FALSE);
  4345. }
  4346. /* to window with first arg */
  4347. if (valid_tabpage(new_curtab))
  4348. goto_tabpage_tp(new_curtab);
  4349. if (win_valid(new_curwin))
  4350. win_enter(new_curwin, FALSE);
  4351. #ifdef FEAT_AUTOCMD
  4352. --autocmd_no_leave;
  4353. #endif
  4354. vim_free(opened);
  4355. }
  4356. # if defined(FEAT_LISTCMDS) || defined(PROTO)
  4357. /*
  4358. * Open a window for a number of buffers.
  4359. */
  4360. void
  4361. ex_buffer_all(eap)
  4362. exarg_T *eap;
  4363. {
  4364. buf_T *buf;
  4365. win_T *wp, *wpnext;
  4366. int split_ret = OK;
  4367. int p_ea_save;
  4368. int open_wins = 0;
  4369. int r;
  4370. int count; /* Maximum number of windows to open. */
  4371. int all; /* When TRUE also load inactive buffers. */
  4372. #ifdef FEAT_WINDOWS
  4373. int had_tab = cmdmod.tab;
  4374. tabpage_T *tpnext;
  4375. #endif
  4376. if (eap->addr_count == 0) /* make as many windows as possible */
  4377. count = 9999;
  4378. else
  4379. count = eap->line2; /* make as many windows as specified */
  4380. if (eap->cmdidx == CMD_unhide || eap->cmdidx == CMD_sunhide)
  4381. all = FALSE;
  4382. else
  4383. all = TRUE;
  4384. setpcmark();
  4385. #ifdef FEAT_GUI
  4386. need_mouse_correct = TRUE;
  4387. #endif
  4388. /*
  4389. * Close superfluous windows (two windows for the same buffer).
  4390. * Also close windows that are not full-width.
  4391. */
  4392. #ifdef FEAT_WINDOWS
  4393. if (had_tab > 0)
  4394. goto_tabpage_tp(first_tabpage);
  4395. for (;;)
  4396. {
  4397. #endif
  4398. tpnext = curtab->tp_next;
  4399. for (wp = firstwin; wp != NULL; wp = wpnext)
  4400. {
  4401. wpnext = wp->w_next;
  4402. if ((wp->w_buffer->b_nwindows > 1
  4403. #ifdef FEAT_VERTSPLIT
  4404. || ((cmdmod.split & WSP_VERT)
  4405. ? wp->w_height + wp->w_status_height < Rows - p_ch
  4406. - tabline_height()
  4407. : wp->w_width != Columns)
  4408. #endif
  4409. #ifdef FEAT_WINDOWS
  4410. || (had_tab > 0 && wp != firstwin)
  4411. #endif
  4412. ) && firstwin != lastwin
  4413. #ifdef FEAT_AUTOCMD
  4414. && !(wp->w_closing || wp->w_buffer->b_closing)
  4415. #endif
  4416. )
  4417. {
  4418. win_close(wp, FALSE);
  4419. #ifdef FEAT_AUTOCMD
  4420. wpnext = firstwin; /* just in case an autocommand does
  4421. something strange with windows */
  4422. tpnext = first_tabpage; /* start all over...*/
  4423. open_wins = 0;
  4424. #endif
  4425. }
  4426. else
  4427. ++open_wins;
  4428. }
  4429. #ifdef FEAT_WINDOWS
  4430. /* Without the ":tab" modifier only do the current tab page. */
  4431. if (had_tab == 0 || tpnext == NULL)
  4432. break;
  4433. goto_tabpage_tp(tpnext);
  4434. }
  4435. #endif
  4436. /*
  4437. * Go through the buffer list. When a buffer doesn't have a window yet,
  4438. * open one. Otherwise move the window to the right position.
  4439. * Watch out for autocommands that delete buffers or windows!
  4440. */
  4441. #ifdef FEAT_AUTOCMD
  4442. /* Don't execute Win/Buf Enter/Leave autocommands here. */
  4443. ++autocmd_no_enter;
  4444. #endif
  4445. win_enter(lastwin, FALSE);
  4446. #ifdef FEAT_AUTOCMD
  4447. ++autocmd_no_leave;
  4448. #endif
  4449. for (buf = firstbuf; buf != NULL && open_wins < count; buf = buf->b_next)
  4450. {
  4451. /* Check if this buffer needs a window */
  4452. if ((!all && buf->b_ml.ml_mfp == NULL) || !buf->b_p_bl)
  4453. continue;
  4454. #ifdef FEAT_WINDOWS
  4455. if (had_tab != 0)
  4456. {
  4457. /* With the ":tab" modifier don't move the window. */
  4458. if (buf->b_nwindows > 0)
  4459. wp = lastwin; /* buffer has a window, skip it */
  4460. else
  4461. wp = NULL;
  4462. }
  4463. else
  4464. #endif
  4465. {
  4466. /* Check if this buffer already has a window */
  4467. for (wp = firstwin; wp != NULL; wp = wp->w_next)
  4468. if (wp->w_buffer == buf)
  4469. break;
  4470. /* If the buffer already has a window, move it */
  4471. if (wp != NULL)
  4472. win_move_after(wp, curwin);
  4473. }
  4474. if (wp == NULL && split_ret == OK)
  4475. {
  4476. /* Split the window and put the buffer in it */
  4477. p_ea_save = p_ea;
  4478. p_ea = TRUE; /* use space from all windows */
  4479. split_ret = win_split(0, WSP_ROOM | WSP_BELOW);
  4480. ++open_wins;
  4481. p_ea = p_ea_save;
  4482. if (split_ret == FAIL)
  4483. continue;
  4484. /* Open the buffer in this window. */
  4485. #if defined(HAS_SWAP_EXISTS_ACTION)
  4486. swap_exists_action = SEA_DIALOG;
  4487. #endif
  4488. set_curbuf(buf, DOBUF_GOTO);
  4489. #ifdef FEAT_AUTOCMD
  4490. if (!buf_valid(buf)) /* autocommands deleted the buffer!!! */
  4491. {
  4492. #if defined(HAS_SWAP_EXISTS_ACTION)
  4493. swap_exists_action = SEA_NONE;
  4494. # endif
  4495. break;
  4496. }
  4497. #endif
  4498. #if defined(HAS_SWAP_EXISTS_ACTION)
  4499. if (swap_exists_action == SEA_QUIT)
  4500. {
  4501. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  4502. cleanup_T cs;
  4503. /* Reset the error/interrupt/exception state here so that
  4504. * aborting() returns FALSE when closing a window. */
  4505. enter_cleanup(&cs);
  4506. # endif
  4507. /* User selected Quit at ATTENTION prompt; close this window. */
  4508. win_close(curwin, TRUE);
  4509. --open_wins;
  4510. swap_exists_action = SEA_NONE;
  4511. swap_exists_did_quit = TRUE;
  4512. # if defined(FEAT_AUTOCMD) && defined(FEAT_EVAL)
  4513. /* Restore the error/interrupt/exception state if not
  4514. * discarded by a new aborting error, interrupt, or uncaught
  4515. * exception. */
  4516. leave_cleanup(&cs);
  4517. # endif
  4518. }
  4519. else
  4520. handle_swap_exists(NULL);
  4521. #endif
  4522. }
  4523. ui_breakcheck();
  4524. if (got_int)
  4525. {
  4526. (void)vgetc(); /* only break the file loading, not the rest */
  4527. break;
  4528. }
  4529. #ifdef FEAT_EVAL
  4530. /* Autocommands deleted the buffer or aborted script processing!!! */
  4531. if (aborting())
  4532. break;
  4533. #endif
  4534. #ifdef FEAT_WINDOWS
  4535. /* When ":tab" was used open a new tab for a new window repeatedly. */
  4536. if (had_tab > 0 && tabpage_index(NULL) <= p_tpm)
  4537. cmdmod.tab = 9999;
  4538. #endif
  4539. }
  4540. #ifdef FEAT_AUTOCMD
  4541. --autocmd_no_enter;
  4542. #endif
  4543. win_enter(firstwin, FALSE); /* back to first window */
  4544. #ifdef FEAT_AUTOCMD
  4545. --autocmd_no_leave;
  4546. #endif
  4547. /*
  4548. * Close superfluous windows.
  4549. */
  4550. for (wp = lastwin; open_wins > count; )
  4551. {
  4552. r = (P_HID(wp->w_buffer) || !bufIsChanged(wp->w_buffer)
  4553. || autowrite(wp->w_buffer, FALSE) == OK);
  4554. #ifdef FEAT_AUTOCMD
  4555. if (!win_valid(wp))
  4556. {
  4557. /* BufWrite Autocommands made the window invalid, start over */
  4558. wp = lastwin;
  4559. }
  4560. else
  4561. #endif
  4562. if (r)
  4563. {
  4564. win_close(wp, !P_HID(wp->w_buffer));
  4565. --open_wins;
  4566. wp = lastwin;
  4567. }
  4568. else
  4569. {
  4570. wp = wp->w_prev;
  4571. if (wp == NULL)
  4572. break;
  4573. }
  4574. }
  4575. }
  4576. # endif /* FEAT_LISTCMDS */
  4577. #endif /* FEAT_WINDOWS */
  4578. static int chk_modeline __ARGS((linenr_T, int));
  4579. /*
  4580. * do_modelines() - process mode lines for the current file
  4581. *
  4582. * "flags" can be:
  4583. * OPT_WINONLY only set options local to window
  4584. * OPT_NOWIN don't set options local to window
  4585. *
  4586. * Returns immediately if the "ml" option isn't set.
  4587. */
  4588. void
  4589. do_modelines(flags)
  4590. int flags;
  4591. {
  4592. linenr_T lnum;
  4593. int nmlines;
  4594. static int entered = 0;
  4595. if (!curbuf->b_p_ml || (nmlines = (int)p_mls) == 0)
  4596. return;
  4597. /* Disallow recursive entry here. Can happen when executing a modeline
  4598. * triggers an autocommand, which reloads modelines with a ":do". */
  4599. if (entered)
  4600. return;
  4601. ++entered;
  4602. for (lnum = 1; lnum <= curbuf->b_ml.ml_line_count && lnum <= nmlines;
  4603. ++lnum)
  4604. if (chk_modeline(lnum, flags) == FAIL)
  4605. nmlines = 0;
  4606. for (lnum = curbuf->b_ml.ml_line_count; lnum > 0 && lnum > nmlines
  4607. && lnum > curbuf->b_ml.ml_line_count - nmlines; --lnum)
  4608. if (chk_modeline(lnum, flags) == FAIL)
  4609. nmlines = 0;
  4610. --entered;
  4611. }
  4612. #include "version.h" /* for version number */
  4613. /*
  4614. * chk_modeline() - check a single line for a mode string
  4615. * Return FAIL if an error encountered.
  4616. */
  4617. static int
  4618. chk_modeline(lnum, flags)
  4619. linenr_T lnum;
  4620. int flags; /* Same as for do_modelines(). */
  4621. {
  4622. char_u *s;
  4623. char_u *e;
  4624. char_u *linecopy; /* local copy of any modeline found */
  4625. int prev;
  4626. int vers;
  4627. int end;
  4628. int retval = OK;
  4629. char_u *save_sourcing_name;
  4630. linenr_T save_sourcing_lnum;
  4631. #ifdef FEAT_EVAL
  4632. scid_T save_SID;
  4633. #endif
  4634. prev = -1;
  4635. for (s = ml_get(lnum); *s != NUL; ++s)
  4636. {
  4637. if (prev == -1 || vim_isspace(prev))
  4638. {
  4639. if ((prev != -1 && STRNCMP(s, "ex:", (size_t)3) == 0)
  4640. || STRNCMP(s, "vi:", (size_t)3) == 0)
  4641. break;
  4642. if (STRNCMP(s, "vim", 3) == 0)
  4643. {
  4644. if (s[3] == '<' || s[3] == '=' || s[3] == '>')
  4645. e = s + 4;
  4646. else
  4647. e = s + 3;
  4648. vers = getdigits(&e);
  4649. if (*e == ':'
  4650. && (s[3] == ':'
  4651. || (VIM_VERSION_100 >= vers && isdigit(s[3]))
  4652. || (VIM_VERSION_100 < vers && s[3] == '<')
  4653. || (VIM_VERSION_100 > vers && s[3] == '>')
  4654. || (VIM_VERSION_100 == vers && s[3] == '=')))
  4655. break;
  4656. }
  4657. }
  4658. prev = *s;
  4659. }
  4660. if (*s)
  4661. {
  4662. do /* skip over "ex:", "vi:" or "vim:" */
  4663. ++s;
  4664. while (s[-1] != ':');
  4665. s = linecopy = vim_strsave(s); /* copy the line, it will change */
  4666. if (linecopy == NULL)
  4667. return FAIL;
  4668. save_sourcing_lnum = sourcing_lnum;
  4669. save_sourcing_name = sourcing_name;
  4670. sourcing_lnum = lnum; /* prepare for emsg() */
  4671. sourcing_name = (char_u *)"modelines";
  4672. end = FALSE;
  4673. while (end == FALSE)
  4674. {
  4675. s = skipwhite(s);
  4676. if (*s == NUL)
  4677. break;
  4678. /*
  4679. * Find end of set command: ':' or end of line.
  4680. * Skip over "\:", replacing it with ":".
  4681. */
  4682. for (e = s; *e != ':' && *e != NUL; ++e)
  4683. if (e[0] == '\\' && e[1] == ':')
  4684. STRMOVE(e, e + 1);
  4685. if (*e == NUL)
  4686. end = TRUE;
  4687. /*
  4688. * If there is a "set" command, require a terminating ':' and
  4689. * ignore the stuff after the ':'.
  4690. * "vi:set opt opt opt: foo" -- foo not interpreted
  4691. * "vi:opt opt opt: foo" -- foo interpreted
  4692. * Accept "se" for compatibility with Elvis.
  4693. */
  4694. if (STRNCMP(s, "set ", (size_t)4) == 0
  4695. || STRNCMP(s, "se ", (size_t)3) == 0)
  4696. {
  4697. if (*e != ':') /* no terminating ':'? */
  4698. break;
  4699. end = TRUE;
  4700. s = vim_strchr(s, ' ') + 1;
  4701. }
  4702. *e = NUL; /* truncate the set command */
  4703. if (*s != NUL) /* skip over an empty "::" */
  4704. {
  4705. #ifdef FEAT_EVAL
  4706. save_SID = current_SID;
  4707. current_SID = SID_MODELINE;
  4708. #endif
  4709. retval = do_set(s, OPT_MODELINE | OPT_LOCAL | flags);
  4710. #ifdef FEAT_EVAL
  4711. current_SID = save_SID;
  4712. #endif
  4713. if (retval == FAIL) /* stop if error found */
  4714. break;
  4715. }
  4716. s = e + 1; /* advance to next part */
  4717. }
  4718. sourcing_lnum = save_sourcing_lnum;
  4719. sourcing_name = save_sourcing_name;
  4720. vim_free(linecopy);
  4721. }
  4722. return retval;
  4723. }
  4724. #if defined(FEAT_VIMINFO) || defined(PROTO)
  4725. int
  4726. read_viminfo_bufferlist(virp, writing)
  4727. vir_T *virp;
  4728. int writing;
  4729. {
  4730. char_u *tab;
  4731. linenr_T lnum;
  4732. colnr_T col;
  4733. buf_T *buf;
  4734. char_u *sfname;
  4735. char_u *xline;
  4736. /* Handle long line and escaped characters. */
  4737. xline = viminfo_readstring(virp, 1, FALSE);
  4738. /* don't read in if there are files on the command-line or if writing: */
  4739. if (xline != NULL && !writing && ARGCOUNT == 0
  4740. && find_viminfo_parameter('%') != NULL)
  4741. {
  4742. /* Format is: <fname> Tab <lnum> Tab <col>.
  4743. * Watch out for a Tab in the file name, work from the end. */
  4744. lnum = 0;
  4745. col = 0;
  4746. tab = vim_strrchr(xline, '\t');
  4747. if (tab != NULL)
  4748. {
  4749. *tab++ = '\0';
  4750. col = (colnr_T)atoi((char *)tab);
  4751. tab = vim_strrchr(xline, '\t');
  4752. if (tab != NULL)
  4753. {
  4754. *tab++ = '\0';
  4755. lnum = atol((char *)tab);
  4756. }
  4757. }
  4758. /* Expand "~/" in the file name at "line + 1" to a full path.
  4759. * Then try shortening it by comparing with the current directory */
  4760. expand_env(xline, NameBuff, MAXPATHL);
  4761. sfname = shorten_fname1(NameBuff);
  4762. buf = buflist_new(NameBuff, sfname, (linenr_T)0, BLN_LISTED);
  4763. if (buf != NULL) /* just in case... */
  4764. {
  4765. buf->b_last_cursor.lnum = lnum;
  4766. buf->b_last_cursor.col = col;
  4767. buflist_setfpos(buf, curwin, lnum, col, FALSE);
  4768. }
  4769. }
  4770. vim_free(xline);
  4771. return viminfo_readline(virp);
  4772. }
  4773. void
  4774. write_viminfo_bufferlist(fp)
  4775. FILE *fp;
  4776. {
  4777. buf_T *buf;
  4778. #ifdef FEAT_WINDOWS
  4779. win_T *win;
  4780. tabpage_T *tp;
  4781. #endif
  4782. char_u *line;
  4783. int max_buffers;
  4784. if (find_viminfo_parameter('%') == NULL)
  4785. return;
  4786. /* Without a number -1 is returned: do all buffers. */
  4787. max_buffers = get_viminfo_parameter('%');
  4788. /* Allocate room for the file name, lnum and col. */
  4789. #define LINE_BUF_LEN (MAXPATHL + 40)
  4790. line = alloc(LINE_BUF_LEN);
  4791. if (line == NULL)
  4792. return;
  4793. #ifdef FEAT_WINDOWS
  4794. FOR_ALL_TAB_WINDOWS(tp, win)
  4795. set_last_cursor(win);
  4796. #else
  4797. set_last_cursor(curwin);
  4798. #endif
  4799. fputs(_("\n# Buffer list:\n"), fp);
  4800. for (buf = firstbuf; buf != NULL ; buf = buf->b_next)
  4801. {
  4802. if (buf->b_fname == NULL
  4803. || !buf->b_p_bl
  4804. #ifdef FEAT_QUICKFIX
  4805. || bt_quickfix(buf)
  4806. #endif
  4807. || removable(buf->b_ffname))
  4808. continue;
  4809. if (max_buffers-- == 0)
  4810. break;
  4811. putc('%', fp);
  4812. home_replace(NULL, buf->b_ffname, line, MAXPATHL, TRUE);
  4813. vim_snprintf_add((char *)line, LINE_BUF_LEN, "\t%ld\t%d",
  4814. (long)buf->b_last_cursor.lnum,
  4815. buf->b_last_cursor.col);
  4816. viminfo_writestring(fp, line);
  4817. }
  4818. vim_free(line);
  4819. }
  4820. #endif
  4821. /*
  4822. * Return special buffer name.
  4823. * Returns NULL when the buffer has a normal file name.
  4824. */
  4825. char *
  4826. buf_spname(buf)
  4827. buf_T *buf;
  4828. {
  4829. #if defined(FEAT_QUICKFIX) && defined(FEAT_WINDOWS)
  4830. if (bt_quickfix(buf))
  4831. {
  4832. win_T *win = NULL;
  4833. tabpage_T *tp;
  4834. /*
  4835. * For location list window, w_llist_ref points to the location list.
  4836. * For quickfix window, w_llist_ref is NULL.
  4837. */
  4838. FOR_ALL_TAB_WINDOWS(tp, win)
  4839. if (win->w_buffer == buf)
  4840. goto win_found;
  4841. win_found:
  4842. if (win != NULL && win->w_llist_ref != NULL)
  4843. return _(msg_loclist);
  4844. else
  4845. return _(msg_qflist);
  4846. }
  4847. #endif
  4848. #ifdef FEAT_QUICKFIX
  4849. /* There is no _file_ when 'buftype' is "nofile", b_sfname
  4850. * contains the name as specified by the user */
  4851. if (bt_nofile(buf))
  4852. {
  4853. if (buf->b_sfname != NULL)
  4854. return (char *)buf->b_sfname;
  4855. return _("[Scratch]");
  4856. }
  4857. #endif
  4858. if (buf->b_fname == NULL)
  4859. return _("[No Name]");
  4860. return NULL;
  4861. }
  4862. #if defined(FEAT_SIGNS) || defined(PROTO)
  4863. /*
  4864. * Insert the sign into the signlist.
  4865. */
  4866. static void
  4867. insert_sign(buf, prev, next, id, lnum, typenr)
  4868. buf_T *buf; /* buffer to store sign in */
  4869. signlist_T *prev; /* previous sign entry */
  4870. signlist_T *next; /* next sign entry */
  4871. int id; /* sign ID */
  4872. linenr_T lnum; /* line number which gets the mark */
  4873. int typenr; /* typenr of sign we are adding */
  4874. {
  4875. signlist_T *newsign;
  4876. newsign = (signlist_T *)lalloc((long_u)sizeof(signlist_T), FALSE);
  4877. if (newsign != NULL)
  4878. {
  4879. newsign->id = id;
  4880. newsign->lnum = lnum;
  4881. newsign->typenr = typenr;
  4882. newsign->next = next;
  4883. #ifdef FEAT_NETBEANS_INTG
  4884. newsign->prev = prev;
  4885. if (next != NULL)
  4886. next->prev = newsign;
  4887. #endif
  4888. if (prev == NULL)
  4889. {
  4890. /* When adding first sign need to redraw the windows to create the
  4891. * column for signs. */
  4892. if (buf->b_signlist == NULL)
  4893. {
  4894. redraw_buf_later(buf, NOT_VALID);
  4895. changed_cline_bef_curs();
  4896. }
  4897. /* first sign in signlist */
  4898. buf->b_signlist = newsign;
  4899. }
  4900. else
  4901. prev->next = newsign;
  4902. }
  4903. }
  4904. /*
  4905. * Add the sign into the signlist. Find the right spot to do it though.
  4906. */
  4907. void
  4908. buf_addsign(buf, id, lnum, typenr)
  4909. buf_T *buf; /* buffer to store sign in */
  4910. int id; /* sign ID */
  4911. linenr_T lnum; /* line number which gets the mark */
  4912. int typenr; /* typenr of sign we are adding */
  4913. {
  4914. signlist_T *sign; /* a sign in the signlist */
  4915. signlist_T *prev; /* the previous sign */
  4916. prev = NULL;
  4917. for (sign = buf->b_signlist; sign != NULL; sign = sign->next)
  4918. {
  4919. if (lnum == sign->lnum && id == sign->id)
  4920. {
  4921. sign->typenr = typenr;
  4922. return;
  4923. }
  4924. else if (
  4925. #ifndef FEAT_NETBEANS_INTG /* keep signs sorted by lnum */
  4926. id < 0 &&
  4927. #endif
  4928. lnum < sign->lnum)
  4929. {
  4930. #ifdef FEAT_NETBEANS_INTG /* insert new sign at head of list for this lnum */
  4931. /* XXX - GRP: Is this because of sign slide problem? Or is it
  4932. * really needed? Or is it because we allow multiple signs per
  4933. * line? If so, should I add that feature to FEAT_SIGNS?
  4934. */
  4935. while (prev != NULL && prev->lnum == lnum)
  4936. prev = prev->prev;
  4937. if (prev == NULL)
  4938. sign = buf->b_signlist;
  4939. else
  4940. sign = prev->next;
  4941. #endif
  4942. insert_sign(buf, prev, sign, id, lnum, typenr);
  4943. return;
  4944. }
  4945. prev = sign;
  4946. }
  4947. #ifdef FEAT_NETBEANS_INTG /* insert new sign at head of list for this lnum */
  4948. /* XXX - GRP: See previous comment */
  4949. while (prev != NULL && prev->lnum == lnum)
  4950. prev = prev->prev;
  4951. if (prev == NULL)
  4952. sign = buf->b_signlist;
  4953. else
  4954. sign = prev->next;
  4955. #endif
  4956. insert_sign(buf, prev, sign, id, lnum, typenr);
  4957. return;
  4958. }
  4959. linenr_T
  4960. buf_change_sign_type(buf, markId, typenr)
  4961. buf_T *buf; /* buffer to store sign in */
  4962. int markId; /* sign ID */
  4963. int typenr; /* typenr of sign we are adding */
  4964. {
  4965. signlist_T *sign; /* a sign in the signlist */
  4966. for (sign = buf->b_signlist; sign != NULL; sign = sign->next)
  4967. {
  4968. if (sign->id == markId)
  4969. {
  4970. sign->typenr = typenr;
  4971. return sign->lnum;
  4972. }
  4973. }
  4974. return (linenr_T)0;
  4975. }
  4976. int
  4977. buf_getsigntype(buf, lnum, type)
  4978. buf_T *buf;
  4979. linenr_T lnum;
  4980. int type; /* SIGN_ICON, SIGN_TEXT, SIGN_ANY, SIGN_LINEHL */
  4981. {
  4982. signlist_T *sign; /* a sign in a b_signlist */
  4983. for (sign = buf->b_signlist; sign != NULL; sign = sign->next)
  4984. if (sign->lnum == lnum
  4985. && (type == SIGN_ANY
  4986. # ifdef FEAT_SIGN_ICONS
  4987. || (type == SIGN_ICON
  4988. && sign_get_image(sign->typenr) != NULL)
  4989. # endif
  4990. || (type == SIGN_TEXT
  4991. && sign_get_text(sign->typenr) != NULL)
  4992. || (type == SIGN_LINEHL
  4993. && sign_get_attr(sign->typenr, TRUE) != 0)))
  4994. return sign->typenr;
  4995. return 0;
  4996. }
  4997. linenr_T
  4998. buf_delsign(buf, id)
  4999. buf_T *buf; /* buffer sign is stored in */
  5000. int id; /* sign id */
  5001. {
  5002. signlist_T **lastp; /* pointer to pointer to current sign */
  5003. signlist_T *sign; /* a sign in a b_signlist */
  5004. signlist_T *next; /* the next sign in a b_signlist */
  5005. linenr_T lnum; /* line number whose sign was deleted */
  5006. lastp = &buf->b_signlist;
  5007. lnum = 0;
  5008. for (sign = buf->b_signlist; sign != NULL; sign = next)
  5009. {
  5010. next = sign->next;
  5011. if (sign->id == id)
  5012. {
  5013. *lastp = next;
  5014. #ifdef FEAT_NETBEANS_INTG
  5015. if (next != NULL)
  5016. next->prev = sign->prev;
  5017. #endif
  5018. lnum = sign->lnum;
  5019. vim_free(sign);
  5020. break;
  5021. }
  5022. else
  5023. lastp = &sign->next;
  5024. }
  5025. /* When deleted the last sign need to redraw the windows to remove the
  5026. * sign column. */
  5027. if (buf->b_signlist == NULL)
  5028. {
  5029. redraw_buf_later(buf, NOT_VALID);
  5030. changed_cline_bef_curs();
  5031. }
  5032. return lnum;
  5033. }
  5034. /*
  5035. * Find the line number of the sign with the requested id. If the sign does
  5036. * not exist, return 0 as the line number. This will still let the correct file
  5037. * get loaded.
  5038. */
  5039. int
  5040. buf_findsign(buf, id)
  5041. buf_T *buf; /* buffer to store sign in */
  5042. int id; /* sign ID */
  5043. {
  5044. signlist_T *sign; /* a sign in the signlist */
  5045. for (sign = buf->b_signlist; sign != NULL; sign = sign->next)
  5046. if (sign->id == id)
  5047. return sign->lnum;
  5048. return 0;
  5049. }
  5050. int
  5051. buf_findsign_id(buf, lnum)
  5052. buf_T *buf; /* buffer whose sign we are searching for */
  5053. linenr_T lnum; /* line number of sign */
  5054. {
  5055. signlist_T *sign; /* a sign in the signlist */
  5056. for (sign = buf->b_signlist; sign != NULL; sign = sign->next)
  5057. if (sign->lnum == lnum)
  5058. return sign->id;
  5059. return 0;
  5060. }
  5061. # if defined(FEAT_NETBEANS_INTG) || defined(PROTO)
  5062. /* see if a given type of sign exists on a specific line */
  5063. int
  5064. buf_findsigntype_id(buf, lnum, typenr)
  5065. buf_T *buf; /* buffer whose sign we are searching for */
  5066. linenr_T lnum; /* line number of sign */
  5067. int typenr; /* sign type number */
  5068. {
  5069. signlist_T *sign; /* a sign in the signlist */
  5070. for (sign = buf->b_signlist; sign != NULL; sign = sign->next)
  5071. if (sign->lnum == lnum && sign->typenr == typenr)
  5072. return sign->id;
  5073. return 0;
  5074. }
  5075. # if defined(FEAT_SIGN_ICONS) || defined(PROTO)
  5076. /* return the number of icons on the given line */
  5077. int
  5078. buf_signcount(buf, lnum)
  5079. buf_T *buf;
  5080. linenr_T lnum;
  5081. {
  5082. signlist_T *sign; /* a sign in the signlist */
  5083. int count = 0;
  5084. for (sign = buf->b_signlist; sign != NULL; sign = sign->next)
  5085. if (sign->lnum == lnum)
  5086. if (sign_get_image(sign->typenr) != NULL)
  5087. count++;
  5088. return count;
  5089. }
  5090. # endif /* FEAT_SIGN_ICONS */
  5091. # endif /* FEAT_NETBEANS_INTG */
  5092. /*
  5093. * Delete signs in buffer "buf".
  5094. */
  5095. static void
  5096. buf_delete_signs(buf)
  5097. buf_T *buf;
  5098. {
  5099. signlist_T *next;
  5100. while (buf->b_signlist != NULL)
  5101. {
  5102. next = buf->b_signlist->next;
  5103. vim_free(buf->b_signlist);
  5104. buf->b_signlist = next;
  5105. }
  5106. }
  5107. /*
  5108. * Delete all signs in all buffers.
  5109. */
  5110. void
  5111. buf_delete_all_signs()
  5112. {
  5113. buf_T *buf; /* buffer we are checking for signs */
  5114. for (buf = firstbuf; buf != NULL; buf = buf->b_next)
  5115. if (buf->b_signlist != NULL)
  5116. {
  5117. /* Need to redraw the windows to remove the sign column. */
  5118. redraw_buf_later(buf, NOT_VALID);
  5119. buf_delete_signs(buf);
  5120. }
  5121. }
  5122. /*
  5123. * List placed signs for "rbuf". If "rbuf" is NULL do it for all buffers.
  5124. */
  5125. void
  5126. sign_list_placed(rbuf)
  5127. buf_T *rbuf;
  5128. {
  5129. buf_T *buf;
  5130. signlist_T *p;
  5131. char lbuf[BUFSIZ];
  5132. MSG_PUTS_TITLE(_("\n--- Signs ---"));
  5133. msg_putchar('\n');
  5134. if (rbuf == NULL)
  5135. buf = firstbuf;
  5136. else
  5137. buf = rbuf;
  5138. while (buf != NULL && !got_int)
  5139. {
  5140. if (buf->b_signlist != NULL)
  5141. {
  5142. vim_snprintf(lbuf, BUFSIZ, _("Signs for %s:"), buf->b_fname);
  5143. MSG_PUTS_ATTR(lbuf, hl_attr(HLF_D));
  5144. msg_putchar('\n');
  5145. }
  5146. for (p = buf->b_signlist; p != NULL && !got_int; p = p->next)
  5147. {
  5148. vim_snprintf(lbuf, BUFSIZ, _(" line=%ld id=%d name=%s"),
  5149. (long)p->lnum, p->id, sign_typenr2name(p->typenr));
  5150. MSG_PUTS(lbuf);
  5151. msg_putchar('\n');
  5152. }
  5153. if (rbuf != NULL)
  5154. break;
  5155. buf = buf->b_next;
  5156. }
  5157. }
  5158. /*
  5159. * Adjust a placed sign for inserted/deleted lines.
  5160. */
  5161. void
  5162. sign_mark_adjust(line1, line2, amount, amount_after)
  5163. linenr_T line1;
  5164. linenr_T line2;
  5165. long amount;
  5166. long amount_after;
  5167. {
  5168. signlist_T *sign; /* a sign in a b_signlist */
  5169. for (sign = curbuf->b_signlist; sign != NULL; sign = sign->next)
  5170. {
  5171. if (sign->lnum >= line1 && sign->lnum <= line2)
  5172. {
  5173. if (amount == MAXLNUM)
  5174. sign->lnum = line1;
  5175. else
  5176. sign->lnum += amount;
  5177. }
  5178. else if (sign->lnum > line2)
  5179. sign->lnum += amount_after;
  5180. }
  5181. }
  5182. #endif /* FEAT_SIGNS */
  5183. /*
  5184. * Set 'buflisted' for curbuf to "on" and trigger autocommands if it changed.
  5185. */
  5186. void
  5187. set_buflisted(on)
  5188. int on;
  5189. {
  5190. if (on != curbuf->b_p_bl)
  5191. {
  5192. curbuf->b_p_bl = on;
  5193. #ifdef FEAT_AUTOCMD
  5194. if (on)
  5195. apply_autocmds(EVENT_BUFADD, NULL, NULL, FALSE, curbuf);
  5196. else
  5197. apply_autocmds(EVENT_BUFDELETE, NULL, NULL, FALSE, curbuf);
  5198. #endif
  5199. }
  5200. }
  5201. /*
  5202. * Read the file for "buf" again and check if the contents changed.
  5203. * Return TRUE if it changed or this could not be checked.
  5204. */
  5205. int
  5206. buf_contents_changed(buf)
  5207. buf_T *buf;
  5208. {
  5209. buf_T *newbuf;
  5210. int differ = TRUE;
  5211. linenr_T lnum;
  5212. aco_save_T aco;
  5213. exarg_T ea;
  5214. /* Allocate a buffer without putting it in the buffer list. */
  5215. newbuf = buflist_new(NULL, NULL, (linenr_T)1, BLN_DUMMY);
  5216. if (newbuf == NULL)
  5217. return TRUE;
  5218. /* Force the 'fileencoding' and 'fileformat' to be equal. */
  5219. if (prep_exarg(&ea, buf) == FAIL)
  5220. {
  5221. wipe_buffer(newbuf, FALSE);
  5222. return TRUE;
  5223. }
  5224. /* set curwin/curbuf to buf and save a few things */
  5225. aucmd_prepbuf(&aco, newbuf);
  5226. if (ml_open(curbuf) == OK
  5227. && readfile(buf->b_ffname, buf->b_fname,
  5228. (linenr_T)0, (linenr_T)0, (linenr_T)MAXLNUM,
  5229. &ea, READ_NEW | READ_DUMMY) == OK)
  5230. {
  5231. /* compare the two files line by line */
  5232. if (buf->b_ml.ml_line_count == curbuf->b_ml.ml_line_count)
  5233. {
  5234. differ = FALSE;
  5235. for (lnum = 1; lnum <= curbuf->b_ml.ml_line_count; ++lnum)
  5236. if (STRCMP(ml_get_buf(buf, lnum, FALSE), ml_get(lnum)) != 0)
  5237. {
  5238. differ = TRUE;
  5239. break;
  5240. }
  5241. }
  5242. }
  5243. vim_free(ea.cmd);
  5244. /* restore curwin/curbuf and a few other things */
  5245. aucmd_restbuf(&aco);
  5246. if (curbuf != newbuf) /* safety check */
  5247. wipe_buffer(newbuf, FALSE);
  5248. return differ;
  5249. }
  5250. /*
  5251. * Wipe out a buffer and decrement the last buffer number if it was used for
  5252. * this buffer. Call this to wipe out a temp buffer that does not contain any
  5253. * marks.
  5254. */
  5255. void
  5256. wipe_buffer(buf, aucmd)
  5257. buf_T *buf;
  5258. int aucmd UNUSED; /* When TRUE trigger autocommands. */
  5259. {
  5260. if (buf->b_fnum == top_file_num - 1)
  5261. --top_file_num;
  5262. #ifdef FEAT_AUTOCMD
  5263. if (!aucmd) /* Don't trigger BufDelete autocommands here. */
  5264. block_autocmds();
  5265. #endif
  5266. close_buffer(NULL, buf, DOBUF_WIPE, FALSE);
  5267. #ifdef FEAT_AUTOCMD
  5268. if (!aucmd)
  5269. unblock_autocmds();
  5270. #endif
  5271. }