1261f346fSBram Moolenaar /* vi:set ts=8 sts=4 sw=4 noet: 2261f346fSBram Moolenaar * 3261f346fSBram Moolenaar * VIM - Vi IMproved by Bram Moolenaar 4261f346fSBram Moolenaar * 5261f346fSBram Moolenaar * Do ":help uganda" in Vim to read copying and usage conditions. 6261f346fSBram Moolenaar * Do ":help credits" in Vim to see a list of people who contributed. 7261f346fSBram Moolenaar * See README.txt for an overview of the Vim source code. 8261f346fSBram Moolenaar */ 9261f346fSBram Moolenaar 10261f346fSBram Moolenaar /* 11261f346fSBram Moolenaar * evalbuffer.c: Buffer related builtin functions 12261f346fSBram Moolenaar */ 13261f346fSBram Moolenaar 14261f346fSBram Moolenaar #include "vim.h" 15261f346fSBram Moolenaar 16261f346fSBram Moolenaar #if defined(FEAT_EVAL) || defined(PROTO) 17261f346fSBram Moolenaar /* 18261f346fSBram Moolenaar * Mark references in functions of buffers. 19261f346fSBram Moolenaar */ 20261f346fSBram Moolenaar int 21261f346fSBram Moolenaar set_ref_in_buffers(int copyID) 22261f346fSBram Moolenaar { 23261f346fSBram Moolenaar int abort = FALSE; 24261f346fSBram Moolenaar buf_T *bp; 25261f346fSBram Moolenaar 26261f346fSBram Moolenaar FOR_ALL_BUFFERS(bp) 27261f346fSBram Moolenaar { 28261f346fSBram Moolenaar listener_T *lnr; 29261f346fSBram Moolenaar typval_T tv; 30261f346fSBram Moolenaar 31261f346fSBram Moolenaar for (lnr = bp->b_listener; !abort && lnr != NULL; lnr = lnr->lr_next) 32261f346fSBram Moolenaar { 33261f346fSBram Moolenaar if (lnr->lr_callback.cb_partial != NULL) 34261f346fSBram Moolenaar { 35261f346fSBram Moolenaar tv.v_type = VAR_PARTIAL; 36261f346fSBram Moolenaar tv.vval.v_partial = lnr->lr_callback.cb_partial; 37261f346fSBram Moolenaar abort = abort || set_ref_in_item(&tv, copyID, NULL, NULL); 38261f346fSBram Moolenaar } 39261f346fSBram Moolenaar } 40261f346fSBram Moolenaar # ifdef FEAT_JOB_CHANNEL 41261f346fSBram Moolenaar if (!abort && bp->b_prompt_callback.cb_partial != NULL) 42261f346fSBram Moolenaar { 43261f346fSBram Moolenaar tv.v_type = VAR_PARTIAL; 44261f346fSBram Moolenaar tv.vval.v_partial = bp->b_prompt_callback.cb_partial; 45261f346fSBram Moolenaar abort = abort || set_ref_in_item(&tv, copyID, NULL, NULL); 46261f346fSBram Moolenaar } 47261f346fSBram Moolenaar if (!abort && bp->b_prompt_interrupt.cb_partial != NULL) 48261f346fSBram Moolenaar { 49261f346fSBram Moolenaar tv.v_type = VAR_PARTIAL; 50261f346fSBram Moolenaar tv.vval.v_partial = bp->b_prompt_interrupt.cb_partial; 51261f346fSBram Moolenaar abort = abort || set_ref_in_item(&tv, copyID, NULL, NULL); 52261f346fSBram Moolenaar } 53261f346fSBram Moolenaar # endif 54261f346fSBram Moolenaar if (abort) 55261f346fSBram Moolenaar break; 56261f346fSBram Moolenaar } 57261f346fSBram Moolenaar return abort; 58261f346fSBram Moolenaar } 59261f346fSBram Moolenaar 60261f346fSBram Moolenaar buf_T * 61261f346fSBram Moolenaar buflist_find_by_name(char_u *name, int curtab_only) 62261f346fSBram Moolenaar { 63261f346fSBram Moolenaar int save_magic; 64261f346fSBram Moolenaar char_u *save_cpo; 65261f346fSBram Moolenaar buf_T *buf; 66261f346fSBram Moolenaar 67261f346fSBram Moolenaar // Ignore 'magic' and 'cpoptions' here to make scripts portable 68261f346fSBram Moolenaar save_magic = p_magic; 69261f346fSBram Moolenaar p_magic = TRUE; 70261f346fSBram Moolenaar save_cpo = p_cpo; 71261f346fSBram Moolenaar p_cpo = (char_u *)""; 72261f346fSBram Moolenaar 73261f346fSBram Moolenaar buf = buflist_findnr(buflist_findpat(name, name + STRLEN(name), 74261f346fSBram Moolenaar TRUE, FALSE, curtab_only)); 75261f346fSBram Moolenaar 76261f346fSBram Moolenaar p_magic = save_magic; 77261f346fSBram Moolenaar p_cpo = save_cpo; 78261f346fSBram Moolenaar return buf; 79261f346fSBram Moolenaar } 80261f346fSBram Moolenaar 81261f346fSBram Moolenaar /* 82261f346fSBram Moolenaar * Find a buffer by number or exact name. 83261f346fSBram Moolenaar */ 84261f346fSBram Moolenaar buf_T * 85261f346fSBram Moolenaar find_buffer(typval_T *avar) 86261f346fSBram Moolenaar { 87261f346fSBram Moolenaar buf_T *buf = NULL; 88261f346fSBram Moolenaar 89261f346fSBram Moolenaar if (avar->v_type == VAR_NUMBER) 90261f346fSBram Moolenaar buf = buflist_findnr((int)avar->vval.v_number); 91261f346fSBram Moolenaar else if (avar->v_type == VAR_STRING && avar->vval.v_string != NULL) 92261f346fSBram Moolenaar { 93261f346fSBram Moolenaar buf = buflist_findname_exp(avar->vval.v_string); 94261f346fSBram Moolenaar if (buf == NULL) 95261f346fSBram Moolenaar { 96261f346fSBram Moolenaar // No full path name match, try a match with a URL or a "nofile" 97261f346fSBram Moolenaar // buffer, these don't use the full path. 98261f346fSBram Moolenaar FOR_ALL_BUFFERS(buf) 99261f346fSBram Moolenaar if (buf->b_fname != NULL 100261f346fSBram Moolenaar && (path_with_url(buf->b_fname) 101261f346fSBram Moolenaar #ifdef FEAT_QUICKFIX 102261f346fSBram Moolenaar || bt_nofilename(buf) 103261f346fSBram Moolenaar #endif 104261f346fSBram Moolenaar ) 105261f346fSBram Moolenaar && STRCMP(buf->b_fname, avar->vval.v_string) == 0) 106261f346fSBram Moolenaar break; 107261f346fSBram Moolenaar } 108261f346fSBram Moolenaar } 109261f346fSBram Moolenaar return buf; 110261f346fSBram Moolenaar } 111261f346fSBram Moolenaar 112261f346fSBram Moolenaar /* 113261f346fSBram Moolenaar * If there is a window for "curbuf", make it the current window. 114261f346fSBram Moolenaar */ 115261f346fSBram Moolenaar static void 116261f346fSBram Moolenaar find_win_for_curbuf(void) 117261f346fSBram Moolenaar { 118261f346fSBram Moolenaar wininfo_T *wip; 119261f346fSBram Moolenaar 120aeea7215SBram Moolenaar FOR_ALL_BUF_WININFO(curbuf, wip) 121261f346fSBram Moolenaar { 122261f346fSBram Moolenaar if (wip->wi_win != NULL) 123261f346fSBram Moolenaar { 124261f346fSBram Moolenaar curwin = wip->wi_win; 125261f346fSBram Moolenaar break; 126261f346fSBram Moolenaar } 127261f346fSBram Moolenaar } 128261f346fSBram Moolenaar } 129261f346fSBram Moolenaar 130261f346fSBram Moolenaar /* 131261f346fSBram Moolenaar * Set line or list of lines in buffer "buf". 132261f346fSBram Moolenaar */ 133261f346fSBram Moolenaar static void 134261f346fSBram Moolenaar set_buffer_lines( 135261f346fSBram Moolenaar buf_T *buf, 136261f346fSBram Moolenaar linenr_T lnum_arg, 137261f346fSBram Moolenaar int append, 138261f346fSBram Moolenaar typval_T *lines, 139261f346fSBram Moolenaar typval_T *rettv) 140261f346fSBram Moolenaar { 141261f346fSBram Moolenaar linenr_T lnum = lnum_arg + (append ? 1 : 0); 142261f346fSBram Moolenaar char_u *line = NULL; 143261f346fSBram Moolenaar list_T *l = NULL; 144261f346fSBram Moolenaar listitem_T *li = NULL; 145261f346fSBram Moolenaar long added = 0; 146261f346fSBram Moolenaar linenr_T append_lnum; 147261f346fSBram Moolenaar buf_T *curbuf_save = NULL; 148261f346fSBram Moolenaar win_T *curwin_save = NULL; 149261f346fSBram Moolenaar int is_curbuf = buf == curbuf; 150261f346fSBram Moolenaar 151261f346fSBram Moolenaar // When using the current buffer ml_mfp will be set if needed. Useful when 152261f346fSBram Moolenaar // setline() is used on startup. For other buffers the buffer must be 153261f346fSBram Moolenaar // loaded. 154261f346fSBram Moolenaar if (buf == NULL || (!is_curbuf && buf->b_ml.ml_mfp == NULL) || lnum < 1) 155261f346fSBram Moolenaar { 156261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 157261f346fSBram Moolenaar return; 158261f346fSBram Moolenaar } 159261f346fSBram Moolenaar 160261f346fSBram Moolenaar if (!is_curbuf) 161261f346fSBram Moolenaar { 162261f346fSBram Moolenaar curbuf_save = curbuf; 163261f346fSBram Moolenaar curwin_save = curwin; 164261f346fSBram Moolenaar curbuf = buf; 165261f346fSBram Moolenaar find_win_for_curbuf(); 166261f346fSBram Moolenaar } 167261f346fSBram Moolenaar 168261f346fSBram Moolenaar if (append) 169261f346fSBram Moolenaar // appendbufline() uses the line number below which we insert 170261f346fSBram Moolenaar append_lnum = lnum - 1; 171261f346fSBram Moolenaar else 172261f346fSBram Moolenaar // setbufline() uses the line number above which we insert, we only 173261f346fSBram Moolenaar // append if it's below the last line 174261f346fSBram Moolenaar append_lnum = curbuf->b_ml.ml_line_count; 175261f346fSBram Moolenaar 176261f346fSBram Moolenaar if (lines->v_type == VAR_LIST) 177261f346fSBram Moolenaar { 178261f346fSBram Moolenaar l = lines->vval.v_list; 179ad48e6c1SBram Moolenaar if (l == NULL || list_len(l) == 0) 180ad48e6c1SBram Moolenaar { 181ad48e6c1SBram Moolenaar // set proper return code 182ad48e6c1SBram Moolenaar if (lnum > curbuf->b_ml.ml_line_count) 183ad48e6c1SBram Moolenaar rettv->vval.v_number = 1; // FAIL 184ad48e6c1SBram Moolenaar goto done; 185ad48e6c1SBram Moolenaar } 1867e9f351bSBram Moolenaar CHECK_LIST_MATERIALIZE(l); 187261f346fSBram Moolenaar li = l->lv_first; 188261f346fSBram Moolenaar } 189261f346fSBram Moolenaar else 190261f346fSBram Moolenaar line = tv_get_string_chk(lines); 191261f346fSBram Moolenaar 192261f346fSBram Moolenaar // default result is zero == OK 193261f346fSBram Moolenaar for (;;) 194261f346fSBram Moolenaar { 195261f346fSBram Moolenaar if (l != NULL) 196261f346fSBram Moolenaar { 197261f346fSBram Moolenaar // list argument, get next string 198261f346fSBram Moolenaar if (li == NULL) 199261f346fSBram Moolenaar break; 200261f346fSBram Moolenaar line = tv_get_string_chk(&li->li_tv); 201261f346fSBram Moolenaar li = li->li_next; 202261f346fSBram Moolenaar } 203261f346fSBram Moolenaar 204261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 205261f346fSBram Moolenaar if (line == NULL || lnum > curbuf->b_ml.ml_line_count + 1) 206261f346fSBram Moolenaar break; 207261f346fSBram Moolenaar 208261f346fSBram Moolenaar // When coming here from Insert mode, sync undo, so that this can be 209261f346fSBram Moolenaar // undone separately from what was previously inserted. 210261f346fSBram Moolenaar if (u_sync_once == 2) 211261f346fSBram Moolenaar { 212261f346fSBram Moolenaar u_sync_once = 1; // notify that u_sync() was called 213261f346fSBram Moolenaar u_sync(TRUE); 214261f346fSBram Moolenaar } 215261f346fSBram Moolenaar 216261f346fSBram Moolenaar if (!append && lnum <= curbuf->b_ml.ml_line_count) 217261f346fSBram Moolenaar { 218261f346fSBram Moolenaar // Existing line, replace it. 219261f346fSBram Moolenaar // Removes any existing text properties. 220261f346fSBram Moolenaar if (u_savesub(lnum) == OK && ml_replace_len( 221261f346fSBram Moolenaar lnum, line, (colnr_T)STRLEN(line) + 1, TRUE, TRUE) == OK) 222261f346fSBram Moolenaar { 223261f346fSBram Moolenaar changed_bytes(lnum, 0); 224261f346fSBram Moolenaar if (is_curbuf && lnum == curwin->w_cursor.lnum) 225261f346fSBram Moolenaar check_cursor_col(); 226261f346fSBram Moolenaar rettv->vval.v_number = 0; // OK 227261f346fSBram Moolenaar } 228261f346fSBram Moolenaar } 229261f346fSBram Moolenaar else if (added > 0 || u_save(lnum - 1, lnum) == OK) 230261f346fSBram Moolenaar { 231261f346fSBram Moolenaar // append the line 232261f346fSBram Moolenaar ++added; 233261f346fSBram Moolenaar if (ml_append(lnum - 1, line, (colnr_T)0, FALSE) == OK) 234261f346fSBram Moolenaar rettv->vval.v_number = 0; // OK 235261f346fSBram Moolenaar } 236261f346fSBram Moolenaar 237261f346fSBram Moolenaar if (l == NULL) // only one string argument 238261f346fSBram Moolenaar break; 239261f346fSBram Moolenaar ++lnum; 240261f346fSBram Moolenaar } 241261f346fSBram Moolenaar 242261f346fSBram Moolenaar if (added > 0) 243261f346fSBram Moolenaar { 244261f346fSBram Moolenaar win_T *wp; 245261f346fSBram Moolenaar tabpage_T *tp; 246261f346fSBram Moolenaar 247261f346fSBram Moolenaar appended_lines_mark(append_lnum, added); 248261f346fSBram Moolenaar 249261f346fSBram Moolenaar // Only adjust the cursor for buffers other than the current, unless it 250261f346fSBram Moolenaar // is the current window. For curbuf and other windows it has been 251261f346fSBram Moolenaar // done in mark_adjust_internal(). 252261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 253261f346fSBram Moolenaar if (wp->w_buffer == buf 254261f346fSBram Moolenaar && (wp->w_buffer != curbuf || wp == curwin) 255261f346fSBram Moolenaar && wp->w_cursor.lnum > append_lnum) 256261f346fSBram Moolenaar wp->w_cursor.lnum += added; 257261f346fSBram Moolenaar check_cursor_col(); 258261f346fSBram Moolenaar update_topline(); 259261f346fSBram Moolenaar } 260261f346fSBram Moolenaar 261ad48e6c1SBram Moolenaar done: 262261f346fSBram Moolenaar if (!is_curbuf) 263261f346fSBram Moolenaar { 264261f346fSBram Moolenaar curbuf = curbuf_save; 265261f346fSBram Moolenaar curwin = curwin_save; 266261f346fSBram Moolenaar } 267261f346fSBram Moolenaar } 268261f346fSBram Moolenaar 269261f346fSBram Moolenaar /* 270261f346fSBram Moolenaar * "append(lnum, string/list)" function 271261f346fSBram Moolenaar */ 272261f346fSBram Moolenaar void 273261f346fSBram Moolenaar f_append(typval_T *argvars, typval_T *rettv) 274261f346fSBram Moolenaar { 275261f346fSBram Moolenaar linenr_T lnum = tv_get_lnum(&argvars[0]); 276261f346fSBram Moolenaar 277261f346fSBram Moolenaar set_buffer_lines(curbuf, lnum, TRUE, &argvars[1], rettv); 278261f346fSBram Moolenaar } 279261f346fSBram Moolenaar 280261f346fSBram Moolenaar /* 281261f346fSBram Moolenaar * "appendbufline(buf, lnum, string/list)" function 282261f346fSBram Moolenaar */ 283261f346fSBram Moolenaar void 284261f346fSBram Moolenaar f_appendbufline(typval_T *argvars, typval_T *rettv) 285261f346fSBram Moolenaar { 286261f346fSBram Moolenaar linenr_T lnum; 287261f346fSBram Moolenaar buf_T *buf; 288261f346fSBram Moolenaar 289261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 290261f346fSBram Moolenaar if (buf == NULL) 291261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 292261f346fSBram Moolenaar else 293261f346fSBram Moolenaar { 294261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 295261f346fSBram Moolenaar set_buffer_lines(buf, lnum, TRUE, &argvars[2], rettv); 296261f346fSBram Moolenaar } 297261f346fSBram Moolenaar } 298261f346fSBram Moolenaar 299261f346fSBram Moolenaar /* 300261f346fSBram Moolenaar * "bufadd(expr)" function 301261f346fSBram Moolenaar */ 302261f346fSBram Moolenaar void 303261f346fSBram Moolenaar f_bufadd(typval_T *argvars, typval_T *rettv) 304261f346fSBram Moolenaar { 305261f346fSBram Moolenaar char_u *name = tv_get_string(&argvars[0]); 306261f346fSBram Moolenaar 307261f346fSBram Moolenaar rettv->vval.v_number = buflist_add(*name == NUL ? NULL : name, 0); 308261f346fSBram Moolenaar } 309261f346fSBram Moolenaar 310261f346fSBram Moolenaar /* 311261f346fSBram Moolenaar * "bufexists(expr)" function 312261f346fSBram Moolenaar */ 313261f346fSBram Moolenaar void 314261f346fSBram Moolenaar f_bufexists(typval_T *argvars, typval_T *rettv) 315261f346fSBram Moolenaar { 316261f346fSBram Moolenaar rettv->vval.v_number = (find_buffer(&argvars[0]) != NULL); 317261f346fSBram Moolenaar } 318261f346fSBram Moolenaar 319261f346fSBram Moolenaar /* 320261f346fSBram Moolenaar * "buflisted(expr)" function 321261f346fSBram Moolenaar */ 322261f346fSBram Moolenaar void 323261f346fSBram Moolenaar f_buflisted(typval_T *argvars, typval_T *rettv) 324261f346fSBram Moolenaar { 325261f346fSBram Moolenaar buf_T *buf; 326261f346fSBram Moolenaar 327261f346fSBram Moolenaar buf = find_buffer(&argvars[0]); 328261f346fSBram Moolenaar rettv->vval.v_number = (buf != NULL && buf->b_p_bl); 329261f346fSBram Moolenaar } 330261f346fSBram Moolenaar 331261f346fSBram Moolenaar /* 332261f346fSBram Moolenaar * "bufload(expr)" function 333261f346fSBram Moolenaar */ 334261f346fSBram Moolenaar void 335261f346fSBram Moolenaar f_bufload(typval_T *argvars, typval_T *rettv UNUSED) 336261f346fSBram Moolenaar { 337261f346fSBram Moolenaar buf_T *buf = get_buf_arg(&argvars[0]); 338261f346fSBram Moolenaar 339261f346fSBram Moolenaar if (buf != NULL) 340261f346fSBram Moolenaar buffer_ensure_loaded(buf); 341261f346fSBram Moolenaar } 342261f346fSBram Moolenaar 343261f346fSBram Moolenaar /* 344261f346fSBram Moolenaar * "bufloaded(expr)" function 345261f346fSBram Moolenaar */ 346261f346fSBram Moolenaar void 347261f346fSBram Moolenaar f_bufloaded(typval_T *argvars, typval_T *rettv) 348261f346fSBram Moolenaar { 349261f346fSBram Moolenaar buf_T *buf; 350261f346fSBram Moolenaar 351261f346fSBram Moolenaar buf = find_buffer(&argvars[0]); 352261f346fSBram Moolenaar rettv->vval.v_number = (buf != NULL && buf->b_ml.ml_mfp != NULL); 353261f346fSBram Moolenaar } 354261f346fSBram Moolenaar 355261f346fSBram Moolenaar /* 356261f346fSBram Moolenaar * "bufname(expr)" function 357261f346fSBram Moolenaar */ 358261f346fSBram Moolenaar void 359261f346fSBram Moolenaar f_bufname(typval_T *argvars, typval_T *rettv) 360261f346fSBram Moolenaar { 361261f346fSBram Moolenaar buf_T *buf; 36202aaad91SBram Moolenaar typval_T *tv = &argvars[0]; 363261f346fSBram Moolenaar 36402aaad91SBram Moolenaar if (tv->v_type == VAR_UNKNOWN) 365261f346fSBram Moolenaar buf = curbuf; 366261f346fSBram Moolenaar else 3673767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(tv); 368261f346fSBram Moolenaar rettv->v_type = VAR_STRING; 369261f346fSBram Moolenaar if (buf != NULL && buf->b_fname != NULL) 370261f346fSBram Moolenaar rettv->vval.v_string = vim_strsave(buf->b_fname); 371261f346fSBram Moolenaar else 372261f346fSBram Moolenaar rettv->vval.v_string = NULL; 373261f346fSBram Moolenaar } 374261f346fSBram Moolenaar 375261f346fSBram Moolenaar /* 376261f346fSBram Moolenaar * "bufnr(expr)" function 377261f346fSBram Moolenaar */ 378261f346fSBram Moolenaar void 379261f346fSBram Moolenaar f_bufnr(typval_T *argvars, typval_T *rettv) 380261f346fSBram Moolenaar { 381261f346fSBram Moolenaar buf_T *buf; 382261f346fSBram Moolenaar int error = FALSE; 383261f346fSBram Moolenaar char_u *name; 384261f346fSBram Moolenaar 385261f346fSBram Moolenaar if (argvars[0].v_type == VAR_UNKNOWN) 386261f346fSBram Moolenaar buf = curbuf; 387261f346fSBram Moolenaar else 3883767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(&argvars[0]); 389261f346fSBram Moolenaar 390261f346fSBram Moolenaar // If the buffer isn't found and the second argument is not zero create a 391261f346fSBram Moolenaar // new buffer. 392261f346fSBram Moolenaar if (buf == NULL 393261f346fSBram Moolenaar && argvars[1].v_type != VAR_UNKNOWN 394fe136c9aSBram Moolenaar && tv_get_bool_chk(&argvars[1], &error) != 0 395261f346fSBram Moolenaar && !error 396261f346fSBram Moolenaar && (name = tv_get_string_chk(&argvars[0])) != NULL 397261f346fSBram Moolenaar && !error) 398261f346fSBram Moolenaar buf = buflist_new(name, NULL, (linenr_T)1, 0); 399261f346fSBram Moolenaar 400261f346fSBram Moolenaar if (buf != NULL) 401261f346fSBram Moolenaar rettv->vval.v_number = buf->b_fnum; 402261f346fSBram Moolenaar else 403261f346fSBram Moolenaar rettv->vval.v_number = -1; 404261f346fSBram Moolenaar } 405261f346fSBram Moolenaar 406261f346fSBram Moolenaar static void 407261f346fSBram Moolenaar buf_win_common(typval_T *argvars, typval_T *rettv, int get_nr) 408261f346fSBram Moolenaar { 409261f346fSBram Moolenaar win_T *wp; 410261f346fSBram Moolenaar int winnr = 0; 411261f346fSBram Moolenaar buf_T *buf; 412261f346fSBram Moolenaar 4133767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(&argvars[0]); 414261f346fSBram Moolenaar FOR_ALL_WINDOWS(wp) 415261f346fSBram Moolenaar { 416261f346fSBram Moolenaar ++winnr; 417261f346fSBram Moolenaar if (wp->w_buffer == buf) 418261f346fSBram Moolenaar break; 419261f346fSBram Moolenaar } 420261f346fSBram Moolenaar rettv->vval.v_number = (wp != NULL ? (get_nr ? winnr : wp->w_id) : -1); 421261f346fSBram Moolenaar } 422261f346fSBram Moolenaar 423261f346fSBram Moolenaar /* 424261f346fSBram Moolenaar * "bufwinid(nr)" function 425261f346fSBram Moolenaar */ 426261f346fSBram Moolenaar void 427261f346fSBram Moolenaar f_bufwinid(typval_T *argvars, typval_T *rettv) 428261f346fSBram Moolenaar { 429261f346fSBram Moolenaar buf_win_common(argvars, rettv, FALSE); 430261f346fSBram Moolenaar } 431261f346fSBram Moolenaar 432261f346fSBram Moolenaar /* 433261f346fSBram Moolenaar * "bufwinnr(nr)" function 434261f346fSBram Moolenaar */ 435261f346fSBram Moolenaar void 436261f346fSBram Moolenaar f_bufwinnr(typval_T *argvars, typval_T *rettv) 437261f346fSBram Moolenaar { 438261f346fSBram Moolenaar buf_win_common(argvars, rettv, TRUE); 439261f346fSBram Moolenaar } 440261f346fSBram Moolenaar 441261f346fSBram Moolenaar /* 442261f346fSBram Moolenaar * "deletebufline()" function 443261f346fSBram Moolenaar */ 444261f346fSBram Moolenaar void 445261f346fSBram Moolenaar f_deletebufline(typval_T *argvars, typval_T *rettv) 446261f346fSBram Moolenaar { 447261f346fSBram Moolenaar buf_T *buf; 448261f346fSBram Moolenaar linenr_T first, last; 449261f346fSBram Moolenaar linenr_T lnum; 450261f346fSBram Moolenaar long count; 451261f346fSBram Moolenaar int is_curbuf; 452261f346fSBram Moolenaar buf_T *curbuf_save = NULL; 453261f346fSBram Moolenaar win_T *curwin_save = NULL; 454261f346fSBram Moolenaar tabpage_T *tp; 455261f346fSBram Moolenaar win_T *wp; 456261f346fSBram Moolenaar 457261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 458261f346fSBram Moolenaar if (buf == NULL) 459261f346fSBram Moolenaar { 460261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 461261f346fSBram Moolenaar return; 462261f346fSBram Moolenaar } 463261f346fSBram Moolenaar is_curbuf = buf == curbuf; 464261f346fSBram Moolenaar 465261f346fSBram Moolenaar first = tv_get_lnum_buf(&argvars[1], buf); 466261f346fSBram Moolenaar if (argvars[2].v_type != VAR_UNKNOWN) 467261f346fSBram Moolenaar last = tv_get_lnum_buf(&argvars[2], buf); 468261f346fSBram Moolenaar else 469261f346fSBram Moolenaar last = first; 470261f346fSBram Moolenaar 471261f346fSBram Moolenaar if (buf->b_ml.ml_mfp == NULL || first < 1 472261f346fSBram Moolenaar || first > buf->b_ml.ml_line_count || last < first) 473261f346fSBram Moolenaar { 474261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 475261f346fSBram Moolenaar return; 476261f346fSBram Moolenaar } 477261f346fSBram Moolenaar 478261f346fSBram Moolenaar if (!is_curbuf) 479261f346fSBram Moolenaar { 480261f346fSBram Moolenaar curbuf_save = curbuf; 481261f346fSBram Moolenaar curwin_save = curwin; 482261f346fSBram Moolenaar curbuf = buf; 483261f346fSBram Moolenaar find_win_for_curbuf(); 484261f346fSBram Moolenaar } 485261f346fSBram Moolenaar if (last > curbuf->b_ml.ml_line_count) 486261f346fSBram Moolenaar last = curbuf->b_ml.ml_line_count; 487261f346fSBram Moolenaar count = last - first + 1; 488261f346fSBram Moolenaar 489261f346fSBram Moolenaar // When coming here from Insert mode, sync undo, so that this can be 490261f346fSBram Moolenaar // undone separately from what was previously inserted. 491261f346fSBram Moolenaar if (u_sync_once == 2) 492261f346fSBram Moolenaar { 493261f346fSBram Moolenaar u_sync_once = 1; // notify that u_sync() was called 494261f346fSBram Moolenaar u_sync(TRUE); 495261f346fSBram Moolenaar } 496261f346fSBram Moolenaar 497261f346fSBram Moolenaar if (u_save(first - 1, last + 1) == FAIL) 498261f346fSBram Moolenaar { 499261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 500261f346fSBram Moolenaar return; 501261f346fSBram Moolenaar } 502261f346fSBram Moolenaar 503261f346fSBram Moolenaar for (lnum = first; lnum <= last; ++lnum) 504ca70c07bSBram Moolenaar ml_delete_flags(first, ML_DEL_MESSAGE); 505261f346fSBram Moolenaar 506261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 507261f346fSBram Moolenaar if (wp->w_buffer == buf) 508261f346fSBram Moolenaar { 509261f346fSBram Moolenaar if (wp->w_cursor.lnum > last) 510261f346fSBram Moolenaar wp->w_cursor.lnum -= count; 511261f346fSBram Moolenaar else if (wp->w_cursor.lnum> first) 512261f346fSBram Moolenaar wp->w_cursor.lnum = first; 513261f346fSBram Moolenaar if (wp->w_cursor.lnum > wp->w_buffer->b_ml.ml_line_count) 514261f346fSBram Moolenaar wp->w_cursor.lnum = wp->w_buffer->b_ml.ml_line_count; 515261f346fSBram Moolenaar } 516261f346fSBram Moolenaar check_cursor_col(); 517261f346fSBram Moolenaar deleted_lines_mark(first, count); 518261f346fSBram Moolenaar 519261f346fSBram Moolenaar if (!is_curbuf) 520261f346fSBram Moolenaar { 521261f346fSBram Moolenaar curbuf = curbuf_save; 522261f346fSBram Moolenaar curwin = curwin_save; 523261f346fSBram Moolenaar } 524261f346fSBram Moolenaar } 525261f346fSBram Moolenaar 526261f346fSBram Moolenaar /* 527261f346fSBram Moolenaar * Returns buffer options, variables and other attributes in a dictionary. 528261f346fSBram Moolenaar */ 529261f346fSBram Moolenaar static dict_T * 530261f346fSBram Moolenaar get_buffer_info(buf_T *buf) 531261f346fSBram Moolenaar { 532261f346fSBram Moolenaar dict_T *dict; 533261f346fSBram Moolenaar tabpage_T *tp; 534261f346fSBram Moolenaar win_T *wp; 535261f346fSBram Moolenaar list_T *windows; 536261f346fSBram Moolenaar 537261f346fSBram Moolenaar dict = dict_alloc(); 538261f346fSBram Moolenaar if (dict == NULL) 539261f346fSBram Moolenaar return NULL; 540261f346fSBram Moolenaar 541261f346fSBram Moolenaar dict_add_number(dict, "bufnr", buf->b_fnum); 542261f346fSBram Moolenaar dict_add_string(dict, "name", buf->b_ffname); 543261f346fSBram Moolenaar dict_add_number(dict, "lnum", buf == curbuf ? curwin->w_cursor.lnum 544261f346fSBram Moolenaar : buflist_findlnum(buf)); 545a9e9679dSBram Moolenaar dict_add_number(dict, "linecount", buf->b_ml.ml_line_count); 546261f346fSBram Moolenaar dict_add_number(dict, "loaded", buf->b_ml.ml_mfp != NULL); 547261f346fSBram Moolenaar dict_add_number(dict, "listed", buf->b_p_bl); 548261f346fSBram Moolenaar dict_add_number(dict, "changed", bufIsChanged(buf)); 549261f346fSBram Moolenaar dict_add_number(dict, "changedtick", CHANGEDTICK(buf)); 550261f346fSBram Moolenaar dict_add_number(dict, "hidden", 551261f346fSBram Moolenaar buf->b_ml.ml_mfp != NULL && buf->b_nwindows == 0); 552261f346fSBram Moolenaar 553261f346fSBram Moolenaar // Get a reference to buffer variables 554261f346fSBram Moolenaar dict_add_dict(dict, "variables", buf->b_vars); 555261f346fSBram Moolenaar 556261f346fSBram Moolenaar // List of windows displaying this buffer 557261f346fSBram Moolenaar windows = list_alloc(); 558261f346fSBram Moolenaar if (windows != NULL) 559261f346fSBram Moolenaar { 560261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 561261f346fSBram Moolenaar if (wp->w_buffer == buf) 562261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 563261f346fSBram Moolenaar dict_add_list(dict, "windows", windows); 564261f346fSBram Moolenaar } 565261f346fSBram Moolenaar 56605ad5ff0SBram Moolenaar #ifdef FEAT_PROP_POPUP 567261f346fSBram Moolenaar // List of popup windows displaying this buffer 568261f346fSBram Moolenaar windows = list_alloc(); 569261f346fSBram Moolenaar if (windows != NULL) 570261f346fSBram Moolenaar { 571aeea7215SBram Moolenaar FOR_ALL_POPUPWINS(wp) 572261f346fSBram Moolenaar if (wp->w_buffer == buf) 573261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 574261f346fSBram Moolenaar FOR_ALL_TABPAGES(tp) 575aeea7215SBram Moolenaar FOR_ALL_POPUPWINS_IN_TAB(tp, wp) 576261f346fSBram Moolenaar if (wp->w_buffer == buf) 577261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 578261f346fSBram Moolenaar 579261f346fSBram Moolenaar dict_add_list(dict, "popups", windows); 580261f346fSBram Moolenaar } 581261f346fSBram Moolenaar #endif 582261f346fSBram Moolenaar 583261f346fSBram Moolenaar #ifdef FEAT_SIGNS 584261f346fSBram Moolenaar if (buf->b_signlist != NULL) 585261f346fSBram Moolenaar { 586261f346fSBram Moolenaar // List of signs placed in this buffer 587261f346fSBram Moolenaar list_T *signs = list_alloc(); 588261f346fSBram Moolenaar if (signs != NULL) 589261f346fSBram Moolenaar { 590261f346fSBram Moolenaar get_buffer_signs(buf, signs); 591261f346fSBram Moolenaar dict_add_list(dict, "signs", signs); 592261f346fSBram Moolenaar } 593261f346fSBram Moolenaar } 594261f346fSBram Moolenaar #endif 595261f346fSBram Moolenaar 59652410575SBram Moolenaar #ifdef FEAT_VIMINFO 59752410575SBram Moolenaar dict_add_number(dict, "lastused", buf->b_last_used); 59852410575SBram Moolenaar #endif 59952410575SBram Moolenaar 600261f346fSBram Moolenaar return dict; 601261f346fSBram Moolenaar } 602261f346fSBram Moolenaar 603261f346fSBram Moolenaar /* 604261f346fSBram Moolenaar * "getbufinfo()" function 605261f346fSBram Moolenaar */ 606261f346fSBram Moolenaar void 607261f346fSBram Moolenaar f_getbufinfo(typval_T *argvars, typval_T *rettv) 608261f346fSBram Moolenaar { 609261f346fSBram Moolenaar buf_T *buf = NULL; 610261f346fSBram Moolenaar buf_T *argbuf = NULL; 611261f346fSBram Moolenaar dict_T *d; 612261f346fSBram Moolenaar int filtered = FALSE; 613261f346fSBram Moolenaar int sel_buflisted = FALSE; 614261f346fSBram Moolenaar int sel_bufloaded = FALSE; 615261f346fSBram Moolenaar int sel_bufmodified = FALSE; 616261f346fSBram Moolenaar 617261f346fSBram Moolenaar if (rettv_list_alloc(rettv) != OK) 618261f346fSBram Moolenaar return; 619261f346fSBram Moolenaar 620261f346fSBram Moolenaar // List of all the buffers or selected buffers 621261f346fSBram Moolenaar if (argvars[0].v_type == VAR_DICT) 622261f346fSBram Moolenaar { 623261f346fSBram Moolenaar dict_T *sel_d = argvars[0].vval.v_dict; 624261f346fSBram Moolenaar 625261f346fSBram Moolenaar if (sel_d != NULL) 626261f346fSBram Moolenaar { 627261f346fSBram Moolenaar filtered = TRUE; 628036c2cf7SBram Moolenaar sel_buflisted = dict_get_bool(sel_d, (char_u *)"buflisted", FALSE); 629036c2cf7SBram Moolenaar sel_bufloaded = dict_get_bool(sel_d, (char_u *)"bufloaded", FALSE); 630036c2cf7SBram Moolenaar sel_bufmodified = dict_get_bool(sel_d, (char_u *)"bufmodified", 631036c2cf7SBram Moolenaar FALSE); 632261f346fSBram Moolenaar } 633261f346fSBram Moolenaar } 634261f346fSBram Moolenaar else if (argvars[0].v_type != VAR_UNKNOWN) 635261f346fSBram Moolenaar { 636261f346fSBram Moolenaar // Information about one buffer. Argument specifies the buffer 6373767e3a3SBram Moolenaar argbuf = tv_get_buf_from_arg(&argvars[0]); 638261f346fSBram Moolenaar if (argbuf == NULL) 639261f346fSBram Moolenaar return; 640261f346fSBram Moolenaar } 641261f346fSBram Moolenaar 642261f346fSBram Moolenaar // Return information about all the buffers or a specified buffer 643261f346fSBram Moolenaar FOR_ALL_BUFFERS(buf) 644261f346fSBram Moolenaar { 645261f346fSBram Moolenaar if (argbuf != NULL && argbuf != buf) 646261f346fSBram Moolenaar continue; 647261f346fSBram Moolenaar if (filtered && ((sel_bufloaded && buf->b_ml.ml_mfp == NULL) 648261f346fSBram Moolenaar || (sel_buflisted && !buf->b_p_bl) 649261f346fSBram Moolenaar || (sel_bufmodified && !buf->b_changed))) 650261f346fSBram Moolenaar continue; 651261f346fSBram Moolenaar 652261f346fSBram Moolenaar d = get_buffer_info(buf); 653261f346fSBram Moolenaar if (d != NULL) 654261f346fSBram Moolenaar list_append_dict(rettv->vval.v_list, d); 655261f346fSBram Moolenaar if (argbuf != NULL) 656261f346fSBram Moolenaar return; 657261f346fSBram Moolenaar } 658261f346fSBram Moolenaar } 659261f346fSBram Moolenaar 660261f346fSBram Moolenaar /* 661261f346fSBram Moolenaar * Get line or list of lines from buffer "buf" into "rettv". 662261f346fSBram Moolenaar * Return a range (from start to end) of lines in rettv from the specified 663261f346fSBram Moolenaar * buffer. 664261f346fSBram Moolenaar * If 'retlist' is TRUE, then the lines are returned as a Vim List. 665261f346fSBram Moolenaar */ 666261f346fSBram Moolenaar static void 667261f346fSBram Moolenaar get_buffer_lines( 668261f346fSBram Moolenaar buf_T *buf, 669261f346fSBram Moolenaar linenr_T start, 670261f346fSBram Moolenaar linenr_T end, 671261f346fSBram Moolenaar int retlist, 672261f346fSBram Moolenaar typval_T *rettv) 673261f346fSBram Moolenaar { 674261f346fSBram Moolenaar char_u *p; 675261f346fSBram Moolenaar 6768a7d6542SBram Moolenaar if (retlist) 6778a7d6542SBram Moolenaar { 6788a7d6542SBram Moolenaar if (rettv_list_alloc(rettv) == FAIL) 6798a7d6542SBram Moolenaar return; 6808a7d6542SBram Moolenaar } 6818a7d6542SBram Moolenaar else 6828a7d6542SBram Moolenaar { 683261f346fSBram Moolenaar rettv->v_type = VAR_STRING; 684261f346fSBram Moolenaar rettv->vval.v_string = NULL; 6858a7d6542SBram Moolenaar } 686261f346fSBram Moolenaar 687261f346fSBram Moolenaar if (buf == NULL || buf->b_ml.ml_mfp == NULL || start < 0) 688261f346fSBram Moolenaar return; 689261f346fSBram Moolenaar 690261f346fSBram Moolenaar if (!retlist) 691261f346fSBram Moolenaar { 692261f346fSBram Moolenaar if (start >= 1 && start <= buf->b_ml.ml_line_count) 693261f346fSBram Moolenaar p = ml_get_buf(buf, start, FALSE); 694261f346fSBram Moolenaar else 695261f346fSBram Moolenaar p = (char_u *)""; 696261f346fSBram Moolenaar rettv->vval.v_string = vim_strsave(p); 697261f346fSBram Moolenaar } 698261f346fSBram Moolenaar else 699261f346fSBram Moolenaar { 700261f346fSBram Moolenaar if (end < start) 701261f346fSBram Moolenaar return; 702261f346fSBram Moolenaar 703261f346fSBram Moolenaar if (start < 1) 704261f346fSBram Moolenaar start = 1; 705261f346fSBram Moolenaar if (end > buf->b_ml.ml_line_count) 706261f346fSBram Moolenaar end = buf->b_ml.ml_line_count; 707261f346fSBram Moolenaar while (start <= end) 708261f346fSBram Moolenaar if (list_append_string(rettv->vval.v_list, 709261f346fSBram Moolenaar ml_get_buf(buf, start++, FALSE), -1) == FAIL) 710261f346fSBram Moolenaar break; 711261f346fSBram Moolenaar } 712261f346fSBram Moolenaar } 713261f346fSBram Moolenaar 714261f346fSBram Moolenaar /* 715261f346fSBram Moolenaar * "getbufline()" function 716261f346fSBram Moolenaar */ 717261f346fSBram Moolenaar void 718261f346fSBram Moolenaar f_getbufline(typval_T *argvars, typval_T *rettv) 719261f346fSBram Moolenaar { 720e6e70a10SBram Moolenaar linenr_T lnum = 1; 721e6e70a10SBram Moolenaar linenr_T end = 1; 722261f346fSBram Moolenaar buf_T *buf; 723261f346fSBram Moolenaar 7243767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(&argvars[0]); 725e6e70a10SBram Moolenaar if (buf != NULL) 726e6e70a10SBram Moolenaar { 727261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 728261f346fSBram Moolenaar if (argvars[2].v_type == VAR_UNKNOWN) 729261f346fSBram Moolenaar end = lnum; 730261f346fSBram Moolenaar else 731261f346fSBram Moolenaar end = tv_get_lnum_buf(&argvars[2], buf); 732e6e70a10SBram Moolenaar } 733261f346fSBram Moolenaar 734261f346fSBram Moolenaar get_buffer_lines(buf, lnum, end, TRUE, rettv); 735261f346fSBram Moolenaar } 736261f346fSBram Moolenaar 737fbdd08edSBram Moolenaar type_T * 738fbdd08edSBram Moolenaar ret_f_getline(int argcount, type_T **argtypes UNUSED) 739fbdd08edSBram Moolenaar { 740fbdd08edSBram Moolenaar return argcount == 1 ? &t_string : &t_list_string; 741fbdd08edSBram Moolenaar } 742fbdd08edSBram Moolenaar 743261f346fSBram Moolenaar /* 744261f346fSBram Moolenaar * "getline(lnum, [end])" function 745261f346fSBram Moolenaar */ 746261f346fSBram Moolenaar void 747261f346fSBram Moolenaar f_getline(typval_T *argvars, typval_T *rettv) 748261f346fSBram Moolenaar { 749261f346fSBram Moolenaar linenr_T lnum; 750261f346fSBram Moolenaar linenr_T end; 751261f346fSBram Moolenaar int retlist; 752261f346fSBram Moolenaar 753261f346fSBram Moolenaar lnum = tv_get_lnum(argvars); 754261f346fSBram Moolenaar if (argvars[1].v_type == VAR_UNKNOWN) 755261f346fSBram Moolenaar { 756261f346fSBram Moolenaar end = 0; 757261f346fSBram Moolenaar retlist = FALSE; 758261f346fSBram Moolenaar } 759261f346fSBram Moolenaar else 760261f346fSBram Moolenaar { 761261f346fSBram Moolenaar end = tv_get_lnum(&argvars[1]); 762261f346fSBram Moolenaar retlist = TRUE; 763261f346fSBram Moolenaar } 764261f346fSBram Moolenaar 765261f346fSBram Moolenaar get_buffer_lines(curbuf, lnum, end, retlist, rettv); 766261f346fSBram Moolenaar } 767261f346fSBram Moolenaar 768261f346fSBram Moolenaar /* 769261f346fSBram Moolenaar * "setbufline()" function 770261f346fSBram Moolenaar */ 771261f346fSBram Moolenaar void 772261f346fSBram Moolenaar f_setbufline(typval_T *argvars, typval_T *rettv) 773261f346fSBram Moolenaar { 774261f346fSBram Moolenaar linenr_T lnum; 775261f346fSBram Moolenaar buf_T *buf; 776261f346fSBram Moolenaar 777261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 778261f346fSBram Moolenaar if (buf == NULL) 779261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 780261f346fSBram Moolenaar else 781261f346fSBram Moolenaar { 782261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 783261f346fSBram Moolenaar set_buffer_lines(buf, lnum, FALSE, &argvars[2], rettv); 784261f346fSBram Moolenaar } 785261f346fSBram Moolenaar } 786261f346fSBram Moolenaar 787261f346fSBram Moolenaar /* 788261f346fSBram Moolenaar * "setline()" function 789261f346fSBram Moolenaar */ 790261f346fSBram Moolenaar void 791261f346fSBram Moolenaar f_setline(typval_T *argvars, typval_T *rettv) 792261f346fSBram Moolenaar { 793261f346fSBram Moolenaar linenr_T lnum = tv_get_lnum(&argvars[0]); 794261f346fSBram Moolenaar 795261f346fSBram Moolenaar set_buffer_lines(curbuf, lnum, FALSE, &argvars[1], rettv); 796261f346fSBram Moolenaar } 797261f346fSBram Moolenaar #endif // FEAT_EVAL 798261f346fSBram Moolenaar 799261f346fSBram Moolenaar #if defined(FEAT_JOB_CHANNEL) \ 800261f346fSBram Moolenaar || defined(FEAT_PYTHON) || defined(FEAT_PYTHON3) \ 801261f346fSBram Moolenaar || defined(PROTO) 802261f346fSBram Moolenaar /* 803261f346fSBram Moolenaar * Make "buf" the current buffer. restore_buffer() MUST be called to undo. 804261f346fSBram Moolenaar * No autocommands will be executed. Use aucmd_prepbuf() if there are any. 805261f346fSBram Moolenaar */ 806261f346fSBram Moolenaar void 807261f346fSBram Moolenaar switch_buffer(bufref_T *save_curbuf, buf_T *buf) 808261f346fSBram Moolenaar { 809261f346fSBram Moolenaar block_autocmds(); 810*3e0107eaSBram Moolenaar #ifdef FEAT_FOLDING 811*3e0107eaSBram Moolenaar ++disable_fold_update; 812*3e0107eaSBram Moolenaar #endif 813261f346fSBram Moolenaar set_bufref(save_curbuf, curbuf); 814261f346fSBram Moolenaar --curbuf->b_nwindows; 815261f346fSBram Moolenaar curbuf = buf; 816261f346fSBram Moolenaar curwin->w_buffer = buf; 817261f346fSBram Moolenaar ++curbuf->b_nwindows; 818261f346fSBram Moolenaar } 819261f346fSBram Moolenaar 820261f346fSBram Moolenaar /* 821261f346fSBram Moolenaar * Restore the current buffer after using switch_buffer(). 822261f346fSBram Moolenaar */ 823261f346fSBram Moolenaar void 824261f346fSBram Moolenaar restore_buffer(bufref_T *save_curbuf) 825261f346fSBram Moolenaar { 826261f346fSBram Moolenaar unblock_autocmds(); 827*3e0107eaSBram Moolenaar #ifdef FEAT_FOLDING 828*3e0107eaSBram Moolenaar --disable_fold_update; 829*3e0107eaSBram Moolenaar #endif 8305d18efecSBram Moolenaar // Check for valid buffer, just in case. 831261f346fSBram Moolenaar if (bufref_valid(save_curbuf)) 832261f346fSBram Moolenaar { 833261f346fSBram Moolenaar --curbuf->b_nwindows; 834261f346fSBram Moolenaar curwin->w_buffer = save_curbuf->br_buf; 835261f346fSBram Moolenaar curbuf = save_curbuf->br_buf; 836261f346fSBram Moolenaar ++curbuf->b_nwindows; 837261f346fSBram Moolenaar } 838261f346fSBram Moolenaar } 839261f346fSBram Moolenaar 840261f346fSBram Moolenaar /* 841261f346fSBram Moolenaar * Find a window for buffer "buf". 842261f346fSBram Moolenaar * If found OK is returned and "wp" and "tp" are set to the window and tabpage. 843261f346fSBram Moolenaar * If not found FAIL is returned. 844261f346fSBram Moolenaar */ 845261f346fSBram Moolenaar static int 846261f346fSBram Moolenaar find_win_for_buf( 847261f346fSBram Moolenaar buf_T *buf, 848261f346fSBram Moolenaar win_T **wp, 849261f346fSBram Moolenaar tabpage_T **tp) 850261f346fSBram Moolenaar { 851261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(*tp, *wp) 852261f346fSBram Moolenaar if ((*wp)->w_buffer == buf) 853261f346fSBram Moolenaar return OK; 854261f346fSBram Moolenaar return FAIL; 855261f346fSBram Moolenaar } 856261f346fSBram Moolenaar 857261f346fSBram Moolenaar /* 858261f346fSBram Moolenaar * Find a window that contains "buf" and switch to it. 859261f346fSBram Moolenaar * If there is no such window, use the current window and change "curbuf". 860261f346fSBram Moolenaar * Caller must initialize save_curbuf to NULL. 861261f346fSBram Moolenaar * restore_win_for_buf() MUST be called later! 862261f346fSBram Moolenaar */ 863261f346fSBram Moolenaar void 864261f346fSBram Moolenaar switch_to_win_for_buf( 865261f346fSBram Moolenaar buf_T *buf, 866261f346fSBram Moolenaar win_T **save_curwinp, 867261f346fSBram Moolenaar tabpage_T **save_curtabp, 868261f346fSBram Moolenaar bufref_T *save_curbuf) 869261f346fSBram Moolenaar { 870261f346fSBram Moolenaar win_T *wp; 871261f346fSBram Moolenaar tabpage_T *tp; 872261f346fSBram Moolenaar 873261f346fSBram Moolenaar if (find_win_for_buf(buf, &wp, &tp) == FAIL) 874261f346fSBram Moolenaar switch_buffer(save_curbuf, buf); 875261f346fSBram Moolenaar else if (switch_win(save_curwinp, save_curtabp, wp, tp, TRUE) == FAIL) 876261f346fSBram Moolenaar { 877261f346fSBram Moolenaar restore_win(*save_curwinp, *save_curtabp, TRUE); 878261f346fSBram Moolenaar switch_buffer(save_curbuf, buf); 879261f346fSBram Moolenaar } 880261f346fSBram Moolenaar } 881261f346fSBram Moolenaar 882261f346fSBram Moolenaar void 883261f346fSBram Moolenaar restore_win_for_buf( 884261f346fSBram Moolenaar win_T *save_curwin, 885261f346fSBram Moolenaar tabpage_T *save_curtab, 886261f346fSBram Moolenaar bufref_T *save_curbuf) 887261f346fSBram Moolenaar { 888261f346fSBram Moolenaar if (save_curbuf->br_buf == NULL) 889261f346fSBram Moolenaar restore_win(save_curwin, save_curtab, TRUE); 890261f346fSBram Moolenaar else 891261f346fSBram Moolenaar restore_buffer(save_curbuf); 892261f346fSBram Moolenaar } 893261f346fSBram Moolenaar #endif 894