3

いくつかのフィールドと削除ボタンを含むTableCellEditorを実装しようとしています。正常に機能しますが、行が削除されても、削除されたセル(TableCellEditorでレンダリング)のコンテンツは更新されません。

fireTableRowsDeleted(row, row)行が削除されたときにモデル内との両方を呼び出そうとしfireTableDataChanged()ましたが、TableCellEditorに通知されないようです。別の行を選択すると機能し、行インデックスはTableCellRendererで再度レンダリングされます。

削除時にTableCellEditorに通知する方法に関する提案はありますか?

  1. 削除ボタンが押されました

    ここに画像の説明を入力してください

  2. 行は削除されましたが、CellEditorのコンテンツは更新されていません

    ここに画像の説明を入力してください

  3. CellRendererが再度使用されると、行の内容が更新されます。

    ここに画像の説明を入力してください

コードは次のとおりです。

public class StringTableDemo extends JFrame {

    public StringTableDemo() {
        
        final StringTableModel model = new StringTableModel();
        model.addRow("Jonas");
        model.addRow("Hello");
        model.addRow("World");
        
        RendererAndEditor rendererAndEditor = new RendererAndEditor(model);
        
        JTable table = new JTable(model);
        table.setDefaultRenderer(Record.class, rendererAndEditor);
        table.setDefaultEditor(Record.class, rendererAndEditor);
        
        add(new JScrollPane(table), BorderLayout.CENTER);
        pack();
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);
    }
    
    class Record {
        String string;
        boolean isDeleted;
    }
    
    class StringTableModel extends AbstractTableModel {

        private final List<Record> data = new ArrayList<Record>();
        
        @Override
        public int getColumnCount() {
            return 1;
        }

        @Override
        public int getRowCount() {
            return data.size();
        }

        @Override
        public Object getValueAt(int row, int column) {
            return data.get(row);
        }
        
        @Override
        public Class<?> getColumnClass(int column) {
            return Record.class;
        }
        
        @Override
        public boolean isCellEditable(int row, int column) {
            return true;
        }
        
        @Override
        public void setValueAt(Object aValue, int row, int column) {
            if(aValue instanceof Record) {
                Record r = (Record)aValue;
                if(!r.isDeleted) {
                    data.set(row, r);
                    fireTableRowsUpdated(row, column);
                }
            } else throw new IllegalStateException("aValue is not a Record");
        }
        
        public void addRow(String s) {
            Record r = new Record();
            r.string = s;
            r.isDeleted = false;
            data.add(r);
            fireTableRowsInserted(data.size()-1, data.size()-1);
        }
        
        public void removeRow(int row) {
            data.remove(row);
            //fireTableRowsDeleted(row, row);
            fireTableDataChanged();
            
            System.out.println("row " + row + " deleted");
        }

    }
    
    class CellPanel extends JPanel {
        private final Action removeAction = new AbstractAction("x") {

            @Override
            public void actionPerformed(ActionEvent arg0) {
                model.removeRow(index);
                isDeleted = true;
            }
            
        };
        private final JButton removeBtn = new JButton(removeAction);
        private final JTextField field = new JTextField();
        private final StringTableModel model;
        private int index;
        private boolean isDeleted = false;
        public CellPanel(StringTableModel model) {
            super(new BorderLayout());
            this.model = model;
            add(field, BorderLayout.CENTER);
            add(removeBtn, BorderLayout.EAST);
        }
        
        public Record getRecord() {
            Record r = new Record();
            r.string = field.getText();
            r.isDeleted = isDeleted;
            return r;
        }
        
        public void setRecord(Record r, int index) {
            field.setText(r.string);
            this.index = index;
        }
    }
    
    class RendererAndEditor extends AbstractCellEditor implements
     TableCellEditor, TableCellRenderer {

        private final CellPanel renderer;
        private final CellPanel editor;
        
        public RendererAndEditor(StringTableModel model) {
            renderer = new CellPanel(model);
            editor = new CellPanel(model);
        }
        
        @Override
        public Object getCellEditorValue() {
            return editor.getRecord();
        }

        @Override
        public Component getTableCellRendererComponent(JTable table,
                Object value, boolean isSelected, boolean hasFocus, 
                       int row, int column) {
            
            renderer.setRecord((Record)value, row);
            return renderer;
        }

        @Override
        public Component getTableCellEditorComponent(JTable table, 
                Object value, boolean isSelected, int row, int column) {
            editor.setRecord((Record)value, row);
            return editor;
        }
        
    }

    
    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {

            @Override
            public void run() {
                new StringTableDemo();
            }
            
        });
    }
}
4

1 に答える 1

4

これはCellEditor、削除ボタンをクリックしたときに、セルの編集を停止する必要があることに気付かないためです。

簡単な解決策は、別のものを追加ActionListenerして、クリックするたびCellEditorに呼び出すことです。stopCellEditing()これはうまくいくはずです:

public RendererAndEditor( StringTableModel model )
{
  renderer = new CellPanel( model );
  editor = new CellPanel( model );

  editor.getRemoveBtn().addActionListener( new ActionListener()
  {
    @Override
    public void actionPerformed( ActionEvent e )
    {
      stopCellEditing();
    }
  });
}
于 2012-01-10T09:58:26.380 に答える