在自定义CellRenderer中覆盖createToolTip()

时间:2012-04-25 12:34:13

标签: java swing jtable tooltip

我正在尝试为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();
    }
}

2 个答案:

答案 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