当DefaultTableModel发生更改时,JTable上的setModel

时间:2012-04-26 10:53:01

标签: java swing model-view-controller oop user-interface

  

TableModelListener定义侦听对象的接口   TableModel中的更改。

TableModelListener

如何将其应用于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,虽然我已经读过这些术语根据具体情况有不同的含义。

2 个答案:

答案 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上运行了。如果没有,您将需要同步对任何共享数据的访问。