Transferring Shell Objects with Drag-and-Drop and the Clipboard
Many applications allow users to transfer data to another application by dragging and dropping the data with the mouse, or by using the Clipboard. Among the many types of data that can be transferred are Shell objects such as files or folders. Shell data transfer can take place between two applications, but users can also transfer Shell data to or from the desktop or Windows Explorer.
Although files are the most commonly transferred Shell object, Shell data transfer can involve any of the variety of objects found in the Shell namespace. For instance, your application might need to transfer a file to a virtual folder such as the Recycle Bin, or accept an object from a non-Microsoft namespace extension. If you are implementing a namespace extension, it must be able to behave properly as a drop source and target.
This document discusses how applications can implement drag-and-drop and Clipboard data transfers with Shell objects.
How Drag-and-Drop Works with Shell Objects
Applications often need to provide users with a way to transfer Shell data. Some examples are:
- Dragging a file from Windows Explorer or the desktop and dropping it on an application.
- Copying a file to the Clipboard in Windows Explorer and pasting it into an application.
- Dragging a file from an application to the Recycle Bin.
For a detailed discussion of how to handle these and other scenarios, see Handling Shell Data Transfer Scenarios. This document focuses on the general principles behind Shell data transfer.
Windows provides two standard ways for applications to transfer Shell data:
- A user cuts or copies Shell data, such as one or more files, to the Clipboard. The other application retrieves the data from the Clipboard.
- A user drags an icon that represents the data from the source application and drops the icon on a window owned by the target.
In both cases, the transferred data is contained in a data object. Data objects are Component Object Model (COM) objects that expose the IDataObject interface. Schematically, there are three essential steps that all Shell data transfers must follow:
- The source creates a data object that represents the data that is to be transferred.
- The target receives a pointer to the data object's IDataObject interface.
- The target calls the IDataObject interface to extract the data from it.
The difference between Clipboard and drag-and-drop data transfers lies primarily in how the IDataObject pointer is transferred from the source to the target.
Clipboard Data Transfers
The Clipboard is the simplest way to transfer Shell data. The basic procedure is similar to standard Clipboard data transfers. However, because you are transferring a pointer to a data object, not the data itself, you must use the OLE clipboard API instead of the standard clipboard API. The following procedure outlines how to use the OLE clipboard API to transfer Shell data with the Clipboard:
- The data source creates a data object to contain the data.
- The data source calls OleSetClipboard, which places a pointer to the data object's IDataObject interface on the Clipboard.
- The target calls OleGetClipboard to retrieve the pointer to the data object's IDataObject interface.
- The target extracts the data by calling the IDataObject::GetData method.
- With some Shell data transfers, the target might also need to call the data object's IDataObject::SetData method to provide feedback to the data object on the outcome of the data transfer. See Handling Optimized Move Operations for an example of this type of operation.
Drag-and-Drop Data Transfers
While somewhat more complex to implement, drag-and-drop data transfer has some significant advantages over the Clipboard:
- Drag-and-drop transfers can be done with a simple mouse movement, making operation more flexible and intuitive to use than the Clipboard.
- Drag-and-drop provides the user with a visual representation of the operation. The user can follow the icon as it moves from source to target.
- Drag-and-drop notifies the target when the data is available.
Drag-and-drop operations also use data objects to transfer data. However, the drop source must provide functionality beyond that required for Clipboard transfers:
- The drop source must also create an object that exposes an IDropSource interface. The system uses IDropSource to communicate with the source while the operation is in progress.
- The drag-and-drop data object is responsible for tracking cursor movement and displaying an icon to represent the data object.
Drop targets must also provide more functionality than is needed to handle Clipboard transfers:
- The drop target must expose an IDropTarget interface. When the cursor is over a target window, the system uses IDropTarget to provide the target with information such as the cursor position, and to notify it when the data is dropped.
- The drop target must register itself with the system by calling RegisterDragDrop. This function provides the system with the handle to a target window and a pointer to the target application's IDropTarget interface.
Note
For drag-and-drop operations, your application must initialize COM with OleInitialize, not CoInitialize.
The following procedure outlines the essential steps that are typically used to transfer Shell data with drag-and-drop:
- The target calls RegisterDragDrop to give the system a pointer to its IDropTarget interface and register a window as a drop target.
- When the user starts a drag-and-drop operation, the source creates a data object and initiates a drag loop by calling DoDragDrop.
- When the cursor is over the target window, the system notifies the target by calling one of the target's IDropTarget methods. The system calls IDropTarget::DragEnter when the cursor enters the target window, and IDropTarget::DragOver as the cursor passes over the target window. Both methods provide the drop target with the current cursor position and the state of keyboard modifier keys such as CTRL or ALT. When the cursor leaves the target window, the system notifies the target by calling IDropTarget::DragLeave. When any of these methods return, the system calls the IDropSource interface to pass the return value to the source.
- When the user releases the mouse button to drop the data, the system calls the target's IDropTarget::Drop method. Among the method's parameters is a pointer to the data object's IDataObject interface.
- The target calls the data object's IDataObject::GetData method to extract the data.
- With some Shell data transfers, the target might also need to call the data object's IDataObject::SetData method to provide feedback to the source on the outcome of the data transfer.
- When the target is finished with the data object, it returns from IDropTarget::Drop. The system returns the source's DoDragDrop call to notify the source that the data transfer is complete.
- Depending on the particular data transfer scenario, the source might need to take additional action based on the value returned by DoDragDrop and the values that are passed to the data object by the target. For instance, when a file is moved, the source must check these values to determine whether it must delete the original file.
- The source releases the data object.
While the procedures outlined above provide a good general model for Shell data transfer, there are many different types of data that can be contained in a Shell data object. There are also a number of different data transfer scenarios that your application might need to handle. Each data type and scenario requires a somewhat different approach to three key steps in the procedure:
- How a source constructs a data object to contain the Shell data.
- How a target extracts Shell data from the data object.
- How the source completes the data transfer operation.
The Shell Data Object provides a general discussion of how a source constructs a Shell data object, and how that data object can be handled by the target. Handling Shell Data Transfer Scenarios discusses in detail how to handle a number of common Shell data transfer scenarios.