blob: 9e8d9eede1286cb9eec94a617363e8ec2811e360 [file] [log] [blame]
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001/* vi:set ts=8 sts=4 sw=4 noet:
2 *
3 * VIM - Vi IMproved by Bram Moolenaar
4 *
5 * Do ":help uganda" in Vim to read copying and usage conditions.
6 * Do ":help credits" in Vim to see a list of people who contributed.
7 * See README.txt for an overview of the Vim source code.
8 */
9
10/*
11 * drawscreen.c: Code for updating all the windows on the screen.
12 * This is the top level, drawline.c is the middle and screen.c the lower
13 * level.
14 *
15 * update_screen() is the function that updates all windows and status lines.
16 * It is called form the main loop when must_redraw is non-zero. It may be
17 * called from other places when an immediate screen update is needed.
18 *
19 * The part of the buffer that is displayed in a window is set with:
20 * - w_topline (first buffer line in window)
21 * - w_topfill (filler lines above the first line)
22 * - w_leftcol (leftmost window cell in window),
23 * - w_skipcol (skipped window cells of first line)
24 *
25 * Commands that only move the cursor around in a window, do not need to take
26 * action to update the display. The main loop will check if w_topline is
27 * valid and update it (scroll the window) when needed.
28 *
29 * Commands that scroll a window change w_topline and must call
30 * check_cursor() to move the cursor into the visible part of the window, and
Bram Moolenaara4d158b2022-08-14 14:17:45 +010031 * call redraw_later(UPD_VALID) to have the window displayed by update_screen()
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020032 * later.
33 *
34 * Commands that change text in the buffer must call changed_bytes() or
35 * changed_lines() to mark the area that changed and will require updating
36 * later. The main loop will call update_screen(), which will update each
37 * window that shows the changed buffer. This assumes text above the change
38 * can remain displayed as it is. Text after the change may need updating for
39 * scrolling, folding and syntax highlighting.
40 *
41 * Commands that change how a window is displayed (e.g., setting 'list') or
42 * invalidate the contents of a window in another way (e.g., change fold
Bram Moolenaara4d158b2022-08-14 14:17:45 +010043 * settings), must call redraw_later(UPD_NOT_VALID) to have the whole window
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020044 * redisplayed by update_screen() later.
45 *
46 * Commands that change how a buffer is displayed (e.g., setting 'tabstop')
Bram Moolenaara4d158b2022-08-14 14:17:45 +010047 * must call redraw_curbuf_later(UPD_NOT_VALID) to have all the windows for the
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020048 * buffer redisplayed by update_screen() later.
49 *
50 * Commands that change highlighting and possibly cause a scroll too must call
Bram Moolenaara4d158b2022-08-14 14:17:45 +010051 * redraw_later(UPD_SOME_VALID) to update the whole window but still use
52 * scrolling to avoid redrawing everything. But the length of displayed lines
53 * must not change, use UPD_NOT_VALID then.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020054 *
Bram Moolenaara4d158b2022-08-14 14:17:45 +010055 * Commands that move the window position must call redraw_later(UPD_NOT_VALID).
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020056 * TODO: should minimize redrawing by scrolling when possible.
57 *
58 * Commands that change everything (e.g., resizing the screen) must call
Bram Moolenaara4d158b2022-08-14 14:17:45 +010059 * redraw_all_later(UPD_NOT_VALID) or redraw_all_later(UPD_CLEAR).
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020060 *
61 * Things that are handled indirectly:
62 * - When messages scroll the screen up, msg_scrolled will be set and
63 * update_screen() called to redraw.
64 */
65
66#include "vim.h"
67
68static void win_update(win_T *wp);
69#ifdef FEAT_STL_OPT
70static void redraw_custom_statusline(win_T *wp);
71#endif
Bram Moolenaare52e0c82020-02-28 22:20:10 +010072#if defined(FEAT_SEARCH_EXTRA) || defined(FEAT_CLIPBOARD)
73static int did_update_one_window;
74#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020075
Bram Moolenaarbdff0122020-04-05 18:56:05 +020076static void win_redr_status(win_T *wp, int ignore_pum);
77
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020078/*
79 * Based on the current value of curwin->w_topline, transfer a screenfull
80 * of stuff from Filemem to ScreenLines[], and update curwin->w_botline.
81 * Return OK when the screen was updated, FAIL if it was not done.
82 */
83 int
84update_screen(int type_arg)
85{
86 int type = type_arg;
87 win_T *wp;
88 static int did_intro = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020089#ifdef FEAT_GUI
Bram Moolenaare52e0c82020-02-28 22:20:10 +010090 int did_one = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020091 int did_undraw = FALSE;
92 int gui_cursor_col = 0;
93 int gui_cursor_row = 0;
94#endif
95 int no_update = FALSE;
Bram Moolenaare0c03c82021-04-23 21:01:34 +020096 int save_pum_will_redraw = pum_will_redraw;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020097
98 // Don't do anything if the screen structures are (not yet) valid.
99 if (!screen_valid(TRUE))
100 return FAIL;
101
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100102 if (type == UPD_VALID_NO_UPDATE)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200103 {
104 no_update = TRUE;
105 type = 0;
106 }
107
108#ifdef FEAT_EVAL
109 {
110 buf_T *buf;
111
112 // Before updating the screen, notify any listeners of changed text.
113 FOR_ALL_BUFFERS(buf)
114 invoke_listeners(buf);
115 }
116#endif
117
118#ifdef FEAT_DIFF
119 // May have postponed updating diffs.
120 if (need_diff_redraw)
121 diff_redraw(TRUE);
122#endif
123
124 if (must_redraw)
125 {
126 if (type < must_redraw) // use maximal type
127 type = must_redraw;
128
129 // must_redraw is reset here, so that when we run into some weird
130 // reason to redraw while busy redrawing (e.g., asynchronous
131 // scrolling), or update_topline() in win_update() will cause a
132 // scroll, the screen will be redrawn later or in win_update().
133 must_redraw = 0;
134 }
135
136 // May need to update w_lines[].
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100137 if (curwin->w_lines_valid == 0 && type < UPD_NOT_VALID
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200138#ifdef FEAT_TERMINAL
139 && !term_do_update_window(curwin)
140#endif
141 )
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100142 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200143
144 // Postpone the redrawing when it's not needed and when being called
145 // recursively.
146 if (!redrawing() || updating_screen)
147 {
148 redraw_later(type); // remember type for next time
149 must_redraw = type;
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100150 if (type > UPD_INVERTED_ALL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200151 curwin->w_lines_valid = 0; // don't use w_lines[].wl_size now
152 return FAIL;
153 }
154 updating_screen = TRUE;
155
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +0100156#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200157 // Update popup_mask if needed. This may set w_redraw_top and w_redraw_bot
158 // in some windows.
159 may_update_popup_mask(type);
160#endif
161
162#ifdef FEAT_SYN_HL
163 ++display_tick; // let syntax code know we're in a next round of
164 // display updating
165#endif
166 if (no_update)
167 ++no_win_do_lines_ins;
168
169 // if the screen was scrolled up when displaying a message, scroll it down
170 if (msg_scrolled)
171 {
172 clear_cmdline = TRUE;
173 if (msg_scrolled > Rows - 5) // clearing is faster
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100174 type = UPD_CLEAR;
175 else if (type != UPD_CLEAR)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200176 {
177 check_for_delay(FALSE);
178 if (screen_ins_lines(0, 0, msg_scrolled, (int)Rows, 0, NULL)
179 == FAIL)
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100180 type = UPD_CLEAR;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200181 FOR_ALL_WINDOWS(wp)
182 {
183 if (wp->w_winrow < msg_scrolled)
184 {
185 if (W_WINROW(wp) + wp->w_height > msg_scrolled
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100186 && wp->w_redr_type < UPD_REDRAW_TOP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200187 && wp->w_lines_valid > 0
188 && wp->w_topline == wp->w_lines[0].wl_lnum)
189 {
190 wp->w_upd_rows = msg_scrolled - W_WINROW(wp);
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100191 wp->w_redr_type = UPD_REDRAW_TOP;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200192 }
193 else
194 {
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100195 wp->w_redr_type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200196 if (W_WINROW(wp) + wp->w_height + wp->w_status_height
197 <= msg_scrolled)
198 wp->w_redr_status = TRUE;
199 }
200 }
201 }
202 if (!no_update)
203 redraw_cmdline = TRUE;
204 redraw_tabline = TRUE;
205 }
206 msg_scrolled = 0;
207 need_wait_return = FALSE;
208 }
209
210 // reset cmdline_row now (may have been changed temporarily)
211 compute_cmdrow();
212
213 // Check for changed highlighting
214 if (need_highlight_changed)
215 highlight_changed();
216
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100217 if (type == UPD_CLEAR) // first clear screen
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200218 {
219 screenclear(); // will reset clear_cmdline
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100220 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200221 // must_redraw may be set indirectly, avoid another redraw later
222 must_redraw = 0;
223 }
224
225 if (clear_cmdline) // going to clear cmdline (done below)
226 check_for_delay(FALSE);
227
228#ifdef FEAT_LINEBREAK
229 // Force redraw when width of 'number' or 'relativenumber' column
230 // changes.
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100231 if (curwin->w_redr_type < UPD_NOT_VALID
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200232 && curwin->w_nrwidth != ((curwin->w_p_nu || curwin->w_p_rnu)
233 ? number_width(curwin) : 0))
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100234 curwin->w_redr_type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200235#endif
236
237 // Only start redrawing if there is really something to do.
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100238 if (type == UPD_INVERTED)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200239 update_curswant();
240 if (curwin->w_redr_type < type
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100241 && !((type == UPD_VALID
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200242 && curwin->w_lines[0].wl_valid
243#ifdef FEAT_DIFF
244 && curwin->w_topfill == curwin->w_old_topfill
245 && curwin->w_botfill == curwin->w_old_botfill
246#endif
247 && curwin->w_topline == curwin->w_lines[0].wl_lnum)
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100248 || (type == UPD_INVERTED
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200249 && VIsual_active
250 && curwin->w_old_cursor_lnum == curwin->w_cursor.lnum
251 && curwin->w_old_visual_mode == VIsual_mode
252 && (curwin->w_valid & VALID_VIRTCOL)
253 && curwin->w_old_curswant == curwin->w_curswant)
254 ))
255 curwin->w_redr_type = type;
256
257 // Redraw the tab pages line if needed.
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100258 if (redraw_tabline || type >= UPD_NOT_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200259 draw_tabline();
260
261#ifdef FEAT_SYN_HL
262 // Correct stored syntax highlighting info for changes in each displayed
263 // buffer. Each buffer must only be done once.
264 FOR_ALL_WINDOWS(wp)
265 {
266 if (wp->w_buffer->b_mod_set)
267 {
268 win_T *wwp;
269
270 // Check if we already did this buffer.
271 for (wwp = firstwin; wwp != wp; wwp = wwp->w_next)
272 if (wwp->w_buffer == wp->w_buffer)
273 break;
274 if (wwp == wp && syntax_present(wp))
275 syn_stack_apply_changes(wp->w_buffer);
276 }
277 }
278#endif
279
Bram Moolenaare0c03c82021-04-23 21:01:34 +0200280 if (pum_redraw_in_same_position())
281 // Avoid flicker if the popup menu is going to be redrawn in the same
282 // position.
283 pum_will_redraw = TRUE;
284
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200285 // Go from top to bottom through the windows, redrawing the ones that need
286 // it.
287#if defined(FEAT_SEARCH_EXTRA) || defined(FEAT_CLIPBOARD)
Bram Moolenaare52e0c82020-02-28 22:20:10 +0100288 did_update_one_window = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200289#endif
290#ifdef FEAT_SEARCH_EXTRA
291 screen_search_hl.rm.regprog = NULL;
292#endif
293 FOR_ALL_WINDOWS(wp)
294 {
295 if (wp->w_redr_type != 0)
296 {
297 cursor_off();
Bram Moolenaare52e0c82020-02-28 22:20:10 +0100298#ifdef FEAT_GUI
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200299 if (!did_one)
300 {
301 did_one = TRUE;
Bram Moolenaare52e0c82020-02-28 22:20:10 +0100302
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200303 // Remove the cursor before starting to do anything, because
304 // scrolling may make it difficult to redraw the text under
305 // it.
Bram Moolenaar09f067f2021-04-11 13:29:18 +0200306 // Also remove the cursor if it needs to be hidden due to an
307 // ongoing cursor-less sleep.
308 if (gui.in_use && (wp == curwin || cursor_is_sleeping()))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200309 {
310 gui_cursor_col = gui.cursor_col;
311 gui_cursor_row = gui.cursor_row;
312 gui_undraw_cursor();
313 did_undraw = TRUE;
314 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200315 }
316#endif
317 win_update(wp);
318 }
319
320 // redraw status line after the window to minimize cursor movement
321 if (wp->w_redr_status)
322 {
323 cursor_off();
324 win_redr_status(wp, TRUE); // any popup menu will be redrawn below
325 }
326 }
327#if defined(FEAT_SEARCH_EXTRA)
328 end_search_hl();
329#endif
Bram Moolenaare0c03c82021-04-23 21:01:34 +0200330
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200331 // May need to redraw the popup menu.
Bram Moolenaare0c03c82021-04-23 21:01:34 +0200332 pum_will_redraw = save_pum_will_redraw;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200333 pum_may_redraw();
334
335 // Reset b_mod_set flags. Going through all windows is probably faster
336 // than going through all buffers (there could be many buffers).
337 FOR_ALL_WINDOWS(wp)
338 wp->w_buffer->b_mod_set = FALSE;
339
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +0100340#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200341 // Display popup windows on top of the windows and command line.
342 update_popups(win_update);
343#endif
344
Bram Moolenaar3194e5b2021-12-13 21:59:09 +0000345#ifdef FEAT_TERMINAL
346 FOR_ALL_WINDOWS(wp)
347 // If this window contains a terminal, after redrawing all windows, the
348 // dirty row range can be reset.
349 term_did_update_window(wp);
350#endif
351
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200352 after_updating_screen(TRUE);
353
354 // Clear or redraw the command line. Done last, because scrolling may
355 // mess up the command line.
356 if (clear_cmdline || redraw_cmdline || redraw_mode)
357 showmode();
358
359 if (no_update)
360 --no_win_do_lines_ins;
361
362 // May put up an introductory message when not editing a file
363 if (!did_intro)
364 maybe_intro_message();
365 did_intro = TRUE;
366
367#ifdef FEAT_GUI
368 // Redraw the cursor and update the scrollbars when all screen updating is
369 // done.
370 if (gui.in_use)
371 {
372 if (did_undraw && !gui_mch_is_blink_off())
373 {
374 mch_disable_flush();
375 out_flush(); // required before updating the cursor
376 mch_enable_flush();
377
378 // Put the GUI position where the cursor was, gui_update_cursor()
379 // uses that.
380 gui.col = gui_cursor_col;
381 gui.row = gui_cursor_row;
382 gui.col = mb_fix_col(gui.col, gui.row);
383 gui_update_cursor(FALSE, FALSE);
384 gui_may_flush();
385 screen_cur_col = gui.col;
386 screen_cur_row = gui.row;
387 }
388 else
389 out_flush();
390 gui_update_scrollbars(FALSE);
391 }
392#endif
393 return OK;
394}
395
396/*
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200397 * Return the row for drawing the statusline and the ruler of window "wp".
398 */
Bram Moolenaar49c51b82021-04-01 16:16:18 +0200399 int
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200400statusline_row(win_T *wp)
401{
402#if defined(FEAT_PROP_POPUP)
403 // If the window is really zero height the winbar isn't displayed.
404 if (wp->w_frame->fr_height == wp->w_status_height && !popup_is_popup(wp))
405 return wp->w_winrow;
406#endif
407 return W_WINROW(wp) + wp->w_height;
408}
409
410/*
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200411 * Redraw the status line of window wp.
412 *
413 * If inversion is possible we use it. Else '=' characters are used.
414 * If "ignore_pum" is TRUE, also redraw statusline when the popup menu is
415 * displayed.
416 */
Bram Moolenaarbdff0122020-04-05 18:56:05 +0200417 static void
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200418win_redr_status(win_T *wp, int ignore_pum UNUSED)
419{
420 int row;
421 char_u *p;
422 int len;
423 int fillchar;
424 int attr;
425 int this_ru_col;
426 static int busy = FALSE;
427
428 // It's possible to get here recursively when 'statusline' (indirectly)
429 // invokes ":redrawstatus". Simply ignore the call then.
430 if (busy)
431 return;
432 busy = TRUE;
433
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200434 row = statusline_row(wp);
435
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200436 wp->w_redr_status = FALSE;
437 if (wp->w_status_height == 0)
438 {
439 // no status line, can only be last window
440 redraw_cmdline = TRUE;
441 }
442 else if (!redrawing()
443 // don't update status line when popup menu is visible and may be
444 // drawn over it, unless it will be redrawn later
445 || (!ignore_pum && pum_visible()))
446 {
447 // Don't redraw right now, do it later.
448 wp->w_redr_status = TRUE;
449 }
450#ifdef FEAT_STL_OPT
451 else if (*p_stl != NUL || *wp->w_p_stl != NUL)
452 {
453 // redraw custom status line
454 redraw_custom_statusline(wp);
455 }
456#endif
457 else
458 {
459 fillchar = fillchar_status(&attr, wp);
460
461 get_trans_bufname(wp->w_buffer);
462 p = NameBuff;
463 len = (int)STRLEN(p);
464
Bram Moolenaarde05bb22022-01-13 13:08:14 +0000465 if ((bt_help(wp->w_buffer)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200466#ifdef FEAT_QUICKFIX
Bram Moolenaarde05bb22022-01-13 13:08:14 +0000467 || wp->w_p_pvw
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200468#endif
Bram Moolenaarde05bb22022-01-13 13:08:14 +0000469 || bufIsChanged(wp->w_buffer)
470 || wp->w_buffer->b_p_ro)
471 && len < MAXPATHL - 1)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200472 *(p + len++) = ' ';
473 if (bt_help(wp->w_buffer))
474 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100475 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[Help]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200476 len += (int)STRLEN(p + len);
477 }
478#ifdef FEAT_QUICKFIX
479 if (wp->w_p_pvw)
480 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100481 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[Preview]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200482 len += (int)STRLEN(p + len);
483 }
484#endif
485 if (bufIsChanged(wp->w_buffer)
486#ifdef FEAT_TERMINAL
487 && !bt_terminal(wp->w_buffer)
488#endif
489 )
490 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100491 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", "[+]");
492 len += (int)STRLEN(p + len);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200493 }
494 if (wp->w_buffer->b_p_ro)
495 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100496 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[RO]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200497 len += (int)STRLEN(p + len);
498 }
499
500 this_ru_col = ru_col - (Columns - wp->w_width);
501 if (this_ru_col < (wp->w_width + 1) / 2)
502 this_ru_col = (wp->w_width + 1) / 2;
503 if (this_ru_col <= 1)
504 {
505 p = (char_u *)"<"; // No room for file name!
506 len = 1;
507 }
508 else if (has_mbyte)
509 {
510 int clen = 0, i;
511
512 // Count total number of display cells.
513 clen = mb_string2cells(p, -1);
514
515 // Find first character that will fit.
516 // Going from start to end is much faster for DBCS.
517 for (i = 0; p[i] != NUL && clen >= this_ru_col - 1;
518 i += (*mb_ptr2len)(p + i))
519 clen -= (*mb_ptr2cells)(p + i);
520 len = clen;
521 if (i > 0)
522 {
523 p = p + i - 1;
524 *p = '<';
525 ++len;
526 }
527
528 }
529 else if (len > this_ru_col - 1)
530 {
531 p += len - (this_ru_col - 1);
532 *p = '<';
533 len = this_ru_col - 1;
534 }
535
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200536 screen_puts(p, row, wp->w_wincol, attr);
537 screen_fill(row, row + 1, len + wp->w_wincol,
538 this_ru_col + wp->w_wincol, fillchar, fillchar, attr);
539
540 if (get_keymap_str(wp, (char_u *)"<%s>", NameBuff, MAXPATHL)
=?UTF-8?q?Dundar=20G=C3=B6c?=420fabc2022-01-28 15:28:04 +0000541 && (this_ru_col - len) > (int)(STRLEN(NameBuff) + 1))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200542 screen_puts(NameBuff, row, (int)(this_ru_col - STRLEN(NameBuff)
543 - 1 + wp->w_wincol), attr);
544
545#ifdef FEAT_CMDL_INFO
546 win_redr_ruler(wp, TRUE, ignore_pum);
547#endif
548 }
549
550 /*
551 * May need to draw the character below the vertical separator.
552 */
553 if (wp->w_vsep_width != 0 && wp->w_status_height != 0 && redrawing())
554 {
555 if (stl_connected(wp))
556 fillchar = fillchar_status(&attr, wp);
557 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +0100558 fillchar = fillchar_vsep(&attr, wp);
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200559 screen_putchar(fillchar, row, W_ENDCOL(wp), attr);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200560 }
561 busy = FALSE;
562}
563
564#ifdef FEAT_STL_OPT
565/*
566 * Redraw the status line according to 'statusline' and take care of any
567 * errors encountered.
568 */
569 static void
570redraw_custom_statusline(win_T *wp)
571{
572 static int entered = FALSE;
573 int saved_did_emsg = did_emsg;
574
575 // When called recursively return. This can happen when the statusline
576 // contains an expression that triggers a redraw.
577 if (entered)
578 return;
579 entered = TRUE;
580
581 did_emsg = FALSE;
582 win_redr_custom(wp, FALSE);
583 if (did_emsg)
584 {
585 // When there is an error disable the statusline, otherwise the
586 // display is messed up with errors and a redraw triggers the problem
587 // again and again.
588 set_string_option_direct((char_u *)"statusline", -1,
589 (char_u *)"", OPT_FREE | (*wp->w_p_stl != NUL
590 ? OPT_LOCAL : OPT_GLOBAL), SID_ERROR);
591 }
592 did_emsg |= saved_did_emsg;
593 entered = FALSE;
594}
595#endif
596
597/*
598 * Show current status info in ruler and various other places
599 * If always is FALSE, only show ruler if position has changed.
600 */
601 void
602showruler(int always)
603{
604 if (!always && !redrawing())
605 return;
606 if (pum_visible())
607 {
608 // Don't redraw right now, do it later.
609 curwin->w_redr_status = TRUE;
610 return;
611 }
612#if defined(FEAT_STL_OPT)
613 if ((*p_stl != NUL || *curwin->w_p_stl != NUL) && curwin->w_status_height)
614 redraw_custom_statusline(curwin);
615 else
616#endif
617#ifdef FEAT_CMDL_INFO
618 win_redr_ruler(curwin, always, FALSE);
619#endif
620
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200621 if (need_maketitle
Bram Moolenaar651fca82021-11-29 20:39:38 +0000622#ifdef FEAT_STL_OPT
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200623 || (p_icon && (stl_syntax & STL_IN_ICON))
624 || (p_title && (stl_syntax & STL_IN_TITLE))
Bram Moolenaar651fca82021-11-29 20:39:38 +0000625#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200626 )
627 maketitle();
Bram Moolenaar651fca82021-11-29 20:39:38 +0000628
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200629 // Redraw the tab pages line if needed.
630 if (redraw_tabline)
631 draw_tabline();
632}
633
634#if defined(FEAT_CMDL_INFO) || defined(PROTO)
635 void
636win_redr_ruler(win_T *wp, int always, int ignore_pum)
637{
638#define RULER_BUF_LEN 70
639 char_u buffer[RULER_BUF_LEN];
640 int row;
641 int fillchar;
642 int attr;
643 int empty_line = FALSE;
644 colnr_T virtcol;
645 int i;
646 size_t len;
647 int o;
648 int this_ru_col;
649 int off = 0;
650 int width;
651
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +0100652 // If 'ruler' off or messages area disabled, don't do anything
653 if (!p_ru || (wp->w_status_height == 0 && p_ch == 0))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200654 return;
655
656 /*
657 * Check if cursor.lnum is valid, since win_redr_ruler() may be called
658 * after deleting lines, before cursor.lnum is corrected.
659 */
660 if (wp->w_cursor.lnum > wp->w_buffer->b_ml.ml_line_count)
661 return;
662
663 // Don't draw the ruler while doing insert-completion, it might overwrite
664 // the (long) mode message.
665 if (wp == lastwin && lastwin->w_status_height == 0)
666 if (edit_submode != NULL)
667 return;
668 // Don't draw the ruler when the popup menu is visible, it may overlap.
669 // Except when the popup menu will be redrawn anyway.
670 if (!ignore_pum && pum_visible())
671 return;
672
673#ifdef FEAT_STL_OPT
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +0100674 if (*p_ruf && p_ch > 0)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200675 {
Bram Moolenaar53989552019-12-23 22:59:18 +0100676 int called_emsg_before = called_emsg;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200677
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200678 win_redr_custom(wp, TRUE);
Bram Moolenaar53989552019-12-23 22:59:18 +0100679 if (called_emsg > called_emsg_before)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200680 set_string_option_direct((char_u *)"rulerformat", -1,
681 (char_u *)"", OPT_FREE, SID_ERROR);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200682 return;
683 }
684#endif
685
686 /*
687 * Check if not in Insert mode and the line is empty (will show "0-1").
688 */
Bram Moolenaar24959102022-05-07 20:01:16 +0100689 if ((State & MODE_INSERT) == 0
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200690 && *ml_get_buf(wp->w_buffer, wp->w_cursor.lnum, FALSE) == NUL)
691 empty_line = TRUE;
692
693 /*
694 * Only draw the ruler when something changed.
695 */
696 validate_virtcol_win(wp);
697 if ( redraw_cmdline
698 || always
699 || wp->w_cursor.lnum != wp->w_ru_cursor.lnum
700 || wp->w_cursor.col != wp->w_ru_cursor.col
701 || wp->w_virtcol != wp->w_ru_virtcol
702 || wp->w_cursor.coladd != wp->w_ru_cursor.coladd
703 || wp->w_topline != wp->w_ru_topline
704 || wp->w_buffer->b_ml.ml_line_count != wp->w_ru_line_count
705#ifdef FEAT_DIFF
706 || wp->w_topfill != wp->w_ru_topfill
707#endif
708 || empty_line != wp->w_ru_empty)
709 {
710 cursor_off();
711 if (wp->w_status_height)
712 {
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200713 row = statusline_row(wp);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200714 fillchar = fillchar_status(&attr, wp);
715 off = wp->w_wincol;
716 width = wp->w_width;
717 }
718 else
719 {
720 row = Rows - 1;
721 fillchar = ' ';
722 attr = 0;
723 width = Columns;
724 off = 0;
725 }
726
727 // In list mode virtcol needs to be recomputed
728 virtcol = wp->w_virtcol;
Bram Moolenaareed9d462021-02-15 20:38:25 +0100729 if (wp->w_p_list && wp->w_lcs_chars.tab1 == NUL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200730 {
731 wp->w_p_list = FALSE;
732 getvvcol(wp, &wp->w_cursor, NULL, &virtcol, NULL);
733 wp->w_p_list = TRUE;
734 }
735
736 /*
737 * Some sprintfs return the length, some return a pointer.
738 * To avoid portability problems we use strlen() here.
739 */
740 vim_snprintf((char *)buffer, RULER_BUF_LEN, "%ld,",
741 (wp->w_buffer->b_ml.ml_flags & ML_EMPTY)
742 ? 0L
743 : (long)(wp->w_cursor.lnum));
744 len = STRLEN(buffer);
745 col_print(buffer + len, RULER_BUF_LEN - len,
746 empty_line ? 0 : (int)wp->w_cursor.col + 1,
747 (int)virtcol + 1);
748
749 /*
750 * Add a "50%" if there is room for it.
751 * On the last line, don't print in the last column (scrolls the
752 * screen up on some terminals).
753 */
754 i = (int)STRLEN(buffer);
755 get_rel_pos(wp, buffer + i + 1, RULER_BUF_LEN - i - 1);
756 o = i + vim_strsize(buffer + i + 1);
757 if (wp->w_status_height == 0) // can't use last char of screen
758 ++o;
759 this_ru_col = ru_col - (Columns - width);
760 if (this_ru_col < 0)
761 this_ru_col = 0;
762 // Never use more than half the window/screen width, leave the other
763 // half for the filename.
764 if (this_ru_col < (width + 1) / 2)
765 this_ru_col = (width + 1) / 2;
766 if (this_ru_col + o < width)
767 {
768 // need at least 3 chars left for get_rel_pos() + NUL
769 while (this_ru_col + o < width && RULER_BUF_LEN > i + 4)
770 {
771 if (has_mbyte)
772 i += (*mb_char2bytes)(fillchar, buffer + i);
773 else
774 buffer[i++] = fillchar;
775 ++o;
776 }
777 get_rel_pos(wp, buffer + i, RULER_BUF_LEN - i);
778 }
779 // Truncate at window boundary.
780 if (has_mbyte)
781 {
782 o = 0;
783 for (i = 0; buffer[i] != NUL; i += (*mb_ptr2len)(buffer + i))
784 {
785 o += (*mb_ptr2cells)(buffer + i);
786 if (this_ru_col + o > width)
787 {
788 buffer[i] = NUL;
789 break;
790 }
791 }
792 }
793 else if (this_ru_col + (int)STRLEN(buffer) > width)
794 buffer[width - this_ru_col] = NUL;
795
796 screen_puts(buffer, row, this_ru_col + off, attr);
797 i = redraw_cmdline;
798 screen_fill(row, row + 1,
799 this_ru_col + off + (int)STRLEN(buffer),
=?UTF-8?q?Dundar=20G=C3=B6c?=420fabc2022-01-28 15:28:04 +0000800 (off + width),
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200801 fillchar, fillchar, attr);
802 // don't redraw the cmdline because of showing the ruler
803 redraw_cmdline = i;
804 wp->w_ru_cursor = wp->w_cursor;
805 wp->w_ru_virtcol = wp->w_virtcol;
806 wp->w_ru_empty = empty_line;
807 wp->w_ru_topline = wp->w_topline;
808 wp->w_ru_line_count = wp->w_buffer->b_ml.ml_line_count;
809#ifdef FEAT_DIFF
810 wp->w_ru_topfill = wp->w_topfill;
811#endif
812 }
813}
814#endif
815
816/*
817 * To be called when "updating_screen" was set before and now the postponed
818 * side effects may take place.
819 */
820 void
821after_updating_screen(int may_resize_shell UNUSED)
822{
823 updating_screen = FALSE;
824#ifdef FEAT_GUI
825 if (may_resize_shell)
826 gui_may_resize_shell();
827#endif
828#ifdef FEAT_TERMINAL
829 term_check_channel_closed_recently();
830#endif
831
832#ifdef HAVE_DROP_FILE
833 // If handle_drop() was called while updating_screen was TRUE need to
834 // handle the drop now.
835 handle_any_postponed_drop();
836#endif
837}
838
839/*
840 * Update all windows that are editing the current buffer.
841 */
842 void
843update_curbuf(int type)
844{
845 redraw_curbuf_later(type);
846 update_screen(type);
847}
848
849#if defined(FEAT_MENU) || defined(FEAT_FOLDING)
850/*
851 * Copy "text" to ScreenLines using "attr".
852 * Returns the next screen column.
853 */
854 static int
855text_to_screenline(win_T *wp, char_u *text, int col)
856{
857 int off = (int)(current_ScreenLine - ScreenLines);
858
859 if (has_mbyte)
860 {
861 int cells;
862 int u8c, u8cc[MAX_MCO];
863 int i;
864 int idx;
865 int c_len;
866 char_u *p;
867# ifdef FEAT_ARABIC
868 int prev_c = 0; // previous Arabic character
869 int prev_c1 = 0; // first composing char for prev_c
870# endif
871
872# ifdef FEAT_RIGHTLEFT
873 if (wp->w_p_rl)
874 idx = off;
875 else
876# endif
877 idx = off + col;
878
879 // Store multibyte characters in ScreenLines[] et al. correctly.
880 for (p = text; *p != NUL; )
881 {
882 cells = (*mb_ptr2cells)(p);
883 c_len = (*mb_ptr2len)(p);
884 if (col + cells > wp->w_width
885# ifdef FEAT_RIGHTLEFT
886 - (wp->w_p_rl ? col : 0)
887# endif
888 )
889 break;
890 ScreenLines[idx] = *p;
891 if (enc_utf8)
892 {
893 u8c = utfc_ptr2char(p, u8cc);
894 if (*p < 0x80 && u8cc[0] == 0)
895 {
896 ScreenLinesUC[idx] = 0;
897#ifdef FEAT_ARABIC
898 prev_c = u8c;
899#endif
900 }
901 else
902 {
903#ifdef FEAT_ARABIC
904 if (p_arshape && !p_tbidi && ARABIC_CHAR(u8c))
905 {
906 // Do Arabic shaping.
907 int pc, pc1, nc;
908 int pcc[MAX_MCO];
909 int firstbyte = *p;
910
911 // The idea of what is the previous and next
912 // character depends on 'rightleft'.
913 if (wp->w_p_rl)
914 {
915 pc = prev_c;
916 pc1 = prev_c1;
917 nc = utf_ptr2char(p + c_len);
918 prev_c1 = u8cc[0];
919 }
920 else
921 {
922 pc = utfc_ptr2char(p + c_len, pcc);
923 nc = prev_c;
924 pc1 = pcc[0];
925 }
926 prev_c = u8c;
927
928 u8c = arabic_shape(u8c, &firstbyte, &u8cc[0],
929 pc, pc1, nc);
930 ScreenLines[idx] = firstbyte;
931 }
932 else
933 prev_c = u8c;
934#endif
935 // Non-BMP character: display as ? or fullwidth ?.
936 ScreenLinesUC[idx] = u8c;
937 for (i = 0; i < Screen_mco; ++i)
938 {
939 ScreenLinesC[i][idx] = u8cc[i];
940 if (u8cc[i] == 0)
941 break;
942 }
943 }
944 if (cells > 1)
945 ScreenLines[idx + 1] = 0;
946 }
947 else if (enc_dbcs == DBCS_JPNU && *p == 0x8e)
948 // double-byte single width character
949 ScreenLines2[idx] = p[1];
950 else if (cells > 1)
951 // double-width character
952 ScreenLines[idx + 1] = p[1];
953 col += cells;
954 idx += cells;
955 p += c_len;
956 }
957 }
958 else
959 {
960 int len = (int)STRLEN(text);
961
962 if (len > wp->w_width - col)
963 len = wp->w_width - col;
964 if (len > 0)
965 {
966#ifdef FEAT_RIGHTLEFT
967 if (wp->w_p_rl)
968 mch_memmove(current_ScreenLine, text, len);
969 else
970#endif
971 mch_memmove(current_ScreenLine + col, text, len);
972 col += len;
973 }
974 }
975 return col;
976}
977#endif
978
979#ifdef FEAT_MENU
980/*
981 * Draw the window toolbar.
982 */
983 static void
984redraw_win_toolbar(win_T *wp)
985{
986 vimmenu_T *menu;
987 int item_idx = 0;
988 int item_count = 0;
989 int col = 0;
990 int next_col;
991 int off = (int)(current_ScreenLine - ScreenLines);
992 int fill_attr = syn_name2attr((char_u *)"ToolbarLine");
993 int button_attr = syn_name2attr((char_u *)"ToolbarButton");
994
995 vim_free(wp->w_winbar_items);
Bram Moolenaar00d253e2020-04-06 22:13:01 +0200996 FOR_ALL_CHILD_MENUS(wp->w_winbar, menu)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200997 ++item_count;
998 wp->w_winbar_items = ALLOC_CLEAR_MULT(winbar_item_T, item_count + 1);
999
1000 // TODO: use fewer spaces if there is not enough room
1001 for (menu = wp->w_winbar->children;
1002 menu != NULL && col < wp->w_width; menu = menu->next)
1003 {
1004 space_to_screenline(off + col, fill_attr);
1005 if (++col >= wp->w_width)
1006 break;
1007 if (col > 1)
1008 {
1009 space_to_screenline(off + col, fill_attr);
1010 if (++col >= wp->w_width)
1011 break;
1012 }
1013
1014 wp->w_winbar_items[item_idx].wb_startcol = col;
1015 space_to_screenline(off + col, button_attr);
1016 if (++col >= wp->w_width)
1017 break;
1018
1019 next_col = text_to_screenline(wp, menu->name, col);
1020 while (col < next_col)
1021 {
1022 ScreenAttrs[off + col] = button_attr;
1023 ++col;
1024 }
1025 wp->w_winbar_items[item_idx].wb_endcol = col;
1026 wp->w_winbar_items[item_idx].wb_menu = menu;
1027 ++item_idx;
1028
1029 if (col >= wp->w_width)
1030 break;
1031 space_to_screenline(off + col, button_attr);
1032 ++col;
1033 }
1034 while (col < wp->w_width)
1035 {
1036 space_to_screenline(off + col, fill_attr);
1037 ++col;
1038 }
1039 wp->w_winbar_items[item_idx].wb_menu = NULL; // end marker
1040
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001041 screen_line(wp, wp->w_winrow, wp->w_wincol, wp->w_width, wp->w_width, 0);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001042}
1043#endif
1044
1045#if defined(FEAT_FOLDING) || defined(PROTO)
1046/*
1047 * Copy "buf[len]" to ScreenLines["off"] and set attributes to "attr".
1048 */
1049 static void
1050copy_text_attr(
1051 int off,
1052 char_u *buf,
1053 int len,
1054 int attr)
1055{
1056 int i;
1057
1058 mch_memmove(ScreenLines + off, buf, (size_t)len);
1059 if (enc_utf8)
1060 vim_memset(ScreenLinesUC + off, 0, sizeof(u8char_T) * (size_t)len);
1061 for (i = 0; i < len; ++i)
1062 ScreenAttrs[off + i] = attr;
1063}
1064
1065/*
1066 * Display one folded line.
1067 */
1068 static void
1069fold_line(
1070 win_T *wp,
1071 long fold_count,
1072 foldinfo_T *foldinfo,
1073 linenr_T lnum,
1074 int row)
1075{
Bram Moolenaar4fa11752021-03-03 13:26:02 +01001076 // Max value of 'foldcolumn' is 12 and maximum number of bytes in a
1077 // multi-byte character is MAX_MCO.
1078 char_u buf[MAX_MCO * 12 + 1];
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001079 pos_T *top, *bot;
1080 linenr_T lnume = lnum + fold_count - 1;
1081 int len;
1082 char_u *text;
1083 int fdc;
1084 int col;
1085 int txtcol;
1086 int off = (int)(current_ScreenLine - ScreenLines);
1087 int ri;
1088
1089 // Build the fold line:
1090 // 1. Add the cmdwin_type for the command-line window
1091 // 2. Add the 'foldcolumn'
1092 // 3. Add the 'number' or 'relativenumber' column
1093 // 4. Compose the text
1094 // 5. Add the text
1095 // 6. set highlighting for the Visual area an other text
1096 col = 0;
1097
1098 // 1. Add the cmdwin_type for the command-line window
1099 // Ignores 'rightleft', this window is never right-left.
1100#ifdef FEAT_CMDWIN
1101 if (cmdwin_type != 0 && wp == curwin)
1102 {
1103 ScreenLines[off] = cmdwin_type;
1104 ScreenAttrs[off] = HL_ATTR(HLF_AT);
1105 if (enc_utf8)
1106 ScreenLinesUC[off] = 0;
1107 ++col;
1108 }
1109#endif
1110
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001111#ifdef FEAT_RIGHTLEFT
1112# define RL_MEMSET(p, v, l) \
1113 do { \
1114 if (wp->w_p_rl) \
kylo252ae6f1d82022-02-16 19:24:07 +00001115 for (ri = 0; ri < (l); ++ri) \
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001116 ScreenAttrs[off + (wp->w_width - (p) - (l)) + ri] = v; \
1117 else \
kylo252ae6f1d82022-02-16 19:24:07 +00001118 for (ri = 0; ri < (l); ++ri) \
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001119 ScreenAttrs[off + (p) + ri] = v; \
1120 } while (0)
1121#else
1122# define RL_MEMSET(p, v, l) \
1123 do { \
1124 for (ri = 0; ri < l; ++ri) \
1125 ScreenAttrs[off + (p) + ri] = v; \
1126 } while (0)
1127#endif
1128
Bram Moolenaar4fa11752021-03-03 13:26:02 +01001129 // 2. Add the 'foldcolumn'
1130 // Reduce the width when there is not enough space.
1131 fdc = compute_foldcolumn(wp, col);
1132 if (fdc > 0)
1133 {
1134 char_u *p;
1135 int i;
1136 int idx;
1137
1138 fill_foldcolumn(buf, wp, TRUE, lnum);
1139 p = buf;
1140 for (i = 0; i < fdc; i++)
1141 {
1142 int ch;
1143
1144 if (has_mbyte)
1145 ch = mb_ptr2char_adv(&p);
1146 else
1147 ch = *p++;
1148#ifdef FEAT_RIGHTLEFT
1149 if (wp->w_p_rl)
1150 idx = off + wp->w_width - i - 1 - col;
1151 else
1152#endif
1153 idx = off + col + i;
1154 if (enc_utf8)
1155 {
1156 if (ch >= 0x80)
1157 {
1158 ScreenLinesUC[idx] = ch;
1159 ScreenLinesC[0][idx] = 0;
1160 ScreenLines[idx] = 0x80;
1161 }
1162 else
1163 {
1164 ScreenLines[idx] = ch;
1165 ScreenLinesUC[idx] = 0;
1166 }
1167 }
1168 else
1169 ScreenLines[idx] = ch;
1170 }
1171
1172 RL_MEMSET(col, HL_ATTR(HLF_FC), fdc);
1173 col += fdc;
1174 }
1175
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001176 // Set all attributes of the 'number' or 'relativenumber' column and the
1177 // text
1178 RL_MEMSET(col, HL_ATTR(HLF_FL), wp->w_width - col);
1179
1180#ifdef FEAT_SIGNS
1181 // If signs are being displayed, add two spaces.
1182 if (signcolumn_on(wp))
1183 {
1184 len = wp->w_width - col;
1185 if (len > 0)
1186 {
1187 if (len > 2)
1188 len = 2;
1189# ifdef FEAT_RIGHTLEFT
1190 if (wp->w_p_rl)
1191 // the line number isn't reversed
1192 copy_text_attr(off + wp->w_width - len - col,
1193 (char_u *)" ", len, HL_ATTR(HLF_FL));
1194 else
1195# endif
1196 copy_text_attr(off + col, (char_u *)" ", len, HL_ATTR(HLF_FL));
1197 col += len;
1198 }
1199 }
1200#endif
1201
1202 // 3. Add the 'number' or 'relativenumber' column
1203 if (wp->w_p_nu || wp->w_p_rnu)
1204 {
1205 len = wp->w_width - col;
1206 if (len > 0)
1207 {
1208 int w = number_width(wp);
1209 long num;
1210 char *fmt = "%*ld ";
1211
1212 if (len > w + 1)
1213 len = w + 1;
1214
1215 if (wp->w_p_nu && !wp->w_p_rnu)
1216 // 'number' + 'norelativenumber'
1217 num = (long)lnum;
1218 else
1219 {
1220 // 'relativenumber', don't use negative numbers
1221 num = labs((long)get_cursor_rel_lnum(wp, lnum));
1222 if (num == 0 && wp->w_p_nu && wp->w_p_rnu)
1223 {
1224 // 'number' + 'relativenumber': cursor line shows absolute
1225 // line number
1226 num = lnum;
1227 fmt = "%-*ld ";
1228 }
1229 }
1230
1231 sprintf((char *)buf, fmt, w, num);
1232#ifdef FEAT_RIGHTLEFT
1233 if (wp->w_p_rl)
1234 // the line number isn't reversed
1235 copy_text_attr(off + wp->w_width - len - col, buf, len,
1236 HL_ATTR(HLF_FL));
1237 else
1238#endif
1239 copy_text_attr(off + col, buf, len, HL_ATTR(HLF_FL));
1240 col += len;
1241 }
1242 }
1243
1244 // 4. Compose the folded-line string with 'foldtext', if set.
1245 text = get_foldtext(wp, lnum, lnume, foldinfo, buf);
1246
1247 txtcol = col; // remember where text starts
1248
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001249 // 5. move the text to current_ScreenLine. Fill up with "fold" from
1250 // 'fillchars'.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001251 // Right-left text is put in columns 0 - number-col, normal text is put
1252 // in columns number-col - window-width.
1253 col = text_to_screenline(wp, text, col);
1254
1255 // Fill the rest of the line with the fold filler
1256#ifdef FEAT_RIGHTLEFT
1257 if (wp->w_p_rl)
1258 col -= txtcol;
1259#endif
1260 while (col < wp->w_width
1261#ifdef FEAT_RIGHTLEFT
1262 - (wp->w_p_rl ? txtcol : 0)
1263#endif
1264 )
1265 {
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001266 int c = wp->w_fill_chars.fold;
1267
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001268 if (enc_utf8)
1269 {
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001270 if (c >= 0x80)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001271 {
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001272 ScreenLinesUC[off + col] = c;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001273 ScreenLinesC[0][off + col] = 0;
1274 ScreenLines[off + col] = 0x80; // avoid storing zero
1275 }
1276 else
1277 {
1278 ScreenLinesUC[off + col] = 0;
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001279 ScreenLines[off + col] = c;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001280 }
1281 col++;
1282 }
1283 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001284 ScreenLines[off + col++] = c;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001285 }
1286
1287 if (text != buf)
1288 vim_free(text);
1289
1290 // 6. set highlighting for the Visual area an other text.
1291 // If all folded lines are in the Visual area, highlight the line.
1292 if (VIsual_active && wp->w_buffer == curwin->w_buffer)
1293 {
1294 if (LTOREQ_POS(curwin->w_cursor, VIsual))
1295 {
1296 // Visual is after curwin->w_cursor
1297 top = &curwin->w_cursor;
1298 bot = &VIsual;
1299 }
1300 else
1301 {
1302 // Visual is before curwin->w_cursor
1303 top = &VIsual;
1304 bot = &curwin->w_cursor;
1305 }
1306 if (lnum >= top->lnum
1307 && lnume <= bot->lnum
1308 && (VIsual_mode != 'v'
1309 || ((lnum > top->lnum
1310 || (lnum == top->lnum
1311 && top->col == 0))
1312 && (lnume < bot->lnum
1313 || (lnume == bot->lnum
1314 && (bot->col - (*p_sel == 'e'))
1315 >= (colnr_T)STRLEN(ml_get_buf(wp->w_buffer, lnume, FALSE)))))))
1316 {
1317 if (VIsual_mode == Ctrl_V)
1318 {
1319 // Visual block mode: highlight the chars part of the block
1320 if (wp->w_old_cursor_fcol + txtcol < (colnr_T)wp->w_width)
1321 {
1322 if (wp->w_old_cursor_lcol != MAXCOL
1323 && wp->w_old_cursor_lcol + txtcol
1324 < (colnr_T)wp->w_width)
1325 len = wp->w_old_cursor_lcol;
1326 else
1327 len = wp->w_width - txtcol;
1328 RL_MEMSET(wp->w_old_cursor_fcol + txtcol, HL_ATTR(HLF_V),
1329 len - (int)wp->w_old_cursor_fcol);
1330 }
1331 }
1332 else
1333 {
1334 // Set all attributes of the text
1335 RL_MEMSET(txtcol, HL_ATTR(HLF_V), wp->w_width - txtcol);
1336 }
1337 }
1338 }
1339
1340#ifdef FEAT_SYN_HL
1341 // Show colorcolumn in the fold line, but let cursorcolumn override it.
1342 if (wp->w_p_cc_cols)
1343 {
1344 int i = 0;
1345 int j = wp->w_p_cc_cols[i];
1346 int old_txtcol = txtcol;
1347
1348 while (j > -1)
1349 {
1350 txtcol += j;
1351 if (wp->w_p_wrap)
1352 txtcol -= wp->w_skipcol;
1353 else
1354 txtcol -= wp->w_leftcol;
1355 if (txtcol >= 0 && txtcol < wp->w_width)
1356 ScreenAttrs[off + txtcol] = hl_combine_attr(
1357 ScreenAttrs[off + txtcol], HL_ATTR(HLF_MC));
1358 txtcol = old_txtcol;
1359 j = wp->w_p_cc_cols[++i];
1360 }
1361 }
1362
1363 // Show 'cursorcolumn' in the fold line.
1364 if (wp->w_p_cuc)
1365 {
1366 txtcol += wp->w_virtcol;
1367 if (wp->w_p_wrap)
1368 txtcol -= wp->w_skipcol;
1369 else
1370 txtcol -= wp->w_leftcol;
1371 if (txtcol >= 0 && txtcol < wp->w_width)
1372 ScreenAttrs[off + txtcol] = hl_combine_attr(
1373 ScreenAttrs[off + txtcol], HL_ATTR(HLF_CUC));
1374 }
1375#endif
1376
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001377 screen_line(wp, row + W_WINROW(wp), wp->w_wincol,
1378 wp->w_width, wp->w_width, 0);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001379
1380 // Update w_cline_height and w_cline_folded if the cursor line was
1381 // updated (saves a call to plines() later).
1382 if (wp == curwin
1383 && lnum <= curwin->w_cursor.lnum
1384 && lnume >= curwin->w_cursor.lnum)
1385 {
1386 curwin->w_cline_row = row;
1387 curwin->w_cline_height = 1;
1388 curwin->w_cline_folded = TRUE;
1389 curwin->w_valid |= (VALID_CHEIGHT|VALID_CROW);
1390 }
Bram Moolenaar00aaa512021-07-03 18:04:11 +02001391
1392# ifdef FEAT_CONCEAL
1393 // When the line was not folded w_wrow may have been set, recompute it.
Bram Moolenaar5cb09622021-07-05 22:03:04 +02001394 if (wp == curwin
1395 && wp->w_cursor.lnum >= lnum
1396 && wp->w_cursor.lnum <= lnume
1397 && conceal_cursor_line(wp))
Bram Moolenaar00aaa512021-07-03 18:04:11 +02001398 curs_columns(TRUE);
1399# endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001400}
1401#endif
1402
1403/*
1404 * Update a single window.
1405 *
1406 * This may cause the windows below it also to be redrawn (when clearing the
1407 * screen or scrolling lines).
1408 *
1409 * How the window is redrawn depends on wp->w_redr_type. Each type also
1410 * implies the one below it.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001411 * UPD_NOT_VALID redraw the whole window
1412 * UPD_SOME_VALID redraw the whole window but do scroll when possible
1413 * UPD_REDRAW_TOP redraw the top w_upd_rows window lines, otherwise like
1414 * UPD_VALID
1415 * UPD_INVERTED redraw the changed part of the Visual area
1416 * UPD_INVERTED_ALL redraw the whole Visual area
1417 * UPD_VALID 1. scroll up/down to adjust for a changed w_topline
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001418 * 2. update lines at the top when scrolled down
1419 * 3. redraw changed text:
1420 * - if wp->w_buffer->b_mod_set set, update lines between
1421 * b_mod_top and b_mod_bot.
1422 * - if wp->w_redraw_top non-zero, redraw lines between
1423 * wp->w_redraw_top and wp->w_redr_bot.
1424 * - continue redrawing when syntax status is invalid.
1425 * 4. if scrolled up, update lines at the bottom.
1426 * This results in three areas that may need updating:
1427 * top: from first row to top_end (when scrolled down)
1428 * mid: from mid_start to mid_end (update inversion or changed text)
1429 * bot: from bot_start to last row (when scrolled up)
1430 */
1431 static void
1432win_update(win_T *wp)
1433{
1434 buf_T *buf = wp->w_buffer;
1435 int type;
1436 int top_end = 0; // Below last row of the top area that needs
1437 // updating. 0 when no top area updating.
1438 int mid_start = 999;// first row of the mid area that needs
1439 // updating. 999 when no mid area updating.
1440 int mid_end = 0; // Below last row of the mid area that needs
1441 // updating. 0 when no mid area updating.
1442 int bot_start = 999;// first row of the bot area that needs
1443 // updating. 999 when no bot area updating
1444 int scrolled_down = FALSE; // TRUE when scrolled down when
1445 // w_topline got smaller a bit
1446#ifdef FEAT_SEARCH_EXTRA
1447 int top_to_mod = FALSE; // redraw above mod_top
1448#endif
1449
1450 int row; // current window row to display
1451 linenr_T lnum; // current buffer lnum to display
1452 int idx; // current index in w_lines[]
1453 int srow; // starting row of the current line
1454
1455 int eof = FALSE; // if TRUE, we hit the end of the file
1456 int didline = FALSE; // if TRUE, we finished the last line
1457 int i;
1458 long j;
1459 static int recursive = FALSE; // being called recursively
Bram Moolenaarcbee6352019-11-12 20:49:15 +01001460 linenr_T old_botline = wp->w_botline;
1461#ifdef FEAT_CONCEAL
1462 int old_wrow = wp->w_wrow;
1463 int old_wcol = wp->w_wcol;
1464#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001465#ifdef FEAT_FOLDING
1466 long fold_count;
1467#endif
1468#ifdef FEAT_SYN_HL
1469 // remember what happened to the previous line, to know if
1470 // check_visual_highlight() can be used
Bram Moolenaare7a74d52022-03-19 11:10:15 +00001471# define DID_NONE 1 // didn't update a line
1472# define DID_LINE 2 // updated a normal line
1473# define DID_FOLD 3 // updated a folded line
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001474 int did_update = DID_NONE;
1475 linenr_T syntax_last_parsed = 0; // last parsed text line
1476#endif
1477 linenr_T mod_top = 0;
1478 linenr_T mod_bot = 0;
1479#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
1480 int save_got_int;
1481#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001482
Bram Moolenaare52e0c82020-02-28 22:20:10 +01001483#if defined(FEAT_SEARCH_EXTRA) || defined(FEAT_CLIPBOARD)
1484 // This needs to be done only for the first window when update_screen() is
1485 // called.
1486 if (!did_update_one_window)
1487 {
1488 did_update_one_window = TRUE;
1489# ifdef FEAT_SEARCH_EXTRA
1490 start_search_hl();
1491# endif
1492# ifdef FEAT_CLIPBOARD
1493 // When Visual area changed, may have to update selection.
1494 if (clip_star.available && clip_isautosel_star())
1495 clip_update_selection(&clip_star);
1496 if (clip_plus.available && clip_isautosel_plus())
1497 clip_update_selection(&clip_plus);
1498# endif
1499 }
1500#endif
1501
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001502 type = wp->w_redr_type;
1503
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001504 if (type == UPD_NOT_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001505 {
1506 wp->w_redr_status = TRUE;
1507 wp->w_lines_valid = 0;
1508 }
1509
Bram Moolenaarae0f1512021-03-30 22:12:12 +02001510 // Window frame is zero-height: nothing to draw.
1511 if (wp->w_height + WINBAR_HEIGHT(wp) == 0
1512 || (wp->w_frame->fr_height == wp->w_status_height
1513#if defined(FEAT_PROP_POPUP)
1514 && !popup_is_popup(wp)
1515#endif
1516 ))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001517 {
1518 wp->w_redr_type = 0;
1519 return;
1520 }
1521
1522 // Window is zero-width: Only need to draw the separator.
1523 if (wp->w_width == 0)
1524 {
1525 // draw the vertical separator right of this window
1526 draw_vsep_win(wp, 0);
1527 wp->w_redr_type = 0;
1528 return;
1529 }
1530
1531#ifdef FEAT_TERMINAL
1532 // If this window contains a terminal, redraw works completely differently.
1533 if (term_do_update_window(wp))
1534 {
1535 term_update_window(wp);
1536# ifdef FEAT_MENU
1537 // Draw the window toolbar, if there is one.
1538 if (winbar_height(wp) > 0)
1539 redraw_win_toolbar(wp);
1540# endif
1541 wp->w_redr_type = 0;
1542 return;
1543 }
1544#endif
1545
1546#ifdef FEAT_SEARCH_EXTRA
1547 init_search_hl(wp, &screen_search_hl);
1548#endif
1549
1550#ifdef FEAT_LINEBREAK
1551 // Force redraw when width of 'number' or 'relativenumber' column
1552 // changes.
1553 i = (wp->w_p_nu || wp->w_p_rnu) ? number_width(wp) : 0;
1554 if (wp->w_nrwidth != i)
1555 {
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001556 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001557 wp->w_nrwidth = i;
1558 }
1559 else
1560#endif
1561
1562 if (buf->b_mod_set && buf->b_mod_xlines != 0 && wp->w_redraw_top != 0)
1563 {
1564 // When there are both inserted/deleted lines and specific lines to be
1565 // redrawn, w_redraw_top and w_redraw_bot may be invalid, just redraw
1566 // everything (only happens when redrawing is off for while).
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001567 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001568 }
1569 else
1570 {
1571 // Set mod_top to the first line that needs displaying because of
1572 // changes. Set mod_bot to the first line after the changes.
1573 mod_top = wp->w_redraw_top;
1574 if (wp->w_redraw_bot != 0)
1575 mod_bot = wp->w_redraw_bot + 1;
1576 else
1577 mod_bot = 0;
1578 if (buf->b_mod_set)
1579 {
1580 if (mod_top == 0 || mod_top > buf->b_mod_top)
1581 {
1582 mod_top = buf->b_mod_top;
1583#ifdef FEAT_SYN_HL
1584 // Need to redraw lines above the change that may be included
1585 // in a pattern match.
1586 if (syntax_present(wp))
1587 {
1588 mod_top -= buf->b_s.b_syn_sync_linebreaks;
1589 if (mod_top < 1)
1590 mod_top = 1;
1591 }
1592#endif
1593 }
1594 if (mod_bot == 0 || mod_bot < buf->b_mod_bot)
1595 mod_bot = buf->b_mod_bot;
1596
1597#ifdef FEAT_SEARCH_EXTRA
1598 // When 'hlsearch' is on and using a multi-line search pattern, a
1599 // change in one line may make the Search highlighting in a
1600 // previous line invalid. Simple solution: redraw all visible
1601 // lines above the change.
1602 // Same for a match pattern.
1603 if (screen_search_hl.rm.regprog != NULL
1604 && re_multiline(screen_search_hl.rm.regprog))
1605 top_to_mod = TRUE;
1606 else
1607 {
1608 matchitem_T *cur = wp->w_match_head;
1609
1610 while (cur != NULL)
1611 {
1612 if (cur->match.regprog != NULL
1613 && re_multiline(cur->match.regprog))
1614 {
1615 top_to_mod = TRUE;
1616 break;
1617 }
1618 cur = cur->next;
1619 }
1620 }
1621#endif
1622 }
Bram Moolenaar368137a2022-05-31 13:43:12 +01001623
1624#ifdef FEAT_SEARCH_EXTRA
1625 if (search_hl_has_cursor_lnum > 0)
1626 {
1627 // CurSearch was used last time, need to redraw the line with it to
1628 // avoid having two matches highlighted with CurSearch.
1629 if (mod_top == 0 || mod_top > search_hl_has_cursor_lnum)
1630 mod_top = search_hl_has_cursor_lnum;
1631 if (mod_bot == 0 || mod_bot < search_hl_has_cursor_lnum + 1)
1632 mod_bot = search_hl_has_cursor_lnum + 1;
1633 }
1634#endif
1635
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001636#ifdef FEAT_FOLDING
1637 if (mod_top != 0 && hasAnyFolding(wp))
1638 {
1639 linenr_T lnumt, lnumb;
1640
1641 // A change in a line can cause lines above it to become folded or
1642 // unfolded. Find the top most buffer line that may be affected.
1643 // If the line was previously folded and displayed, get the first
1644 // line of that fold. If the line is folded now, get the first
1645 // folded line. Use the minimum of these two.
1646
1647 // Find last valid w_lines[] entry above mod_top. Set lnumt to
1648 // the line below it. If there is no valid entry, use w_topline.
1649 // Find the first valid w_lines[] entry below mod_bot. Set lnumb
1650 // to this line. If there is no valid entry, use MAXLNUM.
1651 lnumt = wp->w_topline;
1652 lnumb = MAXLNUM;
1653 for (i = 0; i < wp->w_lines_valid; ++i)
1654 if (wp->w_lines[i].wl_valid)
1655 {
1656 if (wp->w_lines[i].wl_lastlnum < mod_top)
1657 lnumt = wp->w_lines[i].wl_lastlnum + 1;
1658 if (lnumb == MAXLNUM && wp->w_lines[i].wl_lnum >= mod_bot)
1659 {
1660 lnumb = wp->w_lines[i].wl_lnum;
1661 // When there is a fold column it might need updating
1662 // in the next line ("J" just above an open fold).
1663 if (compute_foldcolumn(wp, 0) > 0)
1664 ++lnumb;
1665 }
1666 }
1667
1668 (void)hasFoldingWin(wp, mod_top, &mod_top, NULL, TRUE, NULL);
1669 if (mod_top > lnumt)
1670 mod_top = lnumt;
1671
1672 // Now do the same for the bottom line (one above mod_bot).
1673 --mod_bot;
1674 (void)hasFoldingWin(wp, mod_bot, NULL, &mod_bot, TRUE, NULL);
1675 ++mod_bot;
1676 if (mod_bot < lnumb)
1677 mod_bot = lnumb;
1678 }
1679#endif
1680
1681 // When a change starts above w_topline and the end is below
1682 // w_topline, start redrawing at w_topline.
1683 // If the end of the change is above w_topline: do like no change was
1684 // made, but redraw the first line to find changes in syntax.
1685 if (mod_top != 0 && mod_top < wp->w_topline)
1686 {
1687 if (mod_bot > wp->w_topline)
1688 mod_top = wp->w_topline;
1689#ifdef FEAT_SYN_HL
1690 else if (syntax_present(wp))
1691 top_end = 1;
1692#endif
1693 }
1694
1695 // When line numbers are displayed need to redraw all lines below
1696 // inserted/deleted lines.
1697 if (mod_top != 0 && buf->b_mod_xlines != 0 && wp->w_p_nu)
1698 mod_bot = MAXLNUM;
1699 }
1700 wp->w_redraw_top = 0; // reset for next time
1701 wp->w_redraw_bot = 0;
Bram Moolenaar368137a2022-05-31 13:43:12 +01001702#ifdef FEAT_SEARCH_EXTRA
1703 search_hl_has_cursor_lnum = 0;
1704#endif
1705
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001706
1707 // When only displaying the lines at the top, set top_end. Used when
1708 // window has scrolled down for msg_scrolled.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001709 if (type == UPD_REDRAW_TOP)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001710 {
1711 j = 0;
1712 for (i = 0; i < wp->w_lines_valid; ++i)
1713 {
1714 j += wp->w_lines[i].wl_size;
1715 if (j >= wp->w_upd_rows)
1716 {
1717 top_end = j;
1718 break;
1719 }
1720 }
1721 if (top_end == 0)
1722 // not found (cannot happen?): redraw everything
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001723 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001724 else
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001725 // top area defined, the rest is UPD_VALID
1726 type = UPD_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001727 }
1728
1729 // Trick: we want to avoid clearing the screen twice. screenclear() will
1730 // set "screen_cleared" to TRUE. The special value MAYBE (which is still
1731 // non-zero and thus not FALSE) will indicate that screenclear() was not
1732 // called.
1733 if (screen_cleared)
1734 screen_cleared = MAYBE;
1735
1736 // If there are no changes on the screen that require a complete redraw,
1737 // handle three cases:
1738 // 1: we are off the top of the screen by a few lines: scroll down
1739 // 2: wp->w_topline is below wp->w_lines[0].wl_lnum: may scroll up
1740 // 3: wp->w_topline is wp->w_lines[0].wl_lnum: find first entry in
1741 // w_lines[] that needs updating.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001742 if ((type == UPD_VALID || type == UPD_SOME_VALID
1743 || type == UPD_INVERTED || type == UPD_INVERTED_ALL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001744#ifdef FEAT_DIFF
1745 && !wp->w_botfill && !wp->w_old_botfill
1746#endif
1747 )
1748 {
Bram Moolenaarf8992d42020-08-01 19:14:13 +02001749 if (mod_top != 0
1750 && wp->w_topline == mod_top
1751 && (!wp->w_lines[0].wl_valid
Bram Moolenaarabb6fbd2022-03-25 15:42:27 +00001752 || wp->w_topline == wp->w_lines[0].wl_lnum))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001753 {
Bram Moolenaarf8992d42020-08-01 19:14:13 +02001754 // w_topline is the first changed line and window is not scrolled,
1755 // the scrolling from changed lines will be done further down.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001756 }
1757 else if (wp->w_lines[0].wl_valid
1758 && (wp->w_topline < wp->w_lines[0].wl_lnum
1759#ifdef FEAT_DIFF
1760 || (wp->w_topline == wp->w_lines[0].wl_lnum
1761 && wp->w_topfill > wp->w_old_topfill)
1762#endif
1763 ))
1764 {
1765 // New topline is above old topline: May scroll down.
1766#ifdef FEAT_FOLDING
1767 if (hasAnyFolding(wp))
1768 {
1769 linenr_T ln;
1770
1771 // count the number of lines we are off, counting a sequence
1772 // of folded lines as one
1773 j = 0;
1774 for (ln = wp->w_topline; ln < wp->w_lines[0].wl_lnum; ++ln)
1775 {
1776 ++j;
1777 if (j >= wp->w_height - 2)
1778 break;
1779 (void)hasFoldingWin(wp, ln, NULL, &ln, TRUE, NULL);
1780 }
1781 }
1782 else
1783#endif
1784 j = wp->w_lines[0].wl_lnum - wp->w_topline;
1785 if (j < wp->w_height - 2) // not too far off
1786 {
1787 i = plines_m_win(wp, wp->w_topline, wp->w_lines[0].wl_lnum - 1);
1788#ifdef FEAT_DIFF
1789 // insert extra lines for previously invisible filler lines
1790 if (wp->w_lines[0].wl_lnum != wp->w_topline)
1791 i += diff_check_fill(wp, wp->w_lines[0].wl_lnum)
1792 - wp->w_old_topfill;
1793#endif
1794 if (i < wp->w_height - 2) // less than a screen off
1795 {
1796 // Try to insert the correct number of lines.
1797 // If not the last window, delete the lines at the bottom.
1798 // win_ins_lines may fail when the terminal can't do it.
1799 if (i > 0)
1800 check_for_delay(FALSE);
1801 if (win_ins_lines(wp, 0, i, FALSE, wp == firstwin) == OK)
1802 {
1803 if (wp->w_lines_valid != 0)
1804 {
1805 // Need to update rows that are new, stop at the
1806 // first one that scrolled down.
1807 top_end = i;
1808 scrolled_down = TRUE;
1809
1810 // Move the entries that were scrolled, disable
1811 // the entries for the lines to be redrawn.
Bram Moolenaarfdc5d172022-08-11 15:52:14 +01001812 // Avoid using a wrong index when 'cmdheight' is
1813 // zero and wp->w_height == Rows.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001814 if ((wp->w_lines_valid += j) > wp->w_height)
1815 wp->w_lines_valid = wp->w_height;
Bram Moolenaarfdc5d172022-08-11 15:52:14 +01001816 for (idx = wp->w_lines_valid >= wp->w_height
1817 ? wp->w_height - 1 : wp->w_lines_valid;
1818 idx - j >= 0; idx--)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001819 wp->w_lines[idx] = wp->w_lines[idx - j];
1820 while (idx >= 0)
1821 wp->w_lines[idx--].wl_valid = FALSE;
1822 }
1823 }
1824 else
1825 mid_start = 0; // redraw all lines
1826 }
1827 else
1828 mid_start = 0; // redraw all lines
1829 }
1830 else
1831 mid_start = 0; // redraw all lines
1832 }
1833 else
1834 {
1835 // New topline is at or below old topline: May scroll up.
1836 // When topline didn't change, find first entry in w_lines[] that
1837 // needs updating.
1838
Bram Moolenaarf6ebc822022-01-16 13:58:33 +00001839 // Try to find wp->w_topline in wp->w_lines[].wl_lnum. The check
1840 // for "Rows" is in case "wl_size" is incorrect somehow.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001841 j = -1;
1842 row = 0;
Bram Moolenaarf6ebc822022-01-16 13:58:33 +00001843 for (i = 0; i < wp->w_lines_valid && i < Rows; i++)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001844 {
1845 if (wp->w_lines[i].wl_valid
1846 && wp->w_lines[i].wl_lnum == wp->w_topline)
1847 {
1848 j = i;
1849 break;
1850 }
1851 row += wp->w_lines[i].wl_size;
1852 }
1853 if (j == -1)
1854 {
1855 // if wp->w_topline is not in wp->w_lines[].wl_lnum redraw all
1856 // lines
1857 mid_start = 0;
1858 }
1859 else
1860 {
1861 // Try to delete the correct number of lines.
1862 // wp->w_topline is at wp->w_lines[i].wl_lnum.
1863#ifdef FEAT_DIFF
1864 // If the topline didn't change, delete old filler lines,
1865 // otherwise delete filler lines of the new topline...
1866 if (wp->w_lines[0].wl_lnum == wp->w_topline)
1867 row += wp->w_old_topfill;
1868 else
1869 row += diff_check_fill(wp, wp->w_topline);
1870 // ... but don't delete new filler lines.
1871 row -= wp->w_topfill;
1872#endif
Bram Moolenaarf6ebc822022-01-16 13:58:33 +00001873 if (row > Rows) // just in case
1874 row = Rows;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001875 if (row > 0)
1876 {
1877 check_for_delay(FALSE);
1878 if (win_del_lines(wp, 0, row, FALSE, wp == firstwin, 0)
1879 == OK)
1880 bot_start = wp->w_height - row;
1881 else
1882 mid_start = 0; // redraw all lines
1883 }
1884 if ((row == 0 || bot_start < 999) && wp->w_lines_valid != 0)
1885 {
1886 // Skip the lines (below the deleted lines) that are still
1887 // valid and don't need redrawing. Copy their info
1888 // upwards, to compensate for the deleted lines. Set
1889 // bot_start to the first row that needs redrawing.
1890 bot_start = 0;
1891 idx = 0;
1892 for (;;)
1893 {
1894 wp->w_lines[idx] = wp->w_lines[j];
1895 // stop at line that didn't fit, unless it is still
1896 // valid (no lines deleted)
1897 if (row > 0 && bot_start + row
1898 + (int)wp->w_lines[j].wl_size > wp->w_height)
1899 {
1900 wp->w_lines_valid = idx + 1;
1901 break;
1902 }
1903 bot_start += wp->w_lines[idx++].wl_size;
1904
1905 // stop at the last valid entry in w_lines[].wl_size
1906 if (++j >= wp->w_lines_valid)
1907 {
1908 wp->w_lines_valid = idx;
1909 break;
1910 }
1911 }
1912#ifdef FEAT_DIFF
1913 // Correct the first entry for filler lines at the top
1914 // when it won't get updated below.
1915 if (wp->w_p_diff && bot_start > 0)
1916 wp->w_lines[0].wl_size =
1917 plines_win_nofill(wp, wp->w_topline, TRUE)
1918 + wp->w_topfill;
1919#endif
1920 }
1921 }
1922 }
1923
1924 // When starting redraw in the first line, redraw all lines. When
1925 // there is only one window it's probably faster to clear the screen
1926 // first.
1927 if (mid_start == 0)
1928 {
1929 mid_end = wp->w_height;
1930 if (ONE_WINDOW && !WIN_IS_POPUP(wp))
1931 {
1932 // Clear the screen when it was not done by win_del_lines() or
1933 // win_ins_lines() above, "screen_cleared" is FALSE or MAYBE
1934 // then.
1935 if (screen_cleared != TRUE)
1936 screenclear();
1937 // The screen was cleared, redraw the tab pages line.
1938 if (redraw_tabline)
1939 draw_tabline();
1940 }
1941 }
1942
1943 // When win_del_lines() or win_ins_lines() caused the screen to be
1944 // cleared (only happens for the first window) or when screenclear()
1945 // was called directly above, "must_redraw" will have been set to
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001946 // UPD_NOT_VALID, need to reset it here to avoid redrawing twice.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001947 if (screen_cleared == TRUE)
1948 must_redraw = 0;
1949 }
1950 else
1951 {
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001952 // Not UPD_VALID or UPD_INVERTED: redraw all lines.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001953 mid_start = 0;
1954 mid_end = wp->w_height;
1955 }
1956
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001957 if (type == UPD_SOME_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001958 {
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001959 // UPD_SOME_VALID: redraw all lines.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001960 mid_start = 0;
1961 mid_end = wp->w_height;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001962 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001963 }
1964
1965 // check if we are updating or removing the inverted part
1966 if ((VIsual_active && buf == curwin->w_buffer)
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001967 || (wp->w_old_cursor_lnum != 0 && type != UPD_NOT_VALID))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001968 {
1969 linenr_T from, to;
1970
1971 if (VIsual_active)
1972 {
Bram Moolenaarfe154992022-03-22 20:42:12 +00001973 if (VIsual_mode != wp->w_old_visual_mode
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001974 || type == UPD_INVERTED_ALL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001975 {
1976 // If the type of Visual selection changed, redraw the whole
1977 // selection. Also when the ownership of the X selection is
1978 // gained or lost.
1979 if (curwin->w_cursor.lnum < VIsual.lnum)
1980 {
1981 from = curwin->w_cursor.lnum;
1982 to = VIsual.lnum;
1983 }
1984 else
1985 {
1986 from = VIsual.lnum;
1987 to = curwin->w_cursor.lnum;
1988 }
1989 // redraw more when the cursor moved as well
1990 if (wp->w_old_cursor_lnum < from)
1991 from = wp->w_old_cursor_lnum;
1992 if (wp->w_old_cursor_lnum > to)
1993 to = wp->w_old_cursor_lnum;
1994 if (wp->w_old_visual_lnum < from)
1995 from = wp->w_old_visual_lnum;
1996 if (wp->w_old_visual_lnum > to)
1997 to = wp->w_old_visual_lnum;
1998 }
1999 else
2000 {
2001 // Find the line numbers that need to be updated: The lines
2002 // between the old cursor position and the current cursor
2003 // position. Also check if the Visual position changed.
2004 if (curwin->w_cursor.lnum < wp->w_old_cursor_lnum)
2005 {
2006 from = curwin->w_cursor.lnum;
2007 to = wp->w_old_cursor_lnum;
2008 }
2009 else
2010 {
2011 from = wp->w_old_cursor_lnum;
2012 to = curwin->w_cursor.lnum;
2013 if (from == 0) // Visual mode just started
2014 from = to;
2015 }
2016
2017 if (VIsual.lnum != wp->w_old_visual_lnum
2018 || VIsual.col != wp->w_old_visual_col)
2019 {
2020 if (wp->w_old_visual_lnum < from
2021 && wp->w_old_visual_lnum != 0)
2022 from = wp->w_old_visual_lnum;
2023 if (wp->w_old_visual_lnum > to)
2024 to = wp->w_old_visual_lnum;
2025 if (VIsual.lnum < from)
2026 from = VIsual.lnum;
2027 if (VIsual.lnum > to)
2028 to = VIsual.lnum;
2029 }
2030 }
2031
2032 // If in block mode and changed column or curwin->w_curswant:
2033 // update all lines.
2034 // First compute the actual start and end column.
2035 if (VIsual_mode == Ctrl_V)
2036 {
2037 colnr_T fromc, toc;
2038#if defined(FEAT_LINEBREAK)
Gary Johnson51ad8502021-08-03 18:33:08 +02002039 int save_ve_flags = curwin->w_ve_flags;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002040
2041 if (curwin->w_p_lbr)
Gary Johnson51ad8502021-08-03 18:33:08 +02002042 curwin->w_ve_flags = VE_ALL;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002043#endif
2044 getvcols(wp, &VIsual, &curwin->w_cursor, &fromc, &toc);
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002045 ++toc;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002046#if defined(FEAT_LINEBREAK)
Gary Johnson51ad8502021-08-03 18:33:08 +02002047 curwin->w_ve_flags = save_ve_flags;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002048#endif
Bram Moolenaar9cee4a12021-07-03 15:08:37 +02002049 // Highlight to the end of the line, unless 'virtualedit' has
2050 // "block".
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002051 if (curwin->w_curswant == MAXCOL)
2052 {
Gary Johnson53ba05b2021-07-26 22:19:10 +02002053 if (get_ve_flags() & VE_BLOCK)
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002054 {
2055 pos_T pos;
2056 int cursor_above =
2057 curwin->w_cursor.lnum < VIsual.lnum;
2058
2059 // Need to find the longest line.
2060 toc = 0;
2061 pos.coladd = 0;
2062 for (pos.lnum = curwin->w_cursor.lnum; cursor_above
2063 ? pos.lnum <= VIsual.lnum
2064 : pos.lnum >= VIsual.lnum;
2065 pos.lnum += cursor_above ? 1 : -1)
2066 {
2067 colnr_T t;
2068
Bram Moolenaar6bcb1822021-07-09 15:54:00 +02002069 pos.col = (int)STRLEN(ml_get_buf(wp->w_buffer,
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002070 pos.lnum, FALSE));
2071 getvvcol(wp, &pos, NULL, NULL, &t);
2072 if (toc < t)
2073 toc = t;
2074 }
2075 ++toc;
2076 }
2077 else
2078 toc = MAXCOL;
2079 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002080
2081 if (fromc != wp->w_old_cursor_fcol
2082 || toc != wp->w_old_cursor_lcol)
2083 {
2084 if (from > VIsual.lnum)
2085 from = VIsual.lnum;
2086 if (to < VIsual.lnum)
2087 to = VIsual.lnum;
2088 }
2089 wp->w_old_cursor_fcol = fromc;
2090 wp->w_old_cursor_lcol = toc;
2091 }
2092 }
2093 else
2094 {
2095 // Use the line numbers of the old Visual area.
2096 if (wp->w_old_cursor_lnum < wp->w_old_visual_lnum)
2097 {
2098 from = wp->w_old_cursor_lnum;
2099 to = wp->w_old_visual_lnum;
2100 }
2101 else
2102 {
2103 from = wp->w_old_visual_lnum;
2104 to = wp->w_old_cursor_lnum;
2105 }
2106 }
2107
2108 // There is no need to update lines above the top of the window.
2109 if (from < wp->w_topline)
2110 from = wp->w_topline;
2111
2112 // If we know the value of w_botline, use it to restrict the update to
2113 // the lines that are visible in the window.
2114 if (wp->w_valid & VALID_BOTLINE)
2115 {
2116 if (from >= wp->w_botline)
2117 from = wp->w_botline - 1;
2118 if (to >= wp->w_botline)
2119 to = wp->w_botline - 1;
2120 }
2121
2122 // Find the minimal part to be updated.
2123 // Watch out for scrolling that made entries in w_lines[] invalid.
2124 // E.g., CTRL-U makes the first half of w_lines[] invalid and sets
2125 // top_end; need to redraw from top_end to the "to" line.
2126 // A middle mouse click with a Visual selection may change the text
2127 // above the Visual area and reset wl_valid, do count these for
2128 // mid_end (in srow).
2129 if (mid_start > 0)
2130 {
2131 lnum = wp->w_topline;
2132 idx = 0;
2133 srow = 0;
2134 if (scrolled_down)
2135 mid_start = top_end;
2136 else
2137 mid_start = 0;
2138 while (lnum < from && idx < wp->w_lines_valid) // find start
2139 {
2140 if (wp->w_lines[idx].wl_valid)
2141 mid_start += wp->w_lines[idx].wl_size;
2142 else if (!scrolled_down)
2143 srow += wp->w_lines[idx].wl_size;
2144 ++idx;
2145# ifdef FEAT_FOLDING
2146 if (idx < wp->w_lines_valid && wp->w_lines[idx].wl_valid)
2147 lnum = wp->w_lines[idx].wl_lnum;
2148 else
2149# endif
2150 ++lnum;
2151 }
2152 srow += mid_start;
2153 mid_end = wp->w_height;
2154 for ( ; idx < wp->w_lines_valid; ++idx) // find end
2155 {
2156 if (wp->w_lines[idx].wl_valid
2157 && wp->w_lines[idx].wl_lnum >= to + 1)
2158 {
2159 // Only update until first row of this line
2160 mid_end = srow;
2161 break;
2162 }
2163 srow += wp->w_lines[idx].wl_size;
2164 }
2165 }
2166 }
2167
2168 if (VIsual_active && buf == curwin->w_buffer)
2169 {
2170 wp->w_old_visual_mode = VIsual_mode;
2171 wp->w_old_cursor_lnum = curwin->w_cursor.lnum;
2172 wp->w_old_visual_lnum = VIsual.lnum;
2173 wp->w_old_visual_col = VIsual.col;
2174 wp->w_old_curswant = curwin->w_curswant;
2175 }
2176 else
2177 {
2178 wp->w_old_visual_mode = 0;
2179 wp->w_old_cursor_lnum = 0;
2180 wp->w_old_visual_lnum = 0;
2181 wp->w_old_visual_col = 0;
2182 }
2183
2184#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
2185 // reset got_int, otherwise regexp won't work
2186 save_got_int = got_int;
2187 got_int = 0;
2188#endif
2189#ifdef SYN_TIME_LIMIT
2190 // Set the time limit to 'redrawtime'.
Bram Moolenaar6f0cf622022-06-19 12:27:45 +01002191 redrawtime_limit_set = TRUE;
Paul Ollis65745772022-06-05 16:55:54 +01002192 init_regexp_timeout(p_rdt);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002193#endif
2194#ifdef FEAT_FOLDING
2195 win_foldinfo.fi_level = 0;
2196#endif
2197
2198#ifdef FEAT_MENU
2199 // Draw the window toolbar, if there is one.
2200 // TODO: only when needed.
2201 if (winbar_height(wp) > 0)
2202 redraw_win_toolbar(wp);
2203#endif
2204
2205 // Update all the window rows.
2206 idx = 0; // first entry in w_lines[].wl_size
2207 row = 0;
2208 srow = 0;
2209 lnum = wp->w_topline; // first line shown in window
2210 for (;;)
2211 {
2212 // stop updating when reached the end of the window (check for _past_
2213 // the end of the window is at the end of the loop)
2214 if (row == wp->w_height)
2215 {
2216 didline = TRUE;
2217 break;
2218 }
2219
2220 // stop updating when hit the end of the file
2221 if (lnum > buf->b_ml.ml_line_count)
2222 {
2223 eof = TRUE;
2224 break;
2225 }
2226
2227 // Remember the starting row of the line that is going to be dealt
2228 // with. It is used further down when the line doesn't fit.
2229 srow = row;
2230
2231 // Update a line when it is in an area that needs updating, when it
2232 // has changes or w_lines[idx] is invalid.
2233 // "bot_start" may be halfway a wrapped line after using
2234 // win_del_lines(), check if the current line includes it.
2235 // When syntax folding is being used, the saved syntax states will
2236 // already have been updated, we can't see where the syntax state is
2237 // the same again, just update until the end of the window.
2238 if (row < top_end
2239 || (row >= mid_start && row < mid_end)
2240#ifdef FEAT_SEARCH_EXTRA
2241 || top_to_mod
2242#endif
2243 || idx >= wp->w_lines_valid
2244 || (row + wp->w_lines[idx].wl_size > bot_start)
2245 || (mod_top != 0
2246 && (lnum == mod_top
2247 || (lnum >= mod_top
2248 && (lnum < mod_bot
2249#ifdef FEAT_SYN_HL
2250 || did_update == DID_FOLD
2251 || (did_update == DID_LINE
2252 && syntax_present(wp)
2253 && (
2254# ifdef FEAT_FOLDING
2255 (foldmethodIsSyntax(wp)
2256 && hasAnyFolding(wp)) ||
2257# endif
2258 syntax_check_changed(lnum)))
2259#endif
2260#ifdef FEAT_SEARCH_EXTRA
2261 // match in fixed position might need redraw
2262 // if lines were inserted or deleted
2263 || (wp->w_match_head != NULL
2264 && buf->b_mod_xlines != 0)
2265#endif
Bram Moolenaar11a58af2019-10-24 22:32:31 +02002266 ))))
2267#ifdef FEAT_SYN_HL
zeertzjqc20e46a2022-03-23 14:55:23 +00002268 || (wp->w_p_cul && lnum == wp->w_cursor.lnum)
2269 || lnum == wp->w_last_cursorline
Bram Moolenaar11a58af2019-10-24 22:32:31 +02002270#endif
2271 )
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002272 {
2273#ifdef FEAT_SEARCH_EXTRA
2274 if (lnum == mod_top)
2275 top_to_mod = FALSE;
2276#endif
2277
2278 // When at start of changed lines: May scroll following lines
2279 // up or down to minimize redrawing.
2280 // Don't do this when the change continues until the end.
2281 // Don't scroll when dollar_vcol >= 0, keep the "$".
Bram Moolenaarc9e7e342021-07-22 21:33:03 +02002282 // Don't scroll when redrawing the top, scrolled already above.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002283 if (lnum == mod_top
2284 && mod_bot != MAXLNUM
Bram Moolenaarc9e7e342021-07-22 21:33:03 +02002285 && !(dollar_vcol >= 0 && mod_bot == mod_top + 1)
2286 && row >= top_end)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002287 {
2288 int old_rows = 0;
2289 int new_rows = 0;
2290 int xtra_rows;
2291 linenr_T l;
2292
2293 // Count the old number of window rows, using w_lines[], which
2294 // should still contain the sizes for the lines as they are
2295 // currently displayed.
2296 for (i = idx; i < wp->w_lines_valid; ++i)
2297 {
2298 // Only valid lines have a meaningful wl_lnum. Invalid
2299 // lines are part of the changed area.
2300 if (wp->w_lines[i].wl_valid
2301 && wp->w_lines[i].wl_lnum == mod_bot)
2302 break;
2303 old_rows += wp->w_lines[i].wl_size;
2304#ifdef FEAT_FOLDING
2305 if (wp->w_lines[i].wl_valid
2306 && wp->w_lines[i].wl_lastlnum + 1 == mod_bot)
2307 {
2308 // Must have found the last valid entry above mod_bot.
2309 // Add following invalid entries.
2310 ++i;
2311 while (i < wp->w_lines_valid
2312 && !wp->w_lines[i].wl_valid)
2313 old_rows += wp->w_lines[i++].wl_size;
2314 break;
2315 }
2316#endif
2317 }
2318
2319 if (i >= wp->w_lines_valid)
2320 {
2321 // We can't find a valid line below the changed lines,
2322 // need to redraw until the end of the window.
2323 // Inserting/deleting lines has no use.
2324 bot_start = 0;
2325 }
2326 else
2327 {
2328 // Able to count old number of rows: Count new window
2329 // rows, and may insert/delete lines
2330 j = idx;
2331 for (l = lnum; l < mod_bot; ++l)
2332 {
2333#ifdef FEAT_FOLDING
2334 if (hasFoldingWin(wp, l, NULL, &l, TRUE, NULL))
2335 ++new_rows;
2336 else
2337#endif
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002338 {
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002339#ifdef FEAT_DIFF
2340 if (l == wp->w_topline)
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002341 new_rows += plines_win_nofill(wp, l, TRUE)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002342 + wp->w_topfill;
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002343 else
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002344#endif
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002345 new_rows += plines_win(wp, l, TRUE);
2346 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002347 ++j;
2348 if (new_rows > wp->w_height - row - 2)
2349 {
2350 // it's getting too much, must redraw the rest
2351 new_rows = 9999;
2352 break;
2353 }
2354 }
2355 xtra_rows = new_rows - old_rows;
2356 if (xtra_rows < 0)
2357 {
2358 // May scroll text up. If there is not enough
2359 // remaining text or scrolling fails, must redraw the
2360 // rest. If scrolling works, must redraw the text
2361 // below the scrolled text.
2362 if (row - xtra_rows >= wp->w_height - 2)
2363 mod_bot = MAXLNUM;
2364 else
2365 {
2366 check_for_delay(FALSE);
2367 if (win_del_lines(wp, row,
2368 -xtra_rows, FALSE, FALSE, 0) == FAIL)
2369 mod_bot = MAXLNUM;
2370 else
2371 bot_start = wp->w_height + xtra_rows;
2372 }
2373 }
2374 else if (xtra_rows > 0)
2375 {
2376 // May scroll text down. If there is not enough
2377 // remaining text of scrolling fails, must redraw the
2378 // rest.
2379 if (row + xtra_rows >= wp->w_height - 2)
2380 mod_bot = MAXLNUM;
2381 else
2382 {
2383 check_for_delay(FALSE);
2384 if (win_ins_lines(wp, row + old_rows,
2385 xtra_rows, FALSE, FALSE) == FAIL)
2386 mod_bot = MAXLNUM;
2387 else if (top_end > row + old_rows)
2388 // Scrolled the part at the top that requires
2389 // updating down.
2390 top_end += xtra_rows;
2391 }
2392 }
2393
2394 // When not updating the rest, may need to move w_lines[]
2395 // entries.
2396 if (mod_bot != MAXLNUM && i != j)
2397 {
2398 if (j < i)
2399 {
2400 int x = row + new_rows;
2401
2402 // move entries in w_lines[] upwards
2403 for (;;)
2404 {
2405 // stop at last valid entry in w_lines[]
2406 if (i >= wp->w_lines_valid)
2407 {
2408 wp->w_lines_valid = j;
2409 break;
2410 }
2411 wp->w_lines[j] = wp->w_lines[i];
2412 // stop at a line that won't fit
2413 if (x + (int)wp->w_lines[j].wl_size
2414 > wp->w_height)
2415 {
2416 wp->w_lines_valid = j + 1;
2417 break;
2418 }
2419 x += wp->w_lines[j++].wl_size;
2420 ++i;
2421 }
2422 if (bot_start > x)
2423 bot_start = x;
2424 }
2425 else // j > i
2426 {
2427 // move entries in w_lines[] downwards
2428 j -= i;
2429 wp->w_lines_valid += j;
2430 if (wp->w_lines_valid > wp->w_height)
2431 wp->w_lines_valid = wp->w_height;
2432 for (i = wp->w_lines_valid; i - j >= idx; --i)
2433 wp->w_lines[i] = wp->w_lines[i - j];
2434
2435 // The w_lines[] entries for inserted lines are
2436 // now invalid, but wl_size may be used above.
2437 // Reset to zero.
2438 while (i >= idx)
2439 {
2440 wp->w_lines[i].wl_size = 0;
2441 wp->w_lines[i--].wl_valid = FALSE;
2442 }
2443 }
2444 }
2445 }
2446 }
2447
2448#ifdef FEAT_FOLDING
2449 // When lines are folded, display one line for all of them.
2450 // Otherwise, display normally (can be several display lines when
2451 // 'wrap' is on).
2452 fold_count = foldedCount(wp, lnum, &win_foldinfo);
2453 if (fold_count != 0)
2454 {
2455 fold_line(wp, fold_count, &win_foldinfo, lnum, row);
2456 ++row;
2457 --fold_count;
2458 wp->w_lines[idx].wl_folded = TRUE;
2459 wp->w_lines[idx].wl_lastlnum = lnum + fold_count;
2460# ifdef FEAT_SYN_HL
2461 did_update = DID_FOLD;
2462# endif
2463 }
2464 else
2465#endif
2466 if (idx < wp->w_lines_valid
2467 && wp->w_lines[idx].wl_valid
2468 && wp->w_lines[idx].wl_lnum == lnum
2469 && lnum > wp->w_topline
2470 && !(dy_flags & (DY_LASTLINE | DY_TRUNCATE))
2471 && !WIN_IS_POPUP(wp)
2472 && srow + wp->w_lines[idx].wl_size > wp->w_height
2473#ifdef FEAT_DIFF
2474 && diff_check_fill(wp, lnum) == 0
2475#endif
2476 )
2477 {
2478 // This line is not going to fit. Don't draw anything here,
2479 // will draw "@ " lines below.
2480 row = wp->w_height + 1;
2481 }
2482 else
2483 {
2484#ifdef FEAT_SEARCH_EXTRA
2485 prepare_search_hl(wp, &screen_search_hl, lnum);
2486#endif
2487#ifdef FEAT_SYN_HL
2488 // Let the syntax stuff know we skipped a few lines.
2489 if (syntax_last_parsed != 0 && syntax_last_parsed + 1 < lnum
2490 && syntax_present(wp))
2491 syntax_end_parsing(syntax_last_parsed + 1);
2492#endif
2493
2494 // Display one line.
2495 row = win_line(wp, lnum, srow, wp->w_height,
2496 mod_top == 0, FALSE);
2497
2498#ifdef FEAT_FOLDING
2499 wp->w_lines[idx].wl_folded = FALSE;
2500 wp->w_lines[idx].wl_lastlnum = lnum;
2501#endif
2502#ifdef FEAT_SYN_HL
2503 did_update = DID_LINE;
2504 syntax_last_parsed = lnum;
2505#endif
2506 }
2507
2508 wp->w_lines[idx].wl_lnum = lnum;
2509 wp->w_lines[idx].wl_valid = TRUE;
2510
2511 // Past end of the window or end of the screen. Note that after
2512 // resizing wp->w_height may be end up too big. That's a problem
2513 // elsewhere, but prevent a crash here.
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +01002514 if (row > wp->w_height
2515 || row + wp->w_winrow >= (p_ch > 0 ? Rows : Rows + 1))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002516 {
2517 // we may need the size of that too long line later on
2518 if (dollar_vcol == -1)
2519 wp->w_lines[idx].wl_size = plines_win(wp, lnum, TRUE);
2520 ++idx;
2521 break;
2522 }
2523 if (dollar_vcol == -1)
2524 wp->w_lines[idx].wl_size = row - srow;
2525 ++idx;
2526#ifdef FEAT_FOLDING
2527 lnum += fold_count + 1;
2528#else
2529 ++lnum;
2530#endif
2531 }
2532 else
2533 {
Lewis Russell16246392022-03-29 11:38:17 +01002534 if (wp->w_p_rnu && wp->w_last_cursor_lnum_rnu != wp->w_cursor.lnum)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002535 {
2536#ifdef FEAT_FOLDING
Lewis Russell16246392022-03-29 11:38:17 +01002537 // 'relativenumber' set and the cursor moved vertically: The
2538 // text doesn't need to be drawn, but the number column does.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002539 fold_count = foldedCount(wp, lnum, &win_foldinfo);
2540 if (fold_count != 0)
2541 fold_line(wp, fold_count, &win_foldinfo, lnum, row);
2542 else
2543#endif
2544 (void)win_line(wp, lnum, srow, wp->w_height, TRUE, TRUE);
2545 }
2546
2547 // This line does not need to be drawn, advance to the next one.
2548 row += wp->w_lines[idx++].wl_size;
2549 if (row > wp->w_height) // past end of screen
2550 break;
2551#ifdef FEAT_FOLDING
2552 lnum = wp->w_lines[idx - 1].wl_lastlnum + 1;
2553#else
2554 ++lnum;
2555#endif
2556#ifdef FEAT_SYN_HL
2557 did_update = DID_NONE;
2558#endif
2559 }
2560
2561 if (lnum > buf->b_ml.ml_line_count)
2562 {
2563 eof = TRUE;
2564 break;
2565 }
Bram Moolenaarfa1a4572022-01-16 11:42:20 +00002566
2567 // Safety check: if any of the wl_size values is wrong we might go over
2568 // the end of w_lines[].
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +01002569 if (idx >= (p_ch > 0 ? Rows : Rows + 1))
Bram Moolenaarfa1a4572022-01-16 11:42:20 +00002570 break;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002571 }
2572
2573 // End of loop over all window lines.
2574
zeertzjqc20e46a2022-03-23 14:55:23 +00002575#ifdef FEAT_SYN_HL
2576 // Now that the window has been redrawn with the old and new cursor line,
2577 // update w_last_cursorline.
2578 wp->w_last_cursorline = wp->w_p_cul ? wp->w_cursor.lnum : 0;
2579#endif
Lewis Russell16246392022-03-29 11:38:17 +01002580 wp->w_last_cursor_lnum_rnu = wp->w_p_rnu ? wp->w_cursor.lnum : 0;
zeertzjqc20e46a2022-03-23 14:55:23 +00002581
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002582#ifdef FEAT_VTP
2583 // Rewrite the character at the end of the screen line.
Bram Moolenaar7ed8f592020-04-28 20:44:42 +02002584 // See the version that was fixed.
2585 if (use_vtp() && get_conpty_fix_type() < 1)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002586 {
K.Takata54119102022-02-03 13:33:03 +00002587 int k;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002588
K.Takata54119102022-02-03 13:33:03 +00002589 for (k = 0; k < Rows; ++k)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002590 if (enc_utf8)
K.Takata54119102022-02-03 13:33:03 +00002591 if ((*mb_off2cells)(LineOffset[k] + Columns - 2,
2592 LineOffset[k] + screen_Columns) > 1)
2593 screen_draw_rectangle(k, Columns - 2, 1, 2, FALSE);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002594 else
K.Takata54119102022-02-03 13:33:03 +00002595 screen_draw_rectangle(k, Columns - 1, 1, 1, FALSE);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002596 else
K.Takata54119102022-02-03 13:33:03 +00002597 screen_char(LineOffset[k] + Columns - 1, k, Columns - 1);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002598 }
2599#endif
2600
2601 if (idx > wp->w_lines_valid)
2602 wp->w_lines_valid = idx;
2603
2604#ifdef FEAT_SYN_HL
2605 // Let the syntax stuff know we stop parsing here.
2606 if (syntax_last_parsed != 0 && syntax_present(wp))
2607 syntax_end_parsing(syntax_last_parsed + 1);
2608#endif
2609
2610 // If we didn't hit the end of the file, and we didn't finish the last
2611 // line we were working on, then the line didn't fit.
2612 wp->w_empty_rows = 0;
2613#ifdef FEAT_DIFF
2614 wp->w_filler_rows = 0;
2615#endif
2616 if (!eof && !didline)
2617 {
2618 if (lnum == wp->w_topline)
2619 {
2620 // Single line that does not fit!
2621 // Don't overwrite it, it can be edited.
2622 wp->w_botline = lnum + 1;
2623 }
2624#ifdef FEAT_DIFF
2625 else if (diff_check_fill(wp, lnum) >= wp->w_height - srow)
2626 {
2627 // Window ends in filler lines.
2628 wp->w_botline = lnum;
2629 wp->w_filler_rows = wp->w_height - srow;
2630 }
2631#endif
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002632#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002633 else if (WIN_IS_POPUP(wp))
2634 {
2635 // popup line that doesn't fit is left as-is
2636 wp->w_botline = lnum;
2637 }
2638#endif
2639 else if (dy_flags & DY_TRUNCATE) // 'display' has "truncate"
2640 {
2641 int scr_row = W_WINROW(wp) + wp->w_height - 1;
2642
2643 // Last line isn't finished: Display "@@@" in the last screen line.
Bram Moolenaarcee9c842022-04-09 12:40:13 +01002644 screen_puts_len((char_u *)"@@", wp->w_width > 2 ? 2 : wp->w_width,
2645 scr_row, wp->w_wincol, HL_ATTR(HLF_AT));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002646 screen_fill(scr_row, scr_row + 1,
2647 (int)wp->w_wincol + 2, (int)W_ENDCOL(wp),
2648 '@', ' ', HL_ATTR(HLF_AT));
2649 set_empty_rows(wp, srow);
2650 wp->w_botline = lnum;
2651 }
2652 else if (dy_flags & DY_LASTLINE) // 'display' has "lastline"
2653 {
Bram Moolenaarcee9c842022-04-09 12:40:13 +01002654 int start_col = (int)W_ENDCOL(wp) - 3;
2655
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002656 // Last line isn't finished: Display "@@@" at the end.
2657 screen_fill(W_WINROW(wp) + wp->w_height - 1,
2658 W_WINROW(wp) + wp->w_height,
Bram Moolenaarcee9c842022-04-09 12:40:13 +01002659 start_col < wp->w_wincol ? wp->w_wincol : start_col,
2660 (int)W_ENDCOL(wp),
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002661 '@', '@', HL_ATTR(HLF_AT));
2662 set_empty_rows(wp, srow);
2663 wp->w_botline = lnum;
2664 }
2665 else
2666 {
2667 win_draw_end(wp, '@', ' ', TRUE, srow, wp->w_height, HLF_AT);
2668 wp->w_botline = lnum;
2669 }
2670 }
2671 else
2672 {
2673 draw_vsep_win(wp, row);
2674 if (eof) // we hit the end of the file
2675 {
2676 wp->w_botline = buf->b_ml.ml_line_count + 1;
2677#ifdef FEAT_DIFF
2678 j = diff_check_fill(wp, wp->w_botline);
2679 if (j > 0 && !wp->w_botfill)
2680 {
2681 // Display filler lines at the end of the file.
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002682 if (char2cells(wp->w_fill_chars.diff) > 1)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002683 i = '-';
2684 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002685 i = wp->w_fill_chars.diff;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002686 if (row + j > wp->w_height)
2687 j = wp->w_height - row;
2688 win_draw_end(wp, i, i, TRUE, row, row + (int)j, HLF_DED);
2689 row += j;
2690 }
2691#endif
2692 }
2693 else if (dollar_vcol == -1)
2694 wp->w_botline = lnum;
2695
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002696 // Make sure the rest of the screen is blank.
2697 // write the "eob" character from 'fillchars' to rows that aren't part
2698 // of the file.
Bram Moolenaar1666ac92019-11-10 17:22:31 +01002699 if (WIN_IS_POPUP(wp))
2700 win_draw_end(wp, ' ', ' ', FALSE, row, wp->w_height, HLF_AT);
2701 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002702 win_draw_end(wp, wp->w_fill_chars.eob, ' ', FALSE,
2703 row, wp->w_height, HLF_EOB);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002704 }
2705
2706#ifdef SYN_TIME_LIMIT
Paul Ollis65745772022-06-05 16:55:54 +01002707 disable_regexp_timeout();
Bram Moolenaar6f0cf622022-06-19 12:27:45 +01002708 redrawtime_limit_set = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002709#endif
2710
2711 // Reset the type of redrawing required, the window has been updated.
2712 wp->w_redr_type = 0;
2713#ifdef FEAT_DIFF
2714 wp->w_old_topfill = wp->w_topfill;
2715 wp->w_old_botfill = wp->w_botfill;
2716#endif
2717
2718 if (dollar_vcol == -1)
2719 {
2720 // There is a trick with w_botline. If we invalidate it on each
2721 // change that might modify it, this will cause a lot of expensive
2722 // calls to plines() in update_topline() each time. Therefore the
2723 // value of w_botline is often approximated, and this value is used to
2724 // compute the value of w_topline. If the value of w_botline was
2725 // wrong, check that the value of w_topline is correct (cursor is on
2726 // the visible part of the text). If it's not, we need to redraw
2727 // again. Mostly this just means scrolling up a few lines, so it
2728 // doesn't look too bad. Only do this for the current window (where
2729 // changes are relevant).
2730 wp->w_valid |= VALID_BOTLINE;
2731 if (wp == curwin && wp->w_botline != old_botline && !recursive)
2732 {
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002733 win_T *wwp;
2734#if defined(FEAT_CONCEAL)
2735 linenr_T old_topline = wp->w_topline;
2736 int new_wcol = wp->w_wcol;
2737#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002738 recursive = TRUE;
2739 curwin->w_valid &= ~VALID_TOPLINE;
2740 update_topline(); // may invalidate w_botline again
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002741
2742#if defined(FEAT_CONCEAL)
2743 if (old_wcol != new_wcol && (wp->w_valid & (VALID_WCOL|VALID_WROW))
2744 != (VALID_WCOL|VALID_WROW))
2745 {
2746 // A win_line() call applied a fix to screen cursor column to
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002747 // accommodate concealment of cursor line, but in this call to
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002748 // update_topline() the cursor's row or column got invalidated.
2749 // If they are left invalid, setcursor() will recompute them
2750 // but there won't be any further win_line() call to re-fix the
2751 // column and the cursor will end up misplaced. So we call
2752 // cursor validation now and reapply the fix again (or call
2753 // win_line() to do it for us).
2754 validate_cursor();
2755 if (wp->w_wcol == old_wcol && wp->w_wrow == old_wrow
2756 && old_topline == wp->w_topline)
2757 wp->w_wcol = new_wcol;
2758 else
2759 redrawWinline(wp, wp->w_cursor.lnum);
2760 }
2761#endif
2762 // New redraw either due to updated topline or due to wcol fix.
2763 if (wp->w_redr_type != 0)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002764 {
2765 // Don't update for changes in buffer again.
2766 i = curbuf->b_mod_set;
2767 curbuf->b_mod_set = FALSE;
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002768 j = curbuf->b_mod_xlines;
2769 curbuf->b_mod_xlines = 0;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002770 win_update(curwin);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002771 curbuf->b_mod_set = i;
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002772 curbuf->b_mod_xlines = j;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002773 }
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002774 // Other windows might have w_redr_type raised in update_topline().
2775 must_redraw = 0;
2776 FOR_ALL_WINDOWS(wwp)
2777 if (wwp->w_redr_type > must_redraw)
2778 must_redraw = wwp->w_redr_type;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002779 recursive = FALSE;
2780 }
2781 }
2782
2783#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
2784 // restore got_int, unless CTRL-C was hit while redrawing
2785 if (!got_int)
2786 got_int = save_got_int;
2787#endif
2788}
2789
2790#if defined(FEAT_NETBEANS_INTG) || defined(FEAT_GUI)
2791/*
2792 * Prepare for updating one or more windows.
2793 * Caller must check for "updating_screen" already set to avoid recursiveness.
2794 */
2795 static void
2796update_prepare(void)
2797{
2798 cursor_off();
2799 updating_screen = TRUE;
2800#ifdef FEAT_GUI
2801 // Remove the cursor before starting to do anything, because scrolling may
2802 // make it difficult to redraw the text under it.
2803 if (gui.in_use)
2804 gui_undraw_cursor();
2805#endif
2806#ifdef FEAT_SEARCH_EXTRA
2807 start_search_hl();
2808#endif
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002809#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002810 // Update popup_mask if needed.
2811 may_update_popup_mask(must_redraw);
2812#endif
2813}
2814
2815/*
2816 * Finish updating one or more windows.
2817 */
2818 static void
2819update_finish(void)
2820{
2821 if (redraw_cmdline || redraw_mode)
2822 showmode();
2823
2824# ifdef FEAT_SEARCH_EXTRA
2825 end_search_hl();
2826# endif
2827
2828 after_updating_screen(TRUE);
2829
2830# ifdef FEAT_GUI
2831 // Redraw the cursor and update the scrollbars when all screen updating is
2832 // done.
2833 if (gui.in_use)
2834 {
2835 out_flush_cursor(FALSE, FALSE);
2836 gui_update_scrollbars(FALSE);
2837 }
2838# endif
2839}
2840#endif
2841
2842#if defined(FEAT_NETBEANS_INTG) || defined(PROTO)
2843 void
2844update_debug_sign(buf_T *buf, linenr_T lnum)
2845{
2846 win_T *wp;
2847 int doit = FALSE;
2848
2849# ifdef FEAT_FOLDING
2850 win_foldinfo.fi_level = 0;
2851# endif
2852
2853 // update/delete a specific sign
2854 redraw_buf_line_later(buf, lnum);
2855
2856 // check if it resulted in the need to redraw a window
2857 FOR_ALL_WINDOWS(wp)
2858 if (wp->w_redr_type != 0)
2859 doit = TRUE;
2860
2861 // Return when there is nothing to do, screen updating is already
2862 // happening (recursive call), messages on the screen or still starting up.
2863 if (!doit || updating_screen
Bram Moolenaar24959102022-05-07 20:01:16 +01002864 || State == MODE_ASKMORE || State == MODE_HITRETURN
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002865 || msg_scrolled
2866#ifdef FEAT_GUI
2867 || gui.starting
2868#endif
2869 || starting)
2870 return;
2871
2872 // update all windows that need updating
2873 update_prepare();
2874
2875 FOR_ALL_WINDOWS(wp)
2876 {
2877 if (wp->w_redr_type != 0)
2878 win_update(wp);
2879 if (wp->w_redr_status)
2880 win_redr_status(wp, FALSE);
2881 }
2882
2883 update_finish();
2884}
2885#endif
2886
2887#if defined(FEAT_GUI) || defined(PROTO)
2888/*
2889 * Update a single window, its status line and maybe the command line msg.
2890 * Used for the GUI scrollbar.
2891 */
2892 void
2893updateWindow(win_T *wp)
2894{
2895 // return if already busy updating
2896 if (updating_screen)
2897 return;
2898
2899 update_prepare();
2900
2901#ifdef FEAT_CLIPBOARD
2902 // When Visual area changed, may have to update selection.
2903 if (clip_star.available && clip_isautosel_star())
2904 clip_update_selection(&clip_star);
2905 if (clip_plus.available && clip_isautosel_plus())
2906 clip_update_selection(&clip_plus);
2907#endif
2908
2909 win_update(wp);
2910
2911 // When the screen was cleared redraw the tab pages line.
2912 if (redraw_tabline)
2913 draw_tabline();
2914
2915 if (wp->w_redr_status
2916# ifdef FEAT_CMDL_INFO
2917 || p_ru
2918# endif
2919# ifdef FEAT_STL_OPT
2920 || *p_stl != NUL || *wp->w_p_stl != NUL
2921# endif
2922 )
2923 win_redr_status(wp, FALSE);
2924
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002925#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002926 // Display popup windows on top of everything.
2927 update_popups(win_update);
2928#endif
2929
2930 update_finish();
2931}
2932#endif
2933
2934#if defined(FEAT_TERMRESPONSE) || defined(PROTO)
2935/*
2936 * Redraw as soon as possible. When the command line is not scrolled redraw
2937 * right away and restore what was on the command line.
2938 * Return a code indicating what happened.
2939 */
2940 int
2941redraw_asap(int type)
2942{
2943 int rows;
2944 int cols = screen_Columns;
2945 int r;
2946 int ret = 0;
2947 schar_T *screenline; // copy from ScreenLines[]
2948 sattr_T *screenattr; // copy from ScreenAttrs[]
2949 int i;
2950 u8char_T *screenlineUC = NULL; // copy from ScreenLinesUC[]
2951 u8char_T *screenlineC[MAX_MCO]; // copy from ScreenLinesC[][]
2952 schar_T *screenline2 = NULL; // copy from ScreenLines2[]
2953
2954 redraw_later(type);
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +01002955 if (msg_scrolled
2956 || (State != MODE_NORMAL && State != MODE_NORMAL_BUSY)
2957 || exiting
2958 || p_ch == 0)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002959 return ret;
2960
2961 // Allocate space to save the text displayed in the command line area.
2962 rows = screen_Rows - cmdline_row;
2963 screenline = LALLOC_MULT(schar_T, rows * cols);
2964 screenattr = LALLOC_MULT(sattr_T, rows * cols);
2965 if (screenline == NULL || screenattr == NULL)
2966 ret = 2;
2967 if (enc_utf8)
2968 {
2969 screenlineUC = LALLOC_MULT(u8char_T, rows * cols);
2970 if (screenlineUC == NULL)
2971 ret = 2;
2972 for (i = 0; i < p_mco; ++i)
2973 {
2974 screenlineC[i] = LALLOC_MULT(u8char_T, rows * cols);
2975 if (screenlineC[i] == NULL)
2976 ret = 2;
2977 }
2978 }
2979 if (enc_dbcs == DBCS_JPNU)
2980 {
2981 screenline2 = LALLOC_MULT(schar_T, rows * cols);
2982 if (screenline2 == NULL)
2983 ret = 2;
2984 }
2985
2986 if (ret != 2)
2987 {
2988 // Save the text displayed in the command line area.
2989 for (r = 0; r < rows; ++r)
2990 {
2991 mch_memmove(screenline + r * cols,
2992 ScreenLines + LineOffset[cmdline_row + r],
2993 (size_t)cols * sizeof(schar_T));
2994 mch_memmove(screenattr + r * cols,
2995 ScreenAttrs + LineOffset[cmdline_row + r],
2996 (size_t)cols * sizeof(sattr_T));
2997 if (enc_utf8)
2998 {
2999 mch_memmove(screenlineUC + r * cols,
3000 ScreenLinesUC + LineOffset[cmdline_row + r],
3001 (size_t)cols * sizeof(u8char_T));
3002 for (i = 0; i < p_mco; ++i)
3003 mch_memmove(screenlineC[i] + r * cols,
3004 ScreenLinesC[i] + LineOffset[cmdline_row + r],
3005 (size_t)cols * sizeof(u8char_T));
3006 }
3007 if (enc_dbcs == DBCS_JPNU)
3008 mch_memmove(screenline2 + r * cols,
3009 ScreenLines2 + LineOffset[cmdline_row + r],
3010 (size_t)cols * sizeof(schar_T));
3011 }
3012
3013 update_screen(0);
3014 ret = 3;
3015
3016 if (must_redraw == 0)
3017 {
3018 int off = (int)(current_ScreenLine - ScreenLines);
3019
3020 // Restore the text displayed in the command line area.
3021 for (r = 0; r < rows; ++r)
3022 {
3023 mch_memmove(current_ScreenLine,
3024 screenline + r * cols,
3025 (size_t)cols * sizeof(schar_T));
3026 mch_memmove(ScreenAttrs + off,
3027 screenattr + r * cols,
3028 (size_t)cols * sizeof(sattr_T));
3029 if (enc_utf8)
3030 {
3031 mch_memmove(ScreenLinesUC + off,
3032 screenlineUC + r * cols,
3033 (size_t)cols * sizeof(u8char_T));
3034 for (i = 0; i < p_mco; ++i)
3035 mch_memmove(ScreenLinesC[i] + off,
3036 screenlineC[i] + r * cols,
3037 (size_t)cols * sizeof(u8char_T));
3038 }
3039 if (enc_dbcs == DBCS_JPNU)
3040 mch_memmove(ScreenLines2 + off,
3041 screenline2 + r * cols,
3042 (size_t)cols * sizeof(schar_T));
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01003043 screen_line(curwin, cmdline_row + r, 0, cols, cols, 0);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003044 }
3045 ret = 4;
3046 }
3047 }
3048
3049 vim_free(screenline);
3050 vim_free(screenattr);
3051 if (enc_utf8)
3052 {
3053 vim_free(screenlineUC);
3054 for (i = 0; i < p_mco; ++i)
3055 vim_free(screenlineC[i]);
3056 }
3057 if (enc_dbcs == DBCS_JPNU)
3058 vim_free(screenline2);
3059
3060 // Show the intro message when appropriate.
3061 maybe_intro_message();
3062
3063 setcursor();
3064
3065 return ret;
3066}
3067#endif
3068
3069/*
3070 * Invoked after an asynchronous callback is called.
3071 * If an echo command was used the cursor needs to be put back where
3072 * it belongs. If highlighting was changed a redraw is needed.
3073 * If "call_update_screen" is FALSE don't call update_screen() when at the
3074 * command line.
Bram Moolenaare5050712021-12-09 10:51:05 +00003075 * If "redraw_message" is TRUE.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003076 */
3077 void
Bram Moolenaare5050712021-12-09 10:51:05 +00003078redraw_after_callback(int call_update_screen, int do_message)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003079{
3080 ++redrawing_for_callback;
3081
Bram Moolenaar24959102022-05-07 20:01:16 +01003082 if (State == MODE_HITRETURN || State == MODE_ASKMORE
3083 || State == MODE_SETWSIZE || State == MODE_EXTERNCMD
3084 || State == MODE_CONFIRM || exmode_active)
Bram Moolenaare5050712021-12-09 10:51:05 +00003085 {
3086 if (do_message)
3087 repeat_message();
3088 }
Bram Moolenaar24959102022-05-07 20:01:16 +01003089 else if (State & MODE_CMDLINE)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003090 {
Yegappan Lakshmanan3908ef52022-02-08 12:08:07 +00003091#ifdef FEAT_WILDMENU
3092 if (pum_visible())
3093 cmdline_pum_display();
3094#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003095 // Don't redraw when in prompt_for_number().
3096 if (cmdline_row > 0)
3097 {
3098 // Redrawing only works when the screen didn't scroll. Don't clear
3099 // wildmenu entries.
3100 if (msg_scrolled == 0
3101#ifdef FEAT_WILDMENU
3102 && wild_menu_showing == 0
3103#endif
3104 && call_update_screen)
3105 update_screen(0);
3106
3107 // Redraw in the same position, so that the user can continue
3108 // editing the command.
3109 redrawcmdline_ex(FALSE);
3110 }
3111 }
Bram Moolenaar24959102022-05-07 20:01:16 +01003112 else if (State & (MODE_NORMAL | MODE_INSERT | MODE_TERMINAL))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003113 {
zeertzjq3e559cd2022-03-27 19:26:55 +01003114 update_topline();
3115 validate_cursor();
3116
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003117 // keep the command line if possible
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003118 update_screen(UPD_VALID_NO_UPDATE);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003119 setcursor();
Bram Moolenaar9f284162021-04-22 21:39:30 +02003120
3121 if (msg_scrolled == 0)
3122 {
3123 // don't want a hit-enter prompt when something else is displayed
3124 msg_didany = FALSE;
3125 need_wait_return = FALSE;
3126 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003127 }
3128 cursor_on();
3129#ifdef FEAT_GUI
3130 if (gui.in_use && !gui_mch_is_blink_off())
3131 // Don't update the cursor when it is blinking and off to avoid
3132 // flicker.
3133 out_flush_cursor(FALSE, FALSE);
3134 else
3135#endif
3136 out_flush();
3137
3138 --redrawing_for_callback;
3139}
3140
3141/*
3142 * Redraw the current window later, with update_screen(type).
3143 * Set must_redraw only if not already set to a higher value.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003144 * E.g. if must_redraw is UPD_CLEAR, type UPD_NOT_VALID will do nothing.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003145 */
3146 void
3147redraw_later(int type)
3148{
3149 redraw_win_later(curwin, type);
3150}
3151
3152 void
3153redraw_win_later(
3154 win_T *wp,
3155 int type)
3156{
3157 if (!exiting && wp->w_redr_type < type)
3158 {
3159 wp->w_redr_type = type;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003160 if (type >= UPD_NOT_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003161 wp->w_lines_valid = 0;
3162 if (must_redraw < type) // must_redraw is the maximum of all windows
3163 must_redraw = type;
3164 }
3165}
3166
3167/*
3168 * Force a complete redraw later. Also resets the highlighting. To be used
3169 * after executing a shell command that messes up the screen.
3170 */
3171 void
3172redraw_later_clear(void)
3173{
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003174 redraw_all_later(UPD_CLEAR);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003175 reset_screen_attr();
3176}
3177
3178/*
3179 * Mark all windows to be redrawn later.
3180 */
3181 void
3182redraw_all_later(int type)
3183{
3184 win_T *wp;
3185
3186 FOR_ALL_WINDOWS(wp)
3187 redraw_win_later(wp, type);
3188 // This may be needed when switching tabs.
3189 if (must_redraw < type)
3190 must_redraw = type;
3191}
3192
3193/*
3194 * Mark all windows that are editing the current buffer to be updated later.
3195 */
3196 void
3197redraw_curbuf_later(int type)
3198{
3199 redraw_buf_later(curbuf, type);
3200}
3201
3202 void
3203redraw_buf_later(buf_T *buf, int type)
3204{
3205 win_T *wp;
3206
3207 FOR_ALL_WINDOWS(wp)
3208 {
3209 if (wp->w_buffer == buf)
3210 redraw_win_later(wp, type);
3211 }
Bram Moolenaare52e0c82020-02-28 22:20:10 +01003212#if defined(FEAT_TERMINAL) && defined(FEAT_PROP_POPUP)
3213 // terminal in popup window is not in list of windows
3214 if (curwin->w_buffer == buf)
3215 redraw_win_later(curwin, type);
3216#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003217}
3218
3219#if defined(FEAT_SIGNS) || defined(PROTO)
3220 void
3221redraw_buf_line_later(buf_T *buf, linenr_T lnum)
3222{
3223 win_T *wp;
3224
3225 FOR_ALL_WINDOWS(wp)
3226 if (wp->w_buffer == buf && lnum >= wp->w_topline
3227 && lnum < wp->w_botline)
3228 redrawWinline(wp, lnum);
3229}
3230#endif
3231
3232#if defined(FEAT_JOB_CHANNEL) || defined(PROTO)
3233 void
3234redraw_buf_and_status_later(buf_T *buf, int type)
3235{
3236 win_T *wp;
3237
3238#ifdef FEAT_WILDMENU
3239 if (wild_menu_showing != 0)
3240 // Don't redraw while the command line completion is displayed, it
3241 // would disappear.
3242 return;
3243#endif
3244 FOR_ALL_WINDOWS(wp)
3245 {
3246 if (wp->w_buffer == buf)
3247 {
3248 redraw_win_later(wp, type);
3249 wp->w_redr_status = TRUE;
3250 }
3251 }
3252}
3253#endif
3254
3255/*
3256 * mark all status lines for redraw; used after first :cd
3257 */
3258 void
3259status_redraw_all(void)
3260{
3261 win_T *wp;
3262
3263 FOR_ALL_WINDOWS(wp)
3264 if (wp->w_status_height)
3265 {
3266 wp->w_redr_status = TRUE;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003267 redraw_later(UPD_VALID);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003268 }
3269}
3270
3271/*
3272 * mark all status lines of the current buffer for redraw
3273 */
3274 void
3275status_redraw_curbuf(void)
3276{
3277 win_T *wp;
3278
3279 FOR_ALL_WINDOWS(wp)
3280 if (wp->w_status_height != 0 && wp->w_buffer == curbuf)
3281 {
3282 wp->w_redr_status = TRUE;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003283 redraw_later(UPD_VALID);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003284 }
3285}
3286
3287/*
3288 * Redraw all status lines that need to be redrawn.
3289 */
3290 void
3291redraw_statuslines(void)
3292{
3293 win_T *wp;
3294
3295 FOR_ALL_WINDOWS(wp)
3296 if (wp->w_redr_status)
3297 win_redr_status(wp, FALSE);
3298 if (redraw_tabline)
3299 draw_tabline();
3300}
3301
3302#if defined(FEAT_WILDMENU) || defined(PROTO)
3303/*
3304 * Redraw all status lines at the bottom of frame "frp".
3305 */
3306 void
3307win_redraw_last_status(frame_T *frp)
3308{
3309 if (frp->fr_layout == FR_LEAF)
3310 frp->fr_win->w_redr_status = TRUE;
3311 else if (frp->fr_layout == FR_ROW)
3312 {
3313 FOR_ALL_FRAMES(frp, frp->fr_child)
3314 win_redraw_last_status(frp);
3315 }
3316 else // frp->fr_layout == FR_COL
3317 {
3318 frp = frp->fr_child;
3319 while (frp->fr_next != NULL)
3320 frp = frp->fr_next;
3321 win_redraw_last_status(frp);
3322 }
3323}
3324#endif
3325
3326/*
3327 * Changed something in the current window, at buffer line "lnum", that
3328 * requires that line and possibly other lines to be redrawn.
3329 * Used when entering/leaving Insert mode with the cursor on a folded line.
3330 * Used to remove the "$" from a change command.
3331 * Note that when also inserting/deleting lines w_redraw_top and w_redraw_bot
3332 * may become invalid and the whole window will have to be redrawn.
3333 */
3334 void
3335redrawWinline(
3336 win_T *wp,
3337 linenr_T lnum)
3338{
3339 if (wp->w_redraw_top == 0 || wp->w_redraw_top > lnum)
3340 wp->w_redraw_top = lnum;
3341 if (wp->w_redraw_bot == 0 || wp->w_redraw_bot < lnum)
3342 wp->w_redraw_bot = lnum;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003343 redraw_win_later(wp, UPD_VALID);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003344}