--- /dev/null 2014-07-18 18:29:23.000000000 +0400 +++ new/modules/swing/src/main/java/javafx/embed/swing/CachingTransferable.java 2014-07-18 18:29:22.504132100 +0400 @@ -0,0 +1,128 @@ +/* + * Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package javafx.embed.swing; + +import com.sun.javafx.embed.EmbeddedSceneDSInterface; +import com.sun.javafx.tk.Toolkit; +import java.awt.datatransfer.DataFlavor; +import java.awt.datatransfer.Transferable; +import java.awt.dnd.DropTargetDragEvent; +import java.awt.dnd.DropTargetDropEvent; +import java.util.Arrays; +import java.util.Collections; +import java.util.Map; +import java.util.HashMap; +import java.util.Set; +import javafx.scene.input.TransferMode; +import java.io.UnsupportedEncodingException; +import javafx.scene.input.Clipboard; +import javafx.scene.input.DataFormat; + +/** + * A Transferable implementation backed by a Map. + * The data can be populated either from AWT Transferable + * or from FX Clipboard. + */ +class CachingTransferable implements Transferable { + + @Override + public Object getTransferData(final DataFlavor flavor) throws UnsupportedEncodingException + { + String mimeType = DataFlavorUtils.getFxMimeType(flavor); + return DataFlavorUtils.adjustFxData( + flavor, getData(mimeType)); + } + + @Override + public DataFlavor[] getTransferDataFlavors() { + final String mimeTypes[] = getMimeTypes(); + return DataFlavorUtils.getDataFlavors(mimeTypes); + } + + @Override + public boolean isDataFlavorSupported(final DataFlavor flavor) { + return isMimeTypeAvailable( + DataFlavorUtils.getFxMimeType(flavor)); + } + + private Map mimeType2Data = Collections.EMPTY_MAP; + + void updateData(Transferable t, boolean fetchData) { + final Map mimeType2DataFlavor = + DataFlavorUtils.adjustSwingDataFlavors( + t.getTransferDataFlavors()); + + // If we keep reference to source Transferable in SwingDragSource and + // call Transferable#getTransferData() on it from + // SwingDragSource#getData() we may run into + // "java.awt.dnd.InvalidDnDOperationException" issue as + // SwingDragSource#getData() is called from FX user code and from + // QuantumClipboard#getContent() (sik!). These calls usually take + // place in the context of + // EmbeddedSceneDTInterface#handleDragDrop() method as the + // normal handling of DnD. + // Instead of keeping reference to source Transferable we just read + // all its data while in the context safe for calling + // Transferable#getTransferData(). + // + // This observation is true for standard AWT Transferable-s. + // Things may be totally broken for custom Transferable-s though. + + // For performance reasons, the DRAG_ENTERED and DRAG_OVER event + // handlers pass fetchData == false so as to update the set of + // available MIME types only. The DRAG_DROPPED handler passes + // fetchData == true which also fetches all the data. + // NOTE: Due to JDK-8028585 this code won't be able to fetch data + // when invoked from handlers other than DROPPED in any case. + + try { + mimeType2Data = DataFlavorUtils.readAllData(t, mimeType2DataFlavor, + fetchData); + } catch (Exception e) { + mimeType2Data = Collections.EMPTY_MAP; + } + } + + void updateData(Clipboard cb, boolean fetchData) { + mimeType2Data = new HashMap<>(); + for (DataFormat f : cb.getContentTypes()) { + mimeType2Data.put(DataFlavorUtils.getMimeType(f), + fetchData ? cb.getContent(f) : null); + } + } + + public Object getData(final String mimeType) { + return mimeType2Data.get(mimeType); + } + + public String[] getMimeTypes() { + return mimeType2Data.keySet().toArray(new String[0]); + } + + public boolean isMimeTypeAvailable(final String mimeType) { + return Arrays.asList(getMimeTypes()).contains(mimeType); + } +}