Problem: Slow redrawing with DirectX.
Solution: Avoid calling gui_mch_flush() unnecessarily, especially when
updating the cursor. (Ken Takata, closes #2560)
geom pixelGeometry int 0 - 2 (see below)
renmode renderingMode int 0 - 6 (see below)
taamode textAntialiasMode int 0 - 3 (see below)
- scrlines Scroll Lines int >= 0 (see below)
+ scrlines Scroll Lines int (deprecated)
See this URL for detail (except for scrlines):
https://msdn.microsoft.com/en-us/library/dd368190.aspx
See this URL for detail:
https://msdn.microsoft.com/en-us/library/dd368170.aspx
- For scrlines: threshold for lines to be scrolled.
- 0 - Always use scrolling. (default)
- 1 - Use full page redrawing.
- > 1 - If the lines to be scrolled is grater or equal to the
- specified value, use redrawing. Otherwise use
- scrolling.
-
- If you feel scrolling a page (CTRL-F) is too slow with DirectX
- renderer, try this "scrlines" option.
- When set it "1", Vim uses full page redrawing instead of
- scrolling. Redrawing a page is faster than scrolling a
- page in some environments.
- After that, when you feel scrolling lines (CTRL-Y) becomes
- slow, please try "2" or greater value for this option.
- It works threshold line number to switch scrolling to
- redrawing. Scrolling a few lines might be faster than
- redrawing a page in some environments.
+ For scrlines:
+ This was used for optimizing scrolling behavior, however this
+ is now deprecated. If specified, it is simply ignored.
Example: >
set encoding=utf-8
ex_redraw(&ea);
showruler(FALSE);
setcursor();
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
- }
-#endif
+ out_flush_cursor(TRUE, FALSE);
}
else if (STRCMP(cmd, "expr") == 0 || STRCMP(cmd, "call") == 0)
{
compl_orig_text = NULL;
compl_enter_selects = FALSE;
/* clear v:completed_item */
- set_vim_var_dict(VV_COMPLETED_ITEM, dict_alloc());
+ set_vim_var_dict(VV_COMPLETED_ITEM, dict_alloc_lock(VAR_FIXED));
}
/*
{
/* Show the cursor after the match, not after the redrawn text. */
setcursor();
- out_flush();
- gui_update_cursor(FALSE, FALSE);
+ out_flush_cursor(FALSE, FALSE);
}
#endif
compl_restarting = TRUE;
* flicker, thus we can't do that. */
changed_cline_bef_curs();
/* clear v:completed_item */
- set_vim_var_dict(VV_COMPLETED_ITEM, dict_alloc());
+ set_vim_var_dict(VV_COMPLETED_ITEM, dict_alloc_lock(VAR_FIXED));
}
/*
/* Set completed item. */
/* { word, abbr, menu, kind, info } */
- dict = dict_alloc();
+ dict = dict_alloc_lock(VAR_FIXED);
if (dict != NULL)
{
dict_add_nr_str(dict, "word", 0L,
{
/* Show the cursor after the match, not after the redrawn text. */
setcursor();
- out_flush();
- gui_update_cursor(FALSE, FALSE);
+ out_flush_cursor(FALSE, FALSE);
}
#endif
if (wait_time == -1L || wait_time > 100L) /* flush output before waiting */
{
cursor_on();
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(FALSE, FALSE);
-# ifdef FEAT_MOUSESHAPE
- if (postponed_mouseshape)
- update_mouseshape(-1);
-# endif
- }
+ out_flush_cursor(FALSE, FALSE);
+#if defined(FEAT_GUI) && defined(FEAT_MOUSESHAPE)
+ if (gui.in_use && postponed_mouseshape)
+ update_mouseshape(-1);
#endif
}
static void gui_attempt_start(void);
static int can_update_cursor = TRUE; /* can display the cursor */
+static int disable_flush = 0; /* If > 0, gui_mch_flush() is disabled. */
/*
* The Athena scrollbars can move the thumb to after the end of the scrollbar,
gui.dragged_sb = SBAR_NONE;
#endif
- gui_mch_flush(); /* In case vim decides to take a nap */
+ gui_may_flush(); /* In case vim decides to take a nap */
}
/*
* after scrolling. */
}
+/*
+ * Disable issuing gui_mch_flush().
+ */
+ void
+gui_disable_flush(void)
+{
+ ++disable_flush;
+}
+
+/*
+ * Enable issuing gui_mch_flush().
+ */
+ void
+gui_enable_flush(void)
+{
+ --disable_flush;
+}
+
+/*
+ * Issue gui_mch_flush() if it is not disabled.
+ */
+ void
+gui_may_flush(void)
+{
+ if (disable_flush == 0)
+ gui_mch_flush();
+}
+
static void
gui_outstr(char_u *s, int len)
{
/* Updating the tabline uses direct GUI commands, flush
* outstanding instructions first. (esp. clear screen) */
out_flush();
- gui_mch_flush();
if (!showit != !shown)
gui_mch_show_tabline(showit);
setcursor();
}
# endif
- out_flush();
- gui_update_cursor(FALSE, TRUE);
+ out_flush_cursor(FALSE, TRUE);
#else
add_to_input_buf(bytes, byte_count);
add_long_to_buf((long_u)value, bytes);
* disappear when losing focus after a scrollbar drag. */
if (wp->w_redr_type < type)
wp->w_redr_type = type;
+ mch_disable_flush();
updateWindow(wp); /* update window, status line, and cmdline */
+ mch_enable_flush();
}
#ifdef FEAT_INS_EXPAND
*/
#if 1
gui.in_focus = in_focus;
- out_flush(); /* make sure output has been written */
- gui_update_cursor(TRUE, FALSE);
+ out_flush_cursor(TRUE, FALSE);
# ifdef FEAT_XIM
xim_set_focus(in_focus);
curwin->w_valid &= ~VALID_CROW;
}
# endif
- out_flush(); /* make sure output has been written */
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
+ out_flush_cursor(TRUE, FALSE);
}
#endif
maketitle();
#endif
setcursor();
- out_flush();
- gui_update_cursor(FALSE, FALSE);
- gui_mch_flush();
+ out_flush_cursor(FALSE, FALSE);
}
entered = FALSE;
ID2D1DCRenderTarget *mRT;
ID2D1GdiInteropRenderTarget *mGDIRT;
ID2D1SolidColorBrush *mBrush;
+ ID2D1Bitmap *mBitmap;
IDWriteFactory *mDWriteFactory;
#ifdef FEAT_DIRECTX_COLOR_EMOJI
void SetFont(HFONT hFont);
+ void Rebind();
+
void BindDC(HDC hdc, const RECT *rect);
HRESULT SetDrawingMode(DrawingMode mode);
void SetPixel(int x, int y, COLORREF color);
+ void Scroll(int x, int y, const RECT *rc);
+
void Flush();
void SetRenderingParams(
mRT(NULL),
mGDIRT(NULL),
mBrush(NULL),
+ mBitmap(NULL),
mDWriteFactory(NULL),
#ifdef FEAT_DIRECTX_COLOR_EMOJI
mDWriteFactory2(NULL),
reinterpret_cast<void**>(&mD2D1Factory));
_RPT2(_CRT_WARN, "D2D1CreateFactory: hr=%p p=%p\n", hr, mD2D1Factory);
- if (SUCCEEDED(hr))
- hr = CreateDeviceResources();
-
if (SUCCEEDED(hr))
{
hr = DWriteCreateFactory(
#ifdef FEAT_DIRECTX_COLOR_EMOJI
SafeRelease(&mDWriteFactory2);
#endif
+ SafeRelease(&mBitmap);
SafeRelease(&mBrush);
SafeRelease(&mGDIRT);
SafeRelease(&mRT);
}
if (SUCCEEDED(hr))
- {
- if (mHDC != NULL)
- {
- mRT->BindDC(mHDC, &mBindRect);
- mRT->SetTransform(D2D1::IdentityMatrix());
- }
- }
+ Rebind();
return hr;
}
void
DWriteContext::DiscardDeviceResources()
{
+ SafeRelease(&mBitmap);
SafeRelease(&mBrush);
SafeRelease(&mGDIRT);
SafeRelease(&mRT);
}
void
-DWriteContext::BindDC(HDC hdc, const RECT *rect)
+DWriteContext::Rebind()
{
- Flush();
- mRT->BindDC(hdc, rect);
+ SafeRelease(&mBitmap);
+
+ mRT->BindDC(mHDC, &mBindRect);
mRT->SetTransform(D2D1::IdentityMatrix());
+
+ D2D1_BITMAP_PROPERTIES props = {
+ {DXGI_FORMAT_B8G8R8A8_UNORM, D2D1_ALPHA_MODE_IGNORE},
+ 96.0f, 96.0f
+ };
+ mRT->CreateBitmap(
+ D2D1::SizeU(mBindRect.right - mBindRect.left,
+ mBindRect.bottom - mBindRect.top),
+ props, &mBitmap);
+}
+
+ void
+DWriteContext::BindDC(HDC hdc, const RECT *rect)
+{
mHDC = hdc;
mBindRect = *rect;
+
+ if (mRT == NULL)
+ CreateDeviceResources();
+ else
+ {
+ Flush();
+ Rebind();
+ }
}
HRESULT
}
}
+ void
+DWriteContext::Scroll(int x, int y, const RECT *rc)
+{
+ SetDrawingMode(DM_DIRECTX);
+ mRT->Flush();
+
+ D2D1_RECT_U srcRect;
+ D2D1_POINT_2U destPoint;
+ if (x >= 0)
+ {
+ srcRect.left = rc->left;
+ srcRect.right = rc->right - x;
+ destPoint.x = rc->left + x;
+ }
+ else
+ {
+ srcRect.left = rc->left - x;
+ srcRect.right = rc->right;
+ destPoint.x = rc->left;
+ }
+ if (y >= 0)
+ {
+ srcRect.top = rc->top;
+ srcRect.bottom = rc->bottom - y;
+ destPoint.y = rc->top + y;
+ }
+ else
+ {
+ srcRect.top = rc->top - y;
+ srcRect.bottom = rc->bottom;
+ destPoint.y = rc->top;
+ }
+ mBitmap->CopyFromRenderTarget(&destPoint, mRT, &srcRect);
+
+ D2D1_RECT_F destRect = {
+ FLOAT(destPoint.x), FLOAT(destPoint.y),
+ FLOAT(destPoint.x + srcRect.right - srcRect.left),
+ FLOAT(destPoint.y + srcRect.bottom - srcRect.top)
+ };
+ mRT->DrawBitmap(mBitmap, destRect, 1.0F,
+ D2D1_BITMAP_INTERPOLATION_MODE_NEAREST_NEIGHBOR, destRect);
+}
+
void
DWriteContext::Flush()
{
ctx->SetPixel(x, y, color);
}
+ void
+DWriteContext_Scroll(DWriteContext *ctx, int x, int y, const RECT *rc)
+{
+ if (ctx != NULL)
+ ctx->Scroll(x, y, rc);
+}
+
void
DWriteContext_Flush(DWriteContext *ctx)
{
void DWriteContext_DrawLine(DWriteContext *ctx, int x1, int y1, int x2, int y2,
COLORREF color);
void DWriteContext_SetPixel(DWriteContext *ctx, int x, int y, COLORREF color);
+void DWriteContext_Scroll(DWriteContext *ctx, int x, int y, const RECT *rc);
void DWriteContext_Flush(DWriteContext *ctx);
void DWriteContext_Close(DWriteContext *ctx);
static DWriteContext *s_dwc = NULL;
static int s_directx_enabled = 0;
static int s_directx_load_attempted = 0;
-static int s_directx_scrlines = 0;
# define IS_ENABLE_DIRECTX() (s_directx_enabled && s_dwc != NULL && enc_utf8)
static int directx_enabled(void);
static void directx_binddc(void);
int dx_geom = 0;
int dx_renmode = 0;
int dx_taamode = 0;
- int dx_scrlines = 0;
/* parse string as rendering options. */
for (p = s; p != NULL && *p != NUL; )
}
else if (STRCMP(name, "scrlines") == 0)
{
- dx_scrlines = atoi((char *)value);
+ /* Deprecated. Simply ignore it. */
}
else
return FAIL;
}
}
s_directx_enabled = dx_enable;
- s_directx_scrlines = dx_scrlines;
return OK;
# else
int num_lines)
{
RECT rc;
-#if defined(FEAT_DIRECTX)
- int use_redraw = 0;
-#endif
rc.left = FILL_X(gui.scroll_region_left);
rc.right = FILL_X(gui.scroll_region_right + 1);
#if defined(FEAT_DIRECTX)
if (IS_ENABLE_DIRECTX())
{
- if (s_directx_scrlines > 0 && s_directx_scrlines <= num_lines)
- {
- gui_redraw(rc.left, rc.top,
- rc.right - rc.left + 1, rc.bottom - rc.top + 1);
- use_redraw = 1;
- }
- else
- DWriteContext_Flush(s_dwc);
+ DWriteContext_Scroll(s_dwc, 0, -num_lines * gui.char_height, &rc);
+ DWriteContext_Flush(s_dwc);
}
- if (!use_redraw)
+ else
#endif
{
intel_gpu_workaround();
int num_lines)
{
RECT rc;
-#if defined(FEAT_DIRECTX)
- int use_redraw = 0;
-#endif
rc.left = FILL_X(gui.scroll_region_left);
rc.right = FILL_X(gui.scroll_region_right + 1);
#if defined(FEAT_DIRECTX)
if (IS_ENABLE_DIRECTX())
{
- if (s_directx_scrlines > 0 && s_directx_scrlines <= num_lines)
- {
- gui_redraw(rc.left, rc.top,
- rc.right - rc.left + 1, rc.bottom - rc.top + 1);
- use_redraw = 1;
- }
- else
- DWriteContext_Flush(s_dwc);
+ DWriteContext_Scroll(s_dwc, 0, num_lines * gui.char_height, &rc);
+ DWriteContext_Flush(s_dwc);
}
- if (!use_redraw)
+ else
#endif
{
intel_gpu_workaround();
* position, but don't actually scroll by setting "dont_scroll". */
dont_scroll = !allow_scrollbar;
+ mch_disable_flush();
gui_drag_scrollbar(sb, val, dragging);
+ mch_enable_flush();
+ gui_may_flush();
s_busy_processing = FALSE;
dont_scroll = dont_scroll_save;
if (mouse_scroll_lines == 0)
init_mouse_wheel();
+ mch_disable_flush();
if (mouse_scroll_lines > 0
&& mouse_scroll_lines < (size > 2 ? size - 2 : 1))
{
}
else
_OnScroll(hwnd, hwndCtl, zDelta >= 0 ? SB_PAGEUP : SB_PAGEDOWN, 0);
+ mch_enable_flush();
+ gui_may_flush();
}
#ifdef USE_SYSMENU_FONT
* HIKEY2DI() converts a hashitem key pointer to a dictitem pointer.
* HI2DI() converts a hashitem pointer to a dictitem pointer.
*/
-# define DI2HIKEY(di) ((di)->di_key)
-# define HIKEY2DI(p) ((dictitem_T *)(p - offsetof(dictitem_T, di_key)))
-# define HI2DI(hi) HIKEY2DI((hi)->hi_key)
+#define DI2HIKEY(di) ((di)->di_key)
+#define HIKEY2DI(p) ((dictitem_T *)(p - offsetof(dictitem_T, di_key)))
+#define HI2DI(hi) HIKEY2DI((hi)->hi_key)
+
+/*
+ * Flush control functions.
+ */
+#ifdef FEAT_GUI
+# define mch_enable_flush() gui_enable_flush()
+# define mch_disable_flush() gui_disable_flush()
+#else
+# define mch_enable_flush()
+# define mch_disable_flush()
+#endif
if (VIsual_active)
update_curbuf(INVERTED);/* update inverted part */
else if (must_redraw)
+ {
+ mch_disable_flush(); /* Stop issuing gui_mch_flush(). */
update_screen(0);
+ mch_enable_flush();
+ }
else if (redraw_cmdline || clear_cmdline)
showmode();
redraw_statuslines();
|| conceal_cursor_line(curwin)
|| need_cursor_line_redraw))
{
+ mch_disable_flush(); /* Stop issuing gui_mch_flush(). */
if (conceal_old_cursor_line != conceal_new_cursor_line
&& conceal_old_cursor_line
<= curbuf->b_ml.ml_line_count)
update_single_line(curwin, conceal_old_cursor_line);
update_single_line(curwin, conceal_new_cursor_line);
+ mch_enable_flush();
curwin->w_valid &= ~VALID_CROW;
}
# endif
/* A client can tell us to redraw, but not to display the cursor, so do
* that here. */
setcursor();
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- gui_update_cursor(FALSE, FALSE);
-#endif
+ out_flush_cursor(FALSE, FALSE);
return res;
}
gui_undraw_cursor();
#endif
/* scrolling up always works */
+ mch_disable_flush();
screen_del_lines(0, 0, 1, (int)Rows, TRUE, 0, NULL);
+ mch_enable_flush();
if (!can_clear((char_u *)" "))
{
update_screen(CLEAR);
setcursor();
cursor_on();
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
- }
-#endif
+ out_flush_cursor(TRUE, FALSE);
}
#define NB_DEF_HOST "localhost"
update_screen(VALID);
setcursor();
cursor_on();
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
- }
-#endif
+ out_flush_cursor(TRUE, FALSE);
+
/* Quit a hit-return or more prompt. */
if (State == HITRETURN || State == ASKMORE)
{
update_screen(NOT_VALID);
setcursor();
cursor_on();
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
- }
-#endif
+ out_flush_cursor(TRUE, FALSE);
+
/* Quit a hit-return or more prompt. */
if (State == HITRETURN || State == ASKMORE)
{
/* ALT_INPUT_LOCK_OFF; */
setcursor(); /* restore the cursor position */
- out_flush(); /* make sure output has been written */
-
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
- }
-#endif
+ out_flush_cursor(TRUE, FALSE);
}
update_screen(CLEAR);
setcursor();
cursor_on();
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
- }
-#endif
+ out_flush_cursor(TRUE, FALSE);
}
/*
void gui_write(char_u *s, int len);
void gui_dont_update_cursor(int undraw);
void gui_can_update_cursor(void);
+void gui_disable_flush(void);
+void gui_enable_flush(void);
+void gui_may_flush(void);
int gui_outstr_nowrap(char_u *s, int len, int flags, guicolor_T fg, guicolor_T bg, int back);
void gui_undraw_cursor(void);
void gui_redraw(int x, int y, int w, int h);
char_u *tltoa(unsigned long i);
void termcapinit(char_u *name);
void out_flush(void);
+void out_flush_cursor(int force, int clear_selection);
void out_flush_check(void);
void out_trash(void);
void out_char(unsigned c);
setcursor();
}
cursor_on();
- out_flush();
#ifdef FEAT_GUI
- if (gui.in_use)
- {
+ if (gui.in_use && !gui_mch_is_blink_off())
/* Don't update the cursor when it is blinking and off to avoid
* flicker. */
- if (!gui_mch_is_blink_off())
- gui_update_cursor(FALSE, FALSE);
- gui_mch_flush();
- }
+ out_flush_cursor(FALSE, FALSE);
+ else
+#else
+ out_flush();
#endif
--redrawing_for_callback;
* done. */
if (gui.in_use)
{
- out_flush(); /* required before updating the cursor */
if (did_undraw && !gui_mch_is_blink_off())
{
+ mch_disable_flush();
+ out_flush(); /* required before updating the cursor */
+ mch_enable_flush();
+
/* Put the GUI position where the cursor was, gui_update_cursor()
* uses that. */
gui.col = gui_cursor_col;
gui.col = mb_fix_col(gui.col, gui.row);
# endif
gui_update_cursor(FALSE, FALSE);
+ gui_may_flush();
screen_cur_col = gui.col;
screen_cur_row = gui.row;
}
+ else
+ out_flush();
gui_update_scrollbars(FALSE);
}
#endif
* done. */
if (gui.in_use)
{
- out_flush(); /* required before updating the cursor */
- gui_update_cursor(FALSE, FALSE);
+ out_flush_cursor(FALSE, FALSE);
gui_update_scrollbars(FALSE);
}
# endif
showruler(FALSE);
setcursor();
cursor_on(); /* make sure that the cursor is shown */
- out_flush();
-#ifdef FEAT_GUI
- if (gui.in_use)
- {
- gui_update_cursor(TRUE, FALSE);
- gui_mch_flush();
- }
-#endif
+ out_flush_cursor(TRUE, FALSE);
+
/* Restore dollar_vcol(), because setcursor() may call curs_rows()
* which resets it if the matching position is in a previous line
* and has a higher column number. */
}
}
+/*
+ * out_flush_cursor(): flush the output buffer and redraw the cursor
+ */
+ void
+out_flush_cursor(
+ int force UNUSED, /* when TRUE, update cursor even when not moved */
+ int clear_selection UNUSED) /* clear selection under cursor */
+{
+ mch_disable_flush();
+ out_flush();
+ mch_enable_flush();
+#ifdef FEAT_GUI
+ if (gui.in_use)
+ {
+ gui_update_cursor(force, clear_selection);
+ gui_may_flush();
+ }
+#endif
+}
+
+
#if defined(FEAT_MBYTE) || defined(PROTO)
/*
* Sometimes a byte out of a multi-byte character is written with out_char().
if (interrupted != NULL && *interrupted)
/* Nothing available, but need to return so that side effects get
* handled, such as handling a message on a channel. */
- return FALSE;
+ return FAIL;
if (wtime > 0)
remaining -= due_time;
}
update_curbuf(INVERTED_ALL);
setcursor();
cursor_on();
- out_flush();
-# ifdef FEAT_GUI
- if (gui.in_use)
- gui_update_cursor(TRUE, FALSE);
-# endif
+ out_flush_cursor(TRUE, FALSE);
}
}
#endif
setcursor();
}
cursor_on(); /* redrawing may have switched it off */
- out_flush();
+ out_flush_cursor(FALSE, TRUE);
# ifdef FEAT_GUI
if (gui.in_use)
- {
- gui_update_cursor(FALSE, TRUE);
gui_update_scrollbars(FALSE);
- }
# endif
}
#ifdef FEAT_TITLE
static int included_patches[] =
{ /* Add new patch number below this line */
+/**/
+ 1449,
/**/
1448,
/**/