The JavaTM Tutorial
Previous Page Lesson Contents Next Page Start of Tutorial > Start of Trail > Start of Lesson Search
Feedback Form

Trail: Creating a GUI with JFC/Swing
Lesson: Using Other Swing Features

How to Use Drag and Drop and Data Transfer

Most programs can benefit from the ability to transfer information, either between components, between Java applications, or between Java and native applications. The ability to transfer data takes two forms:

The TransferHandler object, the heart of the data transfer system, is described in more detail later. The arrows in the preceding diagrams show the path of the data.

Many Swing components provide out-of-the-box support for transferring data, as shown in the following table:

Data Transfer Support
Component Drag*
Copy
Drag*
Move
Drop Cut Copy Paste
JColorChooser**

checked

 

checked

     
JEditorPane

checked

checked

checked

checked

checked

checked

JFileChooser***

checked

     

checked

 
JFormattedTextField

checked

checked

checked

checked

checked

checked

JList

checked

     

checked

 
JPasswordField n/a n/a

checked

n/a n/a

checked

JTable

checked

     

checked

 
JTextArea

checked

checked

checked

checked

checked

checked

JTextField

checked

checked

checked

checked

checked

checked

JTextPane

checked

checked

checked

checked

checked

checked

JTree

checked

     

checked

 
  *Enabled by invoking component.setDragEnabled(true) on the component.
 **Imports and exports data of type java.awt.Color.
***Exports both a list of filenames as java.io.File objects (preferred) and as strings for those components that do not accept File objects. The File Name text field in the file chooser accepts strings; the browser in the file chooser does not accept data. Note that as of release 1.4, clipboard copy from a JFileChooser is broken and actually causes the file to be moved when it is pasted. You may want to watch bug #4915992 (outside of the tutorial).


Version note: This section describes the drag and drop architecture implemented in release 1.4. Prior to 1.4, AWT support for data transfer was not well integrated into Swing components.

The data transfer mechanism is built in to every JComponent. For all the components with an empty space in the preceding table only a small amount of code is needed to customize the support. Support can easily be added to JComponents not listed in the table so they can fully participate in data transfer.

The rest of this section covers the following topics:

A Visual Guide to Drag and Drop Cursor Icons

Before delving into drag and drop further, it's useful to take a look at the various cursor icons you may encounter when initiating a drag operation. We expect the Solaris and Linux cursor icons to change for release 1.5, but here is a guide as of release 1.4:

Cursor Icons for Drag and Drop

Microsoft
Windows
Solaris/
Linux
Description
Copy cursor icon for Windows, the component below accepts the drop. Copy cursor icon for Linux/Solaris, the component below accepts the drop. Copy. The component underneath accepts this type of data.
Copy cursor icon for Windows, the component below won't accept. Copy cursor icon for Linux/Solaris, the component below won't accept. Copy. The area underneath will not accept this data.
Move cursor icon for Windows, the component accepts the drop. Move cursor icon for Linux/Solaris, the component accepts the drop. Move. The component underneath accepts this type of data.
Move cursor icon for Windows, the component won't accept. Move cursor icon for Linux/Solaris, the component won't accept. Move. The area underneath will not accept this data.

On a Component supporting both Copy and Move, a normal drag from the component performs a move and a Control-drag performs a copy. The drag behavior from a native application to a Java application is platform dependent. If only one of the operations is supported, a normal drag performs that operation. For more information on the behavior of the drop action, see the class spec for DragSourceDragEvent (in the API reference documentation).

Introduction to Data Transfer Support

The simple demo BasicDnD illustrates default drag and drop behavior for several Swing components. At startup the components do not have drag turned on, but a check box allows you to enable dragging on the fly. Note that at startup, even though drag is not yet enabled, many of the components do support the cut/copy/paste of text using key bindings.


This figure has been reduced to fit on the page.
Click the image to view it at its natural size.


Try this: 
  1. Run BasicDnD using JavaTM Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Select an item in the list, and then release the mouse button so it is now highlighted.
  3. Press the item again, this time holding down the mouse button and begin to drag. Nothing happens because setDragEnabled(true) has not yet been called on the list.
  4. Type Control-C. This puts the text of the selected list item onto the system clipboard.
  5. Click in the text area. The caret cursor blinks showing that this component now has the focus.
  6. Type Control-V. The contents of the previously copied text are pasted at the caret location.
  7. Click the "Turn on Drag and Drop" check box.
  8. Once again, press the selected item in the list and begin to drag. Initially, on Microsoft Windows, you see the Copy cursor icon for Windows, the component won't accept. cursor icon. This indicates that the area below the cursor, in this case the list itself, does not accept drops.
  9. Drag the cursor over a text area. The cursor icon now changes to Copy cursor icon for Windows, the component below accepts the drop.. This text area will accept the data if you release the mouse button.
  10. As you can see in the cursor icon table, the Copy cursor icon gives you a visual clue that it will not disturb the contents of the original component. In the case of Microsoft Windows, the clue is the small + sign.
  11. Drag the selected text over a text area. The insertion point for the text is indicated by a blinking cursor.
  12. Release the mouse and watch the text appear in the text area.
  13. Select some text in one of the text areas.
  14. Press and hold the mouse button while the cursor is over the selected text and begin to drag.
  15. Note that this time, the Move cursor icon for Windows, the component accepts the drop. icon appears. This indicates that the drag is a Move and will remove the text from the original component on a successful drop.
  16. Release the mouse button over an area that will not accept it. The original text is undisturbed.
  17. Hold the Control key down and press again on the selected text. The Copy icon now appears. Move the cursor over the text area and drop. The text appears in the new location but is not removed from the original location. The Control key can be used to change any Move to a Copy.
  18. Select a color from the color chooser. The selected color appears in the Preview panel. Press and hold the mouse button over the color in the Preview panel and drag it over the other components. Note that none of the components accepts color by default.
  19. Play with dragging and dropping between the various components and note which components accept data.

When the Turn drag and drop on check box is checked, BasicDnD demonstrates drag and drop behavior that becomes available to a component with the following line of code:

component.setDragEnabled(true);
Many components support cut/copy/paste of text using the keyboard bindings Control-X, Control-C, and Control-V, respectively. This is because a JTextComponent installs the cut/copy/paste key bindings and action map when it is created. You only need to add a bit more code to create a menu with Cut, Copy and Paste menu items and tie those items to the default text support. We show you how to do that in the DragColorTextFieldDemo example. Cut/copy/paste is further discussed in Adding Cut/Copy/Paste Support.

At the heart of the data transfer mechanism is the TransferHandler (in the API reference documentation) class. A TransferHandler provides an easy mechanism for transferring data to and from a JComponent. The data to be transferred is bundled up into an object that implements the Transferable (in the API reference documentation) interface. The components in the data transfer support table are provided with default transfer handlers, but a transfer handler can be created and installed on any component using the JComponent method setTransferHandler:

component.setTransferHandler(new MyTransferHandler());

The default Swing transfer handlers, such as those used by text components and the color chooser, provide the support considered to be most useful for both importing and exporting of data. If you install a custom TransferHandler onto a Swing component, the default support is replaced. For example, if you replace JTextField's TransferHandler with one that handles colors only, you'll disable its ability to support import and export of text.

This means that if you must replace a default TransferHandler — for example, one that handles text — you'll need to re-implement the text import and export ability. This does not need to be as extensive as what Swing provides — it could be as simple as supporting the StringFlavor data flavor, depending on your application's needs. The DragColorTextFieldDemo gives an example of this. You might also want to watch RFE #4830695 (outside of the tutorial), which requests the ability to add data import on top of an existing TransferHandler.

The remainder of this document describes how to use data transfer in a variety of ways. Here is a list of some common scenarios and where in the document you can find more information:

How do I provide drop support for those components in the Data Transfer Support table that do not have a check-mark in the drop column?
You need to implement a custom transfer handler to provide drop support. See Extending Default DnD Support for an example of how this is done.

I want my component to import only. How do I do that?
You need to provide a custom transfer handler with implementations for the canImport and importData methods. The DragColorDemo example in Importing a New Flavor: Color does this.

How do I create a component that can accept multiple types of data?
A transfer handler can be created to accept more than one type of data. Replacing Default Support: Color and Text shows the example, DragColorTextFieldDemo, which installs a custom transfer handler on a JTextField to import both color and text, and export text. Also, the DragFileDemo in Importing a New Flavor: Files installs a transfer handler on the text area/tabbed pane that imports both files and strings.

How do I create a custom transfer handler to import/export a non-standard type of data?
Specifying the Data Format describes how to create a data flavor with a variety of data types. Also, the DragListDemo example in Data Transfer with a Custom DataFlavor shows how to transfer data in the ArrayList format.

How do I make data transfer work with my custom component?
Data Transfer with a Custom Component discusses the requirements of making a custom component work with the data transfer system.

How do I enable the cut/copy/paste bindings?
Adding Cut/Copy/Paste Support describes how to enable the built-in cut/copy/paste support for text components. Implementing cut/copy/paste for non-text components is also covered.

How do I obtain the drop position in the destination component?
You can obtain the drop location by way of the component's selection. You'll notice that the selection changes in a component as you drag over it. For lists, tables, and trees you can query the current selection at drop time to find the drop position. For text components, you can query the position of the caret. There is an example of this in ArrayListTransferHandler, part of the DragListDemo example. You might also want to watch RFE #4468566 (outside of the tutorial), which requests a better way of indicating (and displaying) the drop location without changing the selection.

This has been a brief introduction to the Swing data transfer mechanism. If you want more details, see the Swing Data Transfer (outside of the tutorial) document in the release notes for your particular J2SE release.

A Simple Example: Adding DnD to JLabel

The JLabel component does not, by default, support drag or drop, but it is a fairly simple exercise to add this support — the following demo shows how to do this.

The LabelDnD example


Try this: 
  1. Run LabelDnD using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Press and hold the mouse button while the cursor is over the label. As you begin to move the cursor the Copy cursor icon appears. Drop the text onto the text field.
  3. Type text into the text field. Select and initiate the drag using the mouse. The Move cursor icon appears because the default behavior for text field is Move. Drop the text onto the label and note that the selected string has been removed from the original text.
  4. Type more text into the text field, if necessary, and select. While holding down the Control key, drag the text. The Copy cursor icon appears. Drop the text as desired.

While it is possible to extend this example to show copy and paste, note that JLabel does not have bindings for copy and paste and does not, by default, receive the focus that is required to support this feature.

The entire example can be found in LabelDnD.java (in a .java source file). Here is the code that creates the label and installs a transfer handler on the label:

label = new JLabel("I'm a Label!", SwingConstants.LEADING);
label.setTransferHandler(new TransferHandler("text"));

MouseListener listener = new DragMouseAdapter();
label.addMouseListener(listener);

To add drag support to JLabel or any custom component, you must add the ability to detect activity on the mouse. LabelDnD implements a mouse listener to detect mouse pressed — when the mouse is pressed, the transfer handler initiates the drag from the label by invoking exportAsDrag with the Copy argument:

public class DragMouseAdapter extends MouseAdapter {
    public void mousePressed(MouseEvent e) {
        JComponent c = (JComponent)e.getSource();
        TransferHandler handler = c.getTransferHandler();
        handler.exportAsDrag(c, e, TransferHandler.COPY);
    }
}

To find what the call to

new TransferHandler("text")
does, see Specifying the Data Format.

Extending Default DnD Support

You've seen in the Data Transfer Support table that several components do not support drop by default. The reason for this is that there is no all-purpose way to handle a drop on those components. For example, what does it mean to drop on a particular node of a JTree? Does it replace the node, insert below it, or insert as a child of that node? Also, we don't know what type of model is behind the tree — it might not be mutable. However, while Swing doesn't provide a default implementation, the framework for drop is there. You need only to provide a custom TransferHandler that deals with the actual transfer of data.

The following example, ExtendedDnDDemo, tweaks the default drag and drop behavior for two components, JList and JTable. This demo shows how to:

If you've run the BasicDnD example previously, you'll see that this tweaked behavior is slightly different than the default Swing behavior which does not use commas as a separator.

The ExtendedDnDDemo example


Try this: 
  1. Run ExtendedDnDDemo using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Select a row in the table
  3. Press the row again and drag. As you drag the cursor icon over the list, the row that is currently under the cursor highlights — the new data will be inserted after the selected row.
  4. Drop the row onto the list. Note that the row has been removed from the table, and now appears in the list with commas separating the columns.
  5. Select two rows from the table and drop onto the list. Now there are two new items in the list with commas separating the columns.
  6. Select one of the items in the list and drag it to the table. As you drag the icon over the table, the row that is under the curson highlights — the new data will be inserted after the selected row.
  7. Drop it onto the table. It is removed from the list and the commas are removed, replaced by column separators.
  8. Type some text into the text area; for example: How, Now, Brown, Cow. Select the line of text and drop it into the table.
  9. Select an item in the list. Hold down the Control key while dragging the item to the text area and drop. The text has been copied to the new location.

The code for the example's main class is in ExtendedDnDDemo.java. An abstract subclass of TransferHandler, StringTransferHandler, defines three abstract methods for importing and exporting strings: exportString, importString, and cleanup. StringTransferHandler also overrides the standard TransferHandler methods: importData and canImport are required to import data; getSourceActions, createTransferable, and exportDone are for export. (Note that exportDone may not be necessary if you only implement Copy and, therefore, do not need to remove data from the source as you would for a Move.) The abstract methods importString, exportString, and cleanup are called by the importData, createTransferable, and exportDone methods, respectively. Here is the code for StringTransferHander.java:

public abstract class StringTransferHandler extends TransferHandler {
    
    protected abstract String exportString(JComponent c);
    protected abstract void importString(JComponent c, String str);
    protected abstract void cleanup(JComponent c, boolean remove);
    
    protected Transferable createTransferable(JComponent c) {
        return new StringSelection(exportString(c));
    }
    
    public int getSourceActions(JComponent c) {
        return COPY_OR_MOVE;
    }
    
    public boolean importData(JComponent c, Transferable t) {
        if (canImport(c, t.getTransferDataFlavors())) {
            try {
                String str = (String)t.getTransferData(DataFlavor.stringFlavor);
                importString(c, str);
                return true;
            } catch (UnsupportedFlavorException ufe) {
            } catch (IOException ioe) {
            }
        }

        return false;
    }
    
    protected void exportDone(JComponent c, Transferable data, int action) {
        cleanup(c, action == MOVE);
    }
    
    public boolean canImport(JComponent c, DataFlavor[] flavors) {
        for (int i = 0; i < flavors.length; i++) {
            if (DataFlavor.stringFlavor.equals(flavors[i])) {
                return true;
            }
        }
        return false;
    }
    
}

The StringSelection (in the API reference documentation) class implements the Transferable interface and handles the details of bundling up the data for transport.

Two subclasses of StringTransferHandler, ListTransferHandler and TableTransferHandler, implement the abstract importString, exportString, and cleanup methods and deal with the specifics of a list and a table, respectively. Here is the code for ListTransferHandler:

public class ListTransferHandler extends StringTransferHandler {
    private int[] indices = null;
    private int addIndex = -1; //Location where items were added
    private int addCount = 0;  //Number of items added.
            
    //Bundle up the selected items in the list
    //as a single string, for export.
    protected String exportString(JComponent c) {
        JList list = (JList)c;
        indices = list.getSelectedIndices();
        Object[] values = list.getSelectedValues();
        
        StringBuffer buff = new StringBuffer();

        for (int i = 0; i < values.length; i++) {
            Object val = values[i];
            buff.append(val == null ? "" : val.toString());
            if (i != values.length - 1) {
                buff.append("\n");
            }
        }
        
        return buff.toString();
    }

    //Take the incoming string and wherever there is a
    //newline, break it into a separate item in the list.
    protected void importString(JComponent c, String str) {
        JList target = (JList)c;
        DefaultListModel listModel = (DefaultListModel)target.getModel();
        int index = target.getSelectedIndex();

        //Prevent the user from dropping data back on itself.
        //For example, if the user is moving items #4,#5,#6 and #7 and
        //attempts to insert the items after item #5, this would
        //be problematic when removing the original items.
        //So this is not allowed.
        if (indices != null && index >= indices[0] - 1 &&
              index <= indices[indices.length - 1]) {
            indices = null;
            return;
        }

        int max = listModel.getSize();
        if (index < 0) {
            index = max;
        } else {
            index++;
            if (index > max) {
                index = max;
            }
        }
        addIndex = index;
        String[] values = str.split("\n");
        addCount = values.length;
        for (int i = 0; i < values.length; i++) {
            listModel.add(index++, values[i]);
        }
    }

    //If the remove argument is true, the drop has been
    //successful and it's time to remove the selected items 
    //from the list. If the remove argument is false, it
    //was a Copy operation and the original list is left
    //intact.
    protected void cleanup(JComponent c, boolean remove) {
        if (remove && indices != null) {
            JList source = (JList)c;
            DefaultListModel model  = (DefaultListModel)source.getModel();
            //If we are moving items around in the same list, we
            //need to adjust the indices accordingly, since those
            //after the insertion point have moved.
            if (addCount > 0) {
                for (int i = 0; i < indices.length; i++) {
                    if (indices[i] > addIndex) {
                        indices[i] += addCount;
                    }
                }
            }
            for (int i = indices.length - 1; i >= 0; i--) {
                model.remove(indices[i]);
            }
        }
        indices = null;
        addCount = 0;
        addIndex = -1;
    }
}

Note that importString uses the split utility method of String to divide the incoming text so that wherever a newline occurs, a new list item is created. To support the moving of text from a list, exportDone calls cleanup, which takes care of removing the dragged items from the original list. The cleanup method has special handling for the case when moving items within the same list and the data is moved to a higher position in the list (with a smaller index). The cleanup method is called after the data has been inserted into the list which changes the indices of the items to be deleted. When the indices of the original items to be moved have changed, they must be adjusted accordingly, before they can be deleted.

The table transfer handler, TableTransferHandler, is implemented in a similar manner, though it is slightly more complex because it has support for text that is newline- and comma-delimited.

Specifying the Data Format

Creating a TransferHandler can be as simple as using the constructor. For example, in the LabelDnD demo, the label supports both importing and exporting Strings to and from its its text property with the following line of code:

label.setTransferHandler(new TransferHandler("text"));

When using the property name form of the constructor, there must be a getProperty method in the component's API to export data and a setProperty method to import data. The label's transfer handler works because JLabel has a getText method. This works with any property, for example, if you had instead created the label's transfer handler like this:

label.setTransferHandler(new TransferHandler("foreground"));

You would then be able to drag a color from the color chooser, drop it on the label, and the label's text color would change, because label has a setForeground method that requires a java.awt.Color object. We have provided another version of LabelDnD, called LabelDnD2, which demonstrates using the TransferHandler constructor, this time for the label's foreground property:

The LabelDnD2 example


Try this: 
  1. Run LabelDnD2 using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Select a color from the palette. The selected color appears in the Preview panel.
  3. Press and hold the mouse button while the cursor is over the Preview panel and begin to drag.
  4. Drop the color onto the label and see the text change color.
  5. Select another color from the palette so the Preview panel now shows a new color.
  6. Press on the label and begin to drag. Drop the color anywhere on the color chooser — in the palette or the preview panel. The Preview panel now shows the color of the label.

If you cannot use the property name form of the TransferHandler constructor, the DataFlavor (in the API reference documentation) class allows you to specify the content-type of your data for both your TransferHandler and, if necessary, your Transferable. Three flavor types are predefined for you:

For example, a TransferHandler imports String data with this line of code:

String str = (String)t.getTransferData(DataFlavor.stringFlavor);

Or it imports a java.awt.Image using imageFlavor with this line of code:

Image image = (Image)t.getTransferData(DataFlavor.imageFlavor);

If you require a flavor other than these predefined types, you need to create your own. The format for specifying a data flavor is this:

DataFlavor(Class representationClass, String humanPresentableName);
For example, to create a data flavor for the java.util.ArrayList class:
new DataFlavor(ArrayList.class, "ArrayList");

To create a data flavor for an integer array:

new DataFlavor(int[].class, "Integer Array");

Transferring the data in this manner uses Object serialization, so the class you use to transfer the data must implement the Serializable interface, as must anything that is serialized with it. If not everything is Serializable, you'll see a NotSerializableException during drop or copy to the clipboard. For more information see Object Serialization (in the Creating a GUI with JFC/Swing trail) in the Essential Java Classes (in the Creating a GUI with JFC/Swing trail) trail.

Note that creating a data flavor using the DataFlavor(Class, String) constructor allows you to transfer data between applications, including native applications. If you want to create a data flavor that transfers data only within an application, you use javaJVMLocalObjectMimeType and the DataFlavor(String) constructor. For example, to specify a data flavor that transfers color from a JColorChooser, you could use this code:

String colorType = DataFlavor.javaJVMLocalObjectMimeType +
                   ";class=java.awt.Color";
DataFlavor colorFlavor = new DataFlavor(colorType);

To create a data flavor for an ArrayList:

new DataFlavor(DataFlavor.javaJVMLocalObjectMimeType +
               ";class=java.util.ArrayList");

To transfer the data as an integer array you would use:

new DataFlavor(DataFlavor.javaJVMLocalObjectMimeType +
               ";class=\"" + int[].class.getName() + "\"");

You'll see that a MIME type containing special characters, (such as [ and ;), must have those characters enclosed in quotes.

Finally, a Transferable can be implemented to support multiple flavors. For example, you can use both local and serialization flavors together, or you can use two forms of the same data, such as the ArrayList and integer array flavors, together, or you can create a TransferHandler that accepts different types of data, such as color and text, as you will later see. When you create the array of DataFlavors to be returned from the Transferable's getTransferDataFlavors method, the flavors should be inserted in preferred order, with the most preferred appearing at element 0 of the array. Generally the preferred order is from the richest or most complex form of the data down to the simplest — the form most likely to be understood by other objects.

See the Components That Support DnD (outside of the tutorial) table in the release notes for your particular J2SE release for further details of which data types each component imports and exports.

Importing a New Flavor: Color

The only Swing component that can, by default, import or export color is JColorChooser. We prevously described how you can create a transfer handler that will transfer data as specified by a named property. While this is easy to do, it has limited functionality. For example, if you specify the "foreground" property, a drop would only change the color of the text. It wouldn't change the background color. And if your component drags and drops text by default, replacing the transfer handler in this manner causes the component to lose this default ability.

To solve this problem you need to write a custom TransferHandler. We have provided an example of how to create a custom transfer handler that can be installed on a component so that it can accept color on a drop. DragColorDemo specifically shows how you can drop a color onto the foreground or background of a button or label.

The DragColorDemo example


Try this: 
  1. Run DragColorDemo using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Select a color from the palette. The selected color appears in the Preview panel.
  3. Press and hold the mouse button while the cursor is over the Preview panel and begin to drag.
  4. Drop the color onto the label and see the text change color.
  5. Select another color. Drop onto a button. The color of the button text changes.
  6. Click the "Change the foreground color" check box so that it is no longer checked.
  7. Drop another color onto a button or the label. The background color changes.

The example's main class can be found in DragColorDemo.java. The custom transfer handler is defined in ColorTransferHandler.java. In this example, we are only implementing import functionality and therefore only need to implement the methods canImport and importData. A single instance of the ColorTransferHandler is created and shared by all nine buttons and the label. Here is a snippet of code where the transfer handler is created and installed on the buttons:

colorHandler = new ColorTransferHandler();
...
for (int i = 0; i < 9; i++) {
    JButton tmp = new JButton("Button "+i);
    tmp.setTransferHandler(colorHandler);
    ....
}
Here is the code for ColorTransferHandler:
class ColorTransferHandler extends TransferHandler {
    //The data type exported from JColorChooser.
    String mimeType = DataFlavor.javaJVMLocalObjectMimeType +
                        ";class=java.awt.Color";
    DataFlavor colorFlavor;
    private boolean changesForegroundColor = true;

    ColorTransferHandler() {
        //Try to create a DataFlavor for color.
        try {
            colorFlavor = new DataFlavor(mimeType);
        } catch (ClassNotFoundException e) { }
    }

    /**
     * Overridden to import a Color if it is available.
     * getChangesForegroundColor is used to determine whether
     * the foreground or the background color is changed.
     */
    public boolean importData(JComponent c, Transferable t) {
        if (hasColorFlavor(t.getTransferDataFlavors())) {
            try {
                Color col = (Color)t.getTransferData(colorFlavor);
                if (getChangesForegroundColor()) {
                    c.setForeground(col);
                } else {
                    c.setBackground(col);
                }
                return true;
            } catch (UnsupportedFlavorException ufe) {
            } catch (IOException ioe) { }
        }
        return false;
    }

    /**
     * Does the flavor list have a Color flavor?
     */
    protected boolean hasColorFlavor(DataFlavor[] flavors) {
        if (colorFlavor == null) {
             return false;
        }

        for (int i = 0; i < flavors.length; i++) {
            if (colorFlavor.equals(flavors[i])) {
                return true;
            }
        }
        return false;
    }

    /**
     * Overridden to include a check for a color flavor.
     */
    public boolean canImport(JComponent c, DataFlavor[] flavors) {
        return hasColorFlavor(flavors);
    }

    protected void setChangesForegroundColor(boolean flag) {
        changesForegroundColor = flag;
    }

    protected boolean getChangesForegroundColor() {
        return changesForegroundColor;
    }
}

The ColorTransferHandler is implemented to support JavaJVMlocalObjectMimeType with the representation class class=java.awt.Color, which is the mechanism JColorChooser uses to export color. For a discussion of how data is specified to the transfer mechanism, see the previous section Specifying the Data Format.

Replacing Default Support: Color and Text

The DragColorDemo example shown in Importing a New Flavor: Color replaces the component's current transfer handler. When DragColorDemo installs the ColorTransferHandler on its components, it clobbers any pre-existing transfer handler. This is not so much a problem with buttons or labels, which don't have any predefined data to transfer, but it can be a problem when you want to add the ability to import/export color on top of a component that already imports/exports other data, such as text. As discussed in Introduction to Data Transfer Support, if you install a custom transfer handler onto a component, such as JTextField, that has a Swing-provided transfer handler, you would need to re-implement the Swing support. We have provided a version of DragColorDemo, called DragColorTextFieldDemo, that creates a transfer handler that accepts color and also re-implements the clobbered support for text.

The DragColorTextFieldDemo example


Try this: 
  1. Run DragColorTextFieldDemo using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Select a color from the palette. Drag the color from the Preview panel to one of the text fields. Each text field can have its own color.
  3. Select some text and drag to move to another text field.
  4. Hold down the Control key while dragging to copy the text to another text field.
  5. Cut some text either using the menu item or the key binding: Control-X.
  6. Select a location to paste the text and paste, either using the menu item or the key binding: Control-V.
  7. Copy some text using Control-C and paste it with Control-V.
  8. Select some text, drag to a native application, such as a text editor, and drop. The text is inserted.
  9. Select some text in the native text editor, drag to the text area and drop. The text is inserted.

This transfer handler descends from ColorTransferHandler which was used in the DragColorDemo example. Since ColorAndTextTransferHandler must export data, it implements createTransferable, getSourceActions, and exportDone (in addition to the two methods it provides for import support). The code is too long to include here, but you can find the main class's source code in DragColorTextFieldDemo.java. The custom transfer handler is in ColorAndTextTransferHandler.java.

Importing a New Flavor: Files

The JFileChooser exports the javaFileListFlavor — a List (in the API reference documentation) of File (in the API reference documentation) objects discussed in Specifying the Data Format. The file chooser also exports its filenames as a list of Strings — both in text/plain and text/html formats. For example, dragging a file from a drag-enabled file chooser and dropping it on a JTextArea causes the file name to be inserted into the text area, but not the contents of the file. However, a custom transfer handler that knows about javaFileListFlavor can be installed to accept the file list provided by a file chooser, open the file, read the contents, and display the contents of the file in the text area. We have provided an example that does this. Note that because this example reads files from your local file system, launching the demo via Java Web Start will bring up a warning panel requiring permission before executing the application. If you prefer, you can instead download the application and run it locally.

The DragFileDemo example


Try this: 
  1. Run DragFileDemo using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Browse your file system and find a text file to select.
  3. Drag the selected file and drop it onto the text area. The text area changes to a JTabbedPane with the file name on the tab and the contents of the file displayed underneath.
  4. Hold the cursor over the tab. The path to the file appears in the tool tip.
  5. Select several files from the file chooser.
  6. Drag and drop them on the text area. Each file is put into its own tab. The current tab is set to the last file added.
  7. Select text from the text area from one of the files. Drop it back on the text area. The text is moved in the expected manner for a text area.
  8. Select a text file from the file chooser. Drag to a native file explorer, and drop. The file will be copied to the location where it was dropped.
  9. From the same native file explorer, select a file and drag to the text area and drop. The contents of the file will appear under a new tab.
  10. Select a text file from the file chooser. Drag to a native application, such as a text editor or web browser on your system, and drop. If that app supports drop, the text of the file should appear in that application.
  11. Select the Clear All button. The tabbed pane is cleared and replaced by the default text area.

The code is too long to include here, but you you can find the main class's source in DragFileDemo.java. The custom TransferHandler for the text area is in FileAndTextTransferHandler.java. DragFileDemo doesn't do anything particularly unusual with the one exception of embedding the file chooser into the main window, rather than running it from a dialog. This allows the file chooser to be interactive without blocking the rest of the application. A separate class, TabbedPaneController manages the JTextArea/JTabbedPane that displays the contents of the files. In the constructor for DragFileDemo, the tab pane controller is created and installed like this:

JTabbedPane tabbedPane = new JTabbedPane();
JPanel tabPanel = new JPanel(new BorderLayout());
...
tpc = new TabbedPaneController(tabbedPane, tabPanel);

You can find the implementation for the tabbed pane controller in TabbedPaneController.java.

The FileandTextTransferHandler that is installed on the text area imports two flavors: javaFileListFlavor and stringFlavor. As you have seen before, stringFlavor is necessary because the new transfer handler clobbers the default behavior for the text area and this re-implements its basic behavior. In the importData method for the transfer handler, the code first checks to see if files are being imported. If so, the files are opened into a BufferedReader and the contents are appended. If the imported data is not files, it then checks for strings.

Data Transfer with a Custom Component

We have seen how to customize data transfer for standard Swing components, but how do you add data transfer to a custom component? The simplest data transfer to implement is drag and drop:

With a bit more code cut/copy/paste support can be added. The DragPictureDemo example shows how to implement full data transfer with a custom component.

The DragPictureDemo example


Try this: 
  1. Run DragPictureDemo using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Drag and drop the pictures using the mouse. Copy a picture by holding the Control key while dragging.
  3. Cut or Copy a picture using the key bindings Control-X or Control-C.
  4. Paste the picture using the key binding Control-V.
  5. You can run this example using only the keyboard by using Tab to move the focus, and the Control-X, Control-C, and Control-V keys to cut, copy, and paste the pictures, respectively.

You can find the main class's source code in DragPictureDemo.java — you may recognize the basic functionality from the TrackFocusDemo example in the How to Use the Focus Subsystem section. The custom component DTPicture is a subclass of the Picture component, modified to support data transfer. A new class is PictureTransferHandler, the custom transfer handler for DTPicture.

The custom component DTPicture enables drag and drop by implementing the MouseMotionListener interface. The MouseMotionListener interface allows you to detect mouse motion by implementing the mouseDragged method. We arbitrarily chose a displacement of 5 pixels to determine whether the user is actually attempting to drag, as opposed to clicking on a picture. Once the cursor has moved a distance of 5 pixels in either direction while the mouse button is down, the transfer handler is called to initiate the drag. The mouseDragged method also checks to see if the Control button is being pushed on the keyboard — if it is, the action is a Copy, otherwise the action is a Move. The mousePressed, mouseDragged and mouseReleased methods in DTPicture look like this:

 
MouseEvent firstMouseEvent = null;

public void mousePressed(MouseEvent e) {
    //Don't bother to drag if there is no image.
    if (image == null) return;

    firstMouseEvent = e;
    e.consume();
}

public void mouseDragged(MouseEvent e) {
    //Don't bother to drag if the component displays no image.
    if (image == null) return;

    if (firstMouseEvent != null) {
        e.consume();

        //If they are holding down the control key, COPY rather than MOVE
        int ctrlMask = InputEvent.CTRL_DOWN_MASK;
        int action = ((e.getModifiersEx() & ctrlMask) == ctrlMask) ?
              TransferHandler.COPY : TransferHandler.MOVE;

        int dx = Math.abs(e.getX() - firstMouseEvent.getX());
        int dy = Math.abs(e.getY() - firstMouseEvent.getY());
        //Arbitrarily define a 5-pixel shift as the
        //official beginning of a drag.
        if (dx > 5 || dy > 5) {
            //This is a drag, not a click.
            JComponent c = (JComponent)e.getSource();
            //Tell the transfer handler to initiate the drag.
            TransferHandler handler = c.getTransferHandler();
            handler.exportAsDrag(c, firstMouseEvent, action);
            firstMouseEvent = null;
        }
    }
}

public void mouseReleased(MouseEvent e) {
    firstMouseEvent = null;
}

The Adding Cut/Copy/Paste Support section discusses how DragPictureDemo implements cut, copy and paste with and without menu support.

The PictureTransferHandler class looks very much like other custom transfer handlers you have seen except this one transfers data using the built-in support for java.awt.Images — DataFlavor.imageFlavor. For more information, see Specifying the Data Format. If you are interested in more discussion on the custom Picture component, see the Tracking Focus Changes to Multiple Components discussion in How to Use the Focus Subsystem.

Data Transfer with a Custom DataFlavor

By now you've seen several examples of transfer handlers that transfer data using conventional formats. This section takes a standard Swing object — a JList — and transfers the data using a content-type based on the java.util.ArrayList (in the API reference documentation) class. To achieve this, a custom Transferable is created. To implement a Transferable you must conform to the Transferable interface and provide implementations for the methods getTransferData, getTransferDataFlavors and isDataFlavorSupported.

The DragListDemo example


Try this: 
  1. Run DragListDemo using Java Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Select one or more items from either list. To select a contiguous group of items, after selecting the first item, select the last item while holding down the Shift key. All the items in between are automatically selected.
  3. As you begin to drag, the selection in the first list changes to show where a drop would occur, but the items being dragged are not affected.
  4. Drop the items on either list. The data is deposited immediately after the selected item. Note that the items are removed from the first list since this was a Move.
  5. Repeat these steps while holding down the Control key to perform a Copy.
  6. Cut, copy, and paste the items using the key bindings Control-X, Control-C, and Control-V.

The DragListDemo.java class creates and displays the lists in the usual manner. On each list is installed a shared instance of ArrayListTransferHandler.java.

Adding Cut/Copy/Paste Support

So far our discussion has centered mostly around drag and drop support. However, it is an easy matter to hook up cut/copy/paste to a transfer handler. The basic steps are:

The DragColorTextFieldDemo, in Replacing Default Support: Color and Text, shows how to use the default cut/copy/paste text support provided by DefaultEditorKit (in the API reference documentation) with the custom TransferHandler installed on the text fields. A nice feature of the DefaultEditorKit methods is that they remember which component last had the focus. Here is the code that creates the Edit menu and uses the cut, copy, and paste Actions defined in DefaultEditorKit to create the menu items:

//Create an Edit menu to support cut/copy/paste.
public JMenuBar createMenuBar () {
    JMenuItem menuItem = null;
    JMenuBar menuBar = new JMenuBar();
    JMenu mainMenu = new JMenu("Edit");
    mainMenu.setMnemonic(KeyEvent.VK_E);

    menuItem = new JMenuItem(new DefaultEditorKit.CutAction());
    menuItem.setText("Cut");
    menuItem.setMnemonic(KeyEvent.VK_T);
    mainMenu.add(menuItem);
    menuItem = new JMenuItem(new DefaultEditorKit.CopyAction());
    menuItem.setText("Copy");
    menuItem.setMnemonic(KeyEvent.VK_C);
    mainMenu.add(menuItem);
    menuItem = new JMenuItem(new DefaultEditorKit.PasteAction());
    menuItem.setText("Paste");
    menuItem.setMnemonic(KeyEvent.VK_P);
    mainMenu.add(menuItem);

    menuBar.add(mainMenu);
    return menuBar;
}

Hooking up cut/copy/paste support in this manner works with any component that descends from JTextComponent.

For any non-text component, you must manually set up the bindings in the input and action maps. The DragPictureDemo example, in Data Transfer with a Custom Component, shows how to do this. Here is a code snippet from the constructor for the DTPicture component:

if (installInputMapBindings) {
    InputMap imap = this.getInputMap();
    imap.put(KeyStroke.getKeyStroke("ctrl X"),
        TransferHandler.getCutAction().getValue(Action.NAME));
    imap.put(KeyStroke.getKeyStroke("ctrl C"),
        TransferHandler.getCopyAction().getValue(Action.NAME));
    imap.put(KeyStroke.getKeyStroke("ctrl V"),
        TransferHandler.getPasteAction().getValue(Action.NAME));
}

ActionMap map = this.getActionMap();
map.put(TransferHandler.getCutAction().getValue(Action.NAME),
        TransferHandler.getCutAction());
map.put(TransferHandler.getCopyAction().getValue(Action.NAME),
        TransferHandler.getCopyAction());
map.put(TransferHandler.getPasteAction().getValue(Action.NAME),
        TransferHandler.getPasteAction());

The boolean installInputMapBindings is true in this case and will be further discussed when we show how to add an Edit menu to support cut/copy/paste.

While you can implement cut/copy/paste to work exclusively using key bindings, it is considered good GUI design to provide menu items as well. We have provided the DragPictureDemo2 example which extends DragPictureDemo with an Edit menu.

The DragPictureDemo2 example


Try this: 
  1. Run DragPictureDemo2 using JavaTM Web Start. Or, to compile and run the example yourself, consult the example index.
  2. Cut/copy an image using the Edit->Cut or Edit->Copy menu item.
  3. Select an empty square. Paste the image using the Edit->paste menu item.

The DragPictureDemo2 example creates the Edit menu like this:

public JMenuBar createMenuBar() {
    JMenuItem menuItem = null;
    JMenuBar menuBar = new JMenuBar();
    JMenu mainMenu = new JMenu("Edit");
    mainMenu.setMnemonic(KeyEvent.VK_E);
    TransferActionListener actionListener = new TransferActionListener();

    menuItem = new JMenuItem("Cut");
    menuItem.setActionCommand((String)TransferHandler.getCutAction().
             getValue(Action.NAME));
    menuItem.addActionListener(actionListener);
    menuItem.setAccelerator(
      KeyStroke.getKeyStroke(KeyEvent.VK_X, ActionEvent.CTRL_MASK));
    menuItem.setMnemonic(KeyEvent.VK_T);
    mainMenu.add(menuItem);
    menuItem = new JMenuItem("Copy");
    menuItem.setActionCommand((String)TransferHandler.getCopyAction().
             getValue(Action.NAME));
    menuItem.addActionListener(actionListener);
    menuItem.setAccelerator(
      KeyStroke.getKeyStroke(KeyEvent.VK_C, ActionEvent.CTRL_MASK));
    menuItem.setMnemonic(KeyEvent.VK_C);
    mainMenu.add(menuItem);
    menuItem = new JMenuItem("Paste");
    menuItem.setActionCommand((String)TransferHandler.getPasteAction().
             getValue(Action.NAME));
    menuItem.addActionListener(actionListener);
    menuItem.setAccelerator(
      KeyStroke.getKeyStroke(KeyEvent.VK_V, ActionEvent.CTRL_MASK));
    menuItem.setMnemonic(KeyEvent.VK_P);
    mainMenu.add(menuItem);

    menuBar.add(mainMenu);
    return menuBar;
}

The line:

menuItem.setActionCommand((String)TransferHandler.getCopyAction().
         getValue(Action.NAME));
ties the copy action to the menu item. The line:
menuItem.setAccelerator(
  KeyStroke.getKeyStroke(KeyEvent.VK_C, ActionEvent.CTRL_MASK));
defines Control-C as the key binding for the action. The keyboard shortcuts, E (Edit), T (cuT), C (Copy), and P (Paste), are installed by calling setMnemonic. For a discussion of mnemonics versus accelerators, see the Enabling Keyboard Operation (in the Creating a GUI with JFC/Swing trail) discussion in the menu portion of the Components (in the Creating a GUI with JFC/Swing trail) lesson.

When you register the key stroke in the input map yourself, you're registering it on a per-component basis — when the component has the focus, and the key stroke is typed, the action is fired. When you set an key binding on a menu, the key stroke information is added to a global input map, and the key binding is active all the time, when the window has the focus. Therefore setting up the input maps on the components yourself is redundant when the menu accelerators are used. If you are not using menu accelerators, you need to set up the input map yourself. The DTPicture class can be used by both DragPictureDemo and DragPictureDemo2 because the static property installInputMapBindings allows one demo to set the bindings on the input map and the other demo to skip that step. For your program, you would choose one approach or the other, but not both.

The final change to DragPictureDemo2 is necessary to ensure that the action goes to the correct component when the user initiates a cut, copy, or paste. The TransferActionListener class is installed as an action listener on the cut/copy/paste menu items, and as a property change listener on the keyboard focus manager. Each time the focus owner changes, TransferActionListener keeps track of the new focus owner. When the user initiates a cut, copy, or paste, through a menu item, TransferActionListener is notified and then fires the appropriate action on the component that has the focus. Here is the code for TransferActionListener:

public class TransferActionListener implements ActionListener,
                                              PropertyChangeListener {
    private JComponent focusOwner = null;
    
    public TransferActionListener() {
        KeyboardFocusManager manager = KeyboardFocusManager.
           getCurrentKeyboardFocusManager();
        manager.addPropertyChangeListener("permanentFocusOwner", this);
    }
    
    public void propertyChange(PropertyChangeEvent e) {
        Object o = e.getNewValue();
        if (o instanceof JComponent) {
            focusOwner = (JComponent)o;
        } else {
            focusOwner = null;
        }
    }
    
    public void actionPerformed(ActionEvent e) {
        if (focusOwner == null)
            return;
        String action = (String)e.getActionCommand();
        Action a = focusOwner.getActionMap().get(action);
        if (a != null) {
            a.actionPerformed(new ActionEvent(focusOwner,
                                              ActionEvent.ACTION_PERFORMED,
                                              null));
        }
    }
}

The Data Transfer API

The following tables list the commonly used constructors and methods related to data transfer. The data transfer API falls into five categories:

For more detailed information about the data transfer mechanism, see the Swing Data Transfer (outside of the tutorial) document in the release notes for your particular J2SE release.

Useful JComponent Methods

All of this API was introduced in release 1.4.
Method Purpose
setTransferHandler(TransferHandler)
getTransferHandler()
Set or get the transfer handler. For those components that have default Swing support, the TransferHandler is installed by the ComponentUI if the value is null or marked by the presence of the UIResource interface. The default TransferHandler implementation installed by the ComponentUI is marked by the UIResource interface, enabling developers to override the default TransferHandler. Note that the same instance of a transfer handler may be shared among components.
setDragEnabled(boolean)
getDragEnabled()
Set or get the dragEnabled property which must be true to enable automatic drag handling. These methods are implemented on the following Swing components only: JColorChooser, JEditorPane, JFileChooser, JFormattedTextField, JList, JPasswordField, JTable, JTextArea, JTextField, JTextPane, and JTree. Even so, some look and feels may not support automatic drag and drop and some components do not have default support for drop. For security reasons, JPasswordField does not support drag. The default value for this property is false.

TransferHandler API

All of this API was introduced in release 1.4.
Constructor or Method Purpose
TransferHandler()
TransferHandler(String)
Create a transfer handler. The constructor that takes a string creates a transfer handler that can transfer a named property from one component to another. The String argument is the name of the property to transfer and may be null.
canImport(JComponent, DataFlavor[]) Returns true if the specified component currently accepts data of at least one of the types specified in the list of data flavors, otherwise returns false. If this method returns true, the data transfer system changes the cursor icon to indicate that this component will accept the data.
protected createTransferable(JComponent) Returns a Transferable, encapsulating the data to be transferred.
importData(JComponent, Transferable) Import the data into the component. The Transferable contains the actual data to import. True is returned if the import was successful, otherwise false is returned.
exportAsDrag(JComponent, InputEvent, int) Initiate the Swing drag support. The int argument specifies either a COPY or a MOVE action. When this method returns, the export may not have completed. The method exportDone is called when the transfer is complete.
exportToClipboard(JComponent, Clipboard, int) Initiate a data transfer from the specified component to the specified clipboard. The int argument specifies either a COPY or a MOVE action. When this method returns, the export is completed. The method exportDone is called with the transfer is complete.
protected exportDone(JComponent, Transferable, int) Called after the export has completed. This method should remove the data that was transferred from the source component if the action was MOVE.
getCutAction()
getCopyAction()
getPasteAction()
Return an Action that implements a cut, copy, or paste operation, respectively. The cut and copy actions cause exportToClipboard to be invoked. The paste action causes importData to be invoked.
getSourceActions(JComponent) Return the type of transfer actions supported by the component when used as the source of data transfer. It will be one of these types: COPY, COPY_OR_MOVE, MOVE, or NONE. The NONE type indicates that the component does not allow exporting of any data.
getVisualRepresentation(Transferable) Returns the Icon that establishes the look of a transfer. In v 1.4, this method does nothing. You might want to watch bug #4816922 (outside of the tutorial) to track the status of this method.

Transferable Classes

Class Purpose
Transferable An interface for classes that can be used to provide data for a transfer operation.
StringSelection A Transferable that implements the capability required to transfer a String. Supports DataFlavor.stringFlavor and equivalent flavors.

Transferable Interface API

Method Purpose
getTransferData(DataFlavor) Return an object that contains the data to be transferred.
getTransferDataFlavors() Return an array of DataFlavors listing the flavors in which the data can be provided. The flavors are listed in preferred order, with the item at element 0 being the most preferred flavor.
isDataFlavorSupported(DataFlavor) Return true if the requested flavor is supported for this object, otherwise return false.

DataFlavor API

Class or Field Purpose
DataFlavor()
DataFlavor(Class, String)
DataFlavor(String)
DataFlavor(String, String)
DataFlavor(String, String, ClassLoader)
Create a new content-type for transferring data. See Specifying the Data Format for details on how to create new data flavors.
stringFlavor The data flavor representing the Java Unicode java.lang.String class.
imageFlavor The data flavor representing the java.awt.Image class.
javaFileListFlavor The data flavor representing the java.util.List class where each element of the list must be a java.io.File.
javaJVMLocalObjectMimeType A data flavor with this MIME type is used to transfer a reference to an arbitrary Java object using the Transferable interface within the same VM.
javaSerializedObjectMimeType A data flavor with this MIME type represents a graph of Java objects that have been made persistent. This is used to transfer serialized objects.
javaRemoteObjectMimeType A data flavor with this MIME type represents a live link to a Remote object, where the representation class of the DataFlavor represents the type of the Remote interface to be transferred.

Examples that Use Data Transfer

The following table lists examples that use data transfer:

Example Where Described Notes
BasicDnD In this section under Introduction to Data Transfer Support Demonstrates basic default drag and drop behavior.
LabelDnD In this section under A Simple Example: Adding DnD to JLabel Demonstrates how to add support for dragging text from and dropping text on a JLabel.
LabelDnD2 In this section under Specifying the Data Format Demonstrates how to add support for dropping color onto a JLabel.
ExtendedDnDDemo In this section under Extended Default DnD Support Demonstrates how to create a custom transfer handler to extend default drag and drop support for JList and JTable.
DragColorDemo In this section under Importing a New Flavor: Color Demonstrates how to drag color onto buttons and labels.
DragColorTextFieldDemo In this section under Replacing Default Support: Color and Text Demonstrates how to drag color onto text fields and how to create a custom transfer handler that allows the text fields to export text. Also demonstrates cut/copy/paste using the built-in text support.
DragFileDemo In this section under Importing a New Flavor: Files Demonstrates how to implement a custom TransferHandler on a text area that accepts a list of files from a file chooser, opens those files and appends the contents.
DragListDemo In this section under Data Transfer with a Custom DataFlavor Demonstrates how to implement a custom TransferHandler with a custom DataFlavorjava.util.ArrayList.
DragPictureDemo In this section under Data Transfer with a Custom Component Demonstrates how to implement a custom TransferHandler for a custom JComponent. Also implements cut/copy/paste on a custom component using key bindings.
DragPictureDemo2 In this section under Adding Cut/Copy/Paste Support Extends DragPictureDemo to support cut/copy/paste using an Edit menu.


Previous Page Lesson Contents Next Page Start of Tutorial > Start of Trail > Start of Lesson Search
Feedback Form

Copyright 1995-2004 Sun Microsystems, Inc. All rights reserved.