Skip to content

Commit 8323787

Browse files
honkar-jdkaivanov-jdk
andcommittedMay 18, 2022
8255439: System Tray icons get corrupted when windows scaling changes
Co-authored-by: Alexey Ivanov <aivanov@openjdk.org> Reviewed-by: kcr, prr, aivanov
1 parent cd5bfe7 commit 8323787

File tree

3 files changed

+166
-2
lines changed

3 files changed

+166
-2
lines changed
 

‎src/java.desktop/windows/native/libawt/windows/awt_TrayIcon.cpp

+32-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2014, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -70,10 +70,12 @@ typedef struct tagBitmapheader {
7070

7171
jfieldID AwtTrayIcon::idID;
7272
jfieldID AwtTrayIcon::actionCommandID;
73+
jmethodID AwtTrayIcon::updateImageID;
7374

7475
HWND AwtTrayIcon::sm_msgWindow = NULL;
7576
AwtTrayIcon::TrayIconListItem* AwtTrayIcon::sm_trayIconList = NULL;
7677
int AwtTrayIcon::sm_instCount = 0;
78+
bool AwtTrayIcon::m_bDPIChanged = false;
7779

7880
/************************************************************************
7981
* AwtTrayIcon methods
@@ -221,6 +223,18 @@ void AwtTrayIcon::InitNID(UINT uID)
221223
m_nid.uVersion = NOTIFYICON_VERSION;
222224
}
223225

226+
// Call updateImage() method on the peer when screen scale changes
227+
void AwtTrayIcon::UpdateImage()
228+
{
229+
JNIEnv *env =(JNIEnv *)JNU_GetEnv(jvm, JNI_VERSION_1_2);
230+
231+
jobject peer = GetPeer(env);
232+
if (peer != NULL) {
233+
env->CallVoidMethod(peer, updateImageID);
234+
env->ExceptionClear();
235+
}
236+
}
237+
224238
BOOL AwtTrayIcon::SendTrayMessage(DWORD dwMessage)
225239
{
226240
return Shell_NotifyIcon(dwMessage, (PNOTIFYICONDATA)&m_nid);
@@ -248,6 +262,10 @@ LRESULT CALLBACK AwtTrayIcon::TrayWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam
248262
}
249263
}
250264
break;
265+
case WM_DPICHANGED:
266+
// Set the flag to update icon images, see WmTaskbarCreated
267+
m_bDPIChanged = true;
268+
break;
251269
default:
252270
if(uMsg == s_msgTaskbarCreated) {
253271
if (hwnd == AwtTrayIcon::sm_msgWindow) {
@@ -474,12 +492,17 @@ MsgRouting AwtTrayIcon::WmContextMenu(UINT flags, int x, int y)
474492
MsgRouting AwtTrayIcon::WmTaskbarCreated() {
475493
TrayIconListItem* item;
476494
for (item = sm_trayIconList; item != NULL; item = item->m_next) {
495+
if (m_bDPIChanged) {
496+
// Update the icon image
497+
item->m_trayIcon->UpdateImage();
498+
}
477499
BOOL result = item->m_trayIcon->SendTrayMessage(NIM_ADD);
478500
// 6270114: Instructs the taskbar to behave according to the Shell version 5.0
479501
if (result) {
480502
item->m_trayIcon->SendTrayMessage(NIM_SETVERSION);
481503
}
482504
}
505+
m_bDPIChanged = false;
483506
return mrDoDefault;
484507
}
485508

@@ -917,6 +940,14 @@ Java_java_awt_TrayIcon_initIDs(JNIEnv *env, jclass cls)
917940
DASSERT(AwtTrayIcon::actionCommandID != NULL);
918941
CHECK_NULL( AwtTrayIcon::actionCommandID);
919942

943+
jclass wPeerCls = env->FindClass("sun/awt/windows/WTrayIconPeer");
944+
DASSERT(wPeerCls != NULL);
945+
CHECK_NULL(wPeerCls);
946+
947+
AwtTrayIcon::updateImageID = env->GetMethodID(wPeerCls, "updateImage", "()V");
948+
DASSERT(AwtTrayIcon::updateImageID != NULL);
949+
CHECK_NULL(AwtTrayIcon::updateImageID);
950+
920951
CATCH_BAD_ALLOC;
921952
}
922953

‎src/java.desktop/windows/native/libawt/windows/awt_TrayIcon.h

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2008, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -86,6 +86,8 @@ class AwtTrayIcon: public AwtObject {
8686

8787
void DisplayMessage(LPCTSTR caption, LPCTSTR text, LPCTSTR msgType);
8888

89+
void UpdateImage();
90+
8991
// Adds to the head of the list
9092
INLINE void AddTrayIconItem(UINT id) {
9193
TrayIconListItem* item = new TrayIconListItem(id, this);
@@ -121,6 +123,7 @@ class AwtTrayIcon: public AwtObject {
121123
*/
122124
static jfieldID idID;
123125
static jfieldID actionCommandID;
126+
static jmethodID updateImageID;
124127

125128
// ************************
126129

@@ -151,6 +154,8 @@ class AwtTrayIcon: public AwtObject {
151154
TrayIconListItem* m_next;
152155
};
153156

157+
static bool m_bDPIChanged;
158+
154159
public:
155160
static TrayIconListItem* sm_trayIconList;
156161
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* @bug 8255439
27+
* @key headful
28+
* @library /java/awt/regtesthelpers
29+
* @build PassFailJFrame
30+
* @summary To test tray icon scaling with on-the-fly DPI/Scale changes on Windows
31+
* @run main/manual TrayIconScalingTest
32+
* @requires (os.family == "windows")
33+
*/
34+
35+
import java.awt.AWTException;
36+
import java.awt.Color;
37+
import java.awt.Font;
38+
import java.awt.Graphics2D;
39+
import java.awt.Image;
40+
import java.awt.RenderingHints;
41+
import java.awt.SystemTray;
42+
import java.awt.TrayIcon;
43+
import java.awt.font.TextLayout;
44+
import java.awt.image.BaseMultiResolutionImage;
45+
import java.awt.image.BufferedImage;
46+
import java.lang.reflect.InvocationTargetException;
47+
import java.util.ArrayList;
48+
49+
public class TrayIconScalingTest {
50+
51+
private static SystemTray tray;
52+
private static TrayIcon icon;
53+
54+
private static final String INSTRUCTIONS =
55+
"This test checks if the tray icon gets updated when DPI / Scale" +
56+
" is changed on the fly.\n\n" +
57+
"STEPS: \n\n" +
58+
"1. Check the system tray / notification area on Windows" +
59+
" taskbar, you should see a white icon which displays a" +
60+
" number.\n\n" +
61+
"2. Navigate to Settings > System > Display and change the" +
62+
" display scale by selecting any value from" +
63+
" Scale & Layout dropdown.\n\n"+
64+
"3. When the scale changes, check the white tray icon," +
65+
" there should be no distortion, it should be displayed sharp,\n" +
66+
" and the displayed number should correspond to the current"+
67+
" scale:\n" +
68+
" 100% - 16, 125% - 20, 150% - 24, 175% - 28, 200% - 32.\n\n"+
69+
" If the icon is displayed sharp and without any distortion," +
70+
" press PASS, otherwise press FAIL.\n";
71+
72+
private static final Font font = new Font("Dialog", Font.BOLD, 12);
73+
74+
public static void main(String[] args)
75+
throws InterruptedException, InvocationTargetException {
76+
// check if SystemTray supported on the machine
77+
if (!SystemTray.isSupported()) {
78+
System.out.println("SystemTray is not supported");
79+
return;
80+
}
81+
PassFailJFrame passFailJFrame = new PassFailJFrame("TrayIcon " +
82+
"Test Instructions", INSTRUCTIONS, 8, 18, 85);
83+
createAndShowGUI();
84+
try {
85+
passFailJFrame.awaitAndCheck();
86+
} finally {
87+
tray.remove(icon);
88+
}
89+
}
90+
91+
private static void createAndShowGUI() {
92+
ArrayList<Image> imageList = new ArrayList<>();
93+
for (int size = 16; size <= 48; size += 4) {
94+
imageList.add(createIcon(size));
95+
}
96+
Image mRImage =
97+
new BaseMultiResolutionImage(imageList.toArray(new Image[0]));
98+
99+
tray = SystemTray.getSystemTray();
100+
icon = new TrayIcon(mRImage);
101+
102+
try {
103+
tray.add(icon);
104+
} catch (AWTException e) {
105+
throw new RuntimeException("Error while adding icon to system tray");
106+
}
107+
}
108+
109+
private static Image createIcon(int size) {
110+
BufferedImage image = new BufferedImage(size, size,
111+
BufferedImage.TYPE_INT_ARGB);
112+
Graphics2D g = image.createGraphics();
113+
g.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,
114+
RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
115+
g.setColor(Color.WHITE);
116+
g.fillRect(0, 0, size, size);
117+
g.setFont(font);
118+
g.setColor(Color.BLACK);
119+
120+
TextLayout layout = new TextLayout(String.valueOf(size),
121+
g.getFont(), g.getFontRenderContext());
122+
int height = (int) layout.getBounds().getHeight();
123+
int width = (int) layout.getBounds().getWidth();
124+
layout.draw(g, (size - width) / 2f - 1, (size + height) / 2f);
125+
g.dispose();
126+
return image;
127+
}
128+
}

0 commit comments

Comments
 (0)
Please sign in to comment.