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