Skip to content

Commit

Permalink
Add static method to configure overlay scrolling for FigureCanvas
Browse files Browse the repository at this point in the history
This allows the user to specify whether new instances of FigureCanvas
should use overlay scrolling on not. This is disabled by default, due to
performance issues on Linux/GTK3.

As a result, the scrollbar is shown as a separate entity and no longer
part of the client area. This setting only affects operating systems
supporting overlay scrolling. This method requires SWT 3.126 or newer.

eclipse-gef#430
  • Loading branch information
ptziegler committed Apr 21, 2024
1 parent 7ca38b7 commit dc302b5
Show file tree
Hide file tree
Showing 2 changed files with 68 additions and 0 deletions.
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,9 @@
# GEF Classic 3.20.0 (Eclipse 2024-06)

## Draw2d
- Clients can disable overlay scrolling by calling FigureCanvas.setScrollbarsMode()
prior to creating the object.

# GEF Classic 3.19.0 (Eclipse 2024-03)

## Draw2d
Expand Down
62 changes: 62 additions & 0 deletions org.eclipse.draw2d/src/org/eclipse/draw2d/FigureCanvas.java
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,9 @@
package org.eclipse.draw2d;

import java.beans.PropertyChangeListener;
import java.lang.reflect.Method;
import java.util.logging.Level;
import java.util.logging.Logger;

import org.eclipse.swt.SWT;
import org.eclipse.swt.events.SelectionAdapter;
Expand All @@ -21,6 +24,7 @@
import org.eclipse.swt.widgets.Canvas;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Control;
import org.eclipse.swt.widgets.Scrollable;

import org.eclipse.draw2d.geometry.Dimension;
import org.eclipse.draw2d.geometry.Point;
Expand Down Expand Up @@ -49,6 +53,7 @@
*/
public class FigureCanvas extends Canvas {

private static final Logger LOGGER = Logger.getLogger(FigureCanvas.class.getSimpleName());
private static final int ACCEPTED_STYLES = SWT.RIGHT_TO_LEFT | SWT.LEFT_TO_RIGHT | SWT.V_SCROLL | SWT.H_SCROLL
| SWT.NO_BACKGROUND | SWT.NO_REDRAW_RESIZE | SWT.DOUBLE_BUFFERED | SWT.BORDER;

Expand All @@ -73,6 +78,16 @@ public class FigureCanvas extends Canvas {
/** Always show scrollbar */
public static int ALWAYS = 2;

/**
* Static property specifying whether new instances of this class should be
* created with overlay scrolling enabled. Possible values are {@link SWT#NONE}
* or {@link SWT#SCROLLBAR_OVERLAY}. Initial value is {@code null}, indicating
* that the default setting should be used.
*
* @see <a href="https://github.com/eclipse/gef-classic/issues/430">here</a>
*/
private static Integer SCROLLBARS_MODE = null;

private int vBarVisibility = AUTOMATIC;
private int hBarVisibility = AUTOMATIC;
private Viewport viewport;
Expand Down Expand Up @@ -155,6 +170,7 @@ public FigureCanvas(Composite parent, LightweightSystem lws) {
*/
public FigureCanvas(int style, Composite parent, LightweightSystem lws) {
super(parent, checkStyle(style));
adjustScrollbarsMode();
getHorizontalBar().setVisible(false);
getVerticalBar().setVisible(false);
this.lws = lws;
Expand Down Expand Up @@ -255,6 +271,25 @@ public Viewport getViewport() {
return viewport;
}

/**
* Calls {@link Scrollable#setScrollbarsMode(int)} if executed with SWT 3.126.
*
* @since 3.2
*/
private void adjustScrollbarsMode() {
// TODO Check version of SWT binary after PR has been merged:
// https://github.com/eclipse-platform/eclipse.platform.swt/pull/1188
if (SWT.getVersion() >= 4965 && SCROLLBARS_MODE != null && SCROLLBARS_MODE != getScrollbarsMode()) {
try {
// Invoke method via reflection to remain compatible with older SWT versions
Method setScrollbarsMode = Scrollable.class.getMethod("setScrollbarsMode", Integer.class); //$NON-NLS-1$
setScrollbarsMode.invoke(this, SCROLLBARS_MODE);
} catch (ReflectiveOperationException e) {
LOGGER.log(Level.SEVERE, e.getMessage(), e);
}
}
}

/**
* Adds listeners for scrolling.
*/
Expand Down Expand Up @@ -552,6 +587,33 @@ public void setViewport(Viewport vp) {
hookViewport();
}

/**
* Global property to enable overlay scrolling for all figure canvas when
* {@link SWT#SCROLLBAR_OVERLAY} and to disable overlay scrolling when
* {@link SWT#NONE} is given. When {@code null}, the default setting of the
* operating system is used.<br>
* Calling this method has no effect on already created instances of this
* class.<br>
* Calling this method has no effect if overlay scrolling is not supported by
* the operating system.
*
* @param mode One of {@link SWT#NONE}, {@link SWT#SCROLLBAR_OVERLAY} or
* {@code null}.
* @since 3.16
* @throws IllegalArgumentException if the given {@code mode} isn't one of
* {@link SWT#NONE},
* {@link SWT#SCROLLBAR_OVERLAY} or
* {@code null}.
* @see #SCROLLBARS_MODE
*/
public static void setScrollbarsMode(Integer mode) throws IllegalArgumentException {
if (mode != null && mode != SWT.NONE && mode != SWT.SCROLLBAR_OVERLAY) {
throw new IllegalArgumentException(
"Scrollbars mode must be either null, SWT.SCROLLBAR_OVERLAY or SWT.NONE"); //$NON-NLS-1$
}
SCROLLBARS_MODE = mode;
}

private static int verifyScrollBarOffset(RangeModel model, int value) {
value = Math.max(model.getMinimum(), value);
return Math.min(model.getMaximum() - model.getExtent(), value);
Expand Down

0 comments on commit dc302b5

Please sign in to comment.