JTable 在选择后将 JComboBox 和 JCheckBox 恢复为值

JTable Reverts JComboBox and JCheckBox to Values after Selection

我有一个包含 JComboBox<Integer> 列和 JCheckBox 列的 JTable。 JTable 设置有适当的渲染器和编辑器。 table 一开始看起来不错,但从组合框或复选框中选择一个值后,单元格似乎恢复为整数和布尔值。这个问题似乎不仅仅是装饰性的,因为预期单元格具有组合框或复选框的方法在查找整数或布尔值时会抛出错误。

这是它的外观图片:

这是代码:

    dataTable.removeAll();
    numberOfVariables = 7;
    Object[] header = new Object[numberOfVariables];
    header[0] = new String("Ring Number");
    header[1] = new String("Radius (cm)");
    header[2] = new String("Plume Distribution");
    header[3] = new String("Thickness (A)");
    header[4] = new String("Deposition Time (s)");
    header[5] = new String("Rate (A/s)");
    header[6] = new String("Optimize (y/n)");

    Object[][] data = new Object[numberOfRings][numberOfVariables];
    for(int k=0;k<numberOfRings;++k){
        Object[] row = new Object[numberOfVariables];
        row[0] = Integer.toString(k+1);
        row[1] = String.format("%.2f", plume.getRingRadius(k));
        row[2] = createDistributionComboBoxForRing(k);
        row[3] = String.format("%.2f", plume.getThicknessOfRing(k));
        row[4] = String.format("%.2f", plume.getTimeForRing(k));
        row[5] = String.format("%.2f", plume.getRateForRing(k));
        row[6] = new JCheckBox();
        ((JCheckBox) row[6]).setSelected(true);
        data[k] = row;
    }
    tableModel = new DefaultTableModel(data,header);
    dataTable.setModel(tableModel);
    dataTable.getColumnModel().getColumn(2).setCellRenderer(new ControlTableRenderer());
    //dataTable.getColumnModel().getColumn(2).setCellEditor(new DefaultCellEditor( createDistributionComboBoxForRing(0) ));
    dataTable.getColumnModel().getColumn(2).setCellEditor(new DefaultCellEditor( (JComboBox<Integer>) data[0][2] ));
    dataTable.getColumnModel().getColumn(6).setCellRenderer(new ControlTableRenderer());
    dataTable.getColumnModel().getColumn(6).setCellEditor(new DefaultCellEditor( (JCheckBox) data[0][6] ));
    dataTable.updateUI();

单元格渲染器和编辑器的想法是您只有一个组件,它会四处移动并且每一行的数据都会更改。该代码为每个单元格使用不同的 JComponent 实例。只需将数据放在 table 模型中,让单元格渲染器和编辑器管理组件。

row[2] = createDistributionComboBoxForRing(k);
row[3] = String.format("%.2f", plume.getThicknessOfRing(k));
row[4] = String.format("%.2f", plume.getTimeForRing(k));
row[5] = String.format("%.2f", plume.getRateForRing(k));
row[6] = new JCheckBox();
((JCheckBox) row[6]).setSelected(true);

JTable 的 TableModel 存储数据,而不是组件。

因此,如果第 3 列包含一个 Integer 对象,那么您将 Integer 存储在 TableModel 中,并将该列的编辑器设置为包含以下列表的组合框有效整数。

包含复选框 renderer/editor 的列也是如此。在这种情况下,您存储一个 Boolean 对象。

例如:

row[2] = new Integer(1);
row[6] = Boolean.TRUE

现在在 TableModel 中,您需要告诉 table 每列中的数据类型,因此您需要重写 getColumnClass(...) 方法。类似于:

tableModel = new DefaultTableModel(data,header)
{
    @Override
    public Class getColumnClass(int column)
    {
        switch (column)
        {
            case 2: return Integer.class;
            case 6: return Boolean.class;
            default: return Object.class;
        }
    }
};

现在 table 可以为每一列选择合适的渲染器和编辑器。

但是,对于组合框,您必须使用组合框的值创建自定义编辑器。有关如何执行此操作的工作示例,请参阅 Using a Combo Box as an Editor 上的 Swing 教程部分。

此外,您还错误地使用了其他一些方法:

dataTable.removeAll();

不确定那是干什么用的。这是一个从面板中删除组件的容器方法。您只需要 setModel(...) 语句来重置 table.

dataTable.updateUI();

不需要使用updateUI()方法。当 LAF 更改时,该方法在内部使用。您没有更改 LAF,因此删除该声明。