Swingでドラッグアンドドロップを使用してファイルパスを取得するにはどうすればよいですか?
-
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
}
}