JavaFX/ScalaFX и буфер обмена: не удается скопировать файлы?

Копирование файлов не работает:

  def toClipboard(selectedLinesOnly: Boolean = false): Unit = {
    val clipboard = Clipboard.systemClipboard
    val content = new ClipboardContent
    val items: Iterable[FileRecord] = selectedLinesOnly match {
      case true => tableView.selectionModel.value.selectedItems.toSeq
      case false => tableView.items.value
    }
    val files = items.map(_.file)
    println(s"COPY $files")
    content.putFiles(files.toSeq)
    clipboard.content = content
  }

Выход: [info] COPY [SFX][/tmp/test/a.txt, /tmp/test/b.txt]

Нет файлов для вставки.

  def toClipboard(selectedLinesOnly: Boolean = false): Unit = {
    val clipboard = Clipboard.systemClipboard
    val content = new ClipboardContent
    val items: Iterable[FileRecord] = selectedLinesOnly match {
      case true => tableView.selectionModel.value.selectedItems.toSeq
      case false => tableView.items.value
    }
    val files = items.map(_.file.getPath)
    println(s"COPY $files")
    content.putFilesByPath(files.toSeq)
    clipboard.content = content
  }

Выход: [info] COPY [SFX][/tmp/test/a.txt, /tmp/test/b.txt]

Нет файлов для вставки.

  def toClipboard(selectedLinesOnly: Boolean = false): Unit = {
    val clipboard = Clipboard.systemClipboard
    val content = new ClipboardContent
    val items: Iterable[FileRecord] = selectedLinesOnly match {
      case true => tableView.selectionModel.value.selectedItems.toSeq
      case false => tableView.items.value
    }
    val files = items.map("file://" + _.file.getPath)
    println(s"COPY $files")
    content.putFilesByPath(files.toSeq)
    clipboard.content = content
  }

Выход: [info] COPY [SFX][file:///tmp/test/a.txt, file:///tmp/test/b.txt]

Нет файлов для вставки.

Но копирование путей в буфер обмена возможно:

  def toClipboard(selectedLinesOnly: Boolean = false): Unit = {
    val clipboard = Clipboard.systemClipboard
    val content = new ClipboardContent
    val items: Iterable[FileRecord] = selectedLinesOnly match {
      case true => tableView.selectionModel.value.selectedItems.toSeq
      case false => tableView.items.value
    }
    val files = items.map(_.file.getPath)
    println(s"COPY $files")
    content.putString(files.mkString(" "))
    clipboard.content = content
  }

Теперь это в моем буфере обмена: "/tmp/test/a.txt /tmp/test/b.txt"

Но мне это нужно в виде файлов, а не строки.

Как сделать так, чтобы копирование файлов работало в моем приложении?

Я работаю с OpenJFX 8 на Ubuntu.


person ideaboxer    schedule 05.06.2016    source источник


Ответы (2)


ClipBoard не имеет такой функциональности, как FileUtils.copyDirectoryToDirectory и FileUtils.moveDirectoryToDirectory (также называемой копированием или вырезанием). Буфер обмена может дать только путь или общие данные. Эти функции можно реализовать с помощью перетаскивания с помощью Dragboard.

Драпборд JavaFX:

Во время жеста перетаскивания могут передаваться различные типы данных, например text, images, URLs, files, bytes, and strings.

Класс javafx.scene.input.DragEvent — это базовый класс, используемый для реализации жеста перетаскивания. Дополнительные сведения о конкретных методах и других классах пакета javafx.scene.input см. в документации по API.

Подробнее см. в этом руководстве: Функция перетаскивания в Приложения JavaFX

Пример кода Dragboard JavaFX: HelloDragAndDrop.java

package hellodraganddrop;

import javafx.application.Application;
import javafx.event.EventHandler;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.input.*;
import javafx.scene.paint.Color;
import javafx.scene.text.Text;
import javafx.stage.Stage;

/**
 * Demonstrates a drag-and-drop feature.
 */
public class HelloDragAndDrop extends Application {

    @Override public void start(Stage stage) {
        stage.setTitle("Hello Drag And Drop");

        Group root = new Group();
        Scene scene = new Scene(root, 400, 200);
        scene.setFill(Color.LIGHTGREEN);

        final Text source = new Text(50, 100, "DRAG ME");
        source.setScaleX(2.0);
        source.setScaleY(2.0);

        final Text target = new Text(250, 100, "DROP HERE");
        target.setScaleX(2.0);
        target.setScaleY(2.0);

        source.setOnDragDetected(new EventHandler <MouseEvent>() {
            public void handle(MouseEvent event) {
                /* drag was detected, start drag-and-drop gesture*/
                System.out.println("onDragDetected");

                /* allow any transfer mode */
                Dragboard db = source.startDragAndDrop(TransferMode.ANY);

                /* put a string on dragboard */
                ClipboardContent content = new ClipboardContent();
                content.putString(source.getText());
                db.setContent(content);

                event.consume();
            }
        });

        target.setOnDragOver(new EventHandler <DragEvent>() {
            public void handle(DragEvent event) {
                /* data is dragged over the target */
                System.out.println("onDragOver");

                /* accept it only if it is  not dragged from the same node 
                 * and if it has a string data */
                if (event.getGestureSource() != target &&
                        event.getDragboard().hasString()) {
                    /* allow for both copying and moving, whatever user chooses */
                    event.acceptTransferModes(TransferMode.COPY_OR_MOVE);
                }

                event.consume();
            }
        });

        target.setOnDragEntered(new EventHandler <DragEvent>() {
            public void handle(DragEvent event) {
                /* the drag-and-drop gesture entered the target */
                System.out.println("onDragEntered");
                /* show to the user that it is an actual gesture target */
                if (event.getGestureSource() != target &&
                        event.getDragboard().hasString()) {
                    target.setFill(Color.GREEN);
                }

                event.consume();
            }
        });

        target.setOnDragExited(new EventHandler <DragEvent>() {
            public void handle(DragEvent event) {
                /* mouse moved away, remove the graphical cues */
                target.setFill(Color.BLACK);

                event.consume();
            }
        });

        target.setOnDragDropped(new EventHandler <DragEvent>() {
            public void handle(DragEvent event) {
                /* data dropped */
                System.out.println("onDragDropped");
                /* if there is a string data on dragboard, read it and use it */
                Dragboard db = event.getDragboard();
                boolean success = false;
                if (db.hasString()) {
                    target.setText(db.getString());
                    success = true;
                }
                /* let the source know whether the string was successfully 
                 * transferred and used */
                event.setDropCompleted(success);

                event.consume();
            }
        });

        source.setOnDragDone(new EventHandler <DragEvent>() {
            public void handle(DragEvent event) {
                /* the drag-and-drop gesture ended */
                System.out.println("onDragDone");
                /* if the data was successfully moved, clear it */
                if (event.getTransferMode() == TransferMode.MOVE) {
                    source.setText("");
                }

                event.consume();
            }
        });

        root.getChildren().add(source);
        root.getChildren().add(target);
        stage.setScene(scene);
        stage.show();
    }

    public static void main(String[] args) {
        Application.launch(args);
    }
}

На самом деле в Java есть несколько способов скопировать файл: 4 способа скопировать файл в Java

Ссылка на ресурс:

  1. Как мне правильно обрабатывать копирование/вырезание файлов и вставить в javafx?
person SkyWalker    schedule 17.06.2016
comment
Я уже реализовал функциональность dragboard. Буфер обмена является еще одним требованием, поскольку перетаскивание не всегда обеспечивает максимальное удобство использования (+ пользователь ожидает средства копирования содержимого). Подводя итог вашему ответу: JavaFX не поддерживает размещение файлов в буфере обмена, Java поддерживает копирование файлов из источника в место назначения, но не помещает их в буфер обмена. Что мне до сих пор непонятно: Почему ClipboardContent предоставляет мне putFilesByPath, хотя он не делает того, что обещает его название? - person ideaboxer; 18.06.2016
comment
Кажется, работает с Java с использованием Toolkit: stackoverflow.com/questions/31798646/ Я попробую. - person ideaboxer; 18.06.2016

Мое решение, которое я успешно протестировал с Ubuntu:

FilesTransferable.scala

package myapp.clipboard

import java.awt.datatransfer._

case class FilesTransferable(files: Iterable[String]) extends Transferable {
  val clipboardString: String = "copy\n" + files.map(path => s"file://$path").mkString("\n")
  val dataFlavor = new DataFlavor("x-special/gnome-copied-files")
  def getTransferDataFlavors(): Array[DataFlavor] = {
    Seq(dataFlavor).toArray
  }
  def isDataFlavorSupported(flavor: DataFlavor): Boolean = {
    dataFlavor.equals(flavor)
  }
  @throws(classOf[UnsupportedFlavorException])
  @throws(classOf[java.io.IOException])
  def getTransferData(flavor: DataFlavor): Object = {
    if(flavor.getRepresentationClass() == classOf[java.io.InputStream]) {
      new java.io.ByteArrayInputStream(clipboardString.getBytes())
    } else {
      return null;
    }
  }
}

Буфер обмена.scala

package myapp.clipboard

import java.awt.Toolkit
import java.awt.datatransfer._

object Clipboard {
  def toClipboard(
    transferable: Transferable,
    lostOwnershipCallback: (Clipboard, Transferable) => Unit =
      { (clipboard: Clipboard, contents: Transferable) => Unit }
  ): Unit = {
    Toolkit.getDefaultToolkit.getSystemClipboard.setContents(
      transferable,
      new ClipboardOwner {
        def lostOwnership(clipboard: Clipboard, contents: Transferable): Unit = {
          lostOwnershipCallback(clipboard, contents)
        }
      }
    )
  }
}

Довольно много кода для такой базовой функциональности (и все еще не независимой от ОС).

Пример использования:

val transferable = FilesTransferable(filePaths)
myapp.clipboard.Clipboard.toClipboard(transferable, { (cb, contents) =>
  println(s"lost clipboard ownership (clipboard: $cb)")
})

Если это правда, что до сих пор не существует решений, являющихся частью Java-AWT/Swing/JavaFX/Scala-Swing/ScalaFX (на самом деле, я до сих пор не могу в это поверить), мой план состоит в том, чтобы построить простую в использовании -используйте библиотеку буфера обмена, которая поддерживает как минимум OS X, Ubuntu и Windows.

person ideaboxer    schedule 21.06.2016