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; 71e5a2dc87SBram Moolenaar p_cpo = empty_option; 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); 917b45d46cSBram Moolenaar else if (in_vim9script() && check_for_string_arg(avar, 0) == FAIL) 927b45d46cSBram Moolenaar return NULL; 93261f346fSBram Moolenaar else if (avar->v_type == VAR_STRING && avar->vval.v_string != NULL) 94261f346fSBram Moolenaar { 95261f346fSBram Moolenaar buf = buflist_findname_exp(avar->vval.v_string); 96261f346fSBram Moolenaar if (buf == NULL) 97261f346fSBram Moolenaar { 98261f346fSBram Moolenaar // No full path name match, try a match with a URL or a "nofile" 99261f346fSBram Moolenaar // buffer, these don't use the full path. 100261f346fSBram Moolenaar FOR_ALL_BUFFERS(buf) 101261f346fSBram Moolenaar if (buf->b_fname != NULL 102261f346fSBram Moolenaar && (path_with_url(buf->b_fname) 103261f346fSBram Moolenaar #ifdef FEAT_QUICKFIX 104261f346fSBram Moolenaar || bt_nofilename(buf) 105261f346fSBram Moolenaar #endif 106261f346fSBram Moolenaar ) 107261f346fSBram Moolenaar && STRCMP(buf->b_fname, avar->vval.v_string) == 0) 108261f346fSBram Moolenaar break; 109261f346fSBram Moolenaar } 110261f346fSBram Moolenaar } 111261f346fSBram Moolenaar return buf; 112261f346fSBram Moolenaar } 113261f346fSBram Moolenaar 114261f346fSBram Moolenaar /* 115261f346fSBram Moolenaar * If there is a window for "curbuf", make it the current window. 116261f346fSBram Moolenaar */ 117261f346fSBram Moolenaar static void 118261f346fSBram Moolenaar find_win_for_curbuf(void) 119261f346fSBram Moolenaar { 120261f346fSBram Moolenaar wininfo_T *wip; 121261f346fSBram Moolenaar 122aeea7215SBram Moolenaar FOR_ALL_BUF_WININFO(curbuf, wip) 123261f346fSBram Moolenaar { 124261f346fSBram Moolenaar if (wip->wi_win != NULL) 125261f346fSBram Moolenaar { 126261f346fSBram Moolenaar curwin = wip->wi_win; 127261f346fSBram Moolenaar break; 128261f346fSBram Moolenaar } 129261f346fSBram Moolenaar } 130261f346fSBram Moolenaar } 131261f346fSBram Moolenaar 132261f346fSBram Moolenaar /* 13334453208SBram Moolenaar * Set line or list of lines in buffer "buf" to "lines". 13434453208SBram Moolenaar * Any type is allowed and converted to a string. 135261f346fSBram Moolenaar */ 136261f346fSBram Moolenaar static void 137261f346fSBram Moolenaar set_buffer_lines( 138261f346fSBram Moolenaar buf_T *buf, 139261f346fSBram Moolenaar linenr_T lnum_arg, 140261f346fSBram Moolenaar int append, 141261f346fSBram Moolenaar typval_T *lines, 142261f346fSBram Moolenaar typval_T *rettv) 143261f346fSBram Moolenaar { 144261f346fSBram Moolenaar linenr_T lnum = lnum_arg + (append ? 1 : 0); 145261f346fSBram Moolenaar char_u *line = NULL; 146261f346fSBram Moolenaar list_T *l = NULL; 147261f346fSBram Moolenaar listitem_T *li = NULL; 148261f346fSBram Moolenaar long added = 0; 149261f346fSBram Moolenaar linenr_T append_lnum; 150261f346fSBram Moolenaar buf_T *curbuf_save = NULL; 151261f346fSBram Moolenaar win_T *curwin_save = NULL; 152261f346fSBram Moolenaar int is_curbuf = buf == curbuf; 153261f346fSBram Moolenaar 154261f346fSBram Moolenaar // When using the current buffer ml_mfp will be set if needed. Useful when 155261f346fSBram Moolenaar // setline() is used on startup. For other buffers the buffer must be 156261f346fSBram Moolenaar // loaded. 157261f346fSBram Moolenaar if (buf == NULL || (!is_curbuf && buf->b_ml.ml_mfp == NULL) || lnum < 1) 158261f346fSBram Moolenaar { 159261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 160261f346fSBram Moolenaar return; 161261f346fSBram Moolenaar } 162261f346fSBram Moolenaar 163261f346fSBram Moolenaar if (!is_curbuf) 164261f346fSBram Moolenaar { 165261f346fSBram Moolenaar curbuf_save = curbuf; 166261f346fSBram Moolenaar curwin_save = curwin; 167261f346fSBram Moolenaar curbuf = buf; 168261f346fSBram Moolenaar find_win_for_curbuf(); 169261f346fSBram Moolenaar } 170261f346fSBram Moolenaar 171261f346fSBram Moolenaar if (append) 172261f346fSBram Moolenaar // appendbufline() uses the line number below which we insert 173261f346fSBram Moolenaar append_lnum = lnum - 1; 174261f346fSBram Moolenaar else 175261f346fSBram Moolenaar // setbufline() uses the line number above which we insert, we only 176261f346fSBram Moolenaar // append if it's below the last line 177261f346fSBram Moolenaar append_lnum = curbuf->b_ml.ml_line_count; 178261f346fSBram Moolenaar 179261f346fSBram Moolenaar if (lines->v_type == VAR_LIST) 180261f346fSBram Moolenaar { 181261f346fSBram Moolenaar l = lines->vval.v_list; 182ad48e6c1SBram Moolenaar if (l == NULL || list_len(l) == 0) 183ad48e6c1SBram Moolenaar { 184ad48e6c1SBram Moolenaar // set proper return code 185ad48e6c1SBram Moolenaar if (lnum > curbuf->b_ml.ml_line_count) 186ad48e6c1SBram Moolenaar rettv->vval.v_number = 1; // FAIL 187ad48e6c1SBram Moolenaar goto done; 188ad48e6c1SBram Moolenaar } 1897e9f351bSBram Moolenaar CHECK_LIST_MATERIALIZE(l); 190261f346fSBram Moolenaar li = l->lv_first; 191261f346fSBram Moolenaar } 192261f346fSBram Moolenaar else 19334453208SBram Moolenaar line = typval_tostring(lines, FALSE); 194261f346fSBram Moolenaar 195261f346fSBram Moolenaar // default result is zero == OK 196261f346fSBram Moolenaar for (;;) 197261f346fSBram Moolenaar { 198261f346fSBram Moolenaar if (l != NULL) 199261f346fSBram Moolenaar { 200261f346fSBram Moolenaar // list argument, get next string 201261f346fSBram Moolenaar if (li == NULL) 202261f346fSBram Moolenaar break; 20334453208SBram Moolenaar vim_free(line); 20434453208SBram Moolenaar line = typval_tostring(&li->li_tv, FALSE); 205261f346fSBram Moolenaar li = li->li_next; 206261f346fSBram Moolenaar } 207261f346fSBram Moolenaar 208261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 209261f346fSBram Moolenaar if (line == NULL || lnum > curbuf->b_ml.ml_line_count + 1) 210261f346fSBram Moolenaar break; 211261f346fSBram Moolenaar 212261f346fSBram Moolenaar // When coming here from Insert mode, sync undo, so that this can be 213261f346fSBram Moolenaar // undone separately from what was previously inserted. 214261f346fSBram Moolenaar if (u_sync_once == 2) 215261f346fSBram Moolenaar { 216261f346fSBram Moolenaar u_sync_once = 1; // notify that u_sync() was called 217261f346fSBram Moolenaar u_sync(TRUE); 218261f346fSBram Moolenaar } 219261f346fSBram Moolenaar 220261f346fSBram Moolenaar if (!append && lnum <= curbuf->b_ml.ml_line_count) 221261f346fSBram Moolenaar { 222261f346fSBram Moolenaar // Existing line, replace it. 223261f346fSBram Moolenaar // Removes any existing text properties. 224261f346fSBram Moolenaar if (u_savesub(lnum) == OK && ml_replace_len( 225261f346fSBram Moolenaar lnum, line, (colnr_T)STRLEN(line) + 1, TRUE, TRUE) == OK) 226261f346fSBram Moolenaar { 227261f346fSBram Moolenaar changed_bytes(lnum, 0); 228261f346fSBram Moolenaar if (is_curbuf && lnum == curwin->w_cursor.lnum) 229261f346fSBram Moolenaar check_cursor_col(); 230261f346fSBram Moolenaar rettv->vval.v_number = 0; // OK 231261f346fSBram Moolenaar } 232261f346fSBram Moolenaar } 233261f346fSBram Moolenaar else if (added > 0 || u_save(lnum - 1, lnum) == OK) 234261f346fSBram Moolenaar { 235261f346fSBram Moolenaar // append the line 236261f346fSBram Moolenaar ++added; 237261f346fSBram Moolenaar if (ml_append(lnum - 1, line, (colnr_T)0, FALSE) == OK) 238261f346fSBram Moolenaar rettv->vval.v_number = 0; // OK 239261f346fSBram Moolenaar } 240261f346fSBram Moolenaar 241261f346fSBram Moolenaar if (l == NULL) // only one string argument 242261f346fSBram Moolenaar break; 243261f346fSBram Moolenaar ++lnum; 244261f346fSBram Moolenaar } 24534453208SBram Moolenaar vim_free(line); 246261f346fSBram Moolenaar 247261f346fSBram Moolenaar if (added > 0) 248261f346fSBram Moolenaar { 249261f346fSBram Moolenaar win_T *wp; 250261f346fSBram Moolenaar tabpage_T *tp; 251261f346fSBram Moolenaar 252261f346fSBram Moolenaar appended_lines_mark(append_lnum, added); 253261f346fSBram Moolenaar 254261f346fSBram Moolenaar // Only adjust the cursor for buffers other than the current, unless it 255261f346fSBram Moolenaar // is the current window. For curbuf and other windows it has been 256261f346fSBram Moolenaar // done in mark_adjust_internal(). 257261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 258261f346fSBram Moolenaar if (wp->w_buffer == buf 259261f346fSBram Moolenaar && (wp->w_buffer != curbuf || wp == curwin) 260261f346fSBram Moolenaar && wp->w_cursor.lnum > append_lnum) 261261f346fSBram Moolenaar wp->w_cursor.lnum += added; 262261f346fSBram Moolenaar check_cursor_col(); 263261f346fSBram Moolenaar update_topline(); 264261f346fSBram Moolenaar } 265261f346fSBram Moolenaar 266ad48e6c1SBram Moolenaar done: 267261f346fSBram Moolenaar if (!is_curbuf) 268261f346fSBram Moolenaar { 269261f346fSBram Moolenaar curbuf = curbuf_save; 270261f346fSBram Moolenaar curwin = curwin_save; 271261f346fSBram Moolenaar } 272261f346fSBram Moolenaar } 273261f346fSBram Moolenaar 274261f346fSBram Moolenaar /* 275261f346fSBram Moolenaar * "append(lnum, string/list)" function 276261f346fSBram Moolenaar */ 277261f346fSBram Moolenaar void 278261f346fSBram Moolenaar f_append(typval_T *argvars, typval_T *rettv) 279261f346fSBram Moolenaar { 280*4490ec4eSYegappan Lakshmanan linenr_T lnum; 281261f346fSBram Moolenaar 282*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_lnum_arg(argvars, 0) == FAIL) 283*4490ec4eSYegappan Lakshmanan return; 284*4490ec4eSYegappan Lakshmanan 285*4490ec4eSYegappan Lakshmanan lnum = tv_get_lnum(&argvars[0]); 286261f346fSBram Moolenaar set_buffer_lines(curbuf, lnum, TRUE, &argvars[1], rettv); 287261f346fSBram Moolenaar } 288261f346fSBram Moolenaar 289261f346fSBram Moolenaar /* 290261f346fSBram Moolenaar * "appendbufline(buf, lnum, string/list)" function 291261f346fSBram Moolenaar */ 292261f346fSBram Moolenaar void 293261f346fSBram Moolenaar f_appendbufline(typval_T *argvars, typval_T *rettv) 294261f346fSBram Moolenaar { 295261f346fSBram Moolenaar linenr_T lnum; 296261f346fSBram Moolenaar buf_T *buf; 297261f346fSBram Moolenaar 298*4490ec4eSYegappan Lakshmanan if (in_vim9script() 299*4490ec4eSYegappan Lakshmanan && (check_for_buffer_arg(argvars, 0) == FAIL 300*4490ec4eSYegappan Lakshmanan || check_for_lnum_arg(argvars, 1) == FAIL 301*4490ec4eSYegappan Lakshmanan || check_for_string_or_number_or_list_arg(argvars, 2) == FAIL)) 302*4490ec4eSYegappan Lakshmanan return; 303*4490ec4eSYegappan Lakshmanan 304261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 305261f346fSBram Moolenaar if (buf == NULL) 306261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 307261f346fSBram Moolenaar else 308261f346fSBram Moolenaar { 309261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 310261f346fSBram Moolenaar set_buffer_lines(buf, lnum, TRUE, &argvars[2], rettv); 311261f346fSBram Moolenaar } 312261f346fSBram Moolenaar } 313261f346fSBram Moolenaar 314261f346fSBram Moolenaar /* 315261f346fSBram Moolenaar * "bufadd(expr)" function 316261f346fSBram Moolenaar */ 317261f346fSBram Moolenaar void 318261f346fSBram Moolenaar f_bufadd(typval_T *argvars, typval_T *rettv) 319261f346fSBram Moolenaar { 320*4490ec4eSYegappan Lakshmanan char_u *name; 321261f346fSBram Moolenaar 322*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_string_arg(argvars, 0) == FAIL) 323*4490ec4eSYegappan Lakshmanan return; 324*4490ec4eSYegappan Lakshmanan 325*4490ec4eSYegappan Lakshmanan name = tv_get_string(&argvars[0]); 326261f346fSBram Moolenaar rettv->vval.v_number = buflist_add(*name == NUL ? NULL : name, 0); 327261f346fSBram Moolenaar } 328261f346fSBram Moolenaar 329261f346fSBram Moolenaar /* 330261f346fSBram Moolenaar * "bufexists(expr)" function 331261f346fSBram Moolenaar */ 332261f346fSBram Moolenaar void 333261f346fSBram Moolenaar f_bufexists(typval_T *argvars, typval_T *rettv) 334261f346fSBram Moolenaar { 335*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_buffer_arg(argvars, 0) == FAIL) 336*4490ec4eSYegappan Lakshmanan return; 337*4490ec4eSYegappan Lakshmanan 338261f346fSBram Moolenaar rettv->vval.v_number = (find_buffer(&argvars[0]) != NULL); 339261f346fSBram Moolenaar } 340261f346fSBram Moolenaar 341261f346fSBram Moolenaar /* 342261f346fSBram Moolenaar * "buflisted(expr)" function 343261f346fSBram Moolenaar */ 344261f346fSBram Moolenaar void 345261f346fSBram Moolenaar f_buflisted(typval_T *argvars, typval_T *rettv) 346261f346fSBram Moolenaar { 347261f346fSBram Moolenaar buf_T *buf; 348261f346fSBram Moolenaar 349*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_buffer_arg(argvars, 0) == FAIL) 350*4490ec4eSYegappan Lakshmanan return; 351*4490ec4eSYegappan Lakshmanan 352261f346fSBram Moolenaar buf = find_buffer(&argvars[0]); 353261f346fSBram Moolenaar rettv->vval.v_number = (buf != NULL && buf->b_p_bl); 354261f346fSBram Moolenaar } 355261f346fSBram Moolenaar 356261f346fSBram Moolenaar /* 357261f346fSBram Moolenaar * "bufload(expr)" function 358261f346fSBram Moolenaar */ 359261f346fSBram Moolenaar void 360261f346fSBram Moolenaar f_bufload(typval_T *argvars, typval_T *rettv UNUSED) 361261f346fSBram Moolenaar { 362*4490ec4eSYegappan Lakshmanan buf_T *buf; 363261f346fSBram Moolenaar 364*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_buffer_arg(argvars, 0) == FAIL) 365*4490ec4eSYegappan Lakshmanan return; 366*4490ec4eSYegappan Lakshmanan 367*4490ec4eSYegappan Lakshmanan buf = get_buf_arg(&argvars[0]); 368261f346fSBram Moolenaar if (buf != NULL) 369261f346fSBram Moolenaar buffer_ensure_loaded(buf); 370261f346fSBram Moolenaar } 371261f346fSBram Moolenaar 372261f346fSBram Moolenaar /* 373261f346fSBram Moolenaar * "bufloaded(expr)" function 374261f346fSBram Moolenaar */ 375261f346fSBram Moolenaar void 376261f346fSBram Moolenaar f_bufloaded(typval_T *argvars, typval_T *rettv) 377261f346fSBram Moolenaar { 378261f346fSBram Moolenaar buf_T *buf; 379261f346fSBram Moolenaar 380*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_buffer_arg(argvars, 0) == FAIL) 381*4490ec4eSYegappan Lakshmanan return; 382*4490ec4eSYegappan Lakshmanan 383261f346fSBram Moolenaar buf = find_buffer(&argvars[0]); 384261f346fSBram Moolenaar rettv->vval.v_number = (buf != NULL && buf->b_ml.ml_mfp != NULL); 385261f346fSBram Moolenaar } 386261f346fSBram Moolenaar 387261f346fSBram Moolenaar /* 388261f346fSBram Moolenaar * "bufname(expr)" function 389261f346fSBram Moolenaar */ 390261f346fSBram Moolenaar void 391261f346fSBram Moolenaar f_bufname(typval_T *argvars, typval_T *rettv) 392261f346fSBram Moolenaar { 393261f346fSBram Moolenaar buf_T *buf; 39402aaad91SBram Moolenaar typval_T *tv = &argvars[0]; 395261f346fSBram Moolenaar 396*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_opt_buffer_arg(argvars, 0) == FAIL) 397*4490ec4eSYegappan Lakshmanan return; 398*4490ec4eSYegappan Lakshmanan 39902aaad91SBram Moolenaar if (tv->v_type == VAR_UNKNOWN) 400261f346fSBram Moolenaar buf = curbuf; 401261f346fSBram Moolenaar else 4023767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(tv); 403261f346fSBram Moolenaar rettv->v_type = VAR_STRING; 404261f346fSBram Moolenaar if (buf != NULL && buf->b_fname != NULL) 405261f346fSBram Moolenaar rettv->vval.v_string = vim_strsave(buf->b_fname); 406261f346fSBram Moolenaar else 407261f346fSBram Moolenaar rettv->vval.v_string = NULL; 408261f346fSBram Moolenaar } 409261f346fSBram Moolenaar 410261f346fSBram Moolenaar /* 411261f346fSBram Moolenaar * "bufnr(expr)" function 412261f346fSBram Moolenaar */ 413261f346fSBram Moolenaar void 414261f346fSBram Moolenaar f_bufnr(typval_T *argvars, typval_T *rettv) 415261f346fSBram Moolenaar { 416261f346fSBram Moolenaar buf_T *buf; 417261f346fSBram Moolenaar int error = FALSE; 418261f346fSBram Moolenaar char_u *name; 419261f346fSBram Moolenaar 42083494b4aSYegappan Lakshmanan if (in_vim9script() 421cd917207SYegappan Lakshmanan && (check_for_opt_buffer_arg(argvars, 0) == FAIL 42283494b4aSYegappan Lakshmanan || (argvars[0].v_type != VAR_UNKNOWN 42383494b4aSYegappan Lakshmanan && check_for_opt_bool_arg(argvars, 1) == FAIL))) 42483494b4aSYegappan Lakshmanan return; 42583494b4aSYegappan Lakshmanan 426261f346fSBram Moolenaar if (argvars[0].v_type == VAR_UNKNOWN) 427261f346fSBram Moolenaar buf = curbuf; 428261f346fSBram Moolenaar else 4293767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(&argvars[0]); 430261f346fSBram Moolenaar 431261f346fSBram Moolenaar // If the buffer isn't found and the second argument is not zero create a 432261f346fSBram Moolenaar // new buffer. 433261f346fSBram Moolenaar if (buf == NULL 434261f346fSBram Moolenaar && argvars[1].v_type != VAR_UNKNOWN 435fe136c9aSBram Moolenaar && tv_get_bool_chk(&argvars[1], &error) != 0 436261f346fSBram Moolenaar && !error 437261f346fSBram Moolenaar && (name = tv_get_string_chk(&argvars[0])) != NULL 438261f346fSBram Moolenaar && !error) 439261f346fSBram Moolenaar buf = buflist_new(name, NULL, (linenr_T)1, 0); 440261f346fSBram Moolenaar 441261f346fSBram Moolenaar if (buf != NULL) 442261f346fSBram Moolenaar rettv->vval.v_number = buf->b_fnum; 443261f346fSBram Moolenaar else 444261f346fSBram Moolenaar rettv->vval.v_number = -1; 445261f346fSBram Moolenaar } 446261f346fSBram Moolenaar 447261f346fSBram Moolenaar static void 448261f346fSBram Moolenaar buf_win_common(typval_T *argvars, typval_T *rettv, int get_nr) 449261f346fSBram Moolenaar { 450261f346fSBram Moolenaar win_T *wp; 451261f346fSBram Moolenaar int winnr = 0; 452261f346fSBram Moolenaar buf_T *buf; 453261f346fSBram Moolenaar 454*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_buffer_arg(argvars, 0) == FAIL) 455*4490ec4eSYegappan Lakshmanan return; 456*4490ec4eSYegappan Lakshmanan 4573767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(&argvars[0]); 458261f346fSBram Moolenaar FOR_ALL_WINDOWS(wp) 459261f346fSBram Moolenaar { 460261f346fSBram Moolenaar ++winnr; 461261f346fSBram Moolenaar if (wp->w_buffer == buf) 462261f346fSBram Moolenaar break; 463261f346fSBram Moolenaar } 464261f346fSBram Moolenaar rettv->vval.v_number = (wp != NULL ? (get_nr ? winnr : wp->w_id) : -1); 465261f346fSBram Moolenaar } 466261f346fSBram Moolenaar 467261f346fSBram Moolenaar /* 468261f346fSBram Moolenaar * "bufwinid(nr)" function 469261f346fSBram Moolenaar */ 470261f346fSBram Moolenaar void 471261f346fSBram Moolenaar f_bufwinid(typval_T *argvars, typval_T *rettv) 472261f346fSBram Moolenaar { 473261f346fSBram Moolenaar buf_win_common(argvars, rettv, FALSE); 474261f346fSBram Moolenaar } 475261f346fSBram Moolenaar 476261f346fSBram Moolenaar /* 477261f346fSBram Moolenaar * "bufwinnr(nr)" function 478261f346fSBram Moolenaar */ 479261f346fSBram Moolenaar void 480261f346fSBram Moolenaar f_bufwinnr(typval_T *argvars, typval_T *rettv) 481261f346fSBram Moolenaar { 482261f346fSBram Moolenaar buf_win_common(argvars, rettv, TRUE); 483261f346fSBram Moolenaar } 484261f346fSBram Moolenaar 485261f346fSBram Moolenaar /* 486261f346fSBram Moolenaar * "deletebufline()" function 487261f346fSBram Moolenaar */ 488261f346fSBram Moolenaar void 489261f346fSBram Moolenaar f_deletebufline(typval_T *argvars, typval_T *rettv) 490261f346fSBram Moolenaar { 491261f346fSBram Moolenaar buf_T *buf; 492261f346fSBram Moolenaar linenr_T first, last; 493261f346fSBram Moolenaar linenr_T lnum; 494261f346fSBram Moolenaar long count; 495261f346fSBram Moolenaar int is_curbuf; 496261f346fSBram Moolenaar buf_T *curbuf_save = NULL; 497261f346fSBram Moolenaar win_T *curwin_save = NULL; 498261f346fSBram Moolenaar tabpage_T *tp; 499261f346fSBram Moolenaar win_T *wp; 500261f346fSBram Moolenaar 50183494b4aSYegappan Lakshmanan if (in_vim9script() 50283494b4aSYegappan Lakshmanan && (check_for_buffer_arg(argvars, 0) == FAIL 50383494b4aSYegappan Lakshmanan || check_for_lnum_arg(argvars, 1) == FAIL 50483494b4aSYegappan Lakshmanan || check_for_opt_lnum_arg(argvars, 2) == FAIL)) 50583494b4aSYegappan Lakshmanan return; 50683494b4aSYegappan Lakshmanan 507261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 508261f346fSBram Moolenaar if (buf == NULL) 509261f346fSBram Moolenaar { 510261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 511261f346fSBram Moolenaar return; 512261f346fSBram Moolenaar } 513261f346fSBram Moolenaar is_curbuf = buf == curbuf; 514261f346fSBram Moolenaar 515261f346fSBram Moolenaar first = tv_get_lnum_buf(&argvars[1], buf); 516261f346fSBram Moolenaar if (argvars[2].v_type != VAR_UNKNOWN) 517261f346fSBram Moolenaar last = tv_get_lnum_buf(&argvars[2], buf); 518261f346fSBram Moolenaar else 519261f346fSBram Moolenaar last = first; 520261f346fSBram Moolenaar 521261f346fSBram Moolenaar if (buf->b_ml.ml_mfp == NULL || first < 1 522261f346fSBram Moolenaar || first > buf->b_ml.ml_line_count || last < first) 523261f346fSBram Moolenaar { 524261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 525261f346fSBram Moolenaar return; 526261f346fSBram Moolenaar } 527261f346fSBram Moolenaar 528261f346fSBram Moolenaar if (!is_curbuf) 529261f346fSBram Moolenaar { 530261f346fSBram Moolenaar curbuf_save = curbuf; 531261f346fSBram Moolenaar curwin_save = curwin; 532261f346fSBram Moolenaar curbuf = buf; 533261f346fSBram Moolenaar find_win_for_curbuf(); 534261f346fSBram Moolenaar } 535261f346fSBram Moolenaar if (last > curbuf->b_ml.ml_line_count) 536261f346fSBram Moolenaar last = curbuf->b_ml.ml_line_count; 537261f346fSBram Moolenaar count = last - first + 1; 538261f346fSBram Moolenaar 539261f346fSBram Moolenaar // When coming here from Insert mode, sync undo, so that this can be 540261f346fSBram Moolenaar // undone separately from what was previously inserted. 541261f346fSBram Moolenaar if (u_sync_once == 2) 542261f346fSBram Moolenaar { 543261f346fSBram Moolenaar u_sync_once = 1; // notify that u_sync() was called 544261f346fSBram Moolenaar u_sync(TRUE); 545261f346fSBram Moolenaar } 546261f346fSBram Moolenaar 547261f346fSBram Moolenaar if (u_save(first - 1, last + 1) == FAIL) 548261f346fSBram Moolenaar { 549261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 550261f346fSBram Moolenaar } 551963ffa0aSBram Moolenaar else 552963ffa0aSBram Moolenaar { 553261f346fSBram Moolenaar for (lnum = first; lnum <= last; ++lnum) 554ca70c07bSBram Moolenaar ml_delete_flags(first, ML_DEL_MESSAGE); 555261f346fSBram Moolenaar 556261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 557261f346fSBram Moolenaar if (wp->w_buffer == buf) 558261f346fSBram Moolenaar { 559261f346fSBram Moolenaar if (wp->w_cursor.lnum > last) 560261f346fSBram Moolenaar wp->w_cursor.lnum -= count; 561261f346fSBram Moolenaar else if (wp->w_cursor.lnum> first) 562261f346fSBram Moolenaar wp->w_cursor.lnum = first; 563261f346fSBram Moolenaar if (wp->w_cursor.lnum > wp->w_buffer->b_ml.ml_line_count) 564261f346fSBram Moolenaar wp->w_cursor.lnum = wp->w_buffer->b_ml.ml_line_count; 565261f346fSBram Moolenaar } 566261f346fSBram Moolenaar check_cursor_col(); 567261f346fSBram Moolenaar deleted_lines_mark(first, count); 568963ffa0aSBram Moolenaar } 569261f346fSBram Moolenaar 570261f346fSBram Moolenaar if (!is_curbuf) 571261f346fSBram Moolenaar { 572261f346fSBram Moolenaar curbuf = curbuf_save; 573261f346fSBram Moolenaar curwin = curwin_save; 574261f346fSBram Moolenaar } 575261f346fSBram Moolenaar } 576261f346fSBram Moolenaar 577261f346fSBram Moolenaar /* 578261f346fSBram Moolenaar * Returns buffer options, variables and other attributes in a dictionary. 579261f346fSBram Moolenaar */ 580261f346fSBram Moolenaar static dict_T * 581261f346fSBram Moolenaar get_buffer_info(buf_T *buf) 582261f346fSBram Moolenaar { 583261f346fSBram Moolenaar dict_T *dict; 584261f346fSBram Moolenaar tabpage_T *tp; 585261f346fSBram Moolenaar win_T *wp; 586261f346fSBram Moolenaar list_T *windows; 587261f346fSBram Moolenaar 588261f346fSBram Moolenaar dict = dict_alloc(); 589261f346fSBram Moolenaar if (dict == NULL) 590261f346fSBram Moolenaar return NULL; 591261f346fSBram Moolenaar 592261f346fSBram Moolenaar dict_add_number(dict, "bufnr", buf->b_fnum); 593261f346fSBram Moolenaar dict_add_string(dict, "name", buf->b_ffname); 594261f346fSBram Moolenaar dict_add_number(dict, "lnum", buf == curbuf ? curwin->w_cursor.lnum 595261f346fSBram Moolenaar : buflist_findlnum(buf)); 596a9e9679dSBram Moolenaar dict_add_number(dict, "linecount", buf->b_ml.ml_line_count); 597261f346fSBram Moolenaar dict_add_number(dict, "loaded", buf->b_ml.ml_mfp != NULL); 598261f346fSBram Moolenaar dict_add_number(dict, "listed", buf->b_p_bl); 599261f346fSBram Moolenaar dict_add_number(dict, "changed", bufIsChanged(buf)); 600261f346fSBram Moolenaar dict_add_number(dict, "changedtick", CHANGEDTICK(buf)); 601261f346fSBram Moolenaar dict_add_number(dict, "hidden", 602261f346fSBram Moolenaar buf->b_ml.ml_mfp != NULL && buf->b_nwindows == 0); 603261f346fSBram Moolenaar 604261f346fSBram Moolenaar // Get a reference to buffer variables 605261f346fSBram Moolenaar dict_add_dict(dict, "variables", buf->b_vars); 606261f346fSBram Moolenaar 607261f346fSBram Moolenaar // List of windows displaying this buffer 608261f346fSBram Moolenaar windows = list_alloc(); 609261f346fSBram Moolenaar if (windows != NULL) 610261f346fSBram Moolenaar { 611261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(tp, wp) 612261f346fSBram Moolenaar if (wp->w_buffer == buf) 613261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 614261f346fSBram Moolenaar dict_add_list(dict, "windows", windows); 615261f346fSBram Moolenaar } 616261f346fSBram Moolenaar 61705ad5ff0SBram Moolenaar #ifdef FEAT_PROP_POPUP 618261f346fSBram Moolenaar // List of popup windows displaying this buffer 619261f346fSBram Moolenaar windows = list_alloc(); 620261f346fSBram Moolenaar if (windows != NULL) 621261f346fSBram Moolenaar { 622aeea7215SBram Moolenaar FOR_ALL_POPUPWINS(wp) 623261f346fSBram Moolenaar if (wp->w_buffer == buf) 624261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 625261f346fSBram Moolenaar FOR_ALL_TABPAGES(tp) 626aeea7215SBram Moolenaar FOR_ALL_POPUPWINS_IN_TAB(tp, wp) 627261f346fSBram Moolenaar if (wp->w_buffer == buf) 628261f346fSBram Moolenaar list_append_number(windows, (varnumber_T)wp->w_id); 629261f346fSBram Moolenaar 630261f346fSBram Moolenaar dict_add_list(dict, "popups", windows); 631261f346fSBram Moolenaar } 632261f346fSBram Moolenaar #endif 633261f346fSBram Moolenaar 634261f346fSBram Moolenaar #ifdef FEAT_SIGNS 635261f346fSBram Moolenaar if (buf->b_signlist != NULL) 636261f346fSBram Moolenaar { 637261f346fSBram Moolenaar // List of signs placed in this buffer 638261f346fSBram Moolenaar list_T *signs = list_alloc(); 639261f346fSBram Moolenaar if (signs != NULL) 640261f346fSBram Moolenaar { 641261f346fSBram Moolenaar get_buffer_signs(buf, signs); 642261f346fSBram Moolenaar dict_add_list(dict, "signs", signs); 643261f346fSBram Moolenaar } 644261f346fSBram Moolenaar } 645261f346fSBram Moolenaar #endif 646261f346fSBram Moolenaar 64752410575SBram Moolenaar #ifdef FEAT_VIMINFO 64852410575SBram Moolenaar dict_add_number(dict, "lastused", buf->b_last_used); 64952410575SBram Moolenaar #endif 65052410575SBram Moolenaar 651261f346fSBram Moolenaar return dict; 652261f346fSBram Moolenaar } 653261f346fSBram Moolenaar 654261f346fSBram Moolenaar /* 655261f346fSBram Moolenaar * "getbufinfo()" function 656261f346fSBram Moolenaar */ 657261f346fSBram Moolenaar void 658261f346fSBram Moolenaar f_getbufinfo(typval_T *argvars, typval_T *rettv) 659261f346fSBram Moolenaar { 660261f346fSBram Moolenaar buf_T *buf = NULL; 661261f346fSBram Moolenaar buf_T *argbuf = NULL; 662261f346fSBram Moolenaar dict_T *d; 663261f346fSBram Moolenaar int filtered = FALSE; 664261f346fSBram Moolenaar int sel_buflisted = FALSE; 665261f346fSBram Moolenaar int sel_bufloaded = FALSE; 666261f346fSBram Moolenaar int sel_bufmodified = FALSE; 667261f346fSBram Moolenaar 668261f346fSBram Moolenaar if (rettv_list_alloc(rettv) != OK) 669261f346fSBram Moolenaar return; 670261f346fSBram Moolenaar 671cd917207SYegappan Lakshmanan if (in_vim9script() 672*4490ec4eSYegappan Lakshmanan && check_for_opt_buffer_or_dict_arg(argvars, 0) == FAIL) 673cd917207SYegappan Lakshmanan return; 674cd917207SYegappan Lakshmanan 675261f346fSBram Moolenaar // List of all the buffers or selected buffers 676261f346fSBram Moolenaar if (argvars[0].v_type == VAR_DICT) 677261f346fSBram Moolenaar { 678261f346fSBram Moolenaar dict_T *sel_d = argvars[0].vval.v_dict; 679261f346fSBram Moolenaar 680261f346fSBram Moolenaar if (sel_d != NULL) 681261f346fSBram Moolenaar { 682261f346fSBram Moolenaar filtered = TRUE; 683036c2cf7SBram Moolenaar sel_buflisted = dict_get_bool(sel_d, (char_u *)"buflisted", FALSE); 684036c2cf7SBram Moolenaar sel_bufloaded = dict_get_bool(sel_d, (char_u *)"bufloaded", FALSE); 685036c2cf7SBram Moolenaar sel_bufmodified = dict_get_bool(sel_d, (char_u *)"bufmodified", 686036c2cf7SBram Moolenaar FALSE); 687261f346fSBram Moolenaar } 688261f346fSBram Moolenaar } 689261f346fSBram Moolenaar else if (argvars[0].v_type != VAR_UNKNOWN) 690261f346fSBram Moolenaar { 691261f346fSBram Moolenaar // Information about one buffer. Argument specifies the buffer 6923767e3a3SBram Moolenaar argbuf = tv_get_buf_from_arg(&argvars[0]); 693261f346fSBram Moolenaar if (argbuf == NULL) 694261f346fSBram Moolenaar return; 695261f346fSBram Moolenaar } 696261f346fSBram Moolenaar 697261f346fSBram Moolenaar // Return information about all the buffers or a specified buffer 698261f346fSBram Moolenaar FOR_ALL_BUFFERS(buf) 699261f346fSBram Moolenaar { 700261f346fSBram Moolenaar if (argbuf != NULL && argbuf != buf) 701261f346fSBram Moolenaar continue; 702261f346fSBram Moolenaar if (filtered && ((sel_bufloaded && buf->b_ml.ml_mfp == NULL) 703261f346fSBram Moolenaar || (sel_buflisted && !buf->b_p_bl) 704261f346fSBram Moolenaar || (sel_bufmodified && !buf->b_changed))) 705261f346fSBram Moolenaar continue; 706261f346fSBram Moolenaar 707261f346fSBram Moolenaar d = get_buffer_info(buf); 708261f346fSBram Moolenaar if (d != NULL) 709261f346fSBram Moolenaar list_append_dict(rettv->vval.v_list, d); 710261f346fSBram Moolenaar if (argbuf != NULL) 711261f346fSBram Moolenaar return; 712261f346fSBram Moolenaar } 713261f346fSBram Moolenaar } 714261f346fSBram Moolenaar 715261f346fSBram Moolenaar /* 716261f346fSBram Moolenaar * Get line or list of lines from buffer "buf" into "rettv". 717261f346fSBram Moolenaar * Return a range (from start to end) of lines in rettv from the specified 718261f346fSBram Moolenaar * buffer. 719261f346fSBram Moolenaar * If 'retlist' is TRUE, then the lines are returned as a Vim List. 720261f346fSBram Moolenaar */ 721261f346fSBram Moolenaar static void 722261f346fSBram Moolenaar get_buffer_lines( 723261f346fSBram Moolenaar buf_T *buf, 724261f346fSBram Moolenaar linenr_T start, 725261f346fSBram Moolenaar linenr_T end, 726261f346fSBram Moolenaar int retlist, 727261f346fSBram Moolenaar typval_T *rettv) 728261f346fSBram Moolenaar { 729261f346fSBram Moolenaar char_u *p; 730261f346fSBram Moolenaar 7318a7d6542SBram Moolenaar if (retlist) 7328a7d6542SBram Moolenaar { 7338a7d6542SBram Moolenaar if (rettv_list_alloc(rettv) == FAIL) 7348a7d6542SBram Moolenaar return; 7358a7d6542SBram Moolenaar } 7368a7d6542SBram Moolenaar else 7378a7d6542SBram Moolenaar { 738261f346fSBram Moolenaar rettv->v_type = VAR_STRING; 739261f346fSBram Moolenaar rettv->vval.v_string = NULL; 7408a7d6542SBram Moolenaar } 741261f346fSBram Moolenaar 742261f346fSBram Moolenaar if (buf == NULL || buf->b_ml.ml_mfp == NULL || start < 0) 743261f346fSBram Moolenaar return; 744261f346fSBram Moolenaar 745261f346fSBram Moolenaar if (!retlist) 746261f346fSBram Moolenaar { 747261f346fSBram Moolenaar if (start >= 1 && start <= buf->b_ml.ml_line_count) 748261f346fSBram Moolenaar p = ml_get_buf(buf, start, FALSE); 749261f346fSBram Moolenaar else 750261f346fSBram Moolenaar p = (char_u *)""; 751261f346fSBram Moolenaar rettv->vval.v_string = vim_strsave(p); 752261f346fSBram Moolenaar } 753261f346fSBram Moolenaar else 754261f346fSBram Moolenaar { 755261f346fSBram Moolenaar if (end < start) 756261f346fSBram Moolenaar return; 757261f346fSBram Moolenaar 758261f346fSBram Moolenaar if (start < 1) 759261f346fSBram Moolenaar start = 1; 760261f346fSBram Moolenaar if (end > buf->b_ml.ml_line_count) 761261f346fSBram Moolenaar end = buf->b_ml.ml_line_count; 762261f346fSBram Moolenaar while (start <= end) 763261f346fSBram Moolenaar if (list_append_string(rettv->vval.v_list, 764261f346fSBram Moolenaar ml_get_buf(buf, start++, FALSE), -1) == FAIL) 765261f346fSBram Moolenaar break; 766261f346fSBram Moolenaar } 767261f346fSBram Moolenaar } 768261f346fSBram Moolenaar 769261f346fSBram Moolenaar /* 770261f346fSBram Moolenaar * "getbufline()" function 771261f346fSBram Moolenaar */ 772261f346fSBram Moolenaar void 773261f346fSBram Moolenaar f_getbufline(typval_T *argvars, typval_T *rettv) 774261f346fSBram Moolenaar { 775e6e70a10SBram Moolenaar linenr_T lnum = 1; 776e6e70a10SBram Moolenaar linenr_T end = 1; 777261f346fSBram Moolenaar buf_T *buf; 778261f346fSBram Moolenaar 77983494b4aSYegappan Lakshmanan if (in_vim9script() 78083494b4aSYegappan Lakshmanan && (check_for_buffer_arg(argvars, 0) == FAIL 78183494b4aSYegappan Lakshmanan || check_for_lnum_arg(argvars, 1) == FAIL 78283494b4aSYegappan Lakshmanan || check_for_opt_lnum_arg(argvars, 2) == FAIL)) 78383494b4aSYegappan Lakshmanan return; 78483494b4aSYegappan Lakshmanan 7853767e3a3SBram Moolenaar buf = tv_get_buf_from_arg(&argvars[0]); 786e6e70a10SBram Moolenaar if (buf != NULL) 787e6e70a10SBram Moolenaar { 788261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 789261f346fSBram Moolenaar if (argvars[2].v_type == VAR_UNKNOWN) 790261f346fSBram Moolenaar end = lnum; 791261f346fSBram Moolenaar else 792261f346fSBram Moolenaar end = tv_get_lnum_buf(&argvars[2], buf); 793e6e70a10SBram Moolenaar } 794261f346fSBram Moolenaar 795261f346fSBram Moolenaar get_buffer_lines(buf, lnum, end, TRUE, rettv); 796261f346fSBram Moolenaar } 797261f346fSBram Moolenaar 798fbdd08edSBram Moolenaar type_T * 799fbdd08edSBram Moolenaar ret_f_getline(int argcount, type_T **argtypes UNUSED) 800fbdd08edSBram Moolenaar { 801fbdd08edSBram Moolenaar return argcount == 1 ? &t_string : &t_list_string; 802fbdd08edSBram Moolenaar } 803fbdd08edSBram Moolenaar 804261f346fSBram Moolenaar /* 805261f346fSBram Moolenaar * "getline(lnum, [end])" function 806261f346fSBram Moolenaar */ 807261f346fSBram Moolenaar void 808261f346fSBram Moolenaar f_getline(typval_T *argvars, typval_T *rettv) 809261f346fSBram Moolenaar { 810261f346fSBram Moolenaar linenr_T lnum; 811261f346fSBram Moolenaar linenr_T end; 812261f346fSBram Moolenaar int retlist; 813261f346fSBram Moolenaar 814cd917207SYegappan Lakshmanan if (in_vim9script() 815cd917207SYegappan Lakshmanan && (check_for_lnum_arg(argvars, 0) == FAIL 816cd917207SYegappan Lakshmanan || check_for_opt_lnum_arg(argvars, 1) == FAIL)) 817cd917207SYegappan Lakshmanan return; 818cd917207SYegappan Lakshmanan 819261f346fSBram Moolenaar lnum = tv_get_lnum(argvars); 820261f346fSBram Moolenaar if (argvars[1].v_type == VAR_UNKNOWN) 821261f346fSBram Moolenaar { 822261f346fSBram Moolenaar end = 0; 823261f346fSBram Moolenaar retlist = FALSE; 824261f346fSBram Moolenaar } 825261f346fSBram Moolenaar else 826261f346fSBram Moolenaar { 827261f346fSBram Moolenaar end = tv_get_lnum(&argvars[1]); 828261f346fSBram Moolenaar retlist = TRUE; 829261f346fSBram Moolenaar } 830261f346fSBram Moolenaar 831261f346fSBram Moolenaar get_buffer_lines(curbuf, lnum, end, retlist, rettv); 832261f346fSBram Moolenaar } 833261f346fSBram Moolenaar 834261f346fSBram Moolenaar /* 835261f346fSBram Moolenaar * "setbufline()" function 836261f346fSBram Moolenaar */ 837261f346fSBram Moolenaar void 838261f346fSBram Moolenaar f_setbufline(typval_T *argvars, typval_T *rettv) 839261f346fSBram Moolenaar { 840261f346fSBram Moolenaar linenr_T lnum; 841261f346fSBram Moolenaar buf_T *buf; 842261f346fSBram Moolenaar 843*4490ec4eSYegappan Lakshmanan if (in_vim9script() 844*4490ec4eSYegappan Lakshmanan && (check_for_buffer_arg(argvars, 0) == FAIL 845*4490ec4eSYegappan Lakshmanan || check_for_lnum_arg(argvars, 1) == FAIL 846*4490ec4eSYegappan Lakshmanan || check_for_string_or_number_or_list_arg(argvars, 2) == FAIL)) 847*4490ec4eSYegappan Lakshmanan return; 848*4490ec4eSYegappan Lakshmanan 849261f346fSBram Moolenaar buf = tv_get_buf(&argvars[0], FALSE); 850261f346fSBram Moolenaar if (buf == NULL) 851261f346fSBram Moolenaar rettv->vval.v_number = 1; // FAIL 852261f346fSBram Moolenaar else 853261f346fSBram Moolenaar { 854261f346fSBram Moolenaar lnum = tv_get_lnum_buf(&argvars[1], buf); 855261f346fSBram Moolenaar set_buffer_lines(buf, lnum, FALSE, &argvars[2], rettv); 856261f346fSBram Moolenaar } 857261f346fSBram Moolenaar } 858261f346fSBram Moolenaar 859261f346fSBram Moolenaar /* 860261f346fSBram Moolenaar * "setline()" function 861261f346fSBram Moolenaar */ 862261f346fSBram Moolenaar void 863261f346fSBram Moolenaar f_setline(typval_T *argvars, typval_T *rettv) 864261f346fSBram Moolenaar { 865*4490ec4eSYegappan Lakshmanan linenr_T lnum; 866261f346fSBram Moolenaar 867*4490ec4eSYegappan Lakshmanan if (in_vim9script() && check_for_lnum_arg(argvars, 0) == FAIL) 868*4490ec4eSYegappan Lakshmanan return; 869*4490ec4eSYegappan Lakshmanan 870*4490ec4eSYegappan Lakshmanan lnum = tv_get_lnum(&argvars[0]); 871261f346fSBram Moolenaar set_buffer_lines(curbuf, lnum, FALSE, &argvars[1], rettv); 872261f346fSBram Moolenaar } 873261f346fSBram Moolenaar #endif // FEAT_EVAL 874261f346fSBram Moolenaar 875261f346fSBram Moolenaar #if defined(FEAT_JOB_CHANNEL) \ 876261f346fSBram Moolenaar || defined(FEAT_PYTHON) || defined(FEAT_PYTHON3) \ 877261f346fSBram Moolenaar || defined(PROTO) 878261f346fSBram Moolenaar /* 879261f346fSBram Moolenaar * Make "buf" the current buffer. restore_buffer() MUST be called to undo. 880261f346fSBram Moolenaar * No autocommands will be executed. Use aucmd_prepbuf() if there are any. 881261f346fSBram Moolenaar */ 882261f346fSBram Moolenaar void 883261f346fSBram Moolenaar switch_buffer(bufref_T *save_curbuf, buf_T *buf) 884261f346fSBram Moolenaar { 885261f346fSBram Moolenaar block_autocmds(); 8863e0107eaSBram Moolenaar #ifdef FEAT_FOLDING 8873e0107eaSBram Moolenaar ++disable_fold_update; 8883e0107eaSBram Moolenaar #endif 889261f346fSBram Moolenaar set_bufref(save_curbuf, curbuf); 890261f346fSBram Moolenaar --curbuf->b_nwindows; 891261f346fSBram Moolenaar curbuf = buf; 892261f346fSBram Moolenaar curwin->w_buffer = buf; 893261f346fSBram Moolenaar ++curbuf->b_nwindows; 894261f346fSBram Moolenaar } 895261f346fSBram Moolenaar 896261f346fSBram Moolenaar /* 897261f346fSBram Moolenaar * Restore the current buffer after using switch_buffer(). 898261f346fSBram Moolenaar */ 899261f346fSBram Moolenaar void 900261f346fSBram Moolenaar restore_buffer(bufref_T *save_curbuf) 901261f346fSBram Moolenaar { 902261f346fSBram Moolenaar unblock_autocmds(); 9033e0107eaSBram Moolenaar #ifdef FEAT_FOLDING 9043e0107eaSBram Moolenaar --disable_fold_update; 9053e0107eaSBram Moolenaar #endif 9065d18efecSBram Moolenaar // Check for valid buffer, just in case. 907261f346fSBram Moolenaar if (bufref_valid(save_curbuf)) 908261f346fSBram Moolenaar { 909261f346fSBram Moolenaar --curbuf->b_nwindows; 910261f346fSBram Moolenaar curwin->w_buffer = save_curbuf->br_buf; 911261f346fSBram Moolenaar curbuf = save_curbuf->br_buf; 912261f346fSBram Moolenaar ++curbuf->b_nwindows; 913261f346fSBram Moolenaar } 914261f346fSBram Moolenaar } 915261f346fSBram Moolenaar 916261f346fSBram Moolenaar /* 917261f346fSBram Moolenaar * Find a window for buffer "buf". 918261f346fSBram Moolenaar * If found OK is returned and "wp" and "tp" are set to the window and tabpage. 919261f346fSBram Moolenaar * If not found FAIL is returned. 920261f346fSBram Moolenaar */ 921261f346fSBram Moolenaar static int 922261f346fSBram Moolenaar find_win_for_buf( 923261f346fSBram Moolenaar buf_T *buf, 924261f346fSBram Moolenaar win_T **wp, 925261f346fSBram Moolenaar tabpage_T **tp) 926261f346fSBram Moolenaar { 927261f346fSBram Moolenaar FOR_ALL_TAB_WINDOWS(*tp, *wp) 928261f346fSBram Moolenaar if ((*wp)->w_buffer == buf) 929261f346fSBram Moolenaar return OK; 930261f346fSBram Moolenaar return FAIL; 931261f346fSBram Moolenaar } 932261f346fSBram Moolenaar 933261f346fSBram Moolenaar /* 934261f346fSBram Moolenaar * Find a window that contains "buf" and switch to it. 935261f346fSBram Moolenaar * If there is no such window, use the current window and change "curbuf". 936261f346fSBram Moolenaar * Caller must initialize save_curbuf to NULL. 937261f346fSBram Moolenaar * restore_win_for_buf() MUST be called later! 938261f346fSBram Moolenaar */ 939261f346fSBram Moolenaar void 940261f346fSBram Moolenaar switch_to_win_for_buf( 941261f346fSBram Moolenaar buf_T *buf, 942261f346fSBram Moolenaar win_T **save_curwinp, 943261f346fSBram Moolenaar tabpage_T **save_curtabp, 944261f346fSBram Moolenaar bufref_T *save_curbuf) 945261f346fSBram Moolenaar { 946261f346fSBram Moolenaar win_T *wp; 947261f346fSBram Moolenaar tabpage_T *tp; 948261f346fSBram Moolenaar 949261f346fSBram Moolenaar if (find_win_for_buf(buf, &wp, &tp) == FAIL) 950261f346fSBram Moolenaar switch_buffer(save_curbuf, buf); 951261f346fSBram Moolenaar else if (switch_win(save_curwinp, save_curtabp, wp, tp, TRUE) == FAIL) 952261f346fSBram Moolenaar { 953261f346fSBram Moolenaar restore_win(*save_curwinp, *save_curtabp, TRUE); 954261f346fSBram Moolenaar switch_buffer(save_curbuf, buf); 955261f346fSBram Moolenaar } 956261f346fSBram Moolenaar } 957261f346fSBram Moolenaar 958261f346fSBram Moolenaar void 959261f346fSBram Moolenaar restore_win_for_buf( 960261f346fSBram Moolenaar win_T *save_curwin, 961261f346fSBram Moolenaar tabpage_T *save_curtab, 962261f346fSBram Moolenaar bufref_T *save_curbuf) 963261f346fSBram Moolenaar { 964261f346fSBram Moolenaar if (save_curbuf->br_buf == NULL) 965261f346fSBram Moolenaar restore_win(save_curwin, save_curtab, TRUE); 966261f346fSBram Moolenaar else 967261f346fSBram Moolenaar restore_buffer(save_curbuf); 968261f346fSBram Moolenaar } 969261f346fSBram Moolenaar #endif 970