gpt4 book ai didi

java - 自定义 TableCellEditor 在添加时显示上一个条目

转载 作者:行者123 更新时间:2023-11-30 04:56:26 24 4
gpt4 key购买 nike

在下面的示例中,我有一个 JTable、一个 JList 和两个 JButton(添加和删除)。列表中有 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() 方法的底部:

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 的源代码,看看编辑停止时会发生什么。它会触发一个事件来通知表,以便可以删除编辑器。你没有这个代码。因此,我建议您扩展 AbstractCellEditor 而不是扩展 customPanel,这样您就可以轻松触发适当的事件。

此外,我相信单击一行将调用编辑器,因此您需要先删除编辑器,然后再从模型中删除该行。请参阅Table Stop Editing有几种方法可以做到这一点。

关于java - 自定义 TableCellEditor 在添加时显示上一个条目,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8389689/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com