From 7300cb4bafbd96a20b7eaaf72b8931e05a3f8972 Mon Sep 17 00:00:00 2001 From: Amartya Parijat Date: Mon, 11 Nov 2024 15:18:13 +0100 Subject: [PATCH] Reduce flickering in embedded controls Ignore calls to Control::setRedraw(false) if the control contains any embedded application as a child. This change does not completely get rid of the flickering but it reduces it drastically. Contributes to https://github.com/eclipse-platform/eclipse.platform.swt/issues/1122 Co-authored-by: Federico Jeanne Co-authored-by: Heiko Klare --- .../org/eclipse/swt/widgets/Control.java | 37 +++++++++++++++++++ 1 file changed, 37 insertions(+) diff --git a/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Control.java b/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Control.java index 9f2d36794a..c48d575b12 100644 --- a/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Control.java +++ b/bundles/org.eclipse.swt/Eclipse SWT/win32/org/eclipse/swt/widgets/Control.java @@ -16,15 +16,18 @@ import java.util.*; +import java.util.stream.*; import org.eclipse.swt.*; import org.eclipse.swt.accessibility.*; +import org.eclipse.swt.browser.*; import org.eclipse.swt.events.*; import org.eclipse.swt.graphics.*; import org.eclipse.swt.internal.*; import org.eclipse.swt.internal.gdip.*; import org.eclipse.swt.internal.ole.win32.*; import org.eclipse.swt.internal.win32.*; +import org.eclipse.swt.ole.win32.*; /** * Control is the abstract superclass of all windowed user interface classes. @@ -3631,6 +3634,21 @@ boolean setRadioSelection (boolean value) { */ public void setRedraw (boolean redraw) { checkWidget (); + + /* + * Some embedded applications like webview2 (Edge), Excel, etc have a renderer + * that will draw the whole application white when setting redraw to false and + * will repaint it completely when setting it to true. This causes flickering, + * which is why turning redraw off is not allowed in case this control has an + * embedded application as a child. + * + * https://github.com/eclipse-platform/eclipse.platform.swt/issues/1122 + */ + boolean isShown = isVisible() && !isDisposed(); + if (!redraw && isShown && embedsWin32Control()) { + return; + } + /* * Feature in Windows. When WM_SETREDRAW is used to turn * off drawing in a widget, it clears the WS_VISIBLE bits @@ -3667,6 +3685,25 @@ public void setRedraw (boolean redraw) { } } +private boolean embedsWin32Control () { + if (this instanceof Browser) { + // The Edge browser embeds webView2 + return (getStyle() & SWT.EDGE) != 0; + } + + if (this instanceof OleClientSite) { + // OLE objects are always embedded by windows + return true; + } + + // This needs to be checked AFTER OleClientSite because OleClientSite itself is a Composite + if (this instanceof Composite comp) { + return Stream.of(comp.getChildren()).anyMatch(Control::embedsWin32Control); + } + + return false; +} + /** * Sets the shape of the control to the region specified * by the argument. When the argument is null, the