Skip to content

Commit

Permalink
8281988: Create a regression test for JDK-4618767
Browse files Browse the repository at this point in the history
Backport-of: 43dc9ef619b823e033cb1e298bbc091eb5a50967
  • Loading branch information
GoeLin committed Jul 21, 2022
1 parent ac3893c commit a240a9e
Showing 1 changed file with 222 additions and 0 deletions.
222 changes: 222 additions & 0 deletions test/jdk/javax/swing/JList/4618767/JListSelectedElementTest.java
@@ -0,0 +1,222 @@
/*
* Copyright (c) 2002, 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/

import java.awt.Robot;
import java.awt.event.FocusAdapter;
import java.awt.event.FocusEvent;
import java.awt.event.KeyEvent;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference;
import java.util.stream.Collectors;
import javax.swing.JFrame;
import javax.swing.JList;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.UIManager.LookAndFeelInfo;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.event.MenuEvent;
import javax.swing.event.MenuListener;


import static javax.swing.UIManager.getInstalledLookAndFeels;

/*
* @test
* @key headful
* @bug 4618767
* @summary This test confirms that typing a letter while a JList has focus now makes the selection
* not jump to the item whose text starts with that letter if that typed letter is accompanied
* by modifier keys such as ALT or CTRL(eg: ALT+F).
* @run main JListSelectedElementTest
*/
public class JListSelectedElementTest {

private static final int FILE_MENU = KeyEvent.VK_F;
private static JFrame frame;
private static JList<String> list;
private static Robot robot;
private static CountDownLatch listGainedFocusLatch;
private static CountDownLatch menuSelectedEventLatch;

public static void main(String[] args) throws Exception {
runTest();
}

public static void runTest() throws Exception {
robot = new Robot();
robot.setAutoWaitForIdle(true);
robot.setAutoDelay(200);

final boolean isMac = System.getProperty("os.name")
.toLowerCase()
.contains("os x");

List<String> lafs = Arrays.stream(getInstalledLookAndFeels())
.map(LookAndFeelInfo::getClassName)
.collect(Collectors.toList());
for (final String laf : lafs) {
listGainedFocusLatch = new CountDownLatch(1);
menuSelectedEventLatch = new CountDownLatch(1);
try {
AtomicBoolean lafSetSuccess = new AtomicBoolean(false);
SwingUtilities.invokeAndWait(() -> {
lafSetSuccess.set(setLookAndFeel(laf));
if (lafSetSuccess.get()) {
createUI();
}
});
if (!lafSetSuccess.get()) {
continue;
}
robot.waitForIdle();

// Wait until the list gains focus.
if (!listGainedFocusLatch.await(3, TimeUnit.SECONDS)) {
throw new RuntimeException("Waited too long, but can't gain focus for list");
}

// Select element named as 'bill'
hitKeys(KeyEvent.VK_B);

// Assertion check to verify that the selected node is 'bill'
AtomicReference<String> elementSel = new AtomicReference<>();
SwingUtilities.invokeAndWait(() -> elementSel.set(list.getSelectedValue()));
final String elementSelBefore = elementSel.get();
if (!"bill".equals(elementSelBefore)) {
throw new RuntimeException("Test failed for " + laf
+ " as the list element selected: " + elementSel
+ " is not the expected one 'bill'"
);
}

// Now operate Menu using Mnemonics, different key combinations for different OSes.
// For most OSes it's ALT+F; on macOS it's ALT+CNTRL+F except for Nimbus LaF.
if (isMac && !laf.contains("Nimbus")) {
hitKeys(KeyEvent.VK_ALT, KeyEvent.VK_CONTROL, FILE_MENU);
} else {
hitKeys(KeyEvent.VK_ALT, FILE_MENU);
}

// Wait until the menu got selected.
if (!menuSelectedEventLatch.await(3, TimeUnit.SECONDS)) {
throw new RuntimeException("Waited too long, but can't select menu using mnemonics for " + laf);
}

hitKeys(KeyEvent.VK_ENTER);

AtomicReference<String> elementSelAfter = new AtomicReference<>();
SwingUtilities.invokeAndWait(() -> elementSelAfter.set(list.getSelectedValue()));

// As per the fix of BugID 4618767, the list element selection should not change
if (!elementSelBefore.equals(elementSelAfter.get())) {
throw new RuntimeException("Test failed for " + laf
+ " as list.getSelectedValue() before = " + elementSel
+ " not equal to list.getSelectedValue() after pressing Enter = " + elementSelAfter
);
}
System.out.println("Test passed for laf: " + laf);

} finally {
SwingUtilities.invokeAndWait(JListSelectedElementTest::disposeFrame);
}
}
}

private static void hitKeys(int... keys) {
for (int key : keys) {
robot.keyPress(key);
}

for (int i = keys.length - 1; i >= 0; i--) {
robot.keyRelease(keys[i]);
}
}

private static void createUI() {
frame = new JFrame();
list = new JList<>(new String[]{"anaheim", "bill", "chicago", "dingo", "ernie", "freak"});
list.addFocusListener(new FocusAdapter() {
@Override
public void focusGained(FocusEvent e) {
listGainedFocusLatch.countDown();
}
});
JMenu menu = new JMenu("File");
menu.setMnemonic(FILE_MENU);
JMenuItem menuItem = new JMenuItem("Dummy");
menu.add(menuItem);
menu.addMenuListener(new MenuListener() {
@Override
public void menuSelected(MenuEvent e) {
menuSelectedEventLatch.countDown();
}

@Override
public void menuDeselected(MenuEvent e) {
}

@Override
public void menuCanceled(MenuEvent e) {
}
});

JMenuBar menuBar = new JMenuBar();
menuBar.add(menu);

frame.setJMenuBar(menuBar);
frame.setContentPane(list);
frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
frame.pack();
frame.setAlwaysOnTop(true);
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}

private static boolean setLookAndFeel(String lafName) {
try {
UIManager.setLookAndFeel(lafName);
} catch (UnsupportedLookAndFeelException ignored) {
System.out.println("Ignoring Unsupported L&F: " + lafName);
return false;
} catch (ClassNotFoundException | InstantiationException
| IllegalAccessException e) {
throw new RuntimeException(e);
}
return true;
}

private static void disposeFrame() {
if (frame != null) {
frame.dispose();
frame = null;
}
}

}

1 comment on commit a240a9e

@openjdk-notifier
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please sign in to comment.