I tried to fix this issue for the past week but somehow I cant seem to find a solution. There is not a lot of information about this topic so its hard to find examples or code to look at.
What I have here is a JList which uses a custom TransferHandler that creates a custom Transferable, for reference here's the code of the involved classes:
Transferable:
package org.dinhware.swing.special;
import java.awt.datatransfer.DataFlavor;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.io.IOException;
/**
* Created by: Niklas
* Date: 20.10.2017
* Alias: Dinh
* Time: 20:03
*/
public class GenericTransferable<T> implements Transferable {
static DataFlavor FLAVOR;
private T object;
GenericTransferable(T object) {
GenericTransferable.FLAVOR = new DataFlavor(object.getClass(), object.getClass().getCanonicalName());
this.object = object;
}
@Override
public DataFlavor[] getTransferDataFlavors() {
return new DataFlavor[]{FLAVOR};
}
@Override
public boolean isDataFlavorSupported(DataFlavor flavor) {
return flavor.equals(FLAVOR);
}
@Override
public Object getTransferData(DataFlavor flavor) throws UnsupportedFlavorException, IOException {
return object;
}
}
TransferHandler:
package org.dinhware.swing.special;
import javax.swing.*;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.io.IOException;
/**
* Created by: Niklas
* Date: 19.10.2017
* Alias: Dinh
* Time: 18:54
*/
@SuppressWarnings("unchecked")
public class HListItemTransferHandler<T> extends TransferHandler {
@Override
protected Transferable createTransferable(JComponent component) {
JList<T> list = (JList<T>) component;
index = list.getSelectedIndex();
T transferredObject = list.getSelectedValue();
return new GenericTransferable<>(transferredObject);
}
@Override
public boolean canImport(TransferSupport info) {
return info.isDataFlavorSupported(GenericTransferable.FLAVOR);
}
@Override
public int getSourceActions(JComponent c) {
return MOVE;
}
@Override
public boolean importData(TransferSupport info) {
if (!canImport(info)) {
return false;
}
JList<Object> target = (JList<Object>) info.getComponent();
JList.DropLocation dl = (JList.DropLocation) info.getDropLocation();
DefaultListModel<Object> listModel = (DefaultListModel<Object>) target.getModel();
int index = dl.getIndex();
int max = listModel.getSize();
if (index < 0 || index > max)
index = max;
addIndex = index;
try {
Object object = info.getTransferable().getTransferData(GenericTransferable.FLAVOR);
listModel.add(index, object);
target.addSelectionInterval(index, index);
return moveAllowed = true;
} catch (UnsupportedFlavorException | IOException e) {
e.printStackTrace();
}
return false;
}
@Override
protected void exportDone(JComponent c, Transferable data, int action) {
if (moveAllowed)
cleanup(c, action == MOVE, false);
}
private void cleanup(JComponent component, boolean remove, boolean bin) {
if (remove && index != -1) {
JList<T> source = (JList<T>) component;
DefaultListModel<T> model = (DefaultListModel<T>) source.getModel();
int removeAt = index > addIndex ? index + 1 : index;
model.remove(bin ? removeAt - 1 : removeAt);
}
index = -1;
addIndex = -1;
moveAllowed = false;
}
private int index = -1;
private int addIndex = -1;
private boolean moveAllowed = false;
}
HBin
package org.dinhware.swing.child;
import org.dinhware.swing.special.HListItemTransferHandler;
import javax.swing.*;
/**
* Created by: Niklas
* Date: 20.10.2017
* Alias: Dinh
* Time: 19:57
*/
public class HBin<T> extends HImageLabel {
public HBin(String text, Icon image, int distance) {
super(text, image, distance);
setTransferHandler(new HListItemTransferHandler<T>());
}
}
And a visualization of how it should work, sadly the container always disappears even when not dragged onto the HBin Container. I thought it was working the entire time until I accidentally moved it outside of my Frame and it still disappeared. The Code above only allows dragging/dropping inside of the List which is intended.
My Question is how to add the functionality to properly make a Container only disappear when dragged onto the HBin Container
The first part of Code I used was this
@Override
protected void exportDone(JComponent c, Transferable data, int action) {
if (moveAllowed) cleanup(c, action == MOVE, false);
else try {
if (data.getTransferData(GenericTransferable.FLAVOR) instanceof RewardItem) {
cleanup(c, true, true);
}
} catch (UnsupportedFlavorException | IOException e) {
e.printStackTrace();
}
}
My logic behind it was that both the List and HBin shared the same Type (RewardItem) which I could compare, later I realized (after making a more generic version of that method) that data will always be of the type RewardItem and will always result in a cleanup call. This results in the bug I am currently still facing.
The approach I took earlier today really made me question my mind and also made me do this post. I added a boolean to the TransferHandler called bin which was false by default. After the canImport check in importData I added bin = info.getComponent() instanceof HBin
which I thought should work. But this field always stood false. I went ahead and added a log for it
System.out.println("IMPORT");
if (info.getComponent() instanceof HBin) {
System.out.println("bin");
return bin = true;
}
It ended up printing IMPORT followed by bin. After importData exportData is called in which I then logged the value of bin, which for whatever reason was now false again. Meanwhile the moveAllowed field seems to change.
This was my full modified TransferHandler
package org.dinhware.swing.special;
import org.dinhware.swing.child.HBin;
import javax.swing.*;
import java.awt.datatransfer.Transferable;
import java.awt.datatransfer.UnsupportedFlavorException;
import java.io.IOException;
/**
* Created by: Niklas Date: 19.10.2017 Alias: Dinh Time: 18:54
*/
@SuppressWarnings("unchecked")
public class HListItemTransferHandler<T> extends TransferHandler {
@Override
protected Transferable createTransferable(JComponent component) {
System.out.println("CREATE");
JList<T> list = (JList<T>) component;
index = list.getSelectedIndex();
T transferredObject = list.getSelectedValue();
return new GenericTransferable<>(transferredObject);
}
@Override
public boolean canImport(TransferSupport info) {
return info.isDataFlavorSupported(GenericTransferable.FLAVOR);
}
@Override
public int getSourceActions(JComponent c) {
System.out.println("ACTION");
return MOVE;
}
@Override
public boolean importData(TransferSupport info) {
System.out.println("IMPORT");
if (!canImport(info)) {
return false;
}
if (info.getComponent() instanceof HBin) {
System.out.println("bin");
return bin = true;
}
JList<Object> target = (JList<Object>) info.getComponent();
JList.DropLocation dl = (JList.DropLocation) info.getDropLocation();
DefaultListModel<Object> listModel = (DefaultListModel<Object>) target.getModel();
int index = dl.getIndex();
int max = listModel.getSize();
if (index < 0 || index > max)
index = max;
addIndex = index;
try {
Object object = info.getTransferable().getTransferData(GenericTransferable.FLAVOR);
listModel.add(index, object);
target.addSelectionInterval(index, index);
return moveAllowed = true;
} catch (UnsupportedFlavorException | IOException e) {
e.printStackTrace();
}
return false;
}
@Override
protected void exportDone(JComponent c, Transferable data, int action) {
System.out.println("EXPORT " + moveAllowed + "/" + bin);
if (moveAllowed)
cleanup(c, action == MOVE, false);
else
cleanup(c, true, true);
}
private void cleanup(JComponent component, boolean remove, boolean bin) {
System.out.println("CLEAN");
if (remove && index != -1) {
JList<T> source = (JList<T>) component;
DefaultListModel<T> model = (DefaultListModel<T>) source.getModel();
int removeAt = index > addIndex ? index + 1 : index;
model.remove(bin ? removeAt - 1 : removeAt);
}
index = -1;
addIndex = -1;
moveAllowed = false;
}
private int index = -1;
private int addIndex = -1;
private boolean moveAllowed = false, bin = false;
}
When moving inside of the List everything works fine (prints)
ACTION
CREATE
IMPORT
EXPORT true/false
CLEAN
But when dropping onto the HBin Container I cant explain whats going on (prints)
ACTION
CREATE
IMPORT
bin
EXPORT false/false
I am farily sure that it should be false/true
Now I am stuck, not able to make the Container only disappear when dropped onto HBin whilst also being confused about the field value not changing when it clearly logs that it has been set to true.
Please.. help...
Drag'n'Drop is complicated and it's not helped by the fact that there are at least two ways to do it.
D'n'D revolves around the idea of "wrapping" an object up in a "transferable" package, which can be "imported" via a number of different means (i.e. DataFlavor
s)
So, in this example, I've focused only on removing items from the JList
, to do this, I created a Trash
object which actually maintains a reference to the item to be removed (I also created a ListTrash
object to demonstrate at least one way you could pass more information)
This object is then wrapped in a TrashTransferable
when a drag occurs on the JList
The main reason for having a Trash
object, it it allows the DataFlavor
to be standardised. The "trash can" only cares about Trash
objects, nothing else. This is especially important if you have more TransferHandler
s doing more operations
One other thing I did was create two TransferHandler
s. One for the "trash can" and one for the JList
, the main reason for this is it isolates the functionality each handler wants to perform and reduces the complexity, as you're not also trying to determine which object is trying to perform which operation.
The example also has another component which is not doing much at all, so it can reject the drop operation.
If you have another components using TransferHandler
s, then those need to reject the TrashTransferable.FLAVOR
import java.awt.Dimension;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.Insets;
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 javax.swing.DefaultListModel;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.SwingUtilities;
import javax.swing.TransferHandler;
import javax.swing.TransferHandler.TransferSupport;
public class Test {
public static void main(String[] args) {
new Test();
}
public Test() {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
JFrame frame = new JFrame("Test");
frame.add(new TestPane());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public class TestPane extends JPanel {
public TestPane() {
setLayout(new GridBagLayout());
DefaultListModel<String> model = new DefaultListModel<>();
model.addElement("Cooks_Assistant");
model.addElement("Romeo_and_Juliet");
model.addElement("Sheep_Shearer");
JList list = new JList(model);
list.setTransferHandler(new HListItemTransferHandler());
list.setDragEnabled(true);
JLabel noDrop = new JLabel("No drop here", JLabel.CENTER);
JLabel trash = new JLabel("All your trash belong to us", JLabel.CENTER);
trash.setTransferHandler(new BinTransferHandler());
GridBagConstraints gbc = new GridBagConstraints();
gbc.gridx = 0;
gbc.gridy = 0;
gbc.weightx = 0.5;
gbc.weighty = 1;
gbc.fill = GridBagConstraints.BOTH;
gbc.insets = new Insets(4, 4, 4, 4);
add(new JScrollPane(list), gbc);
gbc.gridx++;
add(noDrop, gbc);
gbc.gridx = 0;
gbc.gridy++;
gbc.gridwidth = GridBagConstraints.REMAINDER;
add(trash, gbc);
}
@Override
public Dimension getPreferredSize() {
return new Dimension(300, 300);
}
}
public class BinTransferHandler extends TransferHandler {
@Override
public boolean canImport(TransferSupport info) {
return info.isDataFlavorSupported(TrashTransferable.FLAVOR);
}
@Override
public int getSourceActions(JComponent c) {
System.out.println("ACTION");
return DnDConstants.ACTION_MOVE;
}
@Override
public boolean importData(TransferSupport support) {
if (!canImport(support)) {
return false;
}
// Check target component
Transferable transferable = support.getTransferable();
try {
Trash trash = (Trash) transferable.getTransferData(TrashTransferable.FLAVOR);
Object item = trash.getItem();
System.out.println(">> Trash " + item);
return true;
} catch (UnsupportedFlavorException | IOException ex) {
ex.printStackTrace();
}
return false;
}
}
public class HListItemTransferHandler<T> extends TransferHandler {
@Override
protected Transferable createTransferable(JComponent component) {
System.out.println("createTransferable");
JList<T> list = (JList<T>) component;
int index = list.getSelectedIndex();
T transferredObject = list.getSelectedValue();
return new TrashTransferable(new ListTrash<>(list, index, transferredObject));
}
@Override
public boolean canImport(TransferSupport info) {
return info.isDataFlavorSupported(TrashTransferable.FLAVOR);
}
@Override
public int getSourceActions(JComponent c) {
return DnDConstants.ACTION_MOVE;
}
@Override
public boolean importData(TransferSupport info) {
JList<Object> target = (JList<Object>) info.getComponent();
JList.DropLocation dl = (JList.DropLocation) info.getDropLocation();
DefaultListModel<Object> listModel = (DefaultListModel<Object>) target.getModel();
int index = dl.getIndex();
int max = listModel.getSize();
if (index < 0 || index > max) {
index = max;
}
try {
Object object = info.getTransferable().getTransferData(DataFlavor.stringFlavor);
listModel.add(index, object);
target.addSelectionInterval(index, index);
return true;
} catch (UnsupportedFlavorException | IOException e) {
e.printStackTrace();
}
return false;
}
@Override
protected void exportDone(JComponent c, Transferable data, int action) {
System.out.println("Export data");
try {
if (action != MOVE) {
return;
}
if (!(c instanceof JList)) {
return;
}
JList list = (JList) c;
if (!(list.getModel() instanceof DefaultListModel)) {
return;
}
DefaultListModel model = (DefaultListModel) list.getModel();
if (!(data instanceof TrashTransferable)) {
return;
}
Object transferData = data.getTransferData(TrashTransferable.FLAVOR);
if (transferData == null || !(transferData instanceof Trash)) {
return;
}
Trash trash = (Trash) transferData;
Object item = trash.item;
int index = model.indexOf(item);
if (index == -1) {
return;
}
model.remove(index);
} catch (UnsupportedFlavorException | IOException ex) {
ex.printStackTrace();
}
}
}
public static class ListTrash<T> extends Trash<T> {
private JList list;
private int index;
public ListTrash(JList list, int index, T item) {
super(item);
this.list = list;
this.index = index;
}
public JList getList() {
return list;
}
public int getIndex() {
return index;
}
}
public static class Trash<T> {
private T item;
public Trash(T item) {
this.item = item;
}
public T getItem() {
return item;
}
}
public static class TrashTransferable<T> implements Transferable {
public static final DataFlavor FLAVOR = new DataFlavor(Trash.class, "Trash");
private Trash<T> trash;
TrashTransferable(Trash<T> object) {
trash = object;
}
@Override
public DataFlavor[] getTransferDataFlavors() {
return new DataFlavor[]{FLAVOR};
}
@Override
public boolean isDataFlavorSupported(DataFlavor flavor) {
return flavor.equals(flavor);
}
@Override
public Object getTransferData(DataFlavor flavor) throws UnsupportedFlavorException, IOException {
return trash;
}
}
}