Java 自定义TableCellEditor显示添加时的上一项

Java 自定义TableCellEditor显示添加时的上一项,java,swing,tablecellrenderer,tablecelleditor,Java,Swing,Tablecellrenderer,Tablecelleditor,在下面的示例中,我有一个JTableaJList和两个JButtons(添加和删除)。单击“添加”按钮时,列表中有6项(字符串),选定值将添加到表中。 表中的字符串通过使用自定义呈现器(带有按钮和标签的JPanel显示)。按钮的文本和标签的文本将更改为字符串的值。 一切都很顺利,直到编辑发表评论。编辑器允许单击按钮,因此它是必要的。 当一个字符串第一次被正确显示时,该行的高度将被调整为面板的首选高度,并为按钮和标签设置文本。 单击行,然后单击“删除”按钮,从表中删除条目时,一切按预期进行。 现在

在下面的示例中,我有一个
JTable
a
JList
和两个
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);
        }
    }
}
您不应使用:

table.setRowHeight(row, (int)getPreferredSize().height);
在渲染器中。这将导致无限循环,因为无论何时更改行高度,表都将重新绘制()行,并且当调用渲染器时,您将再次更改行高度

去掉那一行代码。相反,您可以在将行添加到表中时计算行高。将以下代码添加到Add()方法的底部:

更新:

没有自定义编辑器,代码工作正常。所以问题是编辑。查看AbstractCellEditor的源代码,了解停止编辑时会发生什么。它触发一个事件来通知表,以便删除编辑器。你没有这个密码。因此,我建议您扩展AbstractCellEditor,而不是扩展
customPanel
,以便可以轻松触发相应的事件


此外,我相信单击一行将调用编辑器,因此在从模型中删除该行之前,需要先删除编辑器。请参阅以了解执行此操作的几种方法。

实现addCellEditorListener()、removeCellEditorListener()、stopCellEditing()和CancellEditing()方法非常有效。删除选定对象之前,需要“断开”编辑器。

在我的一个表中,我有一个上下文菜单来修改表条目,在删除之前,我必须调用CancellEditing()或stopCellEditing(),否则表不会释放该条目,或者它会像以前一样重新出现。

谢谢,我不知道,但很遗憾,它没有解决问题。@siebz0r,确实如此。我在回复之前测试了发布的代码。用新代码更新您的SSCCE。@camickr我这样做了,它只固定了新行的高度设置,但新行仍然显示最后选定的条目。@camickr我已尝试使用
AbstractCellEditor
制作适当的单元格编辑器。但是,当我这样做时,我不能扩展另一个对象。由于我在实际代码中使用了一个自定义面板,其中包含两个UI元素和一个特定的布局,我发现扩展该面板是一个更好的解决方案,因为我必须创建编辑器事件或重新创建容器接口。如果我能扩展AbstractCellEditor,我会的@siebz0r,无需扩展自定义面板。相反,渲染器和编辑器会创建自定义面板的实例。您的自定义编辑器实现无效(这可能是问题所在,也可能不是问题所在)-根据约定,当由于内部原因终止编辑时,它必须通知其侦听器。您的不能简单地遵守该约定……请学习java命名约定并坚持它们(总是!)我总是使用命名约定,除非它不影响可读性,用于测试目的。我会努力寻找丢失的通知,如果它有效的话,这是有意义的。不遵守约定总是会影响可读性——因此,在您向他人展示的代码中不这样做(您希望从他人那里获得帮助,也就是说他们必须阅读和理解它,最好一目了然)是最糟糕的地方
int row = table.getRowCount() - 1;
Component comp = table.prepareRenderer(table.getCellRenderer(row, 0), row, 0);
int rowHeight = comp.getPreferredSize().height;
table.setRowHeight(row, rowHeight);