我有一个 JTable,它使用 RowSorter 进行排序和过滤,并以实时方式更新。当用户尝试使用鼠标(单击 + 拖动行)或通过键盘(SHIFT + 向下箭头多次)创建选择间隔时,我遇到了问题。
一旦发生表模型更新事件,选择间隔开始和结束就会损坏。我发现,如果对表进行了排序并且有多个选定行,则选择模型会触发多个列表选择事件,这些事件会清除然后在每次表模型更新时重新创建选择。
如果一次选择多行 - 一切都很好。用户选择的内容保持选中状态。但是,如果用户尝试通过在多行上单击并向下拖动鼠标来创建选择间隔 - 行将被正确选择,直到触发表模型更新事件。然后选择从当前选定的行开始构建。
我附上了一个小型演示应用程序。尝试使用鼠标选择多行 - 每一秒选择都会被破坏......
这似乎是 Java 类中的一个错误(或不需要的行为)。我希望有人可以提供创造性的解决方案/解决方法。
public class TableTest extends JFrame {
private static final long serialVersionUID = 1L;
public TableTest() {
super("Table Selection Problem");
setBounds(50, 50, 800, 600);
setDefaultCloseOperation(EXIT_ON_CLOSE);
// Create a table model
final DefaultTableModel tableModel = new DefaultTableModel(40,10);
// Create table row sorter sorted on column 1
TableRowSorter<TableModel> sorter = new TableRowSorter<TableModel>(tableModel);
sorter.setSortsOnUpdates(true);
sorter.setSortKeys(Arrays.asList(new RowSorter.SortKey(1,SortOrder.DESCENDING)));
// Create a table and bind the sorter
JTable table = new JTable(tableModel);
table.setRowSorter(sorter);
// Bind a list selection listener, this doesn't do anything besides showing the selection change(s) on every table model update
table.getSelectionModel().addListSelectionListener(new ListSelectionListener() {
@Override
public void valueChanged(ListSelectionEvent e) {
System.out.println(e);
}
});
// place the table in the frame
getContentPane().add(new JScrollPane(table), BorderLayout.CENTER);
// Simulate some application events that trigger a table update...
new Thread() {
public void run() {
try {
final Random r = new Random();
for(int x = 0; x < 1000; x++) {
// update a table cell (not even an add or delete)
SwingUtilities.invokeLater(new Runnable() {
public void run() {
tableModel.setValueAt(r.nextInt(1000), r.nextInt(40), r.nextInt(10));
}
});
// wait 1 second before next table update
Thread.sleep(1000);
}
} catch(Throwable t) {
t.printStackTrace();
}
}
}.start();
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
(new TableTest()).setVisible(true);
}
});
}
}