Skip to content

Commit

Permalink
Implementation of GC to enable multi zoom level support for win32 whi…
Browse files Browse the repository at this point in the history
…ch can be

extended by other resources and widgets.

Contributes to
#62 and #131
  • Loading branch information
amartya4256 committed May 6, 2024
1 parent b58f379 commit 3a3458a
Show file tree
Hide file tree
Showing 10 changed files with 266 additions and 100 deletions.

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@


import org.eclipse.swt.*;
import org.eclipse.swt.internal.*;
import org.eclipse.swt.internal.win32.*;

/**
Expand Down Expand Up @@ -47,6 +48,8 @@ public final class GCData {
public float[] lineDashes;
public float lineMiterLimit = 10;
public int alpha = 0xFF;
public int deviceZoom;
public int nativeDeviceZoom;

public Image image;
public PAINTSTRUCT ps;
Expand All @@ -57,4 +60,9 @@ public final class GCData {
public float gdipXOffset, gdipYOffset;
public int uiState = 0;
public boolean focusDrawn;

public void setNativeDeviceZoom(int nativeDeviceZoom) {
this.nativeDeviceZoom = nativeDeviceZoom;
this.deviceZoom = DPIUtil.getZoomForAutoscaleProperty(nativeDeviceZoom);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -139,7 +139,7 @@ public final class Image extends Resource implements Drawable {
/**
* Attribute to cache current device zoom level
*/
private int currentDeviceZoom = 100;
int currentDeviceZoom = 100;

/**
* width of the image
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1519,6 +1519,7 @@ LRESULT WM_PAINT (long wParam, long lParam) {
long hBufferedPaint = OS.BeginBufferedPaint (hDC, prcTarget, flags, null, phdc);
GCData data = new GCData ();
data.device = display;
data.setNativeDeviceZoom(getShell().getNativeZoom());
data.foreground = getForegroundPixel ();
Control control = findBackgroundControl ();
if (control == null) control = this;
Expand Down Expand Up @@ -1547,6 +1548,7 @@ LRESULT WM_PAINT (long wParam, long lParam) {

/* Create the paint GC */
GCData data = new GCData ();
data.setNativeDeviceZoom(getShell().getNativeZoom());
data.ps = ps;
data.hwnd = handle;
GC gc = GC.win32_new (this, data);
Expand Down Expand Up @@ -1581,6 +1583,7 @@ LRESULT WM_PAINT (long wParam, long lParam) {
paintGC = gc;
gc = new GC (image, paintGC.getStyle() & SWT.RIGHT_TO_LEFT);
GCData gcData = gc.getGCData ();
gcData.setNativeDeviceZoom(getShell().getNativeZoom());
gcData.uiState = data.uiState;
gc.setForeground (getForeground ());
gc.setBackground (getBackground ());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3657,6 +3657,14 @@ public void setRedraw (boolean redraw) {
}
}

@Override
void sendEvent(int eventType, Event event, boolean send) {
if(event != null && event.gc != null && event.gc.getGCData() != null) {
event.gc.getGCData().setNativeDeviceZoom(getShell().getNativeZoom());
}
super.sendEvent(eventType, event, send);
}

/**
* Sets the shape of the control to the region specified
* by the argument. When the argument is null, the
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -563,6 +563,7 @@ void wmDrawChildImage(DRAWITEMSTRUCT struct) {
}

GCData data = new GCData();
data.setNativeDeviceZoom(getShell().getNativeZoom());
data.device = display;
GC gc = GC.win32_new (struct.hDC, data);
Image image = getEnabled () ? this.image : new Image (display, this.image, SWT.IMAGE_DISABLE);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -125,7 +125,20 @@ public class Shell extends Decorations {
ToolTip [] toolTips;
long hwndMDIClient, lpstrTip, toolTipHandle, balloonTipHandle, menuItemToolTipHandle;
int minWidth = SWT.DEFAULT, minHeight = SWT.DEFAULT, maxWidth = SWT.DEFAULT, maxHeight = SWT.DEFAULT;
private int nativeZoom;
/**
* the native zoom of the monitor the shell is drawn on
* (Warning: This field is platform dependent)
* <p>
* <b>IMPORTANT:</b> This field is <em>not</em> part of the SWT
* public API. It is marked public only so that it can be shared
* within the packages provided by SWT. It is not available on all
* platforms and should never be accessed from application code.
* </p>
*
* @noreference This field is not intended to be referenced by clients.
* @since 3.126
*/
public int nativeZoom;
long [] brushes;
boolean showWithParent, fullScreen, wasMaximized, modified, center;
String toolTitle, balloonTitle;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -831,6 +831,7 @@ LRESULT CDDS_ITEMPOSTPAINT (NMTVCUSTOMDRAW nmcd, long wParam, long lParam) {
int nSavedDC = OS.SaveDC (hDC);
GCData data = new GCData ();
data.device = display;
data.setNativeDeviceZoom(getShell().getNativeZoom());
data.font = item.getFont (index);
data.foreground = OS.GetTextColor (hDC);
data.background = OS.GetBkColor (hDC);
Expand Down Expand Up @@ -1050,6 +1051,7 @@ LRESULT CDDS_ITEMPREPAINT (NMTVCUSTOMDRAW nmcd, long wParam, long lParam) {
}
int nSavedDC = OS.SaveDC (hDC);
GCData data = new GCData ();
data.setNativeDeviceZoom(getShell().getNativeZoom());
data.device = display;
if (selected && explorerTheme) {
data.foreground = OS.GetSysColor (OS.COLOR_WINDOWTEXT);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,20 @@
*/
public abstract class Widget {

private int zoom;
/**
* the zoom level of the widget
* (Warning: This field is platform dependent)
* <p>
* <b>IMPORTANT:</b> This field is <em>not</em> part of the SWT
* public API. It is marked public only so that it can be shared
* within the packages provided by SWT. It is not available on all
* platforms and should never be accessed from application code.
* </p>
*
* @noreference This field is not intended to be referenced by clients.
* @since 3.126
*/
public int zoom;
int style, state;
Display display;
EventTable eventTable;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
/*******************************************************************************
* Copyright (c) 2024 Yatta Solutions
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* Yatta Solutions - initial API and implementation
*******************************************************************************/
package org.eclipse.swt.graphics;

import static org.junit.Assert.assertEquals;

import org.eclipse.swt.SWT;
import org.eclipse.swt.internal.DPIUtil;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Shell;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

public class GCWin32Tests {
private Display display;
private int initialZoom;

@Before
public void setUp() {
initialZoom = DPIUtil.getDeviceZoom();
display = Display.getDefault();
DPIUtil.setDeviceZoom(100);
}

@After
public void tearDown() {
DPIUtil.setDeviceZoom(initialZoom);
}

@Test
public void gcZoomLevelMustChangeOnShellZoomChange() {
int zoom = DPIUtil.getDeviceZoom();
Shell shell = new Shell(display);
shell.addListener(SWT.Paint, event -> {
assertEquals("GCData must have a zoom level equal to the actual zoom level of the widget/shell", shell.zoom, event.gc.getGCData().deviceZoom);
});
shell.addListener(SWT.ZoomChanged, event -> {
assertEquals("GCData must have a zoom level equal to the actual zoom level of the widget/shell on zoomChanged event", shell.zoom, event.gc.getGCData().deviceZoom);
});
shell.open();
int newSWTZoom = zoom * 2;
Event swtEvent = new Event();
swtEvent.type = SWT.ZoomChanged;
swtEvent.gc = GC.win32_new(shell, new GCData());
swtEvent.widget = shell;
DPIUtil.setDeviceZoom(newSWTZoom);
shell.zoom = newSWTZoom;
shell.nativeZoom = DPIUtil.getZoomForAutoscaleProperty(newSWTZoom);
shell.notifyListeners(SWT.ZoomChanged, swtEvent);
}

@Test
public void drawnElementsShouldScaleUpToTheRightZoomLevel() {
int zoom = DPIUtil.getDeviceZoom();
int scalingFactor = 2;
Shell shell = new Shell(display);
GC gc = GC.win32_new(shell, new GCData());
gc.getGCData().deviceZoom = zoom * scalingFactor;
gc.getGCData().lineWidth = 10;
assertEquals("DPIUtil calls with getDeviceZoom should scale to the right value", gc.getGCData().lineWidth, gc.getLineWidth() * scalingFactor, 0);
}

}

0 comments on commit 3a3458a

Please sign in to comment.