1

好的。我有一个大型项目,其中一个特定的 jtable 在启动时创建并且永远不会重建。表格模型被刷新并根据各种用户操作重新绘制表格。

我添加了一个自定义 TableCellListener 类来响应单元格更改以及一个 AbstractAction。这是第一次用数据填充表时执行的代码。(如果没有“firstLoad”检查,每次重绘表格时都会附加多个操作)。

if(firstLoad) {
    AbstractAction action = new AbstractAction()
    {
        public void actionPerformed(ActionEvent e)
        {
            TableCellListener tcl = (TableCellListener)e.getSource();


                    sayIt("Row:" + tcl.getRow()+" Column:" + tcl.getColumn()+
                        " Old:" + tcl.getOldValue()+" New:" + tcl.getNewValue());

        }
    };

    firstLoad = false;
    TableCellListener tcl = new TableCellListener(table2, action);
}

TableCellListener 是Rob Camick 在此处发布的自定义侦听器,“sayIt”位是我自己的调试代码。

这一切都很好,但我想在每次重建表时完全删除监听器并再次添加它,因为它正在“记住”最后一个选定单元格的值,现在无效,因为表数据是新的。

我相当确定“removePropertyChangeListener()”类型调用会做到这一点,但它希望侦听器作为参数,我不知道如何找到它。

4

3 回答 3

4

因为它正在“记住”最后一个选定单元格中的值,现在由于表格数据是新的,因此该值无效。

它应该在您开始编辑时保存当前值,并在您停止编辑时生成事件。当您更改 TableModel 时,您不应该编辑任何单元格。因此,当您生成下一个事件时,这意味着您选择并开始在不同的单元格上进行编辑,在这种情况下,您应该拥有新模型的当前值。这对我来说可以:

import java.awt.*;
import java.awt.event.*;
import java.util.Random;
import java.io.*;
import java.net.*;
import javax.swing.*;
import javax.swing.table.*;

public class TableCellListenerTest2
{
    public static void main(String[] args)
    {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                createAndShowGUI();
            }
        });
    }

    public static void createAndShowGUI()
    {
        final JTable table = new JTable( TableCellListenerTest2.createModel());
        table.setPreferredScrollableViewportSize(table.getPreferredSize());
        JScrollPane scrollPane = new JScrollPane(table);

        Action action = new AbstractAction()
        {
            public void actionPerformed(ActionEvent e)
            {
                TableCellListener tcl = (TableCellListener)e.getSource();
                System.out.println( tcl.getOldValue() + " : " + tcl.getNewValue() );
            }
        };

        TableCellListener tcl = new TableCellListener(table, action);

        JButton button = new JButton("Reset Model");
        button.addActionListener( new ActionListener()
        {
            public void actionPerformed(ActionEvent e)
            {
                table.setModel( TableCellListenerTest2.createModel() );
            }
        });

        JFrame.setDefaultLookAndFeelDecorated(true);
        JFrame frame = new JFrame("Table Cell Listener");
        frame.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
        frame.add( scrollPane );
        frame.add(button, BorderLayout.SOUTH);
        frame.setSize(400, 160);
        frame.setLocationRelativeTo( null );
        frame.setVisible(true);
    }

    public static TableModel createModel()
    {
        Random random = new Random();

        DefaultTableModel model = new DefaultTableModel(10, 2);

        for (int i = 0; i < model.getRowCount(); i++)
            model.setValueAt("" + random.nextInt(100), i, 0);

        return model;
    }
}

如果您需要更多帮助,请发布您的 SSCCE。

于 2011-12-25T00:31:43.587 回答
1

为什么不干脆把TableCellListener,tcl,一个类字段,如果重建模型,去掉它,然后重建监听器,重新添加。

于 2011-12-24T22:48:25.290 回答
0

只需记住实例变量中的 TableCellListener 实例:

// side effect: the tcl is added as PropertyChangeListener to the table 
// (bad design, IMHO)
this.tcl = new TableCellListener(table2, action);

// when the table data changes: 
if (this.tcl != null) {
    table2.removePropertyChangeListener(this.tcl);
}
this.tcl = new TableCellListener(table2, action);
于 2011-12-24T22:48:51.153 回答