blob: bfc8bbd1213329bd03ca6a909fd790950fb4b3aa [file] [log] [blame]
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001/* vi:set ts=8 sts=4 sw=4 noet:
2 *
3 * VIM - Vi IMproved by Bram Moolenaar
4 *
5 * Do ":help uganda" in Vim to read copying and usage conditions.
6 * Do ":help credits" in Vim to see a list of people who contributed.
7 * See README.txt for an overview of the Vim source code.
8 */
9
10/*
11 * arglist.c: functions for dealing with the argument list
12 */
13
14#include "vim.h"
15
16#define AL_SET 1
17#define AL_ADD 2
18#define AL_DEL 3
19
Bram Moolenaar5ed58c72021-01-28 14:24:55 +010020// This flag is set whenever the argument list is being changed and calling a
21// function that might trigger an autocommand.
22static int arglist_locked = FALSE;
23
24 static int
25check_arglist_locked(void)
26{
27 if (arglist_locked)
28 {
29 emsg(_(e_cannot_change_arglist_recursively));
30 return FAIL;
31 }
32 return OK;
33}
34
Bram Moolenaar4ad62152019-08-17 14:38:55 +020035/*
36 * Clear an argument list: free all file names and reset it to zero entries.
37 */
38 void
39alist_clear(alist_T *al)
40{
Bram Moolenaar5ed58c72021-01-28 14:24:55 +010041 if (check_arglist_locked() == FAIL)
42 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +020043 while (--al->al_ga.ga_len >= 0)
44 vim_free(AARGLIST(al)[al->al_ga.ga_len].ae_fname);
45 ga_clear(&al->al_ga);
46}
47
48/*
49 * Init an argument list.
50 */
51 void
52alist_init(alist_T *al)
53{
Bram Moolenaar04935fb2022-01-08 16:19:22 +000054 ga_init2(&al->al_ga, sizeof(aentry_T), 5);
Bram Moolenaar4ad62152019-08-17 14:38:55 +020055}
56
57/*
58 * Remove a reference from an argument list.
59 * Ignored when the argument list is the global one.
60 * If the argument list is no longer used by any window, free it.
61 */
62 void
63alist_unlink(alist_T *al)
64{
65 if (al != &global_alist && --al->al_refcount <= 0)
66 {
67 alist_clear(al);
68 vim_free(al);
69 }
70}
71
72/*
73 * Create a new argument list and use it for the current window.
74 */
75 void
76alist_new(void)
77{
78 curwin->w_alist = ALLOC_ONE(alist_T);
79 if (curwin->w_alist == NULL)
80 {
81 curwin->w_alist = &global_alist;
82 ++global_alist.al_refcount;
83 }
84 else
85 {
86 curwin->w_alist->al_refcount = 1;
87 curwin->w_alist->id = ++max_alist_id;
88 alist_init(curwin->w_alist);
89 }
90}
91
92#if !defined(UNIX) || defined(PROTO)
93/*
94 * Expand the file names in the global argument list.
95 * If "fnum_list" is not NULL, use "fnum_list[fnum_len]" as a list of buffer
96 * numbers to be re-used.
97 */
98 void
99alist_expand(int *fnum_list, int fnum_len)
100{
101 char_u **old_arg_files;
102 int old_arg_count;
103 char_u **new_arg_files;
104 int new_arg_file_count;
105 char_u *save_p_su = p_su;
106 int i;
107
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100108 old_arg_files = ALLOC_MULT(char_u *, GARGCOUNT);
109 if (old_arg_files == NULL)
110 return;
111
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200112 // Don't use 'suffixes' here. This should work like the shell did the
113 // expansion. Also, the vimrc file isn't read yet, thus the user
114 // can't set the options.
115 p_su = empty_option;
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100116 for (i = 0; i < GARGCOUNT; ++i)
117 old_arg_files[i] = vim_strsave(GARGLIST[i].ae_fname);
118 old_arg_count = GARGCOUNT;
119 if (expand_wildcards(old_arg_count, old_arg_files,
120 &new_arg_file_count, &new_arg_files,
121 EW_FILE|EW_NOTFOUND|EW_ADDSLASH|EW_NOERROR) == OK
122 && new_arg_file_count > 0)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200123 {
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100124 alist_set(&global_alist, new_arg_file_count, new_arg_files,
125 TRUE, fnum_list, fnum_len);
126 FreeWild(old_arg_count, old_arg_files);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200127 }
128 p_su = save_p_su;
129}
130#endif
131
132/*
133 * Set the argument list for the current window.
134 * Takes over the allocated files[] and the allocated fnames in it.
135 */
136 void
137alist_set(
138 alist_T *al,
139 int count,
140 char_u **files,
141 int use_curbuf,
142 int *fnum_list,
143 int fnum_len)
144{
145 int i;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200146
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100147 if (check_arglist_locked() == FAIL)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200148 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200149
150 alist_clear(al);
Bram Moolenaar35578162021-08-02 19:10:38 +0200151 if (GA_GROW_OK(&al->al_ga, count))
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200152 {
153 for (i = 0; i < count; ++i)
154 {
155 if (got_int)
156 {
157 // When adding many buffers this can take a long time. Allow
158 // interrupting here.
159 while (i < count)
160 vim_free(files[i++]);
161 break;
162 }
163
164 // May set buffer name of a buffer previously used for the
165 // argument list, so that it's re-used by alist_add.
166 if (fnum_list != NULL && i < fnum_len)
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100167 {
168 arglist_locked = TRUE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200169 buf_set_name(fnum_list[i], files[i]);
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100170 arglist_locked = FALSE;
171 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200172
173 alist_add(al, files[i], use_curbuf ? 2 : 1);
174 ui_breakcheck();
175 }
176 vim_free(files);
177 }
178 else
179 FreeWild(count, files);
180 if (al == &global_alist)
181 arg_had_last = FALSE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200182}
183
184/*
185 * Add file "fname" to argument list "al".
186 * "fname" must have been allocated and "al" must have been checked for room.
Christian Brabandt0a6e57b2024-08-15 22:15:28 +0200187 *
188 * May trigger Buf* autocommands
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200189 */
190 void
191alist_add(
192 alist_T *al,
193 char_u *fname,
194 int set_fnum) // 1: set buffer number; 2: re-use curbuf
195{
196 if (fname == NULL) // don't add NULL file names
197 return;
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100198 if (check_arglist_locked() == FAIL)
199 return;
200 arglist_locked = TRUE;
Christian Brabandt0a6e57b2024-08-15 22:15:28 +0200201 curwin->w_locked = TRUE;
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100202
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200203#ifdef BACKSLASH_IN_FILENAME
204 slash_adjust(fname);
205#endif
206 AARGLIST(al)[al->al_ga.ga_len].ae_fname = fname;
207 if (set_fnum > 0)
208 AARGLIST(al)[al->al_ga.ga_len].ae_fnum =
209 buflist_add(fname, BLN_LISTED | (set_fnum == 2 ? BLN_CURBUF : 0));
210 ++al->al_ga.ga_len;
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100211
212 arglist_locked = FALSE;
Christian Brabandt0a6e57b2024-08-15 22:15:28 +0200213 curwin->w_locked = FALSE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200214}
215
216#if defined(BACKSLASH_IN_FILENAME) || defined(PROTO)
217/*
218 * Adjust slashes in file names. Called after 'shellslash' was set.
219 */
220 void
221alist_slash_adjust(void)
222{
223 int i;
224 win_T *wp;
225 tabpage_T *tp;
226
227 for (i = 0; i < GARGCOUNT; ++i)
228 if (GARGLIST[i].ae_fname != NULL)
229 slash_adjust(GARGLIST[i].ae_fname);
230 FOR_ALL_TAB_WINDOWS(tp, wp)
231 if (wp->w_alist != &global_alist)
232 for (i = 0; i < WARGCOUNT(wp); ++i)
233 if (WARGLIST(wp)[i].ae_fname != NULL)
234 slash_adjust(WARGLIST(wp)[i].ae_fname);
235}
236#endif
237
238/*
239 * Isolate one argument, taking backticks.
240 * Changes the argument in-place, puts a NUL after it. Backticks remain.
241 * Return a pointer to the start of the next argument.
242 */
243 static char_u *
244do_one_arg(char_u *str)
245{
246 char_u *p;
247 int inbacktick;
248
249 inbacktick = FALSE;
250 for (p = str; *str; ++str)
251 {
252 // When the backslash is used for escaping the special meaning of a
253 // character we need to keep it until wildcard expansion.
254 if (rem_backslash(str))
255 {
256 *p++ = *str++;
257 *p++ = *str;
258 }
259 else
260 {
261 // An item ends at a space not in backticks
262 if (!inbacktick && vim_isspace(*str))
263 break;
264 if (*str == '`')
265 inbacktick ^= TRUE;
266 *p++ = *str;
267 }
268 }
269 str = skipwhite(str);
270 *p = NUL;
271
272 return str;
273}
274
275/*
276 * Separate the arguments in "str" and return a list of pointers in the
277 * growarray "gap".
278 */
279 static int
280get_arglist(garray_T *gap, char_u *str, int escaped)
281{
Bram Moolenaar04935fb2022-01-08 16:19:22 +0000282 ga_init2(gap, sizeof(char_u *), 20);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200283 while (*str != NUL)
284 {
285 if (ga_grow(gap, 1) == FAIL)
286 {
287 ga_clear(gap);
288 return FAIL;
289 }
290 ((char_u **)gap->ga_data)[gap->ga_len++] = str;
291
292 // If str is escaped, don't handle backslashes or spaces
293 if (!escaped)
294 return OK;
295
296 // Isolate one argument, change it in-place, put a NUL after it.
297 str = do_one_arg(str);
298 }
299 return OK;
300}
301
Martin Tournoijba43e762022-10-13 22:12:15 +0100302#if defined(FEAT_QUICKFIX) || defined(FEAT_SYN_HL) || defined(FEAT_SPELL) || defined(PROTO)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200303/*
304 * Parse a list of arguments (file names), expand them and return in
305 * "fnames[fcountp]". When "wig" is TRUE, removes files matching 'wildignore'.
306 * Return FAIL or OK.
307 */
308 int
309get_arglist_exp(
310 char_u *str,
311 int *fcountp,
312 char_u ***fnamesp,
313 int wig)
314{
315 garray_T ga;
316 int i;
317
318 if (get_arglist(&ga, str, TRUE) == FAIL)
319 return FAIL;
320 if (wig == TRUE)
321 i = expand_wildcards(ga.ga_len, (char_u **)ga.ga_data,
Bram Moolenaarf8c6a172021-01-30 18:09:06 +0100322 fcountp, fnamesp, EW_FILE|EW_NOTFOUND|EW_NOTWILD);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200323 else
324 i = gen_expand_wildcards(ga.ga_len, (char_u **)ga.ga_data,
Bram Moolenaarf8c6a172021-01-30 18:09:06 +0100325 fcountp, fnamesp, EW_FILE|EW_NOTFOUND|EW_NOTWILD);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200326
327 ga_clear(&ga);
328 return i;
329}
330#endif
331
332/*
333 * Check the validity of the arg_idx for each other window.
334 */
335 static void
336alist_check_arg_idx(void)
337{
338 win_T *win;
339 tabpage_T *tp;
340
341 FOR_ALL_TAB_WINDOWS(tp, win)
342 if (win->w_alist == curwin->w_alist)
343 check_arg_idx(win);
344}
345
346/*
347 * Add files[count] to the arglist of the current window after arg "after".
348 * The file names in files[count] must have been allocated and are taken over.
349 * Files[] itself is not taken over.
350 */
351 static void
352alist_add_list(
353 int count,
354 char_u **files,
355 int after, // where to add: 0 = before first one
356 int will_edit) // will edit adding argument
357{
358 int i;
359 int old_argcount = ARGCOUNT;
360
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100361 if (check_arglist_locked() != FAIL
Bram Moolenaar35578162021-08-02 19:10:38 +0200362 && GA_GROW_OK(&ALIST(curwin)->al_ga, count))
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200363 {
364 if (after < 0)
365 after = 0;
366 if (after > ARGCOUNT)
367 after = ARGCOUNT;
368 if (after < ARGCOUNT)
369 mch_memmove(&(ARGLIST[after + count]), &(ARGLIST[after]),
370 (ARGCOUNT - after) * sizeof(aentry_T));
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100371 arglist_locked = TRUE;
Christian Brabandt0a6e57b2024-08-15 22:15:28 +0200372 curwin->w_locked = TRUE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200373 for (i = 0; i < count; ++i)
374 {
375 int flags = BLN_LISTED | (will_edit ? BLN_CURBUF : 0);
376
377 ARGLIST[after + i].ae_fname = files[i];
378 ARGLIST[after + i].ae_fnum = buflist_add(files[i], flags);
379 }
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100380 arglist_locked = FALSE;
Christian Brabandt0a6e57b2024-08-15 22:15:28 +0200381 curwin->w_locked = FALSE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200382 ALIST(curwin)->al_ga.ga_len += count;
383 if (old_argcount > 0 && curwin->w_arg_idx >= after)
384 curwin->w_arg_idx += count;
385 return;
386 }
387
388 for (i = 0; i < count; ++i)
389 vim_free(files[i]);
390}
391
392/*
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100393 * Delete the file names in 'alist_ga' from the argument list.
394 */
395 static void
396arglist_del_files(garray_T *alist_ga)
397{
398 regmatch_T regmatch;
399 int didone;
400 int i;
401 char_u *p;
402 int match;
403
404 // Delete the items: use each item as a regexp and find a match in the
405 // argument list.
406 regmatch.rm_ic = p_fic; // ignore case when 'fileignorecase' is set
407 for (i = 0; i < alist_ga->ga_len && !got_int; ++i)
408 {
409 p = ((char_u **)alist_ga->ga_data)[i];
410 p = file_pat_to_reg_pat(p, NULL, NULL, FALSE);
411 if (p == NULL)
412 break;
413 regmatch.regprog = vim_regcomp(p, magic_isset() ? RE_MAGIC : 0);
414 if (regmatch.regprog == NULL)
415 {
416 vim_free(p);
417 break;
418 }
419
420 didone = FALSE;
421 for (match = 0; match < ARGCOUNT; ++match)
Bram Moolenaar24fe33a2022-11-24 00:09:02 +0000422 if (vim_regexec(&regmatch, alist_name(&ARGLIST[match]), (colnr_T)0))
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100423 {
424 didone = TRUE;
425 vim_free(ARGLIST[match].ae_fname);
426 mch_memmove(ARGLIST + match, ARGLIST + match + 1,
427 (ARGCOUNT - match - 1) * sizeof(aentry_T));
428 --ALIST(curwin)->al_ga.ga_len;
429 if (curwin->w_arg_idx > match)
430 --curwin->w_arg_idx;
431 --match;
432 }
433
434 vim_regfree(regmatch.regprog);
435 vim_free(p);
436 if (!didone)
437 semsg(_(e_no_match_str_2), ((char_u **)alist_ga->ga_data)[i]);
438 }
439 ga_clear(alist_ga);
440}
441
442/*
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200443 * "what" == AL_SET: Redefine the argument list to 'str'.
444 * "what" == AL_ADD: add files in 'str' to the argument list after "after".
445 * "what" == AL_DEL: remove files in 'str' from the argument list.
446 *
447 * Return FAIL for failure, OK otherwise.
448 */
449 static int
450do_arglist(
451 char_u *str,
452 int what,
453 int after UNUSED, // 0 means before first one
454 int will_edit) // will edit added argument
455{
456 garray_T new_ga;
457 int exp_count;
458 char_u **exp_files;
459 int i;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200460 int arg_escaped = TRUE;
461
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100462 if (check_arglist_locked() == FAIL)
463 return FAIL;
464
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200465 // Set default argument for ":argadd" command.
466 if (what == AL_ADD && *str == NUL)
467 {
468 if (curbuf->b_ffname == NULL)
469 return FAIL;
470 str = curbuf->b_fname;
471 arg_escaped = FALSE;
472 }
473
474 // Collect all file name arguments in "new_ga".
475 if (get_arglist(&new_ga, str, arg_escaped) == FAIL)
476 return FAIL;
477
478 if (what == AL_DEL)
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100479 arglist_del_files(&new_ga);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200480 else
481 {
482 i = expand_wildcards(new_ga.ga_len, (char_u **)new_ga.ga_data,
483 &exp_count, &exp_files, EW_DIR|EW_FILE|EW_ADDSLASH|EW_NOTFOUND);
484 ga_clear(&new_ga);
485 if (i == FAIL || exp_count == 0)
486 {
Bram Moolenaar460ae5d2022-01-01 14:19:49 +0000487 emsg(_(e_no_match));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200488 return FAIL;
489 }
490
491 if (what == AL_ADD)
492 {
493 alist_add_list(exp_count, exp_files, after, will_edit);
494 vim_free(exp_files);
495 }
496 else // what == AL_SET
497 alist_set(ALIST(curwin), exp_count, exp_files, will_edit, NULL, 0);
498 }
499
500 alist_check_arg_idx();
501
502 return OK;
503}
504
505/*
506 * Redefine the argument list.
507 */
508 void
509set_arglist(char_u *str)
510{
Rocco Maof96dc8d2024-01-23 21:27:19 +0100511 do_arglist(str, AL_SET, 0, TRUE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200512}
513
514/*
515 * Return TRUE if window "win" is editing the file at the current argument
516 * index.
517 */
518 int
519editing_arg_idx(win_T *win)
520{
521 return !(win->w_arg_idx >= WARGCOUNT(win)
522 || (win->w_buffer->b_fnum
523 != WARGLIST(win)[win->w_arg_idx].ae_fnum
524 && (win->w_buffer->b_ffname == NULL
525 || !(fullpathcmp(
526 alist_name(&WARGLIST(win)[win->w_arg_idx]),
527 win->w_buffer->b_ffname, TRUE, TRUE) & FPC_SAME))));
528}
529
530/*
531 * Check if window "win" is editing the w_arg_idx file in its argument list.
532 */
533 void
534check_arg_idx(win_T *win)
535{
536 if (WARGCOUNT(win) > 1 && !editing_arg_idx(win))
537 {
538 // We are not editing the current entry in the argument list.
539 // Set "arg_had_last" if we are editing the last one.
540 win->w_arg_idx_invalid = TRUE;
541 if (win->w_arg_idx != WARGCOUNT(win) - 1
542 && arg_had_last == FALSE
543 && ALIST(win) == &global_alist
544 && GARGCOUNT > 0
545 && win->w_arg_idx < GARGCOUNT
546 && (win->w_buffer->b_fnum == GARGLIST[GARGCOUNT - 1].ae_fnum
547 || (win->w_buffer->b_ffname != NULL
548 && (fullpathcmp(alist_name(&GARGLIST[GARGCOUNT - 1]),
549 win->w_buffer->b_ffname, TRUE, TRUE) & FPC_SAME))))
550 arg_had_last = TRUE;
551 }
552 else
553 {
554 // We are editing the current entry in the argument list.
555 // Set "arg_had_last" if it's also the last one
556 win->w_arg_idx_invalid = FALSE;
557 if (win->w_arg_idx == WARGCOUNT(win) - 1
558 && win->w_alist == &global_alist)
559 arg_had_last = TRUE;
560 }
561}
562
563/*
zeertzjqa304e492025-06-10 20:31:44 +0200564 * ":args", ":arglocal" and ":argglobal".
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200565 */
566 void
567ex_args(exarg_T *eap)
568{
569 int i;
570
571 if (eap->cmdidx != CMD_args)
572 {
Bram Moolenaar6bcb8772021-02-03 21:23:29 +0100573 if (check_arglist_locked() == FAIL)
574 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200575 alist_unlink(ALIST(curwin));
576 if (eap->cmdidx == CMD_argglobal)
577 ALIST(curwin) = &global_alist;
578 else // eap->cmdidx == CMD_arglocal
579 alist_new();
580 }
581
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100582 // ":args file ..": define new argument list, handle like ":next"
583 // Also for ":argslocal file .." and ":argsglobal file ..".
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200584 if (*eap->arg != NUL)
585 {
Bram Moolenaar6bcb8772021-02-03 21:23:29 +0100586 if (check_arglist_locked() == FAIL)
587 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200588 ex_next(eap);
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100589 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200590 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100591
592 // ":args": list arguments.
593 if (eap->cmdidx == CMD_args)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200594 {
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100595 char_u **items;
596
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100597 if (ARGCOUNT <= 0)
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100598 return; // empty argument list
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200599
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100600 items = ALLOC_MULT(char_u *, ARGCOUNT);
601 if (items == NULL)
602 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200603
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100604 // Overwrite the command, for a short list there is no scrolling
605 // required and no wait_return().
606 gotocmdline(TRUE);
607
608 for (i = 0; i < ARGCOUNT; ++i)
609 items[i] = alist_name(&ARGLIST[i]);
610 list_in_columns(items, ARGCOUNT, curwin->w_arg_idx);
611 vim_free(items);
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100612
613 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200614 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100615
616 // ":argslocal": make a local copy of the global argument list.
617 if (eap->cmdidx == CMD_arglocal)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200618 {
619 garray_T *gap = &curwin->w_alist->al_ga;
620
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100621 if (GA_GROW_FAILS(gap, GARGCOUNT))
622 return;
623
624 for (i = 0; i < GARGCOUNT; ++i)
625 if (GARGLIST[i].ae_fname != NULL)
626 {
627 AARGLIST(curwin->w_alist)[gap->ga_len].ae_fname =
628 vim_strsave(GARGLIST[i].ae_fname);
629 AARGLIST(curwin->w_alist)[gap->ga_len].ae_fnum =
630 GARGLIST[i].ae_fnum;
631 ++gap->ga_len;
632 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200633 }
634}
635
636/*
637 * ":previous", ":sprevious", ":Next" and ":sNext".
638 */
639 void
640ex_previous(exarg_T *eap)
641{
642 // If past the last one already, go to the last one.
643 if (curwin->w_arg_idx - (int)eap->line2 >= ARGCOUNT)
644 do_argfile(eap, ARGCOUNT - 1);
645 else
646 do_argfile(eap, curwin->w_arg_idx - (int)eap->line2);
647}
648
649/*
650 * ":rewind", ":first", ":sfirst" and ":srewind".
651 */
652 void
653ex_rewind(exarg_T *eap)
654{
655 do_argfile(eap, 0);
656}
657
658/*
659 * ":last" and ":slast".
660 */
661 void
662ex_last(exarg_T *eap)
663{
664 do_argfile(eap, ARGCOUNT - 1);
665}
666
667/*
668 * ":argument" and ":sargument".
669 */
670 void
671ex_argument(exarg_T *eap)
672{
673 int i;
674
675 if (eap->addr_count > 0)
676 i = eap->line2 - 1;
677 else
678 i = curwin->w_arg_idx;
679 do_argfile(eap, i);
680}
681
682/*
683 * Edit file "argn" of the argument lists.
684 */
685 void
686do_argfile(exarg_T *eap, int argn)
687{
688 int other;
689 char_u *p;
690 int old_arg_idx = curwin->w_arg_idx;
Colin Kennedy21570352024-03-03 16:16:47 +0100691 int is_split_cmd = *eap->cmd == 's';
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200692
Bram Moolenaar3c01c4a2020-02-01 23:04:24 +0100693 if (ERROR_IF_ANY_POPUP_WINDOW)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200694 return;
695 if (argn < 0 || argn >= ARGCOUNT)
696 {
697 if (ARGCOUNT <= 1)
Bram Moolenaar1a992222021-12-31 17:25:48 +0000698 emsg(_(e_there_is_only_one_file_to_edit));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200699 else if (argn < 0)
Bram Moolenaar1a992222021-12-31 17:25:48 +0000700 emsg(_(e_cannot_go_before_first_file));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200701 else
Bram Moolenaar1a992222021-12-31 17:25:48 +0000702 emsg(_(e_cannot_go_beyond_last_file));
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000703
704 return;
705 }
706
Colin Kennedy21570352024-03-03 16:16:47 +0100707 if (!is_split_cmd
708 && (&ARGLIST[argn])->ae_fnum != curbuf->b_fnum
709 && !check_can_set_curbuf_forceit(eap->forceit))
710 return;
711
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000712 setpcmark();
713#ifdef FEAT_GUI
714 need_mouse_correct = TRUE;
715#endif
716
717 // split window or create new tab page first
Colin Kennedy21570352024-03-03 16:16:47 +0100718 if (is_split_cmd || cmdmod.cmod_tab != 0)
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000719 {
720 if (win_split(0, 0) == FAIL)
721 return;
722 RESET_BINDING(curwin);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200723 }
724 else
725 {
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000726 // if 'hidden' set, only check for changed file when re-editing
727 // the same buffer
728 other = TRUE;
729 if (buf_hide(curbuf))
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200730 {
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000731 p = fix_fname(alist_name(&ARGLIST[argn]));
732 other = otherfile(p);
733 vim_free(p);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200734 }
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000735 if ((!buf_hide(curbuf) || !other)
736 && check_changed(curbuf, CCGD_AW
737 | (other ? 0 : CCGD_MULTWIN)
738 | (eap->forceit ? CCGD_FORCEIT : 0)
739 | CCGD_EXCMD))
740 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200741 }
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000742
743 curwin->w_arg_idx = argn;
744 if (argn == ARGCOUNT - 1 && curwin->w_alist == &global_alist)
745 arg_had_last = TRUE;
746
747 // Edit the file; always use the last known line number.
748 // When it fails (e.g. Abort for already edited file) restore the
749 // argument index.
750 if (do_ecmd(0, alist_name(&ARGLIST[curwin->w_arg_idx]), NULL,
751 eap, ECMD_LAST,
752 (buf_hide(curwin->w_buffer) ? ECMD_HIDE : 0)
753 + (eap->forceit ? ECMD_FORCEIT : 0), curwin) == FAIL)
754 curwin->w_arg_idx = old_arg_idx;
755 // like Vi: set the mark where the cursor is in the file.
756 else if (eap->cmdidx != CMD_argdo)
757 setmark('\'');
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200758}
759
760/*
761 * ":next", and commands that behave like it.
762 */
763 void
764ex_next(exarg_T *eap)
765{
766 int i;
767
768 // check for changed buffer now, if this fails the argument list is not
769 // redefined.
770 if ( buf_hide(curbuf)
771 || eap->cmdidx == CMD_snext
772 || !check_changed(curbuf, CCGD_AW
773 | (eap->forceit ? CCGD_FORCEIT : 0)
774 | CCGD_EXCMD))
775 {
776 if (*eap->arg != NUL) // redefine file list
777 {
778 if (do_arglist(eap->arg, AL_SET, 0, TRUE) == FAIL)
779 return;
780 i = 0;
781 }
782 else
783 i = curwin->w_arg_idx + (int)eap->line2;
784 do_argfile(eap, i);
785 }
786}
787
788/*
Nir Lichtman73a02422021-12-24 20:28:03 +0000789 * ":argdedupe"
790 */
791 void
792ex_argdedupe(exarg_T *eap UNUSED)
793{
794 int i;
795 int j;
796
797 for (i = 0; i < ARGCOUNT; ++i)
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000798 {
799 // Expand each argument to a full path to catch different paths leading
800 // to the same file.
801 char_u *firstFullname = FullName_save(ARGLIST[i].ae_fname, FALSE);
802 if (firstFullname == NULL)
803 return; // out of memory
804
Nir Lichtman73a02422021-12-24 20:28:03 +0000805 for (j = i + 1; j < ARGCOUNT; ++j)
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000806 {
807 char_u *secondFullname = FullName_save(ARGLIST[j].ae_fname, FALSE);
808 if (secondFullname == NULL)
809 break; // out of memory
810 int areNamesDuplicate =
811 fnamecmp(firstFullname, secondFullname) == 0;
812 vim_free(secondFullname);
813
814 if (areNamesDuplicate)
Nir Lichtman73a02422021-12-24 20:28:03 +0000815 {
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000816 // remove one duplicate argument
Nir Lichtman73a02422021-12-24 20:28:03 +0000817 vim_free(ARGLIST[j].ae_fname);
818 mch_memmove(ARGLIST + j, ARGLIST + j + 1,
819 (ARGCOUNT - j - 1) * sizeof(aentry_T));
820 --ARGCOUNT;
821
822 if (curwin->w_arg_idx == j)
823 curwin->w_arg_idx = i;
824 else if (curwin->w_arg_idx > j)
825 --curwin->w_arg_idx;
826
827 --j;
828 }
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000829 }
830
831 vim_free(firstFullname);
832 }
Nir Lichtman73a02422021-12-24 20:28:03 +0000833}
834
835/*
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200836 * ":argedit"
837 */
838 void
839ex_argedit(exarg_T *eap)
840{
841 int i = eap->addr_count ? (int)eap->line2 : curwin->w_arg_idx + 1;
842 // Whether curbuf will be reused, curbuf->b_ffname will be set.
843 int curbuf_is_reusable = curbuf_reusable();
844
845 if (do_arglist(eap->arg, AL_ADD, i, TRUE) == FAIL)
846 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200847 maketitle();
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200848
849 if (curwin->w_arg_idx == 0
850 && (curbuf->b_ml.ml_flags & ML_EMPTY)
851 && (curbuf->b_ffname == NULL || curbuf_is_reusable))
852 i = 0;
853 // Edit the argument.
854 if (i < ARGCOUNT)
855 do_argfile(eap, i);
856}
857
858/*
859 * ":argadd"
860 */
861 void
862ex_argadd(exarg_T *eap)
863{
864 do_arglist(eap->arg, AL_ADD,
865 eap->addr_count > 0 ? (int)eap->line2 : curwin->w_arg_idx + 1,
866 FALSE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200867 maketitle();
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200868}
869
870/*
871 * ":argdelete"
872 */
873 void
874ex_argdelete(exarg_T *eap)
875{
876 int i;
877 int n;
878
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100879 if (check_arglist_locked() == FAIL)
880 return;
881
Bram Moolenaar7b221172020-08-17 19:34:10 +0200882 if (eap->addr_count > 0 || *eap->arg == NUL)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200883 {
Bram Moolenaar02c037a2020-08-30 19:26:45 +0200884 // ":argdel" works like ":.argdel"
Bram Moolenaar7b221172020-08-17 19:34:10 +0200885 if (eap->addr_count == 0)
886 {
887 if (curwin->w_arg_idx >= ARGCOUNT)
888 {
Bram Moolenaarf1474d82021-12-31 19:59:55 +0000889 emsg(_(e_no_argument_to_delete));
Bram Moolenaar7b221172020-08-17 19:34:10 +0200890 return;
891 }
892 eap->line1 = eap->line2 = curwin->w_arg_idx + 1;
893 }
894 else if (eap->line2 > ARGCOUNT)
895 // ":1,4argdel": Delete all arguments in the range.
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200896 eap->line2 = ARGCOUNT;
897 n = eap->line2 - eap->line1 + 1;
898 if (*eap->arg != NUL)
899 // Can't have both a range and an argument.
Bram Moolenaar436b5ad2021-12-31 22:49:24 +0000900 emsg(_(e_invalid_argument));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200901 else if (n <= 0)
902 {
903 // Don't give an error for ":%argdel" if the list is empty.
904 if (eap->line1 != 1 || eap->line2 != 0)
Bram Moolenaar108010a2021-06-27 22:03:33 +0200905 emsg(_(e_invalid_range));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200906 }
907 else
908 {
909 for (i = eap->line1; i <= eap->line2; ++i)
910 vim_free(ARGLIST[i - 1].ae_fname);
911 mch_memmove(ARGLIST + eap->line1 - 1, ARGLIST + eap->line2,
912 (size_t)((ARGCOUNT - eap->line2) * sizeof(aentry_T)));
913 ALIST(curwin)->al_ga.ga_len -= n;
914 if (curwin->w_arg_idx >= eap->line2)
915 curwin->w_arg_idx -= n;
916 else if (curwin->w_arg_idx > eap->line1)
917 curwin->w_arg_idx = eap->line1;
918 if (ARGCOUNT == 0)
919 curwin->w_arg_idx = 0;
920 else if (curwin->w_arg_idx >= ARGCOUNT)
921 curwin->w_arg_idx = ARGCOUNT - 1;
922 }
923 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200924 else
925 do_arglist(eap->arg, AL_DEL, 0, FALSE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200926 maketitle();
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200927}
928
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200929/*
930 * Function given to ExpandGeneric() to obtain the possible arguments of the
931 * argedit and argdelete commands.
932 */
933 char_u *
934get_arglist_name(expand_T *xp UNUSED, int idx)
935{
936 if (idx >= ARGCOUNT)
937 return NULL;
938
939 return alist_name(&ARGLIST[idx]);
940}
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200941
942/*
943 * Get the file name for an argument list entry.
944 */
945 char_u *
946alist_name(aentry_T *aep)
947{
948 buf_T *bp;
949
950 // Use the name from the associated buffer if it exists.
951 bp = buflist_findnr(aep->ae_fnum);
952 if (bp == NULL || bp->b_fname == NULL)
953 return aep->ae_fname;
954 return bp->b_fname;
955}
956
957/*
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100958 * State used by the :all command to open all the files in the argument list in
959 * separate windows.
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200960 */
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100961typedef struct {
962 alist_T *alist; // argument list to be used
963 int had_tab;
964 int keep_tabs;
965 int forceit;
966
967 int use_firstwin; // use first window for arglist
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200968 char_u *opened; // Array of weight for which args are open:
969 // 0: not opened
970 // 1: opened in other tab
971 // 2: opened in curtab
972 // 3: opened in curtab and curwin
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200973 int opened_len; // length of opened[]
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100974 win_T *new_curwin;
975 tabpage_T *new_curtab;
976} arg_all_state_T;
977
978/*
979 * Close all the windows containing files which are not in the argument list.
980 * Used by the ":all" command.
981 */
982 static void
983arg_all_close_unused_windows(arg_all_state_T *aall)
984{
985 win_T *wp;
986 win_T *wpnext;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200987 tabpage_T *tpnext;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100988 buf_T *buf;
989 int i;
990 win_T *old_curwin;
991 tabpage_T *old_curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200992
993 old_curwin = curwin;
994 old_curtab = curtab;
995
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100996 if (aall->had_tab > 0)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200997 goto_tabpage_tp(first_tabpage, TRUE, TRUE);
Christian Brabandtdf12e392023-12-16 13:55:32 +0100998
999 // moving tabpages around in an autocommand may cause an endless loop
1000 tabpage_move_disallowed++;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001001 for (;;)
1002 {
1003 tpnext = curtab->tp_next;
1004 for (wp = firstwin; wp != NULL; wp = wpnext)
1005 {
1006 wpnext = wp->w_next;
1007 buf = wp->w_buffer;
1008 if (buf->b_ffname == NULL
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001009 || (!aall->keep_tabs && (buf->b_nwindows > 1
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001010 || wp->w_width != Columns)))
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001011 i = aall->opened_len;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001012 else
1013 {
1014 // check if the buffer in this window is in the arglist
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001015 for (i = 0; i < aall->opened_len; ++i)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001016 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001017 if (i < aall->alist->al_ga.ga_len
1018 && (AARGLIST(aall->alist)[i].ae_fnum == buf->b_fnum
Bram Moolenaar747f1102022-09-18 13:06:41 +01001019 || fullpathcmp(alist_name(
1020 &AARGLIST(aall->alist)[i]),
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001021 buf->b_ffname, TRUE, TRUE) & FPC_SAME))
1022 {
1023 int weight = 1;
1024
1025 if (old_curtab == curtab)
1026 {
1027 ++weight;
1028 if (old_curwin == wp)
1029 ++weight;
1030 }
1031
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001032 if (weight > (int)aall->opened[i])
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001033 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001034 aall->opened[i] = (char_u)weight;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001035 if (i == 0)
1036 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001037 if (aall->new_curwin != NULL)
Bram Moolenaar747f1102022-09-18 13:06:41 +01001038 aall->new_curwin->w_arg_idx =
1039 aall->opened_len;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001040 aall->new_curwin = wp;
1041 aall->new_curtab = curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001042 }
1043 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001044 else if (aall->keep_tabs)
1045 i = aall->opened_len;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001046
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001047 if (wp->w_alist != aall->alist)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001048 {
1049 // Use the current argument list for all windows
1050 // containing a file from it.
1051 alist_unlink(wp->w_alist);
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001052 wp->w_alist = aall->alist;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001053 ++wp->w_alist->al_refcount;
1054 }
1055 break;
1056 }
1057 }
1058 }
1059 wp->w_arg_idx = i;
1060
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001061 if (i == aall->opened_len && !aall->keep_tabs)// close this window
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001062 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001063 if (buf_hide(buf) || aall->forceit || buf->b_nwindows > 1
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001064 || !bufIsChanged(buf))
1065 {
1066 // If the buffer was changed, and we would like to hide it,
1067 // try autowriting.
1068 if (!buf_hide(buf) && buf->b_nwindows <= 1
1069 && bufIsChanged(buf))
1070 {
1071 bufref_T bufref;
1072
1073 set_bufref(&bufref, buf);
1074
1075 (void)autowrite(buf, FALSE);
1076
1077 // check if autocommands removed the window
1078 if (!win_valid(wp) || !bufref_valid(&bufref))
1079 {
1080 wpnext = firstwin; // start all over...
1081 continue;
1082 }
1083 }
1084 // don't close last window
1085 if (ONE_WINDOW
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001086 && (first_tabpage->tp_next == NULL
1087 || !aall->had_tab))
1088 aall->use_firstwin = TRUE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001089 else
1090 {
1091 win_close(wp, !buf_hide(buf) && !bufIsChanged(buf));
1092
1093 // check if autocommands removed the next window
1094 if (!win_valid(wpnext))
1095 wpnext = firstwin; // start all over...
1096 }
1097 }
1098 }
1099 }
1100
1101 // Without the ":tab" modifier only do the current tab page.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001102 if (aall->had_tab == 0 || tpnext == NULL)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001103 break;
1104
1105 // check if autocommands removed the next tab page
1106 if (!valid_tabpage(tpnext))
1107 tpnext = first_tabpage; // start all over...
1108
1109 goto_tabpage_tp(tpnext, TRUE, TRUE);
1110 }
Christian Brabandtdf12e392023-12-16 13:55:32 +01001111 tabpage_move_disallowed--;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001112}
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001113
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001114/*
1115 * Open upto "count" windows for the files in the argument list 'aall->alist'.
1116 */
1117 static void
1118arg_all_open_windows(arg_all_state_T *aall, int count)
1119{
1120 win_T *wp;
1121 int tab_drop_empty_window = FALSE;
1122 int i;
1123 int split_ret = OK;
1124 int p_ea_save;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001125
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001126 // ":tab drop file" should re-use an empty window to avoid "--remote-tab"
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001127 // leaving an empty tab page when executed locally.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001128 if (aall->keep_tabs && BUFEMPTY() && curbuf->b_nwindows == 1
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001129 && curbuf->b_ffname == NULL && !curbuf->b_changed)
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001130 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001131 aall->use_firstwin = TRUE;
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001132 tab_drop_empty_window = TRUE;
1133 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001134
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001135 for (i = 0; i < count && !got_int; ++i)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001136 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001137 if (aall->alist == &global_alist && i == global_alist.al_ga.ga_len - 1)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001138 arg_had_last = TRUE;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001139 if (aall->opened[i] > 0)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001140 {
1141 // Move the already present window to below the current window
1142 if (curwin->w_arg_idx != i)
1143 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001144 FOR_ALL_WINDOWS(wp)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001145 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001146 if (wp->w_arg_idx == i)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001147 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001148 if (aall->keep_tabs)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001149 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001150 aall->new_curwin = wp;
1151 aall->new_curtab = curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001152 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001153 else if (wp->w_frame->fr_parent
1154 != curwin->w_frame->fr_parent)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001155 {
Bram Moolenaarf1474d82021-12-31 19:59:55 +00001156 emsg(_(e_window_layout_changed_unexpectedly));
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001157 i = count;
1158 break;
1159 }
1160 else
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001161 win_move_after(wp, curwin);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001162 break;
1163 }
1164 }
1165 }
1166 }
1167 else if (split_ret == OK)
1168 {
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001169 // trigger events for tab drop
1170 if (tab_drop_empty_window && i == count - 1)
1171 --autocmd_no_enter;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001172 if (!aall->use_firstwin) // split current window
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001173 {
1174 p_ea_save = p_ea;
1175 p_ea = TRUE; // use space from all windows
1176 split_ret = win_split(0, WSP_ROOM | WSP_BELOW);
1177 p_ea = p_ea_save;
1178 if (split_ret == FAIL)
1179 continue;
1180 }
1181 else // first window: do autocmd for leaving this buffer
1182 --autocmd_no_leave;
1183
1184 // edit file "i"
1185 curwin->w_arg_idx = i;
1186 if (i == 0)
1187 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001188 aall->new_curwin = curwin;
1189 aall->new_curtab = curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001190 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001191 (void)do_ecmd(0, alist_name(&AARGLIST(aall->alist)[i]), NULL, NULL,
1192 ECMD_ONE,
1193 ((buf_hide(curwin->w_buffer)
1194 || bufIsChanged(curwin->w_buffer)) ? ECMD_HIDE : 0)
1195 + ECMD_OLDBUF, curwin);
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001196 if (tab_drop_empty_window && i == count - 1)
1197 ++autocmd_no_enter;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001198 if (aall->use_firstwin)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001199 ++autocmd_no_leave;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001200 aall->use_firstwin = FALSE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001201 }
1202 ui_breakcheck();
1203
1204 // When ":tab" was used open a new tab for a new window repeatedly.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001205 if (aall->had_tab > 0 && tabpage_index(NULL) <= p_tpm)
Bram Moolenaare1004402020-10-24 20:49:43 +02001206 cmdmod.cmod_tab = 9999;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001207 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001208}
1209
1210/*
1211 * do_arg_all(): Open up to "count" windows, one for each argument.
1212 */
1213 static void
1214do_arg_all(
1215 int count,
1216 int forceit, // hide buffers in current windows
1217 int keep_tabs) // keep current tabs, for ":tab drop file"
1218{
1219 arg_all_state_T aall;
1220 win_T *last_curwin;
1221 tabpage_T *last_curtab;
1222 int prev_arglist_locked = arglist_locked;
1223
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001224 if (cmdwin_type != 0)
1225 {
1226 emsg(_(e_invalid_in_cmdline_window));
1227 return;
1228 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001229 if (ARGCOUNT <= 0)
1230 {
1231 // Don't give an error message. We don't want it when the ":all"
1232 // command is in the .vimrc.
1233 return;
1234 }
1235 setpcmark();
1236
1237 aall.use_firstwin = FALSE;
1238 aall.had_tab = cmdmod.cmod_tab;
1239 aall.new_curwin = NULL;
1240 aall.new_curtab = NULL;
1241 aall.forceit = forceit;
1242 aall.keep_tabs = keep_tabs;
1243 aall.opened_len = ARGCOUNT;
1244 aall.opened = alloc_clear(aall.opened_len);
1245 if (aall.opened == NULL)
1246 return;
1247
1248 // Autocommands may do anything to the argument list. Make sure it's not
1249 // freed while we are working here by "locking" it. We still have to
1250 // watch out for its size being changed.
1251 aall.alist = curwin->w_alist;
1252 ++aall.alist->al_refcount;
1253 arglist_locked = TRUE;
1254
1255#ifdef FEAT_GUI
1256 need_mouse_correct = TRUE;
1257#endif
1258
Bram Moolenaar8281a162023-04-20 18:07:57 +01001259 tabpage_T *new_lu_tp = curtab;
1260
Christian Brabandtc9a1e252025-01-11 15:25:00 +01001261 // Stop Visual mode, the cursor and "VIsual" may very well be invalid after
1262 // switching to another buffer.
1263 reset_VIsual_and_resel();
1264
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001265 // Try closing all windows that are not in the argument list.
1266 // Also close windows that are not full width;
1267 // When 'hidden' or "forceit" set the buffer becomes hidden.
1268 // Windows that have a changed buffer and can't be hidden won't be closed.
1269 // When the ":tab" modifier was used do this for all tab pages.
1270 arg_all_close_unused_windows(&aall);
1271
1272 // Open a window for files in the argument list that don't have one.
1273 // ARGCOUNT may change while doing this, because of autocommands.
1274 if (count > aall.opened_len || count <= 0)
1275 count = aall.opened_len;
1276
1277 // Don't execute Win/Buf Enter/Leave autocommands here.
1278 ++autocmd_no_enter;
1279 ++autocmd_no_leave;
1280 last_curwin = curwin;
1281 last_curtab = curtab;
1282 win_enter(lastwin, FALSE);
1283
1284 /*
1285 * Open upto "count" windows.
1286 */
1287 arg_all_open_windows(&aall, count);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001288
1289 // Remove the "lock" on the argument list.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001290 alist_unlink(aall.alist);
Bram Moolenaar6f983712021-12-24 18:11:27 +00001291 arglist_locked = prev_arglist_locked;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001292
1293 --autocmd_no_enter;
1294
1295 // restore last referenced tabpage's curwin
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001296 if (last_curtab != aall.new_curtab)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001297 {
1298 if (valid_tabpage(last_curtab))
1299 goto_tabpage_tp(last_curtab, TRUE, TRUE);
1300 if (win_valid(last_curwin))
1301 win_enter(last_curwin, FALSE);
1302 }
1303 // to window with first arg
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001304 if (valid_tabpage(aall.new_curtab))
1305 goto_tabpage_tp(aall.new_curtab, TRUE, TRUE);
glepnir2975a542024-02-09 19:30:26 +01001306
1307 // Now set the last used tabpage to where we started.
1308 if (valid_tabpage(new_lu_tp))
1309 lastused_tabpage = new_lu_tp;
1310
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001311 if (win_valid(aall.new_curwin))
1312 win_enter(aall.new_curwin, FALSE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001313
1314 --autocmd_no_leave;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001315 vim_free(aall.opened);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001316}
1317
1318/*
1319 * ":all" and ":sall".
1320 * Also used for ":tab drop file ..." after setting the argument list.
1321 */
1322 void
1323ex_all(exarg_T *eap)
1324{
1325 if (eap->addr_count == 0)
1326 eap->line2 = 9999;
1327 do_arg_all((int)eap->line2, eap->forceit, eap->cmdidx == CMD_drop);
1328}
1329
1330/*
1331 * Concatenate all files in the argument list, separated by spaces, and return
1332 * it in one allocated string.
1333 * Spaces and backslashes in the file names are escaped with a backslash.
1334 * Returns NULL when out of memory.
1335 */
1336 char_u *
1337arg_all(void)
1338{
1339 int len;
1340 int idx;
1341 char_u *retval = NULL;
1342 char_u *p;
1343
1344 // Do this loop two times:
1345 // first time: compute the total length
1346 // second time: concatenate the names
1347 for (;;)
1348 {
1349 len = 0;
1350 for (idx = 0; idx < ARGCOUNT; ++idx)
1351 {
1352 p = alist_name(&ARGLIST[idx]);
zeertzjq101d57b2022-07-31 18:34:32 +01001353 if (p == NULL)
1354 continue;
1355 if (len > 0)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001356 {
zeertzjq101d57b2022-07-31 18:34:32 +01001357 // insert a space in between names
1358 if (retval != NULL)
1359 retval[len] = ' ';
1360 ++len;
1361 }
1362 for ( ; *p != NUL; ++p)
1363 {
1364 if (*p == ' '
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001365#ifndef BACKSLASH_IN_FILENAME
zeertzjq101d57b2022-07-31 18:34:32 +01001366 || *p == '\\'
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001367#endif
zeertzjq101d57b2022-07-31 18:34:32 +01001368 || *p == '`')
1369 {
1370 // insert a backslash
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001371 if (retval != NULL)
zeertzjq101d57b2022-07-31 18:34:32 +01001372 retval[len] = '\\';
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001373 ++len;
1374 }
zeertzjq101d57b2022-07-31 18:34:32 +01001375 if (retval != NULL)
1376 retval[len] = *p;
1377 ++len;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001378 }
1379 }
1380
1381 // second time: break here
1382 if (retval != NULL)
1383 {
1384 retval[len] = NUL;
1385 break;
1386 }
1387
1388 // allocate memory
1389 retval = alloc(len + 1);
1390 if (retval == NULL)
1391 break;
1392 }
1393
1394 return retval;
1395}
1396
1397#if defined(FEAT_EVAL) || defined(PROTO)
1398/*
1399 * "argc([window id])" function
1400 */
1401 void
1402f_argc(typval_T *argvars, typval_T *rettv)
1403{
1404 win_T *wp;
1405
Yegappan Lakshmanan4490ec42021-07-27 22:00:44 +02001406 if (in_vim9script() && check_for_opt_number_arg(argvars, 0) == FAIL)
1407 return;
1408
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001409 if (argvars[0].v_type == VAR_UNKNOWN)
1410 // use the current window
1411 rettv->vval.v_number = ARGCOUNT;
1412 else if (argvars[0].v_type == VAR_NUMBER
1413 && tv_get_number(&argvars[0]) == -1)
1414 // use the global argument list
1415 rettv->vval.v_number = GARGCOUNT;
1416 else
1417 {
1418 // use the argument list of the specified window
1419 wp = find_win_by_nr_or_id(&argvars[0]);
1420 if (wp != NULL)
1421 rettv->vval.v_number = WARGCOUNT(wp);
1422 else
1423 rettv->vval.v_number = -1;
1424 }
1425}
1426
1427/*
1428 * "argidx()" function
1429 */
1430 void
1431f_argidx(typval_T *argvars UNUSED, typval_T *rettv)
1432{
1433 rettv->vval.v_number = curwin->w_arg_idx;
1434}
1435
1436/*
1437 * "arglistid()" function
1438 */
1439 void
1440f_arglistid(typval_T *argvars, typval_T *rettv)
1441{
1442 win_T *wp;
1443
Yegappan Lakshmanan4490ec42021-07-27 22:00:44 +02001444 if (in_vim9script()
1445 && (check_for_opt_number_arg(argvars, 0) == FAIL
1446 || (argvars[0].v_type != VAR_UNKNOWN
1447 && check_for_opt_number_arg(argvars, 1) == FAIL)))
1448 return;
1449
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001450 rettv->vval.v_number = -1;
1451 wp = find_tabwin(&argvars[0], &argvars[1], NULL);
1452 if (wp != NULL)
1453 rettv->vval.v_number = wp->w_alist->id;
1454}
1455
1456/*
1457 * Get the argument list for a given window
1458 */
1459 static void
1460get_arglist_as_rettv(aentry_T *arglist, int argcount, typval_T *rettv)
1461{
1462 int idx;
1463
1464 if (rettv_list_alloc(rettv) == OK && arglist != NULL)
1465 for (idx = 0; idx < argcount; ++idx)
1466 list_append_string(rettv->vval.v_list,
1467 alist_name(&arglist[idx]), -1);
1468}
1469
1470/*
1471 * "argv(nr)" function
1472 */
1473 void
1474f_argv(typval_T *argvars, typval_T *rettv)
1475{
1476 int idx;
1477 aentry_T *arglist = NULL;
1478 int argcount = -1;
1479
Yegappan Lakshmanan4490ec42021-07-27 22:00:44 +02001480 if (in_vim9script()
1481 && (check_for_opt_number_arg(argvars, 0) == FAIL
1482 || (argvars[0].v_type != VAR_UNKNOWN
1483 && check_for_opt_number_arg(argvars, 1) == FAIL)))
1484 return;
1485
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001486 if (argvars[0].v_type == VAR_UNKNOWN)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001487 {
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001488 get_arglist_as_rettv(ARGLIST, ARGCOUNT, rettv);
1489 return;
1490 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001491
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001492 if (argvars[1].v_type == VAR_UNKNOWN)
1493 {
1494 arglist = ARGLIST;
1495 argcount = ARGCOUNT;
1496 }
1497 else if (argvars[1].v_type == VAR_NUMBER
1498 && tv_get_number(&argvars[1]) == -1)
1499 {
1500 arglist = GARGLIST;
1501 argcount = GARGCOUNT;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001502 }
1503 else
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001504 {
1505 win_T *wp = find_win_by_nr_or_id(&argvars[1]);
1506
1507 if (wp != NULL)
1508 {
1509 // Use the argument list of the specified window
1510 arglist = WARGLIST(wp);
1511 argcount = WARGCOUNT(wp);
1512 }
1513 }
1514
1515 rettv->v_type = VAR_STRING;
1516 rettv->vval.v_string = NULL;
1517 idx = tv_get_number_chk(&argvars[0], NULL);
1518 if (arglist != NULL && idx >= 0 && idx < argcount)
1519 rettv->vval.v_string = vim_strsave(alist_name(&arglist[idx]));
1520 else if (idx == -1)
1521 get_arglist_as_rettv(arglist, argcount, rettv);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001522}
1523#endif