Skip to content

Commit bd6026c

Browse files
honkar-jdkprsadhuk
authored andcommittedMay 10, 2022
7124282: [macosx] Can't see table cell highlighter when the highlight border is the same color as the cell.
Reviewed-by: psadhukhan, prr
1 parent 9a3cb93 commit bd6026c

File tree

6 files changed

+263
-14
lines changed

6 files changed

+263
-14
lines changed
 

‎src/java.desktop/macosx/classes/com/apple/laf/AquaFocusHandler.java

+35-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2011, 2012, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2011, 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
@@ -25,11 +25,21 @@
2525

2626
package com.apple.laf;
2727

28-
import java.awt.*;
29-
import java.awt.event.*;
30-
import java.beans.*;
31-
32-
import javax.swing.*;
28+
import java.awt.Color;
29+
import java.awt.Window;
30+
import java.awt.event.FocusEvent;
31+
import java.awt.event.FocusListener;
32+
import java.beans.PropertyChangeEvent;
33+
import java.beans.PropertyChangeListener;
34+
35+
import javax.swing.JComponent;
36+
import javax.swing.JTree;
37+
import javax.swing.JTable;
38+
import javax.swing.JList;
39+
import javax.swing.SwingUtilities;
40+
import javax.swing.UIManager;
41+
42+
import javax.swing.plaf.BorderUIResource;
3343
import javax.swing.plaf.UIResource;
3444

3545
/**
@@ -113,22 +123,35 @@ static void swapSelectionColors(final String prefix, final JTable c, final Objec
113123

114124
final Color bg = c.getSelectionBackground();
115125
final Color fg = c.getSelectionForeground();
126+
116127
if (!(bg instanceof UIResource) || !(fg instanceof UIResource)) return;
117128

118129
if (Boolean.FALSE.equals(value)) {
119-
setSelectionColors(c, "Table.selectionInactiveForeground", "Table.selectionInactiveBackground");
130+
setSelectionColors(c, "Table.selectionInactiveForeground",
131+
"Table.selectionInactiveBackground");
120132
return;
121133
}
122134

123135
if (Boolean.TRUE.equals(value)) {
124-
setSelectionColors(c, "Table.selectionForeground", "Table.selectionBackground");
136+
setSelectionColors(c, "Table.selectionForeground",
137+
"Table.selectionBackground");
125138
return;
126139
}
127140
}
128141

129142
static void setSelectionColors(final JTable c, final String fgName, final String bgName) {
143+
130144
c.setSelectionForeground(UIManager.getColor(fgName));
131145
c.setSelectionBackground(UIManager.getColor(bgName));
146+
147+
// focus ring changes for on-the-fly accent color changes
148+
Color prominentFocusRing = AquaLookAndFeel.deriveProminentFocusRing(
149+
UIManager.getColor("Table.cellFocusRing"));
150+
BorderUIResource.LineBorderUIResource focusCellHighlightBorder =
151+
new BorderUIResource.LineBorderUIResource(prominentFocusRing, 2);
152+
UIManager.getDefaults().put("Table.focusCellHighlightBorder",
153+
focusCellHighlightBorder);
154+
132155
}
133156

134157
static void swapSelectionColors(final String prefix, final JList<?> c, final Object value) {
@@ -139,12 +162,14 @@ static void swapSelectionColors(final String prefix, final JList<?> c, final Obj
139162
if (!(bg instanceof UIResource) || !(fg instanceof UIResource)) return;
140163

141164
if (Boolean.FALSE.equals(value)) {
142-
setSelectionColors(c, "List.selectionInactiveForeground", "List.selectionInactiveBackground");
165+
setSelectionColors(c, "List.selectionInactiveForeground",
166+
"List.selectionInactiveBackground");
143167
return;
144168
}
145169

146170
if (Boolean.TRUE.equals(value)) {
147-
setSelectionColors(c, "List.selectionForeground", "List.selectionBackground");
171+
setSelectionColors(c, "List.selectionForeground",
172+
"List.selectionBackground");
148173
return;
149174
}
150175
}

‎src/java.desktop/macosx/classes/com/apple/laf/AquaImageFactory.java

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2011, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2011, 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
@@ -506,6 +506,10 @@ public static Color getFocusRingColorUIResource() {
506506
return new SystemColorProxy(LWCToolkit.getAppleColor(LWCToolkit.KEYBOARD_FOCUS_COLOR));
507507
}
508508

509+
public static Color getCellHighlightColorUIResource() {
510+
return new SystemColorProxy(LWCToolkit.getAppleColor(LWCToolkit.CELL_HIGHLIGHT_COLOR));
511+
}
512+
509513
public static Color getSelectionInactiveBackgroundColorUIResource() {
510514
return new SystemColorProxy(LWCToolkit.getAppleColor(LWCToolkit.INACTIVE_SELECTION_BACKGROUND_COLOR));
511515
}

‎src/java.desktop/macosx/classes/com/apple/laf/AquaLookAndFeel.java

+64-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2011, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2011, 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
@@ -388,6 +388,9 @@ public Object createValue(UIDefaults defaultsTable) {
388388
final Color focusRingColor = AquaImageFactory.getFocusRingColorUIResource();
389389
final Border focusCellHighlightBorder = new BorderUIResource.LineBorderUIResource(focusRingColor);
390390

391+
// for table cell highlighter
392+
final Color cellFocusRingColor = AquaImageFactory.getCellHighlightColorUIResource();
393+
391394
final Color windowBackgroundColor = AquaImageFactory.getWindowBackgroundColorUIResource();
392395
final Color panelBackgroundColor = windowBackgroundColor;
393396
final Color tabBackgroundColor = windowBackgroundColor;
@@ -889,7 +892,9 @@ public Object createValue(UIDefaults defaultsTable) {
889892
"Table.gridColor", white, // grid line color
890893
"Table.focusCellBackground", textHighlightText,
891894
"Table.focusCellForeground", textHighlight,
892-
"Table.focusCellHighlightBorder", focusCellHighlightBorder,
895+
"Table.cellFocusRing", cellFocusRingColor,
896+
"Table.focusCellHighlightBorder", new BorderUIResource.LineBorderUIResource(
897+
deriveProminentFocusRing(cellFocusRingColor), 2),
893898
"Table.scrollPaneBorder", scollListBorder,
894899

895900
"Table.ancestorInputMap", aquaKeyBindings.getTableInputMap(),
@@ -1122,4 +1127,61 @@ protected void initClassDefaults(final UIDefaults table) {
11221127
};
11231128
table.putDefaults(uiDefaults);
11241129
}
1130+
1131+
/**
1132+
* Returns a new cell focus ring color by changing saturation
1133+
* and setting the brightness to 100% for incoming cellFocusRing.
1134+
*
1135+
* If the incoming cellFocusRingColor is equal to white/black/grayish,
1136+
* the returned cellFocusRingColor is Light Gray. For all other colors,
1137+
* new cellFocusRingColor (in the latter case), is obtained by adjusting
1138+
* the saturation levels and setting the brightness to 100% of the
1139+
* incoming cellFocusRingColor.
1140+
*
1141+
* @param cellFocusRingColor - the {@code Color} object
1142+
* @return the {@code Color} object corresponding to new HSB values
1143+
*/
1144+
static Color deriveProminentFocusRing(Color cellFocusRingColor) {
1145+
1146+
// define constants
1147+
float satLowerValue = 0.30f;
1148+
float satUpperValue = 1.0f;
1149+
1150+
// saturation threshold for grayish colors
1151+
float satGrayScale = 0.10f;
1152+
1153+
// used to compare with saturation value of original focus ring and
1154+
// set it to either lower or upper saturation value
1155+
float saturationThreshold = 0.5f;
1156+
1157+
// brightness always set to 100%
1158+
float brightnessValue = 1.0f;
1159+
1160+
float[] hsbValues = new float[3];
1161+
1162+
int redValue = cellFocusRingColor.getRed();
1163+
int greenValue = cellFocusRingColor.getGreen();
1164+
int blueValue = cellFocusRingColor.getBlue();
1165+
1166+
Color.RGBtoHSB(redValue, greenValue, blueValue, hsbValues);
1167+
1168+
// if cellFocusRing is White/Black/Grayish
1169+
if ((hsbValues[0] == 0 && hsbValues[1] == 0)
1170+
|| hsbValues[1] <= satGrayScale) {
1171+
return Color.LIGHT_GRAY;
1172+
}
1173+
1174+
// if cellFocusRing color NOT White/Black/Grayish
1175+
// saturation adjustment - saturation set to either lower or
1176+
// upper saturation value based on current saturation level
1177+
hsbValues[1] = hsbValues[1] >= saturationThreshold ?
1178+
satLowerValue : satUpperValue;
1179+
1180+
// brightness adjustment - brightness set to 100%, always return the
1181+
// brightest color for the new color
1182+
hsbValues[2] = brightnessValue;
1183+
1184+
//create and return color corresponding to new hsbValues
1185+
return Color.getHSBColor(hsbValues[0], hsbValues[1], hsbValues[2]);
1186+
}
11251187
}

‎src/java.desktop/macosx/classes/sun/lwawt/macosx/LWCToolkit.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -209,16 +209,19 @@ public LWCToolkit() {
209209
/*
210210
* System colors with default initial values, overwritten by toolkit if system values differ and are available.
211211
*/
212-
private static final int NUM_APPLE_COLORS = 4;
212+
private static final int NUM_APPLE_COLORS = 5;
213213
public static final int KEYBOARD_FOCUS_COLOR = 0;
214214
public static final int INACTIVE_SELECTION_BACKGROUND_COLOR = 1;
215215
public static final int INACTIVE_SELECTION_FOREGROUND_COLOR = 2;
216216
public static final int SELECTED_CONTROL_TEXT_COLOR = 3;
217+
public static final int CELL_HIGHLIGHT_COLOR = 4;
218+
217219
private static int[] appleColors = {
218220
0xFF808080, // keyboardFocusColor = Color.gray;
219221
0xFFC0C0C0, // secondarySelectedControlColor
220222
0xFF303030, // controlDarkShadowColor
221223
0xFFFFFFFF, // controlTextColor
224+
0xFF808080, // cellHighlightColor = Color.gray;
222225
};
223226

224227
private native void loadNativeColors(final int[] systemColors, final int[] appleColors);

‎src/java.desktop/macosx/native/libawt_lwawt/awt/CSystemColors.m

+6
Original file line numberDiff line numberDiff line change
@@ -120,6 +120,12 @@ + (void)reloadColors {
120120
}
121121
}
122122

123+
// added for JTable Focus Ring
124+
if (@available(macOS 10.14, *)) {
125+
appleColors[sun_lwawt_macosx_LWCToolkit_CELL_HIGHLIGHT_COLOR] = [NSColor controlAccentColor];
126+
} else {
127+
appleColors[sun_lwawt_macosx_LWCToolkit_CELL_HIGHLIGHT_COLOR] = [NSColor keyboardFocusIndicatorColor];
128+
}
123129
appleColors[sun_lwawt_macosx_LWCToolkit_KEYBOARD_FOCUS_COLOR] = [NSColor keyboardFocusIndicatorColor];
124130
appleColors[sun_lwawt_macosx_LWCToolkit_INACTIVE_SELECTION_BACKGROUND_COLOR] = [NSColor secondarySelectedControlColor];
125131
appleColors[sun_lwawt_macosx_LWCToolkit_INACTIVE_SELECTION_FOREGROUND_COLOR] = [NSColor controlDarkShadowColor];
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,149 @@
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+
/*
26+
* @test
27+
* @bug 7124282
28+
* @key headful
29+
* @requires (os.family == "mac")
30+
* @summary Checks whether the JTable's focus ring color's RGB color
31+
* diff with selectionBackground is greater in comparison to original
32+
* focus ring (represented by 'Table.cellFocusRing' property in Aqua LAF
33+
* UIDefaults).
34+
* @run main JTableFocusRingTest
35+
*/
36+
37+
import java.awt.Color;
38+
import java.util.Arrays;
39+
import javax.swing.plaf.BorderUIResource.LineBorderUIResource;
40+
import javax.swing.SwingUtilities;
41+
import javax.swing.UIManager;
42+
import javax.swing.UnsupportedLookAndFeelException;
43+
44+
public class JTableFocusRingTest {
45+
46+
public static void main(String[] args) throws Exception{
47+
48+
try {
49+
UIManager.setLookAndFeel("com.apple.laf.AquaLookAndFeel");
50+
} catch (ClassNotFoundException | InstantiationException |
51+
IllegalAccessException | UnsupportedLookAndFeelException e) {
52+
throw new RuntimeException("Unsupported Look&Feel Class");
53+
}
54+
SwingUtilities.invokeAndWait(() -> {
55+
56+
float[] bckRGB = new float[3];
57+
float[] oldCellRingRGB = new float[3];
58+
float[] newCellRingRGB = new float[3];
59+
60+
Color selectionBck = null;
61+
Color originalRingColor = null;
62+
Color newRingColor = null;
63+
64+
// saturation threshold for grayish colors
65+
float satGrayScale = 0.10f;
66+
67+
if (UIManager.getDefaults().get("Table.selectionBackground") != null
68+
&& UIManager.getDefaults().get("Table.selectionBackground")
69+
instanceof Color) {
70+
selectionBck = (Color) UIManager.getDefaults()
71+
.get("Table.selectionBackground");
72+
}
73+
if (UIManager.getDefaults().get("Table.cellFocusRing") != null
74+
&& UIManager.getDefaults().get("Table.cellFocusRing")
75+
instanceof Color) {
76+
originalRingColor = (Color) UIManager.getDefaults().get("Table.cellFocusRing");
77+
}
78+
79+
if (UIManager.getDefaults()
80+
.get("Table.focusCellHighlightBorder") != null &&
81+
UIManager.getDefaults().get("Table.focusCellHighlightBorder")
82+
instanceof LineBorderUIResource) {
83+
LineBorderUIResource cellFocusBorderObj = (LineBorderUIResource)
84+
UIManager.getDefaults().get("Table.focusCellHighlightBorder");
85+
newRingColor = cellFocusBorderObj.getLineColor();
86+
}
87+
88+
if (selectionBck == null || originalRingColor == null ||
89+
newRingColor == null) {
90+
throw new RuntimeException("One or more color values are null");
91+
}
92+
System.out.println(UIManager.getLookAndFeel().toString());
93+
System.out.println("Selection Background Color: "
94+
+ selectionBck.toString());
95+
96+
System.out.println("Original FocusRing Color: "
97+
+ originalRingColor.toString());
98+
99+
System.out.println("Brighter FocusRing Color: "
100+
+ newRingColor.toString());
101+
102+
int redValue = originalRingColor.getRed();
103+
int greenValue = originalRingColor.getGreen();
104+
int blueValue = originalRingColor.getBlue();
105+
106+
float[] hsbValues = new float[3];
107+
Color.RGBtoHSB(redValue, greenValue, blueValue, hsbValues);
108+
109+
System.out.println("Original Focus Ring Hue, Saturation and" +
110+
" Brightness: "+ Arrays.toString(hsbValues));
111+
112+
// Edge case - Original Focus ring color: WHITE/BLACK/GRAY
113+
if (((hsbValues[0] == 0 && hsbValues[1] == 0)
114+
|| hsbValues[1] <= satGrayScale) &&
115+
newRingColor.equals(Color.LIGHT_GRAY)) {
116+
System.out.println("Original Focus ring color:" +
117+
"WHITE/BLACK/GRAYISH, Cell Focus Ring Color: LIGHT GRAY");
118+
System.out.println("Test case passed");
119+
return;
120+
}
121+
selectionBck.getRGBColorComponents(bckRGB);
122+
originalRingColor.getRGBColorComponents(oldCellRingRGB);
123+
newRingColor.getRGBColorComponents(newCellRingRGB);
124+
125+
float originalRGBDiff = calculateRGBDiff(oldCellRingRGB, bckRGB);
126+
float brighterRGBDiff = calculateRGBDiff(newCellRingRGB, bckRGB);
127+
128+
System.out.println("Original RGB Diff: "+ originalRGBDiff);
129+
System.out.println("Brighter RGB Diff: "+ brighterRGBDiff);
130+
131+
if (brighterRGBDiff <= originalRGBDiff) {
132+
throw new RuntimeException("Cell Focus Ring Not Visible");
133+
}
134+
});
135+
}
136+
137+
/* calculates the difference between individual RGB components of 2 colors
138+
and returns the total difference. A higher RGB difference is preferred
139+
for a prominent cell highlighter */
140+
141+
private static float calculateRGBDiff(float[] focusRingRGB, float[] bckRGB) {
142+
143+
float totalRGBDiff = 0;
144+
for (int i=0; i< focusRingRGB.length; i++) {
145+
totalRGBDiff += Math.abs(focusRingRGB[i] - bckRGB[i]);
146+
}
147+
return totalRGBDiff;
148+
}
149+
}

0 commit comments

Comments
 (0)
Please sign in to comment.