问题 JTable - 拖放


好吧,这个问题超出了我的联盟。我正在尝试在swing中实现一个GUI小部件,允许将文件放到JTable上,并允许拖动JTable的行进行重新排序。想想VLC的播放列表或iTunes中的播放列表。

我从操作系统(Explorer,Finder等)中删除文件工作得很好,但是一旦文件进入,我就不可能重新排列表的行。问题是当我添加时拖动自定义TransferHandler到表  桌子立即被杀死。这是一些示例代码:

import javax.swing.*;

public class TableTest
{
    public static void main (String [] argv)
    {
        // setup table data
        String [] columns = new String [] {"Foo", "Bar", "Baz", "Quux"};
        String [][] data = new String [][] {{"A", "B", "C", "D"},
                        {"1", "2", "3", "4"},
                        {"i", "ii", "iii", "iv"}};
        // create table
        JTable table = new JTable(data, columns);

        // set up drag and drop
        table.setDragEnabled(true);
        table.setDropMode(DropMode.INSERT_ROWS);
        table.setFillsViewportHeight(true);
        TransferHandler dnd = new TransferHandler() {
            // here be code to handle drops, and one would
            // presume drag exporting, too
        };
        table.setTransferHandler(dnd);
        JScrollPane scroll = new JScrollPane(table);

        // create and show window
        JFrame window = new JFrame();
        window.getContentPane().add(scroll);
        window.pack();
        window.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        window.setVisible(true);
    }
}

按原样运行此代码,您将看到无法启动对表的拖动。如果您在表上注释掉对setTransferHandler()的调用,则拖动工作(即,当我开始拖动表行时,我得到X'd出圆圈光标说我不能放弃那里)。但是只要为表设置了TransferHandler,我就无法拖动任何行。问题必须在TransferHandler中,但是我彻底麻烦并调试它,并确定一旦桌面上有TransferHandler就永远不会启动拖动。我究竟做错了什么?


2401
2018-04-03 19:26


起源

有多奇怪我遇到了同样的问题。我想要记住,如果我做了任何工作,我有一个支持拖动和拖动的桌子。 - willcodejavaforfood


答案:


我遇到了同样的问题,它与您自定义的TransferHandler实现无关。当您替换TransferHandler时,您还需要保留默认的DragSource并告诉它识别拖动手势。您可能还需要实现自己的Transferable,因为您需要将它传递给DragGestureEvent.startDrag()方法。

    table.setTransferHandler(new MyTransferHandler());
    table.setDragEnabled(true);
    DragSource source = DragSource.getDefaultDragSource();
    source.createDefaultDragGestureRecognizer(table, DnDConstants.ACTION_COPY, new DragGestureListener() {

        @Override
        public void dragGestureRecognized(DragGestureEvent dge) {
            //grab the selected files from the table model
            ArrayList<File> files = new ArrayList<File>();
            for (int row : table.getSelectedRows()) {
                files.add((File) dm.getValueAt(row, 1));
            }

            //FileTransferable is a custom Transferable implementation
            Transferable transferable = new FileTransferable(files); 

            //and this is the magic right here
            dge.startDrag(null,transferable);
        }
    });

5
2018-03-03 23:35



如果我遗漏前两行,我就会为我工作 - setTransferHandler 和 setDragEnabled。实际上设置拖动已启用 InvalidDnDOperationException: Drag and drop in progress 为了我。我也认为 dm 是指 table.getModel()。 - brian_d
工作完美无瑕。和我前面的海报一样,省略前两行。 - user1052080


它看起来不像你正在使用TransferHandler。尝试阅读教程 这里

请参阅TransferHandler文档 这里。空构造函数看起来不适合在TransferHandler的子类之外使用。

并且您没有实现Swing组件上提供的标准TransferHandler中提供的任何功能。请参阅DnD教程中的exerpt 这里 (我的大胆):

注意:如果将自定义TransferHandler安装到Swing组件上,则会替换默认支持。例如,如果将JTextField的TransferHandler替换为仅处理颜色的TransferHandler,则将禁用其支持文本导入和导出的功能。    如果必须替换默认的TransferHandler(例如,处理文本的TransferHandler),则需要重新实现文本导入和导出功能。这不需要像Swing提供的那样广泛 - 它可以像支持StringFlavor数据风格一样简单,具体取决于您的应用程序的需求。


3
2018-04-07 19:16





我认为问题是空的TransferHandler实际上阻止了DnD事件的发生。这里有一个可能相关的样本。

http://www.java2s.com/Code/Java/Swing-JFC/ExtendedDnDDragandDropDemo.htm


3
2018-04-07 19:26



谢谢。这个例子很棒.. :) - Amarnath


我不想深入了解正在发生的事情,所以我只是委托我对旧的TransferHandler不感兴趣的方法。

tree.setDragEnabled(true);
tree.setDropMode(DropMode.XXXX);
tree.setTransferHandler(new MyTransferHandler(tree.getTransferHandler());

从标准设置开始,但将旧的TransferHandler传递到您的自定义TransferHandler。

private class MyTransferHandler extends TransferHandler {
  private TransferHandler delegate;

  public MyTransferHandler(TransferHandler delegate) {
    this.delegate = delegate;
  }

  public boolean canImport(JComponent comp, DataFlavor[] transferFlavors) {
    return delegate.canImport(comp, transferFlavors);
  }

  public boolean canImport(TransferSupport support) {
    return true;
  }

  protected Transferable createTransferable(JComponent c) {
    try {
      Method method = delegate.getClass().getDeclaredMethod("createTransferable", JComponent.class);
      method.setAccessible(true);
      return (Transferable) method.invoke(delegate, c);
    } catch (Exception e) {
      return super.createTransferable(c);
    }
  }

  public void exportAsDrag(JComponent comp, InputEvent event, int action) {
    delegate.exportAsDrag(comp, event, action);
  }

  protected void exportDone(JComponent source, Transferable data, int action) {
    try {
      Method method = delegate.getClass().getDeclaredMethod("exportDone", JComponent.class, Transferable.class,
          int.class);
      method.setAccessible(true);
      method.invoke(delegate, source, data, action);
    } catch (Exception e) {
      super.exportDone(source, data, action);
    }
  }

  public int getSourceActions(JComponent c) {
    return delegate.getSourceActions(c);
  }

  public Icon getVisualRepresentation(Transferable t) {
    return delegate.getVisualRepresentation(t);
  }

  public boolean importData(JComponent comp, Transferable t) {
    return delegate.importData(comp, t);
  }

  public boolean importData(TransferHandler.TransferSupport support) {
    return delegate.importData(support);
  }
}

一个问题是createTransferable(JComponent)和exportDone(JComponent,Transferable,int)方法受到保护,因此您需要进行反射才能委派给这些方法。当我没有做这个反思代表时,策略不起作用。一旦我执行了这个委托,拖放工作按预期工作而不更改DragSource或必须编写新的Transferable。


1
2017-12-22 13:29



草率的示例代码。 boolean返回类型canImport不返回任何返回类型exportDone返回的类型。 - brian_d
这为什么这么草率?在这个问题中,他明确表示他正在尝试将TransferHandler添加到表中以支持丢弃。他的整个问题是,当他这样做时,桌子的阻力支撑停止工作。封面上发生的事情是,当他提供自己的TransferHandler时,他会破坏桌面上安装的默认TransferHandler。这是如何从表中委托原始TransferHandler的示例,秘密是使用反射来实现createTransferable()和exportDone()。其他方法可以随意覆盖或委派。 - rancidfishbreath