blob: a0801e5865474b81606d28d0540e2bc0cc705be5 [file] [log] [blame]
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001/* vi:set ts=8 sts=4 sw=4 noet:
2 *
3 * VIM - Vi IMproved by Bram Moolenaar
4 *
5 * Do ":help uganda" in Vim to read copying and usage conditions.
6 * Do ":help credits" in Vim to see a list of people who contributed.
7 * See README.txt for an overview of the Vim source code.
8 */
9
10/*
11 * drawscreen.c: Code for updating all the windows on the screen.
12 * This is the top level, drawline.c is the middle and screen.c the lower
13 * level.
14 *
15 * update_screen() is the function that updates all windows and status lines.
16 * It is called form the main loop when must_redraw is non-zero. It may be
17 * called from other places when an immediate screen update is needed.
18 *
19 * The part of the buffer that is displayed in a window is set with:
20 * - w_topline (first buffer line in window)
21 * - w_topfill (filler lines above the first line)
22 * - w_leftcol (leftmost window cell in window),
23 * - w_skipcol (skipped window cells of first line)
24 *
25 * Commands that only move the cursor around in a window, do not need to take
26 * action to update the display. The main loop will check if w_topline is
27 * valid and update it (scroll the window) when needed.
28 *
29 * Commands that scroll a window change w_topline and must call
30 * check_cursor() to move the cursor into the visible part of the window, and
Bram Moolenaara4d158b2022-08-14 14:17:45 +010031 * call redraw_later(UPD_VALID) to have the window displayed by update_screen()
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020032 * later.
33 *
34 * Commands that change text in the buffer must call changed_bytes() or
35 * changed_lines() to mark the area that changed and will require updating
36 * later. The main loop will call update_screen(), which will update each
37 * window that shows the changed buffer. This assumes text above the change
38 * can remain displayed as it is. Text after the change may need updating for
39 * scrolling, folding and syntax highlighting.
40 *
41 * Commands that change how a window is displayed (e.g., setting 'list') or
42 * invalidate the contents of a window in another way (e.g., change fold
Bram Moolenaara4d158b2022-08-14 14:17:45 +010043 * settings), must call redraw_later(UPD_NOT_VALID) to have the whole window
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020044 * redisplayed by update_screen() later.
45 *
46 * Commands that change how a buffer is displayed (e.g., setting 'tabstop')
Bram Moolenaara4d158b2022-08-14 14:17:45 +010047 * must call redraw_curbuf_later(UPD_NOT_VALID) to have all the windows for the
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020048 * buffer redisplayed by update_screen() later.
49 *
50 * Commands that change highlighting and possibly cause a scroll too must call
Bram Moolenaara4d158b2022-08-14 14:17:45 +010051 * redraw_later(UPD_SOME_VALID) to update the whole window but still use
52 * scrolling to avoid redrawing everything. But the length of displayed lines
53 * must not change, use UPD_NOT_VALID then.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020054 *
Bram Moolenaara4d158b2022-08-14 14:17:45 +010055 * Commands that move the window position must call redraw_later(UPD_NOT_VALID).
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020056 * TODO: should minimize redrawing by scrolling when possible.
57 *
58 * Commands that change everything (e.g., resizing the screen) must call
Bram Moolenaara4d158b2022-08-14 14:17:45 +010059 * redraw_all_later(UPD_NOT_VALID) or redraw_all_later(UPD_CLEAR).
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020060 *
61 * Things that are handled indirectly:
62 * - When messages scroll the screen up, msg_scrolled will be set and
63 * update_screen() called to redraw.
64 */
65
66#include "vim.h"
67
68static void win_update(win_T *wp);
69#ifdef FEAT_STL_OPT
70static void redraw_custom_statusline(win_T *wp);
71#endif
Bram Moolenaare52e0c82020-02-28 22:20:10 +010072#if defined(FEAT_SEARCH_EXTRA) || defined(FEAT_CLIPBOARD)
73static int did_update_one_window;
74#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020075
Bram Moolenaarbdff0122020-04-05 18:56:05 +020076static void win_redr_status(win_T *wp, int ignore_pum);
77
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020078/*
79 * Based on the current value of curwin->w_topline, transfer a screenfull
80 * of stuff from Filemem to ScreenLines[], and update curwin->w_botline.
81 * Return OK when the screen was updated, FAIL if it was not done.
82 */
83 int
84update_screen(int type_arg)
85{
86 int type = type_arg;
87 win_T *wp;
88 static int did_intro = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020089#ifdef FEAT_GUI
Bram Moolenaare52e0c82020-02-28 22:20:10 +010090 int did_one = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020091 int did_undraw = FALSE;
92 int gui_cursor_col = 0;
93 int gui_cursor_row = 0;
94#endif
95 int no_update = FALSE;
Bram Moolenaare0c03c82021-04-23 21:01:34 +020096 int save_pum_will_redraw = pum_will_redraw;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +020097
98 // Don't do anything if the screen structures are (not yet) valid.
99 if (!screen_valid(TRUE))
100 return FAIL;
101
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100102 if (type == UPD_VALID_NO_UPDATE)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200103 {
104 no_update = TRUE;
105 type = 0;
106 }
107
108#ifdef FEAT_EVAL
109 {
110 buf_T *buf;
111
112 // Before updating the screen, notify any listeners of changed text.
113 FOR_ALL_BUFFERS(buf)
114 invoke_listeners(buf);
115 }
116#endif
117
118#ifdef FEAT_DIFF
119 // May have postponed updating diffs.
120 if (need_diff_redraw)
121 diff_redraw(TRUE);
122#endif
123
124 if (must_redraw)
125 {
126 if (type < must_redraw) // use maximal type
127 type = must_redraw;
128
129 // must_redraw is reset here, so that when we run into some weird
130 // reason to redraw while busy redrawing (e.g., asynchronous
131 // scrolling), or update_topline() in win_update() will cause a
132 // scroll, the screen will be redrawn later or in win_update().
133 must_redraw = 0;
134 }
135
136 // May need to update w_lines[].
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100137 if (curwin->w_lines_valid == 0 && type < UPD_NOT_VALID
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200138#ifdef FEAT_TERMINAL
139 && !term_do_update_window(curwin)
140#endif
141 )
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100142 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200143
144 // Postpone the redrawing when it's not needed and when being called
145 // recursively.
146 if (!redrawing() || updating_screen)
147 {
148 redraw_later(type); // remember type for next time
149 must_redraw = type;
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100150 if (type > UPD_INVERTED_ALL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200151 curwin->w_lines_valid = 0; // don't use w_lines[].wl_size now
152 return FAIL;
153 }
154 updating_screen = TRUE;
155
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +0100156#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200157 // Update popup_mask if needed. This may set w_redraw_top and w_redraw_bot
158 // in some windows.
159 may_update_popup_mask(type);
160#endif
161
162#ifdef FEAT_SYN_HL
163 ++display_tick; // let syntax code know we're in a next round of
164 // display updating
165#endif
166 if (no_update)
167 ++no_win_do_lines_ins;
168
169 // if the screen was scrolled up when displaying a message, scroll it down
170 if (msg_scrolled)
171 {
172 clear_cmdline = TRUE;
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100173 if (type != UPD_CLEAR)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200174 {
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100175 if (msg_scrolled > Rows - 5) // redrawing is faster
Bram Moolenaarb13d3402022-08-29 13:44:28 +0100176 {
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100177 type = UPD_NOT_VALID;
Bram Moolenaarb13d3402022-08-29 13:44:28 +0100178 redraw_as_cleared();
179 }
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100180 else
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200181 {
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100182 check_for_delay(FALSE);
183 if (screen_ins_lines(0, 0, msg_scrolled, (int)Rows, 0, NULL)
184 == FAIL)
Bram Moolenaarb13d3402022-08-29 13:44:28 +0100185 {
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100186 type = UPD_NOT_VALID;
Bram Moolenaarb13d3402022-08-29 13:44:28 +0100187 redraw_as_cleared();
188 }
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100189 FOR_ALL_WINDOWS(wp)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200190 {
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100191 if (wp->w_winrow < msg_scrolled)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200192 {
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100193 if (W_WINROW(wp) + wp->w_height > msg_scrolled
194 && wp->w_redr_type < UPD_REDRAW_TOP
195 && wp->w_lines_valid > 0
196 && wp->w_topline == wp->w_lines[0].wl_lnum)
197 {
198 wp->w_upd_rows = msg_scrolled - W_WINROW(wp);
199 wp->w_redr_type = UPD_REDRAW_TOP;
200 }
201 else
202 {
203 wp->w_redr_type = UPD_NOT_VALID;
204 if (W_WINROW(wp) + wp->w_height
205 + wp->w_status_height <= msg_scrolled)
206 wp->w_redr_status = TRUE;
207 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200208 }
209 }
Bram Moolenaarf73e5ba2022-08-29 12:41:06 +0100210 if (!no_update)
211 redraw_cmdline = TRUE;
212 redraw_tabline = TRUE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200213 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200214 }
215 msg_scrolled = 0;
216 need_wait_return = FALSE;
217 }
218
219 // reset cmdline_row now (may have been changed temporarily)
220 compute_cmdrow();
221
222 // Check for changed highlighting
223 if (need_highlight_changed)
224 highlight_changed();
225
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100226 if (type == UPD_CLEAR) // first clear screen
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200227 {
228 screenclear(); // will reset clear_cmdline
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100229 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200230 // must_redraw may be set indirectly, avoid another redraw later
231 must_redraw = 0;
232 }
233
234 if (clear_cmdline) // going to clear cmdline (done below)
235 check_for_delay(FALSE);
236
237#ifdef FEAT_LINEBREAK
238 // Force redraw when width of 'number' or 'relativenumber' column
239 // changes.
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100240 if (curwin->w_redr_type < UPD_NOT_VALID
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200241 && curwin->w_nrwidth != ((curwin->w_p_nu || curwin->w_p_rnu)
242 ? number_width(curwin) : 0))
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100243 curwin->w_redr_type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200244#endif
245
246 // Only start redrawing if there is really something to do.
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100247 if (type == UPD_INVERTED)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200248 update_curswant();
249 if (curwin->w_redr_type < type
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100250 && !((type == UPD_VALID
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200251 && curwin->w_lines[0].wl_valid
252#ifdef FEAT_DIFF
253 && curwin->w_topfill == curwin->w_old_topfill
254 && curwin->w_botfill == curwin->w_old_botfill
255#endif
256 && curwin->w_topline == curwin->w_lines[0].wl_lnum)
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100257 || (type == UPD_INVERTED
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200258 && VIsual_active
259 && curwin->w_old_cursor_lnum == curwin->w_cursor.lnum
260 && curwin->w_old_visual_mode == VIsual_mode
261 && (curwin->w_valid & VALID_VIRTCOL)
262 && curwin->w_old_curswant == curwin->w_curswant)
263 ))
264 curwin->w_redr_type = type;
265
266 // Redraw the tab pages line if needed.
Bram Moolenaara4d158b2022-08-14 14:17:45 +0100267 if (redraw_tabline || type >= UPD_NOT_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200268 draw_tabline();
269
270#ifdef FEAT_SYN_HL
271 // Correct stored syntax highlighting info for changes in each displayed
272 // buffer. Each buffer must only be done once.
273 FOR_ALL_WINDOWS(wp)
274 {
275 if (wp->w_buffer->b_mod_set)
276 {
277 win_T *wwp;
278
279 // Check if we already did this buffer.
280 for (wwp = firstwin; wwp != wp; wwp = wwp->w_next)
281 if (wwp->w_buffer == wp->w_buffer)
282 break;
283 if (wwp == wp && syntax_present(wp))
284 syn_stack_apply_changes(wp->w_buffer);
285 }
286 }
287#endif
288
Bram Moolenaare0c03c82021-04-23 21:01:34 +0200289 if (pum_redraw_in_same_position())
290 // Avoid flicker if the popup menu is going to be redrawn in the same
291 // position.
292 pum_will_redraw = TRUE;
293
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200294 // Go from top to bottom through the windows, redrawing the ones that need
295 // it.
296#if defined(FEAT_SEARCH_EXTRA) || defined(FEAT_CLIPBOARD)
Bram Moolenaare52e0c82020-02-28 22:20:10 +0100297 did_update_one_window = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200298#endif
299#ifdef FEAT_SEARCH_EXTRA
300 screen_search_hl.rm.regprog = NULL;
301#endif
302 FOR_ALL_WINDOWS(wp)
303 {
304 if (wp->w_redr_type != 0)
305 {
306 cursor_off();
Bram Moolenaare52e0c82020-02-28 22:20:10 +0100307#ifdef FEAT_GUI
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200308 if (!did_one)
309 {
310 did_one = TRUE;
Bram Moolenaare52e0c82020-02-28 22:20:10 +0100311
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200312 // Remove the cursor before starting to do anything, because
313 // scrolling may make it difficult to redraw the text under
314 // it.
Bram Moolenaar09f067f2021-04-11 13:29:18 +0200315 // Also remove the cursor if it needs to be hidden due to an
316 // ongoing cursor-less sleep.
317 if (gui.in_use && (wp == curwin || cursor_is_sleeping()))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200318 {
319 gui_cursor_col = gui.cursor_col;
320 gui_cursor_row = gui.cursor_row;
321 gui_undraw_cursor();
322 did_undraw = TRUE;
323 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200324 }
325#endif
326 win_update(wp);
327 }
328
329 // redraw status line after the window to minimize cursor movement
330 if (wp->w_redr_status)
331 {
332 cursor_off();
333 win_redr_status(wp, TRUE); // any popup menu will be redrawn below
334 }
335 }
336#if defined(FEAT_SEARCH_EXTRA)
337 end_search_hl();
338#endif
Bram Moolenaare0c03c82021-04-23 21:01:34 +0200339
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200340 // May need to redraw the popup menu.
Bram Moolenaare0c03c82021-04-23 21:01:34 +0200341 pum_will_redraw = save_pum_will_redraw;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200342 pum_may_redraw();
343
344 // Reset b_mod_set flags. Going through all windows is probably faster
345 // than going through all buffers (there could be many buffers).
346 FOR_ALL_WINDOWS(wp)
347 wp->w_buffer->b_mod_set = FALSE;
348
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +0100349#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200350 // Display popup windows on top of the windows and command line.
351 update_popups(win_update);
352#endif
353
Bram Moolenaar3194e5b2021-12-13 21:59:09 +0000354#ifdef FEAT_TERMINAL
355 FOR_ALL_WINDOWS(wp)
356 // If this window contains a terminal, after redrawing all windows, the
357 // dirty row range can be reset.
358 term_did_update_window(wp);
359#endif
360
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200361 after_updating_screen(TRUE);
362
363 // Clear or redraw the command line. Done last, because scrolling may
364 // mess up the command line.
365 if (clear_cmdline || redraw_cmdline || redraw_mode)
366 showmode();
367
368 if (no_update)
369 --no_win_do_lines_ins;
370
371 // May put up an introductory message when not editing a file
372 if (!did_intro)
373 maybe_intro_message();
374 did_intro = TRUE;
375
376#ifdef FEAT_GUI
377 // Redraw the cursor and update the scrollbars when all screen updating is
378 // done.
379 if (gui.in_use)
380 {
381 if (did_undraw && !gui_mch_is_blink_off())
382 {
383 mch_disable_flush();
384 out_flush(); // required before updating the cursor
385 mch_enable_flush();
386
387 // Put the GUI position where the cursor was, gui_update_cursor()
388 // uses that.
389 gui.col = gui_cursor_col;
390 gui.row = gui_cursor_row;
391 gui.col = mb_fix_col(gui.col, gui.row);
392 gui_update_cursor(FALSE, FALSE);
393 gui_may_flush();
394 screen_cur_col = gui.col;
395 screen_cur_row = gui.row;
396 }
397 else
398 out_flush();
399 gui_update_scrollbars(FALSE);
400 }
401#endif
402 return OK;
403}
404
405/*
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200406 * Return the row for drawing the statusline and the ruler of window "wp".
407 */
Bram Moolenaar49c51b82021-04-01 16:16:18 +0200408 int
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200409statusline_row(win_T *wp)
410{
411#if defined(FEAT_PROP_POPUP)
412 // If the window is really zero height the winbar isn't displayed.
413 if (wp->w_frame->fr_height == wp->w_status_height && !popup_is_popup(wp))
414 return wp->w_winrow;
415#endif
416 return W_WINROW(wp) + wp->w_height;
417}
418
419/*
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200420 * Redraw the status line of window wp.
421 *
422 * If inversion is possible we use it. Else '=' characters are used.
423 * If "ignore_pum" is TRUE, also redraw statusline when the popup menu is
424 * displayed.
425 */
Bram Moolenaarbdff0122020-04-05 18:56:05 +0200426 static void
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200427win_redr_status(win_T *wp, int ignore_pum UNUSED)
428{
429 int row;
430 char_u *p;
431 int len;
432 int fillchar;
433 int attr;
434 int this_ru_col;
435 static int busy = FALSE;
436
437 // It's possible to get here recursively when 'statusline' (indirectly)
438 // invokes ":redrawstatus". Simply ignore the call then.
439 if (busy)
440 return;
441 busy = TRUE;
442
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200443 row = statusline_row(wp);
444
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200445 wp->w_redr_status = FALSE;
446 if (wp->w_status_height == 0)
447 {
448 // no status line, can only be last window
449 redraw_cmdline = TRUE;
450 }
451 else if (!redrawing()
452 // don't update status line when popup menu is visible and may be
453 // drawn over it, unless it will be redrawn later
454 || (!ignore_pum && pum_visible()))
455 {
456 // Don't redraw right now, do it later.
457 wp->w_redr_status = TRUE;
458 }
459#ifdef FEAT_STL_OPT
460 else if (*p_stl != NUL || *wp->w_p_stl != NUL)
461 {
462 // redraw custom status line
463 redraw_custom_statusline(wp);
464 }
465#endif
466 else
467 {
468 fillchar = fillchar_status(&attr, wp);
469
470 get_trans_bufname(wp->w_buffer);
471 p = NameBuff;
472 len = (int)STRLEN(p);
473
Bram Moolenaarde05bb22022-01-13 13:08:14 +0000474 if ((bt_help(wp->w_buffer)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200475#ifdef FEAT_QUICKFIX
Bram Moolenaarde05bb22022-01-13 13:08:14 +0000476 || wp->w_p_pvw
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200477#endif
Bram Moolenaarde05bb22022-01-13 13:08:14 +0000478 || bufIsChanged(wp->w_buffer)
479 || wp->w_buffer->b_p_ro)
480 && len < MAXPATHL - 1)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200481 *(p + len++) = ' ';
482 if (bt_help(wp->w_buffer))
483 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100484 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[Help]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200485 len += (int)STRLEN(p + len);
486 }
487#ifdef FEAT_QUICKFIX
488 if (wp->w_p_pvw)
489 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100490 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[Preview]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200491 len += (int)STRLEN(p + len);
492 }
493#endif
Bram Moolenaar6d4b2f52022-08-25 15:11:15 +0100494 if (bufIsChanged(wp->w_buffer) && !bt_terminal(wp->w_buffer))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200495 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100496 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", "[+]");
497 len += (int)STRLEN(p + len);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200498 }
499 if (wp->w_buffer->b_p_ro)
500 {
Bram Moolenaar826bfe42021-10-08 18:39:28 +0100501 vim_snprintf((char *)p + len, MAXPATHL - len, "%s", _("[RO]"));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200502 len += (int)STRLEN(p + len);
503 }
504
505 this_ru_col = ru_col - (Columns - wp->w_width);
506 if (this_ru_col < (wp->w_width + 1) / 2)
507 this_ru_col = (wp->w_width + 1) / 2;
508 if (this_ru_col <= 1)
509 {
510 p = (char_u *)"<"; // No room for file name!
511 len = 1;
512 }
513 else if (has_mbyte)
514 {
515 int clen = 0, i;
516
517 // Count total number of display cells.
518 clen = mb_string2cells(p, -1);
519
520 // Find first character that will fit.
521 // Going from start to end is much faster for DBCS.
522 for (i = 0; p[i] != NUL && clen >= this_ru_col - 1;
523 i += (*mb_ptr2len)(p + i))
524 clen -= (*mb_ptr2cells)(p + i);
525 len = clen;
526 if (i > 0)
527 {
528 p = p + i - 1;
529 *p = '<';
530 ++len;
531 }
532
533 }
534 else if (len > this_ru_col - 1)
535 {
536 p += len - (this_ru_col - 1);
537 *p = '<';
538 len = this_ru_col - 1;
539 }
540
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200541 screen_puts(p, row, wp->w_wincol, attr);
542 screen_fill(row, row + 1, len + wp->w_wincol,
543 this_ru_col + wp->w_wincol, fillchar, fillchar, attr);
544
545 if (get_keymap_str(wp, (char_u *)"<%s>", NameBuff, MAXPATHL)
=?UTF-8?q?Dundar=20G=C3=B6c?=420fabc2022-01-28 15:28:04 +0000546 && (this_ru_col - len) > (int)(STRLEN(NameBuff) + 1))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200547 screen_puts(NameBuff, row, (int)(this_ru_col - STRLEN(NameBuff)
548 - 1 + wp->w_wincol), attr);
549
550#ifdef FEAT_CMDL_INFO
551 win_redr_ruler(wp, TRUE, ignore_pum);
552#endif
553 }
554
555 /*
556 * May need to draw the character below the vertical separator.
557 */
558 if (wp->w_vsep_width != 0 && wp->w_status_height != 0 && redrawing())
559 {
560 if (stl_connected(wp))
561 fillchar = fillchar_status(&attr, wp);
562 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +0100563 fillchar = fillchar_vsep(&attr, wp);
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200564 screen_putchar(fillchar, row, W_ENDCOL(wp), attr);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200565 }
566 busy = FALSE;
567}
568
569#ifdef FEAT_STL_OPT
570/*
571 * Redraw the status line according to 'statusline' and take care of any
572 * errors encountered.
573 */
574 static void
575redraw_custom_statusline(win_T *wp)
576{
577 static int entered = FALSE;
578 int saved_did_emsg = did_emsg;
579
580 // When called recursively return. This can happen when the statusline
581 // contains an expression that triggers a redraw.
582 if (entered)
583 return;
584 entered = TRUE;
585
586 did_emsg = FALSE;
587 win_redr_custom(wp, FALSE);
588 if (did_emsg)
589 {
590 // When there is an error disable the statusline, otherwise the
591 // display is messed up with errors and a redraw triggers the problem
592 // again and again.
593 set_string_option_direct((char_u *)"statusline", -1,
594 (char_u *)"", OPT_FREE | (*wp->w_p_stl != NUL
595 ? OPT_LOCAL : OPT_GLOBAL), SID_ERROR);
596 }
597 did_emsg |= saved_did_emsg;
598 entered = FALSE;
599}
600#endif
601
602/*
603 * Show current status info in ruler and various other places
604 * If always is FALSE, only show ruler if position has changed.
605 */
606 void
607showruler(int always)
608{
609 if (!always && !redrawing())
610 return;
611 if (pum_visible())
612 {
613 // Don't redraw right now, do it later.
614 curwin->w_redr_status = TRUE;
615 return;
616 }
617#if defined(FEAT_STL_OPT)
618 if ((*p_stl != NUL || *curwin->w_p_stl != NUL) && curwin->w_status_height)
619 redraw_custom_statusline(curwin);
620 else
621#endif
622#ifdef FEAT_CMDL_INFO
623 win_redr_ruler(curwin, always, FALSE);
624#endif
625
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200626 if (need_maketitle
Bram Moolenaar651fca82021-11-29 20:39:38 +0000627#ifdef FEAT_STL_OPT
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200628 || (p_icon && (stl_syntax & STL_IN_ICON))
629 || (p_title && (stl_syntax & STL_IN_TITLE))
Bram Moolenaar651fca82021-11-29 20:39:38 +0000630#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200631 )
632 maketitle();
Bram Moolenaar651fca82021-11-29 20:39:38 +0000633
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200634 // Redraw the tab pages line if needed.
635 if (redraw_tabline)
636 draw_tabline();
637}
638
639#if defined(FEAT_CMDL_INFO) || defined(PROTO)
640 void
641win_redr_ruler(win_T *wp, int always, int ignore_pum)
642{
643#define RULER_BUF_LEN 70
644 char_u buffer[RULER_BUF_LEN];
645 int row;
646 int fillchar;
647 int attr;
648 int empty_line = FALSE;
649 colnr_T virtcol;
650 int i;
651 size_t len;
652 int o;
653 int this_ru_col;
654 int off = 0;
655 int width;
656
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +0100657 // If 'ruler' off or messages area disabled, don't do anything
658 if (!p_ru || (wp->w_status_height == 0 && p_ch == 0))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200659 return;
660
661 /*
662 * Check if cursor.lnum is valid, since win_redr_ruler() may be called
663 * after deleting lines, before cursor.lnum is corrected.
664 */
665 if (wp->w_cursor.lnum > wp->w_buffer->b_ml.ml_line_count)
666 return;
667
668 // Don't draw the ruler while doing insert-completion, it might overwrite
669 // the (long) mode message.
670 if (wp == lastwin && lastwin->w_status_height == 0)
671 if (edit_submode != NULL)
672 return;
673 // Don't draw the ruler when the popup menu is visible, it may overlap.
674 // Except when the popup menu will be redrawn anyway.
675 if (!ignore_pum && pum_visible())
676 return;
677
678#ifdef FEAT_STL_OPT
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +0100679 if (*p_ruf && p_ch > 0)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200680 {
Bram Moolenaar53989552019-12-23 22:59:18 +0100681 int called_emsg_before = called_emsg;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200682
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200683 win_redr_custom(wp, TRUE);
Bram Moolenaar53989552019-12-23 22:59:18 +0100684 if (called_emsg > called_emsg_before)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200685 set_string_option_direct((char_u *)"rulerformat", -1,
686 (char_u *)"", OPT_FREE, SID_ERROR);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200687 return;
688 }
689#endif
690
691 /*
692 * Check if not in Insert mode and the line is empty (will show "0-1").
693 */
Bram Moolenaar24959102022-05-07 20:01:16 +0100694 if ((State & MODE_INSERT) == 0
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200695 && *ml_get_buf(wp->w_buffer, wp->w_cursor.lnum, FALSE) == NUL)
696 empty_line = TRUE;
697
698 /*
699 * Only draw the ruler when something changed.
700 */
701 validate_virtcol_win(wp);
702 if ( redraw_cmdline
703 || always
704 || wp->w_cursor.lnum != wp->w_ru_cursor.lnum
705 || wp->w_cursor.col != wp->w_ru_cursor.col
706 || wp->w_virtcol != wp->w_ru_virtcol
707 || wp->w_cursor.coladd != wp->w_ru_cursor.coladd
708 || wp->w_topline != wp->w_ru_topline
709 || wp->w_buffer->b_ml.ml_line_count != wp->w_ru_line_count
710#ifdef FEAT_DIFF
711 || wp->w_topfill != wp->w_ru_topfill
712#endif
713 || empty_line != wp->w_ru_empty)
714 {
715 cursor_off();
716 if (wp->w_status_height)
717 {
Bram Moolenaarae0f1512021-03-30 22:12:12 +0200718 row = statusline_row(wp);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200719 fillchar = fillchar_status(&attr, wp);
720 off = wp->w_wincol;
721 width = wp->w_width;
722 }
723 else
724 {
725 row = Rows - 1;
726 fillchar = ' ';
727 attr = 0;
728 width = Columns;
729 off = 0;
730 }
731
732 // In list mode virtcol needs to be recomputed
733 virtcol = wp->w_virtcol;
Bram Moolenaareed9d462021-02-15 20:38:25 +0100734 if (wp->w_p_list && wp->w_lcs_chars.tab1 == NUL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200735 {
736 wp->w_p_list = FALSE;
737 getvvcol(wp, &wp->w_cursor, NULL, &virtcol, NULL);
738 wp->w_p_list = TRUE;
739 }
740
741 /*
742 * Some sprintfs return the length, some return a pointer.
743 * To avoid portability problems we use strlen() here.
744 */
745 vim_snprintf((char *)buffer, RULER_BUF_LEN, "%ld,",
746 (wp->w_buffer->b_ml.ml_flags & ML_EMPTY)
747 ? 0L
748 : (long)(wp->w_cursor.lnum));
749 len = STRLEN(buffer);
750 col_print(buffer + len, RULER_BUF_LEN - len,
751 empty_line ? 0 : (int)wp->w_cursor.col + 1,
752 (int)virtcol + 1);
753
754 /*
755 * Add a "50%" if there is room for it.
756 * On the last line, don't print in the last column (scrolls the
757 * screen up on some terminals).
758 */
759 i = (int)STRLEN(buffer);
760 get_rel_pos(wp, buffer + i + 1, RULER_BUF_LEN - i - 1);
761 o = i + vim_strsize(buffer + i + 1);
762 if (wp->w_status_height == 0) // can't use last char of screen
763 ++o;
764 this_ru_col = ru_col - (Columns - width);
765 if (this_ru_col < 0)
766 this_ru_col = 0;
767 // Never use more than half the window/screen width, leave the other
768 // half for the filename.
769 if (this_ru_col < (width + 1) / 2)
770 this_ru_col = (width + 1) / 2;
771 if (this_ru_col + o < width)
772 {
773 // need at least 3 chars left for get_rel_pos() + NUL
774 while (this_ru_col + o < width && RULER_BUF_LEN > i + 4)
775 {
776 if (has_mbyte)
777 i += (*mb_char2bytes)(fillchar, buffer + i);
778 else
779 buffer[i++] = fillchar;
780 ++o;
781 }
782 get_rel_pos(wp, buffer + i, RULER_BUF_LEN - i);
783 }
784 // Truncate at window boundary.
785 if (has_mbyte)
786 {
787 o = 0;
788 for (i = 0; buffer[i] != NUL; i += (*mb_ptr2len)(buffer + i))
789 {
790 o += (*mb_ptr2cells)(buffer + i);
791 if (this_ru_col + o > width)
792 {
793 buffer[i] = NUL;
794 break;
795 }
796 }
797 }
798 else if (this_ru_col + (int)STRLEN(buffer) > width)
799 buffer[width - this_ru_col] = NUL;
800
801 screen_puts(buffer, row, this_ru_col + off, attr);
802 i = redraw_cmdline;
803 screen_fill(row, row + 1,
804 this_ru_col + off + (int)STRLEN(buffer),
=?UTF-8?q?Dundar=20G=C3=B6c?=420fabc2022-01-28 15:28:04 +0000805 (off + width),
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200806 fillchar, fillchar, attr);
807 // don't redraw the cmdline because of showing the ruler
808 redraw_cmdline = i;
809 wp->w_ru_cursor = wp->w_cursor;
810 wp->w_ru_virtcol = wp->w_virtcol;
811 wp->w_ru_empty = empty_line;
812 wp->w_ru_topline = wp->w_topline;
813 wp->w_ru_line_count = wp->w_buffer->b_ml.ml_line_count;
814#ifdef FEAT_DIFF
815 wp->w_ru_topfill = wp->w_topfill;
816#endif
817 }
818}
819#endif
820
821/*
822 * To be called when "updating_screen" was set before and now the postponed
823 * side effects may take place.
824 */
825 void
826after_updating_screen(int may_resize_shell UNUSED)
827{
828 updating_screen = FALSE;
829#ifdef FEAT_GUI
830 if (may_resize_shell)
831 gui_may_resize_shell();
832#endif
833#ifdef FEAT_TERMINAL
834 term_check_channel_closed_recently();
835#endif
836
837#ifdef HAVE_DROP_FILE
838 // If handle_drop() was called while updating_screen was TRUE need to
839 // handle the drop now.
840 handle_any_postponed_drop();
841#endif
Bram Moolenaar7a99da42022-08-28 22:21:01 +0100842
Bram Moolenaar926218b2022-08-28 23:36:52 +0100843 if (p_ch == 0)
844 // in case it was changed in dont_use_message_window()
845 cmdline_row = Rows;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +0200846}
847
848/*
849 * Update all windows that are editing the current buffer.
850 */
851 void
852update_curbuf(int type)
853{
854 redraw_curbuf_later(type);
855 update_screen(type);
856}
857
858#if defined(FEAT_MENU) || defined(FEAT_FOLDING)
859/*
860 * Copy "text" to ScreenLines using "attr".
861 * Returns the next screen column.
862 */
863 static int
864text_to_screenline(win_T *wp, char_u *text, int col)
865{
866 int off = (int)(current_ScreenLine - ScreenLines);
867
868 if (has_mbyte)
869 {
870 int cells;
871 int u8c, u8cc[MAX_MCO];
872 int i;
873 int idx;
874 int c_len;
875 char_u *p;
876# ifdef FEAT_ARABIC
877 int prev_c = 0; // previous Arabic character
878 int prev_c1 = 0; // first composing char for prev_c
879# endif
880
881# ifdef FEAT_RIGHTLEFT
882 if (wp->w_p_rl)
883 idx = off;
884 else
885# endif
886 idx = off + col;
887
888 // Store multibyte characters in ScreenLines[] et al. correctly.
889 for (p = text; *p != NUL; )
890 {
891 cells = (*mb_ptr2cells)(p);
892 c_len = (*mb_ptr2len)(p);
893 if (col + cells > wp->w_width
894# ifdef FEAT_RIGHTLEFT
895 - (wp->w_p_rl ? col : 0)
896# endif
897 )
898 break;
899 ScreenLines[idx] = *p;
900 if (enc_utf8)
901 {
902 u8c = utfc_ptr2char(p, u8cc);
903 if (*p < 0x80 && u8cc[0] == 0)
904 {
905 ScreenLinesUC[idx] = 0;
906#ifdef FEAT_ARABIC
907 prev_c = u8c;
908#endif
909 }
910 else
911 {
912#ifdef FEAT_ARABIC
913 if (p_arshape && !p_tbidi && ARABIC_CHAR(u8c))
914 {
915 // Do Arabic shaping.
916 int pc, pc1, nc;
917 int pcc[MAX_MCO];
918 int firstbyte = *p;
919
920 // The idea of what is the previous and next
921 // character depends on 'rightleft'.
922 if (wp->w_p_rl)
923 {
924 pc = prev_c;
925 pc1 = prev_c1;
926 nc = utf_ptr2char(p + c_len);
927 prev_c1 = u8cc[0];
928 }
929 else
930 {
931 pc = utfc_ptr2char(p + c_len, pcc);
932 nc = prev_c;
933 pc1 = pcc[0];
934 }
935 prev_c = u8c;
936
937 u8c = arabic_shape(u8c, &firstbyte, &u8cc[0],
938 pc, pc1, nc);
939 ScreenLines[idx] = firstbyte;
940 }
941 else
942 prev_c = u8c;
943#endif
944 // Non-BMP character: display as ? or fullwidth ?.
945 ScreenLinesUC[idx] = u8c;
946 for (i = 0; i < Screen_mco; ++i)
947 {
948 ScreenLinesC[i][idx] = u8cc[i];
949 if (u8cc[i] == 0)
950 break;
951 }
952 }
953 if (cells > 1)
954 ScreenLines[idx + 1] = 0;
955 }
956 else if (enc_dbcs == DBCS_JPNU && *p == 0x8e)
957 // double-byte single width character
958 ScreenLines2[idx] = p[1];
959 else if (cells > 1)
960 // double-width character
961 ScreenLines[idx + 1] = p[1];
962 col += cells;
963 idx += cells;
964 p += c_len;
965 }
966 }
967 else
968 {
969 int len = (int)STRLEN(text);
970
971 if (len > wp->w_width - col)
972 len = wp->w_width - col;
973 if (len > 0)
974 {
975#ifdef FEAT_RIGHTLEFT
976 if (wp->w_p_rl)
977 mch_memmove(current_ScreenLine, text, len);
978 else
979#endif
980 mch_memmove(current_ScreenLine + col, text, len);
981 col += len;
982 }
983 }
984 return col;
985}
986#endif
987
988#ifdef FEAT_MENU
989/*
990 * Draw the window toolbar.
991 */
992 static void
993redraw_win_toolbar(win_T *wp)
994{
995 vimmenu_T *menu;
996 int item_idx = 0;
997 int item_count = 0;
998 int col = 0;
999 int next_col;
1000 int off = (int)(current_ScreenLine - ScreenLines);
1001 int fill_attr = syn_name2attr((char_u *)"ToolbarLine");
1002 int button_attr = syn_name2attr((char_u *)"ToolbarButton");
1003
1004 vim_free(wp->w_winbar_items);
Bram Moolenaar00d253e2020-04-06 22:13:01 +02001005 FOR_ALL_CHILD_MENUS(wp->w_winbar, menu)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001006 ++item_count;
1007 wp->w_winbar_items = ALLOC_CLEAR_MULT(winbar_item_T, item_count + 1);
1008
1009 // TODO: use fewer spaces if there is not enough room
1010 for (menu = wp->w_winbar->children;
1011 menu != NULL && col < wp->w_width; menu = menu->next)
1012 {
1013 space_to_screenline(off + col, fill_attr);
1014 if (++col >= wp->w_width)
1015 break;
1016 if (col > 1)
1017 {
1018 space_to_screenline(off + col, fill_attr);
1019 if (++col >= wp->w_width)
1020 break;
1021 }
1022
1023 wp->w_winbar_items[item_idx].wb_startcol = col;
1024 space_to_screenline(off + col, button_attr);
1025 if (++col >= wp->w_width)
1026 break;
1027
1028 next_col = text_to_screenline(wp, menu->name, col);
1029 while (col < next_col)
1030 {
1031 ScreenAttrs[off + col] = button_attr;
1032 ++col;
1033 }
1034 wp->w_winbar_items[item_idx].wb_endcol = col;
1035 wp->w_winbar_items[item_idx].wb_menu = menu;
1036 ++item_idx;
1037
1038 if (col >= wp->w_width)
1039 break;
1040 space_to_screenline(off + col, button_attr);
1041 ++col;
1042 }
1043 while (col < wp->w_width)
1044 {
1045 space_to_screenline(off + col, fill_attr);
1046 ++col;
1047 }
1048 wp->w_winbar_items[item_idx].wb_menu = NULL; // end marker
1049
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001050 screen_line(wp, wp->w_winrow, wp->w_wincol, wp->w_width, wp->w_width, 0);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001051}
1052#endif
1053
1054#if defined(FEAT_FOLDING) || defined(PROTO)
1055/*
1056 * Copy "buf[len]" to ScreenLines["off"] and set attributes to "attr".
1057 */
1058 static void
1059copy_text_attr(
1060 int off,
1061 char_u *buf,
1062 int len,
1063 int attr)
1064{
1065 int i;
1066
1067 mch_memmove(ScreenLines + off, buf, (size_t)len);
1068 if (enc_utf8)
1069 vim_memset(ScreenLinesUC + off, 0, sizeof(u8char_T) * (size_t)len);
1070 for (i = 0; i < len; ++i)
1071 ScreenAttrs[off + i] = attr;
1072}
1073
1074/*
1075 * Display one folded line.
1076 */
1077 static void
1078fold_line(
1079 win_T *wp,
1080 long fold_count,
1081 foldinfo_T *foldinfo,
1082 linenr_T lnum,
1083 int row)
1084{
Bram Moolenaar4fa11752021-03-03 13:26:02 +01001085 // Max value of 'foldcolumn' is 12 and maximum number of bytes in a
1086 // multi-byte character is MAX_MCO.
1087 char_u buf[MAX_MCO * 12 + 1];
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001088 pos_T *top, *bot;
1089 linenr_T lnume = lnum + fold_count - 1;
1090 int len;
1091 char_u *text;
1092 int fdc;
1093 int col;
1094 int txtcol;
1095 int off = (int)(current_ScreenLine - ScreenLines);
1096 int ri;
1097
1098 // Build the fold line:
1099 // 1. Add the cmdwin_type for the command-line window
1100 // 2. Add the 'foldcolumn'
1101 // 3. Add the 'number' or 'relativenumber' column
1102 // 4. Compose the text
1103 // 5. Add the text
1104 // 6. set highlighting for the Visual area an other text
1105 col = 0;
1106
1107 // 1. Add the cmdwin_type for the command-line window
1108 // Ignores 'rightleft', this window is never right-left.
1109#ifdef FEAT_CMDWIN
1110 if (cmdwin_type != 0 && wp == curwin)
1111 {
1112 ScreenLines[off] = cmdwin_type;
1113 ScreenAttrs[off] = HL_ATTR(HLF_AT);
1114 if (enc_utf8)
1115 ScreenLinesUC[off] = 0;
1116 ++col;
1117 }
1118#endif
1119
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001120#ifdef FEAT_RIGHTLEFT
1121# define RL_MEMSET(p, v, l) \
1122 do { \
1123 if (wp->w_p_rl) \
kylo252ae6f1d82022-02-16 19:24:07 +00001124 for (ri = 0; ri < (l); ++ri) \
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001125 ScreenAttrs[off + (wp->w_width - (p) - (l)) + ri] = v; \
1126 else \
kylo252ae6f1d82022-02-16 19:24:07 +00001127 for (ri = 0; ri < (l); ++ri) \
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001128 ScreenAttrs[off + (p) + ri] = v; \
1129 } while (0)
1130#else
1131# define RL_MEMSET(p, v, l) \
1132 do { \
1133 for (ri = 0; ri < l; ++ri) \
1134 ScreenAttrs[off + (p) + ri] = v; \
1135 } while (0)
1136#endif
1137
Bram Moolenaar4fa11752021-03-03 13:26:02 +01001138 // 2. Add the 'foldcolumn'
1139 // Reduce the width when there is not enough space.
1140 fdc = compute_foldcolumn(wp, col);
1141 if (fdc > 0)
1142 {
1143 char_u *p;
1144 int i;
1145 int idx;
1146
1147 fill_foldcolumn(buf, wp, TRUE, lnum);
1148 p = buf;
1149 for (i = 0; i < fdc; i++)
1150 {
1151 int ch;
1152
1153 if (has_mbyte)
1154 ch = mb_ptr2char_adv(&p);
1155 else
1156 ch = *p++;
1157#ifdef FEAT_RIGHTLEFT
1158 if (wp->w_p_rl)
1159 idx = off + wp->w_width - i - 1 - col;
1160 else
1161#endif
1162 idx = off + col + i;
1163 if (enc_utf8)
1164 {
1165 if (ch >= 0x80)
1166 {
1167 ScreenLinesUC[idx] = ch;
1168 ScreenLinesC[0][idx] = 0;
1169 ScreenLines[idx] = 0x80;
1170 }
1171 else
1172 {
1173 ScreenLines[idx] = ch;
1174 ScreenLinesUC[idx] = 0;
1175 }
1176 }
1177 else
1178 ScreenLines[idx] = ch;
1179 }
1180
1181 RL_MEMSET(col, HL_ATTR(HLF_FC), fdc);
1182 col += fdc;
1183 }
1184
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001185 // Set all attributes of the 'number' or 'relativenumber' column and the
1186 // text
1187 RL_MEMSET(col, HL_ATTR(HLF_FL), wp->w_width - col);
1188
1189#ifdef FEAT_SIGNS
1190 // If signs are being displayed, add two spaces.
1191 if (signcolumn_on(wp))
1192 {
1193 len = wp->w_width - col;
1194 if (len > 0)
1195 {
1196 if (len > 2)
1197 len = 2;
1198# ifdef FEAT_RIGHTLEFT
1199 if (wp->w_p_rl)
1200 // the line number isn't reversed
1201 copy_text_attr(off + wp->w_width - len - col,
1202 (char_u *)" ", len, HL_ATTR(HLF_FL));
1203 else
1204# endif
1205 copy_text_attr(off + col, (char_u *)" ", len, HL_ATTR(HLF_FL));
1206 col += len;
1207 }
1208 }
1209#endif
1210
1211 // 3. Add the 'number' or 'relativenumber' column
1212 if (wp->w_p_nu || wp->w_p_rnu)
1213 {
1214 len = wp->w_width - col;
1215 if (len > 0)
1216 {
1217 int w = number_width(wp);
1218 long num;
1219 char *fmt = "%*ld ";
1220
1221 if (len > w + 1)
1222 len = w + 1;
1223
1224 if (wp->w_p_nu && !wp->w_p_rnu)
1225 // 'number' + 'norelativenumber'
1226 num = (long)lnum;
1227 else
1228 {
1229 // 'relativenumber', don't use negative numbers
1230 num = labs((long)get_cursor_rel_lnum(wp, lnum));
1231 if (num == 0 && wp->w_p_nu && wp->w_p_rnu)
1232 {
1233 // 'number' + 'relativenumber': cursor line shows absolute
1234 // line number
1235 num = lnum;
1236 fmt = "%-*ld ";
1237 }
1238 }
1239
1240 sprintf((char *)buf, fmt, w, num);
1241#ifdef FEAT_RIGHTLEFT
1242 if (wp->w_p_rl)
1243 // the line number isn't reversed
1244 copy_text_attr(off + wp->w_width - len - col, buf, len,
1245 HL_ATTR(HLF_FL));
1246 else
1247#endif
1248 copy_text_attr(off + col, buf, len, HL_ATTR(HLF_FL));
1249 col += len;
1250 }
1251 }
1252
1253 // 4. Compose the folded-line string with 'foldtext', if set.
1254 text = get_foldtext(wp, lnum, lnume, foldinfo, buf);
1255
1256 txtcol = col; // remember where text starts
1257
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001258 // 5. move the text to current_ScreenLine. Fill up with "fold" from
1259 // 'fillchars'.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001260 // Right-left text is put in columns 0 - number-col, normal text is put
1261 // in columns number-col - window-width.
1262 col = text_to_screenline(wp, text, col);
1263
1264 // Fill the rest of the line with the fold filler
1265#ifdef FEAT_RIGHTLEFT
1266 if (wp->w_p_rl)
1267 col -= txtcol;
1268#endif
1269 while (col < wp->w_width
1270#ifdef FEAT_RIGHTLEFT
1271 - (wp->w_p_rl ? txtcol : 0)
1272#endif
1273 )
1274 {
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001275 int c = wp->w_fill_chars.fold;
1276
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001277 if (enc_utf8)
1278 {
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001279 if (c >= 0x80)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001280 {
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001281 ScreenLinesUC[off + col] = c;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001282 ScreenLinesC[0][off + col] = 0;
1283 ScreenLines[off + col] = 0x80; // avoid storing zero
1284 }
1285 else
1286 {
1287 ScreenLinesUC[off + col] = 0;
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001288 ScreenLines[off + col] = c;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001289 }
1290 col++;
1291 }
1292 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001293 ScreenLines[off + col++] = c;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001294 }
1295
1296 if (text != buf)
1297 vim_free(text);
1298
1299 // 6. set highlighting for the Visual area an other text.
1300 // If all folded lines are in the Visual area, highlight the line.
1301 if (VIsual_active && wp->w_buffer == curwin->w_buffer)
1302 {
1303 if (LTOREQ_POS(curwin->w_cursor, VIsual))
1304 {
1305 // Visual is after curwin->w_cursor
1306 top = &curwin->w_cursor;
1307 bot = &VIsual;
1308 }
1309 else
1310 {
1311 // Visual is before curwin->w_cursor
1312 top = &VIsual;
1313 bot = &curwin->w_cursor;
1314 }
1315 if (lnum >= top->lnum
1316 && lnume <= bot->lnum
1317 && (VIsual_mode != 'v'
1318 || ((lnum > top->lnum
1319 || (lnum == top->lnum
1320 && top->col == 0))
1321 && (lnume < bot->lnum
1322 || (lnume == bot->lnum
1323 && (bot->col - (*p_sel == 'e'))
1324 >= (colnr_T)STRLEN(ml_get_buf(wp->w_buffer, lnume, FALSE)))))))
1325 {
1326 if (VIsual_mode == Ctrl_V)
1327 {
1328 // Visual block mode: highlight the chars part of the block
1329 if (wp->w_old_cursor_fcol + txtcol < (colnr_T)wp->w_width)
1330 {
1331 if (wp->w_old_cursor_lcol != MAXCOL
1332 && wp->w_old_cursor_lcol + txtcol
1333 < (colnr_T)wp->w_width)
1334 len = wp->w_old_cursor_lcol;
1335 else
1336 len = wp->w_width - txtcol;
1337 RL_MEMSET(wp->w_old_cursor_fcol + txtcol, HL_ATTR(HLF_V),
1338 len - (int)wp->w_old_cursor_fcol);
1339 }
1340 }
1341 else
1342 {
1343 // Set all attributes of the text
1344 RL_MEMSET(txtcol, HL_ATTR(HLF_V), wp->w_width - txtcol);
1345 }
1346 }
1347 }
1348
1349#ifdef FEAT_SYN_HL
1350 // Show colorcolumn in the fold line, but let cursorcolumn override it.
1351 if (wp->w_p_cc_cols)
1352 {
1353 int i = 0;
1354 int j = wp->w_p_cc_cols[i];
1355 int old_txtcol = txtcol;
1356
1357 while (j > -1)
1358 {
1359 txtcol += j;
1360 if (wp->w_p_wrap)
1361 txtcol -= wp->w_skipcol;
1362 else
1363 txtcol -= wp->w_leftcol;
1364 if (txtcol >= 0 && txtcol < wp->w_width)
1365 ScreenAttrs[off + txtcol] = hl_combine_attr(
1366 ScreenAttrs[off + txtcol], HL_ATTR(HLF_MC));
1367 txtcol = old_txtcol;
1368 j = wp->w_p_cc_cols[++i];
1369 }
1370 }
1371
1372 // Show 'cursorcolumn' in the fold line.
1373 if (wp->w_p_cuc)
1374 {
1375 txtcol += wp->w_virtcol;
1376 if (wp->w_p_wrap)
1377 txtcol -= wp->w_skipcol;
1378 else
1379 txtcol -= wp->w_leftcol;
1380 if (txtcol >= 0 && txtcol < wp->w_width)
1381 ScreenAttrs[off + txtcol] = hl_combine_attr(
1382 ScreenAttrs[off + txtcol], HL_ATTR(HLF_CUC));
1383 }
1384#endif
1385
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01001386 screen_line(wp, row + W_WINROW(wp), wp->w_wincol,
1387 wp->w_width, wp->w_width, 0);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001388
1389 // Update w_cline_height and w_cline_folded if the cursor line was
1390 // updated (saves a call to plines() later).
1391 if (wp == curwin
1392 && lnum <= curwin->w_cursor.lnum
1393 && lnume >= curwin->w_cursor.lnum)
1394 {
1395 curwin->w_cline_row = row;
1396 curwin->w_cline_height = 1;
1397 curwin->w_cline_folded = TRUE;
1398 curwin->w_valid |= (VALID_CHEIGHT|VALID_CROW);
1399 }
Bram Moolenaar00aaa512021-07-03 18:04:11 +02001400
1401# ifdef FEAT_CONCEAL
1402 // When the line was not folded w_wrow may have been set, recompute it.
Bram Moolenaar5cb09622021-07-05 22:03:04 +02001403 if (wp == curwin
1404 && wp->w_cursor.lnum >= lnum
1405 && wp->w_cursor.lnum <= lnume
1406 && conceal_cursor_line(wp))
Bram Moolenaar00aaa512021-07-03 18:04:11 +02001407 curs_columns(TRUE);
1408# endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001409}
1410#endif
1411
1412/*
1413 * Update a single window.
1414 *
1415 * This may cause the windows below it also to be redrawn (when clearing the
1416 * screen or scrolling lines).
1417 *
1418 * How the window is redrawn depends on wp->w_redr_type. Each type also
1419 * implies the one below it.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001420 * UPD_NOT_VALID redraw the whole window
1421 * UPD_SOME_VALID redraw the whole window but do scroll when possible
1422 * UPD_REDRAW_TOP redraw the top w_upd_rows window lines, otherwise like
1423 * UPD_VALID
1424 * UPD_INVERTED redraw the changed part of the Visual area
1425 * UPD_INVERTED_ALL redraw the whole Visual area
1426 * UPD_VALID 1. scroll up/down to adjust for a changed w_topline
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001427 * 2. update lines at the top when scrolled down
1428 * 3. redraw changed text:
1429 * - if wp->w_buffer->b_mod_set set, update lines between
1430 * b_mod_top and b_mod_bot.
1431 * - if wp->w_redraw_top non-zero, redraw lines between
1432 * wp->w_redraw_top and wp->w_redr_bot.
1433 * - continue redrawing when syntax status is invalid.
1434 * 4. if scrolled up, update lines at the bottom.
1435 * This results in three areas that may need updating:
1436 * top: from first row to top_end (when scrolled down)
1437 * mid: from mid_start to mid_end (update inversion or changed text)
1438 * bot: from bot_start to last row (when scrolled up)
1439 */
1440 static void
1441win_update(win_T *wp)
1442{
1443 buf_T *buf = wp->w_buffer;
1444 int type;
1445 int top_end = 0; // Below last row of the top area that needs
1446 // updating. 0 when no top area updating.
1447 int mid_start = 999;// first row of the mid area that needs
1448 // updating. 999 when no mid area updating.
1449 int mid_end = 0; // Below last row of the mid area that needs
1450 // updating. 0 when no mid area updating.
1451 int bot_start = 999;// first row of the bot area that needs
1452 // updating. 999 when no bot area updating
1453 int scrolled_down = FALSE; // TRUE when scrolled down when
1454 // w_topline got smaller a bit
1455#ifdef FEAT_SEARCH_EXTRA
1456 int top_to_mod = FALSE; // redraw above mod_top
1457#endif
1458
1459 int row; // current window row to display
1460 linenr_T lnum; // current buffer lnum to display
1461 int idx; // current index in w_lines[]
1462 int srow; // starting row of the current line
1463
1464 int eof = FALSE; // if TRUE, we hit the end of the file
1465 int didline = FALSE; // if TRUE, we finished the last line
1466 int i;
1467 long j;
1468 static int recursive = FALSE; // being called recursively
Bram Moolenaarcbee6352019-11-12 20:49:15 +01001469 linenr_T old_botline = wp->w_botline;
1470#ifdef FEAT_CONCEAL
1471 int old_wrow = wp->w_wrow;
1472 int old_wcol = wp->w_wcol;
1473#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001474#ifdef FEAT_FOLDING
1475 long fold_count;
1476#endif
1477#ifdef FEAT_SYN_HL
1478 // remember what happened to the previous line, to know if
1479 // check_visual_highlight() can be used
Bram Moolenaare7a74d52022-03-19 11:10:15 +00001480# define DID_NONE 1 // didn't update a line
1481# define DID_LINE 2 // updated a normal line
1482# define DID_FOLD 3 // updated a folded line
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001483 int did_update = DID_NONE;
1484 linenr_T syntax_last_parsed = 0; // last parsed text line
1485#endif
1486 linenr_T mod_top = 0;
1487 linenr_T mod_bot = 0;
1488#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
1489 int save_got_int;
1490#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001491
Bram Moolenaare52e0c82020-02-28 22:20:10 +01001492#if defined(FEAT_SEARCH_EXTRA) || defined(FEAT_CLIPBOARD)
1493 // This needs to be done only for the first window when update_screen() is
1494 // called.
1495 if (!did_update_one_window)
1496 {
1497 did_update_one_window = TRUE;
1498# ifdef FEAT_SEARCH_EXTRA
1499 start_search_hl();
1500# endif
1501# ifdef FEAT_CLIPBOARD
1502 // When Visual area changed, may have to update selection.
1503 if (clip_star.available && clip_isautosel_star())
1504 clip_update_selection(&clip_star);
1505 if (clip_plus.available && clip_isautosel_plus())
1506 clip_update_selection(&clip_plus);
1507# endif
1508 }
1509#endif
1510
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001511 type = wp->w_redr_type;
1512
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001513 if (type == UPD_NOT_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001514 {
1515 wp->w_redr_status = TRUE;
1516 wp->w_lines_valid = 0;
1517 }
1518
Bram Moolenaarae0f1512021-03-30 22:12:12 +02001519 // Window frame is zero-height: nothing to draw.
1520 if (wp->w_height + WINBAR_HEIGHT(wp) == 0
1521 || (wp->w_frame->fr_height == wp->w_status_height
1522#if defined(FEAT_PROP_POPUP)
1523 && !popup_is_popup(wp)
1524#endif
1525 ))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001526 {
1527 wp->w_redr_type = 0;
1528 return;
1529 }
1530
1531 // Window is zero-width: Only need to draw the separator.
1532 if (wp->w_width == 0)
1533 {
1534 // draw the vertical separator right of this window
1535 draw_vsep_win(wp, 0);
1536 wp->w_redr_type = 0;
1537 return;
1538 }
1539
1540#ifdef FEAT_TERMINAL
1541 // If this window contains a terminal, redraw works completely differently.
1542 if (term_do_update_window(wp))
1543 {
1544 term_update_window(wp);
1545# ifdef FEAT_MENU
1546 // Draw the window toolbar, if there is one.
1547 if (winbar_height(wp) > 0)
1548 redraw_win_toolbar(wp);
1549# endif
1550 wp->w_redr_type = 0;
1551 return;
1552 }
1553#endif
1554
1555#ifdef FEAT_SEARCH_EXTRA
1556 init_search_hl(wp, &screen_search_hl);
1557#endif
1558
1559#ifdef FEAT_LINEBREAK
1560 // Force redraw when width of 'number' or 'relativenumber' column
1561 // changes.
1562 i = (wp->w_p_nu || wp->w_p_rnu) ? number_width(wp) : 0;
1563 if (wp->w_nrwidth != i)
1564 {
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001565 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001566 wp->w_nrwidth = i;
1567 }
1568 else
1569#endif
1570
1571 if (buf->b_mod_set && buf->b_mod_xlines != 0 && wp->w_redraw_top != 0)
1572 {
1573 // When there are both inserted/deleted lines and specific lines to be
1574 // redrawn, w_redraw_top and w_redraw_bot may be invalid, just redraw
1575 // everything (only happens when redrawing is off for while).
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001576 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001577 }
1578 else
1579 {
1580 // Set mod_top to the first line that needs displaying because of
1581 // changes. Set mod_bot to the first line after the changes.
1582 mod_top = wp->w_redraw_top;
1583 if (wp->w_redraw_bot != 0)
1584 mod_bot = wp->w_redraw_bot + 1;
1585 else
1586 mod_bot = 0;
1587 if (buf->b_mod_set)
1588 {
1589 if (mod_top == 0 || mod_top > buf->b_mod_top)
1590 {
1591 mod_top = buf->b_mod_top;
1592#ifdef FEAT_SYN_HL
1593 // Need to redraw lines above the change that may be included
1594 // in a pattern match.
1595 if (syntax_present(wp))
1596 {
1597 mod_top -= buf->b_s.b_syn_sync_linebreaks;
1598 if (mod_top < 1)
1599 mod_top = 1;
1600 }
1601#endif
1602 }
1603 if (mod_bot == 0 || mod_bot < buf->b_mod_bot)
1604 mod_bot = buf->b_mod_bot;
1605
1606#ifdef FEAT_SEARCH_EXTRA
1607 // When 'hlsearch' is on and using a multi-line search pattern, a
1608 // change in one line may make the Search highlighting in a
1609 // previous line invalid. Simple solution: redraw all visible
1610 // lines above the change.
1611 // Same for a match pattern.
1612 if (screen_search_hl.rm.regprog != NULL
1613 && re_multiline(screen_search_hl.rm.regprog))
1614 top_to_mod = TRUE;
1615 else
1616 {
1617 matchitem_T *cur = wp->w_match_head;
1618
1619 while (cur != NULL)
1620 {
1621 if (cur->match.regprog != NULL
1622 && re_multiline(cur->match.regprog))
1623 {
1624 top_to_mod = TRUE;
1625 break;
1626 }
1627 cur = cur->next;
1628 }
1629 }
1630#endif
1631 }
Bram Moolenaar368137a2022-05-31 13:43:12 +01001632
1633#ifdef FEAT_SEARCH_EXTRA
1634 if (search_hl_has_cursor_lnum > 0)
1635 {
1636 // CurSearch was used last time, need to redraw the line with it to
1637 // avoid having two matches highlighted with CurSearch.
1638 if (mod_top == 0 || mod_top > search_hl_has_cursor_lnum)
1639 mod_top = search_hl_has_cursor_lnum;
1640 if (mod_bot == 0 || mod_bot < search_hl_has_cursor_lnum + 1)
1641 mod_bot = search_hl_has_cursor_lnum + 1;
1642 }
1643#endif
1644
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001645#ifdef FEAT_FOLDING
1646 if (mod_top != 0 && hasAnyFolding(wp))
1647 {
1648 linenr_T lnumt, lnumb;
1649
1650 // A change in a line can cause lines above it to become folded or
1651 // unfolded. Find the top most buffer line that may be affected.
1652 // If the line was previously folded and displayed, get the first
1653 // line of that fold. If the line is folded now, get the first
1654 // folded line. Use the minimum of these two.
1655
1656 // Find last valid w_lines[] entry above mod_top. Set lnumt to
1657 // the line below it. If there is no valid entry, use w_topline.
1658 // Find the first valid w_lines[] entry below mod_bot. Set lnumb
1659 // to this line. If there is no valid entry, use MAXLNUM.
1660 lnumt = wp->w_topline;
1661 lnumb = MAXLNUM;
1662 for (i = 0; i < wp->w_lines_valid; ++i)
1663 if (wp->w_lines[i].wl_valid)
1664 {
1665 if (wp->w_lines[i].wl_lastlnum < mod_top)
1666 lnumt = wp->w_lines[i].wl_lastlnum + 1;
1667 if (lnumb == MAXLNUM && wp->w_lines[i].wl_lnum >= mod_bot)
1668 {
1669 lnumb = wp->w_lines[i].wl_lnum;
1670 // When there is a fold column it might need updating
1671 // in the next line ("J" just above an open fold).
1672 if (compute_foldcolumn(wp, 0) > 0)
1673 ++lnumb;
1674 }
1675 }
1676
1677 (void)hasFoldingWin(wp, mod_top, &mod_top, NULL, TRUE, NULL);
1678 if (mod_top > lnumt)
1679 mod_top = lnumt;
1680
1681 // Now do the same for the bottom line (one above mod_bot).
1682 --mod_bot;
1683 (void)hasFoldingWin(wp, mod_bot, NULL, &mod_bot, TRUE, NULL);
1684 ++mod_bot;
1685 if (mod_bot < lnumb)
1686 mod_bot = lnumb;
1687 }
1688#endif
1689
1690 // When a change starts above w_topline and the end is below
1691 // w_topline, start redrawing at w_topline.
1692 // If the end of the change is above w_topline: do like no change was
1693 // made, but redraw the first line to find changes in syntax.
1694 if (mod_top != 0 && mod_top < wp->w_topline)
1695 {
1696 if (mod_bot > wp->w_topline)
1697 mod_top = wp->w_topline;
1698#ifdef FEAT_SYN_HL
1699 else if (syntax_present(wp))
1700 top_end = 1;
1701#endif
1702 }
1703
1704 // When line numbers are displayed need to redraw all lines below
1705 // inserted/deleted lines.
1706 if (mod_top != 0 && buf->b_mod_xlines != 0 && wp->w_p_nu)
1707 mod_bot = MAXLNUM;
1708 }
1709 wp->w_redraw_top = 0; // reset for next time
1710 wp->w_redraw_bot = 0;
Bram Moolenaar368137a2022-05-31 13:43:12 +01001711#ifdef FEAT_SEARCH_EXTRA
1712 search_hl_has_cursor_lnum = 0;
1713#endif
1714
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001715
1716 // When only displaying the lines at the top, set top_end. Used when
1717 // window has scrolled down for msg_scrolled.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001718 if (type == UPD_REDRAW_TOP)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001719 {
1720 j = 0;
1721 for (i = 0; i < wp->w_lines_valid; ++i)
1722 {
1723 j += wp->w_lines[i].wl_size;
1724 if (j >= wp->w_upd_rows)
1725 {
1726 top_end = j;
1727 break;
1728 }
1729 }
1730 if (top_end == 0)
1731 // not found (cannot happen?): redraw everything
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001732 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001733 else
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001734 // top area defined, the rest is UPD_VALID
1735 type = UPD_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001736 }
1737
1738 // Trick: we want to avoid clearing the screen twice. screenclear() will
1739 // set "screen_cleared" to TRUE. The special value MAYBE (which is still
1740 // non-zero and thus not FALSE) will indicate that screenclear() was not
1741 // called.
1742 if (screen_cleared)
1743 screen_cleared = MAYBE;
1744
1745 // If there are no changes on the screen that require a complete redraw,
1746 // handle three cases:
1747 // 1: we are off the top of the screen by a few lines: scroll down
1748 // 2: wp->w_topline is below wp->w_lines[0].wl_lnum: may scroll up
1749 // 3: wp->w_topline is wp->w_lines[0].wl_lnum: find first entry in
1750 // w_lines[] that needs updating.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001751 if ((type == UPD_VALID || type == UPD_SOME_VALID
1752 || type == UPD_INVERTED || type == UPD_INVERTED_ALL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001753#ifdef FEAT_DIFF
1754 && !wp->w_botfill && !wp->w_old_botfill
1755#endif
1756 )
1757 {
Bram Moolenaarf8992d42020-08-01 19:14:13 +02001758 if (mod_top != 0
1759 && wp->w_topline == mod_top
1760 && (!wp->w_lines[0].wl_valid
Bram Moolenaarabb6fbd2022-03-25 15:42:27 +00001761 || wp->w_topline == wp->w_lines[0].wl_lnum))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001762 {
Bram Moolenaarf8992d42020-08-01 19:14:13 +02001763 // w_topline is the first changed line and window is not scrolled,
1764 // the scrolling from changed lines will be done further down.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001765 }
1766 else if (wp->w_lines[0].wl_valid
1767 && (wp->w_topline < wp->w_lines[0].wl_lnum
1768#ifdef FEAT_DIFF
1769 || (wp->w_topline == wp->w_lines[0].wl_lnum
1770 && wp->w_topfill > wp->w_old_topfill)
1771#endif
1772 ))
1773 {
1774 // New topline is above old topline: May scroll down.
1775#ifdef FEAT_FOLDING
1776 if (hasAnyFolding(wp))
1777 {
1778 linenr_T ln;
1779
1780 // count the number of lines we are off, counting a sequence
1781 // of folded lines as one
1782 j = 0;
1783 for (ln = wp->w_topline; ln < wp->w_lines[0].wl_lnum; ++ln)
1784 {
1785 ++j;
1786 if (j >= wp->w_height - 2)
1787 break;
1788 (void)hasFoldingWin(wp, ln, NULL, &ln, TRUE, NULL);
1789 }
1790 }
1791 else
1792#endif
1793 j = wp->w_lines[0].wl_lnum - wp->w_topline;
1794 if (j < wp->w_height - 2) // not too far off
1795 {
1796 i = plines_m_win(wp, wp->w_topline, wp->w_lines[0].wl_lnum - 1);
1797#ifdef FEAT_DIFF
1798 // insert extra lines for previously invisible filler lines
1799 if (wp->w_lines[0].wl_lnum != wp->w_topline)
1800 i += diff_check_fill(wp, wp->w_lines[0].wl_lnum)
1801 - wp->w_old_topfill;
1802#endif
1803 if (i < wp->w_height - 2) // less than a screen off
1804 {
1805 // Try to insert the correct number of lines.
1806 // If not the last window, delete the lines at the bottom.
1807 // win_ins_lines may fail when the terminal can't do it.
1808 if (i > 0)
1809 check_for_delay(FALSE);
1810 if (win_ins_lines(wp, 0, i, FALSE, wp == firstwin) == OK)
1811 {
1812 if (wp->w_lines_valid != 0)
1813 {
1814 // Need to update rows that are new, stop at the
1815 // first one that scrolled down.
1816 top_end = i;
1817 scrolled_down = TRUE;
1818
1819 // Move the entries that were scrolled, disable
1820 // the entries for the lines to be redrawn.
Bram Moolenaarfdc5d172022-08-11 15:52:14 +01001821 // Avoid using a wrong index when 'cmdheight' is
1822 // zero and wp->w_height == Rows.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001823 if ((wp->w_lines_valid += j) > wp->w_height)
1824 wp->w_lines_valid = wp->w_height;
Bram Moolenaarfdc5d172022-08-11 15:52:14 +01001825 for (idx = wp->w_lines_valid >= wp->w_height
1826 ? wp->w_height - 1 : wp->w_lines_valid;
1827 idx - j >= 0; idx--)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001828 wp->w_lines[idx] = wp->w_lines[idx - j];
1829 while (idx >= 0)
1830 wp->w_lines[idx--].wl_valid = FALSE;
1831 }
1832 }
1833 else
1834 mid_start = 0; // redraw all lines
1835 }
1836 else
1837 mid_start = 0; // redraw all lines
1838 }
1839 else
1840 mid_start = 0; // redraw all lines
1841 }
1842 else
1843 {
1844 // New topline is at or below old topline: May scroll up.
1845 // When topline didn't change, find first entry in w_lines[] that
1846 // needs updating.
1847
Bram Moolenaarf6ebc822022-01-16 13:58:33 +00001848 // Try to find wp->w_topline in wp->w_lines[].wl_lnum. The check
1849 // for "Rows" is in case "wl_size" is incorrect somehow.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001850 j = -1;
1851 row = 0;
Bram Moolenaarf6ebc822022-01-16 13:58:33 +00001852 for (i = 0; i < wp->w_lines_valid && i < Rows; i++)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001853 {
1854 if (wp->w_lines[i].wl_valid
1855 && wp->w_lines[i].wl_lnum == wp->w_topline)
1856 {
1857 j = i;
1858 break;
1859 }
1860 row += wp->w_lines[i].wl_size;
1861 }
1862 if (j == -1)
1863 {
1864 // if wp->w_topline is not in wp->w_lines[].wl_lnum redraw all
1865 // lines
1866 mid_start = 0;
1867 }
1868 else
1869 {
1870 // Try to delete the correct number of lines.
1871 // wp->w_topline is at wp->w_lines[i].wl_lnum.
1872#ifdef FEAT_DIFF
1873 // If the topline didn't change, delete old filler lines,
1874 // otherwise delete filler lines of the new topline...
1875 if (wp->w_lines[0].wl_lnum == wp->w_topline)
1876 row += wp->w_old_topfill;
1877 else
1878 row += diff_check_fill(wp, wp->w_topline);
1879 // ... but don't delete new filler lines.
1880 row -= wp->w_topfill;
1881#endif
Bram Moolenaarf6ebc822022-01-16 13:58:33 +00001882 if (row > Rows) // just in case
1883 row = Rows;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001884 if (row > 0)
1885 {
1886 check_for_delay(FALSE);
1887 if (win_del_lines(wp, 0, row, FALSE, wp == firstwin, 0)
1888 == OK)
1889 bot_start = wp->w_height - row;
1890 else
1891 mid_start = 0; // redraw all lines
1892 }
1893 if ((row == 0 || bot_start < 999) && wp->w_lines_valid != 0)
1894 {
1895 // Skip the lines (below the deleted lines) that are still
1896 // valid and don't need redrawing. Copy their info
1897 // upwards, to compensate for the deleted lines. Set
1898 // bot_start to the first row that needs redrawing.
1899 bot_start = 0;
1900 idx = 0;
1901 for (;;)
1902 {
1903 wp->w_lines[idx] = wp->w_lines[j];
1904 // stop at line that didn't fit, unless it is still
1905 // valid (no lines deleted)
1906 if (row > 0 && bot_start + row
1907 + (int)wp->w_lines[j].wl_size > wp->w_height)
1908 {
1909 wp->w_lines_valid = idx + 1;
1910 break;
1911 }
1912 bot_start += wp->w_lines[idx++].wl_size;
1913
1914 // stop at the last valid entry in w_lines[].wl_size
1915 if (++j >= wp->w_lines_valid)
1916 {
1917 wp->w_lines_valid = idx;
1918 break;
1919 }
1920 }
1921#ifdef FEAT_DIFF
1922 // Correct the first entry for filler lines at the top
1923 // when it won't get updated below.
1924 if (wp->w_p_diff && bot_start > 0)
1925 wp->w_lines[0].wl_size =
1926 plines_win_nofill(wp, wp->w_topline, TRUE)
1927 + wp->w_topfill;
1928#endif
1929 }
1930 }
1931 }
1932
1933 // When starting redraw in the first line, redraw all lines. When
1934 // there is only one window it's probably faster to clear the screen
1935 // first.
1936 if (mid_start == 0)
1937 {
1938 mid_end = wp->w_height;
1939 if (ONE_WINDOW && !WIN_IS_POPUP(wp))
1940 {
1941 // Clear the screen when it was not done by win_del_lines() or
1942 // win_ins_lines() above, "screen_cleared" is FALSE or MAYBE
1943 // then.
1944 if (screen_cleared != TRUE)
1945 screenclear();
1946 // The screen was cleared, redraw the tab pages line.
1947 if (redraw_tabline)
1948 draw_tabline();
1949 }
1950 }
1951
1952 // When win_del_lines() or win_ins_lines() caused the screen to be
1953 // cleared (only happens for the first window) or when screenclear()
1954 // was called directly above, "must_redraw" will have been set to
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001955 // UPD_NOT_VALID, need to reset it here to avoid redrawing twice.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001956 if (screen_cleared == TRUE)
1957 must_redraw = 0;
1958 }
1959 else
1960 {
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001961 // Not UPD_VALID or UPD_INVERTED: redraw all lines.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001962 mid_start = 0;
1963 mid_end = wp->w_height;
1964 }
1965
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001966 if (type == UPD_SOME_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001967 {
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001968 // UPD_SOME_VALID: redraw all lines.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001969 mid_start = 0;
1970 mid_end = wp->w_height;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001971 type = UPD_NOT_VALID;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001972 }
1973
1974 // check if we are updating or removing the inverted part
1975 if ((VIsual_active && buf == curwin->w_buffer)
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001976 || (wp->w_old_cursor_lnum != 0 && type != UPD_NOT_VALID))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001977 {
1978 linenr_T from, to;
1979
1980 if (VIsual_active)
1981 {
Bram Moolenaarfe154992022-03-22 20:42:12 +00001982 if (VIsual_mode != wp->w_old_visual_mode
Bram Moolenaara4d158b2022-08-14 14:17:45 +01001983 || type == UPD_INVERTED_ALL)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02001984 {
1985 // If the type of Visual selection changed, redraw the whole
1986 // selection. Also when the ownership of the X selection is
1987 // gained or lost.
1988 if (curwin->w_cursor.lnum < VIsual.lnum)
1989 {
1990 from = curwin->w_cursor.lnum;
1991 to = VIsual.lnum;
1992 }
1993 else
1994 {
1995 from = VIsual.lnum;
1996 to = curwin->w_cursor.lnum;
1997 }
1998 // redraw more when the cursor moved as well
1999 if (wp->w_old_cursor_lnum < from)
2000 from = wp->w_old_cursor_lnum;
2001 if (wp->w_old_cursor_lnum > to)
2002 to = wp->w_old_cursor_lnum;
2003 if (wp->w_old_visual_lnum < from)
2004 from = wp->w_old_visual_lnum;
2005 if (wp->w_old_visual_lnum > to)
2006 to = wp->w_old_visual_lnum;
2007 }
2008 else
2009 {
2010 // Find the line numbers that need to be updated: The lines
2011 // between the old cursor position and the current cursor
2012 // position. Also check if the Visual position changed.
2013 if (curwin->w_cursor.lnum < wp->w_old_cursor_lnum)
2014 {
2015 from = curwin->w_cursor.lnum;
2016 to = wp->w_old_cursor_lnum;
2017 }
2018 else
2019 {
2020 from = wp->w_old_cursor_lnum;
2021 to = curwin->w_cursor.lnum;
2022 if (from == 0) // Visual mode just started
2023 from = to;
2024 }
2025
2026 if (VIsual.lnum != wp->w_old_visual_lnum
2027 || VIsual.col != wp->w_old_visual_col)
2028 {
2029 if (wp->w_old_visual_lnum < from
2030 && wp->w_old_visual_lnum != 0)
2031 from = wp->w_old_visual_lnum;
2032 if (wp->w_old_visual_lnum > to)
2033 to = wp->w_old_visual_lnum;
2034 if (VIsual.lnum < from)
2035 from = VIsual.lnum;
2036 if (VIsual.lnum > to)
2037 to = VIsual.lnum;
2038 }
2039 }
2040
2041 // If in block mode and changed column or curwin->w_curswant:
2042 // update all lines.
2043 // First compute the actual start and end column.
2044 if (VIsual_mode == Ctrl_V)
2045 {
2046 colnr_T fromc, toc;
2047#if defined(FEAT_LINEBREAK)
Gary Johnson51ad8502021-08-03 18:33:08 +02002048 int save_ve_flags = curwin->w_ve_flags;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002049
2050 if (curwin->w_p_lbr)
Gary Johnson51ad8502021-08-03 18:33:08 +02002051 curwin->w_ve_flags = VE_ALL;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002052#endif
2053 getvcols(wp, &VIsual, &curwin->w_cursor, &fromc, &toc);
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002054 ++toc;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002055#if defined(FEAT_LINEBREAK)
Gary Johnson51ad8502021-08-03 18:33:08 +02002056 curwin->w_ve_flags = save_ve_flags;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002057#endif
Bram Moolenaar9cee4a12021-07-03 15:08:37 +02002058 // Highlight to the end of the line, unless 'virtualedit' has
2059 // "block".
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002060 if (curwin->w_curswant == MAXCOL)
2061 {
Gary Johnson53ba05b2021-07-26 22:19:10 +02002062 if (get_ve_flags() & VE_BLOCK)
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002063 {
2064 pos_T pos;
2065 int cursor_above =
2066 curwin->w_cursor.lnum < VIsual.lnum;
2067
2068 // Need to find the longest line.
2069 toc = 0;
2070 pos.coladd = 0;
2071 for (pos.lnum = curwin->w_cursor.lnum; cursor_above
2072 ? pos.lnum <= VIsual.lnum
2073 : pos.lnum >= VIsual.lnum;
2074 pos.lnum += cursor_above ? 1 : -1)
2075 {
2076 colnr_T t;
2077
Bram Moolenaar6bcb1822021-07-09 15:54:00 +02002078 pos.col = (int)STRLEN(ml_get_buf(wp->w_buffer,
Bram Moolenaarb17ab862021-07-03 22:15:17 +02002079 pos.lnum, FALSE));
2080 getvvcol(wp, &pos, NULL, NULL, &t);
2081 if (toc < t)
2082 toc = t;
2083 }
2084 ++toc;
2085 }
2086 else
2087 toc = MAXCOL;
2088 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002089
2090 if (fromc != wp->w_old_cursor_fcol
2091 || toc != wp->w_old_cursor_lcol)
2092 {
2093 if (from > VIsual.lnum)
2094 from = VIsual.lnum;
2095 if (to < VIsual.lnum)
2096 to = VIsual.lnum;
2097 }
2098 wp->w_old_cursor_fcol = fromc;
2099 wp->w_old_cursor_lcol = toc;
2100 }
2101 }
2102 else
2103 {
2104 // Use the line numbers of the old Visual area.
2105 if (wp->w_old_cursor_lnum < wp->w_old_visual_lnum)
2106 {
2107 from = wp->w_old_cursor_lnum;
2108 to = wp->w_old_visual_lnum;
2109 }
2110 else
2111 {
2112 from = wp->w_old_visual_lnum;
2113 to = wp->w_old_cursor_lnum;
2114 }
2115 }
2116
2117 // There is no need to update lines above the top of the window.
2118 if (from < wp->w_topline)
2119 from = wp->w_topline;
2120
2121 // If we know the value of w_botline, use it to restrict the update to
2122 // the lines that are visible in the window.
2123 if (wp->w_valid & VALID_BOTLINE)
2124 {
2125 if (from >= wp->w_botline)
2126 from = wp->w_botline - 1;
2127 if (to >= wp->w_botline)
2128 to = wp->w_botline - 1;
2129 }
2130
2131 // Find the minimal part to be updated.
2132 // Watch out for scrolling that made entries in w_lines[] invalid.
2133 // E.g., CTRL-U makes the first half of w_lines[] invalid and sets
2134 // top_end; need to redraw from top_end to the "to" line.
2135 // A middle mouse click with a Visual selection may change the text
2136 // above the Visual area and reset wl_valid, do count these for
2137 // mid_end (in srow).
2138 if (mid_start > 0)
2139 {
2140 lnum = wp->w_topline;
2141 idx = 0;
2142 srow = 0;
2143 if (scrolled_down)
2144 mid_start = top_end;
2145 else
2146 mid_start = 0;
2147 while (lnum < from && idx < wp->w_lines_valid) // find start
2148 {
2149 if (wp->w_lines[idx].wl_valid)
2150 mid_start += wp->w_lines[idx].wl_size;
2151 else if (!scrolled_down)
2152 srow += wp->w_lines[idx].wl_size;
2153 ++idx;
2154# ifdef FEAT_FOLDING
2155 if (idx < wp->w_lines_valid && wp->w_lines[idx].wl_valid)
2156 lnum = wp->w_lines[idx].wl_lnum;
2157 else
2158# endif
2159 ++lnum;
2160 }
2161 srow += mid_start;
2162 mid_end = wp->w_height;
2163 for ( ; idx < wp->w_lines_valid; ++idx) // find end
2164 {
2165 if (wp->w_lines[idx].wl_valid
2166 && wp->w_lines[idx].wl_lnum >= to + 1)
2167 {
2168 // Only update until first row of this line
2169 mid_end = srow;
2170 break;
2171 }
2172 srow += wp->w_lines[idx].wl_size;
2173 }
2174 }
2175 }
2176
2177 if (VIsual_active && buf == curwin->w_buffer)
2178 {
2179 wp->w_old_visual_mode = VIsual_mode;
2180 wp->w_old_cursor_lnum = curwin->w_cursor.lnum;
2181 wp->w_old_visual_lnum = VIsual.lnum;
2182 wp->w_old_visual_col = VIsual.col;
2183 wp->w_old_curswant = curwin->w_curswant;
2184 }
2185 else
2186 {
2187 wp->w_old_visual_mode = 0;
2188 wp->w_old_cursor_lnum = 0;
2189 wp->w_old_visual_lnum = 0;
2190 wp->w_old_visual_col = 0;
2191 }
2192
2193#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
2194 // reset got_int, otherwise regexp won't work
2195 save_got_int = got_int;
2196 got_int = 0;
2197#endif
2198#ifdef SYN_TIME_LIMIT
2199 // Set the time limit to 'redrawtime'.
Bram Moolenaar6f0cf622022-06-19 12:27:45 +01002200 redrawtime_limit_set = TRUE;
Paul Ollis65745772022-06-05 16:55:54 +01002201 init_regexp_timeout(p_rdt);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002202#endif
2203#ifdef FEAT_FOLDING
2204 win_foldinfo.fi_level = 0;
2205#endif
2206
2207#ifdef FEAT_MENU
2208 // Draw the window toolbar, if there is one.
2209 // TODO: only when needed.
2210 if (winbar_height(wp) > 0)
2211 redraw_win_toolbar(wp);
2212#endif
2213
2214 // Update all the window rows.
2215 idx = 0; // first entry in w_lines[].wl_size
2216 row = 0;
2217 srow = 0;
2218 lnum = wp->w_topline; // first line shown in window
2219 for (;;)
2220 {
2221 // stop updating when reached the end of the window (check for _past_
2222 // the end of the window is at the end of the loop)
2223 if (row == wp->w_height)
2224 {
2225 didline = TRUE;
2226 break;
2227 }
2228
2229 // stop updating when hit the end of the file
2230 if (lnum > buf->b_ml.ml_line_count)
2231 {
2232 eof = TRUE;
2233 break;
2234 }
2235
2236 // Remember the starting row of the line that is going to be dealt
2237 // with. It is used further down when the line doesn't fit.
2238 srow = row;
2239
2240 // Update a line when it is in an area that needs updating, when it
2241 // has changes or w_lines[idx] is invalid.
2242 // "bot_start" may be halfway a wrapped line after using
2243 // win_del_lines(), check if the current line includes it.
2244 // When syntax folding is being used, the saved syntax states will
2245 // already have been updated, we can't see where the syntax state is
2246 // the same again, just update until the end of the window.
2247 if (row < top_end
2248 || (row >= mid_start && row < mid_end)
2249#ifdef FEAT_SEARCH_EXTRA
2250 || top_to_mod
2251#endif
2252 || idx >= wp->w_lines_valid
2253 || (row + wp->w_lines[idx].wl_size > bot_start)
2254 || (mod_top != 0
2255 && (lnum == mod_top
2256 || (lnum >= mod_top
2257 && (lnum < mod_bot
2258#ifdef FEAT_SYN_HL
2259 || did_update == DID_FOLD
2260 || (did_update == DID_LINE
2261 && syntax_present(wp)
2262 && (
2263# ifdef FEAT_FOLDING
2264 (foldmethodIsSyntax(wp)
2265 && hasAnyFolding(wp)) ||
2266# endif
2267 syntax_check_changed(lnum)))
2268#endif
2269#ifdef FEAT_SEARCH_EXTRA
2270 // match in fixed position might need redraw
2271 // if lines were inserted or deleted
2272 || (wp->w_match_head != NULL
2273 && buf->b_mod_xlines != 0)
2274#endif
Bram Moolenaar11a58af2019-10-24 22:32:31 +02002275 ))))
2276#ifdef FEAT_SYN_HL
zeertzjqc20e46a2022-03-23 14:55:23 +00002277 || (wp->w_p_cul && lnum == wp->w_cursor.lnum)
2278 || lnum == wp->w_last_cursorline
Bram Moolenaar11a58af2019-10-24 22:32:31 +02002279#endif
2280 )
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002281 {
2282#ifdef FEAT_SEARCH_EXTRA
2283 if (lnum == mod_top)
2284 top_to_mod = FALSE;
2285#endif
2286
2287 // When at start of changed lines: May scroll following lines
2288 // up or down to minimize redrawing.
2289 // Don't do this when the change continues until the end.
2290 // Don't scroll when dollar_vcol >= 0, keep the "$".
Bram Moolenaarc9e7e342021-07-22 21:33:03 +02002291 // Don't scroll when redrawing the top, scrolled already above.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002292 if (lnum == mod_top
2293 && mod_bot != MAXLNUM
Bram Moolenaarc9e7e342021-07-22 21:33:03 +02002294 && !(dollar_vcol >= 0 && mod_bot == mod_top + 1)
2295 && row >= top_end)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002296 {
2297 int old_rows = 0;
2298 int new_rows = 0;
2299 int xtra_rows;
2300 linenr_T l;
2301
2302 // Count the old number of window rows, using w_lines[], which
2303 // should still contain the sizes for the lines as they are
2304 // currently displayed.
2305 for (i = idx; i < wp->w_lines_valid; ++i)
2306 {
2307 // Only valid lines have a meaningful wl_lnum. Invalid
2308 // lines are part of the changed area.
2309 if (wp->w_lines[i].wl_valid
2310 && wp->w_lines[i].wl_lnum == mod_bot)
2311 break;
2312 old_rows += wp->w_lines[i].wl_size;
2313#ifdef FEAT_FOLDING
2314 if (wp->w_lines[i].wl_valid
2315 && wp->w_lines[i].wl_lastlnum + 1 == mod_bot)
2316 {
2317 // Must have found the last valid entry above mod_bot.
2318 // Add following invalid entries.
2319 ++i;
2320 while (i < wp->w_lines_valid
2321 && !wp->w_lines[i].wl_valid)
2322 old_rows += wp->w_lines[i++].wl_size;
2323 break;
2324 }
2325#endif
2326 }
2327
2328 if (i >= wp->w_lines_valid)
2329 {
2330 // We can't find a valid line below the changed lines,
2331 // need to redraw until the end of the window.
2332 // Inserting/deleting lines has no use.
2333 bot_start = 0;
2334 }
2335 else
2336 {
2337 // Able to count old number of rows: Count new window
2338 // rows, and may insert/delete lines
2339 j = idx;
2340 for (l = lnum; l < mod_bot; ++l)
2341 {
2342#ifdef FEAT_FOLDING
2343 if (hasFoldingWin(wp, l, NULL, &l, TRUE, NULL))
2344 ++new_rows;
2345 else
2346#endif
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002347 {
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002348#ifdef FEAT_DIFF
2349 if (l == wp->w_topline)
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002350 new_rows += plines_win_nofill(wp, l, TRUE)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002351 + wp->w_topfill;
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002352 else
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002353#endif
Dominique Pelle9b0b8442021-10-18 20:56:39 +01002354 new_rows += plines_win(wp, l, TRUE);
2355 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002356 ++j;
2357 if (new_rows > wp->w_height - row - 2)
2358 {
2359 // it's getting too much, must redraw the rest
2360 new_rows = 9999;
2361 break;
2362 }
2363 }
2364 xtra_rows = new_rows - old_rows;
2365 if (xtra_rows < 0)
2366 {
2367 // May scroll text up. If there is not enough
2368 // remaining text or scrolling fails, must redraw the
2369 // rest. If scrolling works, must redraw the text
2370 // below the scrolled text.
2371 if (row - xtra_rows >= wp->w_height - 2)
2372 mod_bot = MAXLNUM;
2373 else
2374 {
2375 check_for_delay(FALSE);
2376 if (win_del_lines(wp, row,
2377 -xtra_rows, FALSE, FALSE, 0) == FAIL)
2378 mod_bot = MAXLNUM;
2379 else
2380 bot_start = wp->w_height + xtra_rows;
2381 }
2382 }
2383 else if (xtra_rows > 0)
2384 {
2385 // May scroll text down. If there is not enough
2386 // remaining text of scrolling fails, must redraw the
2387 // rest.
2388 if (row + xtra_rows >= wp->w_height - 2)
2389 mod_bot = MAXLNUM;
2390 else
2391 {
2392 check_for_delay(FALSE);
2393 if (win_ins_lines(wp, row + old_rows,
2394 xtra_rows, FALSE, FALSE) == FAIL)
2395 mod_bot = MAXLNUM;
2396 else if (top_end > row + old_rows)
2397 // Scrolled the part at the top that requires
2398 // updating down.
2399 top_end += xtra_rows;
2400 }
2401 }
2402
2403 // When not updating the rest, may need to move w_lines[]
2404 // entries.
2405 if (mod_bot != MAXLNUM && i != j)
2406 {
2407 if (j < i)
2408 {
2409 int x = row + new_rows;
2410
2411 // move entries in w_lines[] upwards
2412 for (;;)
2413 {
2414 // stop at last valid entry in w_lines[]
2415 if (i >= wp->w_lines_valid)
2416 {
2417 wp->w_lines_valid = j;
2418 break;
2419 }
2420 wp->w_lines[j] = wp->w_lines[i];
2421 // stop at a line that won't fit
2422 if (x + (int)wp->w_lines[j].wl_size
2423 > wp->w_height)
2424 {
2425 wp->w_lines_valid = j + 1;
2426 break;
2427 }
2428 x += wp->w_lines[j++].wl_size;
2429 ++i;
2430 }
2431 if (bot_start > x)
2432 bot_start = x;
2433 }
2434 else // j > i
2435 {
2436 // move entries in w_lines[] downwards
2437 j -= i;
2438 wp->w_lines_valid += j;
2439 if (wp->w_lines_valid > wp->w_height)
2440 wp->w_lines_valid = wp->w_height;
2441 for (i = wp->w_lines_valid; i - j >= idx; --i)
Bram Moolenaar7a99da42022-08-28 22:21:01 +01002442 if (i < Rows)
2443 wp->w_lines[i] = wp->w_lines[i - j];
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002444
2445 // The w_lines[] entries for inserted lines are
2446 // now invalid, but wl_size may be used above.
2447 // Reset to zero.
2448 while (i >= idx)
2449 {
2450 wp->w_lines[i].wl_size = 0;
2451 wp->w_lines[i--].wl_valid = FALSE;
2452 }
2453 }
2454 }
2455 }
2456 }
2457
2458#ifdef FEAT_FOLDING
2459 // When lines are folded, display one line for all of them.
2460 // Otherwise, display normally (can be several display lines when
2461 // 'wrap' is on).
2462 fold_count = foldedCount(wp, lnum, &win_foldinfo);
2463 if (fold_count != 0)
2464 {
2465 fold_line(wp, fold_count, &win_foldinfo, lnum, row);
2466 ++row;
2467 --fold_count;
2468 wp->w_lines[idx].wl_folded = TRUE;
2469 wp->w_lines[idx].wl_lastlnum = lnum + fold_count;
2470# ifdef FEAT_SYN_HL
2471 did_update = DID_FOLD;
2472# endif
2473 }
2474 else
2475#endif
2476 if (idx < wp->w_lines_valid
2477 && wp->w_lines[idx].wl_valid
2478 && wp->w_lines[idx].wl_lnum == lnum
2479 && lnum > wp->w_topline
2480 && !(dy_flags & (DY_LASTLINE | DY_TRUNCATE))
2481 && !WIN_IS_POPUP(wp)
2482 && srow + wp->w_lines[idx].wl_size > wp->w_height
2483#ifdef FEAT_DIFF
2484 && diff_check_fill(wp, lnum) == 0
2485#endif
2486 )
2487 {
2488 // This line is not going to fit. Don't draw anything here,
2489 // will draw "@ " lines below.
2490 row = wp->w_height + 1;
2491 }
2492 else
2493 {
2494#ifdef FEAT_SEARCH_EXTRA
2495 prepare_search_hl(wp, &screen_search_hl, lnum);
2496#endif
2497#ifdef FEAT_SYN_HL
2498 // Let the syntax stuff know we skipped a few lines.
2499 if (syntax_last_parsed != 0 && syntax_last_parsed + 1 < lnum
2500 && syntax_present(wp))
2501 syntax_end_parsing(syntax_last_parsed + 1);
2502#endif
2503
2504 // Display one line.
2505 row = win_line(wp, lnum, srow, wp->w_height,
2506 mod_top == 0, FALSE);
2507
2508#ifdef FEAT_FOLDING
2509 wp->w_lines[idx].wl_folded = FALSE;
2510 wp->w_lines[idx].wl_lastlnum = lnum;
2511#endif
2512#ifdef FEAT_SYN_HL
2513 did_update = DID_LINE;
2514 syntax_last_parsed = lnum;
2515#endif
2516 }
2517
2518 wp->w_lines[idx].wl_lnum = lnum;
2519 wp->w_lines[idx].wl_valid = TRUE;
2520
2521 // Past end of the window or end of the screen. Note that after
2522 // resizing wp->w_height may be end up too big. That's a problem
2523 // elsewhere, but prevent a crash here.
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +01002524 if (row > wp->w_height
2525 || row + wp->w_winrow >= (p_ch > 0 ? Rows : Rows + 1))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002526 {
2527 // we may need the size of that too long line later on
2528 if (dollar_vcol == -1)
2529 wp->w_lines[idx].wl_size = plines_win(wp, lnum, TRUE);
2530 ++idx;
2531 break;
2532 }
2533 if (dollar_vcol == -1)
2534 wp->w_lines[idx].wl_size = row - srow;
2535 ++idx;
2536#ifdef FEAT_FOLDING
2537 lnum += fold_count + 1;
2538#else
2539 ++lnum;
2540#endif
2541 }
2542 else
2543 {
Lewis Russell16246392022-03-29 11:38:17 +01002544 if (wp->w_p_rnu && wp->w_last_cursor_lnum_rnu != wp->w_cursor.lnum)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002545 {
2546#ifdef FEAT_FOLDING
Lewis Russell16246392022-03-29 11:38:17 +01002547 // 'relativenumber' set and the cursor moved vertically: The
2548 // text doesn't need to be drawn, but the number column does.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002549 fold_count = foldedCount(wp, lnum, &win_foldinfo);
2550 if (fold_count != 0)
2551 fold_line(wp, fold_count, &win_foldinfo, lnum, row);
2552 else
2553#endif
2554 (void)win_line(wp, lnum, srow, wp->w_height, TRUE, TRUE);
2555 }
2556
2557 // This line does not need to be drawn, advance to the next one.
2558 row += wp->w_lines[idx++].wl_size;
2559 if (row > wp->w_height) // past end of screen
2560 break;
2561#ifdef FEAT_FOLDING
2562 lnum = wp->w_lines[idx - 1].wl_lastlnum + 1;
2563#else
2564 ++lnum;
2565#endif
2566#ifdef FEAT_SYN_HL
2567 did_update = DID_NONE;
2568#endif
2569 }
2570
2571 if (lnum > buf->b_ml.ml_line_count)
2572 {
2573 eof = TRUE;
2574 break;
2575 }
Bram Moolenaarfa1a4572022-01-16 11:42:20 +00002576
2577 // Safety check: if any of the wl_size values is wrong we might go over
2578 // the end of w_lines[].
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +01002579 if (idx >= (p_ch > 0 ? Rows : Rows + 1))
Bram Moolenaarfa1a4572022-01-16 11:42:20 +00002580 break;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002581 }
2582
2583 // End of loop over all window lines.
2584
zeertzjqc20e46a2022-03-23 14:55:23 +00002585#ifdef FEAT_SYN_HL
2586 // Now that the window has been redrawn with the old and new cursor line,
2587 // update w_last_cursorline.
2588 wp->w_last_cursorline = wp->w_p_cul ? wp->w_cursor.lnum : 0;
2589#endif
Lewis Russell16246392022-03-29 11:38:17 +01002590 wp->w_last_cursor_lnum_rnu = wp->w_p_rnu ? wp->w_cursor.lnum : 0;
zeertzjqc20e46a2022-03-23 14:55:23 +00002591
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002592#ifdef FEAT_VTP
2593 // Rewrite the character at the end of the screen line.
Bram Moolenaar7ed8f592020-04-28 20:44:42 +02002594 // See the version that was fixed.
2595 if (use_vtp() && get_conpty_fix_type() < 1)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002596 {
K.Takata54119102022-02-03 13:33:03 +00002597 int k;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002598
K.Takata54119102022-02-03 13:33:03 +00002599 for (k = 0; k < Rows; ++k)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002600 if (enc_utf8)
K.Takata54119102022-02-03 13:33:03 +00002601 if ((*mb_off2cells)(LineOffset[k] + Columns - 2,
2602 LineOffset[k] + screen_Columns) > 1)
2603 screen_draw_rectangle(k, Columns - 2, 1, 2, FALSE);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002604 else
K.Takata54119102022-02-03 13:33:03 +00002605 screen_draw_rectangle(k, Columns - 1, 1, 1, FALSE);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002606 else
K.Takata54119102022-02-03 13:33:03 +00002607 screen_char(LineOffset[k] + Columns - 1, k, Columns - 1);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002608 }
2609#endif
2610
2611 if (idx > wp->w_lines_valid)
2612 wp->w_lines_valid = idx;
2613
2614#ifdef FEAT_SYN_HL
2615 // Let the syntax stuff know we stop parsing here.
2616 if (syntax_last_parsed != 0 && syntax_present(wp))
2617 syntax_end_parsing(syntax_last_parsed + 1);
2618#endif
2619
2620 // If we didn't hit the end of the file, and we didn't finish the last
2621 // line we were working on, then the line didn't fit.
2622 wp->w_empty_rows = 0;
2623#ifdef FEAT_DIFF
2624 wp->w_filler_rows = 0;
2625#endif
2626 if (!eof && !didline)
2627 {
2628 if (lnum == wp->w_topline)
2629 {
2630 // Single line that does not fit!
2631 // Don't overwrite it, it can be edited.
2632 wp->w_botline = lnum + 1;
2633 }
2634#ifdef FEAT_DIFF
2635 else if (diff_check_fill(wp, lnum) >= wp->w_height - srow)
2636 {
2637 // Window ends in filler lines.
2638 wp->w_botline = lnum;
2639 wp->w_filler_rows = wp->w_height - srow;
2640 }
2641#endif
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002642#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002643 else if (WIN_IS_POPUP(wp))
2644 {
2645 // popup line that doesn't fit is left as-is
2646 wp->w_botline = lnum;
2647 }
2648#endif
2649 else if (dy_flags & DY_TRUNCATE) // 'display' has "truncate"
2650 {
2651 int scr_row = W_WINROW(wp) + wp->w_height - 1;
2652
2653 // Last line isn't finished: Display "@@@" in the last screen line.
Bram Moolenaarcee9c842022-04-09 12:40:13 +01002654 screen_puts_len((char_u *)"@@", wp->w_width > 2 ? 2 : wp->w_width,
2655 scr_row, wp->w_wincol, HL_ATTR(HLF_AT));
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002656 screen_fill(scr_row, scr_row + 1,
2657 (int)wp->w_wincol + 2, (int)W_ENDCOL(wp),
2658 '@', ' ', HL_ATTR(HLF_AT));
2659 set_empty_rows(wp, srow);
2660 wp->w_botline = lnum;
2661 }
2662 else if (dy_flags & DY_LASTLINE) // 'display' has "lastline"
2663 {
Bram Moolenaarcee9c842022-04-09 12:40:13 +01002664 int start_col = (int)W_ENDCOL(wp) - 3;
2665
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002666 // Last line isn't finished: Display "@@@" at the end.
2667 screen_fill(W_WINROW(wp) + wp->w_height - 1,
2668 W_WINROW(wp) + wp->w_height,
Bram Moolenaarcee9c842022-04-09 12:40:13 +01002669 start_col < wp->w_wincol ? wp->w_wincol : start_col,
2670 (int)W_ENDCOL(wp),
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002671 '@', '@', HL_ATTR(HLF_AT));
2672 set_empty_rows(wp, srow);
2673 wp->w_botline = lnum;
2674 }
2675 else
2676 {
2677 win_draw_end(wp, '@', ' ', TRUE, srow, wp->w_height, HLF_AT);
2678 wp->w_botline = lnum;
2679 }
2680 }
2681 else
2682 {
2683 draw_vsep_win(wp, row);
2684 if (eof) // we hit the end of the file
2685 {
2686 wp->w_botline = buf->b_ml.ml_line_count + 1;
2687#ifdef FEAT_DIFF
2688 j = diff_check_fill(wp, wp->w_botline);
2689 if (j > 0 && !wp->w_botfill)
2690 {
2691 // Display filler lines at the end of the file.
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002692 if (char2cells(wp->w_fill_chars.diff) > 1)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002693 i = '-';
2694 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002695 i = wp->w_fill_chars.diff;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002696 if (row + j > wp->w_height)
2697 j = wp->w_height - row;
2698 win_draw_end(wp, i, i, TRUE, row, row + (int)j, HLF_DED);
2699 row += j;
2700 }
2701#endif
2702 }
2703 else if (dollar_vcol == -1)
2704 wp->w_botline = lnum;
2705
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002706 // Make sure the rest of the screen is blank.
2707 // write the "eob" character from 'fillchars' to rows that aren't part
2708 // of the file.
Bram Moolenaar1666ac92019-11-10 17:22:31 +01002709 if (WIN_IS_POPUP(wp))
2710 win_draw_end(wp, ' ', ' ', FALSE, row, wp->w_height, HLF_AT);
2711 else
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01002712 win_draw_end(wp, wp->w_fill_chars.eob, ' ', FALSE,
2713 row, wp->w_height, HLF_EOB);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002714 }
2715
2716#ifdef SYN_TIME_LIMIT
Paul Ollis65745772022-06-05 16:55:54 +01002717 disable_regexp_timeout();
Bram Moolenaar6f0cf622022-06-19 12:27:45 +01002718 redrawtime_limit_set = FALSE;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002719#endif
2720
2721 // Reset the type of redrawing required, the window has been updated.
2722 wp->w_redr_type = 0;
2723#ifdef FEAT_DIFF
2724 wp->w_old_topfill = wp->w_topfill;
2725 wp->w_old_botfill = wp->w_botfill;
2726#endif
2727
2728 if (dollar_vcol == -1)
2729 {
2730 // There is a trick with w_botline. If we invalidate it on each
2731 // change that might modify it, this will cause a lot of expensive
2732 // calls to plines() in update_topline() each time. Therefore the
2733 // value of w_botline is often approximated, and this value is used to
2734 // compute the value of w_topline. If the value of w_botline was
2735 // wrong, check that the value of w_topline is correct (cursor is on
2736 // the visible part of the text). If it's not, we need to redraw
2737 // again. Mostly this just means scrolling up a few lines, so it
2738 // doesn't look too bad. Only do this for the current window (where
2739 // changes are relevant).
2740 wp->w_valid |= VALID_BOTLINE;
2741 if (wp == curwin && wp->w_botline != old_botline && !recursive)
2742 {
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002743 win_T *wwp;
2744#if defined(FEAT_CONCEAL)
2745 linenr_T old_topline = wp->w_topline;
2746 int new_wcol = wp->w_wcol;
2747#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002748 recursive = TRUE;
2749 curwin->w_valid &= ~VALID_TOPLINE;
2750 update_topline(); // may invalidate w_botline again
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002751
2752#if defined(FEAT_CONCEAL)
2753 if (old_wcol != new_wcol && (wp->w_valid & (VALID_WCOL|VALID_WROW))
2754 != (VALID_WCOL|VALID_WROW))
2755 {
2756 // A win_line() call applied a fix to screen cursor column to
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002757 // accommodate concealment of cursor line, but in this call to
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002758 // update_topline() the cursor's row or column got invalidated.
2759 // If they are left invalid, setcursor() will recompute them
2760 // but there won't be any further win_line() call to re-fix the
2761 // column and the cursor will end up misplaced. So we call
2762 // cursor validation now and reapply the fix again (or call
2763 // win_line() to do it for us).
2764 validate_cursor();
2765 if (wp->w_wcol == old_wcol && wp->w_wrow == old_wrow
2766 && old_topline == wp->w_topline)
2767 wp->w_wcol = new_wcol;
2768 else
2769 redrawWinline(wp, wp->w_cursor.lnum);
2770 }
2771#endif
2772 // New redraw either due to updated topline or due to wcol fix.
2773 if (wp->w_redr_type != 0)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002774 {
2775 // Don't update for changes in buffer again.
2776 i = curbuf->b_mod_set;
2777 curbuf->b_mod_set = FALSE;
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002778 j = curbuf->b_mod_xlines;
2779 curbuf->b_mod_xlines = 0;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002780 win_update(curwin);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002781 curbuf->b_mod_set = i;
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002782 curbuf->b_mod_xlines = j;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002783 }
Bram Moolenaarcbee6352019-11-12 20:49:15 +01002784 // Other windows might have w_redr_type raised in update_topline().
2785 must_redraw = 0;
2786 FOR_ALL_WINDOWS(wwp)
2787 if (wwp->w_redr_type > must_redraw)
2788 must_redraw = wwp->w_redr_type;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002789 recursive = FALSE;
2790 }
2791 }
2792
2793#if defined(FEAT_SYN_HL) || defined(FEAT_SEARCH_EXTRA)
2794 // restore got_int, unless CTRL-C was hit while redrawing
2795 if (!got_int)
2796 got_int = save_got_int;
2797#endif
2798}
2799
2800#if defined(FEAT_NETBEANS_INTG) || defined(FEAT_GUI)
2801/*
2802 * Prepare for updating one or more windows.
2803 * Caller must check for "updating_screen" already set to avoid recursiveness.
2804 */
2805 static void
2806update_prepare(void)
2807{
2808 cursor_off();
2809 updating_screen = TRUE;
2810#ifdef FEAT_GUI
2811 // Remove the cursor before starting to do anything, because scrolling may
2812 // make it difficult to redraw the text under it.
2813 if (gui.in_use)
2814 gui_undraw_cursor();
2815#endif
2816#ifdef FEAT_SEARCH_EXTRA
2817 start_search_hl();
2818#endif
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002819#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002820 // Update popup_mask if needed.
2821 may_update_popup_mask(must_redraw);
2822#endif
2823}
2824
2825/*
2826 * Finish updating one or more windows.
2827 */
2828 static void
2829update_finish(void)
2830{
2831 if (redraw_cmdline || redraw_mode)
2832 showmode();
2833
2834# ifdef FEAT_SEARCH_EXTRA
2835 end_search_hl();
2836# endif
2837
2838 after_updating_screen(TRUE);
2839
2840# ifdef FEAT_GUI
2841 // Redraw the cursor and update the scrollbars when all screen updating is
2842 // done.
2843 if (gui.in_use)
2844 {
2845 out_flush_cursor(FALSE, FALSE);
2846 gui_update_scrollbars(FALSE);
2847 }
2848# endif
2849}
2850#endif
2851
2852#if defined(FEAT_NETBEANS_INTG) || defined(PROTO)
2853 void
2854update_debug_sign(buf_T *buf, linenr_T lnum)
2855{
2856 win_T *wp;
2857 int doit = FALSE;
2858
2859# ifdef FEAT_FOLDING
2860 win_foldinfo.fi_level = 0;
2861# endif
2862
2863 // update/delete a specific sign
2864 redraw_buf_line_later(buf, lnum);
2865
2866 // check if it resulted in the need to redraw a window
2867 FOR_ALL_WINDOWS(wp)
2868 if (wp->w_redr_type != 0)
2869 doit = TRUE;
2870
2871 // Return when there is nothing to do, screen updating is already
2872 // happening (recursive call), messages on the screen or still starting up.
2873 if (!doit || updating_screen
Bram Moolenaar24959102022-05-07 20:01:16 +01002874 || State == MODE_ASKMORE || State == MODE_HITRETURN
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002875 || msg_scrolled
2876#ifdef FEAT_GUI
2877 || gui.starting
2878#endif
2879 || starting)
2880 return;
2881
2882 // update all windows that need updating
2883 update_prepare();
2884
2885 FOR_ALL_WINDOWS(wp)
2886 {
2887 if (wp->w_redr_type != 0)
2888 win_update(wp);
2889 if (wp->w_redr_status)
2890 win_redr_status(wp, FALSE);
2891 }
2892
2893 update_finish();
2894}
2895#endif
2896
2897#if defined(FEAT_GUI) || defined(PROTO)
2898/*
2899 * Update a single window, its status line and maybe the command line msg.
2900 * Used for the GUI scrollbar.
2901 */
2902 void
2903updateWindow(win_T *wp)
2904{
2905 // return if already busy updating
2906 if (updating_screen)
2907 return;
2908
2909 update_prepare();
2910
2911#ifdef FEAT_CLIPBOARD
2912 // When Visual area changed, may have to update selection.
2913 if (clip_star.available && clip_isautosel_star())
2914 clip_update_selection(&clip_star);
2915 if (clip_plus.available && clip_isautosel_plus())
2916 clip_update_selection(&clip_plus);
2917#endif
2918
2919 win_update(wp);
2920
2921 // When the screen was cleared redraw the tab pages line.
2922 if (redraw_tabline)
2923 draw_tabline();
2924
2925 if (wp->w_redr_status
2926# ifdef FEAT_CMDL_INFO
2927 || p_ru
2928# endif
2929# ifdef FEAT_STL_OPT
2930 || *p_stl != NUL || *wp->w_p_stl != NUL
2931# endif
2932 )
2933 win_redr_status(wp, FALSE);
2934
Bram Moolenaar05ad5ff2019-11-30 22:48:27 +01002935#ifdef FEAT_PROP_POPUP
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002936 // Display popup windows on top of everything.
2937 update_popups(win_update);
2938#endif
2939
2940 update_finish();
2941}
2942#endif
2943
2944#if defined(FEAT_TERMRESPONSE) || defined(PROTO)
2945/*
2946 * Redraw as soon as possible. When the command line is not scrolled redraw
2947 * right away and restore what was on the command line.
2948 * Return a code indicating what happened.
2949 */
2950 int
2951redraw_asap(int type)
2952{
2953 int rows;
2954 int cols = screen_Columns;
2955 int r;
2956 int ret = 0;
2957 schar_T *screenline; // copy from ScreenLines[]
2958 sattr_T *screenattr; // copy from ScreenAttrs[]
2959 int i;
2960 u8char_T *screenlineUC = NULL; // copy from ScreenLinesUC[]
2961 u8char_T *screenlineC[MAX_MCO]; // copy from ScreenLinesC[][]
2962 schar_T *screenline2 = NULL; // copy from ScreenLines2[]
2963
2964 redraw_later(type);
Shougo Matsushitaf39cfb72022-07-30 16:54:05 +01002965 if (msg_scrolled
2966 || (State != MODE_NORMAL && State != MODE_NORMAL_BUSY)
2967 || exiting
2968 || p_ch == 0)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02002969 return ret;
2970
2971 // Allocate space to save the text displayed in the command line area.
2972 rows = screen_Rows - cmdline_row;
2973 screenline = LALLOC_MULT(schar_T, rows * cols);
2974 screenattr = LALLOC_MULT(sattr_T, rows * cols);
2975 if (screenline == NULL || screenattr == NULL)
2976 ret = 2;
2977 if (enc_utf8)
2978 {
2979 screenlineUC = LALLOC_MULT(u8char_T, rows * cols);
2980 if (screenlineUC == NULL)
2981 ret = 2;
2982 for (i = 0; i < p_mco; ++i)
2983 {
2984 screenlineC[i] = LALLOC_MULT(u8char_T, rows * cols);
2985 if (screenlineC[i] == NULL)
2986 ret = 2;
2987 }
2988 }
2989 if (enc_dbcs == DBCS_JPNU)
2990 {
2991 screenline2 = LALLOC_MULT(schar_T, rows * cols);
2992 if (screenline2 == NULL)
2993 ret = 2;
2994 }
2995
2996 if (ret != 2)
2997 {
2998 // Save the text displayed in the command line area.
2999 for (r = 0; r < rows; ++r)
3000 {
3001 mch_memmove(screenline + r * cols,
3002 ScreenLines + LineOffset[cmdline_row + r],
3003 (size_t)cols * sizeof(schar_T));
3004 mch_memmove(screenattr + r * cols,
3005 ScreenAttrs + LineOffset[cmdline_row + r],
3006 (size_t)cols * sizeof(sattr_T));
3007 if (enc_utf8)
3008 {
3009 mch_memmove(screenlineUC + r * cols,
3010 ScreenLinesUC + LineOffset[cmdline_row + r],
3011 (size_t)cols * sizeof(u8char_T));
3012 for (i = 0; i < p_mco; ++i)
3013 mch_memmove(screenlineC[i] + r * cols,
3014 ScreenLinesC[i] + LineOffset[cmdline_row + r],
3015 (size_t)cols * sizeof(u8char_T));
3016 }
3017 if (enc_dbcs == DBCS_JPNU)
3018 mch_memmove(screenline2 + r * cols,
3019 ScreenLines2 + LineOffset[cmdline_row + r],
3020 (size_t)cols * sizeof(schar_T));
3021 }
3022
3023 update_screen(0);
3024 ret = 3;
3025
3026 if (must_redraw == 0)
3027 {
3028 int off = (int)(current_ScreenLine - ScreenLines);
3029
3030 // Restore the text displayed in the command line area.
3031 for (r = 0; r < rows; ++r)
3032 {
3033 mch_memmove(current_ScreenLine,
3034 screenline + r * cols,
3035 (size_t)cols * sizeof(schar_T));
3036 mch_memmove(ScreenAttrs + off,
3037 screenattr + r * cols,
3038 (size_t)cols * sizeof(sattr_T));
3039 if (enc_utf8)
3040 {
3041 mch_memmove(ScreenLinesUC + off,
3042 screenlineUC + r * cols,
3043 (size_t)cols * sizeof(u8char_T));
3044 for (i = 0; i < p_mco; ++i)
3045 mch_memmove(ScreenLinesC[i] + off,
3046 screenlineC[i] + r * cols,
3047 (size_t)cols * sizeof(u8char_T));
3048 }
3049 if (enc_dbcs == DBCS_JPNU)
3050 mch_memmove(ScreenLines2 + off,
3051 screenline2 + r * cols,
3052 (size_t)cols * sizeof(schar_T));
Bram Moolenaar96ba25a2022-07-04 17:34:33 +01003053 screen_line(curwin, cmdline_row + r, 0, cols, cols, 0);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003054 }
3055 ret = 4;
3056 }
3057 }
3058
3059 vim_free(screenline);
3060 vim_free(screenattr);
3061 if (enc_utf8)
3062 {
3063 vim_free(screenlineUC);
3064 for (i = 0; i < p_mco; ++i)
3065 vim_free(screenlineC[i]);
3066 }
3067 if (enc_dbcs == DBCS_JPNU)
3068 vim_free(screenline2);
3069
3070 // Show the intro message when appropriate.
3071 maybe_intro_message();
3072
3073 setcursor();
3074
3075 return ret;
3076}
3077#endif
3078
3079/*
3080 * Invoked after an asynchronous callback is called.
3081 * If an echo command was used the cursor needs to be put back where
3082 * it belongs. If highlighting was changed a redraw is needed.
3083 * If "call_update_screen" is FALSE don't call update_screen() when at the
3084 * command line.
Bram Moolenaare5050712021-12-09 10:51:05 +00003085 * If "redraw_message" is TRUE.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003086 */
3087 void
Bram Moolenaare5050712021-12-09 10:51:05 +00003088redraw_after_callback(int call_update_screen, int do_message)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003089{
3090 ++redrawing_for_callback;
3091
Bram Moolenaar24959102022-05-07 20:01:16 +01003092 if (State == MODE_HITRETURN || State == MODE_ASKMORE
3093 || State == MODE_SETWSIZE || State == MODE_EXTERNCMD
3094 || State == MODE_CONFIRM || exmode_active)
Bram Moolenaare5050712021-12-09 10:51:05 +00003095 {
3096 if (do_message)
3097 repeat_message();
3098 }
Bram Moolenaar24959102022-05-07 20:01:16 +01003099 else if (State & MODE_CMDLINE)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003100 {
Yegappan Lakshmanan3908ef52022-02-08 12:08:07 +00003101 if (pum_visible())
3102 cmdline_pum_display();
Bram Moolenaar54162322022-08-26 16:58:51 +01003103
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003104 // Don't redraw when in prompt_for_number().
3105 if (cmdline_row > 0)
3106 {
3107 // Redrawing only works when the screen didn't scroll. Don't clear
3108 // wildmenu entries.
3109 if (msg_scrolled == 0
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003110 && wild_menu_showing == 0
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003111 && call_update_screen)
3112 update_screen(0);
3113
3114 // Redraw in the same position, so that the user can continue
3115 // editing the command.
3116 redrawcmdline_ex(FALSE);
3117 }
3118 }
Bram Moolenaar24959102022-05-07 20:01:16 +01003119 else if (State & (MODE_NORMAL | MODE_INSERT | MODE_TERMINAL))
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003120 {
zeertzjq3e559cd2022-03-27 19:26:55 +01003121 update_topline();
3122 validate_cursor();
3123
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003124 // keep the command line if possible
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003125 update_screen(UPD_VALID_NO_UPDATE);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003126 setcursor();
Bram Moolenaar9f284162021-04-22 21:39:30 +02003127
3128 if (msg_scrolled == 0)
3129 {
3130 // don't want a hit-enter prompt when something else is displayed
3131 msg_didany = FALSE;
3132 need_wait_return = FALSE;
3133 }
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003134 }
3135 cursor_on();
3136#ifdef FEAT_GUI
3137 if (gui.in_use && !gui_mch_is_blink_off())
3138 // Don't update the cursor when it is blinking and off to avoid
3139 // flicker.
3140 out_flush_cursor(FALSE, FALSE);
3141 else
3142#endif
3143 out_flush();
3144
3145 --redrawing_for_callback;
3146}
3147
3148/*
3149 * Redraw the current window later, with update_screen(type).
3150 * Set must_redraw only if not already set to a higher value.
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003151 * E.g. if must_redraw is UPD_CLEAR, type UPD_NOT_VALID will do nothing.
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003152 */
3153 void
3154redraw_later(int type)
3155{
3156 redraw_win_later(curwin, type);
3157}
3158
3159 void
3160redraw_win_later(
3161 win_T *wp,
3162 int type)
3163{
Bram Moolenaar471c0fa2022-08-22 15:19:16 +01003164 if (!exiting && !redraw_not_allowed && wp->w_redr_type < type)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003165 {
3166 wp->w_redr_type = type;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003167 if (type >= UPD_NOT_VALID)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003168 wp->w_lines_valid = 0;
3169 if (must_redraw < type) // must_redraw is the maximum of all windows
3170 must_redraw = type;
3171 }
3172}
3173
3174/*
3175 * Force a complete redraw later. Also resets the highlighting. To be used
3176 * after executing a shell command that messes up the screen.
3177 */
3178 void
3179redraw_later_clear(void)
3180{
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003181 redraw_all_later(UPD_CLEAR);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003182 reset_screen_attr();
3183}
3184
3185/*
3186 * Mark all windows to be redrawn later.
3187 */
3188 void
3189redraw_all_later(int type)
3190{
3191 win_T *wp;
3192
3193 FOR_ALL_WINDOWS(wp)
3194 redraw_win_later(wp, type);
3195 // This may be needed when switching tabs.
Bram Moolenaar471c0fa2022-08-22 15:19:16 +01003196 set_must_redraw(type);
3197}
3198
3199/*
3200 * Set "must_redraw" to "type" unless it already has a higher value
3201 * or it is currently not allowed.
3202 */
3203 void
3204set_must_redraw(int type)
3205{
3206 if (!redraw_not_allowed && must_redraw < type)
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003207 must_redraw = type;
3208}
3209
3210/*
3211 * Mark all windows that are editing the current buffer to be updated later.
3212 */
3213 void
3214redraw_curbuf_later(int type)
3215{
3216 redraw_buf_later(curbuf, type);
3217}
3218
3219 void
3220redraw_buf_later(buf_T *buf, int type)
3221{
3222 win_T *wp;
3223
3224 FOR_ALL_WINDOWS(wp)
3225 {
3226 if (wp->w_buffer == buf)
3227 redraw_win_later(wp, type);
3228 }
Bram Moolenaare52e0c82020-02-28 22:20:10 +01003229#if defined(FEAT_TERMINAL) && defined(FEAT_PROP_POPUP)
3230 // terminal in popup window is not in list of windows
3231 if (curwin->w_buffer == buf)
3232 redraw_win_later(curwin, type);
3233#endif
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003234}
3235
3236#if defined(FEAT_SIGNS) || defined(PROTO)
3237 void
3238redraw_buf_line_later(buf_T *buf, linenr_T lnum)
3239{
3240 win_T *wp;
3241
3242 FOR_ALL_WINDOWS(wp)
3243 if (wp->w_buffer == buf && lnum >= wp->w_topline
3244 && lnum < wp->w_botline)
3245 redrawWinline(wp, lnum);
3246}
3247#endif
3248
3249#if defined(FEAT_JOB_CHANNEL) || defined(PROTO)
3250 void
3251redraw_buf_and_status_later(buf_T *buf, int type)
3252{
3253 win_T *wp;
3254
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003255 if (wild_menu_showing != 0)
3256 // Don't redraw while the command line completion is displayed, it
3257 // would disappear.
3258 return;
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003259 FOR_ALL_WINDOWS(wp)
3260 {
3261 if (wp->w_buffer == buf)
3262 {
3263 redraw_win_later(wp, type);
3264 wp->w_redr_status = TRUE;
3265 }
3266 }
3267}
3268#endif
3269
3270/*
3271 * mark all status lines for redraw; used after first :cd
3272 */
3273 void
3274status_redraw_all(void)
3275{
3276 win_T *wp;
3277
3278 FOR_ALL_WINDOWS(wp)
3279 if (wp->w_status_height)
3280 {
3281 wp->w_redr_status = TRUE;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003282 redraw_later(UPD_VALID);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003283 }
3284}
3285
3286/*
3287 * mark all status lines of the current buffer for redraw
3288 */
3289 void
3290status_redraw_curbuf(void)
3291{
3292 win_T *wp;
3293
3294 FOR_ALL_WINDOWS(wp)
3295 if (wp->w_status_height != 0 && wp->w_buffer == curbuf)
3296 {
3297 wp->w_redr_status = TRUE;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003298 redraw_later(UPD_VALID);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003299 }
3300}
3301
3302/*
3303 * Redraw all status lines that need to be redrawn.
3304 */
3305 void
3306redraw_statuslines(void)
3307{
3308 win_T *wp;
3309
3310 FOR_ALL_WINDOWS(wp)
3311 if (wp->w_redr_status)
3312 win_redr_status(wp, FALSE);
3313 if (redraw_tabline)
3314 draw_tabline();
3315}
3316
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003317/*
3318 * Redraw all status lines at the bottom of frame "frp".
3319 */
3320 void
3321win_redraw_last_status(frame_T *frp)
3322{
3323 if (frp->fr_layout == FR_LEAF)
3324 frp->fr_win->w_redr_status = TRUE;
3325 else if (frp->fr_layout == FR_ROW)
3326 {
3327 FOR_ALL_FRAMES(frp, frp->fr_child)
3328 win_redraw_last_status(frp);
3329 }
3330 else // frp->fr_layout == FR_COL
3331 {
3332 frp = frp->fr_child;
3333 while (frp->fr_next != NULL)
3334 frp = frp->fr_next;
3335 win_redraw_last_status(frp);
3336 }
3337}
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003338
3339/*
3340 * Changed something in the current window, at buffer line "lnum", that
3341 * requires that line and possibly other lines to be redrawn.
3342 * Used when entering/leaving Insert mode with the cursor on a folded line.
3343 * Used to remove the "$" from a change command.
3344 * Note that when also inserting/deleting lines w_redraw_top and w_redraw_bot
3345 * may become invalid and the whole window will have to be redrawn.
3346 */
3347 void
3348redrawWinline(
3349 win_T *wp,
3350 linenr_T lnum)
3351{
3352 if (wp->w_redraw_top == 0 || wp->w_redraw_top > lnum)
3353 wp->w_redraw_top = lnum;
3354 if (wp->w_redraw_bot == 0 || wp->w_redraw_bot < lnum)
3355 wp->w_redraw_bot = lnum;
Bram Moolenaara4d158b2022-08-14 14:17:45 +01003356 redraw_win_later(wp, UPD_VALID);
Bram Moolenaar7528d1f2019-09-19 23:06:20 +02003357}