自定义JTableHeader导致NullPointerException

时间:2015-06-28 00:55:27

标签: java swing jtableheader

我一直在尝试使用Oracle's How To Use Tables中的代码创建一个JTable,其中包含每个列标题的工具提示。 The demo似乎有用,但是我是直接粘贴代码还是抽象我自己的类,我在中调用 getTableCellRendererComponent()时得到 NullPointerException > SynthTableHeaderUI.java 第233行。这是因为调用了 header.getTable(),它在我尝试 setTableHeader()的任何表上都返回null ,即使我是setTableHeader(new JTableHeader(tblWhatever.getColumnModel()));

我从演示中粘贴的函数位于一个自定义的TableModel中,否则它可以很好地工作,看起来像这样:

public class TestTableModel extends AbstractTableModel {
private final String[] columnNames = {"Name", "Height", "Weight", "Age"};
private final String[] columnToolTips = {"Person's Name",
                                     "Height in centimetres.",
                                     "Weight in kilograms.",
                                     "Age in years as of 2015-Jan-01."};
private ToolTipTableHeader ClientTableHeader; // = new ToolTipTableHeader((new JTable()).getColumnModel(), columnToolTips);

    private final Client[] List = {
        new Client("Abigale", 150, 108, 22),
        new Client("Bob", 180, 175, 36),
        new Client("Charles", 150, 210, 52)
    };

    /*
     * Constructors
     */
    public TestTableModel() {
        super();
    }

    public void setTableHeader(JTable tblClients) {
        tblClients.setTableHeader(createDefaultTableHeader(tblClients.getColumnModel()));
    }

    /*
     * AbstractCellEditor Implementations
     */
    @Override
    public Class getColumnClass(int col) throws java.lang.IndexOutOfBoundsException {
        switch(col) {
            case 0: return String.class;  //.ClientName;
            case 1: return Integer.class; //.Height;
            case 2: return Integer.class; //.Weight;
            case 3: return Integer.class; //.Age;
            default: throw new IndexOutOfBoundsException("Column " + col + ": class not accounted for in " + this.getClass().getName() + ".getColumnClass");
        }
    }

    @Override
    public int getColumnCount() { return columnNames.length; }

    @Override
    public String getColumnName(int col) { return columnNames[col]; }

    @Override
    public int getRowCount() { return List.length; }

    @Override
    public Object getValueAt(int row, int col) throws java.lang.IndexOutOfBoundsException {
        switch(col) {
            case 0: return List[row].ClientName;
            case 1: return List[row].Height;
            case 2: return List[row].Weight;
            case 3: return List[row].Age;
            default: throw new IndexOutOfBoundsException("Column " + col + ": value not accounted for in " + this.getClass().getName() + ".getValueAt");
        }
    }

    @Override
    public boolean isCellEditable(int row, int col) { return true; }

    @Override
    public void setValueAt(Object value, int row, int col) {
        switch(col) {
            case 0: List[row].ClientName = (String) value; break;
            case 1: List[row].Height = (Integer) value; break;
            case 2: List[row].Weight = (Integer) value; break;
            case 3: List[row].Age = (Integer) value; break;
            default: throw new IndexOutOfBoundsException("Column " + col + ": value not accounted for in " + this.getClass().getName() + ".setValueAt");
        }
        fireTableCellUpdated(row, col);
    }

    /*
     * Extensions
     */

    //Implement table header tool tips.
    protected JTableHeader createDefaultTableHeader(TableColumnModel tcmThis) {
        return new JTableHeader(tcmThis) {
            @Override
            public String getToolTipText(MouseEvent e) {
                String tip = null;
                java.awt.Point p = e.getPoint();
                int index = columnModel.getColumnIndexAtX(p.x);
                int realIndex = 
                        columnModel.getColumn(index).getModelIndex();
                return columnToolTips[realIndex];
            }
        };
    }
}

自定义类看起来像这样:

public class ToolTipTableHeader extends JTableHeader {
    private final String ColumnToolTips[];

    ToolTipTableHeader(TableColumnModel cm, String iniToolTips[]) {
        super(cm);

        if(iniToolTips.length != cm.getColumnCount()) 
            throw new InvalidParameterException("The size of iniToolTips must be precisely equal to the columnModel column count.");
        ColumnToolTips = iniToolTips;
    }

    @Override
    public String getToolTipText(MouseEvent meToolTipEvent) {
        String tip = null;
        if(columnModel == null) return "columnModel == null";
        if(meToolTipEvent == null) return "meMouseEvent == null";
        Point p = meToolTipEvent.getPoint();
        int index = columnModel.getColumnIndexAtX(p.x);
        int realIndex = columnModel.getColumn(index).getModelIndex();
        return ColumnToolTips[realIndex];
    }
}

初始化在 JDialog 构造函数中完成( JTable tblTest 在设计器中创建):

public TestForm(java.awt.Frame parent, boolean modal) {
    super(parent, modal);
    initComponents();

    TestTableModel htmTest = new TestTableModel();
    tblTest.setModel(new TestTableModel());
    htmTest.setTableHeader(tblTest);
}

我注意到默认JTableHeader的构造函数不需要传递给它,我已经实现了一个构造函数和一个被覆盖的 getTable(),它没有&#39似乎被称为。在我写这篇文章的过程中,我意识到有效的演示将函数放在一个自定义的 JTable 中,我不想这样做,因为我使用的是NetBeans IDE而且我没有&#39 ; t知道一种向设计器添加自定义表的简单方法。

我错过了什么?如何在不创建自定义 JTable 的情况下实现此目的?谢谢你的任何指示。

1 个答案:

答案 0 :(得分:2)

JTableHeader 扩展程序必须作为构建 JTable 的一部分添加。表头的任何后续设置都会导致上述错误。如果使用NetBeans,请转到对话框/表单/面板的设计模式,选择相关表格,然后单击属性代码选项卡窗口。添加(例如)

new javax.swing.JTable() {
    //Implement table header tool tips.
    protected JTableHeader createDefaultTableHeader() {
        return new JTableHeader(columnModel) {
            public String getToolTipText(MouseEvent e) {
                String tip = null;
                java.awt.Point p = e.getPoint();
                int index = columnModel.getColumnIndexAtX(p.x);
                int realIndex =  columnModel.getColumn(index).getModelIndex();
                return myToolTips[realIndex];
            }
        };
    }
};

自定义创建代码字段,并确保定义

private final String[] myToolTips = {
    "Column 0 tool tip",
    "Column 1 tool tip",
    // ...
    "Final column tool tip"
};

对话框/表单/面板类中的某个位置。这可行,但仍然有点不优雅,因为它需要为使用该表模型的每个表单完成,而不是能够将其合并并且只有一个类来设置表。