为什么JXTable会丢失输入而JTable不会?

8
当我使用JXTable来渲染和编辑我的数据时,一些输入到CellEditors中的内容会丢失。如果我点击JXTable-ColumnHeader的调整大小分隔符或改变JFrame的宽度,则CellEditor会在不提交值的情况下终止。如果我使用JTable,则这些值将被保存。
我想使用JXTable,因为它具有其他功能,所以有没有办法修复JXTable?
示例:
package table.columnresize;

import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.table.DefaultTableModel;

import org.jdesktop.swingx.JXTable;

/**
 * Demo of differing behaviour of JXTable and JTable. JXTable loses input in a TableCell where JTable persists
 * it.
 * <p>
 * <table border=1>
 * <tr>
 * <th></th>
 * <th>JXTable</th>
 * <th>JTable</th>
 * </tr>
 * <tr>
 * <td>Click on TableColumnHeader</td>
 * <td>saved</td>
 * <td>saved</td>
 * </tr>
 * <tr>
 * <td>Resizing with Divider of TableColumnHeader</td>
 * <td>lost</td>
 * <td>saved</td>
 * </tr>
 * <tr>
 * <td>Changing the width of JFrame</td>
 * <td>lost</td>
 * <td>saved</td>
 * </tr>
 * 
 * </table>
 * </p>
 * 
 * @author bobndrew 2015-01-29
 */
public class JXTableAndJTableEditLossDemo
{
  private static class DataModel extends DefaultTableModel
  {
    public DataModel( Object[][] data, Object[] columnNames )
    {
      super( data, columnNames );
    }
  }

  private static void createAndShowUI()
  {
    Object[][] DATA = { { "One", 1 }, { "Two", 2 }, { "Three", 3 }, { "Four", 4 }, { "Five", 5 } };
    String[] COLUMNS = { "A", "B" };
    DataModel dataModel = new DataModel( DATA, COLUMNS );

    JFrame frame1 = new JFrame( "JXTable" );
    JXTable jXTable = new JXTable( dataModel );
    //does not change anything:    jXTable.setTerminateEditOnFocusLost( true );
    System.out.println( jXTable.isTerminateEditOnFocusLost() );
    frame1.add( new JScrollPane( jXTable ) );
    frame1.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
    frame1.pack();
    frame1.setVisible( true );

    JFrame frame2 = new JFrame( "JTable" );
    JTable jTable = new JTable( dataModel );
    //does not change anything:    jTable.putClientProperty( "terminateEditOnFocusLost", Boolean.FALSE );
    System.out.println( jTable.getClientProperty( "terminateEditOnFocusLost" ) );
    frame2.add( new JScrollPane( jTable ) );
    frame2.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
    frame2.pack();
    frame2.setLocation( (int) frame1.getLocation().getX() + frame1.getWidth() + 100, (int) frame1
        .getLocation().getY() );
    frame2.setVisible( true );
  }

  public static void main( String[] args )
  {
    java.awt.EventQueue.invokeLater( new Runnable()
    {
      @Override
      public void run()
      {
        createAndShowUI();
      }
    } );
  }

}

你尝试过为JXTable添加自定义焦点监听器吗?请参考此帖子的第二和第三个答案。 https://dev59.com/ZHI-5IYBdhLWcg3w0cOG - a_pradhan
好主意,但是在不同的操作系统下,FocusListener太不可靠了。 - bobndrew
2个回答

3
在调试 JXTable 和 JTable 时,我发现 CellEdits 丢失的原因。区别在于方法 columnMarginChanged():
JXTable:
if (isEditing()) {
  removeEditor();
}

JTable:

if (isEditing() && !getCellEditor().stopCellEditing()) {
  getCellEditor().cancelCellEditing();
} 

起初,我认为 removeEditor()方法是JTable的增强功能...但后来我发现这个2010年9月OpenJDK的更改集修复了错误"4330950: 调整列宽时丢失新输入的单元格数据"。看起来JDK的更改未应用于SwingX源代码。
我会接受自己的答案,因为不同行为的原因现在已经清楚。为了解决我和其他SwingX用户的问题,我将前往SwingX邮件列表和错误跟踪器。

1
当您查看JTable的frameInit()方法时,可以看到它绑定到所有AWTEvent.WINDOW*事件。在JXTable中,initActionsAndBindings()方法绑定到特定操作(如值更改),仅适用于表格。
您需要添加自己的监听器。
    jXTable.getColumnModel().addColumnModelListener(new TableColumnModelListener() {
        @Override
        public void columnMarginChanged(ChangeEvent e) {

        }
    });

然后需要公开表的某些功能,以允许事件触发更新表。或者你可以从那里触发一个TableModelEvent。

frameInit()JFrame.java 中的一个方法,而不是 JTable.java。但是你使用 TableColumnModelListenercolumnMarginChanged() 的想法与我的发现相似;请参见我的回答。 - bobndrew

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接