Swingでドラッグアンドドロップを使用してファイルパスを取得するにはどうすればよいですか?

StackOverflow https://stackoverflow.com/questions/811248

  •  03-07-2019
  •  | 
  •  

質問

swingアプリケーションには、使用するために選択されたファイルのファイルパスを保持するJTextFieldがあります。現在、この値を設定するために使用されるJFileChooserがあります。ただし、ユーザーがこのJTextFieldにファイルをドラッグアンドドロップし、常にJFileChooserを使用する代わりに、そのファイルのファイルパスをJTextFieldに配置できるようにしたいと思います。

これを行うにはどうすればよいですか

役に立ちましたか?

解決

まず、 Swing DragDropサポートを確認する必要があります。 。その後、異なるオペレーティングシステム用の小さなトリックがいくつかあります。処理が完了したら、drop()コールバックを処理します。このコールバックでは、TransferableのDataFlavorを確認する必要があります。

Windowsでは、 DataFlavor.isFlavorJavaFileListType()を実行し、このようなデータを取得します

List<File> dropppedFiles = (List<File>)transferable.getTransferData(DataFlavor.javaFileListFlavor)

Linux(およびおそらくSolaris)の場合、DataFlavorは少し複雑です。独自のDataFlavorを作成する必要があり、Transferableタイプは異なります

nixFileDataFlavor = new DataFlavor("text/uri-list;class=java.lang.String");
String data = (String)transferable.getTransferData(nixFileDataFlavor);
for(StringTokenizer st = new StringTokenizer(data, "\r\n"); st.hasMoreTokens();)
{
    String token = st.nextToken().trim();
    if(token.startsWith("#") || token.isEmpty())
    {
         // comment line, by RFC 2483
         continue;
    }
    try
    {
         File file = new File(new URI(token))
         // store this somewhere
    }
    catch(...)
    {
       // do something good
       ....
    }
}

他のヒント

ファイルおよびフォルダーのドラッグアンドドロップを容易にするために使用できるクラスを含むサンプルプログラムがあります。

http://www.iharder.net/current/java/filedrop/

Windows 7とUbuntu 10.10の両方でこれをテストしましたが、どちらの環境でもうまく機能するようです。

これを使用するには、次のようなコードをコードに追加します。

  JPanel  myPanel = new JPanel();
  new  FileDrop( myPanel, new FileDrop.Listener()
  {   public void  filesDropped( java.io.File[] files )
      {   
          // handle file drop
          ...
      }   // end filesDropped
  }); // end FileDrop.Listener

この比較的複雑なテーマの調査にあまり時間をかけたくなく、Windows(またはJava 7以降を使用)を使用している場合、JTextAreaでドロップされたファイルを受け入れる方法の簡単な例を次に示します。ドロップターゲット:

JTextArea myPanel = new JTextArea();
myPanel.setDropTarget(new DropTarget() {
    public synchronized void drop(DropTargetDropEvent evt) {
        try {
            evt.acceptDrop(DnDConstants.ACTION_COPY);
            List<File> droppedFiles = (List<File>)
                evt.getTransferable().getTransferData(DataFlavor.javaFileListFlavor);
            for (File file : droppedFiles) {
                // process files
            }
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
});

これは古い質問ですが、現在の回答は少し時代遅れです:

  • JDK 1.6以降、クラス「TransferHandler」は新しい(上書きされた)メソッドで使用する必要があります
  • Linuxのサポートが大幅に改善され、カスタム処理が不要になりました

これはLinux(KDE5)およびWindows 7で動作します:

final class FileDropHandler extends TransferHandler {
    @Override
    public boolean canImport(TransferHandler.TransferSupport support) {
        for (DataFlavor flavor : support.getDataFlavors()) {
            if (flavor.isFlavorJavaFileListType()) {
                return true;
            }
        }
        return false;
    }

    @Override
    @SuppressWarnings("unchecked")
    public boolean importData(TransferHandler.TransferSupport support) {
        if (!this.canImport(support))
            return false;

        List<File> files;
        try {
            files = (List<File>) support.getTransferable()
                    .getTransferData(DataFlavor.javaFileListFlavor);
        } catch (UnsupportedFlavorException | IOException ex) {
            // should never happen (or JDK is buggy)
            return false;
        }

        for (File file: files) {
            // do something...
        }
        return true;
    }
}

任意のコンポーネントで使用します

myComponent.setTransferHandler(new FileDropHandler());

これは私には有効です。私はこれを次のように使用しています(スカラ):

def onDrop(files: List[java.io.File]): Unit = { ... }

    val lblDrop = new Label {
      peer.setTransferHandler(new FileDropHandler(onDrop))
      border = EtchedBorder
    }



class FileDropHandler(val onDrop: List[java.io.File] => Unit) extends javax.swing.TransferHandler {
      import javax.swing.JComponent
      import java.awt.datatransfer.{Transferable, DataFlavor}
        import java.net.URI
    import java.io.File

    val stdFileListFlavor = DataFlavor.javaFileListFlavor
    val nixFileListFlavor = new DataFlavor("text/uri-list;class=java.lang.String")

    override def canImport(comp: JComponent, flavors: Array[DataFlavor]): Boolean =
        flavors.exists(flavor =>
            (flavor == stdFileListFlavor) ||
            (flavor == nixFileListFlavor)
        )

    override def importData(comp: JComponent, t: Transferable): Boolean = {

        val flavors = t.getTransferDataFlavors()

        val files = if (flavors.exists(_ == stdFileListFlavor)) {
            val data = t.getTransferData(stdFileListFlavor)
            importStdFileList( data )
        } else if (flavors.exists(_ == nixFileListFlavor)) {
            val data = t.getTransferData(nixFileListFlavor)
            importNixFileList( data )
        } else List()

        onDrop( files )

        !files.isEmpty
    }

    private def importStdFileList(data: Any): List[File] = {
      data.asInstanceOf[List[File]] //XXX NOT TESTED
    }

    private def importNixFileList(data: Any): List[File] = {

        def clean(rawLine: String): Option[String] = {
            val line = rawLine.trim
            if   (line.length == 0 || line == "#") None
            else                                   Some(line)
        }

        def asURI(line: String): Option[URI] = {
            try   { Some(new URI(line)) }
            catch { case e:Exception => println(e); None }
        }

        def asFile(uri: URI): Option[File] = {
            try   { Some(new File(uri)) }
            catch { case e:Exception => println(e); None }
        }

        data.asInstanceOf[java.lang.String].split("\n")
     .toList flatMap clean flatMap asURI flatMap asFile
    }
}
ライセンス: CC-BY-SA帰属
所属していません StackOverflow
scroll top