blob: 187e16e8354b14805f0217dfed4a3351097c6905 [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.
187 */
188 void
189alist_add(
190 alist_T *al,
191 char_u *fname,
192 int set_fnum) // 1: set buffer number; 2: re-use curbuf
193{
194 if (fname == NULL) // don't add NULL file names
195 return;
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100196 if (check_arglist_locked() == FAIL)
197 return;
198 arglist_locked = TRUE;
199
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200200#ifdef BACKSLASH_IN_FILENAME
201 slash_adjust(fname);
202#endif
203 AARGLIST(al)[al->al_ga.ga_len].ae_fname = fname;
204 if (set_fnum > 0)
205 AARGLIST(al)[al->al_ga.ga_len].ae_fnum =
206 buflist_add(fname, BLN_LISTED | (set_fnum == 2 ? BLN_CURBUF : 0));
207 ++al->al_ga.ga_len;
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100208
209 arglist_locked = FALSE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200210}
211
212#if defined(BACKSLASH_IN_FILENAME) || defined(PROTO)
213/*
214 * Adjust slashes in file names. Called after 'shellslash' was set.
215 */
216 void
217alist_slash_adjust(void)
218{
219 int i;
220 win_T *wp;
221 tabpage_T *tp;
222
223 for (i = 0; i < GARGCOUNT; ++i)
224 if (GARGLIST[i].ae_fname != NULL)
225 slash_adjust(GARGLIST[i].ae_fname);
226 FOR_ALL_TAB_WINDOWS(tp, wp)
227 if (wp->w_alist != &global_alist)
228 for (i = 0; i < WARGCOUNT(wp); ++i)
229 if (WARGLIST(wp)[i].ae_fname != NULL)
230 slash_adjust(WARGLIST(wp)[i].ae_fname);
231}
232#endif
233
234/*
235 * Isolate one argument, taking backticks.
236 * Changes the argument in-place, puts a NUL after it. Backticks remain.
237 * Return a pointer to the start of the next argument.
238 */
239 static char_u *
240do_one_arg(char_u *str)
241{
242 char_u *p;
243 int inbacktick;
244
245 inbacktick = FALSE;
246 for (p = str; *str; ++str)
247 {
248 // When the backslash is used for escaping the special meaning of a
249 // character we need to keep it until wildcard expansion.
250 if (rem_backslash(str))
251 {
252 *p++ = *str++;
253 *p++ = *str;
254 }
255 else
256 {
257 // An item ends at a space not in backticks
258 if (!inbacktick && vim_isspace(*str))
259 break;
260 if (*str == '`')
261 inbacktick ^= TRUE;
262 *p++ = *str;
263 }
264 }
265 str = skipwhite(str);
266 *p = NUL;
267
268 return str;
269}
270
271/*
272 * Separate the arguments in "str" and return a list of pointers in the
273 * growarray "gap".
274 */
275 static int
276get_arglist(garray_T *gap, char_u *str, int escaped)
277{
Bram Moolenaar04935fb2022-01-08 16:19:22 +0000278 ga_init2(gap, sizeof(char_u *), 20);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200279 while (*str != NUL)
280 {
281 if (ga_grow(gap, 1) == FAIL)
282 {
283 ga_clear(gap);
284 return FAIL;
285 }
286 ((char_u **)gap->ga_data)[gap->ga_len++] = str;
287
288 // If str is escaped, don't handle backslashes or spaces
289 if (!escaped)
290 return OK;
291
292 // Isolate one argument, change it in-place, put a NUL after it.
293 str = do_one_arg(str);
294 }
295 return OK;
296}
297
Martin Tournoijba43e762022-10-13 22:12:15 +0100298#if defined(FEAT_QUICKFIX) || defined(FEAT_SYN_HL) || defined(FEAT_SPELL) || defined(PROTO)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200299/*
300 * Parse a list of arguments (file names), expand them and return in
301 * "fnames[fcountp]". When "wig" is TRUE, removes files matching 'wildignore'.
302 * Return FAIL or OK.
303 */
304 int
305get_arglist_exp(
306 char_u *str,
307 int *fcountp,
308 char_u ***fnamesp,
309 int wig)
310{
311 garray_T ga;
312 int i;
313
314 if (get_arglist(&ga, str, TRUE) == FAIL)
315 return FAIL;
316 if (wig == TRUE)
317 i = expand_wildcards(ga.ga_len, (char_u **)ga.ga_data,
Bram Moolenaarf8c6a172021-01-30 18:09:06 +0100318 fcountp, fnamesp, EW_FILE|EW_NOTFOUND|EW_NOTWILD);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200319 else
320 i = gen_expand_wildcards(ga.ga_len, (char_u **)ga.ga_data,
Bram Moolenaarf8c6a172021-01-30 18:09:06 +0100321 fcountp, fnamesp, EW_FILE|EW_NOTFOUND|EW_NOTWILD);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200322
323 ga_clear(&ga);
324 return i;
325}
326#endif
327
328/*
329 * Check the validity of the arg_idx for each other window.
330 */
331 static void
332alist_check_arg_idx(void)
333{
334 win_T *win;
335 tabpage_T *tp;
336
337 FOR_ALL_TAB_WINDOWS(tp, win)
338 if (win->w_alist == curwin->w_alist)
339 check_arg_idx(win);
340}
341
342/*
343 * Add files[count] to the arglist of the current window after arg "after".
344 * The file names in files[count] must have been allocated and are taken over.
345 * Files[] itself is not taken over.
346 */
347 static void
348alist_add_list(
349 int count,
350 char_u **files,
351 int after, // where to add: 0 = before first one
352 int will_edit) // will edit adding argument
353{
354 int i;
355 int old_argcount = ARGCOUNT;
356
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100357 if (check_arglist_locked() != FAIL
Bram Moolenaar35578162021-08-02 19:10:38 +0200358 && GA_GROW_OK(&ALIST(curwin)->al_ga, count))
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200359 {
360 if (after < 0)
361 after = 0;
362 if (after > ARGCOUNT)
363 after = ARGCOUNT;
364 if (after < ARGCOUNT)
365 mch_memmove(&(ARGLIST[after + count]), &(ARGLIST[after]),
366 (ARGCOUNT - after) * sizeof(aentry_T));
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100367 arglist_locked = TRUE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200368 for (i = 0; i < count; ++i)
369 {
370 int flags = BLN_LISTED | (will_edit ? BLN_CURBUF : 0);
371
372 ARGLIST[after + i].ae_fname = files[i];
373 ARGLIST[after + i].ae_fnum = buflist_add(files[i], flags);
374 }
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100375 arglist_locked = FALSE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200376 ALIST(curwin)->al_ga.ga_len += count;
377 if (old_argcount > 0 && curwin->w_arg_idx >= after)
378 curwin->w_arg_idx += count;
379 return;
380 }
381
382 for (i = 0; i < count; ++i)
383 vim_free(files[i]);
384}
385
386/*
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100387 * Delete the file names in 'alist_ga' from the argument list.
388 */
389 static void
390arglist_del_files(garray_T *alist_ga)
391{
392 regmatch_T regmatch;
393 int didone;
394 int i;
395 char_u *p;
396 int match;
397
398 // Delete the items: use each item as a regexp and find a match in the
399 // argument list.
400 regmatch.rm_ic = p_fic; // ignore case when 'fileignorecase' is set
401 for (i = 0; i < alist_ga->ga_len && !got_int; ++i)
402 {
403 p = ((char_u **)alist_ga->ga_data)[i];
404 p = file_pat_to_reg_pat(p, NULL, NULL, FALSE);
405 if (p == NULL)
406 break;
407 regmatch.regprog = vim_regcomp(p, magic_isset() ? RE_MAGIC : 0);
408 if (regmatch.regprog == NULL)
409 {
410 vim_free(p);
411 break;
412 }
413
414 didone = FALSE;
415 for (match = 0; match < ARGCOUNT; ++match)
Bram Moolenaar24fe33a2022-11-24 00:09:02 +0000416 if (vim_regexec(&regmatch, alist_name(&ARGLIST[match]), (colnr_T)0))
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100417 {
418 didone = TRUE;
419 vim_free(ARGLIST[match].ae_fname);
420 mch_memmove(ARGLIST + match, ARGLIST + match + 1,
421 (ARGCOUNT - match - 1) * sizeof(aentry_T));
422 --ALIST(curwin)->al_ga.ga_len;
423 if (curwin->w_arg_idx > match)
424 --curwin->w_arg_idx;
425 --match;
426 }
427
428 vim_regfree(regmatch.regprog);
429 vim_free(p);
430 if (!didone)
431 semsg(_(e_no_match_str_2), ((char_u **)alist_ga->ga_data)[i]);
432 }
433 ga_clear(alist_ga);
434}
435
436/*
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200437 * "what" == AL_SET: Redefine the argument list to 'str'.
438 * "what" == AL_ADD: add files in 'str' to the argument list after "after".
439 * "what" == AL_DEL: remove files in 'str' from the argument list.
440 *
441 * Return FAIL for failure, OK otherwise.
442 */
443 static int
444do_arglist(
445 char_u *str,
446 int what,
447 int after UNUSED, // 0 means before first one
448 int will_edit) // will edit added argument
449{
450 garray_T new_ga;
451 int exp_count;
452 char_u **exp_files;
453 int i;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200454 int arg_escaped = TRUE;
455
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100456 if (check_arglist_locked() == FAIL)
457 return FAIL;
458
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200459 // Set default argument for ":argadd" command.
460 if (what == AL_ADD && *str == NUL)
461 {
462 if (curbuf->b_ffname == NULL)
463 return FAIL;
464 str = curbuf->b_fname;
465 arg_escaped = FALSE;
466 }
467
468 // Collect all file name arguments in "new_ga".
469 if (get_arglist(&new_ga, str, arg_escaped) == FAIL)
470 return FAIL;
471
472 if (what == AL_DEL)
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100473 arglist_del_files(&new_ga);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200474 else
475 {
476 i = expand_wildcards(new_ga.ga_len, (char_u **)new_ga.ga_data,
477 &exp_count, &exp_files, EW_DIR|EW_FILE|EW_ADDSLASH|EW_NOTFOUND);
478 ga_clear(&new_ga);
479 if (i == FAIL || exp_count == 0)
480 {
Bram Moolenaar460ae5d2022-01-01 14:19:49 +0000481 emsg(_(e_no_match));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200482 return FAIL;
483 }
484
485 if (what == AL_ADD)
486 {
487 alist_add_list(exp_count, exp_files, after, will_edit);
488 vim_free(exp_files);
489 }
490 else // what == AL_SET
491 alist_set(ALIST(curwin), exp_count, exp_files, will_edit, NULL, 0);
492 }
493
494 alist_check_arg_idx();
495
496 return OK;
497}
498
499/*
500 * Redefine the argument list.
501 */
502 void
503set_arglist(char_u *str)
504{
Rocco Maof96dc8d2024-01-23 21:27:19 +0100505 do_arglist(str, AL_SET, 0, TRUE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200506}
507
508/*
509 * Return TRUE if window "win" is editing the file at the current argument
510 * index.
511 */
512 int
513editing_arg_idx(win_T *win)
514{
515 return !(win->w_arg_idx >= WARGCOUNT(win)
516 || (win->w_buffer->b_fnum
517 != WARGLIST(win)[win->w_arg_idx].ae_fnum
518 && (win->w_buffer->b_ffname == NULL
519 || !(fullpathcmp(
520 alist_name(&WARGLIST(win)[win->w_arg_idx]),
521 win->w_buffer->b_ffname, TRUE, TRUE) & FPC_SAME))));
522}
523
524/*
525 * Check if window "win" is editing the w_arg_idx file in its argument list.
526 */
527 void
528check_arg_idx(win_T *win)
529{
530 if (WARGCOUNT(win) > 1 && !editing_arg_idx(win))
531 {
532 // We are not editing the current entry in the argument list.
533 // Set "arg_had_last" if we are editing the last one.
534 win->w_arg_idx_invalid = TRUE;
535 if (win->w_arg_idx != WARGCOUNT(win) - 1
536 && arg_had_last == FALSE
537 && ALIST(win) == &global_alist
538 && GARGCOUNT > 0
539 && win->w_arg_idx < GARGCOUNT
540 && (win->w_buffer->b_fnum == GARGLIST[GARGCOUNT - 1].ae_fnum
541 || (win->w_buffer->b_ffname != NULL
542 && (fullpathcmp(alist_name(&GARGLIST[GARGCOUNT - 1]),
543 win->w_buffer->b_ffname, TRUE, TRUE) & FPC_SAME))))
544 arg_had_last = TRUE;
545 }
546 else
547 {
548 // We are editing the current entry in the argument list.
549 // Set "arg_had_last" if it's also the last one
550 win->w_arg_idx_invalid = FALSE;
551 if (win->w_arg_idx == WARGCOUNT(win) - 1
552 && win->w_alist == &global_alist)
553 arg_had_last = TRUE;
554 }
555}
556
557/*
558 * ":args", ":argslocal" and ":argsglobal".
559 */
560 void
561ex_args(exarg_T *eap)
562{
563 int i;
564
565 if (eap->cmdidx != CMD_args)
566 {
Bram Moolenaar6bcb8772021-02-03 21:23:29 +0100567 if (check_arglist_locked() == FAIL)
568 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200569 alist_unlink(ALIST(curwin));
570 if (eap->cmdidx == CMD_argglobal)
571 ALIST(curwin) = &global_alist;
572 else // eap->cmdidx == CMD_arglocal
573 alist_new();
574 }
575
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100576 // ":args file ..": define new argument list, handle like ":next"
577 // Also for ":argslocal file .." and ":argsglobal file ..".
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200578 if (*eap->arg != NUL)
579 {
Bram Moolenaar6bcb8772021-02-03 21:23:29 +0100580 if (check_arglist_locked() == FAIL)
581 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200582 ex_next(eap);
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100583 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200584 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100585
586 // ":args": list arguments.
587 if (eap->cmdidx == CMD_args)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200588 {
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100589 char_u **items;
590
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100591 if (ARGCOUNT <= 0)
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100592 return; // empty argument list
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200593
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100594 items = ALLOC_MULT(char_u *, ARGCOUNT);
595 if (items == NULL)
596 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200597
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100598 // Overwrite the command, for a short list there is no scrolling
599 // required and no wait_return().
600 gotocmdline(TRUE);
601
602 for (i = 0; i < ARGCOUNT; ++i)
603 items[i] = alist_name(&ARGLIST[i]);
604 list_in_columns(items, ARGCOUNT, curwin->w_arg_idx);
605 vim_free(items);
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100606
607 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200608 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100609
610 // ":argslocal": make a local copy of the global argument list.
611 if (eap->cmdidx == CMD_arglocal)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200612 {
613 garray_T *gap = &curwin->w_alist->al_ga;
614
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +0100615 if (GA_GROW_FAILS(gap, GARGCOUNT))
616 return;
617
618 for (i = 0; i < GARGCOUNT; ++i)
619 if (GARGLIST[i].ae_fname != NULL)
620 {
621 AARGLIST(curwin->w_alist)[gap->ga_len].ae_fname =
622 vim_strsave(GARGLIST[i].ae_fname);
623 AARGLIST(curwin->w_alist)[gap->ga_len].ae_fnum =
624 GARGLIST[i].ae_fnum;
625 ++gap->ga_len;
626 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200627 }
628}
629
630/*
631 * ":previous", ":sprevious", ":Next" and ":sNext".
632 */
633 void
634ex_previous(exarg_T *eap)
635{
636 // If past the last one already, go to the last one.
637 if (curwin->w_arg_idx - (int)eap->line2 >= ARGCOUNT)
638 do_argfile(eap, ARGCOUNT - 1);
639 else
640 do_argfile(eap, curwin->w_arg_idx - (int)eap->line2);
641}
642
643/*
644 * ":rewind", ":first", ":sfirst" and ":srewind".
645 */
646 void
647ex_rewind(exarg_T *eap)
648{
649 do_argfile(eap, 0);
650}
651
652/*
653 * ":last" and ":slast".
654 */
655 void
656ex_last(exarg_T *eap)
657{
658 do_argfile(eap, ARGCOUNT - 1);
659}
660
661/*
662 * ":argument" and ":sargument".
663 */
664 void
665ex_argument(exarg_T *eap)
666{
667 int i;
668
669 if (eap->addr_count > 0)
670 i = eap->line2 - 1;
671 else
672 i = curwin->w_arg_idx;
673 do_argfile(eap, i);
674}
675
676/*
677 * Edit file "argn" of the argument lists.
678 */
679 void
680do_argfile(exarg_T *eap, int argn)
681{
682 int other;
683 char_u *p;
684 int old_arg_idx = curwin->w_arg_idx;
Colin Kennedy21570352024-03-03 16:16:47 +0100685 int is_split_cmd = *eap->cmd == 's';
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200686
Bram Moolenaar3c01c4a2020-02-01 23:04:24 +0100687 if (ERROR_IF_ANY_POPUP_WINDOW)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200688 return;
689 if (argn < 0 || argn >= ARGCOUNT)
690 {
691 if (ARGCOUNT <= 1)
Bram Moolenaar1a992222021-12-31 17:25:48 +0000692 emsg(_(e_there_is_only_one_file_to_edit));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200693 else if (argn < 0)
Bram Moolenaar1a992222021-12-31 17:25:48 +0000694 emsg(_(e_cannot_go_before_first_file));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200695 else
Bram Moolenaar1a992222021-12-31 17:25:48 +0000696 emsg(_(e_cannot_go_beyond_last_file));
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000697
698 return;
699 }
700
Colin Kennedy21570352024-03-03 16:16:47 +0100701 if (!is_split_cmd
702 && (&ARGLIST[argn])->ae_fnum != curbuf->b_fnum
703 && !check_can_set_curbuf_forceit(eap->forceit))
704 return;
705
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000706 setpcmark();
707#ifdef FEAT_GUI
708 need_mouse_correct = TRUE;
709#endif
710
711 // split window or create new tab page first
Colin Kennedy21570352024-03-03 16:16:47 +0100712 if (is_split_cmd || cmdmod.cmod_tab != 0)
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000713 {
714 if (win_split(0, 0) == FAIL)
715 return;
716 RESET_BINDING(curwin);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200717 }
718 else
719 {
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000720 // if 'hidden' set, only check for changed file when re-editing
721 // the same buffer
722 other = TRUE;
723 if (buf_hide(curbuf))
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200724 {
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000725 p = fix_fname(alist_name(&ARGLIST[argn]));
726 other = otherfile(p);
727 vim_free(p);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200728 }
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000729 if ((!buf_hide(curbuf) || !other)
730 && check_changed(curbuf, CCGD_AW
731 | (other ? 0 : CCGD_MULTWIN)
732 | (eap->forceit ? CCGD_FORCEIT : 0)
733 | CCGD_EXCMD))
734 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200735 }
Yegappan Lakshmanandc4daa32023-01-02 16:54:53 +0000736
737 curwin->w_arg_idx = argn;
738 if (argn == ARGCOUNT - 1 && curwin->w_alist == &global_alist)
739 arg_had_last = TRUE;
740
741 // Edit the file; always use the last known line number.
742 // When it fails (e.g. Abort for already edited file) restore the
743 // argument index.
744 if (do_ecmd(0, alist_name(&ARGLIST[curwin->w_arg_idx]), NULL,
745 eap, ECMD_LAST,
746 (buf_hide(curwin->w_buffer) ? ECMD_HIDE : 0)
747 + (eap->forceit ? ECMD_FORCEIT : 0), curwin) == FAIL)
748 curwin->w_arg_idx = old_arg_idx;
749 // like Vi: set the mark where the cursor is in the file.
750 else if (eap->cmdidx != CMD_argdo)
751 setmark('\'');
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200752}
753
754/*
755 * ":next", and commands that behave like it.
756 */
757 void
758ex_next(exarg_T *eap)
759{
760 int i;
761
762 // check for changed buffer now, if this fails the argument list is not
763 // redefined.
764 if ( buf_hide(curbuf)
765 || eap->cmdidx == CMD_snext
766 || !check_changed(curbuf, CCGD_AW
767 | (eap->forceit ? CCGD_FORCEIT : 0)
768 | CCGD_EXCMD))
769 {
770 if (*eap->arg != NUL) // redefine file list
771 {
772 if (do_arglist(eap->arg, AL_SET, 0, TRUE) == FAIL)
773 return;
774 i = 0;
775 }
776 else
777 i = curwin->w_arg_idx + (int)eap->line2;
778 do_argfile(eap, i);
779 }
780}
781
782/*
Nir Lichtman73a02422021-12-24 20:28:03 +0000783 * ":argdedupe"
784 */
785 void
786ex_argdedupe(exarg_T *eap UNUSED)
787{
788 int i;
789 int j;
790
791 for (i = 0; i < ARGCOUNT; ++i)
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000792 {
793 // Expand each argument to a full path to catch different paths leading
794 // to the same file.
795 char_u *firstFullname = FullName_save(ARGLIST[i].ae_fname, FALSE);
796 if (firstFullname == NULL)
797 return; // out of memory
798
Nir Lichtman73a02422021-12-24 20:28:03 +0000799 for (j = i + 1; j < ARGCOUNT; ++j)
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000800 {
801 char_u *secondFullname = FullName_save(ARGLIST[j].ae_fname, FALSE);
802 if (secondFullname == NULL)
803 break; // out of memory
804 int areNamesDuplicate =
805 fnamecmp(firstFullname, secondFullname) == 0;
806 vim_free(secondFullname);
807
808 if (areNamesDuplicate)
Nir Lichtman73a02422021-12-24 20:28:03 +0000809 {
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000810 // remove one duplicate argument
Nir Lichtman73a02422021-12-24 20:28:03 +0000811 vim_free(ARGLIST[j].ae_fname);
812 mch_memmove(ARGLIST + j, ARGLIST + j + 1,
813 (ARGCOUNT - j - 1) * sizeof(aentry_T));
814 --ARGCOUNT;
815
816 if (curwin->w_arg_idx == j)
817 curwin->w_arg_idx = i;
818 else if (curwin->w_arg_idx > j)
819 --curwin->w_arg_idx;
820
821 --j;
822 }
Nir Lichtmanb3052aa2022-11-12 17:00:31 +0000823 }
824
825 vim_free(firstFullname);
826 }
Nir Lichtman73a02422021-12-24 20:28:03 +0000827}
828
829/*
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200830 * ":argedit"
831 */
832 void
833ex_argedit(exarg_T *eap)
834{
835 int i = eap->addr_count ? (int)eap->line2 : curwin->w_arg_idx + 1;
836 // Whether curbuf will be reused, curbuf->b_ffname will be set.
837 int curbuf_is_reusable = curbuf_reusable();
838
839 if (do_arglist(eap->arg, AL_ADD, i, TRUE) == FAIL)
840 return;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200841 maketitle();
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200842
843 if (curwin->w_arg_idx == 0
844 && (curbuf->b_ml.ml_flags & ML_EMPTY)
845 && (curbuf->b_ffname == NULL || curbuf_is_reusable))
846 i = 0;
847 // Edit the argument.
848 if (i < ARGCOUNT)
849 do_argfile(eap, i);
850}
851
852/*
853 * ":argadd"
854 */
855 void
856ex_argadd(exarg_T *eap)
857{
858 do_arglist(eap->arg, AL_ADD,
859 eap->addr_count > 0 ? (int)eap->line2 : curwin->w_arg_idx + 1,
860 FALSE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200861 maketitle();
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200862}
863
864/*
865 * ":argdelete"
866 */
867 void
868ex_argdelete(exarg_T *eap)
869{
870 int i;
871 int n;
872
Bram Moolenaar5ed58c72021-01-28 14:24:55 +0100873 if (check_arglist_locked() == FAIL)
874 return;
875
Bram Moolenaar7b221172020-08-17 19:34:10 +0200876 if (eap->addr_count > 0 || *eap->arg == NUL)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200877 {
Bram Moolenaar02c037a2020-08-30 19:26:45 +0200878 // ":argdel" works like ":.argdel"
Bram Moolenaar7b221172020-08-17 19:34:10 +0200879 if (eap->addr_count == 0)
880 {
881 if (curwin->w_arg_idx >= ARGCOUNT)
882 {
Bram Moolenaarf1474d82021-12-31 19:59:55 +0000883 emsg(_(e_no_argument_to_delete));
Bram Moolenaar7b221172020-08-17 19:34:10 +0200884 return;
885 }
886 eap->line1 = eap->line2 = curwin->w_arg_idx + 1;
887 }
888 else if (eap->line2 > ARGCOUNT)
889 // ":1,4argdel": Delete all arguments in the range.
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200890 eap->line2 = ARGCOUNT;
891 n = eap->line2 - eap->line1 + 1;
892 if (*eap->arg != NUL)
893 // Can't have both a range and an argument.
Bram Moolenaar436b5ad2021-12-31 22:49:24 +0000894 emsg(_(e_invalid_argument));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200895 else if (n <= 0)
896 {
897 // Don't give an error for ":%argdel" if the list is empty.
898 if (eap->line1 != 1 || eap->line2 != 0)
Bram Moolenaar108010a2021-06-27 22:03:33 +0200899 emsg(_(e_invalid_range));
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200900 }
901 else
902 {
903 for (i = eap->line1; i <= eap->line2; ++i)
904 vim_free(ARGLIST[i - 1].ae_fname);
905 mch_memmove(ARGLIST + eap->line1 - 1, ARGLIST + eap->line2,
906 (size_t)((ARGCOUNT - eap->line2) * sizeof(aentry_T)));
907 ALIST(curwin)->al_ga.ga_len -= n;
908 if (curwin->w_arg_idx >= eap->line2)
909 curwin->w_arg_idx -= n;
910 else if (curwin->w_arg_idx > eap->line1)
911 curwin->w_arg_idx = eap->line1;
912 if (ARGCOUNT == 0)
913 curwin->w_arg_idx = 0;
914 else if (curwin->w_arg_idx >= ARGCOUNT)
915 curwin->w_arg_idx = ARGCOUNT - 1;
916 }
917 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200918 else
919 do_arglist(eap->arg, AL_DEL, 0, FALSE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200920 maketitle();
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200921}
922
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200923/*
924 * Function given to ExpandGeneric() to obtain the possible arguments of the
925 * argedit and argdelete commands.
926 */
927 char_u *
928get_arglist_name(expand_T *xp UNUSED, int idx)
929{
930 if (idx >= ARGCOUNT)
931 return NULL;
932
933 return alist_name(&ARGLIST[idx]);
934}
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200935
936/*
937 * Get the file name for an argument list entry.
938 */
939 char_u *
940alist_name(aentry_T *aep)
941{
942 buf_T *bp;
943
944 // Use the name from the associated buffer if it exists.
945 bp = buflist_findnr(aep->ae_fnum);
946 if (bp == NULL || bp->b_fname == NULL)
947 return aep->ae_fname;
948 return bp->b_fname;
949}
950
951/*
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100952 * State used by the :all command to open all the files in the argument list in
953 * separate windows.
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200954 */
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100955typedef struct {
956 alist_T *alist; // argument list to be used
957 int had_tab;
958 int keep_tabs;
959 int forceit;
960
961 int use_firstwin; // use first window for arglist
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200962 char_u *opened; // Array of weight for which args are open:
963 // 0: not opened
964 // 1: opened in other tab
965 // 2: opened in curtab
966 // 3: opened in curtab and curwin
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200967 int opened_len; // length of opened[]
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100968 win_T *new_curwin;
969 tabpage_T *new_curtab;
970} arg_all_state_T;
971
972/*
973 * Close all the windows containing files which are not in the argument list.
974 * Used by the ":all" command.
975 */
976 static void
977arg_all_close_unused_windows(arg_all_state_T *aall)
978{
979 win_T *wp;
980 win_T *wpnext;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200981 tabpage_T *tpnext;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100982 buf_T *buf;
983 int i;
984 win_T *old_curwin;
985 tabpage_T *old_curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200986
987 old_curwin = curwin;
988 old_curtab = curtab;
989
Yegappan Lakshmanan88947612022-09-05 18:27:47 +0100990 if (aall->had_tab > 0)
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200991 goto_tabpage_tp(first_tabpage, TRUE, TRUE);
Christian Brabandtdf12e392023-12-16 13:55:32 +0100992
993 // moving tabpages around in an autocommand may cause an endless loop
994 tabpage_move_disallowed++;
Bram Moolenaar4ad62152019-08-17 14:38:55 +0200995 for (;;)
996 {
997 tpnext = curtab->tp_next;
998 for (wp = firstwin; wp != NULL; wp = wpnext)
999 {
1000 wpnext = wp->w_next;
1001 buf = wp->w_buffer;
1002 if (buf->b_ffname == NULL
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001003 || (!aall->keep_tabs && (buf->b_nwindows > 1
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001004 || wp->w_width != Columns)))
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001005 i = aall->opened_len;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001006 else
1007 {
1008 // check if the buffer in this window is in the arglist
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001009 for (i = 0; i < aall->opened_len; ++i)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001010 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001011 if (i < aall->alist->al_ga.ga_len
1012 && (AARGLIST(aall->alist)[i].ae_fnum == buf->b_fnum
Bram Moolenaar747f1102022-09-18 13:06:41 +01001013 || fullpathcmp(alist_name(
1014 &AARGLIST(aall->alist)[i]),
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001015 buf->b_ffname, TRUE, TRUE) & FPC_SAME))
1016 {
1017 int weight = 1;
1018
1019 if (old_curtab == curtab)
1020 {
1021 ++weight;
1022 if (old_curwin == wp)
1023 ++weight;
1024 }
1025
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001026 if (weight > (int)aall->opened[i])
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001027 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001028 aall->opened[i] = (char_u)weight;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001029 if (i == 0)
1030 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001031 if (aall->new_curwin != NULL)
Bram Moolenaar747f1102022-09-18 13:06:41 +01001032 aall->new_curwin->w_arg_idx =
1033 aall->opened_len;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001034 aall->new_curwin = wp;
1035 aall->new_curtab = curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001036 }
1037 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001038 else if (aall->keep_tabs)
1039 i = aall->opened_len;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001040
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001041 if (wp->w_alist != aall->alist)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001042 {
1043 // Use the current argument list for all windows
1044 // containing a file from it.
1045 alist_unlink(wp->w_alist);
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001046 wp->w_alist = aall->alist;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001047 ++wp->w_alist->al_refcount;
1048 }
1049 break;
1050 }
1051 }
1052 }
1053 wp->w_arg_idx = i;
1054
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001055 if (i == aall->opened_len && !aall->keep_tabs)// close this window
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001056 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001057 if (buf_hide(buf) || aall->forceit || buf->b_nwindows > 1
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001058 || !bufIsChanged(buf))
1059 {
1060 // If the buffer was changed, and we would like to hide it,
1061 // try autowriting.
1062 if (!buf_hide(buf) && buf->b_nwindows <= 1
1063 && bufIsChanged(buf))
1064 {
1065 bufref_T bufref;
1066
1067 set_bufref(&bufref, buf);
1068
1069 (void)autowrite(buf, FALSE);
1070
1071 // check if autocommands removed the window
1072 if (!win_valid(wp) || !bufref_valid(&bufref))
1073 {
1074 wpnext = firstwin; // start all over...
1075 continue;
1076 }
1077 }
1078 // don't close last window
1079 if (ONE_WINDOW
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001080 && (first_tabpage->tp_next == NULL
1081 || !aall->had_tab))
1082 aall->use_firstwin = TRUE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001083 else
1084 {
1085 win_close(wp, !buf_hide(buf) && !bufIsChanged(buf));
1086
1087 // check if autocommands removed the next window
1088 if (!win_valid(wpnext))
1089 wpnext = firstwin; // start all over...
1090 }
1091 }
1092 }
1093 }
1094
1095 // Without the ":tab" modifier only do the current tab page.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001096 if (aall->had_tab == 0 || tpnext == NULL)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001097 break;
1098
1099 // check if autocommands removed the next tab page
1100 if (!valid_tabpage(tpnext))
1101 tpnext = first_tabpage; // start all over...
1102
1103 goto_tabpage_tp(tpnext, TRUE, TRUE);
1104 }
Christian Brabandtdf12e392023-12-16 13:55:32 +01001105 tabpage_move_disallowed--;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001106}
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001107
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001108/*
1109 * Open upto "count" windows for the files in the argument list 'aall->alist'.
1110 */
1111 static void
1112arg_all_open_windows(arg_all_state_T *aall, int count)
1113{
1114 win_T *wp;
1115 int tab_drop_empty_window = FALSE;
1116 int i;
1117 int split_ret = OK;
1118 int p_ea_save;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001119
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001120 // ":tab drop file" should re-use an empty window to avoid "--remote-tab"
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001121 // leaving an empty tab page when executed locally.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001122 if (aall->keep_tabs && BUFEMPTY() && curbuf->b_nwindows == 1
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001123 && curbuf->b_ffname == NULL && !curbuf->b_changed)
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001124 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001125 aall->use_firstwin = TRUE;
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001126 tab_drop_empty_window = TRUE;
1127 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001128
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001129 for (i = 0; i < count && !got_int; ++i)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001130 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001131 if (aall->alist == &global_alist && i == global_alist.al_ga.ga_len - 1)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001132 arg_had_last = TRUE;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001133 if (aall->opened[i] > 0)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001134 {
1135 // Move the already present window to below the current window
1136 if (curwin->w_arg_idx != i)
1137 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001138 FOR_ALL_WINDOWS(wp)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001139 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001140 if (wp->w_arg_idx == i)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001141 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001142 if (aall->keep_tabs)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001143 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001144 aall->new_curwin = wp;
1145 aall->new_curtab = curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001146 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001147 else if (wp->w_frame->fr_parent
1148 != curwin->w_frame->fr_parent)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001149 {
Bram Moolenaarf1474d82021-12-31 19:59:55 +00001150 emsg(_(e_window_layout_changed_unexpectedly));
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001151 i = count;
1152 break;
1153 }
1154 else
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001155 win_move_after(wp, curwin);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001156 break;
1157 }
1158 }
1159 }
1160 }
1161 else if (split_ret == OK)
1162 {
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001163 // trigger events for tab drop
1164 if (tab_drop_empty_window && i == count - 1)
1165 --autocmd_no_enter;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001166 if (!aall->use_firstwin) // split current window
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001167 {
1168 p_ea_save = p_ea;
1169 p_ea = TRUE; // use space from all windows
1170 split_ret = win_split(0, WSP_ROOM | WSP_BELOW);
1171 p_ea = p_ea_save;
1172 if (split_ret == FAIL)
1173 continue;
1174 }
1175 else // first window: do autocmd for leaving this buffer
1176 --autocmd_no_leave;
1177
1178 // edit file "i"
1179 curwin->w_arg_idx = i;
1180 if (i == 0)
1181 {
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001182 aall->new_curwin = curwin;
1183 aall->new_curtab = curtab;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001184 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001185 (void)do_ecmd(0, alist_name(&AARGLIST(aall->alist)[i]), NULL, NULL,
1186 ECMD_ONE,
1187 ((buf_hide(curwin->w_buffer)
1188 || bufIsChanged(curwin->w_buffer)) ? ECMD_HIDE : 0)
1189 + ECMD_OLDBUF, curwin);
Bram Moolenaarc10b5212020-01-13 20:54:51 +01001190 if (tab_drop_empty_window && i == count - 1)
1191 ++autocmd_no_enter;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001192 if (aall->use_firstwin)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001193 ++autocmd_no_leave;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001194 aall->use_firstwin = FALSE;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001195 }
1196 ui_breakcheck();
1197
1198 // When ":tab" was used open a new tab for a new window repeatedly.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001199 if (aall->had_tab > 0 && tabpage_index(NULL) <= p_tpm)
Bram Moolenaare1004402020-10-24 20:49:43 +02001200 cmdmod.cmod_tab = 9999;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001201 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001202}
1203
1204/*
1205 * do_arg_all(): Open up to "count" windows, one for each argument.
1206 */
1207 static void
1208do_arg_all(
1209 int count,
1210 int forceit, // hide buffers in current windows
1211 int keep_tabs) // keep current tabs, for ":tab drop file"
1212{
1213 arg_all_state_T aall;
1214 win_T *last_curwin;
1215 tabpage_T *last_curtab;
1216 int prev_arglist_locked = arglist_locked;
1217
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001218 if (cmdwin_type != 0)
1219 {
1220 emsg(_(e_invalid_in_cmdline_window));
1221 return;
1222 }
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001223 if (ARGCOUNT <= 0)
1224 {
1225 // Don't give an error message. We don't want it when the ":all"
1226 // command is in the .vimrc.
1227 return;
1228 }
1229 setpcmark();
1230
1231 aall.use_firstwin = FALSE;
1232 aall.had_tab = cmdmod.cmod_tab;
1233 aall.new_curwin = NULL;
1234 aall.new_curtab = NULL;
1235 aall.forceit = forceit;
1236 aall.keep_tabs = keep_tabs;
1237 aall.opened_len = ARGCOUNT;
1238 aall.opened = alloc_clear(aall.opened_len);
1239 if (aall.opened == NULL)
1240 return;
1241
1242 // Autocommands may do anything to the argument list. Make sure it's not
1243 // freed while we are working here by "locking" it. We still have to
1244 // watch out for its size being changed.
1245 aall.alist = curwin->w_alist;
1246 ++aall.alist->al_refcount;
1247 arglist_locked = TRUE;
1248
1249#ifdef FEAT_GUI
1250 need_mouse_correct = TRUE;
1251#endif
1252
Bram Moolenaar8281a162023-04-20 18:07:57 +01001253 tabpage_T *new_lu_tp = curtab;
1254
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001255 // Try closing all windows that are not in the argument list.
1256 // Also close windows that are not full width;
1257 // When 'hidden' or "forceit" set the buffer becomes hidden.
1258 // Windows that have a changed buffer and can't be hidden won't be closed.
1259 // When the ":tab" modifier was used do this for all tab pages.
1260 arg_all_close_unused_windows(&aall);
1261
1262 // Open a window for files in the argument list that don't have one.
1263 // ARGCOUNT may change while doing this, because of autocommands.
1264 if (count > aall.opened_len || count <= 0)
1265 count = aall.opened_len;
1266
1267 // Don't execute Win/Buf Enter/Leave autocommands here.
1268 ++autocmd_no_enter;
1269 ++autocmd_no_leave;
1270 last_curwin = curwin;
1271 last_curtab = curtab;
1272 win_enter(lastwin, FALSE);
1273
1274 /*
1275 * Open upto "count" windows.
1276 */
1277 arg_all_open_windows(&aall, count);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001278
1279 // Remove the "lock" on the argument list.
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001280 alist_unlink(aall.alist);
Bram Moolenaar6f983712021-12-24 18:11:27 +00001281 arglist_locked = prev_arglist_locked;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001282
1283 --autocmd_no_enter;
1284
1285 // restore last referenced tabpage's curwin
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001286 if (last_curtab != aall.new_curtab)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001287 {
1288 if (valid_tabpage(last_curtab))
1289 goto_tabpage_tp(last_curtab, TRUE, TRUE);
1290 if (win_valid(last_curwin))
1291 win_enter(last_curwin, FALSE);
1292 }
1293 // to window with first arg
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001294 if (valid_tabpage(aall.new_curtab))
1295 goto_tabpage_tp(aall.new_curtab, TRUE, TRUE);
glepnir2975a542024-02-09 19:30:26 +01001296
1297 // Now set the last used tabpage to where we started.
1298 if (valid_tabpage(new_lu_tp))
1299 lastused_tabpage = new_lu_tp;
1300
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001301 if (win_valid(aall.new_curwin))
1302 win_enter(aall.new_curwin, FALSE);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001303
1304 --autocmd_no_leave;
Yegappan Lakshmanan88947612022-09-05 18:27:47 +01001305 vim_free(aall.opened);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001306}
1307
1308/*
1309 * ":all" and ":sall".
1310 * Also used for ":tab drop file ..." after setting the argument list.
1311 */
1312 void
1313ex_all(exarg_T *eap)
1314{
1315 if (eap->addr_count == 0)
1316 eap->line2 = 9999;
1317 do_arg_all((int)eap->line2, eap->forceit, eap->cmdidx == CMD_drop);
1318}
1319
1320/*
1321 * Concatenate all files in the argument list, separated by spaces, and return
1322 * it in one allocated string.
1323 * Spaces and backslashes in the file names are escaped with a backslash.
1324 * Returns NULL when out of memory.
1325 */
1326 char_u *
1327arg_all(void)
1328{
1329 int len;
1330 int idx;
1331 char_u *retval = NULL;
1332 char_u *p;
1333
1334 // Do this loop two times:
1335 // first time: compute the total length
1336 // second time: concatenate the names
1337 for (;;)
1338 {
1339 len = 0;
1340 for (idx = 0; idx < ARGCOUNT; ++idx)
1341 {
1342 p = alist_name(&ARGLIST[idx]);
zeertzjq101d57b2022-07-31 18:34:32 +01001343 if (p == NULL)
1344 continue;
1345 if (len > 0)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001346 {
zeertzjq101d57b2022-07-31 18:34:32 +01001347 // insert a space in between names
1348 if (retval != NULL)
1349 retval[len] = ' ';
1350 ++len;
1351 }
1352 for ( ; *p != NUL; ++p)
1353 {
1354 if (*p == ' '
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001355#ifndef BACKSLASH_IN_FILENAME
zeertzjq101d57b2022-07-31 18:34:32 +01001356 || *p == '\\'
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001357#endif
zeertzjq101d57b2022-07-31 18:34:32 +01001358 || *p == '`')
1359 {
1360 // insert a backslash
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001361 if (retval != NULL)
zeertzjq101d57b2022-07-31 18:34:32 +01001362 retval[len] = '\\';
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001363 ++len;
1364 }
zeertzjq101d57b2022-07-31 18:34:32 +01001365 if (retval != NULL)
1366 retval[len] = *p;
1367 ++len;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001368 }
1369 }
1370
1371 // second time: break here
1372 if (retval != NULL)
1373 {
1374 retval[len] = NUL;
1375 break;
1376 }
1377
1378 // allocate memory
1379 retval = alloc(len + 1);
1380 if (retval == NULL)
1381 break;
1382 }
1383
1384 return retval;
1385}
1386
1387#if defined(FEAT_EVAL) || defined(PROTO)
1388/*
1389 * "argc([window id])" function
1390 */
1391 void
1392f_argc(typval_T *argvars, typval_T *rettv)
1393{
1394 win_T *wp;
1395
Yegappan Lakshmanan4490ec42021-07-27 22:00:44 +02001396 if (in_vim9script() && check_for_opt_number_arg(argvars, 0) == FAIL)
1397 return;
1398
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001399 if (argvars[0].v_type == VAR_UNKNOWN)
1400 // use the current window
1401 rettv->vval.v_number = ARGCOUNT;
1402 else if (argvars[0].v_type == VAR_NUMBER
1403 && tv_get_number(&argvars[0]) == -1)
1404 // use the global argument list
1405 rettv->vval.v_number = GARGCOUNT;
1406 else
1407 {
1408 // use the argument list of the specified window
1409 wp = find_win_by_nr_or_id(&argvars[0]);
1410 if (wp != NULL)
1411 rettv->vval.v_number = WARGCOUNT(wp);
1412 else
1413 rettv->vval.v_number = -1;
1414 }
1415}
1416
1417/*
1418 * "argidx()" function
1419 */
1420 void
1421f_argidx(typval_T *argvars UNUSED, typval_T *rettv)
1422{
1423 rettv->vval.v_number = curwin->w_arg_idx;
1424}
1425
1426/*
1427 * "arglistid()" function
1428 */
1429 void
1430f_arglistid(typval_T *argvars, typval_T *rettv)
1431{
1432 win_T *wp;
1433
Yegappan Lakshmanan4490ec42021-07-27 22:00:44 +02001434 if (in_vim9script()
1435 && (check_for_opt_number_arg(argvars, 0) == FAIL
1436 || (argvars[0].v_type != VAR_UNKNOWN
1437 && check_for_opt_number_arg(argvars, 1) == FAIL)))
1438 return;
1439
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001440 rettv->vval.v_number = -1;
1441 wp = find_tabwin(&argvars[0], &argvars[1], NULL);
1442 if (wp != NULL)
1443 rettv->vval.v_number = wp->w_alist->id;
1444}
1445
1446/*
1447 * Get the argument list for a given window
1448 */
1449 static void
1450get_arglist_as_rettv(aentry_T *arglist, int argcount, typval_T *rettv)
1451{
1452 int idx;
1453
1454 if (rettv_list_alloc(rettv) == OK && arglist != NULL)
1455 for (idx = 0; idx < argcount; ++idx)
1456 list_append_string(rettv->vval.v_list,
1457 alist_name(&arglist[idx]), -1);
1458}
1459
1460/*
1461 * "argv(nr)" function
1462 */
1463 void
1464f_argv(typval_T *argvars, typval_T *rettv)
1465{
1466 int idx;
1467 aentry_T *arglist = NULL;
1468 int argcount = -1;
1469
Yegappan Lakshmanan4490ec42021-07-27 22:00:44 +02001470 if (in_vim9script()
1471 && (check_for_opt_number_arg(argvars, 0) == FAIL
1472 || (argvars[0].v_type != VAR_UNKNOWN
1473 && check_for_opt_number_arg(argvars, 1) == FAIL)))
1474 return;
1475
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001476 if (argvars[0].v_type == VAR_UNKNOWN)
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001477 {
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001478 get_arglist_as_rettv(ARGLIST, ARGCOUNT, rettv);
1479 return;
1480 }
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001481
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001482 if (argvars[1].v_type == VAR_UNKNOWN)
1483 {
1484 arglist = ARGLIST;
1485 argcount = ARGCOUNT;
1486 }
1487 else if (argvars[1].v_type == VAR_NUMBER
1488 && tv_get_number(&argvars[1]) == -1)
1489 {
1490 arglist = GARGLIST;
1491 argcount = GARGCOUNT;
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001492 }
1493 else
Yegappan Lakshmananb1f471e2022-09-05 14:33:47 +01001494 {
1495 win_T *wp = find_win_by_nr_or_id(&argvars[1]);
1496
1497 if (wp != NULL)
1498 {
1499 // Use the argument list of the specified window
1500 arglist = WARGLIST(wp);
1501 argcount = WARGCOUNT(wp);
1502 }
1503 }
1504
1505 rettv->v_type = VAR_STRING;
1506 rettv->vval.v_string = NULL;
1507 idx = tv_get_number_chk(&argvars[0], NULL);
1508 if (arglist != NULL && idx >= 0 && idx < argcount)
1509 rettv->vval.v_string = vim_strsave(alist_name(&arglist[idx]));
1510 else if (idx == -1)
1511 get_arglist_as_rettv(arglist, argcount, rettv);
Bram Moolenaar4ad62152019-08-17 14:38:55 +02001512}
1513#endif