Я пытался использовать код от Oracle's How To Use Tables, чтобы создать JTable с подсказками для каждого заголовка столбца. The demo, кажется, работает, но вставить ли я код непосредственно или реферата мой собственный класс, я получаю NullPointerException по вызову к getTableCellRendererComponent() в SynthTableHeaderUI.java линии 233. Это связано с вызовом header.getTable(), который возвращает null в любой таблице, я пытаюсь установить setTableHeader(), даже если я устанавливаю TableHeader (новый JTableHeader (tblWhatever.getColumnModel()));Пользовательский JTableHeader вызывает NullPointerException
Функция Я скопировал из демки находится внутри пользовательского 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];
}
}
инициализация сделано в a 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 не требует JTable быть передано ему, и я реализовал конструктор и преодолено GetTable(), который, кажется, не назовут. Когда я пишу это, я понимаю, что демонстрационная работа, которая работает, помещает функцию в пользовательский JTable, который я не хочу делать, потому что я использую среду IDE NetBeans, и я не знаю простого способа добавить пользовательскую таблицу дизайнеру.
Что мне не хватает? Как это реализовать без создания пользовательского JTable? Спасибо за любые указатели.
Чтобы лучше помочь, опубликуйте [MCVE] (http://stackoverflow.com/help/mcve) (минимальный полный проверяемый пример) или [SSCCE] (http://www.sscce.org/) (Short , Self Contained, Правильный пример). –
Я бы удалил все ссылки на JTable, включая поле thisTable и конструктор, который принимает JTable. Удалите переопределение getTable().JTableHeader уже имеет методы getTable и setTable; не мешают им. В любом случае, ваш код не будет использоваться для этой таблицы. Конечно, как только вы это сделаете, ваш класс будет выглядеть почти идентично анонимному классу в примере Oracle ..., который он должен. – VGR
Спасибо за комментарии. VGR, я согласен с тем, что JTableHeader уже реализует это, и мой собственный код не нуждается в этом. Переопределенная функция getTable() и поддерживающий код являются недавними дополнениями, чтобы попытаться справиться с сообщением об ошибке. Я получаю то же самое с или без него. –