Java 在DragGestureEvent和x27之后,JTable的列不正确;s startDrag呼叫

Java 在DragGestureEvent和x27之后,JTable的列不正确;s startDrag呼叫,java,swing,drag-and-drop,jtable,jtableheader,Java,Swing,Drag And Drop,Jtable,Jtableheader,我已经为JTableHeader实现了拖放功能,因此当用户单击按下Ctrl键的表的标题而不是对列重新排序时,将传输自定义对象 它看起来工作正常,但当调用DraggestureeEvent的startDrag方法时,被拖动的列会发生偏移(请参见第4列和第5列之间的间隙): 如果在此之后单击列或在不使用Ctrl键的情况下拖动它,它将返回到正确的位置。有人知道如何让柱子保持它的位置吗 源代码: public class Tree { private final String name;

我已经为JTableHeader实现了拖放功能,因此当用户单击按下Ctrl键的表的标题而不是对列重新排序时,将传输自定义对象

它看起来工作正常,但当调用DraggestureeEvent的startDrag方法时,被拖动的列会发生偏移(请参见第4列和第5列之间的间隙):

如果在此之后单击列或在不使用Ctrl键的情况下拖动它,它将返回到正确的位置。有人知道如何让柱子保持它的位置吗

源代码:

public class Tree {
    private final String name;

    public Tree (final String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    @Override
    public String toString() {
        return "Tree[" + name + "]";
    };
}
这是:

import java.awt.BorderLayout;
import java.awt.Cursor;
import java.awt.Dimension;
import java.awt.GridLayout;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.awt.dnd.DnDConstants;
import java.awt.dnd.DragGestureEvent;
import java.awt.dnd.DragGestureListener;
import java.awt.dnd.DragSource;
import java.awt.event.ActionEvent;
import java.awt.event.MouseEvent;
import java.io.IOException;

import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.TransferHandler;
import javax.swing.WindowConstants;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.JTableHeader;
import javax.swing.table.TableColumnModel;

public class DnDTest extends JPanel {

    private static final long serialVersionUID = 8341231933303004329L;

    private static final DataFlavor FLAVOR = new DataFlavor(Tree.class,
            Tree.class.getSimpleName());

    private final TableTransferHandler<Integer> customHandler = new TableTransferHandler<Integer>();

    /* ************** CONSTRUCTORS *************** */

    public DnDTest() {
        setLayout(new GridLayout(1, 2));
        final JTable table = createTable();
        table.setTableHeader(new CustomTableHeader(table.getColumnModel()));

        final JPanel right = new JPanel(new BorderLayout());
        final JLabel dropLabel = new JLabel();
        dropLabel.setTransferHandler(customHandler);
        right.add(dropLabel);
        this.add(right);

        new DragSource().createDefaultDragGestureRecognizer(
                table.getTableHeader(), DnDConstants.ACTION_COPY,
                new CustomDragGesture());
    }

    /* ************* PRIVATE METHODS ************* */

    private JTable createTable() {
        final DefaultTableModel model = createTableModel();
        final JTable table = new JTable(model);
        this.add(new JScrollPane(table));
        return table;
    }

    private DefaultTableModel createTableModel() {
        final String[] columnNames = { "Column1", //$NON-NLS-1$
                "Column2", //$NON-NLS-1$
                "Column3", //$NON-NLS-1$
                "Column4", //$NON-NLS-1$
                "Column5" //$NON-NLS-1$
        };
        final Integer[][] data = new Integer[1][columnNames.length];
        for (int i = 0; i < data.length; i++) {
            for (int j = 0; j < data[i].length; j++) {
                data[i][j] = (int) (Math.random() * 30);
            }
        }

        final DefaultTableModel model = new DefaultTableModel(data, columnNames);
        return model;
    }

    /* ***************** CLASSES ***************** */

    private static final class CustomTableHeader extends JTableHeader {

        private static final long serialVersionUID = -781646896813653894L;

        public CustomTableHeader(final TableColumnModel columnModel) {
            super(columnModel);
        }

        public Tree getTree() {
            return new Tree("Test tree");
        }
    }

    private static final class CustomDragGesture implements DragGestureListener {

        @Override
        public void dragGestureRecognized(DragGestureEvent event) {
            if (!(event.getTriggerEvent() instanceof MouseEvent && SwingUtilities
                    .isLeftMouseButton((MouseEvent) event.getTriggerEvent()))
                    || (event.getTriggerEvent().getModifiers() & ActionEvent.CTRL_MASK) != ActionEvent.CTRL_MASK)
                return;

            final CustomTableHeader tableHeader = (CustomTableHeader) event
                    .getComponent();

            Cursor cursor = null;
            if (event.getDragAction() == DnDConstants.ACTION_COPY) {
                cursor = DragSource.DefaultCopyDrop;
            }
            final Tree tree = tableHeader.getTree();

            event.startDrag(cursor, new TransferableTree(tree));
        }
    }

    private static final class TransferableTree extends Tree implements
            Transferable {

        private final Tree tree;

        public TransferableTree(final Tree tree) {
            super(tree.getName());
            this.tree = tree;
        }

        @Override
        public DataFlavor[] getTransferDataFlavors() {
            return new DataFlavor[] { FLAVOR };
        }

        @Override
        public boolean isDataFlavorSupported(DataFlavor flavor) {
            return FLAVOR.equals(flavor);
        }

        @Override
        public Object getTransferData(DataFlavor flavor)
                throws UnsupportedFlavorException, IOException {
            if (FLAVOR.equals(flavor)) {
                return tree;
            }
            throw new UnsupportedFlavorException(flavor);
        }

    }

    private static final class TableTransferHandler<T extends Number & Comparable<T>>
            extends TransferHandler {

        private static final long serialVersionUID = -1794178812912767711L;

        @Override
        public boolean importData(JComponent comp, Transferable t) {
            if (hasTreeFlavor(t.getTransferDataFlavors())) {
                try {
                    final Tree tree = (Tree) t.getTransferData(FLAVOR);
                    if (comp instanceof JLabel) {
                        final JLabel label = (JLabel) comp;
                        label.setText(label.getText() + tree.toString() + ", ");
                    }
                } catch (UnsupportedFlavorException | IOException e) {
                    e.printStackTrace();
                }
            }
            return false;
        }

        @Override
        public boolean canImport(JComponent comp, DataFlavor[] transferFlavors) {
            return hasTreeFlavor(transferFlavors);
        }

        private boolean hasTreeFlavor(DataFlavor[] transferDataFlavors) {
            for (final DataFlavor flavor : transferDataFlavors) {
                if (FLAVOR.equals(flavor)) {
                    return true;
                }
            }
            return false;
        }

    }

    public static void main(String[] args) {
        final JFrame frame = new JFrame("Drag And Drop Test");
        frame.add(new DnDTest());
        frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        frame.setPreferredSize(new Dimension(1000, 800));
        frame.pack();
        frame.setVisible(true);
    }
}
导入java.awt.BorderLayout;
导入java.awt.Cursor;
导入java.awt.Dimension;
导入java.awt.GridLayout;
导入java.awt.datatransfer.DataFlavor;
导入java.awt.datatransfer.transfer;
导入java.awt.datatransfer.UnsupportedFlavorException;
导入java.awt.dnd.DnDConstants;
导入java.awt.dnd.DragGestureEvent;
导入java.awt.dnd.DragGestureListener;
导入java.awt.dnd.DragSource;
导入java.awt.event.ActionEvent;
导入java.awt.event.MouseEvent;
导入java.io.IOException;
导入javax.swing.JComponent;
导入javax.swing.JFrame;
导入javax.swing.JLabel;
导入javax.swing.JPanel;
导入javax.swing.JScrollPane;
导入javax.swing.JTable;
导入javax.swing.SwingUtilities;
导入javax.swing.TransferHandler;
导入javax.swing.WindowConstants;
导入javax.swing.table.DefaultTableModel;
导入javax.swing.table.JTableHeader;
导入javax.swing.table.TableColumnModel;
公共类DnDTest扩展了JPanel{
私有静态最终长serialVersionUID=8341231933303004329L;
私有静态最终DataFlavor=新DataFlavor(Tree.class,
Tree.class.getSimpleName());
私有最终TableTransferHandler customHandler=新TableTransferHandler();
/*****************施工人员********************/
公共DnDTest(){
setLayout(新的GridLayout(1,2));
最终JTable table=createTable();
table.setTableHeader(新的CustomTableHeader(table.getColumnModel());
final JPanel right=new JPanel(new BorderLayout());
最终JLabel dropLabel=新JLabel();
setTransferHandler(customHandler);
右。添加(dropLabel);
这个.加上(右),;
新建DragSource().CreateDefaultDraggestureRecogener(
table.getTableHeader(),DnDConstants.ACTION\u COPY,
新CustomDragesture());
}
/**************私有方法*****************/
私有JTable createTable(){
最终DefaultTableModel=createTableModel();
最终JTable表格=新JTable(模型);
添加(新的JScrollPane(表));
返回表;
}
私有DefaultTableModel createTableModel(){
最终字符串[]columnNames={“Column1”,//$NON-NLS-1$
“第2栏,//$NON-NLS-1$
“Column3”,//$NON-NLS-1$
“Column4”,//$NON-NLS-1$
“第5列”//$NON-NLS-1$
};
最终整数[][]数据=新整数[1][columnNames.length];
对于(int i=0;i