JTable总是无法为内部ComboBox获取正确的rowIndex

时间:2013-09-24 07:31:01

标签: java swing jtable jcombobox

我有一个JTable,其中一个列单元格为JComboBox。 但是当单击表JComboBox单元格时尝试获取行计数时,我发现行索引始终返回错误值(总是最后一次单击行索引)。

public class TableComboBoxTest extends JFrame {

    private JTable table;
    private DefaultTableModel tableModel;
    private Object[][] tableCells;
    private final String[] TABLE_COLUMNS = {"No.1"};
    private final String[] YES_NO_SELECTION = {"Yes", "No"};

    public TableComboBoxTest() {
        Container pane = getContentPane();
        pane.setLayout(new BorderLayout());
        tableModel = new DefaultTableModel(tableCells, TABLE_COLUMNS);

        table = new JTable(tableModel);

        DefaultCellEditor cellEditor = null;

        JComboBox selA = new JComboBox(YES_NO_SELECTION);
        cellEditor = new DefaultCellEditor(selA);
        cellEditor.setClickCountToStart(1);
        table.getColumn(TABLE_COLUMNS[0]).setCellEditor(cellEditor);

        JScrollPane jsp = new JScrollPane();
        jsp.getViewport().add(table, null);
        pane.add(jsp, BorderLayout.CENTER);

        TableCellEditor tce = null;
        addRow("Yes");
        outputDefaultSelection(0, 0);

        addRow("No");
        outputDefaultSelection(1, 0);

        System.out.println("");

        selA.addItemListener(new ItemListener() {
            public void itemStateChanged(ItemEvent e) {
                if (e.getStateChange() == ItemEvent.SELECTED) {
                    JComboBox cb = (JComboBox) e.getSource();
                    String sel = (String) cb.getSelectedItem();
                    int rowIndex = table.getSelectedRow();
                    rowIndex = table.convertRowIndexToModel(rowIndex);

                    if (rowIndex == -1) {
                        return;
                    }

                    outputDefaultSelection(rowIndex, 0);
                    System.out.println("Select: " + sel + " at " + rowIndex);
                }
            }
        });
    }

    private void addRow(String v1) {
        Vector<String> vec = new Vector<String>();
        vec.add(v1);

        tableModel.addRow(vec);
        tableModel.fireTableDataChanged();
    }

    private void outputDefaultSelection(int row, int col) {
        TableCellEditor tce = table.getCellEditor(row, col);
        System.out.println("Default " + row + "-" + col + " Selection: " + tce.getCellEditorValue());
        System.out.println("Default " + row + "-" + col + " Value: " + table.getModel().getValueAt(row, col));
    }

    public static void main(String[] args) {
        TableComboBoxTest stt = new TableComboBoxTest();
        stt.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        stt.setSize(200, 100);
        stt.setVisible(true);
    }
}
Default 0-0 Selection: Yes
Default 0-0 Value: Yes
Default 1-0 Selection: Yes
Default 1-0 Value: No*

单击第一行并选择“是”时,不会发生更改事件触发器。 单击第二行时,更改事件触发器!并且行号错误:0

Default 0-0 Selection: No
Default 0-0 Value: Yes
Select: No at 0*

当继续点击第一行时,更改事件触发器!并且行号错误:1

Default 1-0 Selection: Yes
Default 1-0 Value: No
Select: Yes at 1

如何才能获得正确的点击单元格号?

对于itemStateChanged进程,我还发现单元格设置值是否与默认列值相同(“是”),单击时它不会触发事件。但是如果单元格设置为“否”,则单击它将导致更改事件。这意味着模型数据与默认选择的数据不同。如何使它们保持一致?

感谢〜

1 个答案:

答案 0 :(得分:2)

  

这意味着模型数据与默认选择的数据不同。如何使它们保持一致?

这只表示尚未使用组合框中新选择的值更新模型。

这可以通过使用以下内容来证明:

final String sel = (String) cb.getSelectedItem();
final int rowIndex = table.convertRowIndexToModel(table.getSelectedRow());

if (rowIndex == -1) {
    return;
}

SwingUtilities.invokeLater(new Runnable()
{
    public void run()
    {
        outputDefaultSelection(rowIndex, 0);
        System.out.println("Select: " + sel + " at " + rowIndex);
    }
});

现在,显示代码将添加到Event Dispatch Thread的末尾,这意味着它将在所有其他事件执行完毕后执行,因此现在将更新TableModel。

但是,这不是最好的解决方案。如果您想知道单元格中的数据何时更改,请将TableModelListener添加到TableModel。 不要使用ItemListener。