TableModelListener定义侦听对象的接口 TableModel中的更改。
如何将其应用于JTable,以便它将监听DefaultTableModel中的更改?此外,在调用setModel之后,不应该内置该功能吗?您为什么不希望JTable反映模型的变化?
News中defaultTableModel的引用指向与MessagesController中相同的实例(这是正确的吗?),那么为什么我必须在JTable上显式调用setModel,如果底层对象已被更新?
JTable更新自身的更好方法是什么?也许是tableChanged?
我不明白,如果News中的引用和MessagesController中的引用都指向不同的对象,并且具有相同的值,为什么需要调用setModel()。毕竟,News.defaultTableModel现在已经更新。为什么要重新调用setModel()?
package net.bounceme.dur.nntp;
import java.awt.Dimension;
import java.awt.GridLayout;
import java.awt.Toolkit;
import java.util.logging.Logger;
import javax.swing.*;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;
import javax.swing.table.DefaultTableModel;
public class News {
private static final Logger LOG = Logger.getLogger(News.class.getName());
static JFrame frame = new JFrame();
static JTextPane text = new JTextPane();
static JSlider slider = new JSlider();
static MessagesController messagesController = new MessagesController();
static DefaultTableModel defaultTableModel = new DefaultTableModel();
static JTable table = new JTable();
private static void createAndShowGUI() {
defaultTableModel = messagesController.getDefaultTableModel();
table.setModel(defaultTableModel);
table.getSelectionModel().setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
table.getSelectionModel().addListSelectionListener(new ListSelectionListener() {
public void valueChanged(ListSelectionEvent evt) {
java.awt.EventQueue.invokeLater(new Runnable() {
public void run() {
int row = table.convertRowIndexToModel(table.getSelectedRow());
row = Math.abs(row); //how can this be negative?
LOG.fine("row " + row);
MessageBean messageBean = messagesController.getMessageBean(row);
text.setText(messageBean.getContent());
text.setContentType("text/html");
}
});
}
});
slider.setMinimum(1);
slider.setMaximum(messagesController.getMax());
slider.setValue(messagesController.getMax());
slider.addChangeListener(new javax.swing.event.ChangeListener() {
public void stateChanged(javax.swing.event.ChangeEvent evt) {
java.awt.EventQueue.invokeLater(new Runnable() {
public void run() {
int index = slider.getValue();
LOG.fine("slider " + index);
messagesController.setIndex(index);
defaultTableModel = messagesController.getDefaultTableModel();
table.setModel(defaultTableModel);
table.getSelectionModel().setSelectionInterval(1, 1);
}
});
}
});
table.getSelectionModel().setSelectionInterval(1, 1);
JPanel panel = new JPanel();
panel.setLayout(new GridLayout(3, 1));
panel.add(table);
panel.add(text);
panel.add(slider);
frame.add(panel);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
frame.pack();
frame.setVisible(true);
frame.setSize(screenSize);
}
public static void main(String args[]) {
javax.swing.SwingUtilities.invokeLater(new Runnable() {
public void run() {
createAndShowGUI();
}
});
}
}
可能我误解了pass-by-value,虽然我已经读过这些术语根据具体情况有不同的含义。
答案 0 :(得分:3)
1)row = Math.abs(row); //how can this be negative?
如果没有任何选定的行,那么代码行
int row = table.convertRowIndexToModel(table.getSelectedRow());
返回-1
值,最好是在任何事情之前测试if(table.getSelectedRow() > -1)
2)没有理由替换TableModel,也没有DefaultTableModel,JTable可以返回其TableModel,JTable#getModel
3)不确定来自上下文,但也许你看看JTable Sorting and Filtering
答案 1 :(得分:3)
如果基础对象已更新,为什么我必须在
setModel()
上显式调用JTable
?
你不应该。
JTable
更新自己的更好方法是什么?
更新模型TableModel
应触发通知视图JTable
所需的事件以进行更新。 DefaultTableModel
为你做这件事; AbstractTableModel
为您的模型调用提供了方便的fireXxx()
方法。有一个后者here的例子。另请参阅How to Use Tables: Creating a Table Model。
在DefaultTableModel
中创建一个新的空News
似乎是虚假的,因为(看不见的)MessagesController
模型会立即替换它。
还不清楚为什么在invokeLater()
和ListSelectionListener
中使用ChangeListener
。这些方法应该已经在event dispatch thread上运行了。如果没有,您将需要同步对任何共享数据的访问。