【发布时间】:2010-11-27 14:08:13
【问题描述】:
我正在尝试制作一个像 Excel 一样工作的表格。意思是,当用户开始将数据插入单元格时,插入的新数据会选择并更改其中的内容。
【问题讨论】:
标签: java swing jtable tablecelleditor
我正在尝试制作一个像 Excel 一样工作的表格。意思是,当用户开始将数据插入单元格时,插入的新数据会选择并更改其中的内容。
【问题讨论】:
标签: java swing jtable tablecelleditor
您可以为您的表格创建自定义 TableCellEditor。这个类将有一个TextField 的实例变量,我们称之为textField。那么getTableCellEditorComponent 方法可能如下所示:
public Component getTableCellEditorComponent(JTable table, Object value,
boolean isSelected, int row, int column ) {
textField.setText(value.toString());
textField.selectAll();
return textField;
}
【讨论】:
如果您在表中只有字符串数据并且只需要一个编辑器,那么创建自定义编辑器就可以正常工作。但是,如果您有多种不同类型的数据,例如 String、Integer、Double、货币、百分比等都使用 JTextField 作为编辑器,那么您需要创建多个自定义编辑器。
您可以阅读Table Select All Editor 以了解另一种可能的解决方案。
【讨论】:
请注意,还有另一种可能性,您可以像下面这样覆盖 JTable#prepareEditor:
@Override
public Component prepareEditor(TableCellEditor editor, int row, int column) {
Component c = super.prepareEditor(editor, row, column);
if (c instanceof JTextComponent) {
((JTextComponent) c).selectAll();
}
return c;
}
【讨论】:
当通过鼠标单击开始编辑时,上述解决方案不起作用。
对于某些人来说,解决方案是在 invokeLater() 中调用 selectAll() 以便在分派鼠标事件后选择文本,但这对我不起作用(可能是因为我正在使用 Substance 外观和感觉?)
Swing 内部稍后会收到一个 mouseReleased() 事件并再次更改插入符号,如下堆栈跟踪所示:
at javax.swing.text.JTextComponent.fireCaretUpdate(Unknown Source)
at javax.swing.text.JTextComponent$MutableCaretEvent.fire(Unknown Source)
at javax.swing.text.JTextComponent$MutableCaretEvent.mouseReleased(Unknown Source)
at java.awt.AWTEventMulticaster.mouseReleased(Unknown Source)
at java.awt.AWTEventMulticaster.mouseReleased(Unknown Source)
at java.awt.AWTEventMulticaster.mouseReleased(Unknown Source)
at java.awt.AWTEventMulticaster.mouseReleased(Unknown Source)
at java.awt.AWTEventMulticaster.mouseReleased(Unknown Source)
at java.awt.Component.processMouseEvent(Unknown Source)
at javax.swing.JComponent.processMouseEvent(Unknown Source)
at java.awt.Component.processEvent(Unknown Source)
at java.awt.Container.processEvent(Unknown Source)
at java.awt.Component.dispatchEventImpl(Unknown Source)
at java.awt.Container.dispatchEventImpl(Unknown Source)
at java.awt.Component.dispatchEvent(Unknown Source)
at javax.swing.plaf.basic.BasicTableUI$Handler.repostEvent(Unknown Source)
这是我的解决方案: 监听插入符号位置的变化,在单元格编辑开始后第一次选择从所有选择变为无选择时,再次调用 selectAll()。插入符号侦听器可以通过此处所示的自定义单元格编辑器安装,也可以在自定义 JTable 中的重写 editCellAt() 方法中安装。
private class SelectAllCellEditor extends DefaultCellEditor
{
public SelectAllCellEditor( JTextField tf )
{
super( tf );
}
/**
* Flag to ensure we only install the caret listener on the editor once.
*/
boolean listenerInstalled = false;
/**
* Primes the caret listener to override deselection when the first mouseReleased() event is reposted to the editor.
*/
boolean overrideDeselection = false;
@Override
public Component getTableCellEditorComponent( JTable table , Object value , boolean isSelected , int row , int column )
{
final JFormattedTextField tf = ( JFormattedTextField ) super.getTableCellEditorComponent( table , value , isSelected , row , column );
if( !listenerInstalled )
{
tf.addCaretListener( new CaretListener( )
{
int lastDot = 0;
int lastMark = 0;
@Override
public void caretUpdate( CaretEvent e )
{
if( overrideDeselection )
{
int length = tf.getText( ) == null ? 0 : tf.getText( ).length( );
boolean wasAllSelected = ( lastDot == 0 && lastMark == length ) || ( lastDot == length && lastMark == 0 );
boolean nowNoneSelected = ( e.getDot( ) == 0 && e.getMark( ) == 0 ) || ( e.getDot( ) == length && e.getMark( ) == length );
if( wasAllSelected )
{
// don't try to override again until the next time cell editing is started
overrideDeselection = false;
// only re-select all if the selection went to none; otherwise the user clicked the cell and dragged to select part of the text
if( nowNoneSelected )
{
tf.selectAll( );
}
}
}
lastDot = e.getDot( );
lastMark = e.getMark( );
}
} );
listenerInstalled = true;
}
// Prime the caret listener to override deselection when the first mouseReleased() event is reposted to the editor.
overrideDeselection = true;
tf.selectAll( );
return tf;
}
}
【讨论】:
如果您的目标是在开始编辑时清空单元格,则无需使用selectAll()。只需将值设置为null。
实现示例:
(在DefaultCellEditor 中覆盖getTableCellEditorComponent())
TableCellEditor myCellEditor = new DefaultCellEditor(new JTextField()){
@Override
public Component getTableCellEditorComponent(JTable table, Object value,
boolean isSelected, int row, int column)
{
// empty the cell on edit start
delegate.setValue( (editorComponent instanceof JTextField)? null : value);
return editorComponent;
}
};
【讨论】:
你应该看看 extJS。不过,学习曲线相当陡峭..
【讨论】: