2013-09-20 13:07:11 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2013, 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.
|
|
|
|
*/
|
2016-07-07 14:21:45 +00:00
|
|
|
/*
|
|
|
|
@test
|
2020-05-23 07:57:09 +00:00
|
|
|
@key headful
|
2013-09-20 13:07:11 +00:00
|
|
|
@bug 4927934
|
|
|
|
@summary JTree traversal is unlike Native windows tree traversal
|
|
|
|
@author Andrey Pikalev
|
|
|
|
@run main bug4927934
|
|
|
|
*/
|
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
import java.awt.Robot;
|
|
|
|
import java.awt.event.FocusEvent;
|
|
|
|
import java.awt.event.FocusListener;
|
|
|
|
import java.awt.event.KeyEvent;
|
|
|
|
|
|
|
|
import javax.swing.JFrame;
|
|
|
|
import javax.swing.JTree;
|
|
|
|
import javax.swing.SwingUtilities;
|
|
|
|
import javax.swing.UIManager;
|
|
|
|
import javax.swing.JScrollPane;
|
|
|
|
import javax.swing.event.TreeSelectionListener;
|
|
|
|
import javax.swing.event.TreeExpansionListener;
|
|
|
|
import javax.swing.event.TreeSelectionEvent;
|
|
|
|
import javax.swing.event.TreeExpansionEvent;
|
|
|
|
import javax.swing.tree.DefaultMutableTreeNode;
|
2013-09-20 13:07:11 +00:00
|
|
|
|
|
|
|
public class bug4927934 implements TreeSelectionListener, TreeExpansionListener, FocusListener {
|
|
|
|
|
|
|
|
final static Object listener = new bug4927934();
|
|
|
|
|
|
|
|
static boolean focusGained = false;
|
|
|
|
public static boolean selectionChanged = false;
|
|
|
|
public static boolean treeExpanded = false;
|
|
|
|
public static boolean treeCollapsed = false;
|
|
|
|
|
|
|
|
static JFrame frame;
|
|
|
|
static JTree tree;
|
|
|
|
static Robot robot;
|
|
|
|
|
|
|
|
public static void main(String args[]) throws Exception {
|
|
|
|
UIManager.setLookAndFeel(new javax.swing.plaf.metal.MetalLookAndFeel());
|
|
|
|
|
|
|
|
robot = new Robot();
|
|
|
|
robot.setAutoDelay(50);
|
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
try {
|
|
|
|
SwingUtilities.invokeAndWait(new Runnable() {
|
|
|
|
public void run() {
|
|
|
|
frame = new JFrame();
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
DefaultMutableTreeNode root = new DefaultMutableTreeNode("root");
|
|
|
|
createNodes(root);
|
|
|
|
tree = new JTree(root);
|
|
|
|
JScrollPane scrollPane = new JScrollPane(tree);
|
|
|
|
frame.getContentPane().add(scrollPane);
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
tree.addFocusListener((FocusListener) listener);
|
|
|
|
tree.addTreeSelectionListener((TreeSelectionListener) listener);
|
|
|
|
tree.addTreeExpansionListener((TreeExpansionListener) listener);
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
frame.setSize(300, 300);
|
|
|
|
frame.setVisible(true);
|
|
|
|
}
|
|
|
|
});
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
robot.waitForIdle();
|
|
|
|
Thread.sleep(1000);
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
SwingUtilities.invokeLater(new Runnable() {
|
|
|
|
public void run() {
|
|
|
|
tree.requestFocus();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
synchronized (listener) {
|
|
|
|
if (!focusGained) {
|
|
|
|
System.out.println("waiting focusGained...");
|
|
|
|
try {
|
|
|
|
listener.wait(10000);
|
|
|
|
} catch (InterruptedException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
// GO TO RIGHT
|
|
|
|
selectionChanged = false;
|
|
|
|
hitKey(KeyEvent.VK_RIGHT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!checkSelectionChanged(tree, 0)) {
|
|
|
|
throw new RuntimeException("Root should be selected");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
selectionChanged = false;
|
|
|
|
hitKey(KeyEvent.VK_RIGHT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!checkSelectionChanged(tree, 1)) {
|
|
|
|
throw new RuntimeException("Node should be selected");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
treeExpanded = false;
|
|
|
|
hitKey(KeyEvent.VK_RIGHT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!isTreeExpanded()) {
|
|
|
|
throw new RuntimeException("Node should be expanded");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
selectionChanged = false;
|
|
|
|
hitKey(KeyEvent.VK_RIGHT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!checkSelectionChanged(tree, 2)) {
|
|
|
|
throw new RuntimeException("Leaf1 should be selected");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
selectionChanged = false;
|
|
|
|
hitKey(KeyEvent.VK_RIGHT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!checkSelectionChanged(tree, 2)) {
|
|
|
|
throw new RuntimeException("Leaf1 should be selected");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
// GO TO LEFT
|
|
|
|
selectionChanged = false;
|
|
|
|
hitKey(KeyEvent.VK_LEFT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!checkSelectionChanged(tree, 1)) {
|
|
|
|
throw new RuntimeException("Node should be selected");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
treeCollapsed = false;
|
|
|
|
hitKey(KeyEvent.VK_LEFT);
|
|
|
|
if (!isTreeCollapsed()) {
|
|
|
|
throw new RuntimeException("Node should be collapsed");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
selectionChanged = false;
|
|
|
|
hitKey(KeyEvent.VK_LEFT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!checkSelectionChanged(tree, 0)) {
|
|
|
|
throw new RuntimeException("Root should be selected");
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
|
2020-05-23 07:57:09 +00:00
|
|
|
treeCollapsed = false;
|
|
|
|
hitKey(KeyEvent.VK_LEFT);
|
|
|
|
robot.waitForIdle();
|
|
|
|
if (!isTreeCollapsed()) {
|
|
|
|
throw new RuntimeException("Root should be collapsed");
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (frame != null) {
|
|
|
|
SwingUtilities.invokeAndWait(frame::dispose);
|
|
|
|
}
|
2013-09-20 13:07:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
synchronized public void focusLost(FocusEvent e) {
|
|
|
|
}
|
|
|
|
|
|
|
|
synchronized public void focusGained(FocusEvent e) {
|
|
|
|
focusGained = true;
|
|
|
|
System.out.println("focusGained");
|
|
|
|
listener.notifyAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void createNodes(DefaultMutableTreeNode root) {
|
|
|
|
DefaultMutableTreeNode node = new DefaultMutableTreeNode("Node");
|
|
|
|
node.add(new DefaultMutableTreeNode("Leaf1"));
|
|
|
|
node.add(new DefaultMutableTreeNode("Leaf2"));
|
|
|
|
root.add(node);
|
|
|
|
root.add(new DefaultMutableTreeNode("Leaf3"));
|
|
|
|
}
|
|
|
|
|
|
|
|
synchronized public void valueChanged(TreeSelectionEvent e) {
|
|
|
|
selectionChanged = true;
|
|
|
|
System.out.println("selectionChanged");
|
|
|
|
notifyAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
synchronized public void treeCollapsed(TreeExpansionEvent e) {
|
|
|
|
System.out.println("treeCollapsed");
|
|
|
|
treeCollapsed = true;
|
|
|
|
notifyAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
synchronized public void treeExpanded(TreeExpansionEvent e) {
|
|
|
|
System.out.println("treeExpanded");
|
|
|
|
treeExpanded = true;
|
|
|
|
notifyAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void hitKey(int key) {
|
|
|
|
System.out.println("key " + key + " pressed");
|
|
|
|
robot.keyPress(key);
|
|
|
|
robot.keyRelease(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean checkSelectionChanged(JTree tree, int shouldBeSel) {
|
|
|
|
synchronized(listener) {
|
|
|
|
if (!selectionChanged) {
|
|
|
|
System.out.println("waiting for selectionChanged...");
|
|
|
|
try {
|
|
|
|
listener.wait(5000);
|
|
|
|
} catch (InterruptedException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int selRow = tree.getLeadSelectionRow();
|
|
|
|
System.out.println("Selected row: " + selRow);
|
|
|
|
return selRow == shouldBeSel;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean isTreeExpanded() {
|
|
|
|
synchronized(listener) {
|
|
|
|
if (!treeExpanded) {
|
|
|
|
System.out.println("waiting for treeExpanded...");
|
|
|
|
try {
|
|
|
|
listener.wait(5000);
|
|
|
|
} catch (InterruptedException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return treeExpanded;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean isTreeCollapsed() {
|
|
|
|
synchronized(listener) {
|
|
|
|
if (!treeCollapsed) {
|
|
|
|
System.out.println("waiting for treeCollapsed...");
|
|
|
|
try {
|
|
|
|
listener.wait(5000);
|
|
|
|
} catch (InterruptedException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return treeCollapsed;
|
|
|
|
}
|
|
|
|
}
|