Linux の Java でファイルのドラッグ&ドロップに対応する方法(Swing および AWT のサンプル)

 結果的に、自分でもビックリするくらいニッチなネタになってしまったのですが、調べた時間が勿体無いので、無意味に記事として書き残しておきます。

 さて、と。

 えぇとですね、ドラッグ&ドロップってあるじゃないですか?

 とあるアプリケーションで選択したファイルや何かを、別のアプリケーションまでドラッグして、ポイッとドロップするマウスジェスチャーのことですね。

 Java でも標準で API が用意されているので、割りと簡単に実装できるのですが、あちらこちらでよく紹介されている書き方だと、一部の Linux で上手く機能しないことがあったりします。

 何故か、これを解決する日本語の情報があまり見当たらなかったので、適当に書いてみようかというのが今回の趣旨だったのですが......

 いえね、最初はディストリビューションか、もしくはデスクトップマネージャに依るのかなーと思ってたんですよ。

 ところがどっこい、記事にする為に軽く調査したところ、どうやらそうではなくて、Java のバージョンに依るらしいんですね、これが。

 具体的に言うと、Linux にインストールした Java のバージョンが 6 以前の場合のみ、問題が発生するようなのです。

 えぇ~......

 Java 9 の足音も聞こえてこようかというこの時期に、Java 6 の、しかも Linux でのみ発生する GUI の問題なんて、いったい誰が気にするんだ......

 いや、多分、Mac の Java 6 でも再現していたんじゃないかなーという気がするのですが、今日び Apple が用意していた時代の Java 6 が乗ってる Mac なんて、この世にほとんど存在しないでしょー......

 こんなどうでもいい情報を必要としている人なんて、世界中を見渡しても、余裕で数えられるくらいしかいないよ、きっと......そりゃ、情報が見当たらないのも納得ですね(笑)

 えぇと、気を取り直しまして――

 今回見ていくのは、ファイラーからドラッグしたファイル(複数可)を Java のウィンドウにドロップする方法なので、正確に言うならば、実装的にはドロップだけになります。

 自分でもちょっと引くくらい、恐ろしくニッチなネタですが、運悪く開いてしまった方は、折角なので興味が無くてもお付き合いください(強制)

スポンサーリンク

読み込み中です。少々お待ち下さい

まずは、一般的な書き方から

 まずは、上で触れた問題が発生してしまう、よく見かける書き方を以下に示します。

 ちなみに、Java 6(JDK/JRE 1.6)以降対応です。

 DragAndDropSample.java

  1. import java.awt.BorderLayout;
  2. import java.awt.Font;
  3. import java.awt.datatransfer.DataFlavor;
  4. import java.io.File;
  5. import java.util.List;
  6. import javax.swing.JFrame;
  7. import javax.swing.JLabel;
  8. import javax.swing.SwingConstants;
  9. import javax.swing.TransferHandler;
  10. public class DragAndDropSample extends JFrame {
  11.     @Override protected void frameInit() {
  12.         super.frameInit();
  13.         this.setSize(480, 320);
  14.         this.setLocationByPlatform(true);
  15.         this.setTitle(this.getClass().getName());
  16.         this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  17.         JLabel label = new JLabel("Drop files here.", SwingConstants.CENTER);
  18.         label.setFont(new Font(Font.DIALOG, Font.BOLD, 48));
  19.         this.add(label, BorderLayout.CENTER);
  20.         this.setTransferHandler(new TransferHandler() {
  21.             @Override public boolean importData(TransferSupport support) {
  22.                 try {
  23.                     if (canImport(support)) {
  24.                         if (support.isDataFlavorSupported(DataFlavor.javaFileListFlavor)) {
  25.                             Object list = support.getTransferable().getTransferData(DataFlavor.javaFileListFlavor);
  26.                             if (list instanceof List<?>) {
  27.                                 for (Object o : ((List<?>)list)) {
  28.                                     if (o instanceof File) {
  29.                                         System.out.println(((File)o).getAbsolutePath());
  30.                                     }
  31.                                 }
  32.                             }
  33.                             return true;
  34.                         }
  35.                     }
  36.                 } catch (Exception ex) {
  37.                     ex.printStackTrace();
  38.                 }
  39.                 return false;
  40.             }
  41.             @Override public boolean canImport(TransferSupport support) {
  42.                 return (support.isDataFlavorSupported(DataFlavor.javaFileListFlavor));
  43.             }
  44.         });
  45.     }
  46.     
  47.     public static void main(String... args) {
  48.         (new DragAndDropSample()).setVisible(true);
  49.     }
  50. }

 23 行目からはじまる setTransferHandler 以外の部分はどうでもいいので、背景色を変えてあります。

 まー、普通だったら、こんな風に書いたコードを実行して、ウィンドウにファイルをドロップすれば、そのパスが標準出力に出力される訳です。

ファイラーからファイルをドロップ
Windows なら、Java 6 でも問題ナシ

 ところが、Linux の Java 6 だと、ドロップが上手く処理されません(つまり、標準出力に何も出力されません)。

Linux の Java 6 だと、上手く処理されない
Linux でも Java 7 なら問題ナシ

 これは何故かといいますと、実はイベント自体が発生していない訳ではなく、TransferSupport の isDataFlavorSupported が DataFlavor.javaFileListFlavor を引数に与えると、Linux の Java 6 環境では false を返すからです。

 ならば、どうするかというと、引数に DataFlavor.stringFlavor を指定してあげると良いでしょう(Mime-Type は text/url-list。ちなみに、DataFlavor.javaFileListFlavor の Mime-Type は application/x-java-file-list です)。

Linux の Java 6 対応(Swing 版)

 つまり、こんな風に書き直します。

 DragAndDropSwingSample.java

  1. import java.awt.BorderLayout;
  2. import java.awt.Font;
  3. import java.awt.datatransfer.DataFlavor;
  4. import java.io.File;
  5. import java.net.URL;
  6. import java.net.URLDecoder;
  7. import java.util.LinkedList;
  8. import java.util.List;
  9. import java.util.StringTokenizer;
  10. import javax.swing.JFrame;
  11. import javax.swing.JLabel;
  12. import javax.swing.SwingConstants;
  13. import javax.swing.TransferHandler;
  14. public class DragAndDropSwingSample extends JFrame {
  15.     @Override protected void frameInit() {
  16.         super.frameInit();
  17.         this.setSize(480, 320);
  18.         this.setLocationByPlatform(true);
  19.         this.setTitle(this.getClass().getName());
  20.         this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  21.         JLabel label = new JLabel("Drop files here.", SwingConstants.CENTER);
  22.         label.setFont(new Font(Font.DIALOG, Font.BOLD, 48));
  23.         this.add(label, BorderLayout.CENTER);
  24.         this.setTransferHandler(new TransferHandler() {
  25.             @Override public boolean importData(TransferSupport support) {
  26.                 try {
  27.                     if (canImport(support)) {
  28.                         List<File> files = new LinkedList<File>();
  29.                         if (support.isDataFlavorSupported(DataFlavor.javaFileListFlavor)) {
  30.                             Object list = support.getTransferable().getTransferData(DataFlavor.javaFileListFlavor);
  31.                             if (list instanceof List<?>) {
  32.                                 for (Object o : ((List<?>)list)) {
  33.                                     if (o instanceof File) {
  34.                                         files.add((File)o);
  35.                                     }
  36.                                 }
  37.                             }
  38.                         } else if (support.isDataFlavorSupported(DataFlavor.stringFlavor)) {
  39.                             // for Linux (Java 6 or lower)
  40.                             StringTokenizer tokens = new StringTokenizer((String)support.getTransferable().getTransferData(DataFlavor.stringFlavor));
  41.                             while (tokens.hasMoreTokens()) {
  42.                                 try {
  43.                                     files.add(new File(URLDecoder.decode((new URL(tokens.nextToken()).getFile()), "UTF-8")));
  44.                                 } catch (Exception ex) {
  45.                                     ex.printStackTrace();
  46.                                 }
  47.              }
  48.                         }
  49.                         if (files.size() > 0) {
  50.                             for (File file : files) {
  51.                                 System.out.println(file.getAbsolutePath());
  52.                             }
  53.                             return true;
  54.                         }
  55.                     }
  56.                 } catch (Exception ex) {
  57.                     ex.printStackTrace();
  58.                 }
  59.                 return false;
  60.             }
  61.             @Override public boolean canImport(TransferSupport support) {
  62.                 if (support.isDataFlavorSupported(DataFlavor.javaFileListFlavor)) {
  63.                     return true;
  64.                 }
  65.                 // for Linux (Java 6 or lower)
  66.                 boolean importable = false;
  67.                 if (support.isDataFlavorSupported(DataFlavor.stringFlavor)) {
  68.                     for (DataFlavor flavor : support.getDataFlavors()) {
  69.                         if (flavor.getSubType().equals("uri-list")) {
  70.                             importable = true;
  71.                             break;
  72.                         }
  73.                     }
  74.      }
  75.                 return importable;
  76.             }
  77.         });
  78.     }
  79.     
  80.     public static void main(String... args) {
  81.         (new DragAndDropSwingSample()).setVisible(true);
  82.     }
  83. }

 DataFlavor.stringFlavor なので、SubType の url-list だけで条件判断していますが、気になる方は getPrimaryType も併用するか、若しくは getMimeType を使ってください。

 で、こんな風に書くと、以下のように Linux の Java 6 でも、ドロップを処理できるようになります。

Linux の Java 6 でも問題ナシ

Linux の Java 6 対応(AWT 版)

 世に出回っている D&D サンプルコードのほとんどが Swing のような気がするので、AWT 版も置いておきます(より一層、誰も必要としていない気がしますが)。

 DragAndDropAWTSample.java

  1. import java.awt.BorderLayout;
  2. import java.awt.Font;
  3. import java.awt.Frame;
  4. import java.awt.Label;
  5. import java.awt.datatransfer.DataFlavor;
  6. import java.awt.dnd.DnDConstants;
  7. import java.awt.dnd.DropTarget;
  8. import java.awt.dnd.DropTargetDropEvent;
  9. import java.awt.event.WindowEvent;
  10. import java.awt.event.WindowListener;
  11. import java.io.File;
  12. import java.net.URL;
  13. import java.net.URLDecoder;
  14. import java.util.LinkedList;
  15. import java.util.List;
  16. import java.util.StringTokenizer;
  17. public class DragAndDropAWTSample extends Frame {
  18.     public DragAndDropAWTSample() {
  19.         this.setSize(480, 320);
  20.         this.setLocationByPlatform(true);
  21.         this.setTitle(this.getClass().getName());
  22.         Label label = new Label("Drop files here.", Label.CENTER);
  23.         label.setFont(new Font(Font.DIALOG, Font.BOLD, 48));
  24.         this.add(label, BorderLayout.CENTER);
  25.         this.setDropTarget(new DropTarget() {
  26.             @Override public void drop(DropTargetDropEvent e) {
  27.                 e.acceptDrop(DnDConstants.ACTION_REFERENCE);
  28.                 try {
  29.                     List<File> files = new LinkedList<File>();
  30.                     if (e.isDataFlavorSupported(DataFlavor.javaFileListFlavor)) {
  31.                         Object list = e.getTransferable().getTransferData(DataFlavor.javaFileListFlavor);
  32.                         if (list instanceof List<?>) {
  33.                             for (Object o : ((List<?>)list)) {
  34.                                 if (o instanceof File) {
  35.                                     files.add((File)o);
  36.                                 }
  37.                             }
  38.                         }
  39.                     } else if (e.isDataFlavorSupported(DataFlavor.stringFlavor)) {
  40.                         // for Linux (Java 6 or lower)
  41.                         for (DataFlavor flavor : e.getTransferable().getTransferDataFlavors()) {
  42.                             if (flavor.getSubType().equals("uri-list")) {
  43.                                 StringTokenizer tokens = new StringTokenizer((String)e.getTransferable().getTransferData(DataFlavor.stringFlavor));
  44.                                 while (tokens.hasMoreTokens()) {
  45.                                     try {
  46.                                         files.add(new File(URLDecoder.decode((new URL(tokens.nextToken()).getFile()), "UTF-8")));
  47.                                     } catch (Exception ex) {
  48.                                         ex.printStackTrace();
  49.                                     }
  50.                                 }
  51.                                 break;
  52.                             }
  53.                         }
  54.                     }
  55.                     if (files.size() > 0) {
  56.                         for (File file : files) {
  57.                             System.out.println(file.getAbsolutePath());
  58.                         }
  59.                     }
  60.                 } catch (Throwable t) {
  61.                     t.printStackTrace();
  62.                 }
  63.             }
  64.         });
  65.         final Frame frame = this;
  66.         this.addWindowListener(new WindowListener() {
  67.             @Override public void windowActivated(WindowEvent e) { }
  68.             @Override public void windowClosed(WindowEvent e) { System.exit(0); }
  69.             @Override public void windowClosing(WindowEvent e) { frame.dispose(); }
  70.             @Override public void windowDeactivated(WindowEvent e) { }
  71.             @Override public void windowDeiconified(WindowEvent e) { }
  72.             @Override public void windowIconified(WindowEvent e) { }
  73.             @Override public void windowOpened(WindowEvent e) { }
  74.         });
  75.     }
  76.     
  77.     public static void main(String... args) {
  78.         (new DragAndDropAWTSample()).setVisible(true);
  79.     }
  80. }

おわりに

 ヤケクソ気味に、Stream API とラムダ式を使った Java 8 っぽいサンプルも用意しようかと思ったのですが、書いてみたらあまりにも無意味過ぎたので、危うく思いとどまりました。

 ホントに、いったい誰得なんだ、この記事......

 ちなみに、上で動作確認している Linux は、Debian 8.0 "jessie" になります。

 もしかしたら、Java 6 でも一番目のコードが問題無く動作する Linux 環境もあるかも知れないことはお断りしておきます。

 ますます、誰得なんだ......

この記事をシェア
  • このエントリーをはてなブックマークに追加
  • Share on Google+
  • この記事についてツイート
  • この記事を Facebook でシェア