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