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 120261f346fSBram Moolenaar for (wip = curbuf->b_wininfo; wip != NULL; wip = wip->wi_next) 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; 1798a7d6542SBram Moolenaar range_list_materialize(l); 180261f346fSBram Moolenaar li = l->lv_first; 181261f346fSBram Moolenaar } 182261f346fSBram Moolenaar else 183261f346fSBram Moolenaar line = tv_get_string_chk(lines); 184261f346fSBram Moolenaar 185261f346fSBram Moolenaar // default result is zero == OK 186261f346fSBram Moolenaar for (;;) 187261f346fSBram Moolenaar { 188261f346fSBram Moolenaar if (l != NULL) 189261f346fSBram Moolenaar { 190261f346fSBram Moolenaar // list argument, get next string 191261f346fSBram Moolenaar if (li == NULL) 192261f346fSBram Moolenaar break; 193261f346fSBram Moolenaar line = tv_get_string_chk(&li->li_tv); 194261f346fSBram Moolenaar li = li->li_next; 195261f346fSBram Moolenaar } 196261f346fSBram Moolenaar 197261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 198261f346fSBram Moolenaar if (line == NULL || lnum > curbuf->b_ml.ml_line_count + 1) 199261f346fSBram Moolenaar break; 200261f346fSBram Moolenaar 201261f346fSBram Moolenaar // When coming here from Insert mode, sync undo, so that this can be 202261f346fSBram Moolenaar // undone separately from what was previously inserted. 203261f346fSBram Moolenaar if (u_sync_once == 2) 204261f346fSBram Moolenaar { 205261f346fSBram Moolenaar u_sync_once = 1; // notify that u_sync() was called 206261f346fSBram Moolenaar u_sync(TRUE); 207261f346fSBram Moolenaar } 208261f346fSBram Moolenaar 209261f346fSBram Moolenaar if (!append && lnum <= curbuf->b_ml.ml_line_count) 210261f346fSBram Moolenaar { 211261f346fSBram Moolenaar // Existing line, replace it. 212261f346fSBram Moolenaar // Removes any existing text properties. 213261f346fSBram Moolenaar if (u_savesub(lnum) == OK && ml_replace_len( 214261f346fSBram Moolenaar lnum, line, (colnr_T)STRLEN(line) + 1, TRUE, TRUE) == OK) 215261f346fSBram Moolenaar { 216261f346fSBram Moolenaar changed_bytes(lnum, 0); 217261f346fSBram Moolenaar if (is_curbuf && lnum == curwin->w_cursor.lnum) 218261f346fSBram Moolenaar check_cursor_col(); 219261f346fSBram Moolenaar rettv->vval.v_number = 0; // OK 220261f346fSBram Moolenaar } 221261f346fSBram Moolenaar } 222261f346fSBram Moolenaar else if (added > 0 || u_save(lnum - 1, lnum) == OK) 223261f346fSBram Moolenaar { 224261f346fSBram Moolenaar // append the line 225261f346fSBram Moolenaar ++added; 226261f346fSBram Moolenaar if (ml_append(lnum - 1, line, (colnr_T)0, FALSE) == OK) 227261f346fSBram Moolenaar rettv->vval.v_number = 0; // OK 228261f346fSBram Moolenaar } 229261f346fSBram Moolenaar 230261f346fSBram Moolenaar if (l == NULL) // only one string argument 231261f346fSBram Moolenaar break; 232261f346fSBram Moolenaar ++lnum; 233261f346fSBram Moolenaar } 234261f346fSBram Moolenaar 235261f346fSBram Moolenaar if (added > 0) 236261f346fSBram Moolenaar { 237261f346fSBram Moolenaar win_T *wp; 238261f346fSBram Moolenaar tabpage_T *tp; 239261f346fSBram Moolenaar 240261f346fSBram Moolenaar appended_lines_mark(append_lnum, added); 241261f346fSBram Moolenaar 242261f346fSBram Moolenaar // Only adjust the cursor for buffers other than the current, unless it 243261f346fSBram Moolenaar // is the current window. For curbuf and other windows it has been 244261f346fSBram Moolenaar // done in mark_adjust_internal(). 245261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 246261f346fSBram Moolenaar if (wp->w_buffer == buf 247261f346fSBram Moolenaar && (wp->w_buffer != curbuf || wp == curwin) 248261f346fSBram Moolenaar && wp->w_cursor.lnum > append_lnum) 249261f346fSBram Moolenaar wp->w_cursor.lnum += added; 250261f346fSBram Moolenaar check_cursor_col(); 251261f346fSBram Moolenaar update_topline(); 252261f346fSBram Moolenaar } 253261f346fSBram Moolenaar 254261f346fSBram Moolenaar if (!is_curbuf) 255261f346fSBram Moolenaar { 256261f346fSBram Moolenaar curbuf = curbuf_save; 257261f346fSBram Moolenaar curwin = curwin_save; 258261f346fSBram Moolenaar } 259261f346fSBram Moolenaar } 260261f346fSBram Moolenaar 261261f346fSBram Moolenaar /* 262261f346fSBram Moolenaar * "append(lnum, string/list)" function 263261f346fSBram Moolenaar */ 264261f346fSBram Moolenaar void 265261f346fSBram Moolenaar f_append(typval_T *argvars, typval_T *rettv) 266261f346fSBram Moolenaar { 267261f346fSBram Moolenaar linenr_T lnum = tv_get_lnum(&argvars[0]); 268261f346fSBram Moolenaar 269261f346fSBram Moolenaar set_buffer_lines(curbuf, lnum, TRUE, &argvars[1], rettv); 270261f346fSBram Moolenaar } 271261f346fSBram Moolenaar 272261f346fSBram Moolenaar /* 273261f346fSBram Moolenaar * "appendbufline(buf, lnum, string/list)" function 274261f346fSBram Moolenaar */ 275261f346fSBram Moolenaar void 276261f346fSBram Moolenaar f_appendbufline(typval_T *argvars, typval_T *rettv) 277261f346fSBram Moolenaar { 278261f346fSBram Moolenaar linenr_T lnum; 279261f346fSBram Moolenaar buf_T *buf; 280261f346fSBram Moolenaar 281261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 282261f346fSBram Moolenaar if (buf == NULL) 283261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 284261f346fSBram Moolenaar else 285261f346fSBram Moolenaar { 286261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 287261f346fSBram Moolenaar set_buffer_lines(buf, lnum, TRUE, &argvars[2], rettv); 288261f346fSBram Moolenaar } 289261f346fSBram Moolenaar } 290261f346fSBram Moolenaar 291261f346fSBram Moolenaar /* 292261f346fSBram Moolenaar * "bufadd(expr)" function 293261f346fSBram Moolenaar */ 294261f346fSBram Moolenaar void 295261f346fSBram Moolenaar f_bufadd(typval_T *argvars, typval_T *rettv) 296261f346fSBram Moolenaar { 297261f346fSBram Moolenaar char_u *name = tv_get_string(&argvars[0]); 298261f346fSBram Moolenaar 299261f346fSBram Moolenaar rettv->vval.v_number = buflist_add(*name == NUL ? NULL : name, 0); 300261f346fSBram Moolenaar } 301261f346fSBram Moolenaar 302261f346fSBram Moolenaar /* 303261f346fSBram Moolenaar * "bufexists(expr)" function 304261f346fSBram Moolenaar */ 305261f346fSBram Moolenaar void 306261f346fSBram Moolenaar f_bufexists(typval_T *argvars, typval_T *rettv) 307261f346fSBram Moolenaar { 308261f346fSBram Moolenaar rettv->vval.v_number = (find_buffer(&argvars[0]) != NULL); 309261f346fSBram Moolenaar } 310261f346fSBram Moolenaar 311261f346fSBram Moolenaar /* 312261f346fSBram Moolenaar * "buflisted(expr)" function 313261f346fSBram Moolenaar */ 314261f346fSBram Moolenaar void 315261f346fSBram Moolenaar f_buflisted(typval_T *argvars, typval_T *rettv) 316261f346fSBram Moolenaar { 317261f346fSBram Moolenaar buf_T *buf; 318261f346fSBram Moolenaar 319261f346fSBram Moolenaar buf = find_buffer(&argvars[0]); 320261f346fSBram Moolenaar rettv->vval.v_number = (buf != NULL && buf->b_p_bl); 321261f346fSBram Moolenaar } 322261f346fSBram Moolenaar 323261f346fSBram Moolenaar /* 324261f346fSBram Moolenaar * "bufload(expr)" function 325261f346fSBram Moolenaar */ 326261f346fSBram Moolenaar void 327261f346fSBram Moolenaar f_bufload(typval_T *argvars, typval_T *rettv UNUSED) 328261f346fSBram Moolenaar { 329261f346fSBram Moolenaar buf_T *buf = get_buf_arg(&argvars[0]); 330261f346fSBram Moolenaar 331261f346fSBram Moolenaar if (buf != NULL) 332261f346fSBram Moolenaar buffer_ensure_loaded(buf); 333261f346fSBram Moolenaar } 334261f346fSBram Moolenaar 335261f346fSBram Moolenaar /* 336261f346fSBram Moolenaar * "bufloaded(expr)" function 337261f346fSBram Moolenaar */ 338261f346fSBram Moolenaar void 339261f346fSBram Moolenaar f_bufloaded(typval_T *argvars, typval_T *rettv) 340261f346fSBram Moolenaar { 341261f346fSBram Moolenaar buf_T *buf; 342261f346fSBram Moolenaar 343261f346fSBram Moolenaar buf = find_buffer(&argvars[0]); 344261f346fSBram Moolenaar rettv->vval.v_number = (buf != NULL && buf->b_ml.ml_mfp != NULL); 345261f346fSBram Moolenaar } 346261f346fSBram Moolenaar 347261f346fSBram Moolenaar /* 348261f346fSBram Moolenaar * "bufname(expr)" function 349261f346fSBram Moolenaar */ 350261f346fSBram Moolenaar void 351261f346fSBram Moolenaar f_bufname(typval_T *argvars, typval_T *rettv) 352261f346fSBram Moolenaar { 353261f346fSBram Moolenaar buf_T *buf; 354261f346fSBram Moolenaar 355261f346fSBram Moolenaar if (argvars[0].v_type == VAR_UNKNOWN) 356261f346fSBram Moolenaar buf = curbuf; 357261f346fSBram Moolenaar else 358261f346fSBram Moolenaar { 359261f346fSBram Moolenaar (void)tv_get_number(&argvars[0]); // issue errmsg if type error 360261f346fSBram Moolenaar ++emsg_off; 361261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 362261f346fSBram Moolenaar --emsg_off; 363261f346fSBram Moolenaar } 364261f346fSBram Moolenaar rettv->v_type = VAR_STRING; 365261f346fSBram Moolenaar if (buf != NULL && buf->b_fname != NULL) 366261f346fSBram Moolenaar rettv->vval.v_string = vim_strsave(buf->b_fname); 367261f346fSBram Moolenaar else 368261f346fSBram Moolenaar rettv->vval.v_string = NULL; 369261f346fSBram Moolenaar } 370261f346fSBram Moolenaar 371261f346fSBram Moolenaar /* 372261f346fSBram Moolenaar * "bufnr(expr)" function 373261f346fSBram Moolenaar */ 374261f346fSBram Moolenaar void 375261f346fSBram Moolenaar f_bufnr(typval_T *argvars, typval_T *rettv) 376261f346fSBram Moolenaar { 377261f346fSBram Moolenaar buf_T *buf; 378261f346fSBram Moolenaar int error = FALSE; 379261f346fSBram Moolenaar char_u *name; 380261f346fSBram Moolenaar 381261f346fSBram Moolenaar if (argvars[0].v_type == VAR_UNKNOWN) 382261f346fSBram Moolenaar buf = curbuf; 383261f346fSBram Moolenaar else 384261f346fSBram Moolenaar { 385261f346fSBram Moolenaar (void)tv_get_number(&argvars[0]); // issue errmsg if type error 386261f346fSBram Moolenaar ++emsg_off; 387261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 388261f346fSBram Moolenaar --emsg_off; 389261f346fSBram Moolenaar } 390261f346fSBram Moolenaar 391261f346fSBram Moolenaar // If the buffer isn't found and the second argument is not zero create a 392261f346fSBram Moolenaar // new buffer. 393261f346fSBram Moolenaar if (buf == NULL 394261f346fSBram Moolenaar && argvars[1].v_type != VAR_UNKNOWN 395261f346fSBram Moolenaar && tv_get_number_chk(&argvars[1], &error) != 0 396261f346fSBram Moolenaar && !error 397261f346fSBram Moolenaar && (name = tv_get_string_chk(&argvars[0])) != NULL 398261f346fSBram Moolenaar && !error) 399261f346fSBram Moolenaar buf = buflist_new(name, NULL, (linenr_T)1, 0); 400261f346fSBram Moolenaar 401261f346fSBram Moolenaar if (buf != NULL) 402261f346fSBram Moolenaar rettv->vval.v_number = buf->b_fnum; 403261f346fSBram Moolenaar else 404261f346fSBram Moolenaar rettv->vval.v_number = -1; 405261f346fSBram Moolenaar } 406261f346fSBram Moolenaar 407261f346fSBram Moolenaar static void 408261f346fSBram Moolenaar buf_win_common(typval_T *argvars, typval_T *rettv, int get_nr) 409261f346fSBram Moolenaar { 410261f346fSBram Moolenaar win_T *wp; 411261f346fSBram Moolenaar int winnr = 0; 412261f346fSBram Moolenaar buf_T *buf; 413261f346fSBram Moolenaar 414261f346fSBram Moolenaar (void)tv_get_number(&argvars[0]); // issue errmsg if type error 415261f346fSBram Moolenaar ++emsg_off; 416261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], TRUE); 417261f346fSBram Moolenaar FOR_ALL_WINDOWS(wp) 418261f346fSBram Moolenaar { 419261f346fSBram Moolenaar ++winnr; 420261f346fSBram Moolenaar if (wp->w_buffer == buf) 421261f346fSBram Moolenaar break; 422261f346fSBram Moolenaar } 423261f346fSBram Moolenaar rettv->vval.v_number = (wp != NULL ? (get_nr ? winnr : wp->w_id) : -1); 424261f346fSBram Moolenaar --emsg_off; 425261f346fSBram Moolenaar } 426261f346fSBram Moolenaar 427261f346fSBram Moolenaar /* 428261f346fSBram Moolenaar * "bufwinid(nr)" function 429261f346fSBram Moolenaar */ 430261f346fSBram Moolenaar void 431261f346fSBram Moolenaar f_bufwinid(typval_T *argvars, typval_T *rettv) 432261f346fSBram Moolenaar { 433261f346fSBram Moolenaar buf_win_common(argvars, rettv, FALSE); 434261f346fSBram Moolenaar } 435261f346fSBram Moolenaar 436261f346fSBram Moolenaar /* 437261f346fSBram Moolenaar * "bufwinnr(nr)" function 438261f346fSBram Moolenaar */ 439261f346fSBram Moolenaar void 440261f346fSBram Moolenaar f_bufwinnr(typval_T *argvars, typval_T *rettv) 441261f346fSBram Moolenaar { 442261f346fSBram Moolenaar buf_win_common(argvars, rettv, TRUE); 443261f346fSBram Moolenaar } 444261f346fSBram Moolenaar 445261f346fSBram Moolenaar /* 446261f346fSBram Moolenaar * "deletebufline()" function 447261f346fSBram Moolenaar */ 448261f346fSBram Moolenaar void 449261f346fSBram Moolenaar f_deletebufline(typval_T *argvars, typval_T *rettv) 450261f346fSBram Moolenaar { 451261f346fSBram Moolenaar buf_T *buf; 452261f346fSBram Moolenaar linenr_T first, last; 453261f346fSBram Moolenaar linenr_T lnum; 454261f346fSBram Moolenaar long count; 455261f346fSBram Moolenaar int is_curbuf; 456261f346fSBram Moolenaar buf_T *curbuf_save = NULL; 457261f346fSBram Moolenaar win_T *curwin_save = NULL; 458261f346fSBram Moolenaar tabpage_T *tp; 459261f346fSBram Moolenaar win_T *wp; 460261f346fSBram Moolenaar 461261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 462261f346fSBram Moolenaar if (buf == NULL) 463261f346fSBram Moolenaar { 464261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 465261f346fSBram Moolenaar return; 466261f346fSBram Moolenaar } 467261f346fSBram Moolenaar is_curbuf = buf == curbuf; 468261f346fSBram Moolenaar 469261f346fSBram Moolenaar first = tv_get_lnum_buf(&argvars[1], buf); 470261f346fSBram Moolenaar if (argvars[2].v_type != VAR_UNKNOWN) 471261f346fSBram Moolenaar last = tv_get_lnum_buf(&argvars[2], buf); 472261f346fSBram Moolenaar else 473261f346fSBram Moolenaar last = first; 474261f346fSBram Moolenaar 475261f346fSBram Moolenaar if (buf->b_ml.ml_mfp == NULL || first < 1 476261f346fSBram Moolenaar || first > buf->b_ml.ml_line_count || last < first) 477261f346fSBram Moolenaar { 478261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 479261f346fSBram Moolenaar return; 480261f346fSBram Moolenaar } 481261f346fSBram Moolenaar 482261f346fSBram Moolenaar if (!is_curbuf) 483261f346fSBram Moolenaar { 484261f346fSBram Moolenaar curbuf_save = curbuf; 485261f346fSBram Moolenaar curwin_save = curwin; 486261f346fSBram Moolenaar curbuf = buf; 487261f346fSBram Moolenaar find_win_for_curbuf(); 488261f346fSBram Moolenaar } 489261f346fSBram Moolenaar if (last > curbuf->b_ml.ml_line_count) 490261f346fSBram Moolenaar last = curbuf->b_ml.ml_line_count; 491261f346fSBram Moolenaar count = last - first + 1; 492261f346fSBram Moolenaar 493261f346fSBram Moolenaar // When coming here from Insert mode, sync undo, so that this can be 494261f346fSBram Moolenaar // undone separately from what was previously inserted. 495261f346fSBram Moolenaar if (u_sync_once == 2) 496261f346fSBram Moolenaar { 497261f346fSBram Moolenaar u_sync_once = 1; // notify that u_sync() was called 498261f346fSBram Moolenaar u_sync(TRUE); 499261f346fSBram Moolenaar } 500261f346fSBram Moolenaar 501261f346fSBram Moolenaar if (u_save(first - 1, last + 1) == FAIL) 502261f346fSBram Moolenaar { 503261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 504261f346fSBram Moolenaar return; 505261f346fSBram Moolenaar } 506261f346fSBram Moolenaar 507261f346fSBram Moolenaar for (lnum = first; lnum <= last; ++lnum) 508261f346fSBram Moolenaar ml_delete(first, TRUE); 509261f346fSBram Moolenaar 510261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 511261f346fSBram Moolenaar if (wp->w_buffer == buf) 512261f346fSBram Moolenaar { 513261f346fSBram Moolenaar if (wp->w_cursor.lnum > last) 514261f346fSBram Moolenaar wp->w_cursor.lnum -= count; 515261f346fSBram Moolenaar else if (wp->w_cursor.lnum> first) 516261f346fSBram Moolenaar wp->w_cursor.lnum = first; 517261f346fSBram Moolenaar if (wp->w_cursor.lnum > wp->w_buffer->b_ml.ml_line_count) 518261f346fSBram Moolenaar wp->w_cursor.lnum = wp->w_buffer->b_ml.ml_line_count; 519261f346fSBram Moolenaar } 520261f346fSBram Moolenaar check_cursor_col(); 521261f346fSBram Moolenaar deleted_lines_mark(first, count); 522261f346fSBram Moolenaar 523261f346fSBram Moolenaar if (!is_curbuf) 524261f346fSBram Moolenaar { 525261f346fSBram Moolenaar curbuf = curbuf_save; 526261f346fSBram Moolenaar curwin = curwin_save; 527261f346fSBram Moolenaar } 528261f346fSBram Moolenaar } 529261f346fSBram Moolenaar 530261f346fSBram Moolenaar /* 531261f346fSBram Moolenaar * Returns buffer options, variables and other attributes in a dictionary. 532261f346fSBram Moolenaar */ 533261f346fSBram Moolenaar static dict_T * 534261f346fSBram Moolenaar get_buffer_info(buf_T *buf) 535261f346fSBram Moolenaar { 536261f346fSBram Moolenaar dict_T *dict; 537261f346fSBram Moolenaar tabpage_T *tp; 538261f346fSBram Moolenaar win_T *wp; 539261f346fSBram Moolenaar list_T *windows; 540261f346fSBram Moolenaar 541261f346fSBram Moolenaar dict = dict_alloc(); 542261f346fSBram Moolenaar if (dict == NULL) 543261f346fSBram Moolenaar return NULL; 544261f346fSBram Moolenaar 545261f346fSBram Moolenaar dict_add_number(dict, "bufnr", buf->b_fnum); 546261f346fSBram Moolenaar dict_add_string(dict, "name", buf->b_ffname); 547261f346fSBram Moolenaar dict_add_number(dict, "lnum", buf == curbuf ? curwin->w_cursor.lnum 548261f346fSBram Moolenaar : buflist_findlnum(buf)); 549a9e9679dSBram Moolenaar dict_add_number(dict, "linecount", buf->b_ml.ml_line_count); 550261f346fSBram Moolenaar dict_add_number(dict, "loaded", buf->b_ml.ml_mfp != NULL); 551261f346fSBram Moolenaar dict_add_number(dict, "listed", buf->b_p_bl); 552261f346fSBram Moolenaar dict_add_number(dict, "changed", bufIsChanged(buf)); 553261f346fSBram Moolenaar dict_add_number(dict, "changedtick", CHANGEDTICK(buf)); 554261f346fSBram Moolenaar dict_add_number(dict, "hidden", 555261f346fSBram Moolenaar buf->b_ml.ml_mfp != NULL && buf->b_nwindows == 0); 556261f346fSBram Moolenaar 557261f346fSBram Moolenaar // Get a reference to buffer variables 558261f346fSBram Moolenaar dict_add_dict(dict, "variables", buf->b_vars); 559261f346fSBram Moolenaar 560261f346fSBram Moolenaar // List of windows displaying this buffer 561261f346fSBram Moolenaar windows = list_alloc(); 562261f346fSBram Moolenaar if (windows != NULL) 563261f346fSBram Moolenaar { 564261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 565261f346fSBram Moolenaar if (wp->w_buffer == buf) 566261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 567261f346fSBram Moolenaar dict_add_list(dict, "windows", windows); 568261f346fSBram Moolenaar } 569261f346fSBram Moolenaar 57005ad5ff0SBram Moolenaar #ifdef FEAT_PROP_POPUP 571261f346fSBram Moolenaar // List of popup windows displaying this buffer 572261f346fSBram Moolenaar windows = list_alloc(); 573261f346fSBram Moolenaar if (windows != NULL) 574261f346fSBram Moolenaar { 575261f346fSBram Moolenaar for (wp = first_popupwin; wp != NULL; wp = wp->w_next) 576261f346fSBram Moolenaar if (wp->w_buffer == buf) 577261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 578261f346fSBram Moolenaar FOR_ALL_TABPAGES(tp) 579261f346fSBram Moolenaar for (wp = tp->tp_first_popupwin; wp != NULL; wp = wp->w_next) 580261f346fSBram Moolenaar if (wp->w_buffer == buf) 581261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 582261f346fSBram Moolenaar 583261f346fSBram Moolenaar dict_add_list(dict, "popups", windows); 584261f346fSBram Moolenaar } 585261f346fSBram Moolenaar #endif 586261f346fSBram Moolenaar 587261f346fSBram Moolenaar #ifdef FEAT_SIGNS 588261f346fSBram Moolenaar if (buf->b_signlist != NULL) 589261f346fSBram Moolenaar { 590261f346fSBram Moolenaar // List of signs placed in this buffer 591261f346fSBram Moolenaar list_T *signs = list_alloc(); 592261f346fSBram Moolenaar if (signs != NULL) 593261f346fSBram Moolenaar { 594261f346fSBram Moolenaar get_buffer_signs(buf, signs); 595261f346fSBram Moolenaar dict_add_list(dict, "signs", signs); 596261f346fSBram Moolenaar } 597261f346fSBram Moolenaar } 598261f346fSBram Moolenaar #endif 599261f346fSBram Moolenaar 60052410575SBram Moolenaar #ifdef FEAT_VIMINFO 60152410575SBram Moolenaar dict_add_number(dict, "lastused", buf->b_last_used); 60252410575SBram Moolenaar #endif 60352410575SBram Moolenaar 604261f346fSBram Moolenaar return dict; 605261f346fSBram Moolenaar } 606261f346fSBram Moolenaar 607261f346fSBram Moolenaar /* 608261f346fSBram Moolenaar * "getbufinfo()" function 609261f346fSBram Moolenaar */ 610261f346fSBram Moolenaar void 611261f346fSBram Moolenaar f_getbufinfo(typval_T *argvars, typval_T *rettv) 612261f346fSBram Moolenaar { 613261f346fSBram Moolenaar buf_T *buf = NULL; 614261f346fSBram Moolenaar buf_T *argbuf = NULL; 615261f346fSBram Moolenaar dict_T *d; 616261f346fSBram Moolenaar int filtered = FALSE; 617261f346fSBram Moolenaar int sel_buflisted = FALSE; 618261f346fSBram Moolenaar int sel_bufloaded = FALSE; 619261f346fSBram Moolenaar int sel_bufmodified = FALSE; 620261f346fSBram Moolenaar 621261f346fSBram Moolenaar if (rettv_list_alloc(rettv) != OK) 622261f346fSBram Moolenaar return; 623261f346fSBram Moolenaar 624261f346fSBram Moolenaar // List of all the buffers or selected buffers 625261f346fSBram Moolenaar if (argvars[0].v_type == VAR_DICT) 626261f346fSBram Moolenaar { 627261f346fSBram Moolenaar dict_T *sel_d = argvars[0].vval.v_dict; 628261f346fSBram Moolenaar 629261f346fSBram Moolenaar if (sel_d != NULL) 630261f346fSBram Moolenaar { 631261f346fSBram Moolenaar dictitem_T *di; 632261f346fSBram Moolenaar 633261f346fSBram Moolenaar filtered = TRUE; 634261f346fSBram Moolenaar 635261f346fSBram Moolenaar di = dict_find(sel_d, (char_u *)"buflisted", -1); 636261f346fSBram Moolenaar if (di != NULL && tv_get_number(&di->di_tv)) 637261f346fSBram Moolenaar sel_buflisted = TRUE; 638261f346fSBram Moolenaar 639261f346fSBram Moolenaar di = dict_find(sel_d, (char_u *)"bufloaded", -1); 640261f346fSBram Moolenaar if (di != NULL && tv_get_number(&di->di_tv)) 641261f346fSBram Moolenaar sel_bufloaded = TRUE; 642261f346fSBram Moolenaar 643261f346fSBram Moolenaar di = dict_find(sel_d, (char_u *)"bufmodified", -1); 644261f346fSBram Moolenaar if (di != NULL && tv_get_number(&di->di_tv)) 645261f346fSBram Moolenaar sel_bufmodified = TRUE; 646261f346fSBram Moolenaar } 647261f346fSBram Moolenaar } 648261f346fSBram Moolenaar else if (argvars[0].v_type != VAR_UNKNOWN) 649261f346fSBram Moolenaar { 650261f346fSBram Moolenaar // Information about one buffer. Argument specifies the buffer 651261f346fSBram Moolenaar (void)tv_get_number(&argvars[0]); // issue errmsg if type error 652261f346fSBram Moolenaar ++emsg_off; 653261f346fSBram Moolenaar argbuf = tv_get_buf(&argvars[0], FALSE); 654261f346fSBram Moolenaar --emsg_off; 655261f346fSBram Moolenaar if (argbuf == NULL) 656261f346fSBram Moolenaar return; 657261f346fSBram Moolenaar } 658261f346fSBram Moolenaar 659261f346fSBram Moolenaar // Return information about all the buffers or a specified buffer 660261f346fSBram Moolenaar FOR_ALL_BUFFERS(buf) 661261f346fSBram Moolenaar { 662261f346fSBram Moolenaar if (argbuf != NULL && argbuf != buf) 663261f346fSBram Moolenaar continue; 664261f346fSBram Moolenaar if (filtered && ((sel_bufloaded && buf->b_ml.ml_mfp == NULL) 665261f346fSBram Moolenaar || (sel_buflisted && !buf->b_p_bl) 666261f346fSBram Moolenaar || (sel_bufmodified && !buf->b_changed))) 667261f346fSBram Moolenaar continue; 668261f346fSBram Moolenaar 669261f346fSBram Moolenaar d = get_buffer_info(buf); 670261f346fSBram Moolenaar if (d != NULL) 671261f346fSBram Moolenaar list_append_dict(rettv->vval.v_list, d); 672261f346fSBram Moolenaar if (argbuf != NULL) 673261f346fSBram Moolenaar return; 674261f346fSBram Moolenaar } 675261f346fSBram Moolenaar } 676261f346fSBram Moolenaar 677261f346fSBram Moolenaar /* 678261f346fSBram Moolenaar * Get line or list of lines from buffer "buf" into "rettv". 679261f346fSBram Moolenaar * Return a range (from start to end) of lines in rettv from the specified 680261f346fSBram Moolenaar * buffer. 681261f346fSBram Moolenaar * If 'retlist' is TRUE, then the lines are returned as a Vim List. 682261f346fSBram Moolenaar */ 683261f346fSBram Moolenaar static void 684261f346fSBram Moolenaar get_buffer_lines( 685261f346fSBram Moolenaar buf_T *buf, 686261f346fSBram Moolenaar linenr_T start, 687261f346fSBram Moolenaar linenr_T end, 688261f346fSBram Moolenaar int retlist, 689261f346fSBram Moolenaar typval_T *rettv) 690261f346fSBram Moolenaar { 691261f346fSBram Moolenaar char_u *p; 692261f346fSBram Moolenaar 6938a7d6542SBram Moolenaar if (retlist) 6948a7d6542SBram Moolenaar { 6958a7d6542SBram Moolenaar if (rettv_list_alloc(rettv) == FAIL) 6968a7d6542SBram Moolenaar return; 6978a7d6542SBram Moolenaar } 6988a7d6542SBram Moolenaar else 6998a7d6542SBram Moolenaar { 700261f346fSBram Moolenaar rettv->v_type = VAR_STRING; 701261f346fSBram Moolenaar rettv->vval.v_string = NULL; 7028a7d6542SBram Moolenaar } 703261f346fSBram Moolenaar 704261f346fSBram Moolenaar if (buf == NULL || buf->b_ml.ml_mfp == NULL || start < 0) 705261f346fSBram Moolenaar return; 706261f346fSBram Moolenaar 707261f346fSBram Moolenaar if (!retlist) 708261f346fSBram Moolenaar { 709261f346fSBram Moolenaar if (start >= 1 && start <= buf->b_ml.ml_line_count) 710261f346fSBram Moolenaar p = ml_get_buf(buf, start, FALSE); 711261f346fSBram Moolenaar else 712261f346fSBram Moolenaar p = (char_u *)""; 713261f346fSBram Moolenaar rettv->vval.v_string = vim_strsave(p); 714261f346fSBram Moolenaar } 715261f346fSBram Moolenaar else 716261f346fSBram Moolenaar { 717261f346fSBram Moolenaar if (end < start) 718261f346fSBram Moolenaar return; 719261f346fSBram Moolenaar 720261f346fSBram Moolenaar if (start < 1) 721261f346fSBram Moolenaar start = 1; 722261f346fSBram Moolenaar if (end > buf->b_ml.ml_line_count) 723261f346fSBram Moolenaar end = buf->b_ml.ml_line_count; 724261f346fSBram Moolenaar while (start <= end) 725261f346fSBram Moolenaar if (list_append_string(rettv->vval.v_list, 726261f346fSBram Moolenaar ml_get_buf(buf, start++, FALSE), -1) == FAIL) 727261f346fSBram Moolenaar break; 728261f346fSBram Moolenaar } 729261f346fSBram Moolenaar } 730261f346fSBram Moolenaar 731261f346fSBram Moolenaar /* 732261f346fSBram Moolenaar * "getbufline()" function 733261f346fSBram Moolenaar */ 734261f346fSBram Moolenaar void 735261f346fSBram Moolenaar f_getbufline(typval_T *argvars, typval_T *rettv) 736261f346fSBram Moolenaar { 737261f346fSBram Moolenaar linenr_T lnum; 738261f346fSBram Moolenaar linenr_T end; 739261f346fSBram Moolenaar buf_T *buf; 740261f346fSBram Moolenaar 741261f346fSBram Moolenaar (void)tv_get_number(&argvars[0]); // issue errmsg if type error 742261f346fSBram Moolenaar ++emsg_off; 743261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 744261f346fSBram Moolenaar --emsg_off; 745261f346fSBram Moolenaar 746261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 747261f346fSBram Moolenaar if (argvars[2].v_type == VAR_UNKNOWN) 748261f346fSBram Moolenaar end = lnum; 749261f346fSBram Moolenaar else 750261f346fSBram Moolenaar end = tv_get_lnum_buf(&argvars[2], buf); 751261f346fSBram Moolenaar 752261f346fSBram Moolenaar get_buffer_lines(buf, lnum, end, TRUE, rettv); 753261f346fSBram Moolenaar } 754261f346fSBram Moolenaar 755*fbdd08edSBram Moolenaar type_T * 756*fbdd08edSBram Moolenaar ret_f_getline(int argcount, type_T **argtypes UNUSED) 757*fbdd08edSBram Moolenaar { 758*fbdd08edSBram Moolenaar return argcount == 1 ? &t_string : &t_list_string; 759*fbdd08edSBram Moolenaar } 760*fbdd08edSBram Moolenaar 761261f346fSBram Moolenaar /* 762261f346fSBram Moolenaar * "getline(lnum, [end])" function 763261f346fSBram Moolenaar */ 764261f346fSBram Moolenaar void 765261f346fSBram Moolenaar f_getline(typval_T *argvars, typval_T *rettv) 766261f346fSBram Moolenaar { 767261f346fSBram Moolenaar linenr_T lnum; 768261f346fSBram Moolenaar linenr_T end; 769261f346fSBram Moolenaar int retlist; 770261f346fSBram Moolenaar 771261f346fSBram Moolenaar lnum = tv_get_lnum(argvars); 772261f346fSBram Moolenaar if (argvars[1].v_type == VAR_UNKNOWN) 773261f346fSBram Moolenaar { 774261f346fSBram Moolenaar end = 0; 775261f346fSBram Moolenaar retlist = FALSE; 776261f346fSBram Moolenaar } 777261f346fSBram Moolenaar else 778261f346fSBram Moolenaar { 779261f346fSBram Moolenaar end = tv_get_lnum(&argvars[1]); 780261f346fSBram Moolenaar retlist = TRUE; 781261f346fSBram Moolenaar } 782261f346fSBram Moolenaar 783261f346fSBram Moolenaar get_buffer_lines(curbuf, lnum, end, retlist, rettv); 784261f346fSBram Moolenaar } 785261f346fSBram Moolenaar 786261f346fSBram Moolenaar /* 787261f346fSBram Moolenaar * "setbufline()" function 788261f346fSBram Moolenaar */ 789261f346fSBram Moolenaar void 790261f346fSBram Moolenaar f_setbufline(typval_T *argvars, typval_T *rettv) 791261f346fSBram Moolenaar { 792261f346fSBram Moolenaar linenr_T lnum; 793261f346fSBram Moolenaar buf_T *buf; 794261f346fSBram Moolenaar 795261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 796261f346fSBram Moolenaar if (buf == NULL) 797261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 798261f346fSBram Moolenaar else 799261f346fSBram Moolenaar { 800261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 801261f346fSBram Moolenaar set_buffer_lines(buf, lnum, FALSE, &argvars[2], rettv); 802261f346fSBram Moolenaar } 803261f346fSBram Moolenaar } 804261f346fSBram Moolenaar 805261f346fSBram Moolenaar /* 806261f346fSBram Moolenaar * "setline()" function 807261f346fSBram Moolenaar */ 808261f346fSBram Moolenaar void 809261f346fSBram Moolenaar f_setline(typval_T *argvars, typval_T *rettv) 810261f346fSBram Moolenaar { 811261f346fSBram Moolenaar linenr_T lnum = tv_get_lnum(&argvars[0]); 812261f346fSBram Moolenaar 813261f346fSBram Moolenaar set_buffer_lines(curbuf, lnum, FALSE, &argvars[1], rettv); 814261f346fSBram Moolenaar } 815261f346fSBram Moolenaar #endif // FEAT_EVAL 816261f346fSBram Moolenaar 817261f346fSBram Moolenaar #if defined(FEAT_JOB_CHANNEL) \ 818261f346fSBram Moolenaar || defined(FEAT_PYTHON) || defined(FEAT_PYTHON3) \ 819261f346fSBram Moolenaar || defined(PROTO) 820261f346fSBram Moolenaar /* 821261f346fSBram Moolenaar * Make "buf" the current buffer. restore_buffer() MUST be called to undo. 822261f346fSBram Moolenaar * No autocommands will be executed. Use aucmd_prepbuf() if there are any. 823261f346fSBram Moolenaar */ 824261f346fSBram Moolenaar void 825261f346fSBram Moolenaar switch_buffer(bufref_T *save_curbuf, buf_T *buf) 826261f346fSBram Moolenaar { 827261f346fSBram Moolenaar block_autocmds(); 828261f346fSBram Moolenaar set_bufref(save_curbuf, curbuf); 829261f346fSBram Moolenaar --curbuf->b_nwindows; 830261f346fSBram Moolenaar curbuf = buf; 831261f346fSBram Moolenaar curwin->w_buffer = buf; 832261f346fSBram Moolenaar ++curbuf->b_nwindows; 833261f346fSBram Moolenaar } 834261f346fSBram Moolenaar 835261f346fSBram Moolenaar /* 836261f346fSBram Moolenaar * Restore the current buffer after using switch_buffer(). 837261f346fSBram Moolenaar */ 838261f346fSBram Moolenaar void 839261f346fSBram Moolenaar restore_buffer(bufref_T *save_curbuf) 840261f346fSBram Moolenaar { 841261f346fSBram Moolenaar unblock_autocmds(); 8425d18efecSBram Moolenaar // Check for valid buffer, just in case. 843261f346fSBram Moolenaar if (bufref_valid(save_curbuf)) 844261f346fSBram Moolenaar { 845261f346fSBram Moolenaar --curbuf->b_nwindows; 846261f346fSBram Moolenaar curwin->w_buffer = save_curbuf->br_buf; 847261f346fSBram Moolenaar curbuf = save_curbuf->br_buf; 848261f346fSBram Moolenaar ++curbuf->b_nwindows; 849261f346fSBram Moolenaar } 850261f346fSBram Moolenaar } 851261f346fSBram Moolenaar 852261f346fSBram Moolenaar /* 853261f346fSBram Moolenaar * Find a window for buffer "buf". 854261f346fSBram Moolenaar * If found OK is returned and "wp" and "tp" are set to the window and tabpage. 855261f346fSBram Moolenaar * If not found FAIL is returned. 856261f346fSBram Moolenaar */ 857261f346fSBram Moolenaar static int 858261f346fSBram Moolenaar find_win_for_buf( 859261f346fSBram Moolenaar buf_T *buf, 860261f346fSBram Moolenaar win_T **wp, 861261f346fSBram Moolenaar tabpage_T **tp) 862261f346fSBram Moolenaar { 863261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(*tp, *wp) 864261f346fSBram Moolenaar if ((*wp)->w_buffer == buf) 865261f346fSBram Moolenaar return OK; 866261f346fSBram Moolenaar return FAIL; 867261f346fSBram Moolenaar } 868261f346fSBram Moolenaar 869261f346fSBram Moolenaar /* 870261f346fSBram Moolenaar * Find a window that contains "buf" and switch to it. 871261f346fSBram Moolenaar * If there is no such window, use the current window and change "curbuf". 872261f346fSBram Moolenaar * Caller must initialize save_curbuf to NULL. 873261f346fSBram Moolenaar * restore_win_for_buf() MUST be called later! 874261f346fSBram Moolenaar */ 875261f346fSBram Moolenaar void 876261f346fSBram Moolenaar switch_to_win_for_buf( 877261f346fSBram Moolenaar buf_T *buf, 878261f346fSBram Moolenaar win_T **save_curwinp, 879261f346fSBram Moolenaar tabpage_T **save_curtabp, 880261f346fSBram Moolenaar bufref_T *save_curbuf) 881261f346fSBram Moolenaar { 882261f346fSBram Moolenaar win_T *wp; 883261f346fSBram Moolenaar tabpage_T *tp; 884261f346fSBram Moolenaar 885261f346fSBram Moolenaar if (find_win_for_buf(buf, &wp, &tp) == FAIL) 886261f346fSBram Moolenaar switch_buffer(save_curbuf, buf); 887261f346fSBram Moolenaar else if (switch_win(save_curwinp, save_curtabp, wp, tp, TRUE) == FAIL) 888261f346fSBram Moolenaar { 889261f346fSBram Moolenaar restore_win(*save_curwinp, *save_curtabp, TRUE); 890261f346fSBram Moolenaar switch_buffer(save_curbuf, buf); 891261f346fSBram Moolenaar } 892261f346fSBram Moolenaar } 893261f346fSBram Moolenaar 894261f346fSBram Moolenaar void 895261f346fSBram Moolenaar restore_win_for_buf( 896261f346fSBram Moolenaar win_T *save_curwin, 897261f346fSBram Moolenaar tabpage_T *save_curtab, 898261f346fSBram Moolenaar bufref_T *save_curbuf) 899261f346fSBram Moolenaar { 900261f346fSBram Moolenaar if (save_curbuf->br_buf == NULL) 901261f346fSBram Moolenaar restore_win(save_curwin, save_curtab, TRUE); 902261f346fSBram Moolenaar else 903261f346fSBram Moolenaar restore_buffer(save_curbuf); 904261f346fSBram Moolenaar } 905261f346fSBram Moolenaar #endif 906