1

以下の例では、JTableaJListと2つJButtonのs(追加と削除)があります。リストには、追加ボタンをクリックすると、選択した値がテーブルに追加される6つの項目(文字列)があります。
テーブル内の文字列は、カスタムレンダラー(JPanelボタンとラベル付き)を使用して表示されます。ボタンのテキストとラベルのテキストが文字列の値に変更されます。
編集者がエントリを作成するまで、すべてうまくいきます。エディターはボタンをクリックすることを可能にするので、それは必要です。
初めて正しく表示される文字列をテーブルに追加すると、行の高さがパネルの希望の高さに調整され、ボタンとラベルにテキストが設定されます。
行をクリックしてテーブルからエントリを削除し、[削除]ボタンをクリックすると、すべて期待どおりに実行されます。
ここで問題が発生します。テーブルに(異なる)文字列を追加すると、行の高さが高くなり、ラベルとボタンのテキストが設定されません(レンダラーとエディターの両方が呼び出されないため、チェックしました)ブレークポイントを使用)。
もちろん、カスタムレンダラーを使用して新しい行を表示したいのですが、どうすればよいですか?

package test;

import java.awt.Component;
import java.awt.event.ActionEvent;
import java.util.EventObject;
import javax.swing.AbstractAction;
import javax.swing.DefaultListModel;
import javax.swing.JButton;
import javax.swing.JLabel;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.UIManager;
import javax.swing.event.CellEditorListener;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableCellEditor;
import javax.swing.table.TableCellRenderer;
import javax.swing.table.TableColumn;

public class MainForm
        extends javax.swing.JFrame
{
    private JTable table;
    private JScrollPane tableScrollPane;
    private JList list;
    private JScrollPane listScrollPane;
    private JButton add;
    private JButton remove;

    public MainForm()
    {
        tableScrollPane = new JScrollPane();
        table = new JTable();
        listScrollPane = new JScrollPane();
        list = new JList();
        add = new JButton(new AbstractAction()
        {
            public void actionPerformed(ActionEvent e)
            {
                add();
            }
        });
        add.setText("add");
        remove = new JButton(new AbstractAction()
        {
            public void actionPerformed(ActionEvent e)
            {
                remove();
            }
        });
        remove.setText("remove");

        setLayout(new javax.swing.BoxLayout(getContentPane(), javax.swing.BoxLayout.LINE_AXIS));

        tableScrollPane.setViewportView(table);
        listScrollPane.setViewportView(list);

        add(tableScrollPane);

        DefaultTableModel model = new DefaultTableModel();
        model.addColumn("test");
        table.setModel(model);

        TableColumn col = table.getColumn("test");
        col.setCellRenderer(new CustomTableCellRenderer());
        col.setCellEditor(new CustomTableCellEditor());

        DefaultListModel listModel = new DefaultListModel();
        listModel.addElement("test1");
        listModel.addElement("test2");
        listModel.addElement("test3");
        listModel.addElement("test4");
        listModel.addElement("test5");
        listModel.addElement("test6");
        list.setModel(listModel);

        add(listScrollPane);
        add(add);
        add(remove);
    }

    private void add()
    {
        DefaultTableModel model = (DefaultTableModel) table.getModel();
        model.addRow(new Object[]
                {
                    list.getSelectedValue()
                });
    }

    private void remove()
    {
        int selectedRow = table.getSelectedRow();
        DefaultTableModel model = (DefaultTableModel) table.getModel();
        model.removeRow(selectedRow);
    }

    public static void main(String[] args)
    {
        new MainForm().setVisible(true);
    }

    public class CustomTableCellRenderer
            extends customPanel
            implements TableCellRenderer
    {
        public Component getTableCellRendererComponent(JTable table,
                                                       Object value,
                                                       boolean isSelected,
                                                       boolean hasFocus, int row,
                                                       int column)
        {
            setText((String) value);
            if (isSelected || hasFocus)
            {
                setBackground(UIManager.getColor("List.selectionBackground"));
                setForeground(UIManager.getColor("List.selectionForeground"));
            }
            else
            {
                setBackground(UIManager.getColor("Panel.background"));
                setForeground(UIManager.getColor("Panel.foreground"));
            }
            table.setRowHeight(row, (int)getPreferredSize().height);
            return this;
        }
    }

    public class CustomTableCellEditor
            extends customPanel
            implements TableCellEditor
    {
        Object value;

        public Component getTableCellEditorComponent(JTable table, Object value,
                                                     boolean isSelected, int row,
                                                     int column)
        {
            this.value = value;
            setText((String) value);
            setBackground(UIManager.getColor("List.selectionBackground"));
            setForeground(UIManager.getColor("List.selectionForeground"));
            table.setRowHeight(row, (int)getPreferredSize().height);
            return this;
        }

        public Object getCellEditorValue()
        {
            return value;
        }

        public boolean isCellEditable(EventObject anEvent)
        {
            return true;
        }

        public boolean shouldSelectCell(EventObject anEvent)
        {
            return true;
        }

        public boolean stopCellEditing()
        {
            setBackground(UIManager.getColor("Panel.background"));
            setForeground(UIManager.getColor("Panel.foreground"));
            return true;
        }

        public void cancelCellEditing()
        {
        }

        public void addCellEditorListener(CellEditorListener l)
        {
        }

        public void removeCellEditorListener(CellEditorListener l)
        {
        }
    }

    public class customPanel
            extends JPanel
    {
        private JLabel label;
        private JButton button;

        public customPanel()
        {
            label = new JLabel();
            button = new JButton();
            add(label);
            add(button);
        }

        public void setText(String text)
        {
            label.setText(text);
            button.setText(text);
        }
    }
}
4

2 に答える 2

2

絶対に使用しないでください:

table.setRowHeight(row, (int)getPreferredSize().height);

レンダラーで。行の高さを変更するとテーブルが行をrepaint()し、レンダラーが呼び出されると再び行の高さを変更するため、これにより無限ループが発生します。

そのコード行を削除します。代わりに、行をテーブルに追加するときに行の高さを計算できます。このコードをadd()メソッドの最後に追加します。

int row = table.getRowCount() - 1;
Component comp = table.prepareRenderer(table.getCellRenderer(row, 0), row, 0);
int rowHeight = comp.getPreferredSize().height;
table.setRowHeight(row, rowHeight);

アップデート:

コードは、カスタムエディタがなくても正常に機能します。したがって、問題はエディターです。AbstractCellEditorのソースコードを見て、編集を停止したときに何が起こるかを確認してください。エディターを削除できるように、テーブルに通知するイベントを発生させます。このコードはありません。customPanelしたがって、適切なイベントを簡単に発生させることができるように、拡張する代わりにAbstractCellEditorを拡張することをお勧めします。

また、行をクリックするとエディターが呼び出されると思います。そのため、モデルから行を削除する前に、エディターを削除する必要があります。これを行ういくつかの方法については、テーブルの編集停止を参照してください。

于 2011-12-05T20:57:37.760 に答える
0

addCellEditorListener()、removeCellEditorListener()、stopCellEditing()、cancelCellEditing()メソッドの実装は魅力のように機能します。選択したオブジェクトを削除する前に、エディターを「切断」する必要があります。
私のテーブルの1つに、テーブルエントリを変更するためのコンテキストメニューがあり、削除する前に、cancelCellEditing()またはstopCellEditing()を呼び出す必要があります。そうしないと、テーブルがエントリを解放しないか、以前のように再表示されます。

于 2011-12-06T12:19:17.300 に答える