1

Ok。起動時に特定のjtableが作成され、再構築されない大規模なプロジェクトがあります。テーブルモデルが更新され、さまざまなユーザーアクションに基づいてテーブルが再描画されます。

AbstractActionとともにセルの変更に反応するカスタムTableCellListenerクラスを追加しました。これは、テーブルにデータが最初に入力されたときに実行されるコードです。(「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 に答える