blob: f86d11eb3b206acd086809e3a4ff5776ef43efb5 [file] [log] [blame]
Bram Moolenaaredf3f972016-08-29 22:49:24 +02001/* vi:set ts=8 sts=4 sw=4 noet:
Bram Moolenaar071d4272004-06-13 20:20:40 +00002 *
3 * VIM - Vi IMproved by Bram Moolenaar
4 *
5 * Do ":help uganda" in Vim to read a list of people who contributed.
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#include "vim.h"
11
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010012static int path_is_url(char_u *p);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010013static void cmd_with_count(char *cmd, char_u *bufp, size_t bufsize, long Prenum);
14static void win_init(win_T *newp, win_T *oldp, int flags);
15static void win_init_some(win_T *newp, win_T *oldp);
16static void frame_comp_pos(frame_T *topfrp, int *row, int *col);
17static void frame_setheight(frame_T *curfrp, int height);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010018static void frame_setwidth(frame_T *curfrp, int width);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010019static void win_exchange(long);
20static void win_rotate(int, int);
21static void win_totop(int size, int flags);
22static void win_equal_rec(win_T *next_curwin, int current, frame_T *topfr, int dir, int col, int row, int width, int height);
23static int last_window(void);
24static int close_last_window_tabpage(win_T *win, int free_buf, tabpage_T *prev_curtab);
25static win_T *win_free_mem(win_T *win, int *dirp, tabpage_T *tp);
26static frame_T *win_altframe(win_T *win, tabpage_T *tp);
27static tabpage_T *alt_tabpage(void);
28static win_T *frame2win(frame_T *frp);
29static int frame_has_win(frame_T *frp, win_T *wp);
30static void frame_new_height(frame_T *topfrp, int height, int topfirst, int wfh);
31static int frame_fixed_height(frame_T *frp);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010032static int frame_fixed_width(frame_T *frp);
33static void frame_add_statusline(frame_T *frp);
34static void frame_new_width(frame_T *topfrp, int width, int leftfirst, int wfw);
35static void frame_add_vsep(frame_T *frp);
36static int frame_minwidth(frame_T *topfrp, win_T *next_curwin);
37static void frame_fix_width(win_T *wp);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010038static int win_alloc_firstwin(win_T *oldwin);
39static void new_frame(win_T *wp);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010040static tabpage_T *alloc_tabpage(void);
41static int leave_tabpage(buf_T *new_curbuf, int trigger_leave_autocmds);
42static void enter_tabpage(tabpage_T *tp, buf_T *old_curbuf, int trigger_enter_autocmds, int trigger_leave_autocmds);
43static void frame_fix_height(win_T *wp);
44static int frame_minheight(frame_T *topfrp, win_T *next_curwin);
Bram Moolenaarc917da42016-07-19 22:31:36 +020045static void win_enter_ext(win_T *wp, int undo_sync, int no_curwin, int trigger_new_autocmds, int trigger_enter_autocmds, int trigger_leave_autocmds);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010046static void win_free(win_T *wp, tabpage_T *tp);
47static void frame_append(frame_T *after, frame_T *frp);
48static void frame_insert(frame_T *before, frame_T *frp);
49static void frame_remove(frame_T *frp);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010050static void win_goto_ver(int up, long count);
51static void win_goto_hor(int left, long count);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010052static void frame_add_height(frame_T *frp, int n);
53static void last_status_rec(frame_T *fr, int statusline);
Bram Moolenaar071d4272004-06-13 20:20:40 +000054
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010055static void make_snapshot_rec(frame_T *fr, frame_T **frp);
56static void clear_snapshot(tabpage_T *tp, int idx);
57static void clear_snapshot_rec(frame_T *fr);
58static int check_snapshot_rec(frame_T *sn, frame_T *fr);
59static win_T *restore_snapshot_rec(frame_T *sn, frame_T *fr);
Bram Moolenaar071d4272004-06-13 20:20:40 +000060
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010061static int frame_check_height(frame_T *topfrp, int height);
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010062static int frame_check_width(frame_T *topfrp, int width);
Bram Moolenaarb893ac22013-06-26 14:04:47 +020063
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +010064static win_T *win_alloc(win_T *after, int hidden);
Bram Moolenaar071d4272004-06-13 20:20:40 +000065
66#define URL_SLASH 1 /* path_is_url() has found "://" */
67#define URL_BACKSLASH 2 /* path_is_url() has found ":\\" */
68
Bram Moolenaarb6799ac2007-05-10 16:44:05 +000069#define NOWIN (win_T *)-1 /* non-existing window */
Bram Moolenaar071d4272004-06-13 20:20:40 +000070
Bram Moolenaar4033c552017-09-16 20:54:51 +020071#define ROWS_AVAIL (Rows - p_ch - tabline_height())
Bram Moolenaar4c3f5362006-04-11 21:38:50 +000072
73static char *m_onlyone = N_("Already only one window");
74
Bram Moolenaar071d4272004-06-13 20:20:40 +000075/*
76 * all CTRL-W window commands are handled here, called from normal_cmd().
77 */
78 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +010079do_window(
80 int nchar,
81 long Prenum,
82 int xchar) /* extra char from ":wincmd gx" or NUL */
Bram Moolenaar071d4272004-06-13 20:20:40 +000083{
84 long Prenum1;
85 win_T *wp;
86#if defined(FEAT_SEARCHPATH) || defined(FEAT_FIND_ID)
87 char_u *ptr;
Bram Moolenaard1f56e62006-02-22 21:25:37 +000088 linenr_T lnum = -1;
Bram Moolenaar071d4272004-06-13 20:20:40 +000089#endif
90#ifdef FEAT_FIND_ID
91 int type = FIND_DEFINE;
92 int len;
93#endif
94 char_u cbuf[40];
95
96 if (Prenum == 0)
97 Prenum1 = 1;
98 else
99 Prenum1 = Prenum;
100
101#ifdef FEAT_CMDWIN
Bram Moolenaar6f470022018-04-10 18:47:20 +0200102# define CHECK_CMDWIN \
103 do { \
104 if (cmdwin_type != 0) \
105 { \
106 EMSG(_(e_cmdwin)); \
107 return; \
108 } \
109 } while (0)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000110#else
Bram Moolenaar6f470022018-04-10 18:47:20 +0200111# define CHECK_CMDWIN do { /**/ } while (0)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000112#endif
113
114 switch (nchar)
115 {
116/* split current window in two parts, horizontally */
117 case 'S':
118 case Ctrl_S:
119 case 's':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200120 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000121 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaarb1b715d2006-01-21 22:09:43 +0000122#ifdef FEAT_QUICKFIX
123 /* When splitting the quickfix window open a new buffer in it,
124 * don't replicate the quickfix buffer. */
125 if (bt_quickfix(curbuf))
126 goto newwindow;
127#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000128#ifdef FEAT_GUI
129 need_mouse_correct = TRUE;
130#endif
Bram Moolenaarcde88542015-08-11 19:14:00 +0200131 (void)win_split((int)Prenum, 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000132 break;
133
Bram Moolenaar071d4272004-06-13 20:20:40 +0000134/* split current window in two parts, vertically */
135 case Ctrl_V:
136 case 'v':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200137 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000138 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar44a2f922016-03-19 22:11:51 +0100139#ifdef FEAT_QUICKFIX
Bram Moolenaar990d95c2008-07-07 19:23:37 +0000140 /* When splitting the quickfix window open a new buffer in it,
141 * don't replicate the quickfix buffer. */
142 if (bt_quickfix(curbuf))
143 goto newwindow;
Bram Moolenaar44a2f922016-03-19 22:11:51 +0100144#endif
145#ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +0000146 need_mouse_correct = TRUE;
Bram Moolenaar44a2f922016-03-19 22:11:51 +0100147#endif
Bram Moolenaarcde88542015-08-11 19:14:00 +0200148 (void)win_split((int)Prenum, WSP_VERT);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000149 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000150
151/* split current window and edit alternate file */
152 case Ctrl_HAT:
153 case '^':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200154 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000155 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar2f1e51a2014-12-13 03:58:09 +0100156 cmd_with_count("split #", cbuf, sizeof(cbuf), Prenum);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000157 do_cmdline_cmd(cbuf);
158 break;
159
160/* open new window */
161 case Ctrl_N:
162 case 'n':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200163 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000164 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaarb1b715d2006-01-21 22:09:43 +0000165#ifdef FEAT_QUICKFIX
166newwindow:
167#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000168 if (Prenum)
Bram Moolenaar990d95c2008-07-07 19:23:37 +0000169 /* window height */
170 vim_snprintf((char *)cbuf, sizeof(cbuf) - 5, "%ld", Prenum);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000171 else
172 cbuf[0] = NUL;
Bram Moolenaar44a2f922016-03-19 22:11:51 +0100173#if defined(FEAT_QUICKFIX)
Bram Moolenaar990d95c2008-07-07 19:23:37 +0000174 if (nchar == 'v' || nchar == Ctrl_V)
175 STRCAT(cbuf, "v");
176#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000177 STRCAT(cbuf, "new");
178 do_cmdline_cmd(cbuf);
179 break;
180
181/* quit current window */
182 case Ctrl_Q:
183 case 'q':
Bram Moolenaar071d4272004-06-13 20:20:40 +0000184 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar2f1e51a2014-12-13 03:58:09 +0100185 cmd_with_count("quit", cbuf, sizeof(cbuf), Prenum);
Bram Moolenaarb96a7f32014-11-27 16:22:48 +0100186 do_cmdline_cmd(cbuf);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000187 break;
188
189/* close current window */
190 case Ctrl_C:
191 case 'c':
Bram Moolenaar071d4272004-06-13 20:20:40 +0000192 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar2f1e51a2014-12-13 03:58:09 +0100193 cmd_with_count("close", cbuf, sizeof(cbuf), Prenum);
Bram Moolenaarb96a7f32014-11-27 16:22:48 +0100194 do_cmdline_cmd(cbuf);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000195 break;
196
Bram Moolenaar4033c552017-09-16 20:54:51 +0200197#if defined(FEAT_QUICKFIX)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000198/* close preview window */
199 case Ctrl_Z:
200 case 'z':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200201 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000202 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000203 do_cmdline_cmd((char_u *)"pclose");
204 break;
205
206/* cursor to preview window */
207 case 'P':
Bram Moolenaar29323592016-07-24 22:04:11 +0200208 FOR_ALL_WINDOWS(wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000209 if (wp->w_p_pvw)
210 break;
211 if (wp == NULL)
212 EMSG(_("E441: There is no preview window"));
213 else
214 win_goto(wp);
215 break;
216#endif
217
218/* close all but current window */
219 case Ctrl_O:
220 case 'o':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200221 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000222 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar2f1e51a2014-12-13 03:58:09 +0100223 cmd_with_count("only", cbuf, sizeof(cbuf), Prenum);
Bram Moolenaarb96a7f32014-11-27 16:22:48 +0100224 do_cmdline_cmd(cbuf);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000225 break;
226
227/* cursor to next window with wrap around */
228 case Ctrl_W:
229 case 'w':
230/* cursor to previous window with wrap around */
231 case 'W':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200232 CHECK_CMDWIN;
Bram Moolenaara1f4cb92016-11-06 15:25:42 +0100233 if (ONE_WINDOW && Prenum != 1) /* just one window */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000234 beep_flush();
235 else
236 {
237 if (Prenum) /* go to specified window */
238 {
239 for (wp = firstwin; --Prenum > 0; )
240 {
241 if (wp->w_next == NULL)
242 break;
243 else
244 wp = wp->w_next;
245 }
246 }
247 else
248 {
249 if (nchar == 'W') /* go to previous window */
250 {
251 wp = curwin->w_prev;
252 if (wp == NULL)
253 wp = lastwin; /* wrap around */
254 }
255 else /* go to next window */
256 {
257 wp = curwin->w_next;
258 if (wp == NULL)
259 wp = firstwin; /* wrap around */
260 }
261 }
262 win_goto(wp);
263 }
264 break;
265
266/* cursor to window below */
267 case 'j':
268 case K_DOWN:
269 case Ctrl_J:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200270 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000271 win_goto_ver(FALSE, Prenum1);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000272 break;
273
274/* cursor to window above */
275 case 'k':
276 case K_UP:
277 case Ctrl_K:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200278 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000279 win_goto_ver(TRUE, Prenum1);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000280 break;
281
Bram Moolenaar071d4272004-06-13 20:20:40 +0000282/* cursor to left window */
283 case 'h':
284 case K_LEFT:
285 case Ctrl_H:
286 case K_BS:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200287 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000288 win_goto_hor(TRUE, Prenum1);
289 break;
290
291/* cursor to right window */
292 case 'l':
293 case K_RIGHT:
294 case Ctrl_L:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200295 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000296 win_goto_hor(FALSE, Prenum1);
297 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000298
Bram Moolenaar4c3f5362006-04-11 21:38:50 +0000299/* move window to new tab page */
300 case 'T':
Bram Moolenaar746ebd32009-06-16 14:01:43 +0000301 if (one_window())
Bram Moolenaar4c3f5362006-04-11 21:38:50 +0000302 MSG(_(m_onlyone));
303 else
304 {
305 tabpage_T *oldtab = curtab;
306 tabpage_T *newtab;
Bram Moolenaar4c3f5362006-04-11 21:38:50 +0000307
308 /* First create a new tab with the window, then go back to
309 * the old tab and close the window there. */
Bram Moolenaar89d40322006-08-29 15:30:07 +0000310 wp = curwin;
Bram Moolenaar4c3f5362006-04-11 21:38:50 +0000311 if (win_new_tabpage((int)Prenum) == OK
312 && valid_tabpage(oldtab))
313 {
314 newtab = curtab;
Bram Moolenaar49e649f2013-05-06 04:50:35 +0200315 goto_tabpage_tp(oldtab, TRUE, TRUE);
Bram Moolenaar4c3f5362006-04-11 21:38:50 +0000316 if (curwin == wp)
317 win_close(curwin, FALSE);
318 if (valid_tabpage(newtab))
Bram Moolenaar49e649f2013-05-06 04:50:35 +0200319 goto_tabpage_tp(newtab, TRUE, TRUE);
Bram Moolenaar4c3f5362006-04-11 21:38:50 +0000320 }
321 }
322 break;
323
Bram Moolenaar071d4272004-06-13 20:20:40 +0000324/* cursor to top-left window */
325 case 't':
326 case Ctrl_T:
327 win_goto(firstwin);
328 break;
329
330/* cursor to bottom-right window */
331 case 'b':
332 case Ctrl_B:
333 win_goto(lastwin);
334 break;
335
336/* cursor to last accessed (previous) window */
337 case 'p':
338 case Ctrl_P:
Bram Moolenaar3dda7db2016-04-03 21:22:58 +0200339 if (!win_valid(prevwin))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000340 beep_flush();
341 else
342 win_goto(prevwin);
343 break;
344
345/* exchange current and next window */
346 case 'x':
347 case Ctrl_X:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200348 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000349 win_exchange(Prenum);
350 break;
351
352/* rotate windows downwards */
353 case Ctrl_R:
354 case 'r':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200355 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000356 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000357 win_rotate(FALSE, (int)Prenum1); /* downwards */
358 break;
359
360/* rotate windows upwards */
361 case 'R':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200362 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000363 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000364 win_rotate(TRUE, (int)Prenum1); /* upwards */
365 break;
366
367/* move window to the very top/bottom/left/right */
368 case 'K':
369 case 'J':
Bram Moolenaar071d4272004-06-13 20:20:40 +0000370 case 'H':
371 case 'L':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200372 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000373 win_totop((int)Prenum,
374 ((nchar == 'H' || nchar == 'L') ? WSP_VERT : 0)
375 | ((nchar == 'H' || nchar == 'K') ? WSP_TOP : WSP_BOT));
376 break;
377
378/* make all windows the same height */
379 case '=':
380#ifdef FEAT_GUI
381 need_mouse_correct = TRUE;
382#endif
383 win_equal(NULL, FALSE, 'b');
384 break;
385
386/* increase current window height */
387 case '+':
388#ifdef FEAT_GUI
389 need_mouse_correct = TRUE;
390#endif
391 win_setheight(curwin->w_height + (int)Prenum1);
392 break;
393
394/* decrease current window height */
395 case '-':
396#ifdef FEAT_GUI
397 need_mouse_correct = TRUE;
398#endif
399 win_setheight(curwin->w_height - (int)Prenum1);
400 break;
401
402/* set current window height */
403 case Ctrl__:
404 case '_':
405#ifdef FEAT_GUI
406 need_mouse_correct = TRUE;
407#endif
408 win_setheight(Prenum ? (int)Prenum : 9999);
409 break;
410
Bram Moolenaar071d4272004-06-13 20:20:40 +0000411/* increase current window width */
412 case '>':
413#ifdef FEAT_GUI
414 need_mouse_correct = TRUE;
415#endif
416 win_setwidth(curwin->w_width + (int)Prenum1);
417 break;
418
419/* decrease current window width */
420 case '<':
421#ifdef FEAT_GUI
422 need_mouse_correct = TRUE;
423#endif
424 win_setwidth(curwin->w_width - (int)Prenum1);
425 break;
426
427/* set current window width */
428 case '|':
429#ifdef FEAT_GUI
430 need_mouse_correct = TRUE;
431#endif
432 win_setwidth(Prenum != 0 ? (int)Prenum : 9999);
433 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000434
435/* jump to tag and split window if tag exists (in preview window) */
436#if defined(FEAT_QUICKFIX)
437 case '}':
Bram Moolenaar6f470022018-04-10 18:47:20 +0200438 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000439 if (Prenum)
440 g_do_tagpreview = Prenum;
441 else
442 g_do_tagpreview = p_pvh;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000443#endif
Bram Moolenaar2f40d122017-10-24 21:49:36 +0200444 /* FALLTHROUGH */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000445 case ']':
446 case Ctrl_RSB:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200447 CHECK_CMDWIN;
Bram Moolenaard355c502014-09-23 13:48:43 +0200448 /* keep Visual mode, can select words to use as a tag */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000449 if (Prenum)
450 postponed_split = Prenum;
451 else
452 postponed_split = -1;
Bram Moolenaarda014b92014-09-24 13:26:44 +0200453#ifdef FEAT_QUICKFIX
Bram Moolenaar56095e12014-10-09 10:44:37 +0200454 if (nchar != '}')
455 g_do_tagpreview = 0;
Bram Moolenaarda014b92014-09-24 13:26:44 +0200456#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000457
Bram Moolenaard355c502014-09-23 13:48:43 +0200458 /* Execute the command right here, required when "wincmd ]"
459 * was used in a function. */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000460 do_nv_ident(Ctrl_RSB, NUL);
461 break;
462
463#ifdef FEAT_SEARCHPATH
464/* edit file name under cursor in a new window */
465 case 'f':
Bram Moolenaard1f56e62006-02-22 21:25:37 +0000466 case 'F':
Bram Moolenaar071d4272004-06-13 20:20:40 +0000467 case Ctrl_F:
Bram Moolenaar8dff8182006-04-06 20:18:50 +0000468wingotofile:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200469 CHECK_CMDWIN;
Bram Moolenaard857f0e2005-06-21 22:37:39 +0000470
Bram Moolenaard1f56e62006-02-22 21:25:37 +0000471 ptr = grab_file_name(Prenum1, &lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000472 if (ptr != NULL)
473 {
Bram Moolenaar5d2ca042016-06-26 17:11:21 +0200474 tabpage_T *oldtab = curtab;
475 win_T *oldwin = curwin;
Bram Moolenaar8dff8182006-04-06 20:18:50 +0000476# ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +0000477 need_mouse_correct = TRUE;
Bram Moolenaar8dff8182006-04-06 20:18:50 +0000478# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000479 setpcmark();
480 if (win_split(0, 0) == OK)
481 {
Bram Moolenaar3368ea22010-09-21 16:56:35 +0200482 RESET_BINDING(curwin);
Bram Moolenaar5d2ca042016-06-26 17:11:21 +0200483 if (do_ecmd(0, ptr, NULL, NULL, ECMD_LASTL,
484 ECMD_HIDE, NULL) == FAIL)
485 {
486 /* Failed to open the file, close the window
487 * opened for it. */
488 win_close(curwin, FALSE);
489 goto_tabpage_win(oldtab, oldwin);
490 }
491 else if (nchar == 'F' && lnum >= 0)
Bram Moolenaard1f56e62006-02-22 21:25:37 +0000492 {
493 curwin->w_cursor.lnum = lnum;
494 check_cursor_lnum();
495 beginline(BL_SOL | BL_FIX);
496 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000497 }
498 vim_free(ptr);
499 }
500 break;
501#endif
502
503#ifdef FEAT_FIND_ID
Bram Moolenaarb6799ac2007-05-10 16:44:05 +0000504/* Go to the first occurrence of the identifier under cursor along path in a
Bram Moolenaar071d4272004-06-13 20:20:40 +0000505 * new window -- webb
506 */
507 case 'i': /* Go to any match */
508 case Ctrl_I:
509 type = FIND_ANY;
510 /* FALLTHROUGH */
511 case 'd': /* Go to definition, using 'define' */
512 case Ctrl_D:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200513 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000514 if ((len = find_ident_under_cursor(&ptr, FIND_IDENT)) == 0)
515 break;
516 find_pattern_in_path(ptr, 0, len, TRUE,
517 Prenum == 0 ? TRUE : FALSE, type,
518 Prenum1, ACTION_SPLIT, (linenr_T)1, (linenr_T)MAXLNUM);
519 curwin->w_set_curswant = TRUE;
520 break;
521#endif
522
Bram Moolenaar05159a02005-02-26 23:04:13 +0000523 case K_KENTER:
524 case CAR:
525#if defined(FEAT_QUICKFIX)
526 /*
527 * In a quickfix window a <CR> jumps to the error under the
528 * cursor in a new window.
529 */
530 if (bt_quickfix(curbuf))
531 {
Bram Moolenaar28c258f2006-01-25 22:02:51 +0000532 sprintf((char *)cbuf, "split +%ld%s",
533 (long)curwin->w_cursor.lnum,
534 (curwin->w_llist_ref == NULL) ? "cc" : "ll");
Bram Moolenaar05159a02005-02-26 23:04:13 +0000535 do_cmdline_cmd(cbuf);
536 }
537#endif
538 break;
539
540
Bram Moolenaar071d4272004-06-13 20:20:40 +0000541/* CTRL-W g extended commands */
542 case 'g':
543 case Ctrl_G:
Bram Moolenaar6f470022018-04-10 18:47:20 +0200544 CHECK_CMDWIN;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000545#ifdef USE_ON_FLY_SCROLL
546 dont_scroll = TRUE; /* disallow scrolling here */
547#endif
548 ++no_mapping;
549 ++allow_keys; /* no mapping for xchar, but allow key codes */
550 if (xchar == NUL)
Bram Moolenaar61abfd12007-09-13 16:26:47 +0000551 xchar = plain_vgetc();
Bram Moolenaar071d4272004-06-13 20:20:40 +0000552 LANGMAP_ADJUST(xchar, TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000553 --no_mapping;
554 --allow_keys;
555#ifdef FEAT_CMDL_INFO
556 (void)add_to_showcmd(xchar);
557#endif
558 switch (xchar)
559 {
560#if defined(FEAT_QUICKFIX)
561 case '}':
562 xchar = Ctrl_RSB;
563 if (Prenum)
564 g_do_tagpreview = Prenum;
565 else
566 g_do_tagpreview = p_pvh;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000567#endif
Bram Moolenaar2f40d122017-10-24 21:49:36 +0200568 /* FALLTHROUGH */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000569 case ']':
570 case Ctrl_RSB:
Bram Moolenaard355c502014-09-23 13:48:43 +0200571 /* keep Visual mode, can select words to use as a tag */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000572 if (Prenum)
573 postponed_split = Prenum;
574 else
575 postponed_split = -1;
576
577 /* Execute the command right here, required when
578 * "wincmd g}" was used in a function. */
579 do_nv_ident('g', xchar);
580 break;
581
Bram Moolenaar8dff8182006-04-06 20:18:50 +0000582#ifdef FEAT_SEARCHPATH
583 case 'f': /* CTRL-W gf: "gf" in a new tab page */
Bram Moolenaar57657d82006-04-21 22:12:41 +0000584 case 'F': /* CTRL-W gF: "gF" in a new tab page */
Bram Moolenaar6d1dcff2010-01-27 20:26:46 +0100585 cmdmod.tab = tabpage_index(curtab) + 1;
Bram Moolenaar57657d82006-04-21 22:12:41 +0000586 nchar = xchar;
Bram Moolenaar8dff8182006-04-06 20:18:50 +0000587 goto wingotofile;
588#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000589 default:
590 beep_flush();
591 break;
592 }
593 break;
594
595 default: beep_flush();
596 break;
597 }
598}
599
Bram Moolenaar84c8e5a2015-01-14 15:47:36 +0100600/*
601 * Figure out the address type for ":wnncmd".
602 */
603 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +0100604get_wincmd_addr_type(char_u *arg, exarg_T *eap)
Bram Moolenaar84c8e5a2015-01-14 15:47:36 +0100605{
606 switch (*arg)
607 {
608 case 'S':
609 case Ctrl_S:
610 case 's':
611 case Ctrl_N:
612 case 'n':
613 case 'j':
614 case Ctrl_J:
615 case 'k':
616 case Ctrl_K:
617 case 'T':
618 case Ctrl_R:
619 case 'r':
620 case 'R':
621 case 'K':
622 case 'J':
623 case '+':
624 case '-':
625 case Ctrl__:
626 case '_':
627 case '|':
628 case ']':
629 case Ctrl_RSB:
630 case 'g':
631 case Ctrl_G:
Bram Moolenaar84c8e5a2015-01-14 15:47:36 +0100632 case Ctrl_V:
633 case 'v':
634 case 'h':
635 case Ctrl_H:
636 case 'l':
637 case Ctrl_L:
638 case 'H':
639 case 'L':
640 case '>':
641 case '<':
Bram Moolenaar84c8e5a2015-01-14 15:47:36 +0100642#if defined(FEAT_QUICKFIX)
643 case '}':
644#endif
645#ifdef FEAT_SEARCHPATH
646 case 'f':
647 case 'F':
648 case Ctrl_F:
649#endif
650#ifdef FEAT_FIND_ID
651 case 'i':
652 case Ctrl_I:
653 case 'd':
654 case Ctrl_D:
655#endif
656 /* window size or any count */
657 eap->addr_type = ADDR_LINES;
658 break;
659
660 case Ctrl_HAT:
661 case '^':
662 /* buffer number */
663 eap->addr_type = ADDR_BUFFERS;
664 break;
665
666 case Ctrl_Q:
667 case 'q':
668 case Ctrl_C:
669 case 'c':
670 case Ctrl_O:
671 case 'o':
672 case Ctrl_W:
673 case 'w':
674 case 'W':
675 case 'x':
676 case Ctrl_X:
677 /* window number */
678 eap->addr_type = ADDR_WINDOWS;
679 break;
680
681#if defined(FEAT_QUICKFIX)
682 case Ctrl_Z:
683 case 'z':
684 case 'P':
685#endif
686 case 't':
687 case Ctrl_T:
688 case 'b':
689 case Ctrl_B:
690 case 'p':
691 case Ctrl_P:
692 case '=':
693 case CAR:
694 /* no count */
695 eap->addr_type = 0;
696 break;
697 }
698}
699
Bram Moolenaar2f1e51a2014-12-13 03:58:09 +0100700 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +0100701cmd_with_count(
702 char *cmd,
703 char_u *bufp,
704 size_t bufsize,
705 long Prenum)
Bram Moolenaar2f1e51a2014-12-13 03:58:09 +0100706{
707 size_t len = STRLEN(cmd);
708
709 STRCPY(bufp, cmd);
710 if (Prenum > 0)
711 vim_snprintf((char *)bufp + len, bufsize - len, "%ld", Prenum);
712}
713
Bram Moolenaar071d4272004-06-13 20:20:40 +0000714/*
715 * split the current window, implements CTRL-W s and :split
716 *
717 * "size" is the height or width for the new window, 0 to use half of current
718 * height or width.
719 *
720 * "flags":
721 * WSP_ROOM: require enough room for new window
722 * WSP_VERT: vertical split.
723 * WSP_TOP: open window at the top-left of the shell (help window).
724 * WSP_BOT: open window at the bottom-right of the shell (quickfix window).
725 * WSP_HELP: creating the help window, keep layout snapshot
726 *
727 * return FAIL for failure, OK otherwise
728 */
729 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +0100730win_split(int size, int flags)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000731{
Bram Moolenaar80a94a52006-02-23 21:26:58 +0000732 /* When the ":tab" modifier was used open a new tab page instead. */
733 if (may_open_tabpage() == OK)
734 return OK;
735
Bram Moolenaar071d4272004-06-13 20:20:40 +0000736 /* Add flags from ":vertical", ":topleft" and ":botright". */
737 flags |= cmdmod.split;
738 if ((flags & WSP_TOP) && (flags & WSP_BOT))
739 {
740 EMSG(_("E442: Can't split topleft and botright at the same time"));
741 return FAIL;
742 }
743
744 /* When creating the help window make a snapshot of the window layout.
745 * Otherwise clear the snapshot, it's now invalid. */
746 if (flags & WSP_HELP)
Bram Moolenaar746ebd32009-06-16 14:01:43 +0000747 make_snapshot(SNAP_HELP_IDX);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000748 else
Bram Moolenaar746ebd32009-06-16 14:01:43 +0000749 clear_snapshot(curtab, SNAP_HELP_IDX);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000750
751 return win_split_ins(size, flags, NULL, 0);
752}
753
754/*
Bram Moolenaar70b2a562012-01-10 22:26:17 +0100755 * When "new_wp" is NULL: split the current window in two.
756 * When "new_wp" is not NULL: insert this window at the far
Bram Moolenaar071d4272004-06-13 20:20:40 +0000757 * top/left/right/bottom.
758 * return FAIL for failure, OK otherwise
759 */
Bram Moolenaar746ebd32009-06-16 14:01:43 +0000760 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +0100761win_split_ins(
762 int size,
763 int flags,
764 win_T *new_wp,
765 int dir)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000766{
Bram Moolenaar70b2a562012-01-10 22:26:17 +0100767 win_T *wp = new_wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000768 win_T *oldwin;
769 int new_size = size;
770 int i;
771 int need_status = 0;
772 int do_equal = FALSE;
773 int needed;
774 int available;
775 int oldwin_height = 0;
776 int layout;
Bram Moolenaar54368f22014-07-23 15:21:20 +0200777 frame_T *frp, *curfrp, *frp2, *prevfrp;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000778 int before;
Bram Moolenaarb4d21352014-07-16 14:16:46 +0200779 int minheight;
Bram Moolenaar1f538352014-07-16 18:19:27 +0200780 int wmh1;
Bram Moolenaar98da6ec2018-04-13 22:15:46 +0200781 int did_set_fraction = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000782
783 if (flags & WSP_TOP)
784 oldwin = firstwin;
785 else if (flags & WSP_BOT)
786 oldwin = lastwin;
787 else
788 oldwin = curwin;
789
790 /* add a status line when p_ls == 1 and splitting the first window */
Bram Moolenaar459ca562016-11-10 18:16:33 +0100791 if (ONE_WINDOW && p_ls == 1 && oldwin->w_status_height == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000792 {
Bram Moolenaar415a6932017-12-05 20:31:07 +0100793 if (VISIBLE_HEIGHT(oldwin) <= p_wmh && new_wp == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000794 {
795 EMSG(_(e_noroom));
796 return FAIL;
797 }
798 need_status = STATUS_HEIGHT;
799 }
800
Bram Moolenaaree79cbc2007-05-02 19:50:14 +0000801#ifdef FEAT_GUI
802 /* May be needed for the scrollbars that are going to change. */
803 if (gui.in_use)
804 out_flush();
805#endif
806
Bram Moolenaar071d4272004-06-13 20:20:40 +0000807 if (flags & WSP_VERT)
808 {
Bram Moolenaara0485492014-07-16 23:39:54 +0200809 int wmw1;
810 int minwidth;
811
Bram Moolenaar071d4272004-06-13 20:20:40 +0000812 layout = FR_ROW;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000813
814 /*
815 * Check if we are able to split the current window and compute its
816 * width.
817 */
Bram Moolenaar1f538352014-07-16 18:19:27 +0200818 /* Current window requires at least 1 space. */
819 wmw1 = (p_wmw == 0 ? 1 : p_wmw);
820 needed = wmw1 + 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000821 if (flags & WSP_ROOM)
Bram Moolenaar1f538352014-07-16 18:19:27 +0200822 needed += p_wiw - wmw1;
Bram Moolenaar54368f22014-07-23 15:21:20 +0200823 if (flags & (WSP_BOT | WSP_TOP))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000824 {
Bram Moolenaar1f538352014-07-16 18:19:27 +0200825 minwidth = frame_minwidth(topframe, NOWIN);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000826 available = topframe->fr_width;
Bram Moolenaarb4d21352014-07-16 14:16:46 +0200827 needed += minwidth;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000828 }
Bram Moolenaar54368f22014-07-23 15:21:20 +0200829 else if (p_ea)
830 {
831 minwidth = frame_minwidth(oldwin->w_frame, NOWIN);
832 prevfrp = oldwin->w_frame;
833 for (frp = oldwin->w_frame->fr_parent; frp != NULL;
834 frp = frp->fr_parent)
835 {
836 if (frp->fr_layout == FR_ROW)
837 for (frp2 = frp->fr_child; frp2 != NULL;
838 frp2 = frp2->fr_next)
839 if (frp2 != prevfrp)
840 minwidth += frame_minwidth(frp2, NOWIN);
841 prevfrp = frp;
842 }
843 available = topframe->fr_width;
844 needed += minwidth;
845 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000846 else
Bram Moolenaarb4d21352014-07-16 14:16:46 +0200847 {
Bram Moolenaar1f538352014-07-16 18:19:27 +0200848 minwidth = frame_minwidth(oldwin->w_frame, NOWIN);
849 available = oldwin->w_frame->fr_width;
850 needed += minwidth;
Bram Moolenaarb4d21352014-07-16 14:16:46 +0200851 }
Bram Moolenaar70b2a562012-01-10 22:26:17 +0100852 if (available < needed && new_wp == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000853 {
854 EMSG(_(e_noroom));
855 return FAIL;
856 }
857 if (new_size == 0)
858 new_size = oldwin->w_width / 2;
Bram Moolenaarb4d21352014-07-16 14:16:46 +0200859 if (new_size > available - minwidth - 1)
860 new_size = available - minwidth - 1;
Bram Moolenaar1f538352014-07-16 18:19:27 +0200861 if (new_size < wmw1)
862 new_size = wmw1;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000863
864 /* if it doesn't fit in the current window, need win_equal() */
865 if (oldwin->w_width - new_size - 1 < p_wmw)
866 do_equal = TRUE;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +0000867
868 /* We don't like to take lines for the new window from a
869 * 'winfixwidth' window. Take them from a window to the left or right
Bram Moolenaar38e34832017-03-19 20:22:36 +0100870 * instead, if possible. Add one for the separator. */
Bram Moolenaarbe4d5062006-03-18 21:30:13 +0000871 if (oldwin->w_p_wfw)
Bram Moolenaar38e34832017-03-19 20:22:36 +0100872 win_setwidth_win(oldwin->w_width + new_size + 1, oldwin);
Bram Moolenaar67f71312007-08-12 14:55:56 +0000873
874 /* Only make all windows the same width if one of them (except oldwin)
875 * is wider than one of the split windows. */
876 if (!do_equal && p_ea && size == 0 && *p_ead != 'v'
877 && oldwin->w_frame->fr_parent != NULL)
878 {
879 frp = oldwin->w_frame->fr_parent->fr_child;
880 while (frp != NULL)
881 {
882 if (frp->fr_win != oldwin && frp->fr_win != NULL
883 && (frp->fr_win->w_width > new_size
884 || frp->fr_win->w_width > oldwin->w_width
Bram Moolenaardf46f6f2014-11-19 13:40:08 +0100885 - new_size - 1))
Bram Moolenaar67f71312007-08-12 14:55:56 +0000886 {
887 do_equal = TRUE;
888 break;
889 }
890 frp = frp->fr_next;
891 }
892 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000893 }
894 else
Bram Moolenaar071d4272004-06-13 20:20:40 +0000895 {
896 layout = FR_COL;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000897
898 /*
899 * Check if we are able to split the current window and compute its
900 * height.
901 */
Bram Moolenaar1f538352014-07-16 18:19:27 +0200902 /* Current window requires at least 1 space. */
Bram Moolenaar415a6932017-12-05 20:31:07 +0100903 wmh1 = (p_wmh == 0 ? 1 : p_wmh) + WINBAR_HEIGHT(curwin);
Bram Moolenaar1f538352014-07-16 18:19:27 +0200904 needed = wmh1 + STATUS_HEIGHT;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000905 if (flags & WSP_ROOM)
Bram Moolenaar1f538352014-07-16 18:19:27 +0200906 needed += p_wh - wmh1;
Bram Moolenaar54368f22014-07-23 15:21:20 +0200907 if (flags & (WSP_BOT | WSP_TOP))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000908 {
Bram Moolenaar1f538352014-07-16 18:19:27 +0200909 minheight = frame_minheight(topframe, NOWIN) + need_status;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000910 available = topframe->fr_height;
Bram Moolenaarb4d21352014-07-16 14:16:46 +0200911 needed += minheight;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000912 }
Bram Moolenaar54368f22014-07-23 15:21:20 +0200913 else if (p_ea)
914 {
915 minheight = frame_minheight(oldwin->w_frame, NOWIN) + need_status;
916 prevfrp = oldwin->w_frame;
917 for (frp = oldwin->w_frame->fr_parent; frp != NULL;
918 frp = frp->fr_parent)
919 {
920 if (frp->fr_layout == FR_COL)
921 for (frp2 = frp->fr_child; frp2 != NULL;
922 frp2 = frp2->fr_next)
923 if (frp2 != prevfrp)
924 minheight += frame_minheight(frp2, NOWIN);
925 prevfrp = frp;
926 }
927 available = topframe->fr_height;
928 needed += minheight;
929 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000930 else
931 {
Bram Moolenaar1f538352014-07-16 18:19:27 +0200932 minheight = frame_minheight(oldwin->w_frame, NOWIN) + need_status;
933 available = oldwin->w_frame->fr_height;
934 needed += minheight;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000935 }
Bram Moolenaar70b2a562012-01-10 22:26:17 +0100936 if (available < needed && new_wp == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000937 {
938 EMSG(_(e_noroom));
939 return FAIL;
940 }
941 oldwin_height = oldwin->w_height;
942 if (need_status)
943 {
944 oldwin->w_status_height = STATUS_HEIGHT;
945 oldwin_height -= STATUS_HEIGHT;
946 }
947 if (new_size == 0)
948 new_size = oldwin_height / 2;
Bram Moolenaarb4d21352014-07-16 14:16:46 +0200949 if (new_size > available - minheight - STATUS_HEIGHT)
950 new_size = available - minheight - STATUS_HEIGHT;
Bram Moolenaar1f538352014-07-16 18:19:27 +0200951 if (new_size < wmh1)
952 new_size = wmh1;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000953
954 /* if it doesn't fit in the current window, need win_equal() */
955 if (oldwin_height - new_size - STATUS_HEIGHT < p_wmh)
956 do_equal = TRUE;
957
958 /* We don't like to take lines for the new window from a
959 * 'winfixheight' window. Take them from a window above or below
960 * instead, if possible. */
961 if (oldwin->w_p_wfh)
962 {
Bram Moolenaar98da6ec2018-04-13 22:15:46 +0200963 /* Set w_fraction now so that the cursor keeps the same relative
964 * vertical position using the old height. */
965 set_fraction(oldwin);
966 did_set_fraction = TRUE;
967
Bram Moolenaar071d4272004-06-13 20:20:40 +0000968 win_setheight_win(oldwin->w_height + new_size + STATUS_HEIGHT,
969 oldwin);
970 oldwin_height = oldwin->w_height;
971 if (need_status)
972 oldwin_height -= STATUS_HEIGHT;
973 }
Bram Moolenaar67f71312007-08-12 14:55:56 +0000974
975 /* Only make all windows the same height if one of them (except oldwin)
976 * is higher than one of the split windows. */
Bram Moolenaar44a2f922016-03-19 22:11:51 +0100977 if (!do_equal && p_ea && size == 0 && *p_ead != 'h'
Bram Moolenaar67f71312007-08-12 14:55:56 +0000978 && oldwin->w_frame->fr_parent != NULL)
979 {
980 frp = oldwin->w_frame->fr_parent->fr_child;
981 while (frp != NULL)
982 {
983 if (frp->fr_win != oldwin && frp->fr_win != NULL
984 && (frp->fr_win->w_height > new_size
985 || frp->fr_win->w_height > oldwin_height - new_size
986 - STATUS_HEIGHT))
987 {
988 do_equal = TRUE;
989 break;
990 }
991 frp = frp->fr_next;
992 }
993 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000994 }
995
996 /*
997 * allocate new window structure and link it in the window list
998 */
999 if ((flags & WSP_TOP) == 0
1000 && ((flags & WSP_BOT)
1001 || (flags & WSP_BELOW)
1002 || (!(flags & WSP_ABOVE)
Bram Moolenaar44a2f922016-03-19 22:11:51 +01001003 && ( (flags & WSP_VERT) ? p_spr : p_sb))))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001004 {
1005 /* new window below/right of current one */
Bram Moolenaar70b2a562012-01-10 22:26:17 +01001006 if (new_wp == NULL)
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001007 wp = win_alloc(oldwin, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001008 else
1009 win_append(oldwin, wp);
1010 }
1011 else
1012 {
Bram Moolenaar70b2a562012-01-10 22:26:17 +01001013 if (new_wp == NULL)
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001014 wp = win_alloc(oldwin->w_prev, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001015 else
1016 win_append(oldwin->w_prev, wp);
1017 }
1018
Bram Moolenaar70b2a562012-01-10 22:26:17 +01001019 if (new_wp == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001020 {
1021 if (wp == NULL)
1022 return FAIL;
1023
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001024 new_frame(wp);
1025 if (wp->w_frame == NULL)
1026 {
1027 win_free(wp, NULL);
1028 return FAIL;
1029 }
1030
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001031 /* make the contents of the new window the same as the current one */
Bram Moolenaar884ae642009-02-22 01:37:59 +00001032 win_init(wp, curwin, flags);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001033 }
1034
1035 /*
1036 * Reorganise the tree of frames to insert the new window.
1037 */
1038 if (flags & (WSP_TOP | WSP_BOT))
1039 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001040 if ((topframe->fr_layout == FR_COL && (flags & WSP_VERT) == 0)
1041 || (topframe->fr_layout == FR_ROW && (flags & WSP_VERT) != 0))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001042 {
1043 curfrp = topframe->fr_child;
1044 if (flags & WSP_BOT)
1045 while (curfrp->fr_next != NULL)
1046 curfrp = curfrp->fr_next;
1047 }
1048 else
1049 curfrp = topframe;
1050 before = (flags & WSP_TOP);
1051 }
1052 else
1053 {
1054 curfrp = oldwin->w_frame;
1055 if (flags & WSP_BELOW)
1056 before = FALSE;
1057 else if (flags & WSP_ABOVE)
1058 before = TRUE;
Bram Moolenaar44a2f922016-03-19 22:11:51 +01001059 else if (flags & WSP_VERT)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001060 before = !p_spr;
1061 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001062 before = !p_sb;
1063 }
1064 if (curfrp->fr_parent == NULL || curfrp->fr_parent->fr_layout != layout)
1065 {
1066 /* Need to create a new frame in the tree to make a branch. */
1067 frp = (frame_T *)alloc_clear((unsigned)sizeof(frame_T));
1068 *frp = *curfrp;
1069 curfrp->fr_layout = layout;
1070 frp->fr_parent = curfrp;
1071 frp->fr_next = NULL;
1072 frp->fr_prev = NULL;
1073 curfrp->fr_child = frp;
1074 curfrp->fr_win = NULL;
1075 curfrp = frp;
1076 if (frp->fr_win != NULL)
1077 oldwin->w_frame = frp;
1078 else
1079 for (frp = frp->fr_child; frp != NULL; frp = frp->fr_next)
1080 frp->fr_parent = curfrp;
1081 }
1082
Bram Moolenaar70b2a562012-01-10 22:26:17 +01001083 if (new_wp == NULL)
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001084 frp = wp->w_frame;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001085 else
Bram Moolenaar70b2a562012-01-10 22:26:17 +01001086 frp = new_wp->w_frame;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001087 frp->fr_parent = curfrp->fr_parent;
1088
1089 /* Insert the new frame at the right place in the frame list. */
1090 if (before)
1091 frame_insert(curfrp, frp);
1092 else
1093 frame_append(curfrp, frp);
1094
Bram Moolenaar0215e8e2010-12-17 17:35:10 +01001095 /* Set w_fraction now so that the cursor keeps the same relative
1096 * vertical position. */
Bram Moolenaar98da6ec2018-04-13 22:15:46 +02001097 if (!did_set_fraction)
1098 set_fraction(oldwin);
Bram Moolenaar0215e8e2010-12-17 17:35:10 +01001099 wp->w_fraction = oldwin->w_fraction;
1100
Bram Moolenaar071d4272004-06-13 20:20:40 +00001101 if (flags & WSP_VERT)
1102 {
1103 wp->w_p_scr = curwin->w_p_scr;
Bram Moolenaar0215e8e2010-12-17 17:35:10 +01001104
Bram Moolenaar071d4272004-06-13 20:20:40 +00001105 if (need_status)
1106 {
Bram Moolenaar9b73a782010-03-17 16:54:57 +01001107 win_new_height(oldwin, oldwin->w_height - 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001108 oldwin->w_status_height = need_status;
1109 }
1110 if (flags & (WSP_TOP | WSP_BOT))
1111 {
1112 /* set height and row of new window to full height */
Bram Moolenaar32466aa2006-02-24 23:53:04 +00001113 wp->w_winrow = tabline_height();
Bram Moolenaard326ad62017-09-18 20:31:41 +02001114 win_new_height(wp, curfrp->fr_height - (p_ls > 0)
Bram Moolenaar3167c3e2017-11-25 14:19:43 +01001115 - WINBAR_HEIGHT(wp));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001116 wp->w_status_height = (p_ls > 0);
1117 }
1118 else
1119 {
1120 /* height and row of new window is same as current window */
1121 wp->w_winrow = oldwin->w_winrow;
Bram Moolenaar415a6932017-12-05 20:31:07 +01001122 win_new_height(wp, VISIBLE_HEIGHT(oldwin));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001123 wp->w_status_height = oldwin->w_status_height;
1124 }
1125 frp->fr_height = curfrp->fr_height;
1126
1127 /* "new_size" of the current window goes to the new window, use
1128 * one column for the vertical separator */
Bram Moolenaar9b73a782010-03-17 16:54:57 +01001129 win_new_width(wp, new_size);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001130 if (before)
1131 wp->w_vsep_width = 1;
1132 else
1133 {
1134 wp->w_vsep_width = oldwin->w_vsep_width;
1135 oldwin->w_vsep_width = 1;
1136 }
1137 if (flags & (WSP_TOP | WSP_BOT))
1138 {
1139 if (flags & WSP_BOT)
1140 frame_add_vsep(curfrp);
1141 /* Set width of neighbor frame */
1142 frame_new_width(curfrp, curfrp->fr_width
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001143 - (new_size + ((flags & WSP_TOP) != 0)), flags & WSP_TOP,
1144 FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001145 }
1146 else
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001147 win_new_width(oldwin, oldwin->w_width - (new_size + 1));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001148 if (before) /* new window left of current one */
1149 {
1150 wp->w_wincol = oldwin->w_wincol;
1151 oldwin->w_wincol += new_size + 1;
1152 }
1153 else /* new window right of current one */
1154 wp->w_wincol = oldwin->w_wincol + oldwin->w_width + 1;
1155 frame_fix_width(oldwin);
1156 frame_fix_width(wp);
1157 }
1158 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001159 {
1160 /* width and column of new window is same as current window */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001161 if (flags & (WSP_TOP | WSP_BOT))
1162 {
1163 wp->w_wincol = 0;
Bram Moolenaar9b73a782010-03-17 16:54:57 +01001164 win_new_width(wp, Columns);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001165 wp->w_vsep_width = 0;
1166 }
1167 else
1168 {
1169 wp->w_wincol = oldwin->w_wincol;
Bram Moolenaar9b73a782010-03-17 16:54:57 +01001170 win_new_width(wp, oldwin->w_width);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001171 wp->w_vsep_width = oldwin->w_vsep_width;
1172 }
1173 frp->fr_width = curfrp->fr_width;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001174
1175 /* "new_size" of the current window goes to the new window, use
1176 * one row for the status line */
1177 win_new_height(wp, new_size);
1178 if (flags & (WSP_TOP | WSP_BOT))
Bram Moolenaar991dea32016-05-24 11:31:32 +02001179 {
Bram Moolenaard326ad62017-09-18 20:31:41 +02001180 int new_fr_height = curfrp->fr_height - new_size
Bram Moolenaar3167c3e2017-11-25 14:19:43 +01001181 + WINBAR_HEIGHT(wp) ;
Bram Moolenaar991dea32016-05-24 11:31:32 +02001182
1183 if (!((flags & WSP_BOT) && p_ls == 0))
1184 new_fr_height -= STATUS_HEIGHT;
1185 frame_new_height(curfrp, new_fr_height, flags & WSP_TOP, FALSE);
1186 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001187 else
1188 win_new_height(oldwin, oldwin_height - (new_size + STATUS_HEIGHT));
1189 if (before) /* new window above current one */
1190 {
1191 wp->w_winrow = oldwin->w_winrow;
1192 wp->w_status_height = STATUS_HEIGHT;
1193 oldwin->w_winrow += wp->w_height + STATUS_HEIGHT;
1194 }
1195 else /* new window below current one */
1196 {
Bram Moolenaar415a6932017-12-05 20:31:07 +01001197 wp->w_winrow = oldwin->w_winrow + VISIBLE_HEIGHT(oldwin)
1198 + STATUS_HEIGHT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001199 wp->w_status_height = oldwin->w_status_height;
Bram Moolenaar991dea32016-05-24 11:31:32 +02001200 if (!(flags & WSP_BOT))
1201 oldwin->w_status_height = STATUS_HEIGHT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001202 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001203 if (flags & WSP_BOT)
1204 frame_add_statusline(curfrp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001205 frame_fix_height(wp);
1206 frame_fix_height(oldwin);
1207 }
1208
1209 if (flags & (WSP_TOP | WSP_BOT))
1210 (void)win_comp_pos();
1211
1212 /*
1213 * Both windows need redrawing
1214 */
1215 redraw_win_later(wp, NOT_VALID);
1216 wp->w_redr_status = TRUE;
1217 redraw_win_later(oldwin, NOT_VALID);
1218 oldwin->w_redr_status = TRUE;
1219
1220 if (need_status)
1221 {
1222 msg_row = Rows - 1;
1223 msg_col = sc_col;
1224 msg_clr_eos_force(); /* Old command/ruler may still be there */
1225 comp_col();
1226 msg_row = Rows - 1;
1227 msg_col = 0; /* put position back at start of line */
1228 }
1229
1230 /*
Bram Moolenaar9b73a782010-03-17 16:54:57 +01001231 * equalize the window sizes.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001232 */
1233 if (do_equal || dir != 0)
1234 win_equal(wp, TRUE,
Bram Moolenaar071d4272004-06-13 20:20:40 +00001235 (flags & WSP_VERT) ? (dir == 'v' ? 'b' : 'h')
Bram Moolenaar44a2f922016-03-19 22:11:51 +01001236 : dir == 'h' ? 'b' : 'v');
Bram Moolenaar071d4272004-06-13 20:20:40 +00001237
1238 /* Don't change the window height/width to 'winheight' / 'winwidth' if a
1239 * size was given. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001240 if (flags & WSP_VERT)
1241 {
1242 i = p_wiw;
1243 if (size != 0)
1244 p_wiw = size;
1245
1246# ifdef FEAT_GUI
1247 /* When 'guioptions' includes 'L' or 'R' may have to add scrollbars. */
1248 if (gui.in_use)
1249 gui_init_which_components(NULL);
1250# endif
1251 }
1252 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001253 {
1254 i = p_wh;
1255 if (size != 0)
1256 p_wh = size;
1257 }
Bram Moolenaar9b73a782010-03-17 16:54:57 +01001258
Bram Moolenaar23fb7a92014-07-30 14:05:00 +02001259#ifdef FEAT_JUMPLIST
1260 /* Keep same changelist position in new window. */
1261 wp->w_changelistidx = oldwin->w_changelistidx;
1262#endif
1263
Bram Moolenaar9b73a782010-03-17 16:54:57 +01001264 /*
1265 * make the new window the current window
1266 */
Bram Moolenaarc917da42016-07-19 22:31:36 +02001267 win_enter_ext(wp, FALSE, FALSE, TRUE, TRUE, TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001268 if (flags & WSP_VERT)
1269 p_wiw = i;
1270 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001271 p_wh = i;
1272
1273 return OK;
1274}
1275
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001276
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001277/*
1278 * Initialize window "newp" from window "oldp".
1279 * Used when splitting a window and when creating a new tab page.
1280 * The windows will both edit the same buffer.
Bram Moolenaar884ae642009-02-22 01:37:59 +00001281 * WSP_NEWLOC may be specified in flags to prevent the location list from
1282 * being copied.
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001283 */
1284 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001285win_init(win_T *newp, win_T *oldp, int flags UNUSED)
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001286{
1287 int i;
1288
1289 newp->w_buffer = oldp->w_buffer;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001290#ifdef FEAT_SYN_HL
Bram Moolenaarfd29f462010-06-06 16:11:09 +02001291 newp->w_s = &(oldp->w_buffer->b_s);
Bram Moolenaar860cae12010-06-05 23:22:07 +02001292#endif
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001293 oldp->w_buffer->b_nwindows++;
1294 newp->w_cursor = oldp->w_cursor;
1295 newp->w_valid = 0;
1296 newp->w_curswant = oldp->w_curswant;
1297 newp->w_set_curswant = oldp->w_set_curswant;
1298 newp->w_topline = oldp->w_topline;
1299#ifdef FEAT_DIFF
1300 newp->w_topfill = oldp->w_topfill;
1301#endif
1302 newp->w_leftcol = oldp->w_leftcol;
1303 newp->w_pcmark = oldp->w_pcmark;
1304 newp->w_prev_pcmark = oldp->w_prev_pcmark;
1305 newp->w_alt_fnum = oldp->w_alt_fnum;
Bram Moolenaar4c3f5362006-04-11 21:38:50 +00001306 newp->w_wrow = oldp->w_wrow;
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001307 newp->w_fraction = oldp->w_fraction;
1308 newp->w_prev_fraction_row = oldp->w_prev_fraction_row;
1309#ifdef FEAT_JUMPLIST
1310 copy_jumplist(oldp, newp);
1311#endif
1312#ifdef FEAT_QUICKFIX
Bram Moolenaar884ae642009-02-22 01:37:59 +00001313 if (flags & WSP_NEWLOC)
1314 {
1315 /* Don't copy the location list. */
1316 newp->w_llist = NULL;
1317 newp->w_llist_ref = NULL;
1318 }
1319 else
1320 copy_loclist(oldp, newp);
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001321#endif
Bram Moolenaarbd2dc342014-01-10 15:53:13 +01001322 newp->w_localdir = (oldp->w_localdir == NULL)
1323 ? NULL : vim_strsave(oldp->w_localdir);
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001324
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001325 /* copy tagstack and folds */
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001326 for (i = 0; i < oldp->w_tagstacklen; i++)
1327 {
1328 newp->w_tagstack[i] = oldp->w_tagstack[i];
1329 if (newp->w_tagstack[i].tagname != NULL)
1330 newp->w_tagstack[i].tagname =
1331 vim_strsave(newp->w_tagstack[i].tagname);
1332 }
1333 newp->w_tagstackidx = oldp->w_tagstackidx;
1334 newp->w_tagstacklen = oldp->w_tagstacklen;
Bram Moolenaara971b822011-09-14 14:43:25 +02001335#ifdef FEAT_FOLDING
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001336 copyFoldingState(oldp, newp);
Bram Moolenaara971b822011-09-14 14:43:25 +02001337#endif
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001338
1339 win_init_some(newp, oldp);
Bram Moolenaar1a384422010-07-14 19:53:30 +02001340
Bram Moolenaara971b822011-09-14 14:43:25 +02001341#ifdef FEAT_SYN_HL
Bram Moolenaar1a384422010-07-14 19:53:30 +02001342 check_colorcolumn(newp);
Bram Moolenaara971b822011-09-14 14:43:25 +02001343#endif
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001344}
1345
1346/*
Bram Moolenaar5d2bae82014-09-19 14:26:36 +02001347 * Initialize window "newp" from window "old".
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001348 * Only the essential things are copied.
1349 */
1350 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001351win_init_some(win_T *newp, win_T *oldp)
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001352{
1353 /* Use the same argument list. */
1354 newp->w_alist = oldp->w_alist;
1355 ++newp->w_alist->al_refcount;
1356 newp->w_arg_idx = oldp->w_arg_idx;
1357
1358 /* copy options from existing window */
1359 win_copy_options(oldp, newp);
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00001360}
1361
Bram Moolenaar071d4272004-06-13 20:20:40 +00001362
Bram Moolenaar071d4272004-06-13 20:20:40 +00001363/*
Bram Moolenaare59215c2016-08-14 19:08:45 +02001364 * Check if "win" is a pointer to an existing window in the current tab page.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001365 */
1366 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001367win_valid(win_T *win)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001368{
1369 win_T *wp;
1370
1371 if (win == NULL)
1372 return FALSE;
Bram Moolenaar29323592016-07-24 22:04:11 +02001373 FOR_ALL_WINDOWS(wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001374 if (wp == win)
1375 return TRUE;
1376 return FALSE;
1377}
1378
1379/*
Bram Moolenaare59215c2016-08-14 19:08:45 +02001380 * Check if "win" is a pointer to an existing window in any tab page.
1381 */
1382 int
1383win_valid_any_tab(win_T *win)
1384{
1385 win_T *wp;
1386 tabpage_T *tp;
1387
1388 if (win == NULL)
1389 return FALSE;
1390 FOR_ALL_TABPAGES(tp)
1391 {
1392 FOR_ALL_WINDOWS_IN_TAB(tp, wp)
1393 {
1394 if (wp == win)
1395 return TRUE;
1396 }
1397 }
1398 return FALSE;
1399}
1400
1401/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00001402 * Return the number of windows.
1403 */
1404 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001405win_count(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001406{
1407 win_T *wp;
1408 int count = 0;
1409
Bram Moolenaar29323592016-07-24 22:04:11 +02001410 FOR_ALL_WINDOWS(wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001411 ++count;
1412 return count;
1413}
1414
1415/*
1416 * Make "count" windows on the screen.
1417 * Return actual number of windows on the screen.
1418 * Must be called when there is just one window, filling the whole screen
1419 * (excluding the command line).
1420 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001421 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001422make_windows(
1423 int count,
1424 int vertical UNUSED) /* split windows vertically if TRUE */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001425{
1426 int maxcount;
1427 int todo;
1428
Bram Moolenaar071d4272004-06-13 20:20:40 +00001429 if (vertical)
1430 {
1431 /* Each windows needs at least 'winminwidth' lines and a separator
1432 * column. */
1433 maxcount = (curwin->w_width + curwin->w_vsep_width
1434 - (p_wiw - p_wmw)) / (p_wmw + 1);
1435 }
1436 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001437 {
1438 /* Each window needs at least 'winminheight' lines and a status line. */
Bram Moolenaar415a6932017-12-05 20:31:07 +01001439 maxcount = (VISIBLE_HEIGHT(curwin) + curwin->w_status_height
Bram Moolenaar071d4272004-06-13 20:20:40 +00001440 - (p_wh - p_wmh)) / (p_wmh + STATUS_HEIGHT);
1441 }
1442
1443 if (maxcount < 2)
1444 maxcount = 2;
1445 if (count > maxcount)
1446 count = maxcount;
1447
1448 /*
1449 * add status line now, otherwise first window will be too big
1450 */
1451 if (count > 1)
1452 last_status(TRUE);
1453
Bram Moolenaar071d4272004-06-13 20:20:40 +00001454 /*
1455 * Don't execute autocommands while creating the windows. Must do that
1456 * when putting the buffers in the windows.
1457 */
Bram Moolenaar78ab3312007-09-29 12:16:41 +00001458 block_autocmds();
Bram Moolenaar071d4272004-06-13 20:20:40 +00001459
1460 /* todo is number of windows left to create */
1461 for (todo = count - 1; todo > 0; --todo)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001462 if (vertical)
1463 {
1464 if (win_split(curwin->w_width - (curwin->w_width - todo)
1465 / (todo + 1) - 1, WSP_VERT | WSP_ABOVE) == FAIL)
1466 break;
1467 }
1468 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001469 {
1470 if (win_split(curwin->w_height - (curwin->w_height - todo
1471 * STATUS_HEIGHT) / (todo + 1)
1472 - STATUS_HEIGHT, WSP_ABOVE) == FAIL)
1473 break;
1474 }
1475
Bram Moolenaar78ab3312007-09-29 12:16:41 +00001476 unblock_autocmds();
Bram Moolenaar071d4272004-06-13 20:20:40 +00001477
1478 /* return actual number of windows */
1479 return (count - todo);
1480}
1481
1482/*
1483 * Exchange current and next window
1484 */
1485 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001486win_exchange(long Prenum)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001487{
1488 frame_T *frp;
1489 frame_T *frp2;
1490 win_T *wp;
1491 win_T *wp2;
1492 int temp;
1493
Bram Moolenaar459ca562016-11-10 18:16:33 +01001494 if (ONE_WINDOW) /* just one window */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001495 {
1496 beep_flush();
1497 return;
1498 }
1499
1500#ifdef FEAT_GUI
1501 need_mouse_correct = TRUE;
1502#endif
1503
1504 /*
1505 * find window to exchange with
1506 */
1507 if (Prenum)
1508 {
1509 frp = curwin->w_frame->fr_parent->fr_child;
1510 while (frp != NULL && --Prenum > 0)
1511 frp = frp->fr_next;
1512 }
1513 else if (curwin->w_frame->fr_next != NULL) /* Swap with next */
1514 frp = curwin->w_frame->fr_next;
1515 else /* Swap last window in row/col with previous */
1516 frp = curwin->w_frame->fr_prev;
1517
1518 /* We can only exchange a window with another window, not with a frame
1519 * containing windows. */
1520 if (frp == NULL || frp->fr_win == NULL || frp->fr_win == curwin)
1521 return;
1522 wp = frp->fr_win;
1523
1524/*
1525 * 1. remove curwin from the list. Remember after which window it was in wp2
1526 * 2. insert curwin before wp in the list
1527 * if wp != wp2
1528 * 3. remove wp from the list
1529 * 4. insert wp after wp2
1530 * 5. exchange the status line height and vsep width.
1531 */
1532 wp2 = curwin->w_prev;
1533 frp2 = curwin->w_frame->fr_prev;
1534 if (wp->w_prev != curwin)
1535 {
Bram Moolenaarf740b292006-02-16 22:11:02 +00001536 win_remove(curwin, NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001537 frame_remove(curwin->w_frame);
1538 win_append(wp->w_prev, curwin);
1539 frame_insert(frp, curwin->w_frame);
1540 }
1541 if (wp != wp2)
1542 {
Bram Moolenaarf740b292006-02-16 22:11:02 +00001543 win_remove(wp, NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001544 frame_remove(wp->w_frame);
1545 win_append(wp2, wp);
1546 if (frp2 == NULL)
1547 frame_insert(wp->w_frame->fr_parent->fr_child, wp->w_frame);
1548 else
1549 frame_append(frp2, wp->w_frame);
1550 }
1551 temp = curwin->w_status_height;
1552 curwin->w_status_height = wp->w_status_height;
1553 wp->w_status_height = temp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001554 temp = curwin->w_vsep_width;
1555 curwin->w_vsep_width = wp->w_vsep_width;
1556 wp->w_vsep_width = temp;
1557
1558 /* If the windows are not in the same frame, exchange the sizes to avoid
1559 * messing up the window layout. Otherwise fix the frame sizes. */
1560 if (curwin->w_frame->fr_parent != wp->w_frame->fr_parent)
1561 {
1562 temp = curwin->w_height;
1563 curwin->w_height = wp->w_height;
1564 wp->w_height = temp;
1565 temp = curwin->w_width;
1566 curwin->w_width = wp->w_width;
1567 wp->w_width = temp;
1568 }
1569 else
1570 {
1571 frame_fix_height(curwin);
1572 frame_fix_height(wp);
1573 frame_fix_width(curwin);
1574 frame_fix_width(wp);
1575 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001576
1577 (void)win_comp_pos(); /* recompute window positions */
1578
1579 win_enter(wp, TRUE);
1580 redraw_later(CLEAR);
1581}
1582
1583/*
1584 * rotate windows: if upwards TRUE the second window becomes the first one
1585 * if upwards FALSE the first window becomes the second one
1586 */
1587 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001588win_rotate(int upwards, int count)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001589{
1590 win_T *wp1;
1591 win_T *wp2;
1592 frame_T *frp;
1593 int n;
1594
Bram Moolenaara1f4cb92016-11-06 15:25:42 +01001595 if (ONE_WINDOW) /* nothing to do */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001596 {
1597 beep_flush();
1598 return;
1599 }
1600
1601#ifdef FEAT_GUI
1602 need_mouse_correct = TRUE;
1603#endif
1604
Bram Moolenaar071d4272004-06-13 20:20:40 +00001605 /* Check if all frames in this row/col have one window. */
1606 for (frp = curwin->w_frame->fr_parent->fr_child; frp != NULL;
1607 frp = frp->fr_next)
1608 if (frp->fr_win == NULL)
1609 {
1610 EMSG(_("E443: Cannot rotate when another window is split"));
1611 return;
1612 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001613
1614 while (count--)
1615 {
1616 if (upwards) /* first window becomes last window */
1617 {
1618 /* remove first window/frame from the list */
1619 frp = curwin->w_frame->fr_parent->fr_child;
1620 wp1 = frp->fr_win;
Bram Moolenaarf740b292006-02-16 22:11:02 +00001621 win_remove(wp1, NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001622 frame_remove(frp);
1623
1624 /* find last frame and append removed window/frame after it */
1625 for ( ; frp->fr_next != NULL; frp = frp->fr_next)
1626 ;
1627 win_append(frp->fr_win, wp1);
1628 frame_append(frp, wp1->w_frame);
1629
1630 wp2 = frp->fr_win; /* previously last window */
1631 }
1632 else /* last window becomes first window */
1633 {
1634 /* find last window/frame in the list and remove it */
1635 for (frp = curwin->w_frame; frp->fr_next != NULL;
1636 frp = frp->fr_next)
1637 ;
1638 wp1 = frp->fr_win;
1639 wp2 = wp1->w_prev; /* will become last window */
Bram Moolenaarf740b292006-02-16 22:11:02 +00001640 win_remove(wp1, NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001641 frame_remove(frp);
1642
1643 /* append the removed window/frame before the first in the list */
1644 win_append(frp->fr_parent->fr_child->fr_win->w_prev, wp1);
1645 frame_insert(frp->fr_parent->fr_child, frp);
1646 }
1647
1648 /* exchange status height and vsep width of old and new last window */
1649 n = wp2->w_status_height;
1650 wp2->w_status_height = wp1->w_status_height;
1651 wp1->w_status_height = n;
1652 frame_fix_height(wp1);
1653 frame_fix_height(wp2);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001654 n = wp2->w_vsep_width;
1655 wp2->w_vsep_width = wp1->w_vsep_width;
1656 wp1->w_vsep_width = n;
1657 frame_fix_width(wp1);
1658 frame_fix_width(wp2);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001659
Bram Moolenaar44a2f922016-03-19 22:11:51 +01001660 /* recompute w_winrow and w_wincol for all windows */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001661 (void)win_comp_pos();
1662 }
1663
1664 redraw_later(CLEAR);
1665}
1666
1667/*
1668 * Move the current window to the very top/bottom/left/right of the screen.
1669 */
1670 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001671win_totop(int size, int flags)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001672{
1673 int dir;
1674 int height = curwin->w_height;
1675
Bram Moolenaar459ca562016-11-10 18:16:33 +01001676 if (ONE_WINDOW)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001677 {
1678 beep_flush();
1679 return;
1680 }
1681
1682 /* Remove the window and frame from the tree of frames. */
Bram Moolenaarf740b292006-02-16 22:11:02 +00001683 (void)winframe_remove(curwin, &dir, NULL);
1684 win_remove(curwin, NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001685 last_status(FALSE); /* may need to remove last status line */
1686 (void)win_comp_pos(); /* recompute window positions */
1687
1688 /* Split a window on the desired side and put the window there. */
1689 (void)win_split_ins(size, flags, curwin, dir);
1690 if (!(flags & WSP_VERT))
1691 {
1692 win_setheight(height);
1693 if (p_ea)
1694 win_equal(curwin, TRUE, 'v');
1695 }
1696
Bram Moolenaar44a2f922016-03-19 22:11:51 +01001697#if defined(FEAT_GUI)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001698 /* When 'guioptions' includes 'L' or 'R' may have to remove or add
1699 * scrollbars. Have to update them anyway. */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00001700 gui_may_update_scrollbars();
Bram Moolenaar071d4272004-06-13 20:20:40 +00001701#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001702}
1703
1704/*
1705 * Move window "win1" to below/right of "win2" and make "win1" the current
1706 * window. Only works within the same frame!
1707 */
1708 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001709win_move_after(win_T *win1, win_T *win2)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001710{
1711 int height;
1712
1713 /* check if the arguments are reasonable */
1714 if (win1 == win2)
1715 return;
1716
1717 /* check if there is something to do */
1718 if (win2->w_next != win1)
1719 {
1720 /* may need move the status line/vertical separator of the last window
1721 * */
1722 if (win1 == lastwin)
1723 {
1724 height = win1->w_prev->w_status_height;
1725 win1->w_prev->w_status_height = win1->w_status_height;
1726 win1->w_status_height = height;
Bram Moolenaar0396ab02007-02-19 23:14:18 +00001727 if (win1->w_prev->w_vsep_width == 1)
1728 {
1729 /* Remove the vertical separator from the last-but-one window,
1730 * add it to the last window. Adjust the frame widths. */
1731 win1->w_prev->w_vsep_width = 0;
1732 win1->w_prev->w_frame->fr_width -= 1;
1733 win1->w_vsep_width = 1;
1734 win1->w_frame->fr_width += 1;
1735 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001736 }
1737 else if (win2 == lastwin)
1738 {
1739 height = win1->w_status_height;
1740 win1->w_status_height = win2->w_status_height;
1741 win2->w_status_height = height;
Bram Moolenaar0396ab02007-02-19 23:14:18 +00001742 if (win1->w_vsep_width == 1)
1743 {
1744 /* Remove the vertical separator from win1, add it to the last
1745 * window, win2. Adjust the frame widths. */
1746 win2->w_vsep_width = 1;
1747 win2->w_frame->fr_width += 1;
1748 win1->w_vsep_width = 0;
1749 win1->w_frame->fr_width -= 1;
1750 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001751 }
Bram Moolenaarf740b292006-02-16 22:11:02 +00001752 win_remove(win1, NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001753 frame_remove(win1->w_frame);
1754 win_append(win2, win1);
1755 frame_append(win2->w_frame, win1->w_frame);
1756
1757 (void)win_comp_pos(); /* recompute w_winrow for all windows */
1758 redraw_later(NOT_VALID);
1759 }
1760 win_enter(win1, FALSE);
1761}
1762
1763/*
1764 * Make all windows the same height.
1765 * 'next_curwin' will soon be the current window, make sure it has enough
1766 * rows.
1767 */
1768 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001769win_equal(
1770 win_T *next_curwin, /* pointer to current window to be or NULL */
1771 int current, /* do only frame with current window */
1772 int dir) /* 'v' for vertically, 'h' for horizontally,
Bram Moolenaar071d4272004-06-13 20:20:40 +00001773 'b' for both, 0 for using p_ead */
1774{
1775 if (dir == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001776 dir = *p_ead;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001777 win_equal_rec(next_curwin == NULL ? curwin : next_curwin, current,
Bram Moolenaar32466aa2006-02-24 23:53:04 +00001778 topframe, dir, 0, tabline_height(),
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00001779 (int)Columns, topframe->fr_height);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001780}
1781
1782/*
1783 * Set a frame to a new position and height, spreading the available room
1784 * equally over contained frames.
1785 * The window "next_curwin" (if not NULL) should at least get the size from
1786 * 'winheight' and 'winwidth' if possible.
1787 */
1788 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01001789win_equal_rec(
1790 win_T *next_curwin, /* pointer to current window to be or NULL */
1791 int current, /* do only frame with current window */
1792 frame_T *topfr, /* frame to set size off */
1793 int dir, /* 'v', 'h' or 'b', see win_equal() */
1794 int col, /* horizontal position for frame */
1795 int row, /* vertical position for frame */
1796 int width, /* new width of frame */
1797 int height) /* new height of frame */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001798{
1799 int n, m;
1800 int extra_sep = 0;
1801 int wincount, totwincount = 0;
1802 frame_T *fr;
1803 int next_curwin_size = 0;
1804 int room = 0;
1805 int new_size;
1806 int has_next_curwin = 0;
1807 int hnc;
1808
1809 if (topfr->fr_layout == FR_LEAF)
1810 {
1811 /* Set the width/height of this frame.
1812 * Redraw when size or position changes */
1813 if (topfr->fr_height != height || topfr->fr_win->w_winrow != row
Bram Moolenaar071d4272004-06-13 20:20:40 +00001814 || topfr->fr_width != width || topfr->fr_win->w_wincol != col
Bram Moolenaar071d4272004-06-13 20:20:40 +00001815 )
1816 {
1817 topfr->fr_win->w_winrow = row;
1818 frame_new_height(topfr, height, FALSE, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001819 topfr->fr_win->w_wincol = col;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001820 frame_new_width(topfr, width, FALSE, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001821 redraw_all_later(CLEAR);
1822 }
1823 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001824 else if (topfr->fr_layout == FR_ROW)
1825 {
1826 topfr->fr_width = width;
1827 topfr->fr_height = height;
1828
1829 if (dir != 'v') /* equalize frame widths */
1830 {
1831 /* Compute the maximum number of windows horizontally in this
1832 * frame. */
1833 n = frame_minwidth(topfr, NOWIN);
1834 /* add one for the rightmost window, it doesn't have a separator */
1835 if (col + width == Columns)
1836 extra_sep = 1;
1837 else
1838 extra_sep = 0;
1839 totwincount = (n + extra_sep) / (p_wmw + 1);
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001840 has_next_curwin = frame_has_win(topfr, next_curwin);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001841
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001842 /*
1843 * Compute width for "next_curwin" window and room available for
1844 * other windows.
1845 * "m" is the minimal width when counting p_wiw for "next_curwin".
1846 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001847 m = frame_minwidth(topfr, next_curwin);
1848 room = width - m;
1849 if (room < 0)
1850 {
1851 next_curwin_size = p_wiw + room;
1852 room = 0;
1853 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001854 else
1855 {
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001856 next_curwin_size = -1;
1857 for (fr = topfr->fr_child; fr != NULL; fr = fr->fr_next)
1858 {
1859 /* If 'winfixwidth' set keep the window width if
1860 * possible.
1861 * Watch out for this window being the next_curwin. */
1862 if (frame_fixed_width(fr))
1863 {
1864 n = frame_minwidth(fr, NOWIN);
1865 new_size = fr->fr_width;
1866 if (frame_has_win(fr, next_curwin))
1867 {
1868 room += p_wiw - p_wmw;
1869 next_curwin_size = 0;
1870 if (new_size < p_wiw)
1871 new_size = p_wiw;
1872 }
1873 else
1874 /* These windows don't use up room. */
1875 totwincount -= (n + (fr->fr_next == NULL
1876 ? extra_sep : 0)) / (p_wmw + 1);
1877 room -= new_size - n;
1878 if (room < 0)
1879 {
1880 new_size += room;
1881 room = 0;
1882 }
1883 fr->fr_newwidth = new_size;
1884 }
1885 }
1886 if (next_curwin_size == -1)
1887 {
1888 if (!has_next_curwin)
1889 next_curwin_size = 0;
1890 else if (totwincount > 1
1891 && (room + (totwincount - 2))
1892 / (totwincount - 1) > p_wiw)
1893 {
Bram Moolenaarb21e5842006-04-16 18:30:08 +00001894 /* Can make all windows wider than 'winwidth', spread
1895 * the room equally. */
1896 next_curwin_size = (room + p_wiw
1897 + (totwincount - 1) * p_wmw
1898 + (totwincount - 1)) / totwincount;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001899 room -= next_curwin_size - p_wiw;
1900 }
1901 else
1902 next_curwin_size = p_wiw;
1903 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001904 }
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001905
1906 if (has_next_curwin)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001907 --totwincount; /* don't count curwin */
1908 }
1909
1910 for (fr = topfr->fr_child; fr != NULL; fr = fr->fr_next)
1911 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001912 wincount = 1;
1913 if (fr->fr_next == NULL)
1914 /* last frame gets all that remains (avoid roundoff error) */
1915 new_size = width;
1916 else if (dir == 'v')
1917 new_size = fr->fr_width;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001918 else if (frame_fixed_width(fr))
1919 {
1920 new_size = fr->fr_newwidth;
1921 wincount = 0; /* doesn't count as a sizeable window */
1922 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001923 else
1924 {
1925 /* Compute the maximum number of windows horiz. in "fr". */
1926 n = frame_minwidth(fr, NOWIN);
1927 wincount = (n + (fr->fr_next == NULL ? extra_sep : 0))
1928 / (p_wmw + 1);
1929 m = frame_minwidth(fr, next_curwin);
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001930 if (has_next_curwin)
1931 hnc = frame_has_win(fr, next_curwin);
1932 else
1933 hnc = FALSE;
1934 if (hnc) /* don't count next_curwin */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001935 --wincount;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001936 if (totwincount == 0)
1937 new_size = room;
1938 else
1939 new_size = (wincount * room + ((unsigned)totwincount >> 1))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001940 / totwincount;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001941 if (hnc) /* add next_curwin size */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001942 {
1943 next_curwin_size -= p_wiw - (m - n);
1944 new_size += next_curwin_size;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001945 room -= new_size - next_curwin_size;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001946 }
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001947 else
1948 room -= new_size;
1949 new_size += n;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001950 }
1951
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001952 /* Skip frame that is full width when splitting or closing a
Bram Moolenaar071d4272004-06-13 20:20:40 +00001953 * window, unless equalizing all frames. */
1954 if (!current || dir != 'v' || topfr->fr_parent != NULL
1955 || (new_size != fr->fr_width)
1956 || frame_has_win(fr, next_curwin))
1957 win_equal_rec(next_curwin, current, fr, dir, col, row,
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00001958 new_size, height);
1959 col += new_size;
1960 width -= new_size;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001961 totwincount -= wincount;
1962 }
1963 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001964 else /* topfr->fr_layout == FR_COL */
1965 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001966 topfr->fr_width = width;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001967 topfr->fr_height = height;
1968
1969 if (dir != 'h') /* equalize frame heights */
1970 {
1971 /* Compute maximum number of windows vertically in this frame. */
1972 n = frame_minheight(topfr, NOWIN);
1973 /* add one for the bottom window if it doesn't have a statusline */
1974 if (row + height == cmdline_row && p_ls == 0)
1975 extra_sep = 1;
1976 else
1977 extra_sep = 0;
1978 totwincount = (n + extra_sep) / (p_wmh + 1);
1979 has_next_curwin = frame_has_win(topfr, next_curwin);
1980
1981 /*
1982 * Compute height for "next_curwin" window and room available for
1983 * other windows.
1984 * "m" is the minimal height when counting p_wh for "next_curwin".
1985 */
1986 m = frame_minheight(topfr, next_curwin);
1987 room = height - m;
1988 if (room < 0)
1989 {
1990 /* The room is less then 'winheight', use all space for the
1991 * current window. */
1992 next_curwin_size = p_wh + room;
1993 room = 0;
1994 }
1995 else
1996 {
1997 next_curwin_size = -1;
1998 for (fr = topfr->fr_child; fr != NULL; fr = fr->fr_next)
1999 {
2000 /* If 'winfixheight' set keep the window height if
2001 * possible.
2002 * Watch out for this window being the next_curwin. */
2003 if (frame_fixed_height(fr))
2004 {
2005 n = frame_minheight(fr, NOWIN);
2006 new_size = fr->fr_height;
2007 if (frame_has_win(fr, next_curwin))
2008 {
2009 room += p_wh - p_wmh;
2010 next_curwin_size = 0;
2011 if (new_size < p_wh)
2012 new_size = p_wh;
2013 }
2014 else
2015 /* These windows don't use up room. */
2016 totwincount -= (n + (fr->fr_next == NULL
2017 ? extra_sep : 0)) / (p_wmh + 1);
2018 room -= new_size - n;
2019 if (room < 0)
2020 {
2021 new_size += room;
2022 room = 0;
2023 }
2024 fr->fr_newheight = new_size;
2025 }
2026 }
2027 if (next_curwin_size == -1)
2028 {
2029 if (!has_next_curwin)
2030 next_curwin_size = 0;
2031 else if (totwincount > 1
2032 && (room + (totwincount - 2))
2033 / (totwincount - 1) > p_wh)
2034 {
Bram Moolenaarb21e5842006-04-16 18:30:08 +00002035 /* can make all windows higher than 'winheight',
2036 * spread the room equally. */
2037 next_curwin_size = (room + p_wh
2038 + (totwincount - 1) * p_wmh
Bram Moolenaar071d4272004-06-13 20:20:40 +00002039 + (totwincount - 1)) / totwincount;
2040 room -= next_curwin_size - p_wh;
2041 }
2042 else
2043 next_curwin_size = p_wh;
2044 }
2045 }
2046
2047 if (has_next_curwin)
2048 --totwincount; /* don't count curwin */
2049 }
2050
2051 for (fr = topfr->fr_child; fr != NULL; fr = fr->fr_next)
2052 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00002053 wincount = 1;
2054 if (fr->fr_next == NULL)
2055 /* last frame gets all that remains (avoid roundoff error) */
2056 new_size = height;
2057 else if (dir == 'h')
2058 new_size = fr->fr_height;
2059 else if (frame_fixed_height(fr))
2060 {
2061 new_size = fr->fr_newheight;
2062 wincount = 0; /* doesn't count as a sizeable window */
2063 }
2064 else
2065 {
2066 /* Compute the maximum number of windows vert. in "fr". */
2067 n = frame_minheight(fr, NOWIN);
2068 wincount = (n + (fr->fr_next == NULL ? extra_sep : 0))
2069 / (p_wmh + 1);
2070 m = frame_minheight(fr, next_curwin);
2071 if (has_next_curwin)
2072 hnc = frame_has_win(fr, next_curwin);
2073 else
2074 hnc = FALSE;
2075 if (hnc) /* don't count next_curwin */
2076 --wincount;
2077 if (totwincount == 0)
2078 new_size = room;
2079 else
2080 new_size = (wincount * room + ((unsigned)totwincount >> 1))
2081 / totwincount;
2082 if (hnc) /* add next_curwin size */
2083 {
2084 next_curwin_size -= p_wh - (m - n);
2085 new_size += next_curwin_size;
2086 room -= new_size - next_curwin_size;
2087 }
2088 else
2089 room -= new_size;
2090 new_size += n;
2091 }
2092 /* Skip frame that is full width when splitting or closing a
2093 * window, unless equalizing all frames. */
2094 if (!current || dir != 'h' || topfr->fr_parent != NULL
2095 || (new_size != fr->fr_height)
2096 || frame_has_win(fr, next_curwin))
2097 win_equal_rec(next_curwin, current, fr, dir, col, row,
2098 width, new_size);
2099 row += new_size;
2100 height -= new_size;
2101 totwincount -= wincount;
2102 }
2103 }
2104}
2105
Bram Moolenaar6d41c782018-06-06 09:11:12 +02002106#ifdef FEAT_JOB_CHANNEL
2107 static void
2108leaving_window(win_T *win)
2109{
2110 // When leaving a prompt window stop Insert mode and perhaps restart
2111 // it when entering that window again.
2112 win->w_buffer->b_prompt_insert = restart_edit;
2113 restart_edit = NUL;
2114
2115 // When leaving the window (or closing the window) was done from a
2116 // callback we need to break out of the Insert mode loop.
2117 if (State & INSERT)
2118 stop_insert_mode = TRUE;
2119}
2120
2121 static void
2122entering_window(win_T *win)
2123{
2124 // When entering the prompt window may restart Insert mode.
2125 restart_edit = win->w_buffer->b_prompt_insert;
2126}
2127#endif
2128
Bram Moolenaar071d4272004-06-13 20:20:40 +00002129/*
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01002130 * Close all windows for buffer "buf".
Bram Moolenaar071d4272004-06-13 20:20:40 +00002131 */
2132 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002133close_windows(
2134 buf_T *buf,
2135 int keep_curwin) /* don't close "curwin" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00002136{
Bram Moolenaarf740b292006-02-16 22:11:02 +00002137 win_T *wp;
2138 tabpage_T *tp, *nexttp;
Bram Moolenaar32466aa2006-02-24 23:53:04 +00002139 int h = tabline_height();
Bram Moolenaar12c11d52016-07-19 23:13:03 +02002140 int count = tabpage_index(NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002141
2142 ++RedrawingDisabled;
Bram Moolenaarf740b292006-02-16 22:11:02 +00002143
Bram Moolenaar459ca562016-11-10 18:16:33 +01002144 for (wp = firstwin; wp != NULL && !ONE_WINDOW; )
Bram Moolenaar071d4272004-06-13 20:20:40 +00002145 {
Bram Moolenaar362ce482012-06-06 19:02:45 +02002146 if (wp->w_buffer == buf && (!keep_curwin || wp != curwin)
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01002147 && !(wp->w_closing || wp->w_buffer->b_locked > 0))
Bram Moolenaar071d4272004-06-13 20:20:40 +00002148 {
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01002149 if (win_close(wp, FALSE) == FAIL)
2150 /* If closing the window fails give up, to avoid looping
2151 * forever. */
2152 break;
Bram Moolenaarf740b292006-02-16 22:11:02 +00002153
2154 /* Start all over, autocommands may change the window layout. */
2155 wp = firstwin;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002156 }
2157 else
Bram Moolenaarf740b292006-02-16 22:11:02 +00002158 wp = wp->w_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002159 }
Bram Moolenaarf740b292006-02-16 22:11:02 +00002160
2161 /* Also check windows in other tab pages. */
2162 for (tp = first_tabpage; tp != NULL; tp = nexttp)
2163 {
2164 nexttp = tp->tp_next;
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00002165 if (tp != curtab)
Bram Moolenaarf740b292006-02-16 22:11:02 +00002166 for (wp = tp->tp_firstwin; wp != NULL; wp = wp->w_next)
Bram Moolenaar362ce482012-06-06 19:02:45 +02002167 if (wp->w_buffer == buf
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01002168 && !(wp->w_closing || wp->w_buffer->b_locked > 0))
Bram Moolenaarf740b292006-02-16 22:11:02 +00002169 {
2170 win_close_othertab(wp, FALSE, tp);
2171
2172 /* Start all over, the tab page may be closed and
2173 * autocommands may change the window layout. */
2174 nexttp = first_tabpage;
2175 break;
2176 }
2177 }
2178
Bram Moolenaar071d4272004-06-13 20:20:40 +00002179 --RedrawingDisabled;
Bram Moolenaarf740b292006-02-16 22:11:02 +00002180
Bram Moolenaar12c11d52016-07-19 23:13:03 +02002181 if (count != tabpage_index(NULL))
2182 apply_autocmds(EVENT_TABCLOSED, NULL, NULL, FALSE, curbuf);
Bram Moolenaar12c11d52016-07-19 23:13:03 +02002183
Bram Moolenaar4c7e9db2013-04-15 15:55:19 +02002184 redraw_tabline = TRUE;
Bram Moolenaar32466aa2006-02-24 23:53:04 +00002185 if (h != tabline_height())
Bram Moolenaarf740b292006-02-16 22:11:02 +00002186 shell_new_rows();
Bram Moolenaar071d4272004-06-13 20:20:40 +00002187}
2188
2189/*
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002190 * Return TRUE if the current window is the only window that exists (ignoring
2191 * "aucmd_win").
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00002192 * Returns FALSE if there is a window, possibly in another tab page.
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002193 */
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00002194 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002195last_window(void)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002196{
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002197 return (one_window() && first_tabpage->tp_next == NULL);
2198}
2199
2200/*
2201 * Return TRUE if there is only one window other than "aucmd_win" in the
2202 * current tab page.
2203 */
Bram Moolenaar42ec6562012-02-22 14:58:37 +01002204 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002205one_window(void)
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002206{
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002207 win_T *wp;
2208 int seen_one = FALSE;
2209
2210 FOR_ALL_WINDOWS(wp)
2211 {
2212 if (wp != aucmd_win)
2213 {
2214 if (seen_one)
2215 return FALSE;
2216 seen_one = TRUE;
2217 }
2218 }
2219 return TRUE;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002220}
2221
2222/*
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002223 * Close the possibly last window in a tab page.
2224 * Returns TRUE when the window was closed already.
2225 */
2226 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002227close_last_window_tabpage(
2228 win_T *win,
2229 int free_buf,
2230 tabpage_T *prev_curtab)
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002231{
Bram Moolenaara1f4cb92016-11-06 15:25:42 +01002232 if (ONE_WINDOW)
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002233 {
Bram Moolenaar49e649f2013-05-06 04:50:35 +02002234 buf_T *old_curbuf = curbuf;
2235
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002236 /*
2237 * Closing the last window in a tab page. First go to another tab
2238 * page and then close the window and the tab page. This avoids that
2239 * curwin and curtab are invalid while we are freeing memory, they may
2240 * be used in GUI events.
Bram Moolenaara8596c42012-06-13 14:28:20 +02002241 * Don't trigger autocommands yet, they may use wrong values, so do
2242 * that below.
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002243 */
Bram Moolenaar49e649f2013-05-06 04:50:35 +02002244 goto_tabpage_tp(alt_tabpage(), FALSE, TRUE);
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002245 redraw_tabline = TRUE;
2246
2247 /* Safety check: Autocommands may have closed the window when jumping
2248 * to the other tab page. */
2249 if (valid_tabpage(prev_curtab) && prev_curtab->tp_firstwin == win)
2250 {
2251 int h = tabline_height();
2252
2253 win_close_othertab(win, free_buf, prev_curtab);
2254 if (h != tabline_height())
2255 shell_new_rows();
2256 }
Bram Moolenaar6d41c782018-06-06 09:11:12 +02002257#ifdef FEAT_JOB_CHANNEL
2258 entering_window(curwin);
2259#endif
Bram Moolenaara8596c42012-06-13 14:28:20 +02002260 /* Since goto_tabpage_tp above did not trigger *Enter autocommands, do
2261 * that now. */
Bram Moolenaar12c11d52016-07-19 23:13:03 +02002262 apply_autocmds(EVENT_TABCLOSED, NULL, NULL, FALSE, curbuf);
Bram Moolenaara8596c42012-06-13 14:28:20 +02002263 apply_autocmds(EVENT_WINENTER, NULL, NULL, FALSE, curbuf);
Bram Moolenaar49e649f2013-05-06 04:50:35 +02002264 apply_autocmds(EVENT_TABENTER, NULL, NULL, FALSE, curbuf);
2265 if (old_curbuf != curbuf)
2266 apply_autocmds(EVENT_BUFENTER, NULL, NULL, FALSE, curbuf);
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002267 return TRUE;
2268 }
2269 return FALSE;
2270}
2271
2272/*
Bram Moolenaar910f66f2006-04-05 20:41:53 +00002273 * Close window "win". Only works for the current tab page.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002274 * If "free_buf" is TRUE related buffer may be unloaded.
2275 *
Bram Moolenaar42ec6562012-02-22 14:58:37 +01002276 * Called by :quit, :close, :xit, :wq and findtag().
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002277 * Returns FAIL when the window was not closed.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002278 */
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002279 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002280win_close(win_T *win, int free_buf)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002281{
2282 win_T *wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002283 int other_buffer = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002284 int close_curwin = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002285 int dir;
2286 int help_window = FALSE;
Bram Moolenaarc1b52862006-04-28 22:32:28 +00002287 tabpage_T *prev_curtab = curtab;
Bram Moolenaar41cc0382017-06-26 09:59:35 +02002288 frame_T *win_frame = win->w_frame->fr_parent;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002289
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002290 if (last_window())
Bram Moolenaar071d4272004-06-13 20:20:40 +00002291 {
2292 EMSG(_("E444: Cannot close last window"));
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002293 return FAIL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002294 }
2295
Bram Moolenaare0ab94e2016-09-04 19:50:54 +02002296 if (win->w_closing || (win->w_buffer != NULL
2297 && win->w_buffer->b_locked > 0))
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002298 return FAIL; /* window is already being closed */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002299 if (win == aucmd_win)
2300 {
2301 EMSG(_("E813: Cannot close autocmd window"));
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002302 return FAIL;
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002303 }
2304 if ((firstwin == aucmd_win || lastwin == aucmd_win) && one_window())
2305 {
2306 EMSG(_("E814: Cannot close window, only autocmd window would remain"));
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002307 return FAIL;
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002308 }
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002309
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002310 /* When closing the last window in a tab page first go to another tab page
2311 * and then close the window and the tab page to avoid that curwin and
2312 * curtab are invalid while we are freeing memory. */
2313 if (close_last_window_tabpage(win, free_buf, prev_curtab))
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002314 return FAIL;
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00002315
Bram Moolenaar071d4272004-06-13 20:20:40 +00002316 /* When closing the help window, try restoring a snapshot after closing
2317 * the window. Otherwise clear the snapshot, it's now invalid. */
Bram Moolenaard28cc3f2017-07-27 22:03:50 +02002318 if (bt_help(win->w_buffer))
Bram Moolenaar071d4272004-06-13 20:20:40 +00002319 help_window = TRUE;
2320 else
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002321 clear_snapshot(curtab, SNAP_HELP_IDX);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002322
Bram Moolenaar071d4272004-06-13 20:20:40 +00002323 if (win == curwin)
2324 {
Bram Moolenaar6d41c782018-06-06 09:11:12 +02002325#ifdef FEAT_JOB_CHANNEL
2326 leaving_window(curwin);
2327#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002328 /*
2329 * Guess which window is going to be the new current window.
2330 * This may change because of the autocommands (sigh).
2331 */
Bram Moolenaarf740b292006-02-16 22:11:02 +00002332 wp = frame2win(win_altframe(win, NULL));
Bram Moolenaar071d4272004-06-13 20:20:40 +00002333
2334 /*
Bram Moolenaar362ce482012-06-06 19:02:45 +02002335 * Be careful: If autocommands delete the window or cause this window
2336 * to be the last one left, return now.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002337 */
2338 if (wp->w_buffer != curbuf)
2339 {
2340 other_buffer = TRUE;
Bram Moolenaar362ce482012-06-06 19:02:45 +02002341 win->w_closing = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002342 apply_autocmds(EVENT_BUFLEAVE, NULL, NULL, FALSE, curbuf);
Bram Moolenaar362ce482012-06-06 19:02:45 +02002343 if (!win_valid(win))
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002344 return FAIL;
Bram Moolenaar362ce482012-06-06 19:02:45 +02002345 win->w_closing = FALSE;
2346 if (last_window())
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002347 return FAIL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002348 }
Bram Moolenaar362ce482012-06-06 19:02:45 +02002349 win->w_closing = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002350 apply_autocmds(EVENT_WINLEAVE, NULL, NULL, FALSE, curbuf);
Bram Moolenaar362ce482012-06-06 19:02:45 +02002351 if (!win_valid(win))
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002352 return FAIL;
Bram Moolenaar362ce482012-06-06 19:02:45 +02002353 win->w_closing = FALSE;
2354 if (last_window())
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002355 return FAIL;
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01002356#ifdef FEAT_EVAL
Bram Moolenaar071d4272004-06-13 20:20:40 +00002357 /* autocmds may abort script processing */
2358 if (aborting())
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002359 return FAIL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002360#endif
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01002361 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002362
Bram Moolenaar053b9fa2007-04-26 14:09:42 +00002363#ifdef FEAT_GUI
2364 /* Avoid trouble with scrollbars that are going to be deleted in
2365 * win_free(). */
2366 if (gui.in_use)
2367 out_flush();
2368#endif
2369
Bram Moolenaara971b822011-09-14 14:43:25 +02002370#ifdef FEAT_SYN_HL
2371 /* Free independent synblock before the buffer is freed. */
Bram Moolenaarfc573802011-12-30 15:01:59 +01002372 if (win->w_buffer != NULL)
2373 reset_synblock(win);
Bram Moolenaara971b822011-09-14 14:43:25 +02002374#endif
2375
Bram Moolenaar071d4272004-06-13 20:20:40 +00002376 /*
2377 * Close the link to the buffer.
2378 */
Bram Moolenaarfc573802011-12-30 15:01:59 +01002379 if (win->w_buffer != NULL)
Bram Moolenaar362ce482012-06-06 19:02:45 +02002380 {
Bram Moolenaar7c0a2f32016-07-10 22:11:16 +02002381 bufref_T bufref;
2382
2383 set_bufref(&bufref, curbuf);
Bram Moolenaar362ce482012-06-06 19:02:45 +02002384 win->w_closing = TRUE;
Bram Moolenaar8f913992012-08-29 15:50:26 +02002385 close_buffer(win, win->w_buffer, free_buf ? DOBUF_UNLOAD : 0, TRUE);
Bram Moolenaar11fbc282016-09-02 21:48:32 +02002386 if (win_valid_any_tab(win))
Bram Moolenaar362ce482012-06-06 19:02:45 +02002387 win->w_closing = FALSE;
Bram Moolenaar62ef7972016-01-19 14:51:54 +01002388 /* Make sure curbuf is valid. It can become invalid if 'bufhidden' is
2389 * "wipe". */
Bram Moolenaar7c0a2f32016-07-10 22:11:16 +02002390 if (!bufref_valid(&bufref))
Bram Moolenaar62ef7972016-01-19 14:51:54 +01002391 curbuf = firstbuf;
Bram Moolenaar362ce482012-06-06 19:02:45 +02002392 }
Bram Moolenaarc1b52862006-04-28 22:32:28 +00002393
Bram Moolenaar802418d2013-01-17 14:00:11 +01002394 if (only_one_window() && win_valid(win) && win->w_buffer == NULL
2395 && (last_window() || curtab != prev_curtab
2396 || close_last_window_tabpage(win, free_buf, prev_curtab)))
Bram Moolenaar2b90ed22013-07-24 16:02:36 +02002397 {
Bram Moolenaard28cc3f2017-07-27 22:03:50 +02002398 /* Autocommands have closed all windows, quit now. Restore
Bram Moolenaar2b90ed22013-07-24 16:02:36 +02002399 * curwin->w_buffer, otherwise writing viminfo may fail. */
2400 if (curwin->w_buffer == NULL)
2401 curwin->w_buffer = curbuf;
Bram Moolenaar802418d2013-01-17 14:00:11 +01002402 getout(0);
Bram Moolenaar2b90ed22013-07-24 16:02:36 +02002403 }
Bram Moolenaar802418d2013-01-17 14:00:11 +01002404
Bram Moolenaar11fbc282016-09-02 21:48:32 +02002405 /* Autocommands may have moved to another tab page. */
2406 if (curtab != prev_curtab && win_valid_any_tab(win)
2407 && win->w_buffer == NULL)
2408 {
2409 /* Need to close the window anyway, since the buffer is NULL. */
2410 win_close_othertab(win, FALSE, prev_curtab);
2411 return FAIL;
2412 }
2413
2414 /* Autocommands may have closed the window already or closed the only
2415 * other window. */
2416 if (!win_valid(win) || last_window()
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002417 || close_last_window_tabpage(win, free_buf, prev_curtab))
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002418 return FAIL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002419
Bram Moolenaara971b822011-09-14 14:43:25 +02002420 /* Free the memory used for the window and get the window that received
2421 * the screen space. */
Bram Moolenaarc1b52862006-04-28 22:32:28 +00002422 wp = win_free_mem(win, &dir, NULL);
2423
Bram Moolenaar071d4272004-06-13 20:20:40 +00002424 /* Make sure curwin isn't invalid. It can cause severe trouble when
2425 * printing an error message. For win_equal() curbuf needs to be valid
2426 * too. */
Bram Moolenaarc1b52862006-04-28 22:32:28 +00002427 if (win == curwin)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002428 {
2429 curwin = wp;
2430#ifdef FEAT_QUICKFIX
2431 if (wp->w_p_pvw || bt_quickfix(wp->w_buffer))
2432 {
2433 /*
Bram Moolenaar48cc5fe2007-08-11 11:39:45 +00002434 * If the cursor goes to the preview or the quickfix window, try
Bram Moolenaar071d4272004-06-13 20:20:40 +00002435 * finding another window to go to.
2436 */
2437 for (;;)
2438 {
2439 if (wp->w_next == NULL)
2440 wp = firstwin;
2441 else
2442 wp = wp->w_next;
2443 if (wp == curwin)
2444 break;
2445 if (!wp->w_p_pvw && !bt_quickfix(wp->w_buffer))
2446 {
2447 curwin = wp;
2448 break;
2449 }
2450 }
2451 }
2452#endif
2453 curbuf = curwin->w_buffer;
2454 close_curwin = TRUE;
Bram Moolenaarf79225e2017-03-18 23:11:04 +01002455
2456 /* The cursor position may be invalid if the buffer changed after last
2457 * using the window. */
2458 check_cursor();
Bram Moolenaar071d4272004-06-13 20:20:40 +00002459 }
Bram Moolenaar44a2f922016-03-19 22:11:51 +01002460 if (p_ea && (*p_ead == 'b' || *p_ead == dir))
Bram Moolenaar8eeeba82017-06-25 22:45:39 +02002461 /* If the frame of the closed window contains the new current window,
2462 * only resize that frame. Otherwise resize all windows. */
Bram Moolenaar41cc0382017-06-26 09:59:35 +02002463 win_equal(curwin, curwin->w_frame->fr_parent == win_frame, dir);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002464 else
2465 win_comp_pos();
2466 if (close_curwin)
2467 {
Bram Moolenaarc917da42016-07-19 22:31:36 +02002468 win_enter_ext(wp, FALSE, TRUE, FALSE, TRUE, TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002469 if (other_buffer)
2470 /* careful: after this wp and win may be invalid! */
2471 apply_autocmds(EVENT_BUFENTER, NULL, NULL, FALSE, curbuf);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002472 }
2473
2474 /*
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002475 * If last window has a status line now and we don't want one,
2476 * remove the status line.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002477 */
2478 last_status(FALSE);
2479
2480 /* After closing the help window, try restoring the window layout from
2481 * before it was opened. */
2482 if (help_window)
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002483 restore_snapshot(SNAP_HELP_IDX, close_curwin);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002484
Bram Moolenaar44a2f922016-03-19 22:11:51 +01002485#if defined(FEAT_GUI)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002486 /* When 'guioptions' includes 'L' or 'R' may have to remove scrollbars. */
2487 if (gui.in_use && !win_hasvertsplit())
2488 gui_init_which_components(NULL);
2489#endif
2490
2491 redraw_all_later(NOT_VALID);
Bram Moolenaarc93df6b2013-08-14 17:11:20 +02002492 return OK;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002493}
2494
2495/*
Bram Moolenaarf740b292006-02-16 22:11:02 +00002496 * Close window "win" in tab page "tp", which is not the current tab page.
Bram Moolenaarbef1c362012-05-25 12:39:00 +02002497 * This may be the last window in that tab page and result in closing the tab,
Bram Moolenaarf740b292006-02-16 22:11:02 +00002498 * thus "tp" may become invalid!
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00002499 * Caller must check if buffer is hidden and whether the tabline needs to be
2500 * updated.
Bram Moolenaarf740b292006-02-16 22:11:02 +00002501 */
2502 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002503win_close_othertab(win_T *win, int free_buf, tabpage_T *tp)
Bram Moolenaarf740b292006-02-16 22:11:02 +00002504{
2505 win_T *wp;
2506 int dir;
2507 tabpage_T *ptp = NULL;
Bram Moolenaar4d770fb2010-07-12 21:38:19 +02002508 int free_tp = FALSE;
Bram Moolenaarf740b292006-02-16 22:11:02 +00002509
Bram Moolenaar11fbc282016-09-02 21:48:32 +02002510 /* Get here with win->w_buffer == NULL when win_close() detects the tab
2511 * page changed. */
Bram Moolenaare0ab94e2016-09-04 19:50:54 +02002512 if (win->w_closing || (win->w_buffer != NULL
2513 && win->w_buffer->b_locked > 0))
Bram Moolenaar362ce482012-06-06 19:02:45 +02002514 return; /* window is already being closed */
Bram Moolenaar362ce482012-06-06 19:02:45 +02002515
Bram Moolenaar11fbc282016-09-02 21:48:32 +02002516 if (win->w_buffer != NULL)
2517 /* Close the link to the buffer. */
2518 close_buffer(win, win->w_buffer, free_buf ? DOBUF_UNLOAD : 0, FALSE);
Bram Moolenaarf740b292006-02-16 22:11:02 +00002519
2520 /* Careful: Autocommands may have closed the tab page or made it the
2521 * current tab page. */
2522 for (ptp = first_tabpage; ptp != NULL && ptp != tp; ptp = ptp->tp_next)
2523 ;
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00002524 if (ptp == NULL || tp == curtab)
Bram Moolenaarf740b292006-02-16 22:11:02 +00002525 return;
2526
2527 /* Autocommands may have closed the window already. */
2528 for (wp = tp->tp_firstwin; wp != NULL && wp != win; wp = wp->w_next)
2529 ;
2530 if (wp == NULL)
2531 return;
2532
Bram Moolenaarf740b292006-02-16 22:11:02 +00002533 /* When closing the last window in a tab page remove the tab page. */
Bram Moolenaarcde88542015-08-11 19:14:00 +02002534 if (tp->tp_firstwin == tp->tp_lastwin)
Bram Moolenaarf740b292006-02-16 22:11:02 +00002535 {
2536 if (tp == first_tabpage)
2537 first_tabpage = tp->tp_next;
2538 else
2539 {
2540 for (ptp = first_tabpage; ptp != NULL && ptp->tp_next != tp;
2541 ptp = ptp->tp_next)
2542 ;
2543 if (ptp == NULL)
2544 {
Bram Moolenaar95f09602016-11-10 20:01:45 +01002545 internal_error("win_close_othertab()");
Bram Moolenaarf740b292006-02-16 22:11:02 +00002546 return;
2547 }
2548 ptp->tp_next = tp->tp_next;
2549 }
Bram Moolenaar4d770fb2010-07-12 21:38:19 +02002550 free_tp = TRUE;
Bram Moolenaarf740b292006-02-16 22:11:02 +00002551 }
Bram Moolenaar4d770fb2010-07-12 21:38:19 +02002552
2553 /* Free the memory used for the window. */
2554 win_free_mem(win, &dir, tp);
2555
2556 if (free_tp)
2557 free_tabpage(tp);
Bram Moolenaarf740b292006-02-16 22:11:02 +00002558}
2559
2560/*
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002561 * Free the memory used for a window.
2562 * Returns a pointer to the window that got the freed up space.
2563 */
2564 static win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002565win_free_mem(
2566 win_T *win,
2567 int *dirp, /* set to 'v' or 'h' for direction if 'ea' */
2568 tabpage_T *tp) /* tab page "win" is in, NULL for current */
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002569{
2570 frame_T *frp;
2571 win_T *wp;
2572
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002573 /* Remove the window and its frame from the tree of frames. */
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002574 frp = win->w_frame;
Bram Moolenaarf740b292006-02-16 22:11:02 +00002575 wp = winframe_remove(win, dirp, tp);
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002576 vim_free(frp);
Bram Moolenaarf740b292006-02-16 22:11:02 +00002577 win_free(win, tp);
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002578
Bram Moolenaar910f66f2006-04-05 20:41:53 +00002579 /* When deleting the current window of another tab page select a new
2580 * current window. */
2581 if (tp != NULL && win == tp->tp_curwin)
2582 tp->tp_curwin = wp;
2583
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002584 return wp;
2585}
2586
2587#if defined(EXITFREE) || defined(PROTO)
2588 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002589win_free_all(void)
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002590{
2591 int dummy;
2592
Bram Moolenaarf740b292006-02-16 22:11:02 +00002593 while (first_tabpage->tp_next != NULL)
2594 tabpage_close(TRUE);
Bram Moolenaarf740b292006-02-16 22:11:02 +00002595
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002596 if (aucmd_win != NULL)
2597 {
2598 (void)win_free_mem(aucmd_win, &dummy, NULL);
2599 aucmd_win = NULL;
2600 }
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00002601
2602 while (firstwin != NULL)
2603 (void)win_free_mem(firstwin, &dummy, NULL);
Bram Moolenaar4e036c92014-07-16 16:30:28 +02002604
2605 /* No window should be used after this. Set curwin to NULL to crash
2606 * instead of using freed memory. */
2607 curwin = NULL;
Bram Moolenaar0a5fe212005-06-24 23:01:23 +00002608}
2609#endif
2610
2611/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00002612 * Remove a window and its frame from the tree of frames.
2613 * Returns a pointer to the window that got the freed up space.
2614 */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00002615 win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002616winframe_remove(
2617 win_T *win,
2618 int *dirp UNUSED, /* set to 'v' or 'h' for direction if 'ea' */
2619 tabpage_T *tp) /* tab page "win" is in, NULL for current */
Bram Moolenaar071d4272004-06-13 20:20:40 +00002620{
2621 frame_T *frp, *frp2, *frp3;
2622 frame_T *frp_close = win->w_frame;
2623 win_T *wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002624
2625 /*
Bram Moolenaarf740b292006-02-16 22:11:02 +00002626 * If there is only one window there is nothing to remove.
2627 */
Bram Moolenaara1f4cb92016-11-06 15:25:42 +01002628 if (tp == NULL ? ONE_WINDOW : tp->tp_firstwin == tp->tp_lastwin)
Bram Moolenaarf740b292006-02-16 22:11:02 +00002629 return NULL;
2630
2631 /*
Bram Moolenaar071d4272004-06-13 20:20:40 +00002632 * Remove the window from its frame.
2633 */
Bram Moolenaarf740b292006-02-16 22:11:02 +00002634 frp2 = win_altframe(win, tp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002635 wp = frame2win(frp2);
2636
2637 /* Remove this frame from the list of frames. */
2638 frame_remove(frp_close);
2639
Bram Moolenaar071d4272004-06-13 20:20:40 +00002640 if (frp_close->fr_parent->fr_layout == FR_COL)
2641 {
Bram Moolenaar48cc5fe2007-08-11 11:39:45 +00002642 /* When 'winfixheight' is set, try to find another frame in the column
2643 * (as close to the closed frame as possible) to distribute the height
2644 * to. */
2645 if (frp2->fr_win != NULL && frp2->fr_win->w_p_wfh)
2646 {
2647 frp = frp_close->fr_prev;
2648 frp3 = frp_close->fr_next;
2649 while (frp != NULL || frp3 != NULL)
2650 {
2651 if (frp != NULL)
2652 {
2653 if (frp->fr_win != NULL && !frp->fr_win->w_p_wfh)
2654 {
2655 frp2 = frp;
2656 wp = frp->fr_win;
2657 break;
2658 }
2659 frp = frp->fr_prev;
2660 }
2661 if (frp3 != NULL)
2662 {
2663 if (frp3->fr_win != NULL && !frp3->fr_win->w_p_wfh)
2664 {
2665 frp2 = frp3;
2666 wp = frp3->fr_win;
2667 break;
2668 }
2669 frp3 = frp3->fr_next;
2670 }
2671 }
2672 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002673 frame_new_height(frp2, frp2->fr_height + frp_close->fr_height,
2674 frp2 == frp_close->fr_next ? TRUE : FALSE, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002675 *dirp = 'v';
2676 }
2677 else
2678 {
Bram Moolenaar48cc5fe2007-08-11 11:39:45 +00002679 /* When 'winfixwidth' is set, try to find another frame in the column
2680 * (as close to the closed frame as possible) to distribute the width
2681 * to. */
2682 if (frp2->fr_win != NULL && frp2->fr_win->w_p_wfw)
2683 {
2684 frp = frp_close->fr_prev;
2685 frp3 = frp_close->fr_next;
2686 while (frp != NULL || frp3 != NULL)
2687 {
2688 if (frp != NULL)
2689 {
2690 if (frp->fr_win != NULL && !frp->fr_win->w_p_wfw)
2691 {
2692 frp2 = frp;
2693 wp = frp->fr_win;
2694 break;
2695 }
2696 frp = frp->fr_prev;
2697 }
2698 if (frp3 != NULL)
2699 {
2700 if (frp3->fr_win != NULL && !frp3->fr_win->w_p_wfw)
2701 {
2702 frp2 = frp3;
2703 wp = frp3->fr_win;
2704 break;
2705 }
2706 frp3 = frp3->fr_next;
2707 }
2708 }
2709 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002710 frame_new_width(frp2, frp2->fr_width + frp_close->fr_width,
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00002711 frp2 == frp_close->fr_next ? TRUE : FALSE, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002712 *dirp = 'h';
2713 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002714
2715 /* If rows/columns go to a window below/right its positions need to be
2716 * updated. Can only be done after the sizes have been updated. */
2717 if (frp2 == frp_close->fr_next)
2718 {
2719 int row = win->w_winrow;
Bram Moolenaar53f81742017-09-22 14:35:51 +02002720 int col = win->w_wincol;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002721
2722 frame_comp_pos(frp2, &row, &col);
2723 }
2724
2725 if (frp2->fr_next == NULL && frp2->fr_prev == NULL)
2726 {
2727 /* There is no other frame in this list, move its info to the parent
2728 * and remove it. */
2729 frp2->fr_parent->fr_layout = frp2->fr_layout;
2730 frp2->fr_parent->fr_child = frp2->fr_child;
2731 for (frp = frp2->fr_child; frp != NULL; frp = frp->fr_next)
2732 frp->fr_parent = frp2->fr_parent;
2733 frp2->fr_parent->fr_win = frp2->fr_win;
2734 if (frp2->fr_win != NULL)
2735 frp2->fr_win->w_frame = frp2->fr_parent;
2736 frp = frp2->fr_parent;
Bram Moolenaar6f361c92018-01-31 19:06:50 +01002737 if (topframe->fr_child == frp2)
2738 topframe->fr_child = frp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002739 vim_free(frp2);
2740
2741 frp2 = frp->fr_parent;
2742 if (frp2 != NULL && frp2->fr_layout == frp->fr_layout)
2743 {
2744 /* The frame above the parent has the same layout, have to merge
2745 * the frames into this list. */
2746 if (frp2->fr_child == frp)
2747 frp2->fr_child = frp->fr_child;
2748 frp->fr_child->fr_prev = frp->fr_prev;
2749 if (frp->fr_prev != NULL)
2750 frp->fr_prev->fr_next = frp->fr_child;
2751 for (frp3 = frp->fr_child; ; frp3 = frp3->fr_next)
2752 {
2753 frp3->fr_parent = frp2;
2754 if (frp3->fr_next == NULL)
2755 {
2756 frp3->fr_next = frp->fr_next;
2757 if (frp->fr_next != NULL)
2758 frp->fr_next->fr_prev = frp3;
2759 break;
2760 }
2761 }
Bram Moolenaar6f361c92018-01-31 19:06:50 +01002762 if (topframe->fr_child == frp)
2763 topframe->fr_child = frp2;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002764 vim_free(frp);
2765 }
2766 }
2767
2768 return wp;
2769}
2770
2771/*
Bram Moolenaarc136af22018-05-04 20:15:38 +02002772 * Return a pointer to the frame that will receive the empty screen space that
2773 * is left over after "win" is closed.
2774 *
2775 * If 'splitbelow' or 'splitright' is set, the space goes above or to the left
2776 * by default. Otherwise, the free space goes below or to the right. The
2777 * result is that opening a window and then immediately closing it will
2778 * preserve the initial window layout. The 'wfh' and 'wfw' settings are
2779 * respected when possible.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002780 */
2781 static frame_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002782win_altframe(
2783 win_T *win,
2784 tabpage_T *tp) /* tab page "win" is in, NULL for current */
Bram Moolenaar071d4272004-06-13 20:20:40 +00002785{
2786 frame_T *frp;
Bram Moolenaarc136af22018-05-04 20:15:38 +02002787 frame_T *other_fr, *target_fr;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002788
Bram Moolenaara1f4cb92016-11-06 15:25:42 +01002789 if (tp == NULL ? ONE_WINDOW : tp->tp_firstwin == tp->tp_lastwin)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002790 return alt_tabpage()->tp_curwin->w_frame;
2791
Bram Moolenaar071d4272004-06-13 20:20:40 +00002792 frp = win->w_frame;
Bram Moolenaarc136af22018-05-04 20:15:38 +02002793
2794 if (frp->fr_prev == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002795 return frp->fr_next;
Bram Moolenaarc136af22018-05-04 20:15:38 +02002796 if (frp->fr_next == NULL)
2797 return frp->fr_prev;
2798
2799 target_fr = frp->fr_next;
2800 other_fr = frp->fr_prev;
2801 if (p_spr || p_sb)
2802 {
2803 target_fr = frp->fr_prev;
2804 other_fr = frp->fr_next;
2805 }
2806
2807 /* If 'wfh' or 'wfw' is set for the target and not for the alternate
2808 * window, reverse the selection. */
2809 if (frp->fr_parent != NULL && frp->fr_parent->fr_layout == FR_ROW)
2810 {
2811 if (frame_fixed_width(target_fr) && !frame_fixed_width(other_fr))
2812 target_fr = other_fr;
2813 }
2814 else
2815 {
2816 if (frame_fixed_height(target_fr) && !frame_fixed_height(other_fr))
2817 target_fr = other_fr;
2818 }
2819
2820 return target_fr;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002821}
2822
2823/*
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002824 * Return the tabpage that will be used if the current one is closed.
2825 */
2826 static tabpage_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002827alt_tabpage(void)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002828{
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00002829 tabpage_T *tp;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002830
Bram Moolenaar80a94a52006-02-23 21:26:58 +00002831 /* Use the next tab page if possible. */
2832 if (curtab->tp_next != NULL)
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00002833 return curtab->tp_next;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002834
Bram Moolenaar80a94a52006-02-23 21:26:58 +00002835 /* Find the last but one tab page. */
2836 for (tp = first_tabpage; tp->tp_next != curtab; tp = tp->tp_next)
2837 ;
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00002838 return tp;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00002839}
2840
2841/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00002842 * Find the left-upper window in frame "frp".
2843 */
2844 static win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002845frame2win(frame_T *frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002846{
2847 while (frp->fr_win == NULL)
2848 frp = frp->fr_child;
2849 return frp->fr_win;
2850}
2851
2852/*
2853 * Return TRUE if frame "frp" contains window "wp".
2854 */
2855 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002856frame_has_win(frame_T *frp, win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002857{
2858 frame_T *p;
2859
2860 if (frp->fr_layout == FR_LEAF)
2861 return frp->fr_win == wp;
2862
2863 for (p = frp->fr_child; p != NULL; p = p->fr_next)
2864 if (frame_has_win(p, wp))
2865 return TRUE;
2866 return FALSE;
2867}
2868
2869/*
2870 * Set a new height for a frame. Recursively sets the height for contained
2871 * frames and windows. Caller must take care of positions.
2872 */
2873 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002874frame_new_height(
2875 frame_T *topfrp,
2876 int height,
2877 int topfirst, /* resize topmost contained frame first */
2878 int wfh) /* obey 'winfixheight' when there is a choice;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002879 may cause the height not to be set */
2880{
2881 frame_T *frp;
2882 int extra_lines;
2883 int h;
2884
2885 if (topfrp->fr_win != NULL)
2886 {
2887 /* Simple case: just one window. */
2888 win_new_height(topfrp->fr_win,
Bram Moolenaard326ad62017-09-18 20:31:41 +02002889 height - topfrp->fr_win->w_status_height
Bram Moolenaar3167c3e2017-11-25 14:19:43 +01002890 - WINBAR_HEIGHT(topfrp->fr_win));
Bram Moolenaar071d4272004-06-13 20:20:40 +00002891 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002892 else if (topfrp->fr_layout == FR_ROW)
2893 {
2894 do
2895 {
2896 /* All frames in this row get the same new height. */
2897 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
2898 {
2899 frame_new_height(frp, height, topfirst, wfh);
2900 if (frp->fr_height > height)
2901 {
2902 /* Could not fit the windows, make the whole row higher. */
2903 height = frp->fr_height;
2904 break;
2905 }
2906 }
2907 }
2908 while (frp != NULL);
2909 }
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00002910 else /* fr_layout == FR_COL */
Bram Moolenaar071d4272004-06-13 20:20:40 +00002911 {
2912 /* Complicated case: Resize a column of frames. Resize the bottom
2913 * frame first, frames above that when needed. */
2914
2915 frp = topfrp->fr_child;
2916 if (wfh)
2917 /* Advance past frames with one window with 'wfh' set. */
2918 while (frame_fixed_height(frp))
2919 {
2920 frp = frp->fr_next;
2921 if (frp == NULL)
2922 return; /* no frame without 'wfh', give up */
2923 }
2924 if (!topfirst)
2925 {
2926 /* Find the bottom frame of this column */
2927 while (frp->fr_next != NULL)
2928 frp = frp->fr_next;
2929 if (wfh)
2930 /* Advance back for frames with one window with 'wfh' set. */
2931 while (frame_fixed_height(frp))
2932 frp = frp->fr_prev;
2933 }
2934
2935 extra_lines = height - topfrp->fr_height;
2936 if (extra_lines < 0)
2937 {
2938 /* reduce height of contained frames, bottom or top frame first */
2939 while (frp != NULL)
2940 {
2941 h = frame_minheight(frp, NULL);
2942 if (frp->fr_height + extra_lines < h)
2943 {
2944 extra_lines += frp->fr_height - h;
2945 frame_new_height(frp, h, topfirst, wfh);
2946 }
2947 else
2948 {
2949 frame_new_height(frp, frp->fr_height + extra_lines,
2950 topfirst, wfh);
2951 break;
2952 }
2953 if (topfirst)
2954 {
2955 do
2956 frp = frp->fr_next;
2957 while (wfh && frp != NULL && frame_fixed_height(frp));
2958 }
2959 else
2960 {
2961 do
2962 frp = frp->fr_prev;
2963 while (wfh && frp != NULL && frame_fixed_height(frp));
2964 }
2965 /* Increase "height" if we could not reduce enough frames. */
2966 if (frp == NULL)
2967 height -= extra_lines;
2968 }
2969 }
2970 else if (extra_lines > 0)
2971 {
2972 /* increase height of bottom or top frame */
2973 frame_new_height(frp, frp->fr_height + extra_lines, topfirst, wfh);
2974 }
2975 }
2976 topfrp->fr_height = height;
2977}
2978
2979/*
2980 * Return TRUE if height of frame "frp" should not be changed because of
2981 * the 'winfixheight' option.
2982 */
2983 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01002984frame_fixed_height(frame_T *frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002985{
2986 /* frame with one window: fixed height if 'winfixheight' set. */
2987 if (frp->fr_win != NULL)
2988 return frp->fr_win->w_p_wfh;
2989
2990 if (frp->fr_layout == FR_ROW)
2991 {
2992 /* The frame is fixed height if one of the frames in the row is fixed
2993 * height. */
2994 for (frp = frp->fr_child; frp != NULL; frp = frp->fr_next)
2995 if (frame_fixed_height(frp))
2996 return TRUE;
2997 return FALSE;
2998 }
2999
3000 /* frp->fr_layout == FR_COL: The frame is fixed height if all of the
3001 * frames in the row are fixed height. */
3002 for (frp = frp->fr_child; frp != NULL; frp = frp->fr_next)
3003 if (!frame_fixed_height(frp))
3004 return FALSE;
3005 return TRUE;
3006}
3007
Bram Moolenaar071d4272004-06-13 20:20:40 +00003008/*
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003009 * Return TRUE if width of frame "frp" should not be changed because of
3010 * the 'winfixwidth' option.
3011 */
3012 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003013frame_fixed_width(frame_T *frp)
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003014{
3015 /* frame with one window: fixed width if 'winfixwidth' set. */
3016 if (frp->fr_win != NULL)
3017 return frp->fr_win->w_p_wfw;
3018
3019 if (frp->fr_layout == FR_COL)
3020 {
3021 /* The frame is fixed width if one of the frames in the row is fixed
3022 * width. */
3023 for (frp = frp->fr_child; frp != NULL; frp = frp->fr_next)
3024 if (frame_fixed_width(frp))
3025 return TRUE;
3026 return FALSE;
3027 }
3028
3029 /* frp->fr_layout == FR_ROW: The frame is fixed width if all of the
3030 * frames in the row are fixed width. */
3031 for (frp = frp->fr_child; frp != NULL; frp = frp->fr_next)
3032 if (!frame_fixed_width(frp))
3033 return FALSE;
3034 return TRUE;
3035}
3036
3037/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00003038 * Add a status line to windows at the bottom of "frp".
3039 * Note: Does not check if there is room!
3040 */
3041 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003042frame_add_statusline(frame_T *frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003043{
3044 win_T *wp;
3045
3046 if (frp->fr_layout == FR_LEAF)
3047 {
3048 wp = frp->fr_win;
3049 if (wp->w_status_height == 0)
3050 {
3051 if (wp->w_height > 0) /* don't make it negative */
3052 --wp->w_height;
3053 wp->w_status_height = STATUS_HEIGHT;
3054 }
3055 }
3056 else if (frp->fr_layout == FR_ROW)
3057 {
3058 /* Handle all the frames in the row. */
3059 for (frp = frp->fr_child; frp != NULL; frp = frp->fr_next)
3060 frame_add_statusline(frp);
3061 }
3062 else /* frp->fr_layout == FR_COL */
3063 {
3064 /* Only need to handle the last frame in the column. */
3065 for (frp = frp->fr_child; frp->fr_next != NULL; frp = frp->fr_next)
3066 ;
3067 frame_add_statusline(frp);
3068 }
3069}
3070
3071/*
3072 * Set width of a frame. Handles recursively going through contained frames.
3073 * May remove separator line for windows at the right side (for win_close()).
3074 */
3075 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003076frame_new_width(
3077 frame_T *topfrp,
3078 int width,
3079 int leftfirst, /* resize leftmost contained frame first */
3080 int wfw) /* obey 'winfixwidth' when there is a choice;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003081 may cause the width not to be set */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003082{
3083 frame_T *frp;
3084 int extra_cols;
3085 int w;
3086 win_T *wp;
3087
3088 if (topfrp->fr_layout == FR_LEAF)
3089 {
3090 /* Simple case: just one window. */
3091 wp = topfrp->fr_win;
3092 /* Find out if there are any windows right of this one. */
3093 for (frp = topfrp; frp->fr_parent != NULL; frp = frp->fr_parent)
3094 if (frp->fr_parent->fr_layout == FR_ROW && frp->fr_next != NULL)
3095 break;
3096 if (frp->fr_parent == NULL)
3097 wp->w_vsep_width = 0;
3098 win_new_width(wp, width - wp->w_vsep_width);
3099 }
3100 else if (topfrp->fr_layout == FR_COL)
3101 {
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003102 do
3103 {
3104 /* All frames in this column get the same new width. */
3105 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
3106 {
3107 frame_new_width(frp, width, leftfirst, wfw);
3108 if (frp->fr_width > width)
3109 {
3110 /* Could not fit the windows, make whole column wider. */
3111 width = frp->fr_width;
3112 break;
3113 }
3114 }
3115 } while (frp != NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003116 }
3117 else /* fr_layout == FR_ROW */
3118 {
3119 /* Complicated case: Resize a row of frames. Resize the rightmost
3120 * frame first, frames left of it when needed. */
3121
Bram Moolenaar071d4272004-06-13 20:20:40 +00003122 frp = topfrp->fr_child;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003123 if (wfw)
3124 /* Advance past frames with one window with 'wfw' set. */
3125 while (frame_fixed_width(frp))
3126 {
3127 frp = frp->fr_next;
3128 if (frp == NULL)
3129 return; /* no frame without 'wfw', give up */
3130 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003131 if (!leftfirst)
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003132 {
3133 /* Find the rightmost frame of this row */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003134 while (frp->fr_next != NULL)
3135 frp = frp->fr_next;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003136 if (wfw)
3137 /* Advance back for frames with one window with 'wfw' set. */
3138 while (frame_fixed_width(frp))
3139 frp = frp->fr_prev;
3140 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003141
3142 extra_cols = width - topfrp->fr_width;
3143 if (extra_cols < 0)
3144 {
3145 /* reduce frame width, rightmost frame first */
3146 while (frp != NULL)
3147 {
3148 w = frame_minwidth(frp, NULL);
3149 if (frp->fr_width + extra_cols < w)
3150 {
3151 extra_cols += frp->fr_width - w;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003152 frame_new_width(frp, w, leftfirst, wfw);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003153 }
3154 else
3155 {
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003156 frame_new_width(frp, frp->fr_width + extra_cols,
3157 leftfirst, wfw);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003158 break;
3159 }
3160 if (leftfirst)
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003161 {
3162 do
3163 frp = frp->fr_next;
3164 while (wfw && frp != NULL && frame_fixed_width(frp));
3165 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003166 else
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003167 {
3168 do
3169 frp = frp->fr_prev;
3170 while (wfw && frp != NULL && frame_fixed_width(frp));
3171 }
3172 /* Increase "width" if we could not reduce enough frames. */
3173 if (frp == NULL)
3174 width -= extra_cols;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003175 }
3176 }
3177 else if (extra_cols > 0)
3178 {
3179 /* increase width of rightmost frame */
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00003180 frame_new_width(frp, frp->fr_width + extra_cols, leftfirst, wfw);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003181 }
3182 }
3183 topfrp->fr_width = width;
3184}
3185
3186/*
3187 * Add the vertical separator to windows at the right side of "frp".
3188 * Note: Does not check if there is room!
3189 */
3190 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003191frame_add_vsep(frame_T *frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003192{
3193 win_T *wp;
3194
3195 if (frp->fr_layout == FR_LEAF)
3196 {
3197 wp = frp->fr_win;
3198 if (wp->w_vsep_width == 0)
3199 {
3200 if (wp->w_width > 0) /* don't make it negative */
3201 --wp->w_width;
3202 wp->w_vsep_width = 1;
3203 }
3204 }
3205 else if (frp->fr_layout == FR_COL)
3206 {
3207 /* Handle all the frames in the column. */
3208 for (frp = frp->fr_child; frp != NULL; frp = frp->fr_next)
3209 frame_add_vsep(frp);
3210 }
3211 else /* frp->fr_layout == FR_ROW */
3212 {
3213 /* Only need to handle the last frame in the row. */
3214 frp = frp->fr_child;
3215 while (frp->fr_next != NULL)
3216 frp = frp->fr_next;
3217 frame_add_vsep(frp);
3218 }
3219}
3220
3221/*
3222 * Set frame width from the window it contains.
3223 */
3224 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003225frame_fix_width(win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003226{
3227 wp->w_frame->fr_width = wp->w_width + wp->w_vsep_width;
3228}
Bram Moolenaar071d4272004-06-13 20:20:40 +00003229
3230/*
3231 * Set frame height from the window it contains.
3232 */
3233 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003234frame_fix_height(win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003235{
Bram Moolenaar415a6932017-12-05 20:31:07 +01003236 wp->w_frame->fr_height = VISIBLE_HEIGHT(wp) + wp->w_status_height;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003237}
3238
3239/*
3240 * Compute the minimal height for frame "topfrp".
3241 * Uses the 'winminheight' option.
3242 * When "next_curwin" isn't NULL, use p_wh for this window.
3243 * When "next_curwin" is NOWIN, don't use at least one line for the current
3244 * window.
3245 */
3246 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003247frame_minheight(frame_T *topfrp, win_T *next_curwin)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003248{
3249 frame_T *frp;
3250 int m;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003251 int n;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003252
3253 if (topfrp->fr_win != NULL)
3254 {
3255 if (topfrp->fr_win == next_curwin)
3256 m = p_wh + topfrp->fr_win->w_status_height;
3257 else
3258 {
3259 /* window: minimal height of the window plus status line */
3260 m = p_wmh + topfrp->fr_win->w_status_height;
Bram Moolenaar415a6932017-12-05 20:31:07 +01003261 if (topfrp->fr_win == curwin && next_curwin == NULL)
3262 {
3263 /* Current window is minimal one line high and WinBar is
3264 * visible. */
3265 if (p_wmh == 0)
3266 ++m;
3267 m += WINBAR_HEIGHT(curwin);
3268 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003269 }
3270 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003271 else if (topfrp->fr_layout == FR_ROW)
3272 {
3273 /* get the minimal height from each frame in this row */
3274 m = 0;
3275 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
3276 {
3277 n = frame_minheight(frp, next_curwin);
3278 if (n > m)
3279 m = n;
3280 }
3281 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003282 else
3283 {
3284 /* Add up the minimal heights for all frames in this column. */
3285 m = 0;
3286 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
3287 m += frame_minheight(frp, next_curwin);
3288 }
3289
3290 return m;
3291}
3292
Bram Moolenaar071d4272004-06-13 20:20:40 +00003293/*
3294 * Compute the minimal width for frame "topfrp".
3295 * When "next_curwin" isn't NULL, use p_wiw for this window.
3296 * When "next_curwin" is NOWIN, don't use at least one column for the current
3297 * window.
3298 */
3299 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003300frame_minwidth(
3301 frame_T *topfrp,
3302 win_T *next_curwin) /* use p_wh and p_wiw for next_curwin */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003303{
3304 frame_T *frp;
3305 int m, n;
3306
3307 if (topfrp->fr_win != NULL)
3308 {
3309 if (topfrp->fr_win == next_curwin)
3310 m = p_wiw + topfrp->fr_win->w_vsep_width;
3311 else
3312 {
3313 /* window: minimal width of the window plus separator column */
3314 m = p_wmw + topfrp->fr_win->w_vsep_width;
3315 /* Current window is minimal one column wide */
3316 if (p_wmw == 0 && topfrp->fr_win == curwin && next_curwin == NULL)
3317 ++m;
3318 }
3319 }
3320 else if (topfrp->fr_layout == FR_COL)
3321 {
3322 /* get the minimal width from each frame in this column */
3323 m = 0;
3324 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
3325 {
3326 n = frame_minwidth(frp, next_curwin);
3327 if (n > m)
3328 m = n;
3329 }
3330 }
3331 else
3332 {
3333 /* Add up the minimal widths for all frames in this row. */
3334 m = 0;
3335 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
3336 m += frame_minwidth(frp, next_curwin);
3337 }
3338
3339 return m;
3340}
Bram Moolenaar071d4272004-06-13 20:20:40 +00003341
3342
3343/*
3344 * Try to close all windows except current one.
3345 * Buffers in the other windows become hidden if 'hidden' is set, or '!' is
3346 * used and the buffer was modified.
3347 *
3348 * Used by ":bdel" and ":only".
3349 */
3350 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003351close_others(
3352 int message,
3353 int forceit) /* always hide all other windows */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003354{
3355 win_T *wp;
3356 win_T *nextwp;
3357 int r;
3358
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003359 if (one_window())
Bram Moolenaar071d4272004-06-13 20:20:40 +00003360 {
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01003361 if (message && !autocmd_busy)
Bram Moolenaar4c3f5362006-04-11 21:38:50 +00003362 MSG(_(m_onlyone));
Bram Moolenaar071d4272004-06-13 20:20:40 +00003363 return;
3364 }
3365
3366 /* Be very careful here: autocommands may change the window layout. */
3367 for (wp = firstwin; win_valid(wp); wp = nextwp)
3368 {
3369 nextwp = wp->w_next;
3370 if (wp != curwin) /* don't close current window */
3371 {
3372
3373 /* Check if it's allowed to abandon this window */
3374 r = can_abandon(wp->w_buffer, forceit);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003375 if (!win_valid(wp)) /* autocommands messed wp up */
3376 {
3377 nextwp = firstwin;
3378 continue;
3379 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003380 if (!r)
3381 {
3382#if defined(FEAT_GUI_DIALOG) || defined(FEAT_CON_DIALOG)
3383 if (message && (p_confirm || cmdmod.confirm) && p_write)
3384 {
3385 dialog_changed(wp->w_buffer, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003386 if (!win_valid(wp)) /* autocommands messed wp up */
3387 {
3388 nextwp = firstwin;
3389 continue;
3390 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003391 }
3392 if (bufIsChanged(wp->w_buffer))
3393#endif
3394 continue;
3395 }
Bram Moolenaareb44a682017-08-03 22:44:55 +02003396 win_close(wp, !buf_hide(wp->w_buffer)
3397 && !bufIsChanged(wp->w_buffer));
Bram Moolenaar071d4272004-06-13 20:20:40 +00003398 }
3399 }
3400
Bram Moolenaar459ca562016-11-10 18:16:33 +01003401 if (message && !ONE_WINDOW)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003402 EMSG(_("E445: Other window contains changes"));
3403}
3404
Bram Moolenaar071d4272004-06-13 20:20:40 +00003405/*
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003406 * Init the current window "curwin".
3407 * Called when a new file is being edited.
Bram Moolenaar071d4272004-06-13 20:20:40 +00003408 */
3409 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003410curwin_init(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003411{
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003412 win_init_empty(curwin);
3413}
3414
3415 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003416win_init_empty(win_T *wp)
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003417{
3418 redraw_win_later(wp, NOT_VALID);
3419 wp->w_lines_valid = 0;
3420 wp->w_cursor.lnum = 1;
3421 wp->w_curswant = wp->w_cursor.col = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003422#ifdef FEAT_VIRTUALEDIT
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003423 wp->w_cursor.coladd = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003424#endif
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003425 wp->w_pcmark.lnum = 1; /* pcmark not cleared but set to line 1 */
3426 wp->w_pcmark.col = 0;
3427 wp->w_prev_pcmark.lnum = 0;
3428 wp->w_prev_pcmark.col = 0;
3429 wp->w_topline = 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003430#ifdef FEAT_DIFF
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003431 wp->w_topfill = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003432#endif
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003433 wp->w_botline = 2;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003434#ifdef FEAT_FKMAP
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003435 if (wp->w_p_rl)
3436 wp->w_farsi = W_CONV + W_R_L;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003437 else
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00003438 wp->w_farsi = W_CONV;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003439#endif
Bram Moolenaara971b822011-09-14 14:43:25 +02003440#ifdef FEAT_SYN_HL
3441 wp->w_s = &wp->w_buffer->b_s;
3442#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00003443}
3444
3445/*
3446 * Allocate the first window and put an empty buffer in it.
3447 * Called from main().
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003448 * Return FAIL when something goes wrong (out of memory).
Bram Moolenaar071d4272004-06-13 20:20:40 +00003449 */
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003450 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003451win_alloc_first(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003452{
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003453 if (win_alloc_firstwin(NULL) == FAIL)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003454 return FAIL;
3455
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003456 first_tabpage = alloc_tabpage();
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003457 if (first_tabpage == NULL)
3458 return FAIL;
3459 first_tabpage->tp_topframe = topframe;
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003460 curtab = first_tabpage;
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003461
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003462 return OK;
3463}
3464
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003465/*
3466 * Init "aucmd_win". This can only be done after the first
3467 * window is fully initialized, thus it can't be in win_alloc_first().
3468 */
3469 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003470win_alloc_aucmd_win(void)
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003471{
3472 aucmd_win = win_alloc(NULL, TRUE);
3473 if (aucmd_win != NULL)
3474 {
3475 win_init_some(aucmd_win, curwin);
Bram Moolenaar3368ea22010-09-21 16:56:35 +02003476 RESET_BINDING(aucmd_win);
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003477 new_frame(aucmd_win);
3478 }
3479}
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003480
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003481/*
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003482 * Allocate the first window or the first window in a new tab page.
3483 * When "oldwin" is NULL create an empty buffer for it.
Bram Moolenaar4033c552017-09-16 20:54:51 +02003484 * When "oldwin" is not NULL copy info from it to the new window.
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003485 * Return FAIL when something goes wrong (out of memory).
3486 */
3487 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003488win_alloc_firstwin(win_T *oldwin)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003489{
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003490 curwin = win_alloc(NULL, FALSE);
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003491 if (oldwin == NULL)
3492 {
3493 /* Very first window, need to create an empty buffer for it and
3494 * initialize from scratch. */
3495 curbuf = buflist_new(NULL, NULL, 1L, BLN_LISTED);
3496 if (curwin == NULL || curbuf == NULL)
3497 return FAIL;
3498 curwin->w_buffer = curbuf;
Bram Moolenaar860cae12010-06-05 23:22:07 +02003499#ifdef FEAT_SYN_HL
3500 curwin->w_s = &(curbuf->b_s);
3501#endif
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003502 curbuf->b_nwindows = 1; /* there is one window */
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003503 curwin->w_alist = &global_alist;
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003504 curwin_init(); /* init current window */
3505 }
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003506 else
3507 {
3508 /* First window in new tab page, initialize it from "oldwin". */
Bram Moolenaar884ae642009-02-22 01:37:59 +00003509 win_init(curwin, oldwin, 0);
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003510
Bram Moolenaar3368ea22010-09-21 16:56:35 +02003511 /* We don't want cursor- and scroll-binding in the first window. */
3512 RESET_BINDING(curwin);
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003513 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003514
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003515 new_frame(curwin);
3516 if (curwin->w_frame == NULL)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003517 return FAIL;
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003518 topframe = curwin->w_frame;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003519 topframe->fr_width = Columns;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003520 topframe->fr_height = Rows - p_ch;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003521
3522 return OK;
3523}
3524
3525/*
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003526 * Create a frame for window "wp".
3527 */
3528 static void
3529new_frame(win_T *wp)
3530{
3531 frame_T *frp = (frame_T *)alloc_clear((unsigned)sizeof(frame_T));
3532
3533 wp->w_frame = frp;
3534 if (frp != NULL)
3535 {
3536 frp->fr_layout = FR_LEAF;
3537 frp->fr_win = wp;
3538 }
3539}
3540
3541/*
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003542 * Initialize the window and frame size to the maximum.
3543 */
3544 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003545win_init_size(void)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003546{
3547 firstwin->w_height = ROWS_AVAIL;
3548 topframe->fr_height = ROWS_AVAIL;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003549 firstwin->w_width = Columns;
3550 topframe->fr_width = Columns;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003551}
3552
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003553/*
3554 * Allocate a new tabpage_T and init the values.
3555 * Returns NULL when out of memory.
3556 */
3557 static tabpage_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003558alloc_tabpage(void)
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003559{
3560 tabpage_T *tp;
Bram Moolenaar429fa852013-04-15 12:27:36 +02003561# ifdef FEAT_GUI
3562 int i;
3563# endif
3564
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003565
3566 tp = (tabpage_T *)alloc_clear((unsigned)sizeof(tabpage_T));
Bram Moolenaar429fa852013-04-15 12:27:36 +02003567 if (tp == NULL)
3568 return NULL;
Bram Moolenaar371d5402006-03-20 21:47:49 +00003569
Bram Moolenaar429fa852013-04-15 12:27:36 +02003570# ifdef FEAT_EVAL
3571 /* init t: variables */
3572 tp->tp_vars = dict_alloc();
3573 if (tp->tp_vars == NULL)
3574 {
3575 vim_free(tp);
3576 return NULL;
3577 }
3578 init_var_dict(tp->tp_vars, &tp->tp_winvar, VAR_SCOPE);
3579# endif
3580
3581# ifdef FEAT_GUI
3582 for (i = 0; i < 3; i++)
3583 tp->tp_prev_which_scrollbars[i] = -1;
Bram Moolenaar371d5402006-03-20 21:47:49 +00003584# endif
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003585# ifdef FEAT_DIFF
Bram Moolenaar429fa852013-04-15 12:27:36 +02003586 tp->tp_diff_invalid = TRUE;
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003587# endif
Bram Moolenaar429fa852013-04-15 12:27:36 +02003588 tp->tp_ch_used = p_ch;
3589
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003590 return tp;
3591}
3592
Bram Moolenaard8fc5c02006-04-29 21:55:22 +00003593 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003594free_tabpage(tabpage_T *tp)
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003595{
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003596 int idx;
3597
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003598# ifdef FEAT_DIFF
3599 diff_clear(tp);
3600# endif
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003601 for (idx = 0; idx < SNAP_COUNT; ++idx)
3602 clear_snapshot(tp, idx);
Bram Moolenaar910f66f2006-04-05 20:41:53 +00003603#ifdef FEAT_EVAL
Bram Moolenaar429fa852013-04-15 12:27:36 +02003604 vars_clear(&tp->tp_vars->dv_hashtab); /* free all t: variables */
3605 hash_init(&tp->tp_vars->dv_hashtab);
3606 unref_var_dict(tp->tp_vars);
Bram Moolenaar910f66f2006-04-05 20:41:53 +00003607#endif
Bram Moolenaar5e538ec2013-05-15 15:12:29 +02003608
3609#ifdef FEAT_PYTHON
3610 python_tabpage_free(tp);
3611#endif
3612
3613#ifdef FEAT_PYTHON3
3614 python3_tabpage_free(tp);
3615#endif
3616
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003617 vim_free(tp);
3618}
3619
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003620/*
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003621 * Create a new Tab page with one window.
3622 * It will edit the current buffer, like after ":split".
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003623 * When "after" is 0 put it just after the current Tab page.
3624 * Otherwise put it just before tab page "after".
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003625 * Return FAIL or OK.
3626 */
3627 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003628win_new_tabpage(int after)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003629{
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003630 tabpage_T *tp = curtab;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003631 tabpage_T *newtp;
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003632 int n;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003633
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003634 newtp = alloc_tabpage();
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003635 if (newtp == NULL)
3636 return FAIL;
3637
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003638 /* Remember the current windows in this Tab page. */
Bram Moolenaar49e649f2013-05-06 04:50:35 +02003639 if (leave_tabpage(curbuf, TRUE) == FAIL)
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003640 {
3641 vim_free(newtp);
3642 return FAIL;
3643 }
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003644 curtab = newtp;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003645
3646 /* Create a new empty window. */
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003647 if (win_alloc_firstwin(tp->tp_curwin) == OK)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003648 {
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003649 /* Make the new Tab page the new topframe. */
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003650 if (after == 1)
3651 {
3652 /* New tab page becomes the first one. */
3653 newtp->tp_next = first_tabpage;
3654 first_tabpage = newtp;
3655 }
3656 else
3657 {
3658 if (after > 0)
3659 {
3660 /* Put new tab page before tab page "after". */
3661 n = 2;
3662 for (tp = first_tabpage; tp->tp_next != NULL
3663 && n < after; tp = tp->tp_next)
3664 ++n;
3665 }
3666 newtp->tp_next = tp->tp_next;
3667 tp->tp_next = newtp;
3668 }
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003669 win_init_size();
Bram Moolenaar32466aa2006-02-24 23:53:04 +00003670 firstwin->w_winrow = tabline_height();
Bram Moolenaar910f66f2006-04-05 20:41:53 +00003671 win_comp_scroll(curwin);
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003672
3673 newtp->tp_topframe = topframe;
Bram Moolenaarf740b292006-02-16 22:11:02 +00003674 last_status(FALSE);
Bram Moolenaar607a95ed2006-03-28 20:57:42 +00003675
3676#if defined(FEAT_GUI)
3677 /* When 'guioptions' includes 'L' or 'R' may have to remove or add
3678 * scrollbars. Have to update them anyway. */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003679 gui_may_update_scrollbars();
Bram Moolenaar607a95ed2006-03-28 20:57:42 +00003680#endif
Bram Moolenaar6d41c782018-06-06 09:11:12 +02003681#ifdef FEAT_JOB_CHANNEL
3682 entering_window(curwin);
3683#endif
Bram Moolenaar607a95ed2006-03-28 20:57:42 +00003684
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003685 redraw_all_later(CLEAR);
Bram Moolenaarc917da42016-07-19 22:31:36 +02003686 apply_autocmds(EVENT_WINNEW, NULL, NULL, FALSE, curbuf);
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003687 apply_autocmds(EVENT_WINENTER, NULL, NULL, FALSE, curbuf);
Bram Moolenaarc917da42016-07-19 22:31:36 +02003688 apply_autocmds(EVENT_TABNEW, NULL, NULL, FALSE, curbuf);
Bram Moolenaar49e649f2013-05-06 04:50:35 +02003689 apply_autocmds(EVENT_TABENTER, NULL, NULL, FALSE, curbuf);
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003690 return OK;
3691 }
3692
3693 /* Failed, get back the previous Tab page */
Bram Moolenaar49e649f2013-05-06 04:50:35 +02003694 enter_tabpage(curtab, curbuf, TRUE, TRUE);
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003695 return FAIL;
3696}
3697
3698/*
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003699 * Open a new tab page if ":tab cmd" was used. It will edit the same buffer,
3700 * like with ":split".
3701 * Returns OK if a new tab page was created, FAIL otherwise.
3702 */
3703 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003704may_open_tabpage(void)
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003705{
Bram Moolenaard326ce82007-03-11 14:48:29 +00003706 int n = (cmdmod.tab == 0) ? postponed_split_tab : cmdmod.tab;
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003707
Bram Moolenaard326ce82007-03-11 14:48:29 +00003708 if (n != 0)
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003709 {
3710 cmdmod.tab = 0; /* reset it to avoid doing it twice */
Bram Moolenaard326ce82007-03-11 14:48:29 +00003711 postponed_split_tab = 0;
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003712 return win_new_tabpage(n);
3713 }
3714 return FAIL;
3715}
3716
3717/*
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003718 * Create up to "maxcount" tabpages with empty windows.
3719 * Returns the number of resulting tab pages.
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003720 */
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003721 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003722make_tabpages(int maxcount)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003723{
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003724 int count = maxcount;
3725 int todo;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003726
Bram Moolenaare1438bb2006-03-01 22:01:55 +00003727 /* Limit to 'tabpagemax' tabs. */
3728 if (count > p_tpm)
3729 count = p_tpm;
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003730
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003731 /*
3732 * Don't execute autocommands while creating the tab pages. Must do that
3733 * when putting the buffers in the windows.
3734 */
Bram Moolenaar78ab3312007-09-29 12:16:41 +00003735 block_autocmds();
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003736
3737 for (todo = count - 1; todo > 0; --todo)
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003738 if (win_new_tabpage(0) == FAIL)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003739 break;
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003740
Bram Moolenaar78ab3312007-09-29 12:16:41 +00003741 unblock_autocmds();
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003742
3743 /* return actual number of tab pages */
3744 return (count - todo);
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003745}
3746
3747/*
Bram Moolenaarf740b292006-02-16 22:11:02 +00003748 * Return TRUE when "tpc" points to a valid tab page.
3749 */
3750 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003751valid_tabpage(tabpage_T *tpc)
Bram Moolenaarf740b292006-02-16 22:11:02 +00003752{
3753 tabpage_T *tp;
3754
Bram Moolenaar29323592016-07-24 22:04:11 +02003755 FOR_ALL_TABPAGES(tp)
Bram Moolenaarf740b292006-02-16 22:11:02 +00003756 if (tp == tpc)
3757 return TRUE;
3758 return FALSE;
3759}
3760
3761/*
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01003762 * Return TRUE when "tpc" points to a valid tab page and at least one window is
3763 * valid.
3764 */
3765 int
3766valid_tabpage_win(tabpage_T *tpc)
3767{
3768 tabpage_T *tp;
3769 win_T *wp;
3770
3771 FOR_ALL_TABPAGES(tp)
3772 {
3773 if (tp == tpc)
3774 {
3775 FOR_ALL_WINDOWS_IN_TAB(tp, wp)
3776 {
3777 if (win_valid_any_tab(wp))
3778 return TRUE;
3779 }
3780 return FALSE;
3781 }
3782 }
3783 /* shouldn't happen */
3784 return FALSE;
3785}
3786
3787/*
3788 * Close tabpage "tab", assuming it has no windows in it.
3789 * There must be another tabpage or this will crash.
3790 */
3791 void
3792close_tabpage(tabpage_T *tab)
3793{
3794 tabpage_T *ptp;
3795
3796 if (tab == first_tabpage)
3797 {
3798 first_tabpage = tab->tp_next;
3799 ptp = first_tabpage;
3800 }
3801 else
3802 {
3803 for (ptp = first_tabpage; ptp != NULL && ptp->tp_next != tab;
3804 ptp = ptp->tp_next)
3805 ;
Bram Moolenaara37ffaa2017-03-21 21:58:00 +01003806 assert(ptp != NULL);
Bram Moolenaar8c752bd2017-03-19 17:09:56 +01003807 ptp->tp_next = tab->tp_next;
3808 }
3809
3810 goto_tabpage_tp(ptp, FALSE, FALSE);
3811 free_tabpage(tab);
3812}
3813
3814/*
Bram Moolenaarf740b292006-02-16 22:11:02 +00003815 * Find tab page "n" (first one is 1). Returns NULL when not found.
3816 */
3817 tabpage_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003818find_tabpage(int n)
Bram Moolenaarf740b292006-02-16 22:11:02 +00003819{
3820 tabpage_T *tp;
3821 int i = 1;
3822
3823 for (tp = first_tabpage; tp != NULL && i != n; tp = tp->tp_next)
3824 ++i;
3825 return tp;
3826}
3827
3828/*
Bram Moolenaar32466aa2006-02-24 23:53:04 +00003829 * Get index of tab page "tp". First one has index 1.
Bram Moolenaarba6c0522006-02-25 21:45:02 +00003830 * When not found returns number of tab pages plus one.
Bram Moolenaar32466aa2006-02-24 23:53:04 +00003831 */
3832 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003833tabpage_index(tabpage_T *ftp)
Bram Moolenaar32466aa2006-02-24 23:53:04 +00003834{
3835 int i = 1;
3836 tabpage_T *tp;
3837
3838 for (tp = first_tabpage; tp != NULL && tp != ftp; tp = tp->tp_next)
3839 ++i;
3840 return i;
3841}
3842
3843/*
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003844 * Prepare for leaving the current tab page.
Bram Moolenaar84a05ac2013-05-06 04:24:17 +02003845 * When autocommands change "curtab" we don't leave the tab page and return
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003846 * FAIL.
3847 * Careful: When OK is returned need to get a new tab page very very soon!
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003848 */
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003849 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003850leave_tabpage(
3851 buf_T *new_curbuf UNUSED, /* what is going to be the new curbuf,
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003852 NULL if unknown */
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003853 int trigger_leave_autocmds UNUSED)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003854{
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003855 tabpage_T *tp = curtab;
3856
Bram Moolenaar6d41c782018-06-06 09:11:12 +02003857#ifdef FEAT_JOB_CHANNEL
3858 leaving_window(curwin);
3859#endif
Bram Moolenaar910f66f2006-04-05 20:41:53 +00003860 reset_VIsual_and_resel(); /* stop Visual mode */
Bram Moolenaar49e649f2013-05-06 04:50:35 +02003861 if (trigger_leave_autocmds)
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003862 {
Bram Moolenaar49e649f2013-05-06 04:50:35 +02003863 if (new_curbuf != curbuf)
3864 {
3865 apply_autocmds(EVENT_BUFLEAVE, NULL, NULL, FALSE, curbuf);
3866 if (curtab != tp)
3867 return FAIL;
3868 }
3869 apply_autocmds(EVENT_WINLEAVE, NULL, NULL, FALSE, curbuf);
3870 if (curtab != tp)
3871 return FAIL;
3872 apply_autocmds(EVENT_TABLEAVE, NULL, NULL, FALSE, curbuf);
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003873 if (curtab != tp)
3874 return FAIL;
3875 }
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003876#if defined(FEAT_GUI)
3877 /* Remove the scrollbars. They may be added back later. */
3878 if (gui.in_use)
3879 gui_remove_scrollbars();
3880#endif
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003881 tp->tp_curwin = curwin;
Bram Moolenaarf740b292006-02-16 22:11:02 +00003882 tp->tp_prevwin = prevwin;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003883 tp->tp_firstwin = firstwin;
3884 tp->tp_lastwin = lastwin;
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003885 tp->tp_old_Rows = Rows;
3886 tp->tp_old_Columns = Columns;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003887 firstwin = NULL;
3888 lastwin = NULL;
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003889 return OK;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003890}
3891
3892/*
3893 * Start using tab page "tp".
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003894 * Only to be used after leave_tabpage() or freeing the current tab page.
Bram Moolenaar49e649f2013-05-06 04:50:35 +02003895 * Only trigger *Enter autocommands when trigger_enter_autocmds is TRUE.
3896 * Only trigger *Leave autocommands when trigger_leave_autocmds is TRUE.
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003897 */
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003898 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003899enter_tabpage(
3900 tabpage_T *tp,
3901 buf_T *old_curbuf UNUSED,
Bram Moolenaarf1d25012016-03-03 12:22:53 +01003902 int trigger_enter_autocmds,
3903 int trigger_leave_autocmds)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003904{
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003905 int old_off = tp->tp_firstwin->w_winrow;
Bram Moolenaar773560b2006-05-06 21:38:18 +00003906 win_T *next_prevwin = tp->tp_prevwin;
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003907
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003908 curtab = tp;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003909 firstwin = tp->tp_firstwin;
3910 lastwin = tp->tp_lastwin;
3911 topframe = tp->tp_topframe;
Bram Moolenaar773560b2006-05-06 21:38:18 +00003912
3913 /* We would like doing the TabEnter event first, but we don't have a
3914 * valid current window yet, which may break some commands.
3915 * This triggers autocommands, thus may make "tp" invalid. */
Bram Moolenaarc917da42016-07-19 22:31:36 +02003916 win_enter_ext(tp->tp_curwin, FALSE, TRUE, FALSE,
Bram Moolenaard6949742013-06-16 14:18:28 +02003917 trigger_enter_autocmds, trigger_leave_autocmds);
Bram Moolenaar773560b2006-05-06 21:38:18 +00003918 prevwin = next_prevwin;
3919
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003920 last_status(FALSE); /* status line may appear or disappear */
3921 (void)win_comp_pos(); /* recompute w_winrow for all windows */
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00003922 must_redraw = CLEAR; /* need to redraw everything */
3923#ifdef FEAT_DIFF
3924 diff_need_scrollbind = TRUE;
3925#endif
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003926
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003927 /* The tabpage line may have appeared or disappeared, may need to resize
3928 * the frames for that. When the Vim window was resized need to update
Bram Moolenaarc6fe9192006-04-09 21:54:49 +00003929 * frame sizes too. Use the stored value of p_ch, so that it can be
3930 * different for each tab page. */
3931 p_ch = curtab->tp_ch_used;
Bram Moolenaar32466aa2006-02-24 23:53:04 +00003932 if (curtab->tp_old_Rows != Rows || (old_off != firstwin->w_winrow
3933#ifdef FEAT_GUI_TABLINE
3934 && !gui_use_tabline()
3935#endif
3936 ))
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003937 shell_new_rows();
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003938 if (curtab->tp_old_Columns != Columns && starting == 0)
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003939 shell_new_columns(); /* update window widths */
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00003940
3941#if defined(FEAT_GUI)
3942 /* When 'guioptions' includes 'L' or 'R' may have to remove or add
3943 * scrollbars. Have to update them anyway. */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00003944 gui_may_update_scrollbars();
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003945#endif
3946
Bram Moolenaar5ad15df2012-03-16 19:07:58 +01003947 /* Apply autocommands after updating the display, when 'rows' and
3948 * 'columns' have been set correctly. */
Bram Moolenaar49e649f2013-05-06 04:50:35 +02003949 if (trigger_enter_autocmds)
Bram Moolenaara8596c42012-06-13 14:28:20 +02003950 {
3951 apply_autocmds(EVENT_TABENTER, NULL, NULL, FALSE, curbuf);
3952 if (old_curbuf != curbuf)
3953 apply_autocmds(EVENT_BUFENTER, NULL, NULL, FALSE, curbuf);
3954 }
Bram Moolenaar5ad15df2012-03-16 19:07:58 +01003955
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003956 redraw_all_later(CLEAR);
3957}
3958
3959/*
3960 * Go to tab page "n". For ":tab N" and "Ngt".
Bram Moolenaar32466aa2006-02-24 23:53:04 +00003961 * When "n" is 9999 go to the last tab page.
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003962 */
3963 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01003964goto_tabpage(int n)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003965{
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003966 tabpage_T *tp;
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003967 tabpage_T *ttp;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003968 int i;
3969
Bram Moolenaard68071d2006-05-02 22:08:30 +00003970 if (text_locked())
3971 {
3972 /* Not allowed when editing the command line. */
Bram Moolenaar5a497892016-09-03 16:29:04 +02003973 text_locked_msg();
Bram Moolenaard68071d2006-05-02 22:08:30 +00003974 return;
3975 }
3976
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00003977 /* If there is only one it can't work. */
3978 if (first_tabpage->tp_next == NULL)
3979 {
3980 if (n > 1)
3981 beep_flush();
3982 return;
3983 }
3984
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003985 if (n == 0)
3986 {
3987 /* No count, go to next tab page, wrap around end. */
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003988 if (curtab->tp_next == NULL)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003989 tp = first_tabpage;
3990 else
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00003991 tp = curtab->tp_next;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00003992 }
Bram Moolenaar80a94a52006-02-23 21:26:58 +00003993 else if (n < 0)
3994 {
3995 /* "gT": go to previous tab page, wrap around end. "N gT" repeats
3996 * this N times. */
3997 ttp = curtab;
3998 for (i = n; i < 0; ++i)
3999 {
4000 for (tp = first_tabpage; tp->tp_next != ttp && tp->tp_next != NULL;
4001 tp = tp->tp_next)
4002 ;
4003 ttp = tp;
4004 }
4005 }
Bram Moolenaar32466aa2006-02-24 23:53:04 +00004006 else if (n == 9999)
4007 {
4008 /* Go to last tab page. */
4009 for (tp = first_tabpage; tp->tp_next != NULL; tp = tp->tp_next)
4010 ;
4011 }
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00004012 else
4013 {
4014 /* Go to tab page "n". */
Bram Moolenaar32466aa2006-02-24 23:53:04 +00004015 tp = find_tabpage(n);
Bram Moolenaarf740b292006-02-16 22:11:02 +00004016 if (tp == NULL)
4017 {
4018 beep_flush();
4019 return;
4020 }
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00004021 }
4022
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004023 goto_tabpage_tp(tp, TRUE, TRUE);
Bram Moolenaar32466aa2006-02-24 23:53:04 +00004024
4025#ifdef FEAT_GUI_TABLINE
4026 if (gui_use_tabline())
Bram Moolenaara226a6d2006-02-26 23:59:20 +00004027 gui_mch_set_curtab(tabpage_index(curtab));
Bram Moolenaar32466aa2006-02-24 23:53:04 +00004028#endif
4029}
4030
4031/*
4032 * Go to tabpage "tp".
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004033 * Only trigger *Enter autocommands when trigger_enter_autocmds is TRUE.
4034 * Only trigger *Leave autocommands when trigger_leave_autocmds is TRUE.
Bram Moolenaar32466aa2006-02-24 23:53:04 +00004035 * Note: doesn't update the GUI tab.
4036 */
4037 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004038goto_tabpage_tp(
4039 tabpage_T *tp,
4040 int trigger_enter_autocmds,
4041 int trigger_leave_autocmds)
Bram Moolenaar32466aa2006-02-24 23:53:04 +00004042{
Bram Moolenaarc6af8122010-05-21 12:04:55 +02004043 /* Don't repeat a message in another tab page. */
4044 set_keep_msg(NULL, 0);
4045
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004046 if (tp != curtab && leave_tabpage(tp->tp_curwin->w_buffer,
4047 trigger_leave_autocmds) == OK)
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00004048 {
4049 if (valid_tabpage(tp))
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004050 enter_tabpage(tp, curbuf, trigger_enter_autocmds,
4051 trigger_leave_autocmds);
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00004052 else
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004053 enter_tabpage(curtab, curbuf, trigger_enter_autocmds,
4054 trigger_leave_autocmds);
Bram Moolenaar7e8fd632006-02-18 22:14:51 +00004055 }
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00004056}
Bram Moolenaar071d4272004-06-13 20:20:40 +00004057
4058/*
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004059 * Enter window "wp" in tab page "tp".
4060 * Also updates the GUI tab.
4061 */
4062 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004063goto_tabpage_win(tabpage_T *tp, win_T *wp)
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004064{
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004065 goto_tabpage_tp(tp, TRUE, TRUE);
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004066 if (curtab == tp && win_valid(wp))
4067 {
4068 win_enter(wp, TRUE);
4069# ifdef FEAT_GUI_TABLINE
4070 if (gui_use_tabline())
4071 gui_mch_set_curtab(tabpage_index(curtab));
4072# endif
4073 }
4074}
4075
4076/*
Bram Moolenaar40ce3a42015-04-21 18:08:39 +02004077 * Move the current tab page to after tab page "nr".
Bram Moolenaar80a94a52006-02-23 21:26:58 +00004078 */
4079 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004080tabpage_move(int nr)
Bram Moolenaar80a94a52006-02-23 21:26:58 +00004081{
Bram Moolenaar40ce3a42015-04-21 18:08:39 +02004082 int n = 1;
4083 tabpage_T *tp, *tp_dst;
Bram Moolenaar80a94a52006-02-23 21:26:58 +00004084
4085 if (first_tabpage->tp_next == NULL)
4086 return;
4087
Bram Moolenaar40ce3a42015-04-21 18:08:39 +02004088 for (tp = first_tabpage; tp->tp_next != NULL && n < nr; tp = tp->tp_next)
4089 ++n;
4090
4091 if (tp == curtab || (nr > 0 && tp->tp_next != NULL
4092 && tp->tp_next == curtab))
4093 return;
4094
4095 tp_dst = tp;
4096
Bram Moolenaar80a94a52006-02-23 21:26:58 +00004097 /* Remove the current tab page from the list of tab pages. */
4098 if (curtab == first_tabpage)
4099 first_tabpage = curtab->tp_next;
4100 else
4101 {
Bram Moolenaar29323592016-07-24 22:04:11 +02004102 FOR_ALL_TABPAGES(tp)
Bram Moolenaar80a94a52006-02-23 21:26:58 +00004103 if (tp->tp_next == curtab)
4104 break;
4105 if (tp == NULL) /* "cannot happen" */
4106 return;
4107 tp->tp_next = curtab->tp_next;
4108 }
4109
4110 /* Re-insert it at the specified position. */
Bram Moolenaar40ce3a42015-04-21 18:08:39 +02004111 if (nr <= 0)
Bram Moolenaar80a94a52006-02-23 21:26:58 +00004112 {
4113 curtab->tp_next = first_tabpage;
4114 first_tabpage = curtab;
4115 }
4116 else
4117 {
Bram Moolenaar40ce3a42015-04-21 18:08:39 +02004118 curtab->tp_next = tp_dst->tp_next;
4119 tp_dst->tp_next = curtab;
Bram Moolenaar80a94a52006-02-23 21:26:58 +00004120 }
4121
4122 /* Need to redraw the tabline. Tab page contents doesn't change. */
4123 redraw_tabline = TRUE;
4124}
4125
4126
4127/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00004128 * Go to another window.
4129 * When jumping to another buffer, stop Visual mode. Do this before
4130 * changing windows so we can yank the selection into the '*' register.
4131 * When jumping to another window on the same buffer, adjust its cursor
4132 * position to keep the same Visual area.
4133 */
4134 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004135win_goto(win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004136{
Bram Moolenaar23c347c2010-07-14 20:57:00 +02004137#ifdef FEAT_CONCEAL
4138 win_T *owp = curwin;
4139#endif
4140
Bram Moolenaar2d3f4892006-01-20 23:02:51 +00004141 if (text_locked())
Bram Moolenaar071d4272004-06-13 20:20:40 +00004142 {
4143 beep_flush();
Bram Moolenaar2d3f4892006-01-20 23:02:51 +00004144 text_locked_msg();
Bram Moolenaar071d4272004-06-13 20:20:40 +00004145 return;
4146 }
Bram Moolenaar910f66f2006-04-05 20:41:53 +00004147 if (curbuf_locked())
4148 return;
Bram Moolenaar05a7bb32006-01-19 22:09:32 +00004149
Bram Moolenaar071d4272004-06-13 20:20:40 +00004150 if (wp->w_buffer != curbuf)
4151 reset_VIsual_and_resel();
4152 else if (VIsual_active)
4153 wp->w_cursor = curwin->w_cursor;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004154
4155#ifdef FEAT_GUI
4156 need_mouse_correct = TRUE;
4157#endif
4158 win_enter(wp, TRUE);
Bram Moolenaar23c347c2010-07-14 20:57:00 +02004159
4160#ifdef FEAT_CONCEAL
4161 /* Conceal cursor line in previous window, unconceal in current window. */
Bram Moolenaar530e7df2013-02-06 13:38:02 +01004162 if (win_valid(owp) && owp->w_p_cole > 0 && !msg_scrolled)
Bram Moolenaar23c347c2010-07-14 20:57:00 +02004163 update_single_line(owp, owp->w_cursor.lnum);
Bram Moolenaar530e7df2013-02-06 13:38:02 +01004164 if (curwin->w_p_cole > 0 && !msg_scrolled)
4165 need_cursor_line_redraw = TRUE;
Bram Moolenaar23c347c2010-07-14 20:57:00 +02004166#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00004167}
4168
4169#if defined(FEAT_PERL) || defined(PROTO)
4170/*
4171 * Find window number "winnr" (counting top to bottom).
4172 */
4173 win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004174win_find_nr(int winnr)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004175{
4176 win_T *wp;
4177
Bram Moolenaar29323592016-07-24 22:04:11 +02004178 FOR_ALL_WINDOWS(wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004179 if (--winnr == 0)
4180 break;
4181 return wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004182}
4183#endif
4184
Bram Moolenaar4033c552017-09-16 20:54:51 +02004185#if ((defined(FEAT_PYTHON) || defined(FEAT_PYTHON3))) || defined(PROTO)
Bram Moolenaar105bc352013-05-17 16:03:57 +02004186/*
4187 * Find the tabpage for window "win".
4188 */
4189 tabpage_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004190win_find_tabpage(win_T *win)
Bram Moolenaar105bc352013-05-17 16:03:57 +02004191{
4192 win_T *wp;
4193 tabpage_T *tp;
4194
Bram Moolenaar29323592016-07-24 22:04:11 +02004195 FOR_ALL_TAB_WINDOWS(tp, wp)
Bram Moolenaar105bc352013-05-17 16:03:57 +02004196 if (wp == win)
4197 return tp;
4198 return NULL;
4199}
4200#endif
4201
Bram Moolenaar071d4272004-06-13 20:20:40 +00004202/*
4203 * Move to window above or below "count" times.
4204 */
4205 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004206win_goto_ver(
4207 int up, /* TRUE to go to win above */
4208 long count)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004209{
4210 frame_T *fr;
4211 frame_T *nfr;
4212 frame_T *foundfr;
4213
4214 foundfr = curwin->w_frame;
4215 while (count--)
4216 {
4217 /*
4218 * First go upwards in the tree of frames until we find a upwards or
4219 * downwards neighbor.
4220 */
4221 fr = foundfr;
4222 for (;;)
4223 {
4224 if (fr == topframe)
4225 goto end;
4226 if (up)
4227 nfr = fr->fr_prev;
4228 else
4229 nfr = fr->fr_next;
4230 if (fr->fr_parent->fr_layout == FR_COL && nfr != NULL)
4231 break;
4232 fr = fr->fr_parent;
4233 }
4234
4235 /*
4236 * Now go downwards to find the bottom or top frame in it.
4237 */
4238 for (;;)
4239 {
4240 if (nfr->fr_layout == FR_LEAF)
4241 {
4242 foundfr = nfr;
4243 break;
4244 }
4245 fr = nfr->fr_child;
4246 if (nfr->fr_layout == FR_ROW)
4247 {
4248 /* Find the frame at the cursor row. */
4249 while (fr->fr_next != NULL
4250 && frame2win(fr)->w_wincol + fr->fr_width
4251 <= curwin->w_wincol + curwin->w_wcol)
4252 fr = fr->fr_next;
4253 }
4254 if (nfr->fr_layout == FR_COL && up)
4255 while (fr->fr_next != NULL)
4256 fr = fr->fr_next;
4257 nfr = fr;
4258 }
4259 }
4260end:
4261 if (foundfr != NULL)
4262 win_goto(foundfr->fr_win);
4263}
4264
4265/*
4266 * Move to left or right window.
4267 */
4268 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004269win_goto_hor(
4270 int left, /* TRUE to go to left win */
4271 long count)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004272{
4273 frame_T *fr;
4274 frame_T *nfr;
4275 frame_T *foundfr;
4276
4277 foundfr = curwin->w_frame;
4278 while (count--)
4279 {
4280 /*
4281 * First go upwards in the tree of frames until we find a left or
4282 * right neighbor.
4283 */
4284 fr = foundfr;
4285 for (;;)
4286 {
4287 if (fr == topframe)
4288 goto end;
4289 if (left)
4290 nfr = fr->fr_prev;
4291 else
4292 nfr = fr->fr_next;
4293 if (fr->fr_parent->fr_layout == FR_ROW && nfr != NULL)
4294 break;
4295 fr = fr->fr_parent;
4296 }
4297
4298 /*
4299 * Now go downwards to find the leftmost or rightmost frame in it.
4300 */
4301 for (;;)
4302 {
4303 if (nfr->fr_layout == FR_LEAF)
4304 {
4305 foundfr = nfr;
4306 break;
4307 }
4308 fr = nfr->fr_child;
4309 if (nfr->fr_layout == FR_COL)
4310 {
4311 /* Find the frame at the cursor row. */
4312 while (fr->fr_next != NULL
4313 && frame2win(fr)->w_winrow + fr->fr_height
4314 <= curwin->w_winrow + curwin->w_wrow)
4315 fr = fr->fr_next;
4316 }
4317 if (nfr->fr_layout == FR_ROW && left)
4318 while (fr->fr_next != NULL)
4319 fr = fr->fr_next;
4320 nfr = fr;
4321 }
4322 }
4323end:
4324 if (foundfr != NULL)
4325 win_goto(foundfr->fr_win);
4326}
Bram Moolenaar071d4272004-06-13 20:20:40 +00004327
4328/*
4329 * Make window "wp" the current window.
4330 */
4331 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004332win_enter(win_T *wp, int undo_sync)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004333{
Bram Moolenaarc917da42016-07-19 22:31:36 +02004334 win_enter_ext(wp, undo_sync, FALSE, FALSE, TRUE, TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004335}
4336
4337/*
4338 * Make window wp the current window.
4339 * Can be called with "curwin_invalid" TRUE, which means that curwin has just
4340 * been closed and isn't valid.
4341 */
4342 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004343win_enter_ext(
4344 win_T *wp,
4345 int undo_sync,
4346 int curwin_invalid,
Bram Moolenaar6f470022018-04-10 18:47:20 +02004347 int trigger_new_autocmds,
4348 int trigger_enter_autocmds,
4349 int trigger_leave_autocmds)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004350{
Bram Moolenaar071d4272004-06-13 20:20:40 +00004351 int other_buffer = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004352
4353 if (wp == curwin && !curwin_invalid) /* nothing to do */
4354 return;
4355
Bram Moolenaar6d41c782018-06-06 09:11:12 +02004356#ifdef FEAT_JOB_CHANNEL
4357 if (!curwin_invalid)
4358 leaving_window(curwin);
4359#endif
4360
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004361 if (!curwin_invalid && trigger_leave_autocmds)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004362 {
4363 /*
4364 * Be careful: If autocommands delete the window, return now.
4365 */
4366 if (wp->w_buffer != curbuf)
4367 {
4368 apply_autocmds(EVENT_BUFLEAVE, NULL, NULL, FALSE, curbuf);
4369 other_buffer = TRUE;
4370 if (!win_valid(wp))
4371 return;
4372 }
4373 apply_autocmds(EVENT_WINLEAVE, NULL, NULL, FALSE, curbuf);
4374 if (!win_valid(wp))
4375 return;
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01004376#ifdef FEAT_EVAL
Bram Moolenaar071d4272004-06-13 20:20:40 +00004377 /* autocmds may abort script processing */
4378 if (aborting())
4379 return;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004380#endif
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01004381 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004382
4383 /* sync undo before leaving the current buffer */
4384 if (undo_sync && curbuf != wp->w_buffer)
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004385 u_sync(FALSE);
Bram Moolenaarec1561c2014-06-17 13:52:40 +02004386
4387 /* Might need to scroll the old window before switching, e.g., when the
4388 * cursor was moved. */
4389 update_topline();
4390
Bram Moolenaar071d4272004-06-13 20:20:40 +00004391 /* may have to copy the buffer options when 'cpo' contains 'S' */
4392 if (wp->w_buffer != curbuf)
4393 buf_copy_options(wp->w_buffer, BCO_ENTER | BCO_NOHELP);
4394 if (!curwin_invalid)
4395 {
4396 prevwin = curwin; /* remember for CTRL-W p */
4397 curwin->w_redr_status = TRUE;
4398 }
4399 curwin = wp;
4400 curbuf = wp->w_buffer;
4401 check_cursor();
4402#ifdef FEAT_VIRTUALEDIT
4403 if (!virtual_active())
4404 curwin->w_cursor.coladd = 0;
4405#endif
4406 changed_line_abv_curs(); /* assume cursor position needs updating */
4407
4408 if (curwin->w_localdir != NULL)
4409 {
4410 /* Window has a local directory: Save current directory as global
4411 * directory (unless that was done already) and change to the local
4412 * directory. */
4413 if (globaldir == NULL)
4414 {
4415 char_u cwd[MAXPATHL];
4416
4417 if (mch_dirname(cwd, MAXPATHL) == OK)
4418 globaldir = vim_strsave(cwd);
4419 }
Bram Moolenaarfe86f2d2008-11-28 20:29:07 +00004420 if (mch_chdir((char *)curwin->w_localdir) == 0)
4421 shorten_fnames(TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004422 }
4423 else if (globaldir != NULL)
4424 {
4425 /* Window doesn't have a local directory and we are not in the global
4426 * directory: Change to the global directory. */
Bram Moolenaarfe86f2d2008-11-28 20:29:07 +00004427 ignored = mch_chdir((char *)globaldir);
Bram Moolenaard23a8232018-02-10 18:45:26 +01004428 VIM_CLEAR(globaldir);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004429 shorten_fnames(TRUE);
4430 }
4431
Bram Moolenaar6d41c782018-06-06 09:11:12 +02004432#ifdef FEAT_JOB_CHANNEL
4433 entering_window(curwin);
4434#endif
Bram Moolenaarc917da42016-07-19 22:31:36 +02004435 if (trigger_new_autocmds)
4436 apply_autocmds(EVENT_WINNEW, NULL, NULL, FALSE, curbuf);
Bram Moolenaar49e649f2013-05-06 04:50:35 +02004437 if (trigger_enter_autocmds)
4438 {
4439 apply_autocmds(EVENT_WINENTER, NULL, NULL, FALSE, curbuf);
4440 if (other_buffer)
4441 apply_autocmds(EVENT_BUFENTER, NULL, NULL, FALSE, curbuf);
4442 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004443
4444#ifdef FEAT_TITLE
4445 maketitle();
4446#endif
4447 curwin->w_redr_status = TRUE;
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00004448 redraw_tabline = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004449 if (restart_edit)
4450 redraw_later(VALID); /* causes status line redraw */
4451
4452 /* set window height to desired minimal value */
4453 if (curwin->w_height < p_wh && !curwin->w_p_wfh)
4454 win_setheight((int)p_wh);
4455 else if (curwin->w_height == 0)
4456 win_setheight(1);
4457
Bram Moolenaar071d4272004-06-13 20:20:40 +00004458 /* set window width to desired minimal value */
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00004459 if (curwin->w_width < p_wiw && !curwin->w_p_wfw)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004460 win_setwidth((int)p_wiw);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004461
4462#ifdef FEAT_MOUSE
4463 setmouse(); /* in case jumped to/from help buffer */
4464#endif
4465
Bram Moolenaar498efdb2006-09-05 14:31:54 +00004466 /* Change directories when the 'acd' option is set. */
Bram Moolenaar6f470022018-04-10 18:47:20 +02004467 DO_AUTOCHDIR;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004468}
4469
Bram Moolenaar071d4272004-06-13 20:20:40 +00004470
Bram Moolenaar071d4272004-06-13 20:20:40 +00004471/*
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004472 * Jump to the first open window that contains buffer "buf", if one exists.
4473 * Returns a pointer to the window found, otherwise NULL.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004474 */
4475 win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004476buf_jump_open_win(buf_T *buf)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004477{
Bram Moolenaar482a2b52014-10-21 20:57:15 +02004478 win_T *wp = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004479
Bram Moolenaar482a2b52014-10-21 20:57:15 +02004480 if (curwin->w_buffer == buf)
4481 wp = curwin;
Bram Moolenaar482a2b52014-10-21 20:57:15 +02004482 else
Bram Moolenaar29323592016-07-24 22:04:11 +02004483 FOR_ALL_WINDOWS(wp)
Bram Moolenaar482a2b52014-10-21 20:57:15 +02004484 if (wp->w_buffer == buf)
4485 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004486 if (wp != NULL)
4487 win_enter(wp, FALSE);
Bram Moolenaar482a2b52014-10-21 20:57:15 +02004488 return wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004489}
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004490
4491/*
4492 * Jump to the first open window in any tab page that contains buffer "buf",
4493 * if one exists.
4494 * Returns a pointer to the window found, otherwise NULL.
4495 */
4496 win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004497buf_jump_open_tab(buf_T *buf)
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004498{
Bram Moolenaar482a2b52014-10-21 20:57:15 +02004499 win_T *wp = buf_jump_open_win(buf);
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004500 tabpage_T *tp;
4501
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004502 if (wp != NULL)
4503 return wp;
4504
Bram Moolenaar29323592016-07-24 22:04:11 +02004505 FOR_ALL_TABPAGES(tp)
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004506 if (tp != curtab)
4507 {
4508 for (wp = tp->tp_firstwin; wp != NULL; wp = wp->w_next)
4509 if (wp->w_buffer == buf)
4510 break;
4511 if (wp != NULL)
4512 {
4513 goto_tabpage_win(tp, wp);
4514 if (curwin != wp)
4515 wp = NULL; /* something went wrong */
4516 break;
4517 }
4518 }
Bram Moolenaar482a2b52014-10-21 20:57:15 +02004519 return wp;
Bram Moolenaar779b74b2006-04-10 14:55:34 +00004520}
Bram Moolenaar071d4272004-06-13 20:20:40 +00004521
Bram Moolenaar888ccac2016-06-04 18:49:36 +02004522static int last_win_id = LOWEST_WIN_ID - 1;
Bram Moolenaar86edef62016-03-13 18:07:30 +01004523
Bram Moolenaar071d4272004-06-13 20:20:40 +00004524/*
Bram Moolenaar746ebd32009-06-16 14:01:43 +00004525 * Allocate a window structure and link it in the window list when "hidden" is
4526 * FALSE.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004527 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004528 static win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004529win_alloc(win_T *after UNUSED, int hidden UNUSED)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004530{
Bram Moolenaar70b2a562012-01-10 22:26:17 +01004531 win_T *new_wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004532
4533 /*
4534 * allocate window structure and linesizes arrays
4535 */
Bram Moolenaar70b2a562012-01-10 22:26:17 +01004536 new_wp = (win_T *)alloc_clear((unsigned)sizeof(win_T));
Bram Moolenaar429fa852013-04-15 12:27:36 +02004537 if (new_wp == NULL)
4538 return NULL;
4539
4540 if (win_alloc_lines(new_wp) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004541 {
Bram Moolenaar70b2a562012-01-10 22:26:17 +01004542 vim_free(new_wp);
Bram Moolenaar429fa852013-04-15 12:27:36 +02004543 return NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004544 }
4545
Bram Moolenaar86edef62016-03-13 18:07:30 +01004546 new_wp->w_id = ++last_win_id;
4547
Bram Moolenaar429fa852013-04-15 12:27:36 +02004548#ifdef FEAT_EVAL
4549 /* init w: variables */
4550 new_wp->w_vars = dict_alloc();
4551 if (new_wp->w_vars == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004552 {
Bram Moolenaar429fa852013-04-15 12:27:36 +02004553 win_free_lsize(new_wp);
4554 vim_free(new_wp);
4555 return NULL;
4556 }
4557 init_var_dict(new_wp->w_vars, &new_wp->w_winvar, VAR_SCOPE);
Bram Moolenaaree79cbc2007-05-02 19:50:14 +00004558#endif
Bram Moolenaar429fa852013-04-15 12:27:36 +02004559
Bram Moolenaar429fa852013-04-15 12:27:36 +02004560 /* Don't execute autocommands while the window is not properly
4561 * initialized yet. gui_create_scrollbar() may trigger a FocusGained
4562 * event. */
4563 block_autocmds();
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01004564
Bram Moolenaar429fa852013-04-15 12:27:36 +02004565 /*
4566 * link the window in the window list
4567 */
Bram Moolenaar429fa852013-04-15 12:27:36 +02004568 if (!hidden)
4569 win_append(after, new_wp);
Bram Moolenaar429fa852013-04-15 12:27:36 +02004570 new_wp->w_wincol = 0;
4571 new_wp->w_width = Columns;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004572
Bram Moolenaar429fa852013-04-15 12:27:36 +02004573 /* position the display and the cursor at the top of the file. */
4574 new_wp->w_topline = 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004575#ifdef FEAT_DIFF
Bram Moolenaar429fa852013-04-15 12:27:36 +02004576 new_wp->w_topfill = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004577#endif
Bram Moolenaar429fa852013-04-15 12:27:36 +02004578 new_wp->w_botline = 2;
4579 new_wp->w_cursor.lnum = 1;
Bram Moolenaar429fa852013-04-15 12:27:36 +02004580 new_wp->w_scbind_pos = 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004581
Bram Moolenaar429fa852013-04-15 12:27:36 +02004582 /* We won't calculate w_fraction until resizing the window */
4583 new_wp->w_fraction = 0;
4584 new_wp->w_prev_fraction_row = -1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004585
4586#ifdef FEAT_GUI
Bram Moolenaar429fa852013-04-15 12:27:36 +02004587 if (gui.in_use)
4588 {
4589 gui_create_scrollbar(&new_wp->w_scrollbars[SBAR_LEFT],
4590 SBAR_LEFT, new_wp);
4591 gui_create_scrollbar(&new_wp->w_scrollbars[SBAR_RIGHT],
4592 SBAR_RIGHT, new_wp);
4593 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004594#endif
4595#ifdef FEAT_FOLDING
Bram Moolenaar429fa852013-04-15 12:27:36 +02004596 foldInitWin(new_wp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004597#endif
Bram Moolenaar429fa852013-04-15 12:27:36 +02004598 unblock_autocmds();
Bram Moolenaar6ee10162007-07-26 20:58:42 +00004599#ifdef FEAT_SEARCH_EXTRA
Bram Moolenaar429fa852013-04-15 12:27:36 +02004600 new_wp->w_match_head = NULL;
4601 new_wp->w_next_match_id = 4;
Bram Moolenaar6ee10162007-07-26 20:58:42 +00004602#endif
Bram Moolenaar70b2a562012-01-10 22:26:17 +01004603 return new_wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004604}
4605
Bram Moolenaar071d4272004-06-13 20:20:40 +00004606/*
Bram Moolenaarff18df02013-07-24 17:51:57 +02004607 * Remove window 'wp' from the window list and free the structure.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004608 */
4609 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004610win_free(
4611 win_T *wp,
4612 tabpage_T *tp) /* tab page "win" is in, NULL for current */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004613{
4614 int i;
Bram Moolenaarff18df02013-07-24 17:51:57 +02004615 buf_T *buf;
4616 wininfo_T *wip;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004617
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00004618#ifdef FEAT_FOLDING
4619 clearFolding(wp);
4620#endif
4621
4622 /* reduce the reference count to the argument list. */
4623 alist_unlink(wp->w_alist);
4624
Bram Moolenaaree79cbc2007-05-02 19:50:14 +00004625 /* Don't execute autocommands while the window is halfway being deleted.
4626 * gui_mch_destroy_scrollbar() may trigger a FocusGained event. */
Bram Moolenaar78ab3312007-09-29 12:16:41 +00004627 block_autocmds();
Bram Moolenaaree79cbc2007-05-02 19:50:14 +00004628
Bram Moolenaar0ba04292010-07-14 23:23:17 +02004629#ifdef FEAT_LUA
4630 lua_window_free(wp);
4631#endif
4632
Bram Moolenaar325b7a22004-07-05 15:58:32 +00004633#ifdef FEAT_MZSCHEME
4634 mzscheme_window_free(wp);
4635#endif
4636
Bram Moolenaar071d4272004-06-13 20:20:40 +00004637#ifdef FEAT_PERL
4638 perl_win_free(wp);
4639#endif
4640
4641#ifdef FEAT_PYTHON
4642 python_window_free(wp);
4643#endif
4644
Bram Moolenaarbd5e15f2010-07-17 21:19:38 +02004645#ifdef FEAT_PYTHON3
4646 python3_window_free(wp);
4647#endif
4648
Bram Moolenaar071d4272004-06-13 20:20:40 +00004649#ifdef FEAT_TCL
4650 tcl_window_free(wp);
4651#endif
4652
4653#ifdef FEAT_RUBY
4654 ruby_window_free(wp);
4655#endif
4656
4657 clear_winopt(&wp->w_onebuf_opt);
4658 clear_winopt(&wp->w_allbuf_opt);
4659
4660#ifdef FEAT_EVAL
Bram Moolenaar429fa852013-04-15 12:27:36 +02004661 vars_clear(&wp->w_vars->dv_hashtab); /* free all w: variables */
4662 hash_init(&wp->w_vars->dv_hashtab);
4663 unref_var_dict(wp->w_vars);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004664#endif
4665
Bram Moolenaar3dda7db2016-04-03 21:22:58 +02004666 {
4667 tabpage_T *ttp;
4668
4669 if (prevwin == wp)
4670 prevwin = NULL;
Bram Moolenaar29323592016-07-24 22:04:11 +02004671 FOR_ALL_TABPAGES(ttp)
Bram Moolenaar3dda7db2016-04-03 21:22:58 +02004672 if (ttp->tp_prevwin == wp)
4673 ttp->tp_prevwin = NULL;
4674 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004675 win_free_lsize(wp);
4676
4677 for (i = 0; i < wp->w_tagstacklen; ++i)
4678 vim_free(wp->w_tagstack[i].tagname);
4679
4680 vim_free(wp->w_localdir);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00004681
Bram Moolenaarff18df02013-07-24 17:51:57 +02004682 /* Remove the window from the b_wininfo lists, it may happen that the
4683 * freed memory is re-used for another window. */
Bram Moolenaar29323592016-07-24 22:04:11 +02004684 FOR_ALL_BUFFERS(buf)
Bram Moolenaarff18df02013-07-24 17:51:57 +02004685 for (wip = buf->b_wininfo; wip != NULL; wip = wip->wi_next)
4686 if (wip->wi_win == wp)
4687 wip->wi_win = NULL;
4688
Bram Moolenaar071d4272004-06-13 20:20:40 +00004689#ifdef FEAT_SEARCH_EXTRA
Bram Moolenaar6ee10162007-07-26 20:58:42 +00004690 clear_matches(wp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004691#endif
Bram Moolenaar6ee10162007-07-26 20:58:42 +00004692
Bram Moolenaar071d4272004-06-13 20:20:40 +00004693#ifdef FEAT_JUMPLIST
4694 free_jumplist(wp);
4695#endif
4696
Bram Moolenaar28c258f2006-01-25 22:02:51 +00004697#ifdef FEAT_QUICKFIX
4698 qf_free_all(wp);
4699#endif
4700
Bram Moolenaar071d4272004-06-13 20:20:40 +00004701#ifdef FEAT_GUI
4702 if (gui.in_use)
4703 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00004704 gui_mch_destroy_scrollbar(&wp->w_scrollbars[SBAR_LEFT]);
4705 gui_mch_destroy_scrollbar(&wp->w_scrollbars[SBAR_RIGHT]);
4706 }
4707#endif /* FEAT_GUI */
4708
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02004709#ifdef FEAT_MENU
4710 remove_winbar(wp);
4711#endif
4712
Bram Moolenaar860cae12010-06-05 23:22:07 +02004713#ifdef FEAT_SYN_HL
Bram Moolenaar1a384422010-07-14 19:53:30 +02004714 vim_free(wp->w_p_cc_cols);
Bram Moolenaar860cae12010-06-05 23:22:07 +02004715#endif
4716
Bram Moolenaarf061e0b2009-06-24 15:32:01 +00004717 if (wp != aucmd_win)
Bram Moolenaarfd29f462010-06-06 16:11:09 +02004718 win_remove(wp, tp);
Bram Moolenaar3be85852014-06-12 14:01:31 +02004719 if (autocmd_busy)
4720 {
4721 wp->w_next = au_pending_free_win;
4722 au_pending_free_win = wp;
4723 }
4724 else
4725 vim_free(wp);
Bram Moolenaaree79cbc2007-05-02 19:50:14 +00004726
Bram Moolenaar78ab3312007-09-29 12:16:41 +00004727 unblock_autocmds();
Bram Moolenaar071d4272004-06-13 20:20:40 +00004728}
4729
4730/*
4731 * Append window "wp" in the window list after window "after".
4732 */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00004733 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004734win_append(win_T *after, win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004735{
4736 win_T *before;
4737
4738 if (after == NULL) /* after NULL is in front of the first */
4739 before = firstwin;
4740 else
4741 before = after->w_next;
4742
4743 wp->w_next = before;
4744 wp->w_prev = after;
4745 if (after == NULL)
4746 firstwin = wp;
4747 else
4748 after->w_next = wp;
4749 if (before == NULL)
4750 lastwin = wp;
4751 else
4752 before->w_prev = wp;
4753}
4754
4755/*
4756 * Remove a window from the window list.
4757 */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00004758 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004759win_remove(
4760 win_T *wp,
4761 tabpage_T *tp) /* tab page "win" is in, NULL for current */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004762{
4763 if (wp->w_prev != NULL)
4764 wp->w_prev->w_next = wp->w_next;
Bram Moolenaarf740b292006-02-16 22:11:02 +00004765 else if (tp == NULL)
Bram Moolenaar816968d2017-09-29 21:29:18 +02004766 firstwin = curtab->tp_firstwin = wp->w_next;
Bram Moolenaarf740b292006-02-16 22:11:02 +00004767 else
4768 tp->tp_firstwin = wp->w_next;
Bram Moolenaar816968d2017-09-29 21:29:18 +02004769
Bram Moolenaar071d4272004-06-13 20:20:40 +00004770 if (wp->w_next != NULL)
4771 wp->w_next->w_prev = wp->w_prev;
Bram Moolenaarf740b292006-02-16 22:11:02 +00004772 else if (tp == NULL)
Bram Moolenaar816968d2017-09-29 21:29:18 +02004773 lastwin = curtab->tp_lastwin = wp->w_prev;
Bram Moolenaarf740b292006-02-16 22:11:02 +00004774 else
4775 tp->tp_lastwin = wp->w_prev;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004776}
4777
4778/*
4779 * Append frame "frp" in a frame list after frame "after".
4780 */
4781 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004782frame_append(frame_T *after, frame_T *frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004783{
4784 frp->fr_next = after->fr_next;
4785 after->fr_next = frp;
4786 if (frp->fr_next != NULL)
4787 frp->fr_next->fr_prev = frp;
4788 frp->fr_prev = after;
4789}
4790
4791/*
4792 * Insert frame "frp" in a frame list before frame "before".
4793 */
4794 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004795frame_insert(frame_T *before, frame_T *frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004796{
4797 frp->fr_next = before;
4798 frp->fr_prev = before->fr_prev;
4799 before->fr_prev = frp;
4800 if (frp->fr_prev != NULL)
4801 frp->fr_prev->fr_next = frp;
4802 else
4803 frp->fr_parent->fr_child = frp;
4804}
4805
4806/*
4807 * Remove a frame from a frame list.
4808 */
4809 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004810frame_remove(frame_T *frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004811{
4812 if (frp->fr_prev != NULL)
4813 frp->fr_prev->fr_next = frp->fr_next;
4814 else
Bram Moolenaar6f361c92018-01-31 19:06:50 +01004815 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00004816 frp->fr_parent->fr_child = frp->fr_next;
Bram Moolenaar6f361c92018-01-31 19:06:50 +01004817 /* special case: topframe->fr_child == frp */
4818 if (topframe->fr_child == frp)
4819 topframe->fr_child = frp->fr_next;
4820 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004821 if (frp->fr_next != NULL)
4822 frp->fr_next->fr_prev = frp->fr_prev;
4823}
4824
Bram Moolenaar071d4272004-06-13 20:20:40 +00004825/*
4826 * Allocate w_lines[] for window "wp".
4827 * Return FAIL for failure, OK for success.
4828 */
4829 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004830win_alloc_lines(win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004831{
4832 wp->w_lines_valid = 0;
Bram Moolenaar9334c342006-11-21 19:57:30 +00004833 wp->w_lines = (wline_T *)alloc_clear((unsigned)(Rows * sizeof(wline_T)));
Bram Moolenaar071d4272004-06-13 20:20:40 +00004834 if (wp->w_lines == NULL)
4835 return FAIL;
4836 return OK;
4837}
4838
4839/*
4840 * free lsize arrays for a window
4841 */
4842 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004843win_free_lsize(win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004844{
Bram Moolenaar06e4a6d2014-06-12 11:49:46 +02004845 /* TODO: why would wp be NULL here? */
4846 if (wp != NULL)
Bram Moolenaard23a8232018-02-10 18:45:26 +01004847 VIM_CLEAR(wp->w_lines);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004848}
4849
4850/*
4851 * Called from win_new_shellsize() after Rows changed.
Bram Moolenaarf740b292006-02-16 22:11:02 +00004852 * This only does the current tab page, others must be done when made active.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004853 */
4854 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004855shell_new_rows(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004856{
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00004857 int h = (int)ROWS_AVAIL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004858
4859 if (firstwin == NULL) /* not initialized yet */
4860 return;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004861 if (h < frame_minheight(topframe, NULL))
4862 h = frame_minheight(topframe, NULL);
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00004863
4864 /* First try setting the heights of windows with 'winfixheight'. If
Bram Moolenaar071d4272004-06-13 20:20:40 +00004865 * that doesn't result in the right height, forget about that option. */
4866 frame_new_height(topframe, h, FALSE, TRUE);
Bram Moolenaarb893ac22013-06-26 14:04:47 +02004867 if (!frame_check_height(topframe, h))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004868 frame_new_height(topframe, h, FALSE, FALSE);
4869
4870 (void)win_comp_pos(); /* recompute w_winrow and w_wincol */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004871 compute_cmdrow();
Bram Moolenaarc6fe9192006-04-09 21:54:49 +00004872 curtab->tp_ch_used = p_ch;
Bram Moolenaar05159a02005-02-26 23:04:13 +00004873
Bram Moolenaar071d4272004-06-13 20:20:40 +00004874#if 0
4875 /* Disabled: don't want making the screen smaller make a window larger. */
4876 if (p_ea)
4877 win_equal(curwin, FALSE, 'v');
4878#endif
4879}
4880
Bram Moolenaar071d4272004-06-13 20:20:40 +00004881/*
4882 * Called from win_new_shellsize() after Columns changed.
4883 */
4884 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004885shell_new_columns(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004886{
4887 if (firstwin == NULL) /* not initialized yet */
4888 return;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00004889
4890 /* First try setting the widths of windows with 'winfixwidth'. If that
4891 * doesn't result in the right width, forget about that option. */
4892 frame_new_width(topframe, (int)Columns, FALSE, TRUE);
Bram Moolenaarb893ac22013-06-26 14:04:47 +02004893 if (!frame_check_width(topframe, Columns))
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00004894 frame_new_width(topframe, (int)Columns, FALSE, FALSE);
4895
Bram Moolenaar071d4272004-06-13 20:20:40 +00004896 (void)win_comp_pos(); /* recompute w_winrow and w_wincol */
4897#if 0
4898 /* Disabled: don't want making the screen smaller make a window larger. */
4899 if (p_ea)
4900 win_equal(curwin, FALSE, 'h');
4901#endif
4902}
Bram Moolenaar071d4272004-06-13 20:20:40 +00004903
4904#if defined(FEAT_CMDWIN) || defined(PROTO)
4905/*
4906 * Save the size of all windows in "gap".
4907 */
4908 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004909win_size_save(garray_T *gap)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004910
4911{
4912 win_T *wp;
4913
4914 ga_init2(gap, (int)sizeof(int), 1);
4915 if (ga_grow(gap, win_count() * 2) == OK)
Bram Moolenaar29323592016-07-24 22:04:11 +02004916 FOR_ALL_WINDOWS(wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004917 {
4918 ((int *)gap->ga_data)[gap->ga_len++] =
4919 wp->w_width + wp->w_vsep_width;
4920 ((int *)gap->ga_data)[gap->ga_len++] = wp->w_height;
4921 }
4922}
4923
4924/*
4925 * Restore window sizes, but only if the number of windows is still the same.
4926 * Does not free the growarray.
4927 */
4928 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004929win_size_restore(garray_T *gap)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004930{
4931 win_T *wp;
Bram Moolenaarb643e772014-07-16 15:18:26 +02004932 int i, j;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004933
4934 if (win_count() * 2 == gap->ga_len)
4935 {
Bram Moolenaarb643e772014-07-16 15:18:26 +02004936 /* The order matters, because frames contain other frames, but it's
4937 * difficult to get right. The easy way out is to do it twice. */
4938 for (j = 0; j < 2; ++j)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004939 {
Bram Moolenaarb643e772014-07-16 15:18:26 +02004940 i = 0;
Bram Moolenaar29323592016-07-24 22:04:11 +02004941 FOR_ALL_WINDOWS(wp)
Bram Moolenaarb643e772014-07-16 15:18:26 +02004942 {
4943 frame_setwidth(wp->w_frame, ((int *)gap->ga_data)[i++]);
4944 win_setheight_win(((int *)gap->ga_data)[i++], wp);
4945 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004946 }
4947 /* recompute the window positions */
4948 (void)win_comp_pos();
4949 }
4950}
4951#endif /* FEAT_CMDWIN */
4952
Bram Moolenaar071d4272004-06-13 20:20:40 +00004953/*
4954 * Update the position for all windows, using the width and height of the
4955 * frames.
4956 * Returns the row just after the last window.
4957 */
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00004958 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004959win_comp_pos(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004960{
Bram Moolenaar32466aa2006-02-24 23:53:04 +00004961 int row = tabline_height();
Bram Moolenaar071d4272004-06-13 20:20:40 +00004962 int col = 0;
4963
4964 frame_comp_pos(topframe, &row, &col);
4965 return row;
4966}
4967
4968/*
4969 * Update the position of the windows in frame "topfrp", using the width and
4970 * height of the frames.
4971 * "*row" and "*col" are the top-left position of the frame. They are updated
4972 * to the bottom-right position plus one.
4973 */
4974 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01004975frame_comp_pos(frame_T *topfrp, int *row, int *col)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004976{
4977 win_T *wp;
4978 frame_T *frp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004979 int startcol;
4980 int startrow;
Bram Moolenaar415a6932017-12-05 20:31:07 +01004981 int h;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004982
4983 wp = topfrp->fr_win;
4984 if (wp != NULL)
4985 {
Bram Moolenaar44a2f922016-03-19 22:11:51 +01004986 if (wp->w_winrow != *row || wp->w_wincol != *col)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004987 {
4988 /* position changed, redraw */
4989 wp->w_winrow = *row;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004990 wp->w_wincol = *col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004991 redraw_win_later(wp, NOT_VALID);
4992 wp->w_redr_status = TRUE;
4993 }
Bram Moolenaar415a6932017-12-05 20:31:07 +01004994 /* WinBar will not show if the window height is zero */
4995 h = VISIBLE_HEIGHT(wp) + wp->w_status_height;
4996 *row += h > topfrp->fr_height ? topfrp->fr_height : h;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004997 *col += wp->w_width + wp->w_vsep_width;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004998 }
4999 else
5000 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00005001 startrow = *row;
5002 startcol = *col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005003 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
5004 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00005005 if (topfrp->fr_layout == FR_ROW)
5006 *row = startrow; /* all frames are at the same row */
5007 else
5008 *col = startcol; /* all frames are at the same col */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005009 frame_comp_pos(frp, row, col);
5010 }
5011 }
5012}
5013
Bram Moolenaar071d4272004-06-13 20:20:40 +00005014/*
5015 * Set current window height and take care of repositioning other windows to
5016 * fit around it.
5017 */
5018 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005019win_setheight(int height)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005020{
5021 win_setheight_win(height, curwin);
5022}
5023
5024/*
5025 * Set the window height of window "win" and take care of repositioning other
5026 * windows to fit around it.
5027 */
5028 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005029win_setheight_win(int height, win_T *win)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005030{
5031 int row;
5032
5033 if (win == curwin)
5034 {
5035 /* Always keep current window at least one line high, even when
5036 * 'winminheight' is zero. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005037 if (height < p_wmh)
5038 height = p_wmh;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005039 if (height == 0)
5040 height = 1;
Bram Moolenaar415a6932017-12-05 20:31:07 +01005041 height += WINBAR_HEIGHT(curwin);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005042 }
5043
Bram Moolenaar071d4272004-06-13 20:20:40 +00005044 frame_setheight(win->w_frame, height + win->w_status_height);
5045
5046 /* recompute the window positions */
5047 row = win_comp_pos();
Bram Moolenaar071d4272004-06-13 20:20:40 +00005048
5049 /*
5050 * If there is extra space created between the last window and the command
5051 * line, clear it.
5052 */
5053 if (full_screen && msg_scrolled == 0 && row < cmdline_row)
5054 screen_fill(row, cmdline_row, 0, (int)Columns, ' ', ' ', 0);
5055 cmdline_row = row;
5056 msg_row = row;
5057 msg_col = 0;
5058
5059 redraw_all_later(NOT_VALID);
5060}
5061
Bram Moolenaar071d4272004-06-13 20:20:40 +00005062/*
5063 * Set the height of a frame to "height" and take care that all frames and
5064 * windows inside it are resized. Also resize frames on the left and right if
5065 * the are in the same FR_ROW frame.
5066 *
5067 * Strategy:
5068 * If the frame is part of a FR_COL frame, try fitting the frame in that
5069 * frame. If that doesn't work (the FR_COL frame is too small), recursively
5070 * go to containing frames to resize them and make room.
5071 * If the frame is part of a FR_ROW frame, all frames must be resized as well.
5072 * Check for the minimal height of the FR_ROW frame.
5073 * At the top level we can also use change the command line height.
5074 */
5075 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005076frame_setheight(frame_T *curfrp, int height)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005077{
5078 int room; /* total number of lines available */
5079 int take; /* number of lines taken from other windows */
5080 int room_cmdline; /* lines available from cmdline */
5081 int run;
5082 frame_T *frp;
5083 int h;
5084 int room_reserved;
5085
5086 /* If the height already is the desired value, nothing to do. */
5087 if (curfrp->fr_height == height)
5088 return;
5089
5090 if (curfrp->fr_parent == NULL)
5091 {
5092 /* topframe: can only change the command line */
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00005093 if (height > ROWS_AVAIL)
5094 height = ROWS_AVAIL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005095 if (height > 0)
5096 frame_new_height(curfrp, height, FALSE, FALSE);
5097 }
5098 else if (curfrp->fr_parent->fr_layout == FR_ROW)
5099 {
5100 /* Row of frames: Also need to resize frames left and right of this
5101 * one. First check for the minimal height of these. */
5102 h = frame_minheight(curfrp->fr_parent, NULL);
5103 if (height < h)
5104 height = h;
5105 frame_setheight(curfrp->fr_parent, height);
5106 }
5107 else
5108 {
5109 /*
5110 * Column of frames: try to change only frames in this column.
5111 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005112 /*
5113 * Do this twice:
5114 * 1: compute room available, if it's not enough try resizing the
5115 * containing frame.
5116 * 2: compute the room available and adjust the height to it.
5117 * Try not to reduce the height of a window with 'winfixheight' set.
5118 */
5119 for (run = 1; run <= 2; ++run)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005120 {
5121 room = 0;
5122 room_reserved = 0;
5123 for (frp = curfrp->fr_parent->fr_child; frp != NULL;
5124 frp = frp->fr_next)
5125 {
5126 if (frp != curfrp
5127 && frp->fr_win != NULL
5128 && frp->fr_win->w_p_wfh)
5129 room_reserved += frp->fr_height;
5130 room += frp->fr_height;
5131 if (frp != curfrp)
5132 room -= frame_minheight(frp, NULL);
5133 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00005134 if (curfrp->fr_width != Columns)
5135 room_cmdline = 0;
5136 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00005137 {
5138 room_cmdline = Rows - p_ch - (lastwin->w_winrow
Bram Moolenaar415a6932017-12-05 20:31:07 +01005139 + VISIBLE_HEIGHT(lastwin)
5140 + lastwin->w_status_height);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005141 if (room_cmdline < 0)
5142 room_cmdline = 0;
5143 }
5144
5145 if (height <= room + room_cmdline)
5146 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005147 if (run == 2 || curfrp->fr_width == Columns)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005148 {
5149 if (height > room + room_cmdline)
5150 height = room + room_cmdline;
5151 break;
5152 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00005153 frame_setheight(curfrp->fr_parent, height
5154 + frame_minheight(curfrp->fr_parent, NOWIN) - (int)p_wmh - 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005155 }
5156
5157 /*
5158 * Compute the number of lines we will take from others frames (can be
5159 * negative!).
5160 */
5161 take = height - curfrp->fr_height;
5162
5163 /* If there is not enough room, also reduce the height of a window
5164 * with 'winfixheight' set. */
5165 if (height > room + room_cmdline - room_reserved)
5166 room_reserved = room + room_cmdline - height;
5167 /* If there is only a 'winfixheight' window and making the
5168 * window smaller, need to make the other window taller. */
5169 if (take < 0 && room - curfrp->fr_height < room_reserved)
5170 room_reserved = 0;
5171
5172 if (take > 0 && room_cmdline > 0)
5173 {
5174 /* use lines from cmdline first */
5175 if (take < room_cmdline)
5176 room_cmdline = take;
5177 take -= room_cmdline;
5178 topframe->fr_height += room_cmdline;
5179 }
5180
5181 /*
5182 * set the current frame to the new height
5183 */
5184 frame_new_height(curfrp, height, FALSE, FALSE);
5185
5186 /*
5187 * First take lines from the frames after the current frame. If
5188 * that is not enough, takes lines from frames above the current
5189 * frame.
5190 */
5191 for (run = 0; run < 2; ++run)
5192 {
5193 if (run == 0)
5194 frp = curfrp->fr_next; /* 1st run: start with next window */
5195 else
5196 frp = curfrp->fr_prev; /* 2nd run: start with prev window */
5197 while (frp != NULL && take != 0)
5198 {
5199 h = frame_minheight(frp, NULL);
5200 if (room_reserved > 0
5201 && frp->fr_win != NULL
5202 && frp->fr_win->w_p_wfh)
5203 {
5204 if (room_reserved >= frp->fr_height)
5205 room_reserved -= frp->fr_height;
5206 else
5207 {
5208 if (frp->fr_height - room_reserved > take)
5209 room_reserved = frp->fr_height - take;
5210 take -= frp->fr_height - room_reserved;
5211 frame_new_height(frp, room_reserved, FALSE, FALSE);
5212 room_reserved = 0;
5213 }
5214 }
5215 else
5216 {
5217 if (frp->fr_height - take < h)
5218 {
5219 take -= frp->fr_height - h;
5220 frame_new_height(frp, h, FALSE, FALSE);
5221 }
5222 else
5223 {
5224 frame_new_height(frp, frp->fr_height - take,
5225 FALSE, FALSE);
5226 take = 0;
5227 }
5228 }
5229 if (run == 0)
5230 frp = frp->fr_next;
5231 else
5232 frp = frp->fr_prev;
5233 }
5234 }
5235 }
5236}
5237
Bram Moolenaar071d4272004-06-13 20:20:40 +00005238/*
5239 * Set current window width and take care of repositioning other windows to
5240 * fit around it.
5241 */
5242 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005243win_setwidth(int width)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005244{
5245 win_setwidth_win(width, curwin);
5246}
5247
5248 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005249win_setwidth_win(int width, win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005250{
5251 /* Always keep current window at least one column wide, even when
5252 * 'winminwidth' is zero. */
5253 if (wp == curwin)
5254 {
5255 if (width < p_wmw)
5256 width = p_wmw;
5257 if (width == 0)
5258 width = 1;
5259 }
5260
5261 frame_setwidth(wp->w_frame, width + wp->w_vsep_width);
5262
5263 /* recompute the window positions */
5264 (void)win_comp_pos();
5265
5266 redraw_all_later(NOT_VALID);
5267}
5268
5269/*
5270 * Set the width of a frame to "width" and take care that all frames and
5271 * windows inside it are resized. Also resize frames above and below if the
5272 * are in the same FR_ROW frame.
5273 *
5274 * Strategy is similar to frame_setheight().
5275 */
5276 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005277frame_setwidth(frame_T *curfrp, int width)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005278{
5279 int room; /* total number of lines available */
5280 int take; /* number of lines taken from other windows */
5281 int run;
5282 frame_T *frp;
5283 int w;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005284 int room_reserved;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005285
5286 /* If the width already is the desired value, nothing to do. */
5287 if (curfrp->fr_width == width)
5288 return;
5289
5290 if (curfrp->fr_parent == NULL)
5291 /* topframe: can't change width */
5292 return;
5293
5294 if (curfrp->fr_parent->fr_layout == FR_COL)
5295 {
5296 /* Column of frames: Also need to resize frames above and below of
5297 * this one. First check for the minimal width of these. */
5298 w = frame_minwidth(curfrp->fr_parent, NULL);
5299 if (width < w)
5300 width = w;
5301 frame_setwidth(curfrp->fr_parent, width);
5302 }
5303 else
5304 {
5305 /*
5306 * Row of frames: try to change only frames in this row.
5307 *
5308 * Do this twice:
5309 * 1: compute room available, if it's not enough try resizing the
5310 * containing frame.
5311 * 2: compute the room available and adjust the width to it.
5312 */
5313 for (run = 1; run <= 2; ++run)
5314 {
5315 room = 0;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005316 room_reserved = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005317 for (frp = curfrp->fr_parent->fr_child; frp != NULL;
5318 frp = frp->fr_next)
5319 {
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005320 if (frp != curfrp
5321 && frp->fr_win != NULL
5322 && frp->fr_win->w_p_wfw)
5323 room_reserved += frp->fr_width;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005324 room += frp->fr_width;
5325 if (frp != curfrp)
5326 room -= frame_minwidth(frp, NULL);
5327 }
5328
5329 if (width <= room)
5330 break;
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00005331 if (run == 2 || curfrp->fr_height >= ROWS_AVAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005332 {
5333 if (width > room)
5334 width = room;
5335 break;
5336 }
5337 frame_setwidth(curfrp->fr_parent, width
5338 + frame_minwidth(curfrp->fr_parent, NOWIN) - (int)p_wmw - 1);
5339 }
5340
Bram Moolenaar071d4272004-06-13 20:20:40 +00005341 /*
5342 * Compute the number of lines we will take from others frames (can be
5343 * negative!).
5344 */
5345 take = width - curfrp->fr_width;
5346
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005347 /* If there is not enough room, also reduce the width of a window
5348 * with 'winfixwidth' set. */
5349 if (width > room - room_reserved)
5350 room_reserved = room - width;
5351 /* If there is only a 'winfixwidth' window and making the
5352 * window smaller, need to make the other window narrower. */
5353 if (take < 0 && room - curfrp->fr_width < room_reserved)
5354 room_reserved = 0;
5355
Bram Moolenaar071d4272004-06-13 20:20:40 +00005356 /*
5357 * set the current frame to the new width
5358 */
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005359 frame_new_width(curfrp, width, FALSE, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005360
5361 /*
5362 * First take lines from the frames right of the current frame. If
5363 * that is not enough, takes lines from frames left of the current
5364 * frame.
5365 */
5366 for (run = 0; run < 2; ++run)
5367 {
5368 if (run == 0)
5369 frp = curfrp->fr_next; /* 1st run: start with next window */
5370 else
5371 frp = curfrp->fr_prev; /* 2nd run: start with prev window */
5372 while (frp != NULL && take != 0)
5373 {
5374 w = frame_minwidth(frp, NULL);
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005375 if (room_reserved > 0
5376 && frp->fr_win != NULL
5377 && frp->fr_win->w_p_wfw)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005378 {
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005379 if (room_reserved >= frp->fr_width)
5380 room_reserved -= frp->fr_width;
5381 else
5382 {
5383 if (frp->fr_width - room_reserved > take)
5384 room_reserved = frp->fr_width - take;
5385 take -= frp->fr_width - room_reserved;
5386 frame_new_width(frp, room_reserved, FALSE, FALSE);
5387 room_reserved = 0;
5388 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00005389 }
5390 else
5391 {
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005392 if (frp->fr_width - take < w)
5393 {
5394 take -= frp->fr_width - w;
5395 frame_new_width(frp, w, FALSE, FALSE);
5396 }
5397 else
5398 {
5399 frame_new_width(frp, frp->fr_width - take,
5400 FALSE, FALSE);
5401 take = 0;
5402 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00005403 }
5404 if (run == 0)
5405 frp = frp->fr_next;
5406 else
5407 frp = frp->fr_prev;
5408 }
5409 }
5410 }
5411}
Bram Moolenaar071d4272004-06-13 20:20:40 +00005412
5413/*
5414 * Check 'winminheight' for a valid value.
5415 */
5416 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005417win_setminheight(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005418{
5419 int room;
5420 int first = TRUE;
5421 win_T *wp;
5422
5423 /* loop until there is a 'winminheight' that is possible */
5424 while (p_wmh > 0)
5425 {
5426 /* TODO: handle vertical splits */
5427 room = -p_wh;
Bram Moolenaar29323592016-07-24 22:04:11 +02005428 FOR_ALL_WINDOWS(wp)
Bram Moolenaar415a6932017-12-05 20:31:07 +01005429 room += VISIBLE_HEIGHT(wp) - p_wmh;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005430 if (room >= 0)
5431 break;
5432 --p_wmh;
5433 if (first)
5434 {
5435 EMSG(_(e_noroom));
5436 first = FALSE;
5437 }
5438 }
5439}
5440
Bram Moolenaar6a2697f2015-11-19 13:14:30 +01005441#if defined(FEAT_MOUSE) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005442
5443/*
5444 * Status line of dragwin is dragged "offset" lines down (negative is up).
5445 */
5446 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005447win_drag_status_line(win_T *dragwin, int offset)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005448{
5449 frame_T *curfr;
5450 frame_T *fr;
5451 int room;
5452 int row;
5453 int up; /* if TRUE, drag status line up, otherwise down */
5454 int n;
5455
5456 fr = dragwin->w_frame;
5457 curfr = fr;
5458 if (fr != topframe) /* more than one window */
5459 {
5460 fr = fr->fr_parent;
5461 /* When the parent frame is not a column of frames, its parent should
5462 * be. */
5463 if (fr->fr_layout != FR_COL)
5464 {
5465 curfr = fr;
5466 if (fr != topframe) /* only a row of windows, may drag statusline */
5467 fr = fr->fr_parent;
5468 }
5469 }
5470
5471 /* If this is the last frame in a column, may want to resize the parent
5472 * frame instead (go two up to skip a row of frames). */
5473 while (curfr != topframe && curfr->fr_next == NULL)
5474 {
5475 if (fr != topframe)
5476 fr = fr->fr_parent;
5477 curfr = fr;
5478 if (fr != topframe)
5479 fr = fr->fr_parent;
5480 }
5481
5482 if (offset < 0) /* drag up */
5483 {
5484 up = TRUE;
5485 offset = -offset;
5486 /* sum up the room of the current frame and above it */
5487 if (fr == curfr)
5488 {
5489 /* only one window */
5490 room = fr->fr_height - frame_minheight(fr, NULL);
5491 }
5492 else
5493 {
5494 room = 0;
5495 for (fr = fr->fr_child; ; fr = fr->fr_next)
5496 {
5497 room += fr->fr_height - frame_minheight(fr, NULL);
5498 if (fr == curfr)
5499 break;
5500 }
5501 }
5502 fr = curfr->fr_next; /* put fr at frame that grows */
5503 }
5504 else /* drag down */
5505 {
5506 up = FALSE;
5507 /*
5508 * Only dragging the last status line can reduce p_ch.
5509 */
5510 room = Rows - cmdline_row;
5511 if (curfr->fr_next == NULL)
5512 room -= 1;
5513 else
5514 room -= p_ch;
5515 if (room < 0)
5516 room = 0;
5517 /* sum up the room of frames below of the current one */
5518 for (fr = curfr->fr_next; fr != NULL; fr = fr->fr_next)
5519 room += fr->fr_height - frame_minheight(fr, NULL);
5520 fr = curfr; /* put fr at window that grows */
5521 }
5522
5523 if (room < offset) /* Not enough room */
5524 offset = room; /* Move as far as we can */
5525 if (offset <= 0)
5526 return;
5527
5528 /*
5529 * Grow frame fr by "offset" lines.
5530 * Doesn't happen when dragging the last status line up.
5531 */
5532 if (fr != NULL)
5533 frame_new_height(fr, fr->fr_height + offset, up, FALSE);
5534
5535 if (up)
5536 fr = curfr; /* current frame gets smaller */
5537 else
5538 fr = curfr->fr_next; /* next frame gets smaller */
5539
5540 /*
5541 * Now make the other frames smaller.
5542 */
5543 while (fr != NULL && offset > 0)
5544 {
5545 n = frame_minheight(fr, NULL);
5546 if (fr->fr_height - offset <= n)
5547 {
5548 offset -= fr->fr_height - n;
5549 frame_new_height(fr, n, !up, FALSE);
5550 }
5551 else
5552 {
5553 frame_new_height(fr, fr->fr_height - offset, !up, FALSE);
5554 break;
5555 }
5556 if (up)
5557 fr = fr->fr_prev;
5558 else
5559 fr = fr->fr_next;
5560 }
5561 row = win_comp_pos();
5562 screen_fill(row, cmdline_row, 0, (int)Columns, ' ', ' ', 0);
5563 cmdline_row = row;
5564 p_ch = Rows - cmdline_row;
5565 if (p_ch < 1)
5566 p_ch = 1;
Bram Moolenaarc6fe9192006-04-09 21:54:49 +00005567 curtab->tp_ch_used = p_ch;
Bram Moolenaarf71a3db2006-03-12 21:50:18 +00005568 redraw_all_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005569 showmode();
5570}
5571
Bram Moolenaar071d4272004-06-13 20:20:40 +00005572/*
5573 * Separator line of dragwin is dragged "offset" lines right (negative is left).
5574 */
5575 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005576win_drag_vsep_line(win_T *dragwin, int offset)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005577{
5578 frame_T *curfr;
5579 frame_T *fr;
5580 int room;
5581 int left; /* if TRUE, drag separator line left, otherwise right */
5582 int n;
5583
5584 fr = dragwin->w_frame;
Bram Moolenaareb3593b2006-04-22 22:33:57 +00005585 if (fr == topframe) /* only one window (cannot happen?) */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005586 return;
5587 curfr = fr;
5588 fr = fr->fr_parent;
5589 /* When the parent frame is not a row of frames, its parent should be. */
5590 if (fr->fr_layout != FR_ROW)
5591 {
5592 if (fr == topframe) /* only a column of windows (cannot happen?) */
5593 return;
5594 curfr = fr;
5595 fr = fr->fr_parent;
5596 }
5597
5598 /* If this is the last frame in a row, may want to resize a parent
5599 * frame instead. */
5600 while (curfr->fr_next == NULL)
5601 {
5602 if (fr == topframe)
5603 break;
5604 curfr = fr;
5605 fr = fr->fr_parent;
5606 if (fr != topframe)
5607 {
5608 curfr = fr;
5609 fr = fr->fr_parent;
5610 }
5611 }
5612
5613 if (offset < 0) /* drag left */
5614 {
5615 left = TRUE;
5616 offset = -offset;
5617 /* sum up the room of the current frame and left of it */
5618 room = 0;
5619 for (fr = fr->fr_child; ; fr = fr->fr_next)
5620 {
5621 room += fr->fr_width - frame_minwidth(fr, NULL);
5622 if (fr == curfr)
5623 break;
5624 }
5625 fr = curfr->fr_next; /* put fr at frame that grows */
5626 }
5627 else /* drag right */
5628 {
5629 left = FALSE;
5630 /* sum up the room of frames right of the current one */
5631 room = 0;
5632 for (fr = curfr->fr_next; fr != NULL; fr = fr->fr_next)
5633 room += fr->fr_width - frame_minwidth(fr, NULL);
5634 fr = curfr; /* put fr at window that grows */
5635 }
5636
5637 if (room < offset) /* Not enough room */
5638 offset = room; /* Move as far as we can */
5639 if (offset <= 0) /* No room at all, quit. */
5640 return;
Bram Moolenaar294a7e52015-11-22 19:39:38 +01005641 if (fr == NULL)
5642 return; /* Safety check, should not happen. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005643
5644 /* grow frame fr by offset lines */
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005645 frame_new_width(fr, fr->fr_width + offset, left, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005646
5647 /* shrink other frames: current and at the left or at the right */
5648 if (left)
5649 fr = curfr; /* current frame gets smaller */
5650 else
5651 fr = curfr->fr_next; /* next frame gets smaller */
5652
5653 while (fr != NULL && offset > 0)
5654 {
5655 n = frame_minwidth(fr, NULL);
5656 if (fr->fr_width - offset <= n)
5657 {
5658 offset -= fr->fr_width - n;
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005659 frame_new_width(fr, n, !left, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005660 }
5661 else
5662 {
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00005663 frame_new_width(fr, fr->fr_width - offset, !left, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005664 break;
5665 }
5666 if (left)
5667 fr = fr->fr_prev;
5668 else
5669 fr = fr->fr_next;
5670 }
5671 (void)win_comp_pos();
5672 redraw_all_later(NOT_VALID);
5673}
Bram Moolenaar071d4272004-06-13 20:20:40 +00005674#endif /* FEAT_MOUSE */
5675
Bram Moolenaar0215e8e2010-12-17 17:35:10 +01005676#define FRACTION_MULT 16384L
5677
5678/*
5679 * Set wp->w_fraction for the current w_wrow and w_height.
Bram Moolenaar3679c172017-11-22 22:22:11 +01005680 * Has no effect when the window is less than two lines.
Bram Moolenaar0215e8e2010-12-17 17:35:10 +01005681 */
Bram Moolenaar9dc2ce32015-12-05 19:47:04 +01005682 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005683set_fraction(win_T *wp)
Bram Moolenaar0215e8e2010-12-17 17:35:10 +01005684{
Bram Moolenaar3679c172017-11-22 22:22:11 +01005685 if (wp->w_height > 1)
5686 wp->w_fraction = ((long)wp->w_wrow * FRACTION_MULT
Bram Moolenaar56b3bf82014-05-07 20:25:35 +02005687 + wp->w_height / 2) / (long)wp->w_height;
Bram Moolenaar0215e8e2010-12-17 17:35:10 +01005688}
5689
Bram Moolenaar071d4272004-06-13 20:20:40 +00005690/*
5691 * Set the height of a window.
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02005692 * "height" excludes any window toolbar.
Bram Moolenaar071d4272004-06-13 20:20:40 +00005693 * This takes care of the things inside the window, not what happens to the
5694 * window position, the frame or to other windows.
5695 */
Bram Moolenaar6763c142012-07-19 18:05:44 +02005696 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005697win_new_height(win_T *wp, int height)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005698{
Bram Moolenaar56b3bf82014-05-07 20:25:35 +02005699 int prev_height = wp->w_height;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005700
5701 /* Don't want a negative height. Happens when splitting a tiny window.
5702 * Will equalize heights soon to fix it. */
5703 if (height < 0)
5704 height = 0;
Bram Moolenaar4c3f5362006-04-11 21:38:50 +00005705 if (wp->w_height == height)
5706 return; /* nothing to do */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005707
Bram Moolenaar56b3bf82014-05-07 20:25:35 +02005708 if (wp->w_height > 0)
5709 {
5710 if (wp == curwin)
Bram Moolenaar0ae36a52014-06-13 20:08:45 +02005711 /* w_wrow needs to be valid. When setting 'laststatus' this may
5712 * call win_new_height() recursively. */
5713 validate_cursor();
5714 if (wp->w_height != prev_height)
5715 return; /* Recursive call already changed the size, bail out here
5716 to avoid the following to mess things up. */
Bram Moolenaar56b3bf82014-05-07 20:25:35 +02005717 if (wp->w_wrow != wp->w_prev_fraction_row)
5718 set_fraction(wp);
5719 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00005720
5721 wp->w_height = height;
5722 wp->w_skipcol = 0;
5723
Bram Moolenaar955f1982017-02-05 15:10:51 +01005724 /* There is no point in adjusting the scroll position when exiting. Some
5725 * values might be invalid. */
5726 if (!exiting)
5727 scroll_to_fraction(wp, prev_height);
Bram Moolenaar46328f92016-08-28 15:39:57 +02005728}
5729
5730 void
5731scroll_to_fraction(win_T *wp, int prev_height)
5732{
5733 linenr_T lnum;
5734 int sline, line_size;
5735 int height = wp->w_height;
5736
Bram Moolenaar071d4272004-06-13 20:20:40 +00005737 /* Don't change w_topline when height is zero. Don't set w_topline when
5738 * 'scrollbind' is set and this isn't the current window. */
Bram Moolenaar8a3bb562018-03-04 20:14:14 +01005739 if (height > 0 && (!wp->w_p_scb || wp == curwin))
Bram Moolenaar071d4272004-06-13 20:20:40 +00005740 {
Bram Moolenaar34114692005-01-02 11:28:13 +00005741 /*
5742 * Find a value for w_topline that shows the cursor at the same
5743 * relative position in the window as before (more or less).
5744 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005745 lnum = wp->w_cursor.lnum;
5746 if (lnum < 1) /* can happen when starting up */
5747 lnum = 1;
Bram Moolenaar56b3bf82014-05-07 20:25:35 +02005748 wp->w_wrow = ((long)wp->w_fraction * (long)height - 1L
5749 + FRACTION_MULT / 2) / FRACTION_MULT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005750 line_size = plines_win_col(wp, lnum, (long)(wp->w_cursor.col)) - 1;
5751 sline = wp->w_wrow - line_size;
Bram Moolenaar26470632006-10-24 19:12:40 +00005752
5753 if (sline >= 0)
5754 {
5755 /* Make sure the whole cursor line is visible, if possible. */
5756 int rows = plines_win(wp, lnum, FALSE);
5757
5758 if (sline > wp->w_height - rows)
5759 {
5760 sline = wp->w_height - rows;
5761 wp->w_wrow -= rows - line_size;
5762 }
5763 }
5764
Bram Moolenaar071d4272004-06-13 20:20:40 +00005765 if (sline < 0)
5766 {
5767 /*
5768 * Cursor line would go off top of screen if w_wrow was this high.
Bram Moolenaar26470632006-10-24 19:12:40 +00005769 * Make cursor line the first line in the window. If not enough
5770 * room use w_skipcol;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005771 */
5772 wp->w_wrow = line_size;
Bram Moolenaar26470632006-10-24 19:12:40 +00005773 if (wp->w_wrow >= wp->w_height
Bram Moolenaar02631462017-09-22 15:20:32 +02005774 && (wp->w_width - win_col_off(wp)) > 0)
Bram Moolenaar26470632006-10-24 19:12:40 +00005775 {
Bram Moolenaar02631462017-09-22 15:20:32 +02005776 wp->w_skipcol += wp->w_width - win_col_off(wp);
Bram Moolenaar26470632006-10-24 19:12:40 +00005777 --wp->w_wrow;
5778 while (wp->w_wrow >= wp->w_height)
5779 {
Bram Moolenaar02631462017-09-22 15:20:32 +02005780 wp->w_skipcol += wp->w_width - win_col_off(wp)
Bram Moolenaar26470632006-10-24 19:12:40 +00005781 + win_col_off2(wp);
5782 --wp->w_wrow;
5783 }
5784 }
Bram Moolenaarb4d21352014-07-16 14:16:46 +02005785 set_topline(wp, lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005786 }
Bram Moolenaardd0402a2014-05-28 13:43:04 +02005787 else if (sline > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005788 {
Bram Moolenaar26470632006-10-24 19:12:40 +00005789 while (sline > 0 && lnum > 1)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005790 {
5791#ifdef FEAT_FOLDING
5792 hasFoldingWin(wp, lnum, &lnum, NULL, TRUE, NULL);
5793 if (lnum == 1)
5794 {
5795 /* first line in buffer is folded */
5796 line_size = 1;
5797 --sline;
5798 break;
5799 }
5800#endif
5801 --lnum;
5802#ifdef FEAT_DIFF
5803 if (lnum == wp->w_topline)
5804 line_size = plines_win_nofill(wp, lnum, TRUE)
5805 + wp->w_topfill;
5806 else
5807#endif
5808 line_size = plines_win(wp, lnum, TRUE);
5809 sline -= line_size;
5810 }
Bram Moolenaar34114692005-01-02 11:28:13 +00005811
Bram Moolenaar071d4272004-06-13 20:20:40 +00005812 if (sline < 0)
5813 {
5814 /*
5815 * Line we want at top would go off top of screen. Use next
5816 * line instead.
5817 */
5818#ifdef FEAT_FOLDING
5819 hasFoldingWin(wp, lnum, NULL, &lnum, TRUE, NULL);
5820#endif
5821 lnum++;
5822 wp->w_wrow -= line_size + sline;
5823 }
Bram Moolenaardd0402a2014-05-28 13:43:04 +02005824 else if (sline > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005825 {
5826 /* First line of file reached, use that as topline. */
5827 lnum = 1;
5828 wp->w_wrow -= sline;
5829 }
Bram Moolenaardd0402a2014-05-28 13:43:04 +02005830
Bram Moolenaarb4d21352014-07-16 14:16:46 +02005831 set_topline(wp, lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005832 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00005833 }
5834
5835 if (wp == curwin)
5836 {
5837 if (p_so)
5838 update_topline();
5839 curs_columns(FALSE); /* validate w_wrow */
5840 }
Bram Moolenaar56b3bf82014-05-07 20:25:35 +02005841 if (prev_height > 0)
5842 wp->w_prev_fraction_row = wp->w_wrow;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005843
5844 win_comp_scroll(wp);
Bram Moolenaarf71a3db2006-03-12 21:50:18 +00005845 redraw_win_later(wp, SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005846 wp->w_redr_status = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005847 invalidate_botline_win(wp);
5848}
5849
Bram Moolenaar071d4272004-06-13 20:20:40 +00005850/*
5851 * Set the width of a window.
5852 */
Bram Moolenaar6763c142012-07-19 18:05:44 +02005853 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005854win_new_width(win_T *wp, int width)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005855{
5856 wp->w_width = width;
5857 wp->w_lines_valid = 0;
5858 changed_line_abv_curs_win(wp);
5859 invalidate_botline_win(wp);
5860 if (wp == curwin)
5861 {
5862 update_topline();
5863 curs_columns(TRUE); /* validate w_wrow */
5864 }
5865 redraw_win_later(wp, NOT_VALID);
5866 wp->w_redr_status = TRUE;
5867}
Bram Moolenaar071d4272004-06-13 20:20:40 +00005868
5869 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005870win_comp_scroll(win_T *wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005871{
5872 wp->w_p_scr = ((unsigned)wp->w_height >> 1);
5873 if (wp->w_p_scr == 0)
5874 wp->w_p_scr = 1;
5875}
5876
5877/*
5878 * command_height: called whenever p_ch has been changed
5879 */
5880 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005881command_height(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005882{
Bram Moolenaar071d4272004-06-13 20:20:40 +00005883 int h;
5884 frame_T *frp;
Bram Moolenaarc6fe9192006-04-09 21:54:49 +00005885 int old_p_ch = curtab->tp_ch_used;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005886
Bram Moolenaarc6fe9192006-04-09 21:54:49 +00005887 /* Use the value of p_ch that we remembered. This is needed for when the
5888 * GUI starts up, we can't be sure in what order things happen. And when
5889 * p_ch was changed in another tab page. */
5890 curtab->tp_ch_used = p_ch;
Bram Moolenaar05159a02005-02-26 23:04:13 +00005891
Bram Moolenaar071d4272004-06-13 20:20:40 +00005892 /* Find bottom frame with width of screen. */
5893 frp = lastwin->w_frame;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005894 while (frp->fr_width != Columns && frp->fr_parent != NULL)
5895 frp = frp->fr_parent;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005896
5897 /* Avoid changing the height of a window with 'winfixheight' set. */
5898 while (frp->fr_prev != NULL && frp->fr_layout == FR_LEAF
5899 && frp->fr_win->w_p_wfh)
5900 frp = frp->fr_prev;
5901
5902 if (starting != NO_SCREEN)
5903 {
5904 cmdline_row = Rows - p_ch;
5905
5906 if (p_ch > old_p_ch) /* p_ch got bigger */
5907 {
5908 while (p_ch > old_p_ch)
5909 {
5910 if (frp == NULL)
5911 {
5912 EMSG(_(e_noroom));
5913 p_ch = old_p_ch;
Bram Moolenaar719939c2007-09-25 12:51:28 +00005914 curtab->tp_ch_used = p_ch;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005915 cmdline_row = Rows - p_ch;
5916 break;
5917 }
5918 h = frp->fr_height - frame_minheight(frp, NULL);
5919 if (h > p_ch - old_p_ch)
5920 h = p_ch - old_p_ch;
5921 old_p_ch += h;
5922 frame_add_height(frp, -h);
5923 frp = frp->fr_prev;
5924 }
5925
5926 /* Recompute window positions. */
5927 (void)win_comp_pos();
5928
5929 /* clear the lines added to cmdline */
5930 if (full_screen)
5931 screen_fill((int)(cmdline_row), (int)Rows, 0,
5932 (int)Columns, ' ', ' ', 0);
5933 msg_row = cmdline_row;
5934 redraw_cmdline = TRUE;
5935 return;
5936 }
5937
5938 if (msg_row < cmdline_row)
5939 msg_row = cmdline_row;
5940 redraw_cmdline = TRUE;
5941 }
5942 frame_add_height(frp, (int)(old_p_ch - p_ch));
5943
5944 /* Recompute window positions. */
5945 if (frp != lastwin->w_frame)
5946 (void)win_comp_pos();
Bram Moolenaar071d4272004-06-13 20:20:40 +00005947}
5948
Bram Moolenaar071d4272004-06-13 20:20:40 +00005949/*
5950 * Resize frame "frp" to be "n" lines higher (negative for less high).
5951 * Also resize the frames it is contained in.
5952 */
5953 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005954frame_add_height(frame_T *frp, int n)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005955{
5956 frame_new_height(frp, frp->fr_height + n, FALSE, FALSE);
5957 for (;;)
5958 {
5959 frp = frp->fr_parent;
5960 if (frp == NULL)
5961 break;
5962 frp->fr_height += n;
5963 }
5964}
5965
5966/*
5967 * Add or remove a status line for the bottom window(s), according to the
5968 * value of 'laststatus'.
5969 */
5970 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005971last_status(
5972 int morewin) /* pretend there are two or more windows */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005973{
5974 /* Don't make a difference between horizontal or vertical split. */
5975 last_status_rec(topframe, (p_ls == 2
Bram Moolenaar459ca562016-11-10 18:16:33 +01005976 || (p_ls == 1 && (morewin || !ONE_WINDOW))));
Bram Moolenaar071d4272004-06-13 20:20:40 +00005977}
5978
5979 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01005980last_status_rec(frame_T *fr, int statusline)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005981{
5982 frame_T *fp;
5983 win_T *wp;
5984
5985 if (fr->fr_layout == FR_LEAF)
5986 {
5987 wp = fr->fr_win;
5988 if (wp->w_status_height != 0 && !statusline)
5989 {
5990 /* remove status line */
5991 win_new_height(wp, wp->w_height + 1);
5992 wp->w_status_height = 0;
5993 comp_col();
5994 }
5995 else if (wp->w_status_height == 0 && statusline)
5996 {
5997 /* Find a frame to take a line from. */
5998 fp = fr;
5999 while (fp->fr_height <= frame_minheight(fp, NULL))
6000 {
6001 if (fp == topframe)
6002 {
6003 EMSG(_(e_noroom));
6004 return;
6005 }
6006 /* In a column of frames: go to frame above. If already at
6007 * the top or in a row of frames: go to parent. */
6008 if (fp->fr_parent->fr_layout == FR_COL && fp->fr_prev != NULL)
6009 fp = fp->fr_prev;
6010 else
6011 fp = fp->fr_parent;
6012 }
6013 wp->w_status_height = 1;
6014 if (fp != fr)
6015 {
6016 frame_new_height(fp, fp->fr_height - 1, FALSE, FALSE);
6017 frame_fix_height(wp);
6018 (void)win_comp_pos();
6019 }
6020 else
6021 win_new_height(wp, wp->w_height - 1);
6022 comp_col();
Bram Moolenaarf71a3db2006-03-12 21:50:18 +00006023 redraw_all_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006024 }
6025 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006026 else if (fr->fr_layout == FR_ROW)
6027 {
6028 /* vertically split windows, set status line for each one */
6029 for (fp = fr->fr_child; fp != NULL; fp = fp->fr_next)
6030 last_status_rec(fp, statusline);
6031 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006032 else
6033 {
6034 /* horizontally split window, set status line for last one */
6035 for (fp = fr->fr_child; fp->fr_next != NULL; fp = fp->fr_next)
6036 ;
6037 last_status_rec(fp, statusline);
6038 }
6039}
6040
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00006041/*
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00006042 * Return the number of lines used by the tab page line.
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00006043 */
6044 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006045tabline_height(void)
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00006046{
Bram Moolenaar32466aa2006-02-24 23:53:04 +00006047#ifdef FEAT_GUI_TABLINE
6048 /* When the GUI has the tabline then this always returns zero. */
6049 if (gui_use_tabline())
6050 return 0;
6051#endif
Bram Moolenaar2a0449d2006-02-20 21:27:21 +00006052 switch (p_stal)
Bram Moolenaar98ea5de2006-02-15 22:11:25 +00006053 {
6054 case 0: return 0;
6055 case 1: return (first_tabpage->tp_next == NULL) ? 0 : 1;
6056 }
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00006057 return 1;
6058}
6059
Bram Moolenaar071d4272004-06-13 20:20:40 +00006060#if defined(FEAT_SEARCHPATH) || defined(PROTO)
6061/*
Bram Moolenaard857f0e2005-06-21 22:37:39 +00006062 * Get the file name at the cursor.
6063 * If Visual mode is active, use the selected text if it's in one line.
6064 * Returns the name in allocated memory, NULL for failure.
6065 */
6066 char_u *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006067grab_file_name(long count, linenr_T *file_lnum)
Bram Moolenaard857f0e2005-06-21 22:37:39 +00006068{
Bram Moolenaard45c07a2015-02-27 17:19:10 +01006069 int options = FNAME_MESS|FNAME_EXP|FNAME_REL|FNAME_UNESC;
6070
Bram Moolenaard857f0e2005-06-21 22:37:39 +00006071 if (VIsual_active)
6072 {
6073 int len;
6074 char_u *ptr;
6075
6076 if (get_visual_text(NULL, &ptr, &len) == FAIL)
6077 return NULL;
Bram Moolenaard45c07a2015-02-27 17:19:10 +01006078 return find_file_name_in_path(ptr, len, options,
6079 count, curbuf->b_ffname);
Bram Moolenaard857f0e2005-06-21 22:37:39 +00006080 }
Bram Moolenaard45c07a2015-02-27 17:19:10 +01006081 return file_name_at_cursor(options | FNAME_HYP, count, file_lnum);
Bram Moolenaard857f0e2005-06-21 22:37:39 +00006082}
6083
6084/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00006085 * Return the file name under or after the cursor.
6086 *
6087 * The 'path' option is searched if the file name is not absolute.
6088 * The string returned has been alloc'ed and should be freed by the caller.
6089 * NULL is returned if the file name or file is not found.
6090 *
6091 * options:
6092 * FNAME_MESS give error messages
6093 * FNAME_EXP expand to path
6094 * FNAME_HYP check for hypertext link
6095 * FNAME_INCL apply "includeexpr"
6096 */
6097 char_u *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006098file_name_at_cursor(int options, long count, linenr_T *file_lnum)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006099{
6100 return file_name_in_line(ml_get_curline(),
Bram Moolenaard1f56e62006-02-22 21:25:37 +00006101 curwin->w_cursor.col, options, count, curbuf->b_ffname,
6102 file_lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006103}
6104
6105/*
6106 * Return the name of the file under or after ptr[col].
6107 * Otherwise like file_name_at_cursor().
6108 */
6109 char_u *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006110file_name_in_line(
6111 char_u *line,
6112 int col,
6113 int options,
6114 long count,
6115 char_u *rel_fname, /* file we are searching relative to */
6116 linenr_T *file_lnum) /* line number after the file name */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006117{
6118 char_u *ptr;
6119 int len;
Bram Moolenaar9e3dfc62017-12-25 14:29:18 +01006120 int in_type = TRUE;
6121 int is_url = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006122
6123 /*
6124 * search forward for what could be the start of a file name
6125 */
6126 ptr = line + col;
6127 while (*ptr != NUL && !vim_isfilec(*ptr))
Bram Moolenaar91acfff2017-03-12 19:22:36 +01006128 MB_PTR_ADV(ptr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006129 if (*ptr == NUL) /* nothing found */
6130 {
6131 if (options & FNAME_MESS)
6132 EMSG(_("E446: No file name under cursor"));
6133 return NULL;
6134 }
6135
6136 /*
6137 * Search backward for first char of the file name.
6138 * Go one char back to ":" before "//" even when ':' is not in 'isfname'.
6139 */
6140 while (ptr > line)
6141 {
6142#ifdef FEAT_MBYTE
6143 if (has_mbyte && (len = (*mb_head_off)(line, ptr - 1)) > 0)
6144 ptr -= len + 1;
6145 else
6146#endif
6147 if (vim_isfilec(ptr[-1])
6148 || ((options & FNAME_HYP) && path_is_url(ptr - 1)))
6149 --ptr;
6150 else
6151 break;
6152 }
6153
6154 /*
6155 * Search forward for the last char of the file name.
6156 * Also allow "://" when ':' is not in 'isfname'.
6157 */
6158 len = 0;
Bram Moolenaard45c07a2015-02-27 17:19:10 +01006159 while (vim_isfilec(ptr[len]) || (ptr[len] == '\\' && ptr[len + 1] == ' ')
Bram Moolenaar9e3dfc62017-12-25 14:29:18 +01006160 || ((options & FNAME_HYP) && path_is_url(ptr + len))
6161 || (is_url && vim_strchr((char_u *)"?&=", ptr[len]) != NULL))
Bram Moolenaard45c07a2015-02-27 17:19:10 +01006162 {
Bram Moolenaar9e3dfc62017-12-25 14:29:18 +01006163 /* After type:// we also include ?, & and = as valid characters, so that
6164 * http://google.com?q=this&that=ok works. */
6165 if ((ptr[len] >= 'A' && ptr[len] <= 'Z') || (ptr[len] >= 'a' && ptr[len] <= 'z'))
6166 {
6167 if (in_type && path_is_url(ptr + len + 1))
6168 is_url = TRUE;
6169 }
6170 else
6171 in_type = FALSE;
6172
Bram Moolenaard45c07a2015-02-27 17:19:10 +01006173 if (ptr[len] == '\\')
6174 /* Skip over the "\" in "\ ". */
6175 ++len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006176#ifdef FEAT_MBYTE
6177 if (has_mbyte)
Bram Moolenaar0fa313a2005-08-10 21:07:57 +00006178 len += (*mb_ptr2len)(ptr + len);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006179 else
6180#endif
6181 ++len;
Bram Moolenaard45c07a2015-02-27 17:19:10 +01006182 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006183
6184 /*
6185 * If there is trailing punctuation, remove it.
6186 * But don't remove "..", could be a directory name.
6187 */
6188 if (len > 2 && vim_strchr((char_u *)".,:;!", ptr[len - 1]) != NULL
6189 && ptr[len - 2] != '.')
6190 --len;
6191
Bram Moolenaard1f56e62006-02-22 21:25:37 +00006192 if (file_lnum != NULL)
6193 {
6194 char_u *p;
6195
6196 /* Get the number after the file name and a separator character */
6197 p = ptr + len;
6198 p = skipwhite(p);
6199 if (*p != NUL)
6200 {
6201 if (!isdigit(*p))
6202 ++p; /* skip the separator */
6203 p = skipwhite(p);
6204 if (isdigit(*p))
6205 *file_lnum = (int)getdigits(&p);
6206 }
6207 }
6208
Bram Moolenaar071d4272004-06-13 20:20:40 +00006209 return find_file_name_in_path(ptr, len, options, count, rel_fname);
6210}
6211
6212# if defined(FEAT_FIND_ID) && defined(FEAT_EVAL)
Bram Moolenaarbaaa7e92016-01-29 22:47:03 +01006213static char_u *eval_includeexpr(char_u *ptr, int len);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006214
6215 static char_u *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006216eval_includeexpr(char_u *ptr, int len)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006217{
6218 char_u *res;
6219
6220 set_vim_var_string(VV_FNAME, ptr, len);
Bram Moolenaar2d3f4892006-01-20 23:02:51 +00006221 res = eval_to_string_safe(curbuf->b_p_inex, NULL,
Bram Moolenaard1f56e62006-02-22 21:25:37 +00006222 was_set_insecurely((char_u *)"includeexpr", OPT_LOCAL));
Bram Moolenaar071d4272004-06-13 20:20:40 +00006223 set_vim_var_string(VV_FNAME, NULL, 0);
6224 return res;
6225}
6226#endif
6227
6228/*
6229 * Return the name of the file ptr[len] in 'path'.
6230 * Otherwise like file_name_at_cursor().
6231 */
6232 char_u *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006233find_file_name_in_path(
6234 char_u *ptr,
6235 int len,
6236 int options,
6237 long count,
6238 char_u *rel_fname) /* file we are searching relative to */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006239{
6240 char_u *file_name;
6241 int c;
6242# if defined(FEAT_FIND_ID) && defined(FEAT_EVAL)
6243 char_u *tofree = NULL;
6244
6245 if ((options & FNAME_INCL) && *curbuf->b_p_inex != NUL)
6246 {
6247 tofree = eval_includeexpr(ptr, len);
6248 if (tofree != NULL)
6249 {
6250 ptr = tofree;
6251 len = (int)STRLEN(ptr);
6252 }
6253 }
6254# endif
6255
6256 if (options & FNAME_EXP)
6257 {
6258 file_name = find_file_in_path(ptr, len, options & ~FNAME_MESS,
6259 TRUE, rel_fname);
6260
6261# if defined(FEAT_FIND_ID) && defined(FEAT_EVAL)
6262 /*
6263 * If the file could not be found in a normal way, try applying
6264 * 'includeexpr' (unless done already).
6265 */
6266 if (file_name == NULL
6267 && !(options & FNAME_INCL) && *curbuf->b_p_inex != NUL)
6268 {
6269 tofree = eval_includeexpr(ptr, len);
6270 if (tofree != NULL)
6271 {
6272 ptr = tofree;
6273 len = (int)STRLEN(ptr);
6274 file_name = find_file_in_path(ptr, len, options & ~FNAME_MESS,
6275 TRUE, rel_fname);
6276 }
6277 }
6278# endif
6279 if (file_name == NULL && (options & FNAME_MESS))
6280 {
6281 c = ptr[len];
6282 ptr[len] = NUL;
6283 EMSG2(_("E447: Can't find file \"%s\" in path"), ptr);
6284 ptr[len] = c;
6285 }
6286
6287 /* Repeat finding the file "count" times. This matters when it
6288 * appears several times in the path. */
6289 while (file_name != NULL && --count > 0)
6290 {
6291 vim_free(file_name);
6292 file_name = find_file_in_path(ptr, len, options, FALSE, rel_fname);
6293 }
6294 }
6295 else
6296 file_name = vim_strnsave(ptr, len);
6297
6298# if defined(FEAT_FIND_ID) && defined(FEAT_EVAL)
6299 vim_free(tofree);
6300# endif
6301
6302 return file_name;
6303}
6304#endif /* FEAT_SEARCHPATH */
6305
6306/*
6307 * Check if the "://" of a URL is at the pointer, return URL_SLASH.
6308 * Also check for ":\\", which MS Internet Explorer accepts, return
6309 * URL_BACKSLASH.
6310 */
6311 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006312path_is_url(char_u *p)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006313{
6314 if (STRNCMP(p, "://", (size_t)3) == 0)
6315 return URL_SLASH;
6316 else if (STRNCMP(p, ":\\\\", (size_t)3) == 0)
6317 return URL_BACKSLASH;
6318 return 0;
6319}
6320
6321/*
6322 * Check if "fname" starts with "name://". Return URL_SLASH if it does.
6323 * Return URL_BACKSLASH for "name:\\".
6324 * Return zero otherwise.
6325 */
6326 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006327path_with_url(char_u *fname)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006328{
6329 char_u *p;
6330
6331 for (p = fname; isalpha(*p); ++p)
6332 ;
6333 return path_is_url(p);
6334}
6335
6336/*
6337 * Return TRUE if "name" is a full (absolute) path name or URL.
6338 */
6339 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006340vim_isAbsName(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006341{
6342 return (path_with_url(name) != 0 || mch_isFullName(name));
6343}
6344
6345/*
Bram Moolenaar5b962cf2005-12-12 21:58:40 +00006346 * Get absolute file name into buffer "buf[len]".
Bram Moolenaar071d4272004-06-13 20:20:40 +00006347 *
6348 * return FAIL for failure, OK otherwise
6349 */
6350 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006351vim_FullName(
6352 char_u *fname,
6353 char_u *buf,
6354 int len,
6355 int force) /* force expansion even when already absolute */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006356{
6357 int retval = OK;
6358 int url;
6359
6360 *buf = NUL;
6361 if (fname == NULL)
6362 return FAIL;
6363
6364 url = path_with_url(fname);
6365 if (!url)
6366 retval = mch_FullName(fname, buf, len, force);
6367 if (url || retval == FAIL)
6368 {
6369 /* something failed; use the file name (truncate when too long) */
Bram Moolenaarb6356332005-07-18 21:40:44 +00006370 vim_strncpy(buf, fname, len - 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006371 }
Bram Moolenaard0573012017-10-28 21:11:06 +02006372#if defined(MSWIN)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006373 slash_adjust(buf);
6374#endif
6375 return retval;
6376}
6377
6378/*
6379 * Return the minimal number of rows that is needed on the screen to display
6380 * the current number of windows.
6381 */
6382 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006383min_rows(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006384{
6385 int total;
Bram Moolenaarf740b292006-02-16 22:11:02 +00006386 tabpage_T *tp;
6387 int n;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006388
6389 if (firstwin == NULL) /* not initialized yet */
6390 return MIN_LINES;
6391
Bram Moolenaarf740b292006-02-16 22:11:02 +00006392 total = 0;
Bram Moolenaar29323592016-07-24 22:04:11 +02006393 FOR_ALL_TABPAGES(tp)
Bram Moolenaarf740b292006-02-16 22:11:02 +00006394 {
6395 n = frame_minheight(tp->tp_topframe, NULL);
6396 if (total < n)
6397 total = n;
6398 }
Bram Moolenaar32466aa2006-02-24 23:53:04 +00006399 total += tabline_height();
Bram Moolenaarf740b292006-02-16 22:11:02 +00006400 total += 1; /* count the room for the command line */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006401 return total;
6402}
6403
6404/*
Bram Moolenaar49d7bf12006-02-17 21:45:41 +00006405 * Return TRUE if there is only one window (in the current tab page), not
6406 * counting a help or preview window, unless it is the current window.
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006407 * Does not count "aucmd_win".
Bram Moolenaar071d4272004-06-13 20:20:40 +00006408 */
6409 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006410only_one_window(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006411{
Bram Moolenaar071d4272004-06-13 20:20:40 +00006412 int count = 0;
6413 win_T *wp;
6414
Bram Moolenaar1d2ba7f2006-02-14 22:29:30 +00006415 /* If there is another tab page there always is another window. */
6416 if (first_tabpage->tp_next != NULL)
6417 return FALSE;
6418
Bram Moolenaar29323592016-07-24 22:04:11 +02006419 FOR_ALL_WINDOWS(wp)
Bram Moolenaar802418d2013-01-17 14:00:11 +01006420 if (wp->w_buffer != NULL
Bram Moolenaard28cc3f2017-07-27 22:03:50 +02006421 && (!((bt_help(wp->w_buffer) && !bt_help(curbuf))
Bram Moolenaar071d4272004-06-13 20:20:40 +00006422# ifdef FEAT_QUICKFIX
6423 || wp->w_p_pvw
6424# endif
Bram Moolenaarf2bd8ef2018-03-04 18:08:14 +01006425 ) || wp == curwin) && wp != aucmd_win)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006426 ++count;
6427 return (count <= 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006428}
6429
Bram Moolenaar071d4272004-06-13 20:20:40 +00006430/*
6431 * Correct the cursor line number in other windows. Used after changing the
6432 * current buffer, and before applying autocommands.
6433 * When "do_curwin" is TRUE, also check current window.
6434 */
6435 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006436check_lnums(int do_curwin)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006437{
6438 win_T *wp;
Bram Moolenaarf740b292006-02-16 22:11:02 +00006439 tabpage_T *tp;
6440
6441 FOR_ALL_TAB_WINDOWS(tp, wp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006442 if ((do_curwin || wp != curwin) && wp->w_buffer == curbuf)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006443 {
6444 if (wp->w_cursor.lnum > curbuf->b_ml.ml_line_count)
6445 wp->w_cursor.lnum = curbuf->b_ml.ml_line_count;
6446 if (wp->w_topline > curbuf->b_ml.ml_line_count)
6447 wp->w_topline = curbuf->b_ml.ml_line_count;
6448 }
6449}
Bram Moolenaar071d4272004-06-13 20:20:40 +00006450
6451/*
6452 * A snapshot of the window sizes, to restore them after closing the help
6453 * window.
6454 * Only these fields are used:
6455 * fr_layout
6456 * fr_width
6457 * fr_height
6458 * fr_next
6459 * fr_child
6460 * fr_win (only valid for the old curwin, NULL otherwise)
6461 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006462
6463/*
6464 * Create a snapshot of the current frame sizes.
6465 */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006466 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006467make_snapshot(int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006468{
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006469 clear_snapshot(curtab, idx);
6470 make_snapshot_rec(topframe, &curtab->tp_snapshot[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006471}
6472
6473 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006474make_snapshot_rec(frame_T *fr, frame_T **frp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006475{
6476 *frp = (frame_T *)alloc_clear((unsigned)sizeof(frame_T));
6477 if (*frp == NULL)
6478 return;
6479 (*frp)->fr_layout = fr->fr_layout;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006480 (*frp)->fr_width = fr->fr_width;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006481 (*frp)->fr_height = fr->fr_height;
6482 if (fr->fr_next != NULL)
6483 make_snapshot_rec(fr->fr_next, &((*frp)->fr_next));
6484 if (fr->fr_child != NULL)
6485 make_snapshot_rec(fr->fr_child, &((*frp)->fr_child));
6486 if (fr->fr_layout == FR_LEAF && fr->fr_win == curwin)
6487 (*frp)->fr_win = curwin;
6488}
6489
6490/*
6491 * Remove any existing snapshot.
6492 */
6493 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006494clear_snapshot(tabpage_T *tp, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006495{
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006496 clear_snapshot_rec(tp->tp_snapshot[idx]);
6497 tp->tp_snapshot[idx] = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006498}
6499
6500 static void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006501clear_snapshot_rec(frame_T *fr)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006502{
6503 if (fr != NULL)
6504 {
6505 clear_snapshot_rec(fr->fr_next);
6506 clear_snapshot_rec(fr->fr_child);
6507 vim_free(fr);
6508 }
6509}
6510
6511/*
6512 * Restore a previously created snapshot, if there is any.
6513 * This is only done if the screen size didn't change and the window layout is
6514 * still the same.
6515 */
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006516 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006517restore_snapshot(
6518 int idx,
6519 int close_curwin) /* closing current window */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006520{
6521 win_T *wp;
6522
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006523 if (curtab->tp_snapshot[idx] != NULL
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006524 && curtab->tp_snapshot[idx]->fr_width == topframe->fr_width
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006525 && curtab->tp_snapshot[idx]->fr_height == topframe->fr_height
6526 && check_snapshot_rec(curtab->tp_snapshot[idx], topframe) == OK)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006527 {
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006528 wp = restore_snapshot_rec(curtab->tp_snapshot[idx], topframe);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006529 win_comp_pos();
6530 if (wp != NULL && close_curwin)
6531 win_goto(wp);
6532 redraw_all_later(CLEAR);
6533 }
Bram Moolenaar746ebd32009-06-16 14:01:43 +00006534 clear_snapshot(curtab, idx);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006535}
6536
6537/*
6538 * Check if frames "sn" and "fr" have the same layout, same following frames
Bram Moolenaar343b8c02017-02-17 12:04:56 +01006539 * and same children. And the window pointer is valid.
Bram Moolenaar071d4272004-06-13 20:20:40 +00006540 */
6541 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006542check_snapshot_rec(frame_T *sn, frame_T *fr)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006543{
6544 if (sn->fr_layout != fr->fr_layout
6545 || (sn->fr_next == NULL) != (fr->fr_next == NULL)
6546 || (sn->fr_child == NULL) != (fr->fr_child == NULL)
6547 || (sn->fr_next != NULL
6548 && check_snapshot_rec(sn->fr_next, fr->fr_next) == FAIL)
6549 || (sn->fr_child != NULL
Bram Moolenaar343b8c02017-02-17 12:04:56 +01006550 && check_snapshot_rec(sn->fr_child, fr->fr_child) == FAIL)
Bram Moolenaar2c90d512017-03-18 22:35:30 +01006551 || (sn->fr_win != NULL && !win_valid(sn->fr_win)))
Bram Moolenaar071d4272004-06-13 20:20:40 +00006552 return FAIL;
6553 return OK;
6554}
6555
6556/*
6557 * Copy the size of snapshot frame "sn" to frame "fr". Do the same for all
6558 * following frames and children.
6559 * Returns a pointer to the old current window, or NULL.
6560 */
6561 static win_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006562restore_snapshot_rec(frame_T *sn, frame_T *fr)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006563{
6564 win_T *wp = NULL;
6565 win_T *wp2;
6566
6567 fr->fr_height = sn->fr_height;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006568 fr->fr_width = sn->fr_width;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006569 if (fr->fr_layout == FR_LEAF)
6570 {
6571 frame_new_height(fr, fr->fr_height, FALSE, FALSE);
Bram Moolenaarbe4d5062006-03-18 21:30:13 +00006572 frame_new_width(fr, fr->fr_width, FALSE, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006573 wp = sn->fr_win;
6574 }
6575 if (sn->fr_next != NULL)
6576 {
6577 wp2 = restore_snapshot_rec(sn->fr_next, fr->fr_next);
6578 if (wp2 != NULL)
6579 wp = wp2;
6580 }
6581 if (sn->fr_child != NULL)
6582 {
6583 wp2 = restore_snapshot_rec(sn->fr_child, fr->fr_child);
6584 if (wp2 != NULL)
6585 wp = wp2;
6586 }
6587 return wp;
6588}
6589
Bram Moolenaar95064ec2013-07-17 17:15:25 +02006590#if defined(FEAT_EVAL) || defined(FEAT_PYTHON) || defined(FEAT_PYTHON3) \
6591 || defined(PROTO)
Bram Moolenaar105bc352013-05-17 16:03:57 +02006592/*
6593 * Set "win" to be the curwin and "tp" to be the current tab page.
Bram Moolenaar5d2bae82014-09-19 14:26:36 +02006594 * restore_win() MUST be called to undo, also when FAIL is returned.
6595 * No autocommands will be executed until restore_win() is called.
Bram Moolenaard6949742013-06-16 14:18:28 +02006596 * When "no_display" is TRUE the display won't be affected, no redraw is
6597 * triggered, another tabpage access is limited.
Bram Moolenaar105bc352013-05-17 16:03:57 +02006598 * Returns FAIL if switching to "win" failed.
6599 */
6600 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006601switch_win(
Bram Moolenaar816968d2017-09-29 21:29:18 +02006602 win_T **save_curwin,
6603 tabpage_T **save_curtab,
6604 win_T *win,
6605 tabpage_T *tp,
6606 int no_display)
Bram Moolenaar105bc352013-05-17 16:03:57 +02006607{
Bram Moolenaar105bc352013-05-17 16:03:57 +02006608 block_autocmds();
Bram Moolenaar105bc352013-05-17 16:03:57 +02006609 *save_curwin = curwin;
6610 if (tp != NULL)
6611 {
6612 *save_curtab = curtab;
Bram Moolenaard6949742013-06-16 14:18:28 +02006613 if (no_display)
6614 {
6615 curtab->tp_firstwin = firstwin;
6616 curtab->tp_lastwin = lastwin;
6617 curtab = tp;
6618 firstwin = curtab->tp_firstwin;
6619 lastwin = curtab->tp_lastwin;
6620 }
6621 else
6622 goto_tabpage_tp(tp, FALSE, FALSE);
Bram Moolenaar105bc352013-05-17 16:03:57 +02006623 }
6624 if (!win_valid(win))
Bram Moolenaar105bc352013-05-17 16:03:57 +02006625 return FAIL;
Bram Moolenaar105bc352013-05-17 16:03:57 +02006626 curwin = win;
6627 curbuf = curwin->w_buffer;
Bram Moolenaar105bc352013-05-17 16:03:57 +02006628 return OK;
6629}
6630
6631/*
6632 * Restore current tabpage and window saved by switch_win(), if still valid.
Bram Moolenaard6949742013-06-16 14:18:28 +02006633 * When "no_display" is TRUE the display won't be affected, no redraw is
6634 * triggered.
Bram Moolenaar105bc352013-05-17 16:03:57 +02006635 */
6636 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006637restore_win(
6638 win_T *save_curwin UNUSED,
6639 tabpage_T *save_curtab UNUSED,
6640 int no_display UNUSED)
Bram Moolenaar105bc352013-05-17 16:03:57 +02006641{
Bram Moolenaar105bc352013-05-17 16:03:57 +02006642 if (save_curtab != NULL && valid_tabpage(save_curtab))
Bram Moolenaard6949742013-06-16 14:18:28 +02006643 {
6644 if (no_display)
6645 {
6646 curtab->tp_firstwin = firstwin;
6647 curtab->tp_lastwin = lastwin;
6648 curtab = save_curtab;
6649 firstwin = curtab->tp_firstwin;
6650 lastwin = curtab->tp_lastwin;
6651 }
6652 else
6653 goto_tabpage_tp(save_curtab, FALSE, FALSE);
6654 }
Bram Moolenaar105bc352013-05-17 16:03:57 +02006655 if (win_valid(save_curwin))
6656 {
6657 curwin = save_curwin;
6658 curbuf = curwin->w_buffer;
6659 }
Bram Moolenaar105bc352013-05-17 16:03:57 +02006660 unblock_autocmds();
Bram Moolenaar105bc352013-05-17 16:03:57 +02006661}
6662
6663/*
6664 * Make "buf" the current buffer. restore_buffer() MUST be called to undo.
6665 * No autocommands will be executed. Use aucmd_prepbuf() if there are any.
6666 */
6667 void
Bram Moolenaar7c0a2f32016-07-10 22:11:16 +02006668switch_buffer(bufref_T *save_curbuf, buf_T *buf)
Bram Moolenaar105bc352013-05-17 16:03:57 +02006669{
Bram Moolenaar105bc352013-05-17 16:03:57 +02006670 block_autocmds();
Bram Moolenaar7c0a2f32016-07-10 22:11:16 +02006671 set_bufref(save_curbuf, curbuf);
Bram Moolenaar105bc352013-05-17 16:03:57 +02006672 --curbuf->b_nwindows;
6673 curbuf = buf;
6674 curwin->w_buffer = buf;
6675 ++curbuf->b_nwindows;
6676}
6677
6678/*
6679 * Restore the current buffer after using switch_buffer().
6680 */
6681 void
Bram Moolenaar7c0a2f32016-07-10 22:11:16 +02006682restore_buffer(bufref_T *save_curbuf)
Bram Moolenaar105bc352013-05-17 16:03:57 +02006683{
Bram Moolenaar105bc352013-05-17 16:03:57 +02006684 unblock_autocmds();
Bram Moolenaar105bc352013-05-17 16:03:57 +02006685 /* Check for valid buffer, just in case. */
Bram Moolenaar7c0a2f32016-07-10 22:11:16 +02006686 if (bufref_valid(save_curbuf))
Bram Moolenaar105bc352013-05-17 16:03:57 +02006687 {
6688 --curbuf->b_nwindows;
Bram Moolenaar7c0a2f32016-07-10 22:11:16 +02006689 curwin->w_buffer = save_curbuf->br_buf;
6690 curbuf = save_curbuf->br_buf;
Bram Moolenaar105bc352013-05-17 16:03:57 +02006691 ++curbuf->b_nwindows;
6692 }
6693}
6694#endif
6695
Bram Moolenaar4033c552017-09-16 20:54:51 +02006696#if defined(FEAT_GUI) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006697/*
6698 * Return TRUE if there is any vertically split window.
6699 */
6700 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006701win_hasvertsplit(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006702{
6703 frame_T *fr;
6704
6705 if (topframe->fr_layout == FR_ROW)
6706 return TRUE;
6707
6708 if (topframe->fr_layout == FR_COL)
6709 for (fr = topframe->fr_child; fr != NULL; fr = fr->fr_next)
6710 if (fr->fr_layout == FR_ROW)
6711 return TRUE;
6712
6713 return FALSE;
6714}
6715#endif
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006716
6717#if defined(FEAT_SEARCH_EXTRA) || defined(PROTO)
6718/*
6719 * Add match to the match list of window 'wp'. The pattern 'pat' will be
Bram Moolenaare37d50a2008-08-06 17:06:04 +00006720 * highlighted with the group 'grp' with priority 'prio'.
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006721 * Optionally, a desired ID 'id' can be specified (greater than or equal to 1).
6722 * If no particular ID is desired, -1 must be specified for 'id'.
6723 * Return ID of added match, -1 on failure.
6724 */
6725 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006726match_add(
6727 win_T *wp,
6728 char_u *grp,
6729 char_u *pat,
6730 int prio,
6731 int id,
6732 list_T *pos_list,
6733 char_u *conceal_char UNUSED) /* pointer to conceal replacement char */
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006734{
Bram Moolenaarb3414592014-06-17 17:48:32 +02006735 matchitem_T *cur;
6736 matchitem_T *prev;
6737 matchitem_T *m;
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006738 int hlg_id;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006739 regprog_T *regprog = NULL;
6740 int rtype = SOME_VALID;
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006741
Bram Moolenaarb3414592014-06-17 17:48:32 +02006742 if (*grp == NUL || (pat != NULL && *pat == NUL))
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006743 return -1;
6744 if (id < -1 || id == 0)
6745 {
Bram Moolenaar5b302912016-08-24 22:11:55 +02006746 EMSGN(_("E799: Invalid ID: %ld (must be greater than or equal to 1)"), id);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006747 return -1;
6748 }
6749 if (id != -1)
6750 {
6751 cur = wp->w_match_head;
6752 while (cur != NULL)
6753 {
6754 if (cur->id == id)
6755 {
Bram Moolenaar5b302912016-08-24 22:11:55 +02006756 EMSGN(_("E801: ID already taken: %ld"), id);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006757 return -1;
6758 }
6759 cur = cur->next;
6760 }
6761 }
Bram Moolenaarcb4cef22008-03-16 15:04:34 +00006762 if ((hlg_id = syn_namen2id(grp, (int)STRLEN(grp))) == 0)
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006763 {
6764 EMSG2(_(e_nogroup), grp);
6765 return -1;
6766 }
Bram Moolenaarb3414592014-06-17 17:48:32 +02006767 if (pat != NULL && (regprog = vim_regcomp(pat, RE_MAGIC)) == NULL)
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006768 {
6769 EMSG2(_(e_invarg2), pat);
6770 return -1;
6771 }
6772
6773 /* Find available match ID. */
6774 while (id == -1)
6775 {
6776 cur = wp->w_match_head;
6777 while (cur != NULL && cur->id != wp->w_next_match_id)
6778 cur = cur->next;
6779 if (cur == NULL)
6780 id = wp->w_next_match_id;
6781 wp->w_next_match_id++;
6782 }
6783
6784 /* Build new match. */
Bram Moolenaardeae0f22014-06-18 21:20:11 +02006785 m = (matchitem_T *)alloc_clear(sizeof(matchitem_T));
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006786 m->id = id;
6787 m->priority = prio;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006788 m->pattern = pat == NULL ? NULL : vim_strsave(pat);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006789 m->hlg_id = hlg_id;
Bram Moolenaar0963cd92007-08-05 16:49:43 +00006790 m->match.regprog = regprog;
6791 m->match.rmm_ic = FALSE;
6792 m->match.rmm_maxcol = 0;
Bram Moolenaar42356152016-03-31 22:27:40 +02006793# if defined(FEAT_CONCEAL) && defined(FEAT_MBYTE)
Bram Moolenaar6561d522015-07-21 15:48:27 +02006794 m->conceal_char = 0;
6795 if (conceal_char != NULL)
6796 m->conceal_char = (*mb_ptr2char)(conceal_char);
Bram Moolenaar42356152016-03-31 22:27:40 +02006797# endif
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006798
Bram Moolenaarb3414592014-06-17 17:48:32 +02006799 /* Set up position matches */
6800 if (pos_list != NULL)
6801 {
6802 linenr_T toplnum = 0;
6803 linenr_T botlnum = 0;
6804 listitem_T *li;
6805 int i;
6806
Bram Moolenaarb6da44a2014-06-25 18:15:22 +02006807 for (i = 0, li = pos_list->lv_first; li != NULL && i < MAXPOSMATCH;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006808 i++, li = li->li_next)
6809 {
6810 linenr_T lnum = 0;
6811 colnr_T col = 0;
6812 int len = 1;
6813 list_T *subl;
6814 listitem_T *subli;
Bram Moolenaardeae0f22014-06-18 21:20:11 +02006815 int error = FALSE;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006816
Bram Moolenaarb3414592014-06-17 17:48:32 +02006817 if (li->li_tv.v_type == VAR_LIST)
6818 {
6819 subl = li->li_tv.vval.v_list;
6820 if (subl == NULL)
6821 goto fail;
6822 subli = subl->lv_first;
6823 if (subli == NULL)
6824 goto fail;
6825 lnum = get_tv_number_chk(&subli->li_tv, &error);
6826 if (error == TRUE)
6827 goto fail;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006828 if (lnum == 0)
6829 {
6830 --i;
6831 continue;
6832 }
Bram Moolenaarb6da44a2014-06-25 18:15:22 +02006833 m->pos.pos[i].lnum = lnum;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006834 subli = subli->li_next;
6835 if (subli != NULL)
6836 {
6837 col = get_tv_number_chk(&subli->li_tv, &error);
6838 if (error == TRUE)
6839 goto fail;
6840 subli = subli->li_next;
6841 if (subli != NULL)
6842 {
6843 len = get_tv_number_chk(&subli->li_tv, &error);
6844 if (error == TRUE)
6845 goto fail;
6846 }
6847 }
6848 m->pos.pos[i].col = col;
6849 m->pos.pos[i].len = len;
6850 }
6851 else if (li->li_tv.v_type == VAR_NUMBER)
6852 {
6853 if (li->li_tv.vval.v_number == 0)
Bram Moolenaarb6da44a2014-06-25 18:15:22 +02006854 {
6855 --i;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006856 continue;
Bram Moolenaarb6da44a2014-06-25 18:15:22 +02006857 }
Bram Moolenaarb3414592014-06-17 17:48:32 +02006858 m->pos.pos[i].lnum = li->li_tv.vval.v_number;
6859 m->pos.pos[i].col = 0;
6860 m->pos.pos[i].len = 0;
6861 }
6862 else
6863 {
6864 EMSG(_("List or number required"));
6865 goto fail;
6866 }
6867 if (toplnum == 0 || lnum < toplnum)
6868 toplnum = lnum;
Bram Moolenaar41d75232014-06-25 17:58:11 +02006869 if (botlnum == 0 || lnum >= botlnum)
6870 botlnum = lnum + 1;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006871 }
6872
6873 /* Calculate top and bottom lines for redrawing area */
6874 if (toplnum != 0)
6875 {
6876 if (wp->w_buffer->b_mod_set)
6877 {
6878 if (wp->w_buffer->b_mod_top > toplnum)
6879 wp->w_buffer->b_mod_top = toplnum;
6880 if (wp->w_buffer->b_mod_bot < botlnum)
6881 wp->w_buffer->b_mod_bot = botlnum;
6882 }
6883 else
6884 {
Bram Moolenaardab70c62014-07-02 17:16:58 +02006885 wp->w_buffer->b_mod_set = TRUE;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006886 wp->w_buffer->b_mod_top = toplnum;
6887 wp->w_buffer->b_mod_bot = botlnum;
Bram Moolenaardab70c62014-07-02 17:16:58 +02006888 wp->w_buffer->b_mod_xlines = 0;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006889 }
6890 m->pos.toplnum = toplnum;
6891 m->pos.botlnum = botlnum;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006892 rtype = VALID;
6893 }
6894 }
6895
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006896 /* Insert new match. The match list is in ascending order with regard to
6897 * the match priorities. */
6898 cur = wp->w_match_head;
6899 prev = cur;
6900 while (cur != NULL && prio >= cur->priority)
6901 {
6902 prev = cur;
6903 cur = cur->next;
6904 }
6905 if (cur == prev)
6906 wp->w_match_head = m;
6907 else
6908 prev->next = m;
6909 m->next = cur;
6910
Bram Moolenaarb3414592014-06-17 17:48:32 +02006911 redraw_later(rtype);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006912 return id;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006913
6914fail:
6915 vim_free(m);
6916 return -1;
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006917}
6918
6919/*
6920 * Delete match with ID 'id' in the match list of window 'wp'.
6921 * Print error messages if 'perr' is TRUE.
6922 */
6923 int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006924match_delete(win_T *wp, int id, int perr)
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006925{
Bram Moolenaarb3414592014-06-17 17:48:32 +02006926 matchitem_T *cur = wp->w_match_head;
6927 matchitem_T *prev = cur;
6928 int rtype = SOME_VALID;
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006929
6930 if (id < 1)
6931 {
6932 if (perr == TRUE)
Bram Moolenaar5b302912016-08-24 22:11:55 +02006933 EMSGN(_("E802: Invalid ID: %ld (must be greater than or equal to 1)"),
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006934 id);
6935 return -1;
6936 }
6937 while (cur != NULL && cur->id != id)
6938 {
6939 prev = cur;
6940 cur = cur->next;
6941 }
6942 if (cur == NULL)
6943 {
6944 if (perr == TRUE)
Bram Moolenaar5b302912016-08-24 22:11:55 +02006945 EMSGN(_("E803: ID not found: %ld"), id);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006946 return -1;
6947 }
6948 if (cur == prev)
6949 wp->w_match_head = cur->next;
6950 else
6951 prev->next = cur->next;
Bram Moolenaar473de612013-06-08 18:19:48 +02006952 vim_regfree(cur->match.regprog);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006953 vim_free(cur->pattern);
Bram Moolenaarb3414592014-06-17 17:48:32 +02006954 if (cur->pos.toplnum != 0)
6955 {
6956 if (wp->w_buffer->b_mod_set)
6957 {
6958 if (wp->w_buffer->b_mod_top > cur->pos.toplnum)
6959 wp->w_buffer->b_mod_top = cur->pos.toplnum;
6960 if (wp->w_buffer->b_mod_bot < cur->pos.botlnum)
6961 wp->w_buffer->b_mod_bot = cur->pos.botlnum;
6962 }
6963 else
6964 {
Bram Moolenaardab70c62014-07-02 17:16:58 +02006965 wp->w_buffer->b_mod_set = TRUE;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006966 wp->w_buffer->b_mod_top = cur->pos.toplnum;
6967 wp->w_buffer->b_mod_bot = cur->pos.botlnum;
Bram Moolenaardab70c62014-07-02 17:16:58 +02006968 wp->w_buffer->b_mod_xlines = 0;
Bram Moolenaarb3414592014-06-17 17:48:32 +02006969 }
Bram Moolenaarb3414592014-06-17 17:48:32 +02006970 rtype = VALID;
6971 }
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006972 vim_free(cur);
Bram Moolenaarb3414592014-06-17 17:48:32 +02006973 redraw_later(rtype);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006974 return 0;
6975}
6976
6977/*
6978 * Delete all matches in the match list of window 'wp'.
6979 */
6980 void
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01006981clear_matches(win_T *wp)
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006982{
6983 matchitem_T *m;
6984
6985 while (wp->w_match_head != NULL)
6986 {
6987 m = wp->w_match_head->next;
Bram Moolenaar473de612013-06-08 18:19:48 +02006988 vim_regfree(wp->w_match_head->match.regprog);
Bram Moolenaar6ee10162007-07-26 20:58:42 +00006989 vim_free(wp->w_match_head->pattern);
6990 vim_free(wp->w_match_head);
6991 wp->w_match_head = m;
6992 }
6993 redraw_later(SOME_VALID);
6994}
6995
6996/*
6997 * Get match from ID 'id' in window 'wp'.
6998 * Return NULL if match not found.
6999 */
7000 matchitem_T *
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01007001get_match(win_T *wp, int id)
Bram Moolenaar6ee10162007-07-26 20:58:42 +00007002{
7003 matchitem_T *cur = wp->w_match_head;
7004
7005 while (cur != NULL && cur->id != id)
7006 cur = cur->next;
7007 return cur;
7008}
7009#endif
Bram Moolenaar6d216452013-05-12 19:00:41 +02007010
7011#if defined(FEAT_PYTHON) || defined(FEAT_PYTHON3) || defined(PROTO)
7012 int
Bram Moolenaar5e538ec2013-05-15 15:12:29 +02007013get_win_number(win_T *wp, win_T *first_win)
Bram Moolenaar6d216452013-05-12 19:00:41 +02007014{
7015 int i = 1;
7016 win_T *w;
7017
Bram Moolenaar5e538ec2013-05-15 15:12:29 +02007018 for (w = first_win; w != NULL && w != wp; w = W_NEXT(w))
Bram Moolenaar6d216452013-05-12 19:00:41 +02007019 ++i;
7020
7021 if (w == NULL)
7022 return 0;
7023 else
7024 return i;
7025}
Bram Moolenaar5e538ec2013-05-15 15:12:29 +02007026
7027 int
Bram Moolenaar8c0e3222013-06-16 17:32:40 +02007028get_tab_number(tabpage_T *tp UNUSED)
Bram Moolenaar5e538ec2013-05-15 15:12:29 +02007029{
7030 int i = 1;
7031 tabpage_T *t;
7032
7033 for (t = first_tabpage; t != NULL && t != tp; t = t->tp_next)
7034 ++i;
7035
7036 if (t == NULL)
7037 return 0;
7038 else
7039 return i;
7040}
Bram Moolenaar6d216452013-05-12 19:00:41 +02007041#endif
Bram Moolenaarb893ac22013-06-26 14:04:47 +02007042
7043/*
7044 * Return TRUE if "topfrp" and its children are at the right height.
7045 */
7046 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01007047frame_check_height(frame_T *topfrp, int height)
Bram Moolenaarb893ac22013-06-26 14:04:47 +02007048{
7049 frame_T *frp;
7050
7051 if (topfrp->fr_height != height)
7052 return FALSE;
7053
7054 if (topfrp->fr_layout == FR_ROW)
7055 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
7056 if (frp->fr_height != height)
7057 return FALSE;
7058
7059 return TRUE;
7060}
7061
Bram Moolenaarb893ac22013-06-26 14:04:47 +02007062/*
7063 * Return TRUE if "topfrp" and its children are at the right width.
7064 */
7065 static int
Bram Moolenaarb638a7b2016-01-30 21:29:58 +01007066frame_check_width(frame_T *topfrp, int width)
Bram Moolenaarb893ac22013-06-26 14:04:47 +02007067{
7068 frame_T *frp;
7069
7070 if (topfrp->fr_width != width)
7071 return FALSE;
7072
7073 if (topfrp->fr_layout == FR_COL)
7074 for (frp = topfrp->fr_child; frp != NULL; frp = frp->fr_next)
7075 if (frp->fr_width != width)
7076 return FALSE;
7077
7078 return TRUE;
7079}
Bram Moolenaarb893ac22013-06-26 14:04:47 +02007080
Bram Moolenaar86edef62016-03-13 18:07:30 +01007081#if defined(FEAT_EVAL) || defined(PROTO)
7082 int
7083win_getid(typval_T *argvars)
7084{
7085 int winnr;
7086 win_T *wp;
7087
7088 if (argvars[0].v_type == VAR_UNKNOWN)
7089 return curwin->w_id;
7090 winnr = get_tv_number(&argvars[0]);
7091 if (winnr > 0)
7092 {
7093 if (argvars[1].v_type == VAR_UNKNOWN)
7094 wp = firstwin;
7095 else
7096 {
7097 tabpage_T *tp;
7098 int tabnr = get_tv_number(&argvars[1]);
7099
Bram Moolenaar29323592016-07-24 22:04:11 +02007100 FOR_ALL_TABPAGES(tp)
Bram Moolenaar86edef62016-03-13 18:07:30 +01007101 if (--tabnr == 0)
7102 break;
7103 if (tp == NULL)
7104 return -1;
Bram Moolenaar8e639052016-11-13 14:31:40 +01007105 if (tp == curtab)
7106 wp = firstwin;
7107 else
7108 wp = tp->tp_firstwin;
Bram Moolenaar86edef62016-03-13 18:07:30 +01007109 }
7110 for ( ; wp != NULL; wp = wp->w_next)
7111 if (--winnr == 0)
7112 return wp->w_id;
7113 }
7114 return 0;
7115}
7116
7117 int
7118win_gotoid(typval_T *argvars)
7119{
7120 win_T *wp;
7121 tabpage_T *tp;
7122 int id = get_tv_number(&argvars[0]);
7123
Bram Moolenaar29323592016-07-24 22:04:11 +02007124 FOR_ALL_TAB_WINDOWS(tp, wp)
Bram Moolenaar86edef62016-03-13 18:07:30 +01007125 if (wp->w_id == id)
7126 {
7127 goto_tabpage_win(tp, wp);
7128 return 1;
7129 }
7130 return 0;
7131}
7132
7133 void
7134win_id2tabwin(typval_T *argvars, list_T *list)
7135{
7136 win_T *wp;
7137 tabpage_T *tp;
7138 int winnr = 1;
7139 int tabnr = 1;
7140 int id = get_tv_number(&argvars[0]);
7141
Bram Moolenaar29323592016-07-24 22:04:11 +02007142 FOR_ALL_TABPAGES(tp)
Bram Moolenaar86edef62016-03-13 18:07:30 +01007143 {
Bram Moolenaar29323592016-07-24 22:04:11 +02007144 FOR_ALL_WINDOWS_IN_TAB(tp, wp)
Bram Moolenaar86edef62016-03-13 18:07:30 +01007145 {
7146 if (wp->w_id == id)
7147 {
7148 list_append_number(list, tabnr);
7149 list_append_number(list, winnr);
7150 return;
7151 }
7152 ++winnr;
7153 }
7154 ++tabnr;
7155 winnr = 1;
7156 }
7157 list_append_number(list, 0);
7158 list_append_number(list, 0);
7159}
7160
Bram Moolenaarb5ae48e2016-08-12 22:23:25 +02007161 win_T *
7162win_id2wp(typval_T *argvars)
7163{
7164 win_T *wp;
7165 tabpage_T *tp;
7166 int id = get_tv_number(&argvars[0]);
7167
7168 FOR_ALL_TAB_WINDOWS(tp, wp)
7169 if (wp->w_id == id)
7170 return wp;
7171
7172 return NULL;
7173}
7174
Bram Moolenaar86edef62016-03-13 18:07:30 +01007175 int
7176win_id2win(typval_T *argvars)
7177{
7178 win_T *wp;
7179 int nr = 1;
7180 int id = get_tv_number(&argvars[0]);
7181
Bram Moolenaar29323592016-07-24 22:04:11 +02007182 FOR_ALL_WINDOWS(wp)
Bram Moolenaar86edef62016-03-13 18:07:30 +01007183 {
7184 if (wp->w_id == id)
7185 return nr;
7186 ++nr;
7187 }
7188 return 0;
7189}
Bram Moolenaar9cdf86b2016-03-13 19:04:51 +01007190
7191 void
7192win_findbuf(typval_T *argvars, list_T *list)
7193{
7194 win_T *wp;
7195 tabpage_T *tp;
7196 int bufnr = get_tv_number(&argvars[0]);
7197
Bram Moolenaar29323592016-07-24 22:04:11 +02007198 FOR_ALL_TAB_WINDOWS(tp, wp)
Bram Moolenaar9cdf86b2016-03-13 19:04:51 +01007199 if (wp->w_buffer->b_fnum == bufnr)
7200 list_append_number(list, wp->w_id);
7201}
7202
Bram Moolenaar86edef62016-03-13 18:07:30 +01007203#endif