我正在尝试为JTable的特定列获取自定义工具提示。我已经创建了一个CellRenderer(我已经成功地更改了其他特定于细胞的属性):
private class CustomCellRenderer extends DefaultTableCellRenderer
{
private static final long serialVersionUID = 1L;
@Override
public Component getTableCellRendererComponent(JTable table, Object value,
boolean isSelected, boolean hasFocus, int row, int column)
{
JComponent c = (JComponent) super.getTableCellRendererComponent(table, value,
isSelected, hasFocus, row, column);
if (value != null)
{
if(column == 1 && value instanceof Date)
{
final DateFormat df = new SimpleDateFormat("h:mm aa");
table.setValueAt(df.format(value), row, column);
}
else if(column == 2)
{
c.setToolTipText((String) value);
}
else if(column == 4)
{
final Mail m = main.selectedPage.messages.get(row);
JCheckBox checkBox;
if((Boolean) value)
{
checkBox = new JCheckBox()
{
@Override
public JToolTip createToolTip()
{
System.out.println("Passed");
return new ImageToolTip(m.getImage());
}
};
checkBox.setToolTipText(m.attachName);
}
else
checkBox = new JCheckBox();
checkBox.setSelected(((Boolean)value).booleanValue());
c = checkBox;
}
}
else
{
c.setToolTipText(null);
}
return c;
}
}
当我覆盖任何其他JComponent的createTooltip()方法时,它在渲染器之外都可以正常工作。
checkBox = new JCheckBox()
{
@Override
public JToolTip createToolTip()
{
System.out.println("Passed");
return new ImageToolTip(m.getImage());
}
};
据我所知,工具提示是在别处创建的,因为“Passed”从未打印过。 checkBox.setToolTipText(m.attachName);
仅导致带有该字符串的默认工具提示。
我找到了similar question的人,但我不能说我完全理解唯一的解决方法。我是否需要扩展JTable并覆盖getToolTipText(MouseEvent e)
?如果是这样,我不知道如何获得正确的(我的)工具提示。
请原谅我任何一种自学成才的怪异。提前致谢。 : - )
感谢Robin,我能够基于JTable的getToolTipText(MouseEvent e)
代码拼凑出一些东西。我会把它留给其他有类似问题的人。同样,我不确定这是最好的方法,所以请随意批评它。 : - )
messageTable = new JTable()
{
@Override
public JToolTip createToolTip()
{
Point p = getMousePosition();
// Locate the renderer under the event location
int hitColumnIndex = columnAtPoint(p);
int hitRowIndex = rowAtPoint(p);
if ((hitColumnIndex != -1) && (hitRowIndex != -1))
{
TableCellRenderer renderer = getCellRenderer(hitRowIndex, hitColumnIndex);
Component component = prepareRenderer(renderer, hitRowIndex, hitColumnIndex);
if (component instanceof JCheckBox)
{
Image img = main.selectedPage.messages.get(hitRowIndex).getImage();
if(((JCheckBox) component).isSelected())
return new ImageToolTip(img);
}
}
return super.createToolTip();
}
}
答案 0 :(得分:4)
您无法为单元格渲染器中的复选框创建工具提示。实际上,当您尝试将鼠标移到它上时,该组件不存在。这只是一个形象。您需要为JTable创建工具提示
private void tableMouseMoved(java.awt.event.MouseEvent evt) {
String toolTipText;
int row = table.rowAtPoint(evt.getPoint());
int column = table.columnAtPoint(evt.getPoint());
if (row >= 0) {
Object o = table.getValueAt(row, column);
if (column == YourTableModel.COLUMN_INDEX_WITH_CHECKBOX) {
Boolean value = (Boolean) o;
if (value == Boolean.TRUE) {
toolTipText = "Tooltip text for true value";
} else {
toolTipText = "Tooltip text for false value";
}
}
}
}
当然,你需要为MouseEvent注册监听器:
javax.swing.JTable table = new JTable();
table.addMouseMotionListener(new java.awt.event.MouseMotionAdapter() {
public void mouseMoved(java.awt.event.MouseEvent evt) {
tableMouseMoved(evt);
}
});
答案 1 :(得分:3)
在实现中可以看到JTable
不使用您的工具提示的原因。 JTable
确实会使用渲染器返回的组件,但它会询问它的工具提示文本。因此,如果您坚持使用默认的JTable
实现,则只有自定义工具提示文本的设置才有效。只需快速复制粘贴JTable
源代码的相关部分即可说明:
if (component instanceof JComponent) {
// Convert the event to the renderer's coordinate system
Rectangle cellRect = getCellRect(hitRowIndex, hitColumnIndex, false);
p.translate(-cellRect.x, -cellRect.y);
MouseEvent newEvent = new MouseEvent(component, event.getID(),
event.getWhen(), event.getModifiers(),
p.x, p.y,
event.getXOnScreen(),
event.getYOnScreen(),
event.getClickCount(),
event.isPopupTrigger(),
MouseEvent.NOBUTTON);
tip = ((JComponent)component).getToolTipText(newEvent);
}
所以是的,如果你真的想要一个图像作为复选框的工具提示,你必须覆盖JTable
方法。
在旁注:您的渲染器代码有奇怪的行为。
final DateFormat df = new SimpleDateFormat("h:mm aa");
table.setValueAt(df.format(value), row, column);
似乎不正确。您应该用
替换setValueAt
来电
JLabel label = new JLabel();//a field in your renderer
//in the getTableCellRendererComponent method
label.setText( df.format( value ) );
return label;
或类似的东西。渲染器不应调整表值,而是创建适当的组件以可视化数据。在这种情况下,JLabel
似乎就足够了。正如斯坦尼斯拉夫在评论中注意到的那样,你不应该不断创造新的组件。这违背了引入的渲染器的目的,该目的是为了避免为每个行/列组合创建新组件。请注意,该方法称为getTableCellRendererComponent
(强调get
)而不是createTableCellRendererComponent