com.sun.jna.platform.dnd.DragHandler Maven / Gradle / Ivy
/* Copyright (c) 2007 Timothy Wall, All Rights Reserved
*
* The contents of this file is dual-licensed under 2
* alternative Open Source/Free licenses: LGPL 2.1 or later and
* Apache License 2.0. (starting with JNA version 4.0.0).
*
* You can freely decide which license you want to apply to
* the project.
*
* You may obtain a copy of the LGPL License at:
*
* http://www.gnu.org/licenses/licenses.html
*
* A copy is also included in the downloadable source code package
* containing JNA, in file "LGPL2.1".
*
* You may obtain a copy of the Apache License at:
*
* http://www.apache.org/licenses/
*
* A copy is also included in the downloadable source code package
* containing JNA, in file "AL2.0".
*/
package com.sun.jna.platform.dnd;
import com.sun.jna.Platform;
import java.awt.AlphaComposite;
import java.awt.Component;
import java.awt.Cursor;
import java.awt.Dimension;
import java.awt.Graphics2D;
import java.awt.GraphicsConfiguration;
import java.awt.Image;
import java.awt.Point;
import java.awt.Transparency;
import java.awt.datatransfer.Transferable;
import java.awt.dnd.DnDConstants;
import java.awt.dnd.DragGestureEvent;
import java.awt.dnd.DragGestureListener;
import java.awt.dnd.DragSource;
import java.awt.dnd.DragSourceContext;
import java.awt.dnd.DragSourceDragEvent;
import java.awt.dnd.DragSourceDropEvent;
import java.awt.dnd.DragSourceEvent;
import java.awt.dnd.DragSourceListener;
import java.awt.dnd.DragSourceMotionListener;
import java.awt.dnd.DropTargetDragEvent;
import java.awt.dnd.DropTargetDropEvent;
import java.awt.dnd.DropTargetEvent;
import java.awt.dnd.InvalidDnDOperationException;
import java.awt.event.InputEvent;
import java.awt.image.BufferedImage;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.Icon;
import javax.swing.JColorChooser;
import javax.swing.JFileChooser;
import javax.swing.JList;
import javax.swing.JTable;
import javax.swing.JTree;
import javax.swing.text.JTextComponent;
/** Provides simplified drag handling for a component.
* Usage:
*
* int actions = DnDConstants.MOVE_OR_COPY;
* Component component = ...;
* DragHandler handler = new DragHandler(component, actions);
*
*
* - Supports painting an arbitrary {@link Icon} with transparency to
* represent the item being dragged (restricted to the {@link java.awt.Window}
* of the drag source if the platform doesn't support drag images).
*
- Disallow starting a drag if the user requests an unsupported action.
*
- Adjusts the cursor on drags with no modifier for which the default action
* is disallowed but where one or more non-default actions are allowed, e.g. a
* drag (with no modifiers) to a target which supports "link" should change the
* cursor to "link" (prior to 1.6, the JRE behavior is to display a
* "not allowed" cursor, even though the action actually depends on how the
* drop target responds).
*
* The bug is fixed in java 1.6.
*
- Disallow drops to targets if the non-default (user-requested) action
* is not supported by the target, e.g. the user requests a "copy" when the
* target only supports "move". This is generally the responsibility of the
* drop handler, which decides whether or not to accept a drag. The DragHandler
* provides static modifier state information since the drop handler doesn't
* have access to it.
*
* NOTE: Fundamentally, the active action is determined by the drop handler
* in {@link DropTargetDragEvent#acceptDrag}, but often client code
* simply relies on {@link DropTargetDragEvent#getDropAction}.
*/
//TODO: separate into the the following drag pieces:
// * default cursor changing behavior fix (fixed in jre1.6) (global DSL)
// * drag unselected item (fixed in 1.6, flag in 1.5.0.05+) bug not an issue w/DragHandler
// * multi-selection drag workaround (if not using swing drag gesture recognizer)
// proper multi-selection drag is provided for swing dnd as of 1.4
// (use swing drag gesture recognizer if possible?) DSL+mouse listener
//TODO: identify overlap with default Swing support + TransferHandler
// * TransferHandler doesn't have drop location information (included in 1.6)
// * SwingDropTarget allows listeners, so target highlighting is possible
//NOTE: acceptDrag(int) is simply indicates the drag is accepted; the action
// value doesn't get propagated anywhere, it's certainly not communicated to
// the drag source
//MAYBE: should Transferable/Icon provision be a separate interface
//(i.e. TransferHandler)? only if the rest of the drag handler is constant and
// only the image needs to change (for standard components, e.g. tree cells,
// table cells, etc.)
public abstract class DragHandler
implements DragSourceListener, DragSourceMotionListener, DragGestureListener {
private static final Logger LOG = Logger.getLogger(DragHandler.class.getName());
/** Default maximum size for ghosted images. */
public static final Dimension MAX_GHOST_SIZE = new Dimension(250, 250);
/** Default transparency for ghosting. */
public static final float DEFAULT_GHOST_ALPHA = 0.5f;
/** {@link #getModifiers} returns this value when the current
* modifiers state is unknown.
*/
public static final int UNKNOWN_MODIFIERS = -1;
/** {@link #getTransferable} returns this value when
* the current {@link Transferable} is unknown.
*/
public static final Transferable UNKNOWN_TRANSFERABLE = null;
/** Convenience to reference {@link DnDConstants#ACTION_MOVE}. */
protected static final int MOVE = DnDConstants.ACTION_MOVE;
/** Convenience to reference {@link DnDConstants#ACTION_COPY}. */
protected static final int COPY = DnDConstants.ACTION_COPY;
/** Convenience to reference {@link DnDConstants#ACTION_LINK}. */
protected static final int LINK = DnDConstants.ACTION_LINK;
/** Convenience to reference {@link DnDConstants#ACTION_NONE}. */
protected static final int NONE = DnDConstants.ACTION_NONE;
// TODO: w32 explorer: link=alt or ctrl+shift, copy=ctrl or shift
// w32 others: copy=ctrl
/** Modifier mask for a user-requested move. */
static final int MOVE_MASK = InputEvent.SHIFT_DOWN_MASK;
static final boolean OSX = Platform.isMac();
/** Modifier mask for a user-requested copy. */
static final int COPY_MASK =
OSX ? InputEvent.ALT_DOWN_MASK : InputEvent.CTRL_DOWN_MASK;
/** Modifier mask for a user-requested link. */
static final int LINK_MASK =
OSX ? InputEvent.ALT_DOWN_MASK|InputEvent.META_DOWN_MASK
: InputEvent.CTRL_DOWN_MASK|InputEvent.SHIFT_DOWN_MASK;
/** Modifier mask for any user-requested action. */
static final int KEY_MASK =
InputEvent.ALT_DOWN_MASK|InputEvent.META_DOWN_MASK
|InputEvent.CTRL_DOWN_MASK|InputEvent.SHIFT_DOWN_MASK
|InputEvent.ALT_GRAPH_DOWN_MASK;
private static int modifiers = UNKNOWN_MODIFIERS;
private static Transferable transferable = UNKNOWN_TRANSFERABLE;
/** Used to communicate modifier state to {@link DropHandler}. Note that
* this field will only be accurate when a {@link DragHandler} in
* the same VM started the drag. Otherwise, {@link #UNKNOWN_MODIFIERS}
* will be returned.
* @return Current drag modifiers.
*/
static int getModifiers() {
return modifiers;
}
/** Used to communicate the current {@link Transferable} during a drag,
* if available. Work around absence of access to the data when dragging
* pre-1.5.
* @param e event
* @return {@link Transferable} representation of the item being dragged.
*/
public static Transferable getTransferable(DropTargetEvent e) {
if (e instanceof DropTargetDragEvent) {
try {
return ((DropTargetDragEvent) e).getTransferable();
} catch (Exception ex) {
// Method not available
}
} else if (e instanceof DropTargetDropEvent) {
return ((DropTargetDropEvent) e).getTransferable();
}
return transferable;
}
private int supportedActions;
private boolean fixCursor = true;
private Component dragSource;
private GhostedDragImage ghost;
private Point imageOffset;
private Dimension maxGhostSize = MAX_GHOST_SIZE;
private float ghostAlpha = DEFAULT_GHOST_ALPHA;
/** Enable drags from the given component, supporting the actions in
* the given action mask.
* @param dragSource source of the drag.
* @param actions actions which should be supported.
*/
protected DragHandler(Component dragSource, int actions) {
this.dragSource = dragSource;
this.supportedActions = actions;
try {
String alpha = System.getProperty("DragHandler.alpha");
if (alpha != null) {
try {
ghostAlpha = Float.parseFloat(alpha);
}
catch(NumberFormatException e) { }
}
String max = System.getProperty("DragHandler.maxDragImageSize");
if (max != null) {
String[] size = max.split("x");
if (size.length == 2) {
try {
maxGhostSize = new Dimension(Integer.parseInt(size[0]),
Integer.parseInt(size[1]));
}
catch(NumberFormatException e) { }
}
}
}
catch(SecurityException e) { }
// Avoid having more than one gesture recognizer active
disableSwingDragSupport(dragSource);
DragSource src = DragSource.getDefaultDragSource();
src.createDefaultDragGestureRecognizer(dragSource, supportedActions, this);
}
private void disableSwingDragSupport(Component comp) {
if (comp instanceof JTree) {
((JTree)comp).setDragEnabled(false);
}
else if (comp instanceof JList) {
((JList)comp).setDragEnabled(false);
}
else if (comp instanceof JTable) {
((JTable)comp).setDragEnabled(false);
}
else if (comp instanceof JTextComponent) {
((JTextComponent)comp).setDragEnabled(false);
}
else if (comp instanceof JColorChooser) {
((JColorChooser)comp).setDragEnabled(false);
}
else if (comp instanceof JFileChooser) {
((JFileChooser)comp).setDragEnabled(false);
}
}
/** Override to control whether a drag is started. The default
* implementation disallows the drag if the user is applying modifiers
* and the user-requested action is not supported.
* @param e event
* @return Whether to allow a drag
*/
protected boolean canDrag(DragGestureEvent e) {
int mods = e.getTriggerEvent().getModifiersEx() & KEY_MASK;
if (mods == MOVE_MASK)
return (supportedActions & MOVE) != 0;
if (mods == COPY_MASK)
return (supportedActions & COPY) != 0;
if (mods == LINK_MASK)
return (supportedActions & LINK) != 0;
return true;
}
/** Update the modifiers hint.
* @param mods Current modifiers
*/
protected void setModifiers(int mods) {
modifiers = mods;
}
/** Override to provide an appropriate {@link Transferable} representing
* the data being dragged.
* @param e event
* @return {@link Transferable} representation of item being dragged.
*/
protected abstract Transferable getTransferable(DragGestureEvent e);
/** Override this to provide a custom image. The {@link Icon}
* returned by this method by default is null
, which results
* in no drag image.
* @param e event
* @param srcOffset set this to be the offset from the drag source
* component's upper left corner to the image's upper left corner.
* For example, when dragging a row from a list, the offset would be the
* row's bounding rectangle's (x,y) coordinate.
* The default value is (0,0), so if unchanged, the image is will
* use the same origin as the drag source component.
* @return drag icon (defaults to none)
*/
protected Icon getDragIcon(DragGestureEvent e, Point srcOffset) {
return null;
}
/** Override to perform any decoration of the target at the start of a drag,
* if desired.
* @param e event
*/
protected void dragStarted(DragGestureEvent e) { }
/** Called when a user drag gesture is recognized. This method is
* responsible for initiating the drag operation.
* @param e event
*/
@Override
public void dragGestureRecognized(DragGestureEvent e) {
if ((e.getDragAction() & supportedActions) != 0
&& canDrag(e)) {
setModifiers(e.getTriggerEvent().getModifiersEx() & KEY_MASK);
Transferable transferable = getTransferable(e);
if (transferable == null)
return;
try {
Point srcOffset = new Point(0, 0);
Icon icon = getDragIcon(e, srcOffset);
Point origin = e.getDragOrigin();
// offset of the image origin from the cursor
imageOffset = new Point(srcOffset.x - origin.x,
srcOffset.y - origin.y);
Icon dragIcon = scaleDragIcon(icon, imageOffset);
Cursor cursor = null;
if (dragIcon != null && DragSource.isDragImageSupported()) {
GraphicsConfiguration gc = e.getComponent().getGraphicsConfiguration();
e.startDrag(cursor, createDragImage(gc, dragIcon),
imageOffset, transferable, this);
}
else {
if (dragIcon != null) {
Point screen = dragSource.getLocationOnScreen();
screen.translate(origin.x, origin.y);
Point cursorOffset = new Point(-imageOffset.x, -imageOffset.y);
ghost = new GhostedDragImage(dragSource, dragIcon,
getImageLocation(screen), cursorOffset);
ghost.setAlpha(ghostAlpha);
}
e.startDrag(cursor, transferable, this);
}
dragStarted(e);
moved = false;
e.getDragSource().addDragSourceMotionListener(this);
DragHandler.transferable = transferable;
}
catch (InvalidDnDOperationException ex) {
if (ghost != null) {
ghost.dispose();
ghost = null;
}
}
}
}
/** Change the size of the given drag icon, if appropriate. When using
* a differently-sized drag icon, we also need to adjust the cursor offset within
* the icon.
* @param icon Icon to be scaled.
* @param imageOffset Modified to account for the new icon's size.
* @return Scaled {@link Icon}, or the original if there was no change.
*/
protected Icon scaleDragIcon(Icon icon, Point imageOffset) {
/*
if (icon != null && maxGhostSize != null) {
if (icon.getIconWidth() > maxGhostSize.width
|| icon.getIconHeight() > maxGhostSize.height) {
Icon scaled = new ScaledIcon(icon, maxGhostSize.width,
maxGhostSize.height);
double scale = (double)scaled.getIconWidth()/icon.getIconWidth();
imageOffset.x *= scale;
imageOffset.y *= scale;
return scaled;
}
}*/
return icon;
}
/** Create an image from the given icon. The image is provided to the
* native handler if drag images are supported natively.
* @param gc current graphics configuration.
* @param icon Icon on which to base the drag image.
* @return image based on the given icon.
*/
protected Image createDragImage(GraphicsConfiguration gc, Icon icon) {
int w = icon.getIconWidth();
int h = icon.getIconHeight();
BufferedImage image = gc.createCompatibleImage(w, h, Transparency.TRANSLUCENT);
Graphics2D g = (Graphics2D)image.getGraphics();
g.setComposite(AlphaComposite.Clear);
g.fillRect(0, 0, w, h);
// Ignore pixels in the buffered image
g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, ghostAlpha));
icon.paintIcon(dragSource, g, 0, 0);
g.dispose();
return image;
}
/** Reduce a multiply-set bit mask to a single bit. */
private int reduce(int actions) {
if ((actions & MOVE) != 0 && actions != MOVE) {
return MOVE;
}
else if ((actions & COPY) != 0 && actions != COPY) {
return COPY;
}
return actions;
}
protected Cursor getCursorForAction(int actualAction) {
switch(actualAction) {
case MOVE:
return DragSource.DefaultMoveDrop;
case COPY:
return DragSource.DefaultCopyDrop;
case LINK:
return DragSource.DefaultLinkDrop;
default:
return DragSource.DefaultMoveNoDrop;
}
}
/** Returns the first available action supported by source and target.
* @param targetActions current actions requested
* @return subset of actions supported based on the input
*/
protected int getAcceptableDropAction(int targetActions) {
return reduce(supportedActions & targetActions);
}
/** Get the currently requested drop action.
* @param ev event
* @return effective drop action
*/
protected int getDropAction(DragSourceEvent ev) {
if (ev instanceof DragSourceDragEvent) {
DragSourceDragEvent e = (DragSourceDragEvent)ev;
return e.getDropAction();
}
if (ev instanceof DragSourceDropEvent) {
return ((DragSourceDropEvent)ev).getDropAction();
}
return NONE;
}
/** Pick a different drop action if the target doesn't support the current
* one and there are no modifiers.
* @param ev event
* @return effective drop action
*/
protected int adjustDropAction(DragSourceEvent ev) {
int action = getDropAction(ev);
if (ev instanceof DragSourceDragEvent) {
DragSourceDragEvent e = (DragSourceDragEvent)ev;
if (action == NONE) {
int mods = e.getGestureModifiersEx() & KEY_MASK;
if (mods == 0) {
action = getAcceptableDropAction(e.getTargetActions());
}
}
}
return action;
}
/**
* Hook to update the cursor on various {@link DragSourceEvent} updates.
* @param ev event
*/
protected void updateCursor(DragSourceEvent ev) {
if (!fixCursor)
return;
Cursor cursor = getCursorForAction(adjustDropAction(ev));
ev.getDragSourceContext().setCursor(cursor);
}
static String actionString(int action) {
switch(action) {
case MOVE: return "MOVE";
case MOVE|COPY: return "MOVE|COPY";
case MOVE|LINK: return "MOVE|LINK";
case MOVE|COPY|LINK: return "MOVE|COPY|LINK";
case COPY: return "COPY";
case COPY|LINK: return "COPY|LINK";
case LINK: return "LINK";
default: return "NONE";
}
}
private String lastAction;
private void describe(String type, DragSourceEvent e) {
if (LOG.isLoggable(Level.FINE)) {
StringBuilder msgBuilder = new StringBuilder();
msgBuilder.append("drag: ");
msgBuilder.append(type);
DragSourceContext ds = e.getDragSourceContext();
if (e instanceof DragSourceDragEvent) {
DragSourceDragEvent ev = (DragSourceDragEvent)e;
msgBuilder.append(": src=");
msgBuilder.append(actionString(ds.getSourceActions()));
msgBuilder.append(" usr=");
msgBuilder.append(actionString(ev.getUserAction()));
msgBuilder.append(" tgt=");
msgBuilder.append(actionString(ev.getTargetActions()));
msgBuilder.append(" act=");
msgBuilder.append(actionString(ev.getDropAction()));
msgBuilder.append(" mods=");
msgBuilder.append(ev.getGestureModifiersEx());
}
else {
msgBuilder.append(": e=");
msgBuilder.append(e);
}
String msg = msgBuilder.toString();
if (!msg.equals(lastAction)) {
LOG.log(Level.FINE, msg);
lastAction = msg;
}
}
}
@Override
public void dragDropEnd(DragSourceDropEvent e) {
describe("end", e);
setModifiers(UNKNOWN_MODIFIERS);
transferable = UNKNOWN_TRANSFERABLE;
if (ghost != null) {
if (e.getDropSuccess()) {
ghost.dispose();
}
else {
ghost.returnToOrigin();
}
ghost = null;
}
DragSource src = e.getDragSourceContext().getDragSource();
src.removeDragSourceMotionListener(this);
moved = false;
}
private Point getImageLocation(Point where) {
where.translate(imageOffset.x, imageOffset.y);
return where;
}
@Override
public void dragEnter(DragSourceDragEvent e) {
describe("enter", e);
if (ghost != null) {
ghost.move(getImageLocation(e.getLocation()));
}
updateCursor(e);
}
// bug workaround; need to skip initial dragMouseMoved event,
// which has reports "0" for the available target actions (1.4+?)
// filed a bug for this
private boolean moved;
@Override
public void dragMouseMoved(DragSourceDragEvent e) {
describe("move", e);
if (ghost != null) {
ghost.move(getImageLocation(e.getLocation()));
}
if (moved)
updateCursor(e);
moved = true;
}
@Override
public void dragOver(DragSourceDragEvent e) {
describe("over", e);
if (ghost != null) {
ghost.move(getImageLocation(e.getLocation()));
}
updateCursor(e);
}
@Override
public void dragExit(DragSourceEvent e) {
describe("exit", e);
}
@Override
public void dropActionChanged(DragSourceDragEvent e) {
describe("change", e);
setModifiers(e.getGestureModifiersEx() & KEY_MASK);
if (ghost != null) {
ghost.move(getImageLocation(e.getLocation()));
}
updateCursor(e);
}
}