org.openide.nodes.NodeTransfer Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.openide.nodes;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.awt.dnd.DnDConstants;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
import org.openide.util.datatransfer.ExTransferable;
import org.openide.util.datatransfer.MultiTransferObject;
import org.openide.util.datatransfer.PasteType;
/** Class that contains specific datatransfer flavors and methods to work with
* nodes. There are flavors to allow a node
* to be copied or cut, and to decide its paste types.
* This is a dummy utility class--no instances are possible.
*
* @author Jaroslav Tulach
*/
public abstract class NodeTransfer extends Object {
/** Constants for drag-n-drop operations.
* Are exactly the same as constants
* in {@link DnDConstants}.
*/
public static final int DND_NONE = DnDConstants.ACTION_NONE;
public static final int DND_COPY = DnDConstants.ACTION_COPY;
public static final int DND_MOVE = DnDConstants.ACTION_MOVE;
public static final int DND_COPY_OR_MOVE = DnDConstants.ACTION_COPY | DnDConstants.ACTION_MOVE;
public static final int DND_LINK = DnDConstants.ACTION_LINK;
public static final int DND_REFERENCE = DnDConstants.ACTION_LINK;
/** Constant indicating copying to the clipboard.
* Equal to {@link #DND_COPY}, because
* copy to clipboard and d'n'd copy should be the same.
*/
public static final int CLIPBOARD_COPY = DND_COPY;
/** Constant indicating cutting to the clipboard.
*/
public static final int CLIPBOARD_CUT = 0x04;
/** Generic mask for copying nodes (do not destroy the original).
* Equal to {@link #CLIPBOARD_COPY} or {@link #DND_COPY}.
*/
public static final int COPY = CLIPBOARD_COPY | DND_COPY;
/** Generic mask for moving nodes (destroy the original).
* Equal to {@link #CLIPBOARD_CUT} or {@link #DND_MOVE}.
*/
public static final int MOVE = CLIPBOARD_CUT | DND_MOVE;
/** Flavor for representation class {@link NodeTransfer.Paste}.
* Provides methods for obtaining a set of {@link PasteType}s when
* the target node is known.
*/
private static final DataFlavor nodePasteFlavor;
static {
try {
nodePasteFlavor = new DataFlavor(
"application/x-java-openide-nodepaste;class=org.openide.nodes.Node", // NOI18N
Node.getString("LBL_nodePasteFlavor"),
Node.class.getClassLoader());
} catch (ClassNotFoundException e) {
throw new AssertionError(e);
}
}
private static final String dndMimeType = "application/x-java-openide-nodednd;class=org.openide.nodes.Node;mask="; //NOI18N
private NodeTransfer() {
}
/** Creates data flavor for given mask of dnd actions.
* @param actions any mask of dnd constants DND_* and CLIPBOARD_*
*/
private static DataFlavor createDndFlavor(int actions) {
Exception ex;
try {
return new DataFlavor(dndMimeType + actions,
null, Node.class.getClassLoader());
} catch (IllegalArgumentException iae) {
ex = iae;
} catch (ClassNotFoundException cnfE) {
ex = cnfE;
}
throw new IllegalStateException("Cannot createDndFlavor(" + actions + ")", ex); // NOI18N
}
/** Creates transferable that represents a node operation, such as cut-to-clipboard.
* The transferable will be recognizable by {@link #node}, {@link #nodes}, and {@link #cookie}.
*
* @param n the node to create a transferable for
* @param actions the action performed on the node
* @return the transferable
*/
public static ExTransferable.Single transferable(final Node n, int actions) {
return new ExTransferable.Single(createDndFlavor(actions)) {
public Object getData() {
return n;
}
};
}
/** Obtain a node from a transferable.
* Probes the transferable in case it includes a flavor corresponding
* to a node operation (which you must specify a mask for).
*
* @param t transferable
* @param action one of the DND_*
or CLIPBOARD_*
constants
* @return the node or null
*/
public static Node node(Transferable t, int action) {
DataFlavor[] flavors = t.getTransferDataFlavors();
if (flavors == null) {
return null;
}
int len = flavors.length;
String subtype = "x-java-openide-nodednd"; // NOI18N
String primary = "application"; // NOI18N
String mask = "mask"; // NOI18N
for (int i = 0; i < len; i++) {
DataFlavor df = flavors[i];
if (df.getSubType().equals(subtype) && df.getPrimaryType().equals(primary)) {
try {
int m = Integer.valueOf(df.getParameter(mask)).intValue();
if ((m & action) != 0) {
// found the node
return (Node) t.getTransferData(df);
}
} catch (NumberFormatException nfe) {
maybeReportException(nfe);
} catch (ClassCastException cce) {
maybeReportException(cce);
} catch (IOException ioe) {
maybeReportException(ioe);
} catch (UnsupportedFlavorException ufe) {
maybeReportException(ufe);
}
}
}
return null;
}
/** Obtain a list of nodes from a transferable.
* If there is only a single node in the transferable, this will just return a singleton
* array like {@link #node}.
* If there is a {@link ExTransferable#multiFlavor multiple transfer} (of at least one element),
* each element of which
* contains a node, then an array of these will be returned.
* If neither of these things is true, null
will be returned.
*
This is a convenience method intended for those who wish to specially support pastes
* of multiple nodes at once. (By default, an explorer will
* fall back to presenting each component of a multiple-item transferable separately when checking for paste
* types on a target node, so if you have only one paste type and it makes no difference whether all of the nodes
* are pasted together or separately, you can just use {@link #node}.)
*
If you wish to test for cookies, you should do so manually
* according to your specific logic.
* @param t the transferable to probe
* @param action a DnD or clipboard constant
* @return a non-empty array of nodes, or null
*/
public static Node[] nodes(Transferable t, int action) {
try {
if (t.isDataFlavorSupported(ExTransferable.multiFlavor)) {
MultiTransferObject mto = (MultiTransferObject) t.getTransferData(ExTransferable.multiFlavor);
int count = mto.getCount();
Node[] ns = new Node[count];
boolean ok = true;
for (int i = 0; i < count; i++) {
Node n = node(mto.getTransferableAt(i), action);
if (n == null) {
ok = false;
break;
} else {
ns[i] = n;
}
}
if (ok && (count > 0)) {
return ns;
}
} else {
Node n = node(t, action);
if (n != null) {
return new Node[] { n };
}
}
} catch (ClassCastException cce) {
maybeReportException(cce);
} catch (IOException ioe) {
maybeReportException(ioe);
} catch (UnsupportedFlavorException ufe) {
maybeReportException(ufe);
}
return null;
}
/** Obtain a cookie instance from the copied node in a transferable.
*
* First of all it checks whether the given transferable contains
* a node and then asks for the cookie.
*
If you wish to specially support multiple-node transfers, please use {@link #nodes}
* and manually check for the desired combination of cookies.
*
* @param t transferable to check in
* @param cookie cookie representation class to look for
* @param action the action which was used to store the node
*
* @return cookie or null
if it does not exist
*/
public static T cookie(Transferable t, int action, Class cookie) {
Node n = node(t, action);
return (n == null) ? null : n.getCookie(cookie);
}
/** Creates transfer object that is used to carry an intelligent
* paste source through transferable or clipboard.
* {@link #findPaste} can retrieve it.
* @param paste the intelligent source of paste types
* @return the transferable
*/
public static ExTransferable.Single createPaste(final Paste paste) {
return new ExTransferable.Single(nodePasteFlavor) {
public Object getData() {
return paste;
}
};
}
/** Find an intelligent source of paste types in a transferable.
* Note that {@link AbstractNode#createPasteTypes} looks for this
* by default, so cut/copied nodes may specify how they may be pasted
* to some external node target.
* @param t the transferable to test
* @return the intelligent source or null
if none is in the transferable
*/
public static Paste findPaste(Transferable t) {
try {
if (t.isDataFlavorSupported(nodePasteFlavor)) {
return (Paste) t.getTransferData(nodePasteFlavor);
}
} catch (ClassCastException cce) {
maybeReportException(cce);
} catch (IOException ioe) {
maybeReportException(ioe);
} catch (UnsupportedFlavorException ufe) {
maybeReportException(ufe);
}
return null;
}
/** Print a stack trace if debugging is on.
* Used for exceptions that could occur when probing transferables,
* which should not interrupt the probing with an error, but
* indicate a bug elsewhere and should be reported somehow.
* @param e the exception
*/
private static void maybeReportException(Exception e) {
Logger.getLogger(NodeTransfer.class.getName()).log(Level.WARNING, "Node transfer error: {0}", e.getMessage());
Logger.getLogger(NodeTransfer.class.getName()).log(Level.INFO, null, e);
}
/** An intelligent source of paste types (ways how to paste)
* for a target node.
*
* Each node should check for this
* type in a paste operation to allow anyone to insert something
* into it.
*
* Sample example of implementation of {@link Node#getPasteTypes}:
*
* public PasteType[] getPasteTypes (Transferable t) {
* NodeTransfer.Paste p = (NodeTransfer.Paste)t.getTransferData (
* NodeTransfer.nodePasteFlavor
* );
* return p.types (this);
* }
*
*/
public interface Paste {
/** Method that checks the type of target node and can
* decide which paste types it supports.
*
* @param target the target node
* @return array of paste types that are valid for such a target node
*/
public PasteType[] types(Node target);
}
}