blob: f56ef91f290f5c4cfb4ae56a46c199aa16d34b61 [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
31 * call redraw_later(VALID) to have the window displayed by update_screen()
32 * 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
43 * settings), must call redraw_later(NOT_VALID) to have the whole window
44 * redisplayed by update_screen() later.
45 *
46 * Commands that change how a buffer is displayed (e.g., setting 'tabstop')
47 * must call redraw_curbuf_later(NOT_VALID) to have all the windows for the
48 * buffer redisplayed by update_screen() later.
49 *
50 * Commands that change highlighting and possibly cause a scroll too must call
51 * redraw_later(SOME_VALID) to update the whole window but still use scrolling
52 * to avoid redrawing everything. But the length of displayed lines must not
53 * change, use NOT_VALID then.
54 *
55 * Commands that move the window position must call redraw_later(NOT_VALID).
56 * TODO: should minimize redrawing by scrolling when possible.
57 *
58 * Commands that change everything (e.g., resizing the screen) must call
59 * redraw_all_later(NOT_VALID) or redraw_all_later(CLEAR).
60 *
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
102 if (type == VALID_NO_UPDATE)
103 {
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[].
137 if (curwin->w_lines_valid == 0 && type < NOT_VALID
138#ifdef FEAT_TERMINAL
139 && !term_do_update_window(curwin)
140#endif
141 )
142 type = NOT_VALID;
143
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;
150 if (type > INVERTED_ALL)
151 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
174 type = CLEAR;
175 else if (type != CLEAR)
176 {
177 check_for_delay(FALSE);
178 if (screen_ins_lines(0, 0, msg_scrolled, (int)Rows, 0, NULL)
179 == FAIL)
180 type = CLEAR;
181 FOR_ALL_WINDOWS(wp)
182 {
183 if (wp->w_winrow < msg_scrolled)
184 {
185 if (W_WINROW(wp) + wp->w_height > msg_scrolled
186 && wp->w_redr_type < REDRAW_TOP
187 && 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);
191 wp->w_redr_type = REDRAW_TOP;
192 }
193 else
194 {
195 wp->w_redr_type = NOT_VALID;
196 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
217 if (type == CLEAR) // first clear screen
218 {
219 screenclear(); // will reset clear_cmdline
220 type = NOT_VALID;
221 // 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.
231 if (curwin->w_redr_type < NOT_VALID
232 && curwin->w_nrwidth != ((curwin->w_p_nu || curwin->w_p_rnu)
233 ? number_width(curwin) : 0))
234 curwin->w_redr_type = NOT_VALID;
235#endif
236
237 // Only start redrawing if there is really something to do.
238 if (type == INVERTED)
239 update_curswant();
240 if (curwin->w_redr_type < type
241 && !((type == VALID
242 && 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)
248 || (type == INVERTED
249 && 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.
258 if (redraw_tabline || type >= NOT_VALID)
259 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
345 after_updating_screen(TRUE);
346
347 // Clear or redraw the command line. Done last, because scrolling may
348 // mess up the command line.
349 if (clear_cmdline || redraw_cmdline || redraw_mode)
350 showmode();
351
352 if (no_update)
353 --no_win_do_lines_ins;
354
355 // May put up an introductory message when not editing a file
356 if (!did_intro)
357 maybe_intro_message();
358 did_intro = TRUE;
359
360#ifdef FEAT_GUI
361 // Redraw the cursor and update the scrollbars when all screen updating is
362 // done.
363 if (gui.in_use)
364 {
365 if (did_undraw && !gui_mch_is_blink_off())
366 {
367 mch_disable_flush();
368 out_flush(); // required before updating the cursor
369 mch_enable_flush();
370
371 // Put the GUI position where the cursor was, gui_update_cursor()
372 // uses that.
373 gui.col = gui_cursor_col;
374 gui.row = gui_cursor_row;
375 gui.col = mb_fix_col(gui.col, gui.row);
376 gui_update_cursor(FALSE, FALSE);
377 gui_may_flush();
378 screen_cur_col = gui.col;
379 screen_cur_row = gui.row;
380 }
381 else
382 out_flush();
383 gui_update_scrollbars(FALSE);
384 }
385#endif
386 return OK;
387}
388
389/*
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200390 * Return the row for drawing the statusline and the ruler of window "wp".
391 */
Bram Moolenaar49c51b82021-04-01 16:16:18 +0200392 int
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200393statusline_row(win_T *wp)
394{
395#if defined(FEAT_PROP_POPUP)
396 // If the window is really zero height the winbar isn't displayed.
397 if (wp->w_frame->fr_height == wp->w_status_height && !popup_is_popup(wp))
398 return wp->w_winrow;
399#endif
400 return W_WINROW(wp) + wp->w_height;
401}
402
403/*
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200404 * Redraw the status line of window wp.
405 *
406 * If inversion is possible we use it. Else '=' characters are used.
407 * If "ignore_pum" is TRUE, also redraw statusline when the popup menu is
408 * displayed.
409 */
Bram Moolenaarbdff0122020-04-05 18:56:05 +0200410 static void
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200411win_redr_status(win_T *wp, int ignore_pum UNUSED)
412{
413 int row;
414 char_u *p;
415 int len;
416 int fillchar;
417 int attr;
418 int this_ru_col;
419 static int busy = FALSE;
420
421 // It's possible to get here recursively when 'statusline' (indirectly)
422 // invokes ":redrawstatus". Simply ignore the call then.
423 if (busy)
424 return;
425 busy = TRUE;
426
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200427 row = statusline_row(wp);
428
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200429 wp->w_redr_status = FALSE;
430 if (wp->w_status_height == 0)
431 {
432 // no status line, can only be last window
433 redraw_cmdline = TRUE;
434 }
435 else if (!redrawing()
436 // don't update status line when popup menu is visible and may be
437 // drawn over it, unless it will be redrawn later
438 || (!ignore_pum && pum_visible()))
439 {
440 // Don't redraw right now, do it later.
441 wp->w_redr_status = TRUE;
442 }
443#ifdef FEAT_STL_OPT
444 else if (*p_stl != NUL || *wp->w_p_stl != NUL)
445 {
446 // redraw custom status line
447 redraw_custom_statusline(wp);
448 }
449#endif
450 else
451 {
452 fillchar = fillchar_status(&attr, wp);
453
454 get_trans_bufname(wp->w_buffer);
455 p = NameBuff;
456 len = (int)STRLEN(p);
457
458 if (bt_help(wp->w_buffer)
459#ifdef FEAT_QUICKFIX
460 || wp->w_p_pvw
461#endif
462 || bufIsChanged(wp->w_buffer)
463 || wp->w_buffer->b_p_ro)
464 *(p + len++) = ' ';
465 if (bt_help(wp->w_buffer))
466 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100467 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[Help]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200468 len += (int)STRLEN(p + len);
469 }
470#ifdef FEAT_QUICKFIX
471 if (wp->w_p_pvw)
472 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100473 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[Preview]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200474 len += (int)STRLEN(p + len);
475 }
476#endif
477 if (bufIsChanged(wp->w_buffer)
478#ifdef FEAT_TERMINAL
479 && !bt_terminal(wp->w_buffer)
480#endif
481 )
482 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100483 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", "[+]");
484 len += (int)STRLEN(p + len);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200485 }
486 if (wp->w_buffer->b_p_ro)
487 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100488 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[RO]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200489 len += (int)STRLEN(p + len);
490 }
491
492 this_ru_col = ru_col - (Columns - wp->w_width);
493 if (this_ru_col < (wp->w_width + 1) / 2)
494 this_ru_col = (wp->w_width + 1) / 2;
495 if (this_ru_col <= 1)
496 {
497 p = (char_u *)"<"; // No room for file name!
498 len = 1;
499 }
500 else if (has_mbyte)
501 {
502 int clen = 0, i;
503
504 // Count total number of display cells.
505 clen = mb_string2cells(p, -1);
506
507 // Find first character that will fit.
508 // Going from start to end is much faster for DBCS.
509 for (i = 0; p[i] != NUL && clen >= this_ru_col - 1;
510 i += (*mb_ptr2len)(p + i))
511 clen -= (*mb_ptr2cells)(p + i);
512 len = clen;
513 if (i > 0)
514 {
515 p = p + i - 1;
516 *p = '<';
517 ++len;
518 }
519
520 }
521 else if (len > this_ru_col - 1)
522 {
523 p += len - (this_ru_col - 1);
524 *p = '<';
525 len = this_ru_col - 1;
526 }
527
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200528 screen_puts(p, row, wp->w_wincol, attr);
529 screen_fill(row, row + 1, len + wp->w_wincol,
530 this_ru_col + wp->w_wincol, fillchar, fillchar, attr);
531
532 if (get_keymap_str(wp, (char_u *)"<%s>", NameBuff, MAXPATHL)
533 && (int)(this_ru_col - len) > (int)(STRLEN(NameBuff) + 1))
534 screen_puts(NameBuff, row, (int)(this_ru_col - STRLEN(NameBuff)
535 - 1 + wp->w_wincol), attr);
536
537#ifdef FEAT_CMDL_INFO
538 win_redr_ruler(wp, TRUE, ignore_pum);
539#endif
540 }
541
542 /*
543 * May need to draw the character below the vertical separator.
544 */
545 if (wp->w_vsep_width != 0 && wp->w_status_height != 0 && redrawing())
546 {
547 if (stl_connected(wp))
548 fillchar = fillchar_status(&attr, wp);
549 else
550 fillchar = fillchar_vsep(&attr);
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200551 screen_putchar(fillchar, row, W_ENDCOL(wp), attr);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200552 }
553 busy = FALSE;
554}
555
556#ifdef FEAT_STL_OPT
557/*
558 * Redraw the status line according to 'statusline' and take care of any
559 * errors encountered.
560 */
561 static void
562redraw_custom_statusline(win_T *wp)
563{
564 static int entered = FALSE;
565 int saved_did_emsg = did_emsg;
566
567 // When called recursively return. This can happen when the statusline
568 // contains an expression that triggers a redraw.
569 if (entered)
570 return;
571 entered = TRUE;
572
573 did_emsg = FALSE;
574 win_redr_custom(wp, FALSE);
575 if (did_emsg)
576 {
577 // When there is an error disable the statusline, otherwise the
578 // display is messed up with errors and a redraw triggers the problem
579 // again and again.
580 set_string_option_direct((char_u *)"statusline", -1,
581 (char_u *)"", OPT_FREE | (*wp->w_p_stl != NUL
582 ? OPT_LOCAL : OPT_GLOBAL), SID_ERROR);
583 }
584 did_emsg |= saved_did_emsg;
585 entered = FALSE;
586}
587#endif
588
589/*
590 * Show current status info in ruler and various other places
591 * If always is FALSE, only show ruler if position has changed.
592 */
593 void
594showruler(int always)
595{
596 if (!always && !redrawing())
597 return;
598 if (pum_visible())
599 {
600 // Don't redraw right now, do it later.
601 curwin->w_redr_status = TRUE;
602 return;
603 }
604#if defined(FEAT_STL_OPT)
605 if ((*p_stl != NUL || *curwin->w_p_stl != NUL) && curwin->w_status_height)
606 redraw_custom_statusline(curwin);
607 else
608#endif
609#ifdef FEAT_CMDL_INFO
610 win_redr_ruler(curwin, always, FALSE);
611#endif
612
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200613 if (need_maketitle
Bram Moolenaar651fca82021-11-29 20:39:38 +0000614#ifdef FEAT_STL_OPT
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200615 || (p_icon && (stl_syntax & STL_IN_ICON))
616 || (p_title && (stl_syntax & STL_IN_TITLE))
Bram Moolenaar651fca82021-11-29 20:39:38 +0000617#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200618 )
619 maketitle();
Bram Moolenaar651fca82021-11-29 20:39:38 +0000620
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200621 // Redraw the tab pages line if needed.
622 if (redraw_tabline)
623 draw_tabline();
624}
625
626#if defined(FEAT_CMDL_INFO) || defined(PROTO)
627 void
628win_redr_ruler(win_T *wp, int always, int ignore_pum)
629{
630#define RULER_BUF_LEN 70
631 char_u buffer[RULER_BUF_LEN];
632 int row;
633 int fillchar;
634 int attr;
635 int empty_line = FALSE;
636 colnr_T virtcol;
637 int i;
638 size_t len;
639 int o;
640 int this_ru_col;
641 int off = 0;
642 int width;
643
644 // If 'ruler' off or redrawing disabled, don't do anything
645 if (!p_ru)
646 return;
647
648 /*
649 * Check if cursor.lnum is valid, since win_redr_ruler() may be called
650 * after deleting lines, before cursor.lnum is corrected.
651 */
652 if (wp->w_cursor.lnum > wp->w_buffer->b_ml.ml_line_count)
653 return;
654
655 // Don't draw the ruler while doing insert-completion, it might overwrite
656 // the (long) mode message.
657 if (wp == lastwin && lastwin->w_status_height == 0)
658 if (edit_submode != NULL)
659 return;
660 // Don't draw the ruler when the popup menu is visible, it may overlap.
661 // Except when the popup menu will be redrawn anyway.
662 if (!ignore_pum && pum_visible())
663 return;
664
665#ifdef FEAT_STL_OPT
666 if (*p_ruf)
667 {
Bram Moolenaar53989552019-12-23 22:59:18 +0100668 int called_emsg_before = called_emsg;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200669
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200670 win_redr_custom(wp, TRUE);
Bram Moolenaar53989552019-12-23 22:59:18 +0100671 if (called_emsg > called_emsg_before)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200672 set_string_option_direct((char_u *)"rulerformat", -1,
673 (char_u *)"", OPT_FREE, SID_ERROR);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200674 return;
675 }
676#endif
677
678 /*
679 * Check if not in Insert mode and the line is empty (will show "0-1").
680 */
681 if (!(State & INSERT)
682 && *ml_get_buf(wp->w_buffer, wp->w_cursor.lnum, FALSE) == NUL)
683 empty_line = TRUE;
684
685 /*
686 * Only draw the ruler when something changed.
687 */
688 validate_virtcol_win(wp);
689 if ( redraw_cmdline
690 || always
691 || wp->w_cursor.lnum != wp->w_ru_cursor.lnum
692 || wp->w_cursor.col != wp->w_ru_cursor.col
693 || wp->w_virtcol != wp->w_ru_virtcol
694 || wp->w_cursor.coladd != wp->w_ru_cursor.coladd
695 || wp->w_topline != wp->w_ru_topline
696 || wp->w_buffer->b_ml.ml_line_count != wp->w_ru_line_count
697#ifdef FEAT_DIFF
698 || wp->w_topfill != wp->w_ru_topfill
699#endif
700 || empty_line != wp->w_ru_empty)
701 {
702 cursor_off();
703 if (wp->w_status_height)
704 {
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200705 row = statusline_row(wp);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200706 fillchar = fillchar_status(&attr, wp);
707 off = wp->w_wincol;
708 width = wp->w_width;
709 }
710 else
711 {
712 row = Rows - 1;
713 fillchar = ' ';
714 attr = 0;
715 width = Columns;
716 off = 0;
717 }
718
719 // In list mode virtcol needs to be recomputed
720 virtcol = wp->w_virtcol;
Bram Moolenaareed9d462021-02-15 20:38:25 +0100721 if (wp->w_p_list && wp->w_lcs_chars.tab1 == NUL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200722 {
723 wp->w_p_list = FALSE;
724 getvvcol(wp, &wp->w_cursor, NULL, &virtcol, NULL);
725 wp->w_p_list = TRUE;
726 }
727
728 /*
729 * Some sprintfs return the length, some return a pointer.
730 * To avoid portability problems we use strlen() here.
731 */
732 vim_snprintf((char *)buffer, RULER_BUF_LEN, "%ld,",
733 (wp->w_buffer->b_ml.ml_flags & ML_EMPTY)
734 ? 0L
735 : (long)(wp->w_cursor.lnum));
736 len = STRLEN(buffer);
737 col_print(buffer + len, RULER_BUF_LEN - len,
738 empty_line ? 0 : (int)wp->w_cursor.col + 1,
739 (int)virtcol + 1);
740
741 /*
742 * Add a "50%" if there is room for it.
743 * On the last line, don't print in the last column (scrolls the
744 * screen up on some terminals).
745 */
746 i = (int)STRLEN(buffer);
747 get_rel_pos(wp, buffer + i + 1, RULER_BUF_LEN - i - 1);
748 o = i + vim_strsize(buffer + i + 1);
749 if (wp->w_status_height == 0) // can't use last char of screen
750 ++o;
751 this_ru_col = ru_col - (Columns - width);
752 if (this_ru_col < 0)
753 this_ru_col = 0;
754 // Never use more than half the window/screen width, leave the other
755 // half for the filename.
756 if (this_ru_col < (width + 1) / 2)
757 this_ru_col = (width + 1) / 2;
758 if (this_ru_col + o < width)
759 {
760 // need at least 3 chars left for get_rel_pos() + NUL
761 while (this_ru_col + o < width && RULER_BUF_LEN > i + 4)
762 {
763 if (has_mbyte)
764 i += (*mb_char2bytes)(fillchar, buffer + i);
765 else
766 buffer[i++] = fillchar;
767 ++o;
768 }
769 get_rel_pos(wp, buffer + i, RULER_BUF_LEN - i);
770 }
771 // Truncate at window boundary.
772 if (has_mbyte)
773 {
774 o = 0;
775 for (i = 0; buffer[i] != NUL; i += (*mb_ptr2len)(buffer + i))
776 {
777 o += (*mb_ptr2cells)(buffer + i);
778 if (this_ru_col + o > width)
779 {
780 buffer[i] = NUL;
781 break;
782 }
783 }
784 }
785 else if (this_ru_col + (int)STRLEN(buffer) > width)
786 buffer[width - this_ru_col] = NUL;
787
788 screen_puts(buffer, row, this_ru_col + off, attr);
789 i = redraw_cmdline;
790 screen_fill(row, row + 1,
791 this_ru_col + off + (int)STRLEN(buffer),
792 (int)(off + width),
793 fillchar, fillchar, attr);
794 // don't redraw the cmdline because of showing the ruler
795 redraw_cmdline = i;
796 wp->w_ru_cursor = wp->w_cursor;
797 wp->w_ru_virtcol = wp->w_virtcol;
798 wp->w_ru_empty = empty_line;
799 wp->w_ru_topline = wp->w_topline;
800 wp->w_ru_line_count = wp->w_buffer->b_ml.ml_line_count;
801#ifdef FEAT_DIFF
802 wp->w_ru_topfill = wp->w_topfill;
803#endif
804 }
805}
806#endif
807
808/*
809 * To be called when "updating_screen" was set before and now the postponed
810 * side effects may take place.
811 */
812 void
813after_updating_screen(int may_resize_shell UNUSED)
814{
815 updating_screen = FALSE;
816#ifdef FEAT_GUI
817 if (may_resize_shell)
818 gui_may_resize_shell();
819#endif
820#ifdef FEAT_TERMINAL
821 term_check_channel_closed_recently();
822#endif
823
824#ifdef HAVE_DROP_FILE
825 // If handle_drop() was called while updating_screen was TRUE need to
826 // handle the drop now.
827 handle_any_postponed_drop();
828#endif
829}
830
831/*
832 * Update all windows that are editing the current buffer.
833 */
834 void
835update_curbuf(int type)
836{
837 redraw_curbuf_later(type);
838 update_screen(type);
839}
840
841#if defined(FEAT_MENU) || defined(FEAT_FOLDING)
842/*
843 * Copy "text" to ScreenLines using "attr".
844 * Returns the next screen column.
845 */
846 static int
847text_to_screenline(win_T *wp, char_u *text, int col)
848{
849 int off = (int)(current_ScreenLine - ScreenLines);
850
851 if (has_mbyte)
852 {
853 int cells;
854 int u8c, u8cc[MAX_MCO];
855 int i;
856 int idx;
857 int c_len;
858 char_u *p;
859# ifdef FEAT_ARABIC
860 int prev_c = 0; // previous Arabic character
861 int prev_c1 = 0; // first composing char for prev_c
862# endif
863
864# ifdef FEAT_RIGHTLEFT
865 if (wp->w_p_rl)
866 idx = off;
867 else
868# endif
869 idx = off + col;
870
871 // Store multibyte characters in ScreenLines[] et al. correctly.
872 for (p = text; *p != NUL; )
873 {
874 cells = (*mb_ptr2cells)(p);
875 c_len = (*mb_ptr2len)(p);
876 if (col + cells > wp->w_width
877# ifdef FEAT_RIGHTLEFT
878 - (wp->w_p_rl ? col : 0)
879# endif
880 )
881 break;
882 ScreenLines[idx] = *p;
883 if (enc_utf8)
884 {
885 u8c = utfc_ptr2char(p, u8cc);
886 if (*p < 0x80 && u8cc[0] == 0)
887 {
888 ScreenLinesUC[idx] = 0;
889#ifdef FEAT_ARABIC
890 prev_c = u8c;
891#endif
892 }
893 else
894 {
895#ifdef FEAT_ARABIC
896 if (p_arshape && !p_tbidi && ARABIC_CHAR(u8c))
897 {
898 // Do Arabic shaping.
899 int pc, pc1, nc;
900 int pcc[MAX_MCO];
901 int firstbyte = *p;
902
903 // The idea of what is the previous and next
904 // character depends on 'rightleft'.
905 if (wp->w_p_rl)
906 {
907 pc = prev_c;
908 pc1 = prev_c1;
909 nc = utf_ptr2char(p + c_len);
910 prev_c1 = u8cc[0];
911 }
912 else
913 {
914 pc = utfc_ptr2char(p + c_len, pcc);
915 nc = prev_c;
916 pc1 = pcc[0];
917 }
918 prev_c = u8c;
919
920 u8c = arabic_shape(u8c, &firstbyte, &u8cc[0],
921 pc, pc1, nc);
922 ScreenLines[idx] = firstbyte;
923 }
924 else
925 prev_c = u8c;
926#endif
927 // Non-BMP character: display as ? or fullwidth ?.
928 ScreenLinesUC[idx] = u8c;
929 for (i = 0; i < Screen_mco; ++i)
930 {
931 ScreenLinesC[i][idx] = u8cc[i];
932 if (u8cc[i] == 0)
933 break;
934 }
935 }
936 if (cells > 1)
937 ScreenLines[idx + 1] = 0;
938 }
939 else if (enc_dbcs == DBCS_JPNU && *p == 0x8e)
940 // double-byte single width character
941 ScreenLines2[idx] = p[1];
942 else if (cells > 1)
943 // double-width character
944 ScreenLines[idx + 1] = p[1];
945 col += cells;
946 idx += cells;
947 p += c_len;
948 }
949 }
950 else
951 {
952 int len = (int)STRLEN(text);
953
954 if (len > wp->w_width - col)
955 len = wp->w_width - col;
956 if (len > 0)
957 {
958#ifdef FEAT_RIGHTLEFT
959 if (wp->w_p_rl)
960 mch_memmove(current_ScreenLine, text, len);
961 else
962#endif
963 mch_memmove(current_ScreenLine + col, text, len);
964 col += len;
965 }
966 }
967 return col;
968}
969#endif
970
971#ifdef FEAT_MENU
972/*
973 * Draw the window toolbar.
974 */
975 static void
976redraw_win_toolbar(win_T *wp)
977{
978 vimmenu_T *menu;
979 int item_idx = 0;
980 int item_count = 0;
981 int col = 0;
982 int next_col;
983 int off = (int)(current_ScreenLine - ScreenLines);
984 int fill_attr = syn_name2attr((char_u *)"ToolbarLine");
985 int button_attr = syn_name2attr((char_u *)"ToolbarButton");
986
987 vim_free(wp->w_winbar_items);
Bram Moolenaar00d253e2020-04-06 22:13:01 +0200988 FOR_ALL_CHILD_MENUS(wp->w_winbar, menu)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200989 ++item_count;
990 wp->w_winbar_items = ALLOC_CLEAR_MULT(winbar_item_T, item_count + 1);
991
992 // TODO: use fewer spaces if there is not enough room
993 for (menu = wp->w_winbar->children;
994 menu != NULL && col < wp->w_width; menu = menu->next)
995 {
996 space_to_screenline(off + col, fill_attr);
997 if (++col >= wp->w_width)
998 break;
999 if (col > 1)
1000 {
1001 space_to_screenline(off + col, fill_attr);
1002 if (++col >= wp->w_width)
1003 break;
1004 }
1005
1006 wp->w_winbar_items[item_idx].wb_startcol = col;
1007 space_to_screenline(off + col, button_attr);
1008 if (++col >= wp->w_width)
1009 break;
1010
1011 next_col = text_to_screenline(wp, menu->name, col);
1012 while (col < next_col)
1013 {
1014 ScreenAttrs[off + col] = button_attr;
1015 ++col;
1016 }
1017 wp->w_winbar_items[item_idx].wb_endcol = col;
1018 wp->w_winbar_items[item_idx].wb_menu = menu;
1019 ++item_idx;
1020
1021 if (col >= wp->w_width)
1022 break;
1023 space_to_screenline(off + col, button_attr);
1024 ++col;
1025 }
1026 while (col < wp->w_width)
1027 {
1028 space_to_screenline(off + col, fill_attr);
1029 ++col;
1030 }
1031 wp->w_winbar_items[item_idx].wb_menu = NULL; // end marker
1032
1033 screen_line(wp->w_winrow, wp->w_wincol, (int)wp->w_width,
1034 (int)wp->w_width, 0);
1035}
1036#endif
1037
1038#if defined(FEAT_FOLDING) || defined(PROTO)
1039/*
1040 * Copy "buf[len]" to ScreenLines["off"] and set attributes to "attr".
1041 */
1042 static void
1043copy_text_attr(
1044 int off,
1045 char_u *buf,
1046 int len,
1047 int attr)
1048{
1049 int i;
1050
1051 mch_memmove(ScreenLines + off, buf, (size_t)len);
1052 if (enc_utf8)
1053 vim_memset(ScreenLinesUC + off, 0, sizeof(u8char_T) * (size_t)len);
1054 for (i = 0; i < len; ++i)
1055 ScreenAttrs[off + i] = attr;
1056}
1057
1058/*
1059 * Display one folded line.
1060 */
1061 static void
1062fold_line(
1063 win_T *wp,
1064 long fold_count,
1065 foldinfo_T *foldinfo,
1066 linenr_T lnum,
1067 int row)
1068{
Bram Moolenaar4fa11752021-03-03 13:26:02 +01001069 // Max value of 'foldcolumn' is 12 and maximum number of bytes in a
1070 // multi-byte character is MAX_MCO.
1071 char_u buf[MAX_MCO * 12 + 1];
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001072 pos_T *top, *bot;
1073 linenr_T lnume = lnum + fold_count - 1;
1074 int len;
1075 char_u *text;
1076 int fdc;
1077 int col;
1078 int txtcol;
1079 int off = (int)(current_ScreenLine - ScreenLines);
1080 int ri;
1081
1082 // Build the fold line:
1083 // 1. Add the cmdwin_type for the command-line window
1084 // 2. Add the 'foldcolumn'
1085 // 3. Add the 'number' or 'relativenumber' column
1086 // 4. Compose the text
1087 // 5. Add the text
1088 // 6. set highlighting for the Visual area an other text
1089 col = 0;
1090
1091 // 1. Add the cmdwin_type for the command-line window
1092 // Ignores 'rightleft', this window is never right-left.
1093#ifdef FEAT_CMDWIN
1094 if (cmdwin_type != 0 && wp == curwin)
1095 {
1096 ScreenLines[off] = cmdwin_type;
1097 ScreenAttrs[off] = HL_ATTR(HLF_AT);
1098 if (enc_utf8)
1099 ScreenLinesUC[off] = 0;
1100 ++col;
1101 }
1102#endif
1103
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001104#ifdef FEAT_RIGHTLEFT
1105# define RL_MEMSET(p, v, l) \
1106 do { \
1107 if (wp->w_p_rl) \
1108 for (ri = 0; ri < l; ++ri) \
1109 ScreenAttrs[off + (wp->w_width - (p) - (l)) + ri] = v; \
1110 else \
1111 for (ri = 0; ri < l; ++ri) \
1112 ScreenAttrs[off + (p) + ri] = v; \
1113 } while (0)
1114#else
1115# define RL_MEMSET(p, v, l) \
1116 do { \
1117 for (ri = 0; ri < l; ++ri) \
1118 ScreenAttrs[off + (p) + ri] = v; \
1119 } while (0)
1120#endif
1121
Bram Moolenaar4fa11752021-03-03 13:26:02 +01001122 // 2. Add the 'foldcolumn'
1123 // Reduce the width when there is not enough space.
1124 fdc = compute_foldcolumn(wp, col);
1125 if (fdc > 0)
1126 {
1127 char_u *p;
1128 int i;
1129 int idx;
1130
1131 fill_foldcolumn(buf, wp, TRUE, lnum);
1132 p = buf;
1133 for (i = 0; i < fdc; i++)
1134 {
1135 int ch;
1136
1137 if (has_mbyte)
1138 ch = mb_ptr2char_adv(&p);
1139 else
1140 ch = *p++;
1141#ifdef FEAT_RIGHTLEFT
1142 if (wp->w_p_rl)
1143 idx = off + wp->w_width - i - 1 - col;
1144 else
1145#endif
1146 idx = off + col + i;
1147 if (enc_utf8)
1148 {
1149 if (ch >= 0x80)
1150 {
1151 ScreenLinesUC[idx] = ch;
1152 ScreenLinesC[0][idx] = 0;
1153 ScreenLines[idx] = 0x80;
1154 }
1155 else
1156 {
1157 ScreenLines[idx] = ch;
1158 ScreenLinesUC[idx] = 0;
1159 }
1160 }
1161 else
1162 ScreenLines[idx] = ch;
1163 }
1164
1165 RL_MEMSET(col, HL_ATTR(HLF_FC), fdc);
1166 col += fdc;
1167 }
1168
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001169 // Set all attributes of the 'number' or 'relativenumber' column and the
1170 // text
1171 RL_MEMSET(col, HL_ATTR(HLF_FL), wp->w_width - col);
1172
1173#ifdef FEAT_SIGNS
1174 // If signs are being displayed, add two spaces.
1175 if (signcolumn_on(wp))
1176 {
1177 len = wp->w_width - col;
1178 if (len > 0)
1179 {
1180 if (len > 2)
1181 len = 2;
1182# ifdef FEAT_RIGHTLEFT
1183 if (wp->w_p_rl)
1184 // the line number isn't reversed
1185 copy_text_attr(off + wp->w_width - len - col,
1186 (char_u *)" ", len, HL_ATTR(HLF_FL));
1187 else
1188# endif
1189 copy_text_attr(off + col, (char_u *)" ", len, HL_ATTR(HLF_FL));
1190 col += len;
1191 }
1192 }
1193#endif
1194
1195 // 3. Add the 'number' or 'relativenumber' column
1196 if (wp->w_p_nu || wp->w_p_rnu)
1197 {
1198 len = wp->w_width - col;
1199 if (len > 0)
1200 {
1201 int w = number_width(wp);
1202 long num;
1203 char *fmt = "%*ld ";
1204
1205 if (len > w + 1)
1206 len = w + 1;
1207
1208 if (wp->w_p_nu && !wp->w_p_rnu)
1209 // 'number' + 'norelativenumber'
1210 num = (long)lnum;
1211 else
1212 {
1213 // 'relativenumber', don't use negative numbers
1214 num = labs((long)get_cursor_rel_lnum(wp, lnum));
1215 if (num == 0 && wp->w_p_nu && wp->w_p_rnu)
1216 {
1217 // 'number' + 'relativenumber': cursor line shows absolute
1218 // line number
1219 num = lnum;
1220 fmt = "%-*ld ";
1221 }
1222 }
1223
1224 sprintf((char *)buf, fmt, w, num);
1225#ifdef FEAT_RIGHTLEFT
1226 if (wp->w_p_rl)
1227 // the line number isn't reversed
1228 copy_text_attr(off + wp->w_width - len - col, buf, len,
1229 HL_ATTR(HLF_FL));
1230 else
1231#endif
1232 copy_text_attr(off + col, buf, len, HL_ATTR(HLF_FL));
1233 col += len;
1234 }
1235 }
1236
1237 // 4. Compose the folded-line string with 'foldtext', if set.
1238 text = get_foldtext(wp, lnum, lnume, foldinfo, buf);
1239
1240 txtcol = col; // remember where text starts
1241
1242 // 5. move the text to current_ScreenLine. Fill up with "fill_fold".
1243 // Right-left text is put in columns 0 - number-col, normal text is put
1244 // in columns number-col - window-width.
1245 col = text_to_screenline(wp, text, col);
1246
1247 // Fill the rest of the line with the fold filler
1248#ifdef FEAT_RIGHTLEFT
1249 if (wp->w_p_rl)
1250 col -= txtcol;
1251#endif
1252 while (col < wp->w_width
1253#ifdef FEAT_RIGHTLEFT
1254 - (wp->w_p_rl ? txtcol : 0)
1255#endif
1256 )
1257 {
1258 if (enc_utf8)
1259 {
1260 if (fill_fold >= 0x80)
1261 {
1262 ScreenLinesUC[off + col] = fill_fold;
1263 ScreenLinesC[0][off + col] = 0;
1264 ScreenLines[off + col] = 0x80; // avoid storing zero
1265 }
1266 else
1267 {
1268 ScreenLinesUC[off + col] = 0;
1269 ScreenLines[off + col] = fill_fold;
1270 }
1271 col++;
1272 }
1273 else
1274 ScreenLines[off + col++] = fill_fold;
1275 }
1276
1277 if (text != buf)
1278 vim_free(text);
1279
1280 // 6. set highlighting for the Visual area an other text.
1281 // If all folded lines are in the Visual area, highlight the line.
1282 if (VIsual_active && wp->w_buffer == curwin->w_buffer)
1283 {
1284 if (LTOREQ_POS(curwin->w_cursor, VIsual))
1285 {
1286 // Visual is after curwin->w_cursor
1287 top = &curwin->w_cursor;
1288 bot = &VIsual;
1289 }
1290 else
1291 {
1292 // Visual is before curwin->w_cursor
1293 top = &VIsual;
1294 bot = &curwin->w_cursor;
1295 }
1296 if (lnum >= top->lnum
1297 && lnume <= bot->lnum
1298 && (VIsual_mode != 'v'
1299 || ((lnum > top->lnum
1300 || (lnum == top->lnum
1301 && top->col == 0))
1302 && (lnume < bot->lnum
1303 || (lnume == bot->lnum
1304 && (bot->col - (*p_sel == 'e'))
1305 >= (colnr_T)STRLEN(ml_get_buf(wp->w_buffer, lnume, FALSE)))))))
1306 {
1307 if (VIsual_mode == Ctrl_V)
1308 {
1309 // Visual block mode: highlight the chars part of the block
1310 if (wp->w_old_cursor_fcol + txtcol < (colnr_T)wp->w_width)
1311 {
1312 if (wp->w_old_cursor_lcol != MAXCOL
1313 && wp->w_old_cursor_lcol + txtcol
1314 < (colnr_T)wp->w_width)
1315 len = wp->w_old_cursor_lcol;
1316 else
1317 len = wp->w_width - txtcol;
1318 RL_MEMSET(wp->w_old_cursor_fcol + txtcol, HL_ATTR(HLF_V),
1319 len - (int)wp->w_old_cursor_fcol);
1320 }
1321 }
1322 else
1323 {
1324 // Set all attributes of the text
1325 RL_MEMSET(txtcol, HL_ATTR(HLF_V), wp->w_width - txtcol);
1326 }
1327 }
1328 }
1329
1330#ifdef FEAT_SYN_HL
1331 // Show colorcolumn in the fold line, but let cursorcolumn override it.
1332 if (wp->w_p_cc_cols)
1333 {
1334 int i = 0;
1335 int j = wp->w_p_cc_cols[i];
1336 int old_txtcol = txtcol;
1337
1338 while (j > -1)
1339 {
1340 txtcol += j;
1341 if (wp->w_p_wrap)
1342 txtcol -= wp->w_skipcol;
1343 else
1344 txtcol -= wp->w_leftcol;
1345 if (txtcol >= 0 && txtcol < wp->w_width)
1346 ScreenAttrs[off + txtcol] = hl_combine_attr(
1347 ScreenAttrs[off + txtcol], HL_ATTR(HLF_MC));
1348 txtcol = old_txtcol;
1349 j = wp->w_p_cc_cols[++i];
1350 }
1351 }
1352
1353 // Show 'cursorcolumn' in the fold line.
1354 if (wp->w_p_cuc)
1355 {
1356 txtcol += wp->w_virtcol;
1357 if (wp->w_p_wrap)
1358 txtcol -= wp->w_skipcol;
1359 else
1360 txtcol -= wp->w_leftcol;
1361 if (txtcol >= 0 && txtcol < wp->w_width)
1362 ScreenAttrs[off + txtcol] = hl_combine_attr(
1363 ScreenAttrs[off + txtcol], HL_ATTR(HLF_CUC));
1364 }
1365#endif
1366
1367 screen_line(row + W_WINROW(wp), wp->w_wincol, (int)wp->w_width,
1368 (int)wp->w_width, 0);
1369
1370 // Update w_cline_height and w_cline_folded if the cursor line was
1371 // updated (saves a call to plines() later).
1372 if (wp == curwin
1373 && lnum <= curwin->w_cursor.lnum
1374 && lnume >= curwin->w_cursor.lnum)
1375 {
1376 curwin->w_cline_row = row;
1377 curwin->w_cline_height = 1;
1378 curwin->w_cline_folded = TRUE;
1379 curwin->w_valid |= (VALID_CHEIGHT|VALID_CROW);
1380 }
Bram Moolenaar00aaa512021-07-03 18:04:11 +02001381
1382# ifdef FEAT_CONCEAL
1383 // When the line was not folded w_wrow may have been set, recompute it.
Bram Moolenaar5cb09622021-07-05 22:03:04 +02001384 if (wp == curwin
1385 && wp->w_cursor.lnum >= lnum
1386 && wp->w_cursor.lnum <= lnume
1387 && conceal_cursor_line(wp))
Bram Moolenaar00aaa512021-07-03 18:04:11 +02001388 curs_columns(TRUE);
1389# endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001390}
1391#endif
1392
1393/*
1394 * Update a single window.
1395 *
1396 * This may cause the windows below it also to be redrawn (when clearing the
1397 * screen or scrolling lines).
1398 *
1399 * How the window is redrawn depends on wp->w_redr_type. Each type also
1400 * implies the one below it.
1401 * NOT_VALID redraw the whole window
1402 * SOME_VALID redraw the whole window but do scroll when possible
1403 * REDRAW_TOP redraw the top w_upd_rows window lines, otherwise like VALID
1404 * INVERTED redraw the changed part of the Visual area
1405 * INVERTED_ALL redraw the whole Visual area
1406 * VALID 1. scroll up/down to adjust for a changed w_topline
1407 * 2. update lines at the top when scrolled down
1408 * 3. redraw changed text:
1409 * - if wp->w_buffer->b_mod_set set, update lines between
1410 * b_mod_top and b_mod_bot.
1411 * - if wp->w_redraw_top non-zero, redraw lines between
1412 * wp->w_redraw_top and wp->w_redr_bot.
1413 * - continue redrawing when syntax status is invalid.
1414 * 4. if scrolled up, update lines at the bottom.
1415 * This results in three areas that may need updating:
1416 * top: from first row to top_end (when scrolled down)
1417 * mid: from mid_start to mid_end (update inversion or changed text)
1418 * bot: from bot_start to last row (when scrolled up)
1419 */
1420 static void
1421win_update(win_T *wp)
1422{
1423 buf_T *buf = wp->w_buffer;
1424 int type;
1425 int top_end = 0; // Below last row of the top area that needs
1426 // updating. 0 when no top area updating.
1427 int mid_start = 999;// first row of the mid area that needs
1428 // updating. 999 when no mid area updating.
1429 int mid_end = 0; // Below last row of the mid area that needs
1430 // updating. 0 when no mid area updating.
1431 int bot_start = 999;// first row of the bot area that needs
1432 // updating. 999 when no bot area updating
1433 int scrolled_down = FALSE; // TRUE when scrolled down when
1434 // w_topline got smaller a bit
1435#ifdef FEAT_SEARCH_EXTRA
1436 int top_to_mod = FALSE; // redraw above mod_top
1437#endif
1438
1439 int row; // current window row to display
1440 linenr_T lnum; // current buffer lnum to display
1441 int idx; // current index in w_lines[]
1442 int srow; // starting row of the current line
1443
1444 int eof = FALSE; // if TRUE, we hit the end of the file
1445 int didline = FALSE; // if TRUE, we finished the last line
1446 int i;
1447 long j;
1448 static int recursive = FALSE; // being called recursively
Bram Moolenaarcbee6352019-11-12 20:49:15 +01001449 linenr_T old_botline = wp->w_botline;
1450#ifdef FEAT_CONCEAL
1451 int old_wrow = wp->w_wrow;
1452 int old_wcol = wp->w_wcol;
1453#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001454#ifdef FEAT_FOLDING
1455 long fold_count;
1456#endif
1457#ifdef FEAT_SYN_HL
1458 // remember what happened to the previous line, to know if
1459 // check_visual_highlight() can be used
1460#define DID_NONE 1 // didn't update a line
1461#define DID_LINE 2 // updated a normal line
1462#define DID_FOLD 3 // updated a folded line
1463 int did_update = DID_NONE;
1464 linenr_T syntax_last_parsed = 0; // last parsed text line
1465#endif
1466 linenr_T mod_top = 0;
1467 linenr_T mod_bot = 0;
1468#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
1469 int save_got_int;
1470#endif
1471#ifdef SYN_TIME_LIMIT
1472 proftime_T syntax_tm;
1473#endif
1474
Bram Moolenaare52e0c82020-02-28 22:20:10 +01001475#if defined(FEAT_SEARCH_EXTRA) || defined(FEAT_CLIPBOARD)
1476 // This needs to be done only for the first window when update_screen() is
1477 // called.
1478 if (!did_update_one_window)
1479 {
1480 did_update_one_window = TRUE;
1481# ifdef FEAT_SEARCH_EXTRA
1482 start_search_hl();
1483# endif
1484# ifdef FEAT_CLIPBOARD
1485 // When Visual area changed, may have to update selection.
1486 if (clip_star.available && clip_isautosel_star())
1487 clip_update_selection(&clip_star);
1488 if (clip_plus.available && clip_isautosel_plus())
1489 clip_update_selection(&clip_plus);
1490# endif
1491 }
1492#endif
1493
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001494 type = wp->w_redr_type;
1495
1496 if (type == NOT_VALID)
1497 {
1498 wp->w_redr_status = TRUE;
1499 wp->w_lines_valid = 0;
1500 }
1501
Bram Moolenaarae0f1512021-03-30 22:12:12 +02001502 // Window frame is zero-height: nothing to draw.
1503 if (wp->w_height + WINBAR_HEIGHT(wp) == 0
1504 || (wp->w_frame->fr_height == wp->w_status_height
1505#if defined(FEAT_PROP_POPUP)
1506 && !popup_is_popup(wp)
1507#endif
1508 ))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001509 {
1510 wp->w_redr_type = 0;
1511 return;
1512 }
1513
1514 // Window is zero-width: Only need to draw the separator.
1515 if (wp->w_width == 0)
1516 {
1517 // draw the vertical separator right of this window
1518 draw_vsep_win(wp, 0);
1519 wp->w_redr_type = 0;
1520 return;
1521 }
1522
1523#ifdef FEAT_TERMINAL
1524 // If this window contains a terminal, redraw works completely differently.
1525 if (term_do_update_window(wp))
1526 {
1527 term_update_window(wp);
1528# ifdef FEAT_MENU
1529 // Draw the window toolbar, if there is one.
1530 if (winbar_height(wp) > 0)
1531 redraw_win_toolbar(wp);
1532# endif
1533 wp->w_redr_type = 0;
1534 return;
1535 }
1536#endif
1537
1538#ifdef FEAT_SEARCH_EXTRA
1539 init_search_hl(wp, &screen_search_hl);
1540#endif
1541
1542#ifdef FEAT_LINEBREAK
1543 // Force redraw when width of 'number' or 'relativenumber' column
1544 // changes.
1545 i = (wp->w_p_nu || wp->w_p_rnu) ? number_width(wp) : 0;
1546 if (wp->w_nrwidth != i)
1547 {
1548 type = NOT_VALID;
1549 wp->w_nrwidth = i;
1550 }
1551 else
1552#endif
1553
1554 if (buf->b_mod_set && buf->b_mod_xlines != 0 && wp->w_redraw_top != 0)
1555 {
1556 // When there are both inserted/deleted lines and specific lines to be
1557 // redrawn, w_redraw_top and w_redraw_bot may be invalid, just redraw
1558 // everything (only happens when redrawing is off for while).
1559 type = NOT_VALID;
1560 }
1561 else
1562 {
1563 // Set mod_top to the first line that needs displaying because of
1564 // changes. Set mod_bot to the first line after the changes.
1565 mod_top = wp->w_redraw_top;
1566 if (wp->w_redraw_bot != 0)
1567 mod_bot = wp->w_redraw_bot + 1;
1568 else
1569 mod_bot = 0;
1570 if (buf->b_mod_set)
1571 {
1572 if (mod_top == 0 || mod_top > buf->b_mod_top)
1573 {
1574 mod_top = buf->b_mod_top;
1575#ifdef FEAT_SYN_HL
1576 // Need to redraw lines above the change that may be included
1577 // in a pattern match.
1578 if (syntax_present(wp))
1579 {
1580 mod_top -= buf->b_s.b_syn_sync_linebreaks;
1581 if (mod_top < 1)
1582 mod_top = 1;
1583 }
1584#endif
1585 }
1586 if (mod_bot == 0 || mod_bot < buf->b_mod_bot)
1587 mod_bot = buf->b_mod_bot;
1588
1589#ifdef FEAT_SEARCH_EXTRA
1590 // When 'hlsearch' is on and using a multi-line search pattern, a
1591 // change in one line may make the Search highlighting in a
1592 // previous line invalid. Simple solution: redraw all visible
1593 // lines above the change.
1594 // Same for a match pattern.
1595 if (screen_search_hl.rm.regprog != NULL
1596 && re_multiline(screen_search_hl.rm.regprog))
1597 top_to_mod = TRUE;
1598 else
1599 {
1600 matchitem_T *cur = wp->w_match_head;
1601
1602 while (cur != NULL)
1603 {
1604 if (cur->match.regprog != NULL
1605 && re_multiline(cur->match.regprog))
1606 {
1607 top_to_mod = TRUE;
1608 break;
1609 }
1610 cur = cur->next;
1611 }
1612 }
1613#endif
1614 }
1615#ifdef FEAT_FOLDING
1616 if (mod_top != 0 && hasAnyFolding(wp))
1617 {
1618 linenr_T lnumt, lnumb;
1619
1620 // A change in a line can cause lines above it to become folded or
1621 // unfolded. Find the top most buffer line that may be affected.
1622 // If the line was previously folded and displayed, get the first
1623 // line of that fold. If the line is folded now, get the first
1624 // folded line. Use the minimum of these two.
1625
1626 // Find last valid w_lines[] entry above mod_top. Set lnumt to
1627 // the line below it. If there is no valid entry, use w_topline.
1628 // Find the first valid w_lines[] entry below mod_bot. Set lnumb
1629 // to this line. If there is no valid entry, use MAXLNUM.
1630 lnumt = wp->w_topline;
1631 lnumb = MAXLNUM;
1632 for (i = 0; i < wp->w_lines_valid; ++i)
1633 if (wp->w_lines[i].wl_valid)
1634 {
1635 if (wp->w_lines[i].wl_lastlnum < mod_top)
1636 lnumt = wp->w_lines[i].wl_lastlnum + 1;
1637 if (lnumb == MAXLNUM && wp->w_lines[i].wl_lnum >= mod_bot)
1638 {
1639 lnumb = wp->w_lines[i].wl_lnum;
1640 // When there is a fold column it might need updating
1641 // in the next line ("J" just above an open fold).
1642 if (compute_foldcolumn(wp, 0) > 0)
1643 ++lnumb;
1644 }
1645 }
1646
1647 (void)hasFoldingWin(wp, mod_top, &mod_top, NULL, TRUE, NULL);
1648 if (mod_top > lnumt)
1649 mod_top = lnumt;
1650
1651 // Now do the same for the bottom line (one above mod_bot).
1652 --mod_bot;
1653 (void)hasFoldingWin(wp, mod_bot, NULL, &mod_bot, TRUE, NULL);
1654 ++mod_bot;
1655 if (mod_bot < lnumb)
1656 mod_bot = lnumb;
1657 }
1658#endif
1659
1660 // When a change starts above w_topline and the end is below
1661 // w_topline, start redrawing at w_topline.
1662 // If the end of the change is above w_topline: do like no change was
1663 // made, but redraw the first line to find changes in syntax.
1664 if (mod_top != 0 && mod_top < wp->w_topline)
1665 {
1666 if (mod_bot > wp->w_topline)
1667 mod_top = wp->w_topline;
1668#ifdef FEAT_SYN_HL
1669 else if (syntax_present(wp))
1670 top_end = 1;
1671#endif
1672 }
1673
1674 // When line numbers are displayed need to redraw all lines below
1675 // inserted/deleted lines.
1676 if (mod_top != 0 && buf->b_mod_xlines != 0 && wp->w_p_nu)
1677 mod_bot = MAXLNUM;
1678 }
1679 wp->w_redraw_top = 0; // reset for next time
1680 wp->w_redraw_bot = 0;
1681
1682 // When only displaying the lines at the top, set top_end. Used when
1683 // window has scrolled down for msg_scrolled.
1684 if (type == REDRAW_TOP)
1685 {
1686 j = 0;
1687 for (i = 0; i < wp->w_lines_valid; ++i)
1688 {
1689 j += wp->w_lines[i].wl_size;
1690 if (j >= wp->w_upd_rows)
1691 {
1692 top_end = j;
1693 break;
1694 }
1695 }
1696 if (top_end == 0)
1697 // not found (cannot happen?): redraw everything
1698 type = NOT_VALID;
1699 else
1700 // top area defined, the rest is VALID
1701 type = VALID;
1702 }
1703
1704 // Trick: we want to avoid clearing the screen twice. screenclear() will
1705 // set "screen_cleared" to TRUE. The special value MAYBE (which is still
1706 // non-zero and thus not FALSE) will indicate that screenclear() was not
1707 // called.
1708 if (screen_cleared)
1709 screen_cleared = MAYBE;
1710
1711 // If there are no changes on the screen that require a complete redraw,
1712 // handle three cases:
1713 // 1: we are off the top of the screen by a few lines: scroll down
1714 // 2: wp->w_topline is below wp->w_lines[0].wl_lnum: may scroll up
1715 // 3: wp->w_topline is wp->w_lines[0].wl_lnum: find first entry in
1716 // w_lines[] that needs updating.
1717 if ((type == VALID || type == SOME_VALID
1718 || type == INVERTED || type == INVERTED_ALL)
1719#ifdef FEAT_DIFF
1720 && !wp->w_botfill && !wp->w_old_botfill
1721#endif
1722 )
1723 {
Bram Moolenaarf8992d42020-08-01 19:14:13 +02001724 if (mod_top != 0
1725 && wp->w_topline == mod_top
1726 && (!wp->w_lines[0].wl_valid
Bram Moolenaar9dc19172020-08-19 20:19:48 +02001727 || wp->w_topline <= wp->w_lines[0].wl_lnum))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001728 {
Bram Moolenaarf8992d42020-08-01 19:14:13 +02001729 // w_topline is the first changed line and window is not scrolled,
1730 // the scrolling from changed lines will be done further down.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001731 }
1732 else if (wp->w_lines[0].wl_valid
1733 && (wp->w_topline < wp->w_lines[0].wl_lnum
1734#ifdef FEAT_DIFF
1735 || (wp->w_topline == wp->w_lines[0].wl_lnum
1736 && wp->w_topfill > wp->w_old_topfill)
1737#endif
1738 ))
1739 {
1740 // New topline is above old topline: May scroll down.
1741#ifdef FEAT_FOLDING
1742 if (hasAnyFolding(wp))
1743 {
1744 linenr_T ln;
1745
1746 // count the number of lines we are off, counting a sequence
1747 // of folded lines as one
1748 j = 0;
1749 for (ln = wp->w_topline; ln < wp->w_lines[0].wl_lnum; ++ln)
1750 {
1751 ++j;
1752 if (j >= wp->w_height - 2)
1753 break;
1754 (void)hasFoldingWin(wp, ln, NULL, &ln, TRUE, NULL);
1755 }
1756 }
1757 else
1758#endif
1759 j = wp->w_lines[0].wl_lnum - wp->w_topline;
1760 if (j < wp->w_height - 2) // not too far off
1761 {
1762 i = plines_m_win(wp, wp->w_topline, wp->w_lines[0].wl_lnum - 1);
1763#ifdef FEAT_DIFF
1764 // insert extra lines for previously invisible filler lines
1765 if (wp->w_lines[0].wl_lnum != wp->w_topline)
1766 i += diff_check_fill(wp, wp->w_lines[0].wl_lnum)
1767 - wp->w_old_topfill;
1768#endif
1769 if (i < wp->w_height - 2) // less than a screen off
1770 {
1771 // Try to insert the correct number of lines.
1772 // If not the last window, delete the lines at the bottom.
1773 // win_ins_lines may fail when the terminal can't do it.
1774 if (i > 0)
1775 check_for_delay(FALSE);
1776 if (win_ins_lines(wp, 0, i, FALSE, wp == firstwin) == OK)
1777 {
1778 if (wp->w_lines_valid != 0)
1779 {
1780 // Need to update rows that are new, stop at the
1781 // first one that scrolled down.
1782 top_end = i;
1783 scrolled_down = TRUE;
1784
1785 // Move the entries that were scrolled, disable
1786 // the entries for the lines to be redrawn.
1787 if ((wp->w_lines_valid += j) > wp->w_height)
1788 wp->w_lines_valid = wp->w_height;
1789 for (idx = wp->w_lines_valid; idx - j >= 0; idx--)
1790 wp->w_lines[idx] = wp->w_lines[idx - j];
1791 while (idx >= 0)
1792 wp->w_lines[idx--].wl_valid = FALSE;
1793 }
1794 }
1795 else
1796 mid_start = 0; // redraw all lines
1797 }
1798 else
1799 mid_start = 0; // redraw all lines
1800 }
1801 else
1802 mid_start = 0; // redraw all lines
1803 }
1804 else
1805 {
1806 // New topline is at or below old topline: May scroll up.
1807 // When topline didn't change, find first entry in w_lines[] that
1808 // needs updating.
1809
1810 // try to find wp->w_topline in wp->w_lines[].wl_lnum
1811 j = -1;
1812 row = 0;
1813 for (i = 0; i < wp->w_lines_valid; i++)
1814 {
1815 if (wp->w_lines[i].wl_valid
1816 && wp->w_lines[i].wl_lnum == wp->w_topline)
1817 {
1818 j = i;
1819 break;
1820 }
1821 row += wp->w_lines[i].wl_size;
1822 }
1823 if (j == -1)
1824 {
1825 // if wp->w_topline is not in wp->w_lines[].wl_lnum redraw all
1826 // lines
1827 mid_start = 0;
1828 }
1829 else
1830 {
1831 // Try to delete the correct number of lines.
1832 // wp->w_topline is at wp->w_lines[i].wl_lnum.
1833#ifdef FEAT_DIFF
1834 // If the topline didn't change, delete old filler lines,
1835 // otherwise delete filler lines of the new topline...
1836 if (wp->w_lines[0].wl_lnum == wp->w_topline)
1837 row += wp->w_old_topfill;
1838 else
1839 row += diff_check_fill(wp, wp->w_topline);
1840 // ... but don't delete new filler lines.
1841 row -= wp->w_topfill;
1842#endif
1843 if (row > 0)
1844 {
1845 check_for_delay(FALSE);
1846 if (win_del_lines(wp, 0, row, FALSE, wp == firstwin, 0)
1847 == OK)
1848 bot_start = wp->w_height - row;
1849 else
1850 mid_start = 0; // redraw all lines
1851 }
1852 if ((row == 0 || bot_start < 999) && wp->w_lines_valid != 0)
1853 {
1854 // Skip the lines (below the deleted lines) that are still
1855 // valid and don't need redrawing. Copy their info
1856 // upwards, to compensate for the deleted lines. Set
1857 // bot_start to the first row that needs redrawing.
1858 bot_start = 0;
1859 idx = 0;
1860 for (;;)
1861 {
1862 wp->w_lines[idx] = wp->w_lines[j];
1863 // stop at line that didn't fit, unless it is still
1864 // valid (no lines deleted)
1865 if (row > 0 && bot_start + row
1866 + (int)wp->w_lines[j].wl_size > wp->w_height)
1867 {
1868 wp->w_lines_valid = idx + 1;
1869 break;
1870 }
1871 bot_start += wp->w_lines[idx++].wl_size;
1872
1873 // stop at the last valid entry in w_lines[].wl_size
1874 if (++j >= wp->w_lines_valid)
1875 {
1876 wp->w_lines_valid = idx;
1877 break;
1878 }
1879 }
1880#ifdef FEAT_DIFF
1881 // Correct the first entry for filler lines at the top
1882 // when it won't get updated below.
1883 if (wp->w_p_diff && bot_start > 0)
1884 wp->w_lines[0].wl_size =
1885 plines_win_nofill(wp, wp->w_topline, TRUE)
1886 + wp->w_topfill;
1887#endif
1888 }
1889 }
1890 }
1891
1892 // When starting redraw in the first line, redraw all lines. When
1893 // there is only one window it's probably faster to clear the screen
1894 // first.
1895 if (mid_start == 0)
1896 {
1897 mid_end = wp->w_height;
1898 if (ONE_WINDOW && !WIN_IS_POPUP(wp))
1899 {
1900 // Clear the screen when it was not done by win_del_lines() or
1901 // win_ins_lines() above, "screen_cleared" is FALSE or MAYBE
1902 // then.
1903 if (screen_cleared != TRUE)
1904 screenclear();
1905 // The screen was cleared, redraw the tab pages line.
1906 if (redraw_tabline)
1907 draw_tabline();
1908 }
1909 }
1910
1911 // When win_del_lines() or win_ins_lines() caused the screen to be
1912 // cleared (only happens for the first window) or when screenclear()
1913 // was called directly above, "must_redraw" will have been set to
1914 // NOT_VALID, need to reset it here to avoid redrawing twice.
1915 if (screen_cleared == TRUE)
1916 must_redraw = 0;
1917 }
1918 else
1919 {
1920 // Not VALID or INVERTED: redraw all lines.
1921 mid_start = 0;
1922 mid_end = wp->w_height;
1923 }
1924
1925 if (type == SOME_VALID)
1926 {
1927 // SOME_VALID: redraw all lines.
1928 mid_start = 0;
1929 mid_end = wp->w_height;
1930 type = NOT_VALID;
1931 }
1932
1933 // check if we are updating or removing the inverted part
1934 if ((VIsual_active && buf == curwin->w_buffer)
1935 || (wp->w_old_cursor_lnum != 0 && type != NOT_VALID))
1936 {
1937 linenr_T from, to;
1938
1939 if (VIsual_active)
1940 {
1941 if (VIsual_active
1942 && (VIsual_mode != wp->w_old_visual_mode
1943 || type == INVERTED_ALL))
1944 {
1945 // If the type of Visual selection changed, redraw the whole
1946 // selection. Also when the ownership of the X selection is
1947 // gained or lost.
1948 if (curwin->w_cursor.lnum < VIsual.lnum)
1949 {
1950 from = curwin->w_cursor.lnum;
1951 to = VIsual.lnum;
1952 }
1953 else
1954 {
1955 from = VIsual.lnum;
1956 to = curwin->w_cursor.lnum;
1957 }
1958 // redraw more when the cursor moved as well
1959 if (wp->w_old_cursor_lnum < from)
1960 from = wp->w_old_cursor_lnum;
1961 if (wp->w_old_cursor_lnum > to)
1962 to = wp->w_old_cursor_lnum;
1963 if (wp->w_old_visual_lnum < from)
1964 from = wp->w_old_visual_lnum;
1965 if (wp->w_old_visual_lnum > to)
1966 to = wp->w_old_visual_lnum;
1967 }
1968 else
1969 {
1970 // Find the line numbers that need to be updated: The lines
1971 // between the old cursor position and the current cursor
1972 // position. Also check if the Visual position changed.
1973 if (curwin->w_cursor.lnum < wp->w_old_cursor_lnum)
1974 {
1975 from = curwin->w_cursor.lnum;
1976 to = wp->w_old_cursor_lnum;
1977 }
1978 else
1979 {
1980 from = wp->w_old_cursor_lnum;
1981 to = curwin->w_cursor.lnum;
1982 if (from == 0) // Visual mode just started
1983 from = to;
1984 }
1985
1986 if (VIsual.lnum != wp->w_old_visual_lnum
1987 || VIsual.col != wp->w_old_visual_col)
1988 {
1989 if (wp->w_old_visual_lnum < from
1990 && wp->w_old_visual_lnum != 0)
1991 from = wp->w_old_visual_lnum;
1992 if (wp->w_old_visual_lnum > to)
1993 to = wp->w_old_visual_lnum;
1994 if (VIsual.lnum < from)
1995 from = VIsual.lnum;
1996 if (VIsual.lnum > to)
1997 to = VIsual.lnum;
1998 }
1999 }
2000
2001 // If in block mode and changed column or curwin->w_curswant:
2002 // update all lines.
2003 // First compute the actual start and end column.
2004 if (VIsual_mode == Ctrl_V)
2005 {
2006 colnr_T fromc, toc;
2007#if defined(FEAT_LINEBREAK)
Gary Johnson51ad8502021-08-03 18:33:08 +02002008 int save_ve_flags = curwin->w_ve_flags;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002009
2010 if (curwin->w_p_lbr)
Gary Johnson51ad8502021-08-03 18:33:08 +02002011 curwin->w_ve_flags = VE_ALL;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002012#endif
2013 getvcols(wp, &VIsual, &curwin->w_cursor, &fromc, &toc);
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002014 ++toc;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002015#if defined(FEAT_LINEBREAK)
Gary Johnson51ad8502021-08-03 18:33:08 +02002016 curwin->w_ve_flags = save_ve_flags;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002017#endif
Bram Moolenaar9cee4a12021-07-03 15:08:37 +02002018 // Highlight to the end of the line, unless 'virtualedit' has
2019 // "block".
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002020 if (curwin->w_curswant == MAXCOL)
2021 {
Gary Johnson53ba05b2021-07-26 22:19:10 +02002022 if (get_ve_flags() & VE_BLOCK)
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002023 {
2024 pos_T pos;
2025 int cursor_above =
2026 curwin->w_cursor.lnum < VIsual.lnum;
2027
2028 // Need to find the longest line.
2029 toc = 0;
2030 pos.coladd = 0;
2031 for (pos.lnum = curwin->w_cursor.lnum; cursor_above
2032 ? pos.lnum <= VIsual.lnum
2033 : pos.lnum >= VIsual.lnum;
2034 pos.lnum += cursor_above ? 1 : -1)
2035 {
2036 colnr_T t;
2037
Bram Moolenaar6bcb1822021-07-09 15:54:00 +02002038 pos.col = (int)STRLEN(ml_get_buf(wp->w_buffer,
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002039 pos.lnum, FALSE));
2040 getvvcol(wp, &pos, NULL, NULL, &t);
2041 if (toc < t)
2042 toc = t;
2043 }
2044 ++toc;
2045 }
2046 else
2047 toc = MAXCOL;
2048 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002049
2050 if (fromc != wp->w_old_cursor_fcol
2051 || toc != wp->w_old_cursor_lcol)
2052 {
2053 if (from > VIsual.lnum)
2054 from = VIsual.lnum;
2055 if (to < VIsual.lnum)
2056 to = VIsual.lnum;
2057 }
2058 wp->w_old_cursor_fcol = fromc;
2059 wp->w_old_cursor_lcol = toc;
2060 }
2061 }
2062 else
2063 {
2064 // Use the line numbers of the old Visual area.
2065 if (wp->w_old_cursor_lnum < wp->w_old_visual_lnum)
2066 {
2067 from = wp->w_old_cursor_lnum;
2068 to = wp->w_old_visual_lnum;
2069 }
2070 else
2071 {
2072 from = wp->w_old_visual_lnum;
2073 to = wp->w_old_cursor_lnum;
2074 }
2075 }
2076
2077 // There is no need to update lines above the top of the window.
2078 if (from < wp->w_topline)
2079 from = wp->w_topline;
2080
2081 // If we know the value of w_botline, use it to restrict the update to
2082 // the lines that are visible in the window.
2083 if (wp->w_valid & VALID_BOTLINE)
2084 {
2085 if (from >= wp->w_botline)
2086 from = wp->w_botline - 1;
2087 if (to >= wp->w_botline)
2088 to = wp->w_botline - 1;
2089 }
2090
2091 // Find the minimal part to be updated.
2092 // Watch out for scrolling that made entries in w_lines[] invalid.
2093 // E.g., CTRL-U makes the first half of w_lines[] invalid and sets
2094 // top_end; need to redraw from top_end to the "to" line.
2095 // A middle mouse click with a Visual selection may change the text
2096 // above the Visual area and reset wl_valid, do count these for
2097 // mid_end (in srow).
2098 if (mid_start > 0)
2099 {
2100 lnum = wp->w_topline;
2101 idx = 0;
2102 srow = 0;
2103 if (scrolled_down)
2104 mid_start = top_end;
2105 else
2106 mid_start = 0;
2107 while (lnum < from && idx < wp->w_lines_valid) // find start
2108 {
2109 if (wp->w_lines[idx].wl_valid)
2110 mid_start += wp->w_lines[idx].wl_size;
2111 else if (!scrolled_down)
2112 srow += wp->w_lines[idx].wl_size;
2113 ++idx;
2114# ifdef FEAT_FOLDING
2115 if (idx < wp->w_lines_valid && wp->w_lines[idx].wl_valid)
2116 lnum = wp->w_lines[idx].wl_lnum;
2117 else
2118# endif
2119 ++lnum;
2120 }
2121 srow += mid_start;
2122 mid_end = wp->w_height;
2123 for ( ; idx < wp->w_lines_valid; ++idx) // find end
2124 {
2125 if (wp->w_lines[idx].wl_valid
2126 && wp->w_lines[idx].wl_lnum >= to + 1)
2127 {
2128 // Only update until first row of this line
2129 mid_end = srow;
2130 break;
2131 }
2132 srow += wp->w_lines[idx].wl_size;
2133 }
2134 }
2135 }
2136
2137 if (VIsual_active && buf == curwin->w_buffer)
2138 {
2139 wp->w_old_visual_mode = VIsual_mode;
2140 wp->w_old_cursor_lnum = curwin->w_cursor.lnum;
2141 wp->w_old_visual_lnum = VIsual.lnum;
2142 wp->w_old_visual_col = VIsual.col;
2143 wp->w_old_curswant = curwin->w_curswant;
2144 }
2145 else
2146 {
2147 wp->w_old_visual_mode = 0;
2148 wp->w_old_cursor_lnum = 0;
2149 wp->w_old_visual_lnum = 0;
2150 wp->w_old_visual_col = 0;
2151 }
2152
2153#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
2154 // reset got_int, otherwise regexp won't work
2155 save_got_int = got_int;
2156 got_int = 0;
2157#endif
2158#ifdef SYN_TIME_LIMIT
2159 // Set the time limit to 'redrawtime'.
2160 profile_setlimit(p_rdt, &syntax_tm);
2161 syn_set_timeout(&syntax_tm);
2162#endif
2163#ifdef FEAT_FOLDING
2164 win_foldinfo.fi_level = 0;
2165#endif
2166
2167#ifdef FEAT_MENU
2168 // Draw the window toolbar, if there is one.
2169 // TODO: only when needed.
2170 if (winbar_height(wp) > 0)
2171 redraw_win_toolbar(wp);
2172#endif
2173
2174 // Update all the window rows.
2175 idx = 0; // first entry in w_lines[].wl_size
2176 row = 0;
2177 srow = 0;
2178 lnum = wp->w_topline; // first line shown in window
2179 for (;;)
2180 {
2181 // stop updating when reached the end of the window (check for _past_
2182 // the end of the window is at the end of the loop)
2183 if (row == wp->w_height)
2184 {
2185 didline = TRUE;
2186 break;
2187 }
2188
2189 // stop updating when hit the end of the file
2190 if (lnum > buf->b_ml.ml_line_count)
2191 {
2192 eof = TRUE;
2193 break;
2194 }
2195
2196 // Remember the starting row of the line that is going to be dealt
2197 // with. It is used further down when the line doesn't fit.
2198 srow = row;
2199
2200 // Update a line when it is in an area that needs updating, when it
2201 // has changes or w_lines[idx] is invalid.
2202 // "bot_start" may be halfway a wrapped line after using
2203 // win_del_lines(), check if the current line includes it.
2204 // When syntax folding is being used, the saved syntax states will
2205 // already have been updated, we can't see where the syntax state is
2206 // the same again, just update until the end of the window.
2207 if (row < top_end
2208 || (row >= mid_start && row < mid_end)
2209#ifdef FEAT_SEARCH_EXTRA
2210 || top_to_mod
2211#endif
2212 || idx >= wp->w_lines_valid
2213 || (row + wp->w_lines[idx].wl_size > bot_start)
2214 || (mod_top != 0
2215 && (lnum == mod_top
2216 || (lnum >= mod_top
2217 && (lnum < mod_bot
2218#ifdef FEAT_SYN_HL
2219 || did_update == DID_FOLD
2220 || (did_update == DID_LINE
2221 && syntax_present(wp)
2222 && (
2223# ifdef FEAT_FOLDING
2224 (foldmethodIsSyntax(wp)
2225 && hasAnyFolding(wp)) ||
2226# endif
2227 syntax_check_changed(lnum)))
2228#endif
2229#ifdef FEAT_SEARCH_EXTRA
2230 // match in fixed position might need redraw
2231 // if lines were inserted or deleted
2232 || (wp->w_match_head != NULL
2233 && buf->b_mod_xlines != 0)
2234#endif
Bram Moolenaar11a58af2019-10-24 22:32:31 +02002235 ))))
2236#ifdef FEAT_SYN_HL
2237 || (wp->w_p_cul && (lnum == wp->w_cursor.lnum
2238 || lnum == wp->w_last_cursorline))
2239#endif
2240 )
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002241 {
2242#ifdef FEAT_SEARCH_EXTRA
2243 if (lnum == mod_top)
2244 top_to_mod = FALSE;
2245#endif
2246
2247 // When at start of changed lines: May scroll following lines
2248 // up or down to minimize redrawing.
2249 // Don't do this when the change continues until the end.
2250 // Don't scroll when dollar_vcol >= 0, keep the "$".
Bram Moolenaarc9e7e342021-07-22 21:33:03 +02002251 // Don't scroll when redrawing the top, scrolled already above.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002252 if (lnum == mod_top
2253 && mod_bot != MAXLNUM
Bram Moolenaarc9e7e342021-07-22 21:33:03 +02002254 && !(dollar_vcol >= 0 && mod_bot == mod_top + 1)
2255 && row >= top_end)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002256 {
2257 int old_rows = 0;
2258 int new_rows = 0;
2259 int xtra_rows;
2260 linenr_T l;
2261
2262 // Count the old number of window rows, using w_lines[], which
2263 // should still contain the sizes for the lines as they are
2264 // currently displayed.
2265 for (i = idx; i < wp->w_lines_valid; ++i)
2266 {
2267 // Only valid lines have a meaningful wl_lnum. Invalid
2268 // lines are part of the changed area.
2269 if (wp->w_lines[i].wl_valid
2270 && wp->w_lines[i].wl_lnum == mod_bot)
2271 break;
2272 old_rows += wp->w_lines[i].wl_size;
2273#ifdef FEAT_FOLDING
2274 if (wp->w_lines[i].wl_valid
2275 && wp->w_lines[i].wl_lastlnum + 1 == mod_bot)
2276 {
2277 // Must have found the last valid entry above mod_bot.
2278 // Add following invalid entries.
2279 ++i;
2280 while (i < wp->w_lines_valid
2281 && !wp->w_lines[i].wl_valid)
2282 old_rows += wp->w_lines[i++].wl_size;
2283 break;
2284 }
2285#endif
2286 }
2287
2288 if (i >= wp->w_lines_valid)
2289 {
2290 // We can't find a valid line below the changed lines,
2291 // need to redraw until the end of the window.
2292 // Inserting/deleting lines has no use.
2293 bot_start = 0;
2294 }
2295 else
2296 {
2297 // Able to count old number of rows: Count new window
2298 // rows, and may insert/delete lines
2299 j = idx;
2300 for (l = lnum; l < mod_bot; ++l)
2301 {
2302#ifdef FEAT_FOLDING
2303 if (hasFoldingWin(wp, l, NULL, &l, TRUE, NULL))
2304 ++new_rows;
2305 else
2306#endif
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002307 {
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002308#ifdef FEAT_DIFF
2309 if (l == wp->w_topline)
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002310 new_rows += plines_win_nofill(wp, l, TRUE)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002311 + wp->w_topfill;
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002312 else
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002313#endif
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002314 new_rows += plines_win(wp, l, TRUE);
2315 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002316 ++j;
2317 if (new_rows > wp->w_height - row - 2)
2318 {
2319 // it's getting too much, must redraw the rest
2320 new_rows = 9999;
2321 break;
2322 }
2323 }
2324 xtra_rows = new_rows - old_rows;
2325 if (xtra_rows < 0)
2326 {
2327 // May scroll text up. If there is not enough
2328 // remaining text or scrolling fails, must redraw the
2329 // rest. If scrolling works, must redraw the text
2330 // below the scrolled text.
2331 if (row - xtra_rows >= wp->w_height - 2)
2332 mod_bot = MAXLNUM;
2333 else
2334 {
2335 check_for_delay(FALSE);
2336 if (win_del_lines(wp, row,
2337 -xtra_rows, FALSE, FALSE, 0) == FAIL)
2338 mod_bot = MAXLNUM;
2339 else
2340 bot_start = wp->w_height + xtra_rows;
2341 }
2342 }
2343 else if (xtra_rows > 0)
2344 {
2345 // May scroll text down. If there is not enough
2346 // remaining text of scrolling fails, must redraw the
2347 // rest.
2348 if (row + xtra_rows >= wp->w_height - 2)
2349 mod_bot = MAXLNUM;
2350 else
2351 {
2352 check_for_delay(FALSE);
2353 if (win_ins_lines(wp, row + old_rows,
2354 xtra_rows, FALSE, FALSE) == FAIL)
2355 mod_bot = MAXLNUM;
2356 else if (top_end > row + old_rows)
2357 // Scrolled the part at the top that requires
2358 // updating down.
2359 top_end += xtra_rows;
2360 }
2361 }
2362
2363 // When not updating the rest, may need to move w_lines[]
2364 // entries.
2365 if (mod_bot != MAXLNUM && i != j)
2366 {
2367 if (j < i)
2368 {
2369 int x = row + new_rows;
2370
2371 // move entries in w_lines[] upwards
2372 for (;;)
2373 {
2374 // stop at last valid entry in w_lines[]
2375 if (i >= wp->w_lines_valid)
2376 {
2377 wp->w_lines_valid = j;
2378 break;
2379 }
2380 wp->w_lines[j] = wp->w_lines[i];
2381 // stop at a line that won't fit
2382 if (x + (int)wp->w_lines[j].wl_size
2383 > wp->w_height)
2384 {
2385 wp->w_lines_valid = j + 1;
2386 break;
2387 }
2388 x += wp->w_lines[j++].wl_size;
2389 ++i;
2390 }
2391 if (bot_start > x)
2392 bot_start = x;
2393 }
2394 else // j > i
2395 {
2396 // move entries in w_lines[] downwards
2397 j -= i;
2398 wp->w_lines_valid += j;
2399 if (wp->w_lines_valid > wp->w_height)
2400 wp->w_lines_valid = wp->w_height;
2401 for (i = wp->w_lines_valid; i - j >= idx; --i)
2402 wp->w_lines[i] = wp->w_lines[i - j];
2403
2404 // The w_lines[] entries for inserted lines are
2405 // now invalid, but wl_size may be used above.
2406 // Reset to zero.
2407 while (i >= idx)
2408 {
2409 wp->w_lines[i].wl_size = 0;
2410 wp->w_lines[i--].wl_valid = FALSE;
2411 }
2412 }
2413 }
2414 }
2415 }
2416
2417#ifdef FEAT_FOLDING
2418 // When lines are folded, display one line for all of them.
2419 // Otherwise, display normally (can be several display lines when
2420 // 'wrap' is on).
2421 fold_count = foldedCount(wp, lnum, &win_foldinfo);
2422 if (fold_count != 0)
2423 {
2424 fold_line(wp, fold_count, &win_foldinfo, lnum, row);
2425 ++row;
2426 --fold_count;
2427 wp->w_lines[idx].wl_folded = TRUE;
2428 wp->w_lines[idx].wl_lastlnum = lnum + fold_count;
2429# ifdef FEAT_SYN_HL
2430 did_update = DID_FOLD;
2431# endif
2432 }
2433 else
2434#endif
2435 if (idx < wp->w_lines_valid
2436 && wp->w_lines[idx].wl_valid
2437 && wp->w_lines[idx].wl_lnum == lnum
2438 && lnum > wp->w_topline
2439 && !(dy_flags & (DY_LASTLINE | DY_TRUNCATE))
2440 && !WIN_IS_POPUP(wp)
2441 && srow + wp->w_lines[idx].wl_size > wp->w_height
2442#ifdef FEAT_DIFF
2443 && diff_check_fill(wp, lnum) == 0
2444#endif
2445 )
2446 {
2447 // This line is not going to fit. Don't draw anything here,
2448 // will draw "@ " lines below.
2449 row = wp->w_height + 1;
2450 }
2451 else
2452 {
2453#ifdef FEAT_SEARCH_EXTRA
2454 prepare_search_hl(wp, &screen_search_hl, lnum);
2455#endif
2456#ifdef FEAT_SYN_HL
2457 // Let the syntax stuff know we skipped a few lines.
2458 if (syntax_last_parsed != 0 && syntax_last_parsed + 1 < lnum
2459 && syntax_present(wp))
2460 syntax_end_parsing(syntax_last_parsed + 1);
2461#endif
2462
2463 // Display one line.
2464 row = win_line(wp, lnum, srow, wp->w_height,
2465 mod_top == 0, FALSE);
2466
2467#ifdef FEAT_FOLDING
2468 wp->w_lines[idx].wl_folded = FALSE;
2469 wp->w_lines[idx].wl_lastlnum = lnum;
2470#endif
2471#ifdef FEAT_SYN_HL
2472 did_update = DID_LINE;
2473 syntax_last_parsed = lnum;
2474#endif
2475 }
2476
2477 wp->w_lines[idx].wl_lnum = lnum;
2478 wp->w_lines[idx].wl_valid = TRUE;
2479
2480 // Past end of the window or end of the screen. Note that after
2481 // resizing wp->w_height may be end up too big. That's a problem
2482 // elsewhere, but prevent a crash here.
2483 if (row > wp->w_height || row + wp->w_winrow >= Rows)
2484 {
2485 // we may need the size of that too long line later on
2486 if (dollar_vcol == -1)
2487 wp->w_lines[idx].wl_size = plines_win(wp, lnum, TRUE);
2488 ++idx;
2489 break;
2490 }
2491 if (dollar_vcol == -1)
2492 wp->w_lines[idx].wl_size = row - srow;
2493 ++idx;
2494#ifdef FEAT_FOLDING
2495 lnum += fold_count + 1;
2496#else
2497 ++lnum;
2498#endif
2499 }
2500 else
2501 {
2502 if (wp->w_p_rnu)
2503 {
2504#ifdef FEAT_FOLDING
2505 // 'relativenumber' set: The text doesn't need to be drawn, but
2506 // the number column nearly always does.
2507 fold_count = foldedCount(wp, lnum, &win_foldinfo);
2508 if (fold_count != 0)
2509 fold_line(wp, fold_count, &win_foldinfo, lnum, row);
2510 else
2511#endif
2512 (void)win_line(wp, lnum, srow, wp->w_height, TRUE, TRUE);
2513 }
2514
2515 // This line does not need to be drawn, advance to the next one.
2516 row += wp->w_lines[idx++].wl_size;
2517 if (row > wp->w_height) // past end of screen
2518 break;
2519#ifdef FEAT_FOLDING
2520 lnum = wp->w_lines[idx - 1].wl_lastlnum + 1;
2521#else
2522 ++lnum;
2523#endif
2524#ifdef FEAT_SYN_HL
2525 did_update = DID_NONE;
2526#endif
2527 }
2528
2529 if (lnum > buf->b_ml.ml_line_count)
2530 {
2531 eof = TRUE;
2532 break;
2533 }
2534 }
2535
2536 // End of loop over all window lines.
2537
2538#ifdef FEAT_VTP
2539 // Rewrite the character at the end of the screen line.
Bram Moolenaar7ed8f592020-04-28 20:44:42 +02002540 // See the version that was fixed.
2541 if (use_vtp() && get_conpty_fix_type() < 1)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002542 {
2543 int i;
2544
2545 for (i = 0; i < Rows; ++i)
2546 if (enc_utf8)
2547 if ((*mb_off2cells)(LineOffset[i] + Columns - 2,
2548 LineOffset[i] + screen_Columns) > 1)
2549 screen_draw_rectangle(i, Columns - 2, 1, 2, FALSE);
2550 else
2551 screen_draw_rectangle(i, Columns - 1, 1, 1, FALSE);
2552 else
2553 screen_char(LineOffset[i] + Columns - 1, i, Columns - 1);
2554 }
2555#endif
2556
2557 if (idx > wp->w_lines_valid)
2558 wp->w_lines_valid = idx;
2559
2560#ifdef FEAT_SYN_HL
2561 // Let the syntax stuff know we stop parsing here.
2562 if (syntax_last_parsed != 0 && syntax_present(wp))
2563 syntax_end_parsing(syntax_last_parsed + 1);
2564#endif
2565
2566 // If we didn't hit the end of the file, and we didn't finish the last
2567 // line we were working on, then the line didn't fit.
2568 wp->w_empty_rows = 0;
2569#ifdef FEAT_DIFF
2570 wp->w_filler_rows = 0;
2571#endif
2572 if (!eof && !didline)
2573 {
2574 if (lnum == wp->w_topline)
2575 {
2576 // Single line that does not fit!
2577 // Don't overwrite it, it can be edited.
2578 wp->w_botline = lnum + 1;
2579 }
2580#ifdef FEAT_DIFF
2581 else if (diff_check_fill(wp, lnum) >= wp->w_height - srow)
2582 {
2583 // Window ends in filler lines.
2584 wp->w_botline = lnum;
2585 wp->w_filler_rows = wp->w_height - srow;
2586 }
2587#endif
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002588#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002589 else if (WIN_IS_POPUP(wp))
2590 {
2591 // popup line that doesn't fit is left as-is
2592 wp->w_botline = lnum;
2593 }
2594#endif
2595 else if (dy_flags & DY_TRUNCATE) // 'display' has "truncate"
2596 {
2597 int scr_row = W_WINROW(wp) + wp->w_height - 1;
2598
2599 // Last line isn't finished: Display "@@@" in the last screen line.
2600 screen_puts_len((char_u *)"@@", 2, scr_row, wp->w_wincol,
2601 HL_ATTR(HLF_AT));
2602 screen_fill(scr_row, scr_row + 1,
2603 (int)wp->w_wincol + 2, (int)W_ENDCOL(wp),
2604 '@', ' ', HL_ATTR(HLF_AT));
2605 set_empty_rows(wp, srow);
2606 wp->w_botline = lnum;
2607 }
2608 else if (dy_flags & DY_LASTLINE) // 'display' has "lastline"
2609 {
2610 // Last line isn't finished: Display "@@@" at the end.
2611 screen_fill(W_WINROW(wp) + wp->w_height - 1,
2612 W_WINROW(wp) + wp->w_height,
2613 (int)W_ENDCOL(wp) - 3, (int)W_ENDCOL(wp),
2614 '@', '@', HL_ATTR(HLF_AT));
2615 set_empty_rows(wp, srow);
2616 wp->w_botline = lnum;
2617 }
2618 else
2619 {
2620 win_draw_end(wp, '@', ' ', TRUE, srow, wp->w_height, HLF_AT);
2621 wp->w_botline = lnum;
2622 }
2623 }
2624 else
2625 {
2626 draw_vsep_win(wp, row);
2627 if (eof) // we hit the end of the file
2628 {
2629 wp->w_botline = buf->b_ml.ml_line_count + 1;
2630#ifdef FEAT_DIFF
2631 j = diff_check_fill(wp, wp->w_botline);
2632 if (j > 0 && !wp->w_botfill)
2633 {
2634 // Display filler lines at the end of the file.
2635 if (char2cells(fill_diff) > 1)
2636 i = '-';
2637 else
2638 i = fill_diff;
2639 if (row + j > wp->w_height)
2640 j = wp->w_height - row;
2641 win_draw_end(wp, i, i, TRUE, row, row + (int)j, HLF_DED);
2642 row += j;
2643 }
2644#endif
2645 }
2646 else if (dollar_vcol == -1)
2647 wp->w_botline = lnum;
2648
2649 // Make sure the rest of the screen is blank
Bram Moolenaara98f8a22021-02-13 18:24:23 +01002650 // write the 'fill_eob' character to rows that aren't part of the file
Bram Moolenaar1666ac92019-11-10 17:22:31 +01002651 if (WIN_IS_POPUP(wp))
2652 win_draw_end(wp, ' ', ' ', FALSE, row, wp->w_height, HLF_AT);
2653 else
Bram Moolenaara98f8a22021-02-13 18:24:23 +01002654 win_draw_end(wp, fill_eob, ' ', FALSE, row, wp->w_height, HLF_EOB);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002655 }
2656
2657#ifdef SYN_TIME_LIMIT
2658 syn_set_timeout(NULL);
2659#endif
2660
2661 // Reset the type of redrawing required, the window has been updated.
2662 wp->w_redr_type = 0;
2663#ifdef FEAT_DIFF
2664 wp->w_old_topfill = wp->w_topfill;
2665 wp->w_old_botfill = wp->w_botfill;
2666#endif
2667
2668 if (dollar_vcol == -1)
2669 {
2670 // There is a trick with w_botline. If we invalidate it on each
2671 // change that might modify it, this will cause a lot of expensive
2672 // calls to plines() in update_topline() each time. Therefore the
2673 // value of w_botline is often approximated, and this value is used to
2674 // compute the value of w_topline. If the value of w_botline was
2675 // wrong, check that the value of w_topline is correct (cursor is on
2676 // the visible part of the text). If it's not, we need to redraw
2677 // again. Mostly this just means scrolling up a few lines, so it
2678 // doesn't look too bad. Only do this for the current window (where
2679 // changes are relevant).
2680 wp->w_valid |= VALID_BOTLINE;
2681 if (wp == curwin && wp->w_botline != old_botline && !recursive)
2682 {
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002683 win_T *wwp;
2684#if defined(FEAT_CONCEAL)
2685 linenr_T old_topline = wp->w_topline;
2686 int new_wcol = wp->w_wcol;
2687#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002688 recursive = TRUE;
2689 curwin->w_valid &= ~VALID_TOPLINE;
2690 update_topline(); // may invalidate w_botline again
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002691
2692#if defined(FEAT_CONCEAL)
2693 if (old_wcol != new_wcol && (wp->w_valid & (VALID_WCOL|VALID_WROW))
2694 != (VALID_WCOL|VALID_WROW))
2695 {
2696 // A win_line() call applied a fix to screen cursor column to
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002697 // accommodate concealment of cursor line, but in this call to
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002698 // update_topline() the cursor's row or column got invalidated.
2699 // If they are left invalid, setcursor() will recompute them
2700 // but there won't be any further win_line() call to re-fix the
2701 // column and the cursor will end up misplaced. So we call
2702 // cursor validation now and reapply the fix again (or call
2703 // win_line() to do it for us).
2704 validate_cursor();
2705 if (wp->w_wcol == old_wcol && wp->w_wrow == old_wrow
2706 && old_topline == wp->w_topline)
2707 wp->w_wcol = new_wcol;
2708 else
2709 redrawWinline(wp, wp->w_cursor.lnum);
2710 }
2711#endif
2712 // New redraw either due to updated topline or due to wcol fix.
2713 if (wp->w_redr_type != 0)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002714 {
2715 // Don't update for changes in buffer again.
2716 i = curbuf->b_mod_set;
2717 curbuf->b_mod_set = FALSE;
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002718 j = curbuf->b_mod_xlines;
2719 curbuf->b_mod_xlines = 0;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002720 win_update(curwin);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002721 curbuf->b_mod_set = i;
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002722 curbuf->b_mod_xlines = j;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002723 }
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002724 // Other windows might have w_redr_type raised in update_topline().
2725 must_redraw = 0;
2726 FOR_ALL_WINDOWS(wwp)
2727 if (wwp->w_redr_type > must_redraw)
2728 must_redraw = wwp->w_redr_type;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002729 recursive = FALSE;
2730 }
2731 }
2732
2733#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
2734 // restore got_int, unless CTRL-C was hit while redrawing
2735 if (!got_int)
2736 got_int = save_got_int;
2737#endif
2738}
2739
2740#if defined(FEAT_NETBEANS_INTG) || defined(FEAT_GUI)
2741/*
2742 * Prepare for updating one or more windows.
2743 * Caller must check for "updating_screen" already set to avoid recursiveness.
2744 */
2745 static void
2746update_prepare(void)
2747{
2748 cursor_off();
2749 updating_screen = TRUE;
2750#ifdef FEAT_GUI
2751 // Remove the cursor before starting to do anything, because scrolling may
2752 // make it difficult to redraw the text under it.
2753 if (gui.in_use)
2754 gui_undraw_cursor();
2755#endif
2756#ifdef FEAT_SEARCH_EXTRA
2757 start_search_hl();
2758#endif
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002759#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002760 // Update popup_mask if needed.
2761 may_update_popup_mask(must_redraw);
2762#endif
2763}
2764
2765/*
2766 * Finish updating one or more windows.
2767 */
2768 static void
2769update_finish(void)
2770{
2771 if (redraw_cmdline || redraw_mode)
2772 showmode();
2773
2774# ifdef FEAT_SEARCH_EXTRA
2775 end_search_hl();
2776# endif
2777
2778 after_updating_screen(TRUE);
2779
2780# ifdef FEAT_GUI
2781 // Redraw the cursor and update the scrollbars when all screen updating is
2782 // done.
2783 if (gui.in_use)
2784 {
2785 out_flush_cursor(FALSE, FALSE);
2786 gui_update_scrollbars(FALSE);
2787 }
2788# endif
2789}
2790#endif
2791
2792#if defined(FEAT_NETBEANS_INTG) || defined(PROTO)
2793 void
2794update_debug_sign(buf_T *buf, linenr_T lnum)
2795{
2796 win_T *wp;
2797 int doit = FALSE;
2798
2799# ifdef FEAT_FOLDING
2800 win_foldinfo.fi_level = 0;
2801# endif
2802
2803 // update/delete a specific sign
2804 redraw_buf_line_later(buf, lnum);
2805
2806 // check if it resulted in the need to redraw a window
2807 FOR_ALL_WINDOWS(wp)
2808 if (wp->w_redr_type != 0)
2809 doit = TRUE;
2810
2811 // Return when there is nothing to do, screen updating is already
2812 // happening (recursive call), messages on the screen or still starting up.
2813 if (!doit || updating_screen
2814 || State == ASKMORE || State == HITRETURN
2815 || msg_scrolled
2816#ifdef FEAT_GUI
2817 || gui.starting
2818#endif
2819 || starting)
2820 return;
2821
2822 // update all windows that need updating
2823 update_prepare();
2824
2825 FOR_ALL_WINDOWS(wp)
2826 {
2827 if (wp->w_redr_type != 0)
2828 win_update(wp);
2829 if (wp->w_redr_status)
2830 win_redr_status(wp, FALSE);
2831 }
2832
2833 update_finish();
2834}
2835#endif
2836
2837#if defined(FEAT_GUI) || defined(PROTO)
2838/*
2839 * Update a single window, its status line and maybe the command line msg.
2840 * Used for the GUI scrollbar.
2841 */
2842 void
2843updateWindow(win_T *wp)
2844{
2845 // return if already busy updating
2846 if (updating_screen)
2847 return;
2848
2849 update_prepare();
2850
2851#ifdef FEAT_CLIPBOARD
2852 // When Visual area changed, may have to update selection.
2853 if (clip_star.available && clip_isautosel_star())
2854 clip_update_selection(&clip_star);
2855 if (clip_plus.available && clip_isautosel_plus())
2856 clip_update_selection(&clip_plus);
2857#endif
2858
2859 win_update(wp);
2860
2861 // When the screen was cleared redraw the tab pages line.
2862 if (redraw_tabline)
2863 draw_tabline();
2864
2865 if (wp->w_redr_status
2866# ifdef FEAT_CMDL_INFO
2867 || p_ru
2868# endif
2869# ifdef FEAT_STL_OPT
2870 || *p_stl != NUL || *wp->w_p_stl != NUL
2871# endif
2872 )
2873 win_redr_status(wp, FALSE);
2874
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002875#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002876 // Display popup windows on top of everything.
2877 update_popups(win_update);
2878#endif
2879
2880 update_finish();
2881}
2882#endif
2883
2884#if defined(FEAT_TERMRESPONSE) || defined(PROTO)
2885/*
2886 * Redraw as soon as possible. When the command line is not scrolled redraw
2887 * right away and restore what was on the command line.
2888 * Return a code indicating what happened.
2889 */
2890 int
2891redraw_asap(int type)
2892{
2893 int rows;
2894 int cols = screen_Columns;
2895 int r;
2896 int ret = 0;
2897 schar_T *screenline; // copy from ScreenLines[]
2898 sattr_T *screenattr; // copy from ScreenAttrs[]
2899 int i;
2900 u8char_T *screenlineUC = NULL; // copy from ScreenLinesUC[]
2901 u8char_T *screenlineC[MAX_MCO]; // copy from ScreenLinesC[][]
2902 schar_T *screenline2 = NULL; // copy from ScreenLines2[]
2903
2904 redraw_later(type);
2905 if (msg_scrolled || (State != NORMAL && State != NORMAL_BUSY) || exiting)
2906 return ret;
2907
2908 // Allocate space to save the text displayed in the command line area.
2909 rows = screen_Rows - cmdline_row;
2910 screenline = LALLOC_MULT(schar_T, rows * cols);
2911 screenattr = LALLOC_MULT(sattr_T, rows * cols);
2912 if (screenline == NULL || screenattr == NULL)
2913 ret = 2;
2914 if (enc_utf8)
2915 {
2916 screenlineUC = LALLOC_MULT(u8char_T, rows * cols);
2917 if (screenlineUC == NULL)
2918 ret = 2;
2919 for (i = 0; i < p_mco; ++i)
2920 {
2921 screenlineC[i] = LALLOC_MULT(u8char_T, rows * cols);
2922 if (screenlineC[i] == NULL)
2923 ret = 2;
2924 }
2925 }
2926 if (enc_dbcs == DBCS_JPNU)
2927 {
2928 screenline2 = LALLOC_MULT(schar_T, rows * cols);
2929 if (screenline2 == NULL)
2930 ret = 2;
2931 }
2932
2933 if (ret != 2)
2934 {
2935 // Save the text displayed in the command line area.
2936 for (r = 0; r < rows; ++r)
2937 {
2938 mch_memmove(screenline + r * cols,
2939 ScreenLines + LineOffset[cmdline_row + r],
2940 (size_t)cols * sizeof(schar_T));
2941 mch_memmove(screenattr + r * cols,
2942 ScreenAttrs + LineOffset[cmdline_row + r],
2943 (size_t)cols * sizeof(sattr_T));
2944 if (enc_utf8)
2945 {
2946 mch_memmove(screenlineUC + r * cols,
2947 ScreenLinesUC + LineOffset[cmdline_row + r],
2948 (size_t)cols * sizeof(u8char_T));
2949 for (i = 0; i < p_mco; ++i)
2950 mch_memmove(screenlineC[i] + r * cols,
2951 ScreenLinesC[i] + LineOffset[cmdline_row + r],
2952 (size_t)cols * sizeof(u8char_T));
2953 }
2954 if (enc_dbcs == DBCS_JPNU)
2955 mch_memmove(screenline2 + r * cols,
2956 ScreenLines2 + LineOffset[cmdline_row + r],
2957 (size_t)cols * sizeof(schar_T));
2958 }
2959
2960 update_screen(0);
2961 ret = 3;
2962
2963 if (must_redraw == 0)
2964 {
2965 int off = (int)(current_ScreenLine - ScreenLines);
2966
2967 // Restore the text displayed in the command line area.
2968 for (r = 0; r < rows; ++r)
2969 {
2970 mch_memmove(current_ScreenLine,
2971 screenline + r * cols,
2972 (size_t)cols * sizeof(schar_T));
2973 mch_memmove(ScreenAttrs + off,
2974 screenattr + r * cols,
2975 (size_t)cols * sizeof(sattr_T));
2976 if (enc_utf8)
2977 {
2978 mch_memmove(ScreenLinesUC + off,
2979 screenlineUC + r * cols,
2980 (size_t)cols * sizeof(u8char_T));
2981 for (i = 0; i < p_mco; ++i)
2982 mch_memmove(ScreenLinesC[i] + off,
2983 screenlineC[i] + r * cols,
2984 (size_t)cols * sizeof(u8char_T));
2985 }
2986 if (enc_dbcs == DBCS_JPNU)
2987 mch_memmove(ScreenLines2 + off,
2988 screenline2 + r * cols,
2989 (size_t)cols * sizeof(schar_T));
2990 screen_line(cmdline_row + r, 0, cols, cols, 0);
2991 }
2992 ret = 4;
2993 }
2994 }
2995
2996 vim_free(screenline);
2997 vim_free(screenattr);
2998 if (enc_utf8)
2999 {
3000 vim_free(screenlineUC);
3001 for (i = 0; i < p_mco; ++i)
3002 vim_free(screenlineC[i]);
3003 }
3004 if (enc_dbcs == DBCS_JPNU)
3005 vim_free(screenline2);
3006
3007 // Show the intro message when appropriate.
3008 maybe_intro_message();
3009
3010 setcursor();
3011
3012 return ret;
3013}
3014#endif
3015
3016/*
3017 * Invoked after an asynchronous callback is called.
3018 * If an echo command was used the cursor needs to be put back where
3019 * it belongs. If highlighting was changed a redraw is needed.
3020 * If "call_update_screen" is FALSE don't call update_screen() when at the
3021 * command line.
3022 */
3023 void
3024redraw_after_callback(int call_update_screen)
3025{
3026 ++redrawing_for_callback;
3027
3028 if (State == HITRETURN || State == ASKMORE)
3029 ; // do nothing
3030 else if (State & CMDLINE)
3031 {
3032 // Don't redraw when in prompt_for_number().
3033 if (cmdline_row > 0)
3034 {
3035 // Redrawing only works when the screen didn't scroll. Don't clear
3036 // wildmenu entries.
3037 if (msg_scrolled == 0
3038#ifdef FEAT_WILDMENU
3039 && wild_menu_showing == 0
3040#endif
3041 && call_update_screen)
3042 update_screen(0);
3043
3044 // Redraw in the same position, so that the user can continue
3045 // editing the command.
3046 redrawcmdline_ex(FALSE);
3047 }
3048 }
3049 else if (State & (NORMAL | INSERT | TERMINAL))
3050 {
3051 // keep the command line if possible
3052 update_screen(VALID_NO_UPDATE);
3053 setcursor();
Bram Moolenaar9f284162021-04-22 21:39:30 +02003054
3055 if (msg_scrolled == 0)
3056 {
3057 // don't want a hit-enter prompt when something else is displayed
3058 msg_didany = FALSE;
3059 need_wait_return = FALSE;
3060 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003061 }
3062 cursor_on();
3063#ifdef FEAT_GUI
3064 if (gui.in_use && !gui_mch_is_blink_off())
3065 // Don't update the cursor when it is blinking and off to avoid
3066 // flicker.
3067 out_flush_cursor(FALSE, FALSE);
3068 else
3069#endif
3070 out_flush();
3071
3072 --redrawing_for_callback;
3073}
3074
3075/*
3076 * Redraw the current window later, with update_screen(type).
3077 * Set must_redraw only if not already set to a higher value.
3078 * E.g. if must_redraw is CLEAR, type NOT_VALID will do nothing.
3079 */
3080 void
3081redraw_later(int type)
3082{
3083 redraw_win_later(curwin, type);
3084}
3085
3086 void
3087redraw_win_later(
3088 win_T *wp,
3089 int type)
3090{
3091 if (!exiting && wp->w_redr_type < type)
3092 {
3093 wp->w_redr_type = type;
3094 if (type >= NOT_VALID)
3095 wp->w_lines_valid = 0;
3096 if (must_redraw < type) // must_redraw is the maximum of all windows
3097 must_redraw = type;
3098 }
3099}
3100
3101/*
3102 * Force a complete redraw later. Also resets the highlighting. To be used
3103 * after executing a shell command that messes up the screen.
3104 */
3105 void
3106redraw_later_clear(void)
3107{
3108 redraw_all_later(CLEAR);
3109 reset_screen_attr();
3110}
3111
3112/*
3113 * Mark all windows to be redrawn later.
3114 */
3115 void
3116redraw_all_later(int type)
3117{
3118 win_T *wp;
3119
3120 FOR_ALL_WINDOWS(wp)
3121 redraw_win_later(wp, type);
3122 // This may be needed when switching tabs.
3123 if (must_redraw < type)
3124 must_redraw = type;
3125}
3126
3127/*
3128 * Mark all windows that are editing the current buffer to be updated later.
3129 */
3130 void
3131redraw_curbuf_later(int type)
3132{
3133 redraw_buf_later(curbuf, type);
3134}
3135
3136 void
3137redraw_buf_later(buf_T *buf, int type)
3138{
3139 win_T *wp;
3140
3141 FOR_ALL_WINDOWS(wp)
3142 {
3143 if (wp->w_buffer == buf)
3144 redraw_win_later(wp, type);
3145 }
Bram Moolenaare52e0c82020-02-28 22:20:10 +01003146#if defined(FEAT_TERMINAL) && defined(FEAT_PROP_POPUP)
3147 // terminal in popup window is not in list of windows
3148 if (curwin->w_buffer == buf)
3149 redraw_win_later(curwin, type);
3150#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003151}
3152
3153#if defined(FEAT_SIGNS) || defined(PROTO)
3154 void
3155redraw_buf_line_later(buf_T *buf, linenr_T lnum)
3156{
3157 win_T *wp;
3158
3159 FOR_ALL_WINDOWS(wp)
3160 if (wp->w_buffer == buf && lnum >= wp->w_topline
3161 && lnum < wp->w_botline)
3162 redrawWinline(wp, lnum);
3163}
3164#endif
3165
3166#if defined(FEAT_JOB_CHANNEL) || defined(PROTO)
3167 void
3168redraw_buf_and_status_later(buf_T *buf, int type)
3169{
3170 win_T *wp;
3171
3172#ifdef FEAT_WILDMENU
3173 if (wild_menu_showing != 0)
3174 // Don't redraw while the command line completion is displayed, it
3175 // would disappear.
3176 return;
3177#endif
3178 FOR_ALL_WINDOWS(wp)
3179 {
3180 if (wp->w_buffer == buf)
3181 {
3182 redraw_win_later(wp, type);
3183 wp->w_redr_status = TRUE;
3184 }
3185 }
3186}
3187#endif
3188
3189/*
3190 * mark all status lines for redraw; used after first :cd
3191 */
3192 void
3193status_redraw_all(void)
3194{
3195 win_T *wp;
3196
3197 FOR_ALL_WINDOWS(wp)
3198 if (wp->w_status_height)
3199 {
3200 wp->w_redr_status = TRUE;
3201 redraw_later(VALID);
3202 }
3203}
3204
3205/*
3206 * mark all status lines of the current buffer for redraw
3207 */
3208 void
3209status_redraw_curbuf(void)
3210{
3211 win_T *wp;
3212
3213 FOR_ALL_WINDOWS(wp)
3214 if (wp->w_status_height != 0 && wp->w_buffer == curbuf)
3215 {
3216 wp->w_redr_status = TRUE;
3217 redraw_later(VALID);
3218 }
3219}
3220
3221/*
3222 * Redraw all status lines that need to be redrawn.
3223 */
3224 void
3225redraw_statuslines(void)
3226{
3227 win_T *wp;
3228
3229 FOR_ALL_WINDOWS(wp)
3230 if (wp->w_redr_status)
3231 win_redr_status(wp, FALSE);
3232 if (redraw_tabline)
3233 draw_tabline();
3234}
3235
3236#if defined(FEAT_WILDMENU) || defined(PROTO)
3237/*
3238 * Redraw all status lines at the bottom of frame "frp".
3239 */
3240 void
3241win_redraw_last_status(frame_T *frp)
3242{
3243 if (frp->fr_layout == FR_LEAF)
3244 frp->fr_win->w_redr_status = TRUE;
3245 else if (frp->fr_layout == FR_ROW)
3246 {
3247 FOR_ALL_FRAMES(frp, frp->fr_child)
3248 win_redraw_last_status(frp);
3249 }
3250 else // frp->fr_layout == FR_COL
3251 {
3252 frp = frp->fr_child;
3253 while (frp->fr_next != NULL)
3254 frp = frp->fr_next;
3255 win_redraw_last_status(frp);
3256 }
3257}
3258#endif
3259
3260/*
3261 * Changed something in the current window, at buffer line "lnum", that
3262 * requires that line and possibly other lines to be redrawn.
3263 * Used when entering/leaving Insert mode with the cursor on a folded line.
3264 * Used to remove the "$" from a change command.
3265 * Note that when also inserting/deleting lines w_redraw_top and w_redraw_bot
3266 * may become invalid and the whole window will have to be redrawn.
3267 */
3268 void
3269redrawWinline(
3270 win_T *wp,
3271 linenr_T lnum)
3272{
3273 if (wp->w_redraw_top == 0 || wp->w_redraw_top > lnum)
3274 wp->w_redraw_top = lnum;
3275 if (wp->w_redraw_bot == 0 || wp->w_redraw_bot < lnum)
3276 wp->w_redraw_bot = lnum;
3277 redraw_win_later(wp, VALID);
3278}