TableModel更新后的JTable行选择

时间:2011-08-25 21:22:38

标签: swing jtable

如何在表模式更新(fireTableDataChanes)后保留表行selecetion?我知道我应该在开火前保存选择并在之后恢复(从there)。但是当我尝试在TableModelListener中恢复选择时,它不起作用。那么我应该在哪里恢复选择?

更新: 现在我尝试以这种方式恢复选择:         table.setModel(模型);

    model.addTableModelListener(new TableModelListener() {

      @Override
      public void tableChanged(TableModelEvent e) {
        table.addRowSelectionInterval(1, 1);
      }
    });

但它不起作用。

3 个答案:

答案 0 :(得分:2)

这是第一个答案的更新版本。您可以使用字段来存储行,因为内部类可以访问该行。我在使用这个解决方案时遇到了一些麻烦,因为我使用了一个定期线程来更新表。

private int selectedRow = -1;

public void mymethod() {
//put code to create the table here...
    JTable tableList = new JTable();


// Save selected row table
    tableList.getSelectionModel().addListSelectionListener(new ListSelectionListener() {
        @Override
        public void valueChanged(ListSelectionEvent e) {
            selectedRow = e.getFirstIndex();
        }
    });

// Restore selected raw table
model.addTableModelListener(new TableModelListener() {      
    @Override
    public void tableChanged(TableModelEvent e) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                if (selectedRow >= 0) {
                            tableList.addRowSelectionInterval(index, index);
                }
             }
        });
    }
    });
}

答案 1 :(得分:0)

决策很简单:我们应该使用Swing EventQueue:

final int selectedRow = 0;
// Save selected row table
tableList.getSelectionModel().addListSelectionListener(new ListSelectionListener() {
    @Override
    public void valueChanged(ListSelectionEvent e) {
        selectedRow = e.getFirstIndex();
    }
});

// Restore selected raw table
model.addTableModelListener(new TableModelListener() {      
    @Override
    public void tableChanged(TableModelEvent e) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                if (selectedRow >= 0) {
                            tableList.addRowSelectionInterval(index, index);
                }
             }
        });
    }
});

答案 2 :(得分:0)

第一个答案中的示例显示了基本解决方案,但不会开箱即用。无法更改最终的“selectedRow”。因此应该使用可变整数(例如AtomicInteger)。 tamleModelListener将调度到任务以进行摆动,因此稍后会调用它 - 在大多数情况下,在表更改已删除选择之后。所以selectedRow在大多数情况下都是-1。

我稍微改变了上面的例子(并添加了单个单元格选择,但它也适用于整行):

    final AtomicInteger selectedRow=new AtomicInteger(-1);
    final AtomicInteger selectedCol=new AtomicInteger(-1);
    tblZeiten.getSelectionModel().addListSelectionListener(new ListSelectionListener() {
        @Override
        public void valueChanged(ListSelectionEvent e)
        {
            selectedRow.set(tblZeiten.getSelectedRow());
            selectedCol.set(tblZeiten.getSelectedColumn());
        }
    });
    tblZeiten.getModel().addTableModelListener(new TableModelListener() {      
        @Override
        public void tableChanged(TableModelEvent e)
        {
            TableCellEditor editor=tblZeiten.getCellEditor();
            if (editor!=null) editor.cancelCellEditing();

            final int row=selectedRow.get();
            final int col=selectedCol.get();
            if (row<0||col<0) return;

            SwingUtilities.invokeLater(new Runnable() {
                @Override
                public void run() {
                    // http://book.javanb.com/the-java-developers-almanac-1-4/egs/javax.swing.table/Sel.html
                    tblZeiten.changeSelection(row,col, false, false);
                 }
            });
        }
    });