-
Bug
-
Resolution: Fixed
-
P3
-
1.3.1, 1.3.1_03
-
04
-
x86, sparc
-
linux, solaris_8, windows_nt, windows_2000, windows_xp
SelectedNode();
DefaultMutableTreeNode parent;
/* Determine where to create the new node. */
if(lastItem != null) {
parent = (DefaultMutableTreeNode)lastItem.getParent();
if(parent == null) {
parent = (DefaultMutableTreeNode)treeModel.getRoot();
lastItem = null;
}
}
else
parent = (DefaultMutableTreeNode)treeModel.getRoot();
if(lastItem == null)
newIndex = treeModel.getChildCount(parent);
else
newIndex = parent.getIndex(lastItem);
/* Let the treemodel know. */
treeModel.insertNodeInto(createNewNode("Inserted " +
Integer.toString(insertCount++)),
parent, newIndex);
}
} // End of SampleTree.InsertAction
/**
* ReloadAction is used to reload from the selected node. If nothing
* is selected, reload is not issued.
*/
class ReloadAction extends Object implements ActionListener
{
/**
* Messaged when the user clicks on the Reload menu item.
* Determines the selection from the Tree and asks the treemodel
* to reload from that node.
*/
public void actionPerformed(ActionEvent e) {
DefaultMutableTreeNode lastItem = getSelectedNode();
if(lastItem != null)
treeModel.reload(lastItem);
}
} // End of SampleTree.ReloadAction
/**
* RemoveAction removes the selected node from the tree. If
* The root or nothing is selected nothing is removed.
*/
class RemoveAction extends Object implements ActionListener
{
/**
* Removes the selected item as long as it isn't root.
*/
public void actionPerformed(ActionEvent e) {
DefaultMutableTreeNode lastItem = getSelectedNode();
if(lastItem != null && lastItem != (DefaultMutableTreeNode)
treeModel.getRoot()) {
treeModel.removeNodeFromParent(lastItem);
}
}
} // End of SampleTree.RemoveAction
/**
* ShowHandlesChangeListener implements the ChangeListener interface
* to toggle the state of showing the handles in the tree.
*/
class ShowHandlesChangeListener extends Object implements ChangeListener
{
public void stateChanged(ChangeEvent e) {
tree.setShowsRootHandles(((JCheckBox)e.getSource()).isSelected());
}
} // End of class SampleTree.ShowHandlesChangeListener
/**
* ShowRootChangeListener implements the ChangeListener interface
* to toggle the state of showing the root node in the tree.
*/
class ShowRootChangeListener extends Object implements ChangeListener
{
public void stateChanged(ChangeEvent e) {
tree.setRootVisible(((JCheckBox)e.getSource()).isSelected());
}
} // End of class SampleTree.ShowRootChangeListener
/**
* TreeEditableChangeListener implements the ChangeListener interface
* to toggle between allowing editing and now allowing editing in
* the tree.
*/
class TreeEditableChangeListener extends Object implements ChangeListener
{
public void stateChanged(ChangeEvent e) {
tree.setEditable(((JCheckBox)e.getSource()).isSelected());
}
} // End of class SampleTree.TreeEditableChangeListener
// *************************** !CANDLE ***********************************
// * BEGIN DEFINITION OF JTREE EXTENSION TO DEMONSTRATE TOOLTIP PROBLEM *
// * UNDER 1.3.1_03. *
// *************************** !CANDLE ***********************************
/**
* ToolTipTree
*/
class ToolTipTree extends JTree
{
private Point currentTipLocation;
public ToolTipTree( TreeModel model )
{
super( model );
}
/**
* Returns the location for the origin of the tooltip, or
* null to hide the tooltip.
*/
public Point getToolTipLocation( MouseEvent event )
{
// Determine if we are hovering over a row in the tree...
int row = tree.getRowForLocation( event.getX(), event.getY() );
if (row > -1)
{
// set tool tip origin to be at current mouse position...
currentTipLocation = event.getPoint();
}
else
{
// set tool tip location to null to hide tool tip...
currentTipLocation = null;
}
return currentTipLocation;
}
/**
* Returns the string to be used as the tooltip text.
*/
public String getToolTipText( MouseEvent event )
{
String tooltiptext = null;
Point tipLocation = null;
// See if hovering over a row in the tree...
tipLocation = getToolTipLocation( event );
if (tipLocation != null)
{
// Use label associated with tree row as tip text...
TreePath path = getPathForLocation( tipLocation.x, tipLocation.y );
tooltiptext = path.getLastPathComponent().toString();
}
return tooltiptext;
}
} // End of class SampleTree.ToolTipTree
// *************************** !CANDLE ***********************************
// * END DEFINITION OF JTREE EXTENSION TO DEMONSTRATE TOOLTIP PROBLEM *
// * UNDER 1.3.1_03. *
// *************************** !CANDLE ***********************************
static public void main(String args[]) {
new SampleTree();
}
}
---------- END SOURCE ----------
CUSTOMER WORKAROUND :
No workaround found, but desparately needed. Falling back
to 1.3.1_02 run-time does work, but is not desired.
(Review ID: 145250)
======================================================================
Name: jk109818 Date: 04/19/2002
FULL PRODUCT VERSION :
java version "1.3.1_03"
Java(TM) 2 Runtime Environment, Standard Edition (build 1.3.1_03-b03)
Java HotSpot(TM) Client VM (build 1.3.1_03-b03, mixed mode)
FULL OPERATING SYSTEM VERSION : Windows NT 4.0, SP6a
A DESCRIPTION OF THE PROBLEM :
We have an application where tooltips are displayed when a
user positions the mouse over a row (node) in JTree. The
information in the tooltip varies depending on the state of
the underlying model associated with the row being
targeted. When we determine that the mouse position is no
longer hovering over a JTree row, the tooltip display is
dismissed.
Under 1.3.1_02 and prior 1.3.1 releases, this tooltip
feature worked fine. However, under 1.3.1_03, this feature
has been regressed. In short, the tooltip either never
appears when hovering over a visible tree row, or it will
appear only once over the first targeted tree row.
After comparing the source changes made to the
javax\swing\ToolTipManager class between the 1.3.1_02 and
1.3.1_03, the regression appears to have been introduced by
the addition and removal of the following statement
relative to the 1.3.1_02 level release:
insideComponent = null;
If I change the 1.3.1_03 version of the ToolTipManager
class to be identical to the 1.3.1_02 version with respect
to the above statement only, the regression disappears and
the tooltips work normally.
I have included a modified version of the SampleTree demo
application packaged with the 1.3.1_03 SDK. Specifically,
I included the modified source code for the SampleTree.java
module found in the SampleTree\src directory. The sample
was modified to show a tooltip whenever the mouse is
positioned over a tree row. The text associated with the
tooltip varies based on the label associated with the
visible tree row itself. All changed and/or added
statements made to SampleTree.java are tagged with the
literal "!CANDLE" for easy identification.
If you compile and run the SampleTree demo with this change
under the 1.3.1_03 run-time, you will observe that no
tooltips are displayed. However, if you run the exact same
test with the 1.3.1_02 run-time, you will observe that the
tooltips are displayed as the mouse hovers over each
visible tree row and, further, that the tooltip is
dismissed (hidden) when the mouse is not positioned over a
visible tree row.
Again, the regression is related to the excessive
nullification of the insideComponent field in the 1.3.1_03
run-time relative to the more sparse use of this statement
in the 1.3.1_02 run-time.
Thank you in advance for your assistance.
REGRESSION. Last worked in version 1.3.1_02
STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
1. Recompile the SampleTree demo with attached version of
modified SampleTree.java module.
2. Using the 1.3.1_03 run-time, run the SampleTree demo
and position the mouse over any visible tree row. Observe
that no visible tooltip will be displayed.
3. Using the 1.3.1_02 run-time, run the same SampleTree
demo and position the mouse over any visible tree row.
Observe that a tooltip appears with the name of the tree
row label text.
EXPECTED VERSUS ACTUAL BEHAVIOR :
Tooltips should be displayed when mouse is positioned over
visible tree row. Tooltips should be hidden when mouse is
no longer over a visible tree row.
This bug can be reproduced always.
---------- BEGIN SOURCE ----------
/*
* Copyright 2002 Sun Microsystems, Inc. All rights reserved.
* SUN PROPRIETARY/CONFIDENTIAL. Use is subject to license terms.
*/
import javax.swing.*;
import javax.swing.event.*;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import javax.swing.tree.*;
import java.awt.event.MouseEvent; // !CANDLE
import java.awt.Point; // !CANDLE
/**
* A demo for illustrating how to do different things with JTree.
* The data that this displays is rather boring, that is each node will
* have 7 children that have random names based on the fonts. Each node
* is then drawn with that font and in a different color.
* While the data isn't interesting the example illustrates a number
* of things:
*
* For an example of dynamicaly loading children refer to DynamicTreeNode.
* For an example of adding/removing/inserting/reloading refer to the inner
* classes of this class, AddAction, RemovAction, InsertAction and
* ReloadAction.
* For an example of creating your own cell renderer refer to
* SampleTreeCellRenderer.
* For an example of subclassing JTreeModel for editing refer to
* SampleTreeModel.
*
* @version 1.18 02/06/02
* @author Scott Violet
*/
public class SampleTree
{
/** Window for showing Tree. */
protected JFrame frame;
/** Tree used for the example. */
protected JTree tree;
/** Tree model. */
protected DefaultTreeModel treeModel;
/**
* Constructs a new instance of SampleTree.
*/
public SampleTree() {
// Force SampleTree to come up in the Cross Platform L&F
try {
UIManager.setLookAndFeel
(UIManager.getCrossPlatformLookAndFeelClassName());
// If you want the System L&F instead, comment out the above line
and
// uncomment the following:
// UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName
());
} catch (Exception exc) {
System.err.println("Error loading L&F: " + exc);
}
JMenuBar menuBar = constructMenuBar();
JPanel panel = new JPanel(true);
frame = new JFrame("SampleTree");
frame.getContentPane().add("Center", panel);
frame.setJMenuBar(menuBar);
frame.setBackground(Color.lightGray);
/* Create the JTreeModel. */
DefaultMutableTreeNode root = createNewNode("Root");
treeModel = new SampleTreeModel(root);
/* Create the tree. */
// tree = new JTree(treeModel); !CANDLE
tree = new SampleTree.ToolTipTree(treeModel); // !CANDLE
/* Enable tool tips for the tree, without this tool tips will not
be picked up. */
ToolTipManager.sharedInstance().registerComponent(tree);
/* Make the tree use an instance of SampleTreeCellRenderer for
drawing. */
tree.setCellRenderer(new SampleTreeCellRenderer());
/* Make tree ask for the height of each row. */
tree.setRowHeight(-1);
/* Put the Tree in a scroller. */
JScrollPane sp = new JScrollPane();
sp.setPreferredSize(new Dimension(300, 300));
sp.getViewport().add(tree);
/* And show it. */
panel.setLayout(new BorderLayout());
panel.add("Center", sp);
panel.add("South", constructOptionsPanel());
frame.addWindowListener( new WindowAdapter() {
public void windowClosing(WindowEvent e) {System.exit(0);}});
frame.pack();
frame.show();
}
/** Constructs a JPanel containing check boxes for the different
* options that tree supports. */
private JPanel constructOptionsPanel() {
JCheckBox aCheckbox;
JPanel retPanel = new JPanel(false);
JPanel borderPane = new JPanel(false);
borderPane.setLayout(new BorderLayout());
retPanel.setLayout(new FlowLayout());
aCheckbox = new JCheckBox("show handles");
aCheckbox.setSelected(tree.getShowsRootHandles());
aCheckbox.addChangeListener(new ShowHandlesChangeListener());
retPanel.add(aCheckbox);
aCheckbox = new JCheckBox("show root");
aCheckbox.setSelected(tree.isRootVisible());
aCheckbox.addChangeListener(new ShowRootChangeListener());
retPanel.add(aCheckbox);
aCheckbox = new JCheckBox("editable");
aCheckbox.setSelected(tree.isEditable());
aCheckbox.addChangeListener(new TreeEditableChangeListener());
aCheckbox.setToolTipText("Triple click to edit");
retPanel.add(aCheckbox);
borderPane.add(retPanel, BorderLayout.CENTER);
/* Create a set of radio buttons that dictate what selection should
be allowed in the tree. */
ButtonGroup group = new ButtonGroup();
JPanel buttonPane = new JPanel(false);
JRadioButton button;
buttonPane.setLayout(new FlowLayout());
button = new JRadioButton("Single");
button.addActionListener(new AbstractAction() {
public boolean isEnabled() { return true; }
public void actionPerformed(ActionEvent e) {
tree.getSelectionModel().setSelectionMode
(TreeSelectionModel.SINGLE_TREE_SELECTION);
}
});
group.add(button);
buttonPane.add(button);
button = new JRadioButton("Contiguous");
button.addActionListener(new AbstractAction() {
public boolean isEnabled() { return true; }
public void actionPerformed(ActionEvent e) {
tree.getSelectionModel().setSelectionMode
(TreeSelectionModel.CONTIGUOUS_TREE_SELECTION);
}
});
group.add(button);
buttonPane.add(button);
button = new JRadioButton("Discontiguous");
button.addActionListener(new AbstractAction() {
public boolean isEnabled() { return true; }
public void actionPerformed(ActionEvent e) {
tree.getSelectionModel().setSelectionMode
(TreeSelectionModel.DISCONTIGUOUS_TREE_SELECTION);
}
});
button.setSelected(true);
group.add(button);
buttonPane.add(button);
borderPane.add(buttonPane, BorderLayout.SOUTH);
// NOTE: This will be enabled in a future release.
// Create a label and combobox to determine how many clicks are
// needed to expand.
/*
JPanel clickPanel = new JPanel();
Object[] values = { "Never", new Integer(1),
new Integer(2), new Integer(3) };
final JComboBox clickCBox = new JComboBox(values);
clickPanel.setLayout(new FlowLayout());
clickPanel.add(new JLabel("Click count to expand:"));
clickCBox.setSelectedIndex(2);
clickCBox.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent ae) {
Object selItem = clickCBox.getSelectedItem();
if(selItem instanceof Integer)
tree.setToggleClickCount(((Integer)selItem).intValue());
else // Don't toggle
tree.setToggleClickCount(0);
}
});
clickPanel.add(clickCBox);
borderPane.add(clickPanel, BorderLayout.NORTH);
*/
return borderPane;
}
/** Construct a menu. */
private JMenuBar constructMenuBar() {
JMenu menu;
JMenuBar menuBar = new JMenuBar();
JMenuItem menuItem;
/* Good ol exit. */
menu = new JMenu("File");
menuBar.add(menu);
menuItem = menu.add(new JMenuItem("Exit"));
menuItem.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
System.exit(0);
}});
/* Tree related stuff. */
menu = new JMenu("Tree");
menuBar.add(menu);
menuItem = menu.add(new JMenuItem("Add"));
menuItem.addActionListener(new AddAction());
menuItem = menu.add(new JMenuItem("Insert"));
menuItem.addActionListener(new InsertAction());
menuItem = menu.add(new JMenuItem("Reload"));
menuItem.addActionListener(new ReloadAction());
menuItem = menu.add(new JMenuItem("Remove"));
menuItem.addActionListener(new RemoveAction());
return menuBar;
}
/**
* Returns the TreeNode instance that is selected in the tree.
* If nothing is selected, null is returned.
*/
protected DefaultMutableTreeNode getSelectedNode() {
TreePath selPath = tree.getSelectionPath();
if(selPath != null)
return (DefaultMutableTreeNode)selPath.getLastPathComponent();
return null;
}
protected DefaultMutableTreeNode createNewNode(String name) {
return new DynamicTreeNode(new SampleData(null, Color.black, name));
}
/**
* AddAction is used to add a new item after the selected item.
*/
class AddAction extends Object implements ActionListener
{
/** Number of nodes that have been added. */
public int addCount;
/**
* Messaged when the user clicks on the Add menu item.
* Determines the selection from the Tree and adds an item
* after that. If nothing is selected, an item is added to
* the root.
*/
public void actionPerformed(ActionEvent e) {
int newIndex;
DefaultMutableTreeNode lastItem = getSelectedNode();
DefaultMutableTreeNode parent;
/* Determine where to create the new node. */
if(lastItem != null) {
parent = (DefaultMutableTreeNode)lastItem.getParent();
if(parent == null) {
parent = (DefaultMutableTreeNode)treeModel.getRoot();
lastItem = null;
}
}
else
parent = (DefaultMutableTreeNode)treeModel.getRoot();
if(lastItem == null)
newIndex = treeModel.getChildCount(parent);
else
newIndex = parent.getIndex(lastItem) + 1;
/* Let the treemodel know. */
treeModel.insertNodeInto(createNewNode("Added " +
Integer.toString(addCount++)),
parent, newIndex);
}
} // End of SampleTree.AddAction
/**
* InsertAction is used to insert a new item before the selected item.
*/
class InsertAction extends Object implements ActionListener
{
/** Number of nodes that have been added. */
public int insertCount;
/**
* Messaged when the user clicks on the Insert menu item.
* Determines the selection from the Tree and inserts an item
* after that. If nothing is selected, an item is added to
* the root.
*/
public void actionPerformed(ActionEvent e) {
int newIndex;
DefaultMutableTreeNode lastItem = get
DefaultMutableTreeNode parent;
/* Determine where to create the new node. */
if(lastItem != null) {
parent = (DefaultMutableTreeNode)lastItem.getParent();
if(parent == null) {
parent = (DefaultMutableTreeNode)treeModel.getRoot();
lastItem = null;
}
}
else
parent = (DefaultMutableTreeNode)treeModel.getRoot();
if(lastItem == null)
newIndex = treeModel.getChildCount(parent);
else
newIndex = parent.getIndex(lastItem);
/* Let the treemodel know. */
treeModel.insertNodeInto(createNewNode("Inserted " +
Integer.toString(insertCount++)),
parent, newIndex);
}
} // End of SampleTree.InsertAction
/**
* ReloadAction is used to reload from the selected node. If nothing
* is selected, reload is not issued.
*/
class ReloadAction extends Object implements ActionListener
{
/**
* Messaged when the user clicks on the Reload menu item.
* Determines the selection from the Tree and asks the treemodel
* to reload from that node.
*/
public void actionPerformed(ActionEvent e) {
DefaultMutableTreeNode lastItem = getSelectedNode();
if(lastItem != null)
treeModel.reload(lastItem);
}
} // End of SampleTree.ReloadAction
/**
* RemoveAction removes the selected node from the tree. If
* The root or nothing is selected nothing is removed.
*/
class RemoveAction extends Object implements ActionListener
{
/**
* Removes the selected item as long as it isn't root.
*/
public void actionPerformed(ActionEvent e) {
DefaultMutableTreeNode lastItem = getSelectedNode();
if(lastItem != null && lastItem != (DefaultMutableTreeNode)
treeModel.getRoot()) {
treeModel.removeNodeFromParent(lastItem);
}
}
} // End of SampleTree.RemoveAction
/**
* ShowHandlesChangeListener implements the ChangeListener interface
* to toggle the state of showing the handles in the tree.
*/
class ShowHandlesChangeListener extends Object implements ChangeListener
{
public void stateChanged(ChangeEvent e) {
tree.setShowsRootHandles(((JCheckBox)e.getSource()).isSelected());
}
} // End of class SampleTree.ShowHandlesChangeListener
/**
* ShowRootChangeListener implements the ChangeListener interface
* to toggle the state of showing the root node in the tree.
*/
class ShowRootChangeListener extends Object implements ChangeListener
{
public void stateChanged(ChangeEvent e) {
tree.setRootVisible(((JCheckBox)e.getSource()).isSelected());
}
} // End of class SampleTree.ShowRootChangeListener
/**
* TreeEditableChangeListener implements the ChangeListener interface
* to toggle between allowing editing and now allowing editing in
* the tree.
*/
class TreeEditableChangeListener extends Object implements ChangeListener
{
public void stateChanged(ChangeEvent e) {
tree.setEditable(((JCheckBox)e.getSource()).isSelected());
}
} // End of class SampleTree.TreeEditableChangeListener
// *************************** !CANDLE ***********************************
// * BEGIN DEFINITION OF JTREE EXTENSION TO DEMONSTRATE TOOLTIP PROBLEM *
// * UNDER 1.3.1_03. *
// *************************** !CANDLE ***********************************
/**
* ToolTipTree
*/
class ToolTipTree extends JTree
{
private Point currentTipLocation;
public ToolTipTree( TreeModel model )
{
super( model );
}
/**
* Returns the location for the origin of the tooltip, or
* null to hide the tooltip.
*/
public Point getToolTipLocation( MouseEvent event )
{
// Determine if we are hovering over a row in the tree...
int row = tree.getRowForLocation( event.getX(), event.getY() );
if (row > -1)
{
// set tool tip origin to be at current mouse position...
currentTipLocation = event.getPoint();
}
else
{
// set tool tip location to null to hide tool tip...
currentTipLocation = null;
}
return currentTipLocation;
}
/**
* Returns the string to be used as the tooltip text.
*/
public String getToolTipText( MouseEvent event )
{
String tooltiptext = null;
Point tipLocation = null;
// See if hovering over a row in the tree...
tipLocation = getToolTipLocation( event );
if (tipLocation != null)
{
// Use label associated with tree row as tip text...
TreePath path = getPathForLocation( tipLocation.x, tipLocation.y );
tooltiptext = path.getLastPathComponent().toString();
}
return tooltiptext;
}
} // End of class SampleTree.ToolTipTree
// *************************** !CANDLE ***********************************
// * END DEFINITION OF JTREE EXTENSION TO DEMONSTRATE TOOLTIP PROBLEM *
// * UNDER 1.3.1_03. *
// *************************** !CANDLE ***********************************
static public void main(String args[]) {
new SampleTree();
}
}
---------- END SOURCE ----------
CUSTOMER WORKAROUND :
No workaround found, but desparately needed. Falling back
to 1.3.1_02 run-time does work, but is not desired.
(Review ID: 145250)
======================================================================
Name: jk109818 Date: 04/19/2002
FULL PRODUCT VERSION :
java version "1.3.1_03"
Java(TM) 2 Runtime Environment, Standard Edition (build 1.3.1_03-b03)
Java HotSpot(TM) Client VM (build 1.3.1_03-b03, mixed mode)
FULL OPERATING SYSTEM VERSION : Windows NT 4.0, SP6a
A DESCRIPTION OF THE PROBLEM :
We have an application where tooltips are displayed when a
user positions the mouse over a row (node) in JTree. The
information in the tooltip varies depending on the state of
the underlying model associated with the row being
targeted. When we determine that the mouse position is no
longer hovering over a JTree row, the tooltip display is
dismissed.
Under 1.3.1_02 and prior 1.3.1 releases, this tooltip
feature worked fine. However, under 1.3.1_03, this feature
has been regressed. In short, the tooltip either never
appears when hovering over a visible tree row, or it will
appear only once over the first targeted tree row.
After comparing the source changes made to the
javax\swing\ToolTipManager class between the 1.3.1_02 and
1.3.1_03, the regression appears to have been introduced by
the addition and removal of the following statement
relative to the 1.3.1_02 level release:
insideComponent = null;
If I change the 1.3.1_03 version of the ToolTipManager
class to be identical to the 1.3.1_02 version with respect
to the above statement only, the regression disappears and
the tooltips work normally.
I have included a modified version of the SampleTree demo
application packaged with the 1.3.1_03 SDK. Specifically,
I included the modified source code for the SampleTree.java
module found in the SampleTree\src directory. The sample
was modified to show a tooltip whenever the mouse is
positioned over a tree row. The text associated with the
tooltip varies based on the label associated with the
visible tree row itself. All changed and/or added
statements made to SampleTree.java are tagged with the
literal "!CANDLE" for easy identification.
If you compile and run the SampleTree demo with this change
under the 1.3.1_03 run-time, you will observe that no
tooltips are displayed. However, if you run the exact same
test with the 1.3.1_02 run-time, you will observe that the
tooltips are displayed as the mouse hovers over each
visible tree row and, further, that the tooltip is
dismissed (hidden) when the mouse is not positioned over a
visible tree row.
Again, the regression is related to the excessive
nullification of the insideComponent field in the 1.3.1_03
run-time relative to the more sparse use of this statement
in the 1.3.1_02 run-time.
Thank you in advance for your assistance.
REGRESSION. Last worked in version 1.3.1_02
STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
1. Recompile the SampleTree demo with attached version of
modified SampleTree.java module.
2. Using the 1.3.1_03 run-time, run the SampleTree demo
and position the mouse over any visible tree row. Observe
that no visible tooltip will be displayed.
3. Using the 1.3.1_02 run-time, run the same SampleTree
demo and position the mouse over any visible tree row.
Observe that a tooltip appears with the name of the tree
row label text.
EXPECTED VERSUS ACTUAL BEHAVIOR :
Tooltips should be displayed when mouse is positioned over
visible tree row. Tooltips should be hidden when mouse is
no longer over a visible tree row.
This bug can be reproduced always.
---------- BEGIN SOURCE ----------
/*
* Copyright 2002 Sun Microsystems, Inc. All rights reserved.
* SUN PROPRIETARY/CONFIDENTIAL. Use is subject to license terms.
*/
import javax.swing.*;
import javax.swing.event.*;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import javax.swing.tree.*;
import java.awt.event.MouseEvent; // !CANDLE
import java.awt.Point; // !CANDLE
/**
* A demo for illustrating how to do different things with JTree.
* The data that this displays is rather boring, that is each node will
* have 7 children that have random names based on the fonts. Each node
* is then drawn with that font and in a different color.
* While the data isn't interesting the example illustrates a number
* of things:
*
* For an example of dynamicaly loading children refer to DynamicTreeNode.
* For an example of adding/removing/inserting/reloading refer to the inner
* classes of this class, AddAction, RemovAction, InsertAction and
* ReloadAction.
* For an example of creating your own cell renderer refer to
* SampleTreeCellRenderer.
* For an example of subclassing JTreeModel for editing refer to
* SampleTreeModel.
*
* @version 1.18 02/06/02
* @author Scott Violet
*/
public class SampleTree
{
/** Window for showing Tree. */
protected JFrame frame;
/** Tree used for the example. */
protected JTree tree;
/** Tree model. */
protected DefaultTreeModel treeModel;
/**
* Constructs a new instance of SampleTree.
*/
public SampleTree() {
// Force SampleTree to come up in the Cross Platform L&F
try {
UIManager.setLookAndFeel
(UIManager.getCrossPlatformLookAndFeelClassName());
// If you want the System L&F instead, comment out the above line
and
// uncomment the following:
// UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName
());
} catch (Exception exc) {
System.err.println("Error loading L&F: " + exc);
}
JMenuBar menuBar = constructMenuBar();
JPanel panel = new JPanel(true);
frame = new JFrame("SampleTree");
frame.getContentPane().add("Center", panel);
frame.setJMenuBar(menuBar);
frame.setBackground(Color.lightGray);
/* Create the JTreeModel. */
DefaultMutableTreeNode root = createNewNode("Root");
treeModel = new SampleTreeModel(root);
/* Create the tree. */
// tree = new JTree(treeModel); !CANDLE
tree = new SampleTree.ToolTipTree(treeModel); // !CANDLE
/* Enable tool tips for the tree, without this tool tips will not
be picked up. */
ToolTipManager.sharedInstance().registerComponent(tree);
/* Make the tree use an instance of SampleTreeCellRenderer for
drawing. */
tree.setCellRenderer(new SampleTreeCellRenderer());
/* Make tree ask for the height of each row. */
tree.setRowHeight(-1);
/* Put the Tree in a scroller. */
JScrollPane sp = new JScrollPane();
sp.setPreferredSize(new Dimension(300, 300));
sp.getViewport().add(tree);
/* And show it. */
panel.setLayout(new BorderLayout());
panel.add("Center", sp);
panel.add("South", constructOptionsPanel());
frame.addWindowListener( new WindowAdapter() {
public void windowClosing(WindowEvent e) {System.exit(0);}});
frame.pack();
frame.show();
}
/** Constructs a JPanel containing check boxes for the different
* options that tree supports. */
private JPanel constructOptionsPanel() {
JCheckBox aCheckbox;
JPanel retPanel = new JPanel(false);
JPanel borderPane = new JPanel(false);
borderPane.setLayout(new BorderLayout());
retPanel.setLayout(new FlowLayout());
aCheckbox = new JCheckBox("show handles");
aCheckbox.setSelected(tree.getShowsRootHandles());
aCheckbox.addChangeListener(new ShowHandlesChangeListener());
retPanel.add(aCheckbox);
aCheckbox = new JCheckBox("show root");
aCheckbox.setSelected(tree.isRootVisible());
aCheckbox.addChangeListener(new ShowRootChangeListener());
retPanel.add(aCheckbox);
aCheckbox = new JCheckBox("editable");
aCheckbox.setSelected(tree.isEditable());
aCheckbox.addChangeListener(new TreeEditableChangeListener());
aCheckbox.setToolTipText("Triple click to edit");
retPanel.add(aCheckbox);
borderPane.add(retPanel, BorderLayout.CENTER);
/* Create a set of radio buttons that dictate what selection should
be allowed in the tree. */
ButtonGroup group = new ButtonGroup();
JPanel buttonPane = new JPanel(false);
JRadioButton button;
buttonPane.setLayout(new FlowLayout());
button = new JRadioButton("Single");
button.addActionListener(new AbstractAction() {
public boolean isEnabled() { return true; }
public void actionPerformed(ActionEvent e) {
tree.getSelectionModel().setSelectionMode
(TreeSelectionModel.SINGLE_TREE_SELECTION);
}
});
group.add(button);
buttonPane.add(button);
button = new JRadioButton("Contiguous");
button.addActionListener(new AbstractAction() {
public boolean isEnabled() { return true; }
public void actionPerformed(ActionEvent e) {
tree.getSelectionModel().setSelectionMode
(TreeSelectionModel.CONTIGUOUS_TREE_SELECTION);
}
});
group.add(button);
buttonPane.add(button);
button = new JRadioButton("Discontiguous");
button.addActionListener(new AbstractAction() {
public boolean isEnabled() { return true; }
public void actionPerformed(ActionEvent e) {
tree.getSelectionModel().setSelectionMode
(TreeSelectionModel.DISCONTIGUOUS_TREE_SELECTION);
}
});
button.setSelected(true);
group.add(button);
buttonPane.add(button);
borderPane.add(buttonPane, BorderLayout.SOUTH);
// NOTE: This will be enabled in a future release.
// Create a label and combobox to determine how many clicks are
// needed to expand.
/*
JPanel clickPanel = new JPanel();
Object[] values = { "Never", new Integer(1),
new Integer(2), new Integer(3) };
final JComboBox clickCBox = new JComboBox(values);
clickPanel.setLayout(new FlowLayout());
clickPanel.add(new JLabel("Click count to expand:"));
clickCBox.setSelectedIndex(2);
clickCBox.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent ae) {
Object selItem = clickCBox.getSelectedItem();
if(selItem instanceof Integer)
tree.setToggleClickCount(((Integer)selItem).intValue());
else // Don't toggle
tree.setToggleClickCount(0);
}
});
clickPanel.add(clickCBox);
borderPane.add(clickPanel, BorderLayout.NORTH);
*/
return borderPane;
}
/** Construct a menu. */
private JMenuBar constructMenuBar() {
JMenu menu;
JMenuBar menuBar = new JMenuBar();
JMenuItem menuItem;
/* Good ol exit. */
menu = new JMenu("File");
menuBar.add(menu);
menuItem = menu.add(new JMenuItem("Exit"));
menuItem.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
System.exit(0);
}});
/* Tree related stuff. */
menu = new JMenu("Tree");
menuBar.add(menu);
menuItem = menu.add(new JMenuItem("Add"));
menuItem.addActionListener(new AddAction());
menuItem = menu.add(new JMenuItem("Insert"));
menuItem.addActionListener(new InsertAction());
menuItem = menu.add(new JMenuItem("Reload"));
menuItem.addActionListener(new ReloadAction());
menuItem = menu.add(new JMenuItem("Remove"));
menuItem.addActionListener(new RemoveAction());
return menuBar;
}
/**
* Returns the TreeNode instance that is selected in the tree.
* If nothing is selected, null is returned.
*/
protected DefaultMutableTreeNode getSelectedNode() {
TreePath selPath = tree.getSelectionPath();
if(selPath != null)
return (DefaultMutableTreeNode)selPath.getLastPathComponent();
return null;
}
protected DefaultMutableTreeNode createNewNode(String name) {
return new DynamicTreeNode(new SampleData(null, Color.black, name));
}
/**
* AddAction is used to add a new item after the selected item.
*/
class AddAction extends Object implements ActionListener
{
/** Number of nodes that have been added. */
public int addCount;
/**
* Messaged when the user clicks on the Add menu item.
* Determines the selection from the Tree and adds an item
* after that. If nothing is selected, an item is added to
* the root.
*/
public void actionPerformed(ActionEvent e) {
int newIndex;
DefaultMutableTreeNode lastItem = getSelectedNode();
DefaultMutableTreeNode parent;
/* Determine where to create the new node. */
if(lastItem != null) {
parent = (DefaultMutableTreeNode)lastItem.getParent();
if(parent == null) {
parent = (DefaultMutableTreeNode)treeModel.getRoot();
lastItem = null;
}
}
else
parent = (DefaultMutableTreeNode)treeModel.getRoot();
if(lastItem == null)
newIndex = treeModel.getChildCount(parent);
else
newIndex = parent.getIndex(lastItem) + 1;
/* Let the treemodel know. */
treeModel.insertNodeInto(createNewNode("Added " +
Integer.toString(addCount++)),
parent, newIndex);
}
} // End of SampleTree.AddAction
/**
* InsertAction is used to insert a new item before the selected item.
*/
class InsertAction extends Object implements ActionListener
{
/** Number of nodes that have been added. */
public int insertCount;
/**
* Messaged when the user clicks on the Insert menu item.
* Determines the selection from the Tree and inserts an item
* after that. If nothing is selected, an item is added to
* the root.
*/
public void actionPerformed(ActionEvent e) {
int newIndex;
DefaultMutableTreeNode lastItem = get
- duplicates
-
JDK-4673672 Tool-Tip fails to appear for 1.3.1_04 b01 in test case JTabbedPaneTest0002.
-
- Closed
-
-
JDK-4673904 Tooltip do not appear for the tabs in the JTabbedPane in 1.3.1_03 release.
-
- Closed
-
-
JDK-4682451 REGRESSION:ToolTipManager behaves incorrectly in 1.3.1_03
-
- Closed
-