Конвертировать modelRowIndex в viewRowIndex для отсортированного JTable
У меня есть сортируемый JTable; когда строка добавлена, я хочу знать ее представление-index. Я попытался использовать слушатель модели таблицы следующим образом:
@Override
public void tableChanged(TableModelEvent event)
{
if (event.getType() == TableModelEvent.INSERT)
{
int modelRowIndex = event.getFirstRow();
int viewRowIndex = table.convertRowIndexToView(modelRowIndex);
System.out.println("viewRowIndex: " + viewRowIndex);
}
}
Это работает, если таблица не отсортирована. К сожалению, если таблица отсортирована, метод преобразования приводит к ArrayIndexOutOfBoundsException
вызванный DefaultRowSorter.java:503
,
Я предполагаю, что это происходит, потому что табличная модель уведомляет моего слушателя, прежде чем она уведомляет сортировщик строк. Есть идеи? Спасибо!
Вот SSCCE:
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.ArrayList;
import java.util.List;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.event.TableModelEvent;
import javax.swing.event.TableModelListener;
import javax.swing.table.AbstractTableModel;
public class RowIndexSSCCE extends JFrame
{
public static void main(String[] args)
{
SwingUtilities.invokeLater(new Runnable()
{
@Override
public void run()
{
new RowIndexSSCCE().setVisible(true);
}
});
}
private final JTable table;
public RowIndexSSCCE()
{
JButton button = new JButton("Add");
button.addActionListener(new ButtonListener());
table = new JTable(new Model());
table.setAutoCreateRowSorter(true);
table.getModel().addTableModelListener(new ModelListener());
JScrollPane scrollPane = new JScrollPane(table);
JPanel panel = new JPanel();
panel.add(button);
panel.add(scrollPane);
add(panel);
pack();
setLocationRelativeTo(null);
setDefaultCloseOperation(EXIT_ON_CLOSE);
}
private class ButtonListener implements ActionListener
{
@Override
public void actionPerformed(ActionEvent event)
{
((Model) table.getModel()).addRow("asdf" + table.getRowCount());
}
}
private class ModelListener implements TableModelListener
{
@Override
public void tableChanged(TableModelEvent event)
{
if (event.getType() == TableModelEvent.INSERT)
{
int modelRowIndex = event.getFirstRow();
int viewRowIndex = table.convertRowIndexToView(modelRowIndex);
System.out.println("viewRowIndex: " + viewRowIndex);
}
}
}
private class Model extends AbstractTableModel
{
private final List<String> data = new ArrayList<String>();
public void addRow(String string)
{
int oldSize = data.size();
data.add(string);
fireTableRowsInserted(oldSize, oldSize);
}
@Override
public int getRowCount()
{
return data.size();
}
@Override
public int getColumnCount()
{
return 1;
}
@Override
public Object getValueAt(int rowIndex, int columnIndex)
{
return data.get(rowIndex);
}
}
}
Редактировать:
Я просто понял это сам... используя SwingUtilities.invokeLater
гарантирует, что преобразование индекса строки выполняется после обновления сортировщика строк:
@Override
public void tableChanged(final TableModelEvent event)
{
if (event.getType() == TableModelEvent.INSERT)
{
SwingUtilities.invokeLater(new Runnable()
{
@Override
public void run()
{
int modelRowIndex = event.getFirstRow();
int viewRowIndex = table.convertRowIndexToView(modelRowIndex);
System.out.println("viewRowIndex: " + viewRowIndex);
}
});
}
}
1 ответ
Вы правы, слушатели модели стола информированы в неправильном порядке. Самое простое решение, которое приходит мне в голову, - это реализовать слушателей самостоятельно. Что касается меня, я никогда не реализую табличные модели, расширяя AbstractTableModel. Я нахожу более прозрачным для реализации функциональности самостоятельно.
private class Model extends AbstractTableModel {
private final List<TableModelListener> listeners = new ArrayList<TableModelListener>();
@Override
public void addTableModelListener(TableModelListener l) {
listeners.add(l);
}
@Override
public void fireTableRowsInserted(int firstRow, int lastRow) {
for (TableModelListener l : listeners) {
l.tableChanged(new TableModelEvent(Model.this, firstRow,
lastRow, TableModelEvent.ALL_COLUMNS, TableModelEvent.INSERT));
}
}
...
}