Как выбрать весь текст в ячейке JTable при редактировании, но не при наборе текста?

Поведение JTable по умолчанию заключается в добавлении к содержимому, когда вы начинаете вводить текст, и поместить каретку в щелканное место при нажатии. Я хочу, чтобы поведение обеих этих вещей изменилось, поэтому содержимое заменяется, когда я редактирую ячейку, либо набрав, либо щелкнув, а затем набрав. Однако, когда я нажимаю на ячейку, а затем меняю позицию каретки, я хочу, чтобы содержимое оставалось таким, чтобы я мог его изменить.

Я знаю, как выбрать все, когда ячейка будет редактироваться, заменив редактор ячейки тем, который выбирает все внутри SwingUtilities.invokeLater(см. в другом месте), но что приводит к нарушению типизации. Когда я это сделаю и начну вводить в ячейку, сначала введенный символ добавляется к строке, затем выбирается (но выбор невидим!), И при вводе другого символа содержимое заменяется на это.

Есть ли способ немедленно заменить содержимое при вводе выделенной (но не редактируемой) ячейки, но выбрать все, щелкнув ячейку?

Вот код, который я использую для CellEditor:

public class TextFieldCellEditor extends JTextField implements TableCellEditor
{
    private CellEditorListener  cellEditorListener  = null;

    private boolean             isInteger           = false;
    private Object              oldValue;

    // Start editing
    @Override
    public Component getTableCellEditorComponent(JTable table, Object obj, boolean isSelected, int row, int column)
    {
        Color color2 = DefaultLookup.getColor(this, ui, "Table.alternateRowColor");
        super.setBackground(color2 != null && (row & 1) == 1? color2 : table.getBackground());
        super.setForeground(table.getForeground());
        super.setBorder(DefaultLookup.getBorder(this, ui, "Table.focusCellHighlightBorder"));

        super.setText(obj.toString());

        isInteger = obj instanceof Integer;
        if (isInteger)
        {
            super.setHorizontalAlignment(SwingConstants.RIGHT);
            oldValue = obj;
        }

        // SwingUtilities.invokeLater(new Runnable()
        // {
        // public void run()
        // {
        // TextFieldCellEditor.this.selectAll();
        // }
        // });

        return this;
    }

    // Retrieve e dited value
    @Override
    public Object getCellEditorValue()
    {
        if (isInteger)
        {
            // Try to convert to integer. If input is invalid, revert.
            try
            {
                return new Integer(super.getText());
            }
            catch (NumberFormatException e)
            {
                return oldValue;
            }
        }
        return super.getText();
    }

    @Override
    public boolean isCellEditable(EventObject e)
    {
        return true;
    }

    @Override
    public boolean shouldSelectCell(EventObject e)
    {
        return true;
    }

    @Override
    public boolean stopCellEditing()
    {
        cellEditorListener.editingStopped(new ChangeEvent(this));
        return true;
    }

    @Override
    public void cancelCellEditing()
    {
        cellEditorListener.editingCanceled(new ChangeEvent(this));
    }

    @Override
    public void addCellEditorListener(CellEditorListener celleditorlistener)
    {
        cellEditorListener = celleditorlistener;
    }

    @Override
    public void removeCellEditorListener(CellEditorListener celleditorlistener)
    {
        if (cellEditorListener == cellEditorListener) cellEditorListener = null;
    }
}

Ответ 1

В вашей реализации getTableCellEditorComponent() добавьте следующее:

if (isSelected) {
    this.selectAll();
}

В стороне, почему бы не расширить AbstractCellEditor или DefaultCellEditor(JTextField textField)? См. Также Как использовать таблицы: использование других редакторов.

Приложение: См. также Таблица Выбрать все рендеринг и Таблица Выбрать все Редактор.

Ответ 2

Самое чистое решение, которое я смог найти для этого случая, это перезаписать JTable editCellAt и сообщить CellEditor о том, как было вызвано редактирование:

@Override
public boolean editCellAt(int row, int column, EventObject e) {
    cellEditor.setKeyTriggered(e instanceof KeyEvent);
    return super.editCellAt(row, column, e);
}

И вот соответствующий код CellEditor:

public class MyCellEditor extends DefaultCellEditor {

    private boolean keyTriggered;

    public MyCellEditor() {
        super(new JTextField());
        final JTextField textField = (JTextField) getComponent();
        textField.addFocusListener(new FocusAdapter() {
            @Override
            public void focusGained(FocusEvent e) {
                SwingUtilities.invokeLater(new Runnable() {
                    @Override
                    public void run() {
                        if (!keyTriggered) {
                            textField.selectAll();
                        }
                    }
                });
            }
        });
    }

    public void setKeyTriggered(boolean keyTriggered) {
        this.keyTriggered = keyTriggered;
    }

    @Override
    public Component getTableCellEditorComponent(
            JTable table, Object value, boolean isSelected, int row, int column) {
        final JTextField textField = (JTextField)
                super.getTableCellEditorComponent(table, value, isSelected, row, column);
        textField.selectAll();
        return textField;
    }
}