Events
May 19, 6 PM - May 23, 12 AM
Calling all developers, creators, and AI innovators to join us in Seattle @Microsoft Build May 19-22.
Register todayThis browser is no longer supported.
Upgrade to Microsoft Edge to take advantage of the latest features, security updates, and technical support.
The Shell Data Object document discussed the general approach that is used to transfer Shell data with drag-and-drop or the Clipboard. However, to implement Shell data transfer in your application, you must also understand how to apply these general principles and techniques to the variety of ways that Shell data can be transferred. This document presents common Shell data transfer scenarios and discusses how to implement each one in your application.
Note
Although each of these scenarios discusses a specific data transfer operation, many of them apply to a variety of related scenarios. For instance, the primary difference between most Clipboard and drag-and-drop transfers is in how the data object arrives at the target. Once the target has a pointer to the data object's IDataObject interface, the procedures for extracting information are largely the same for both types of data transfer. However, some of the scenarios are limited to a specific type of operation. Refer to the individual scenario for details.
Each of the following sections discusses a single, fairly specific data transfer scenario. However, data transfers are often more complex and might involve aspects of several scenarios. You typically do not know, in advance, which scenario you will actually need to handle. Here are a few general guidelines to keep in mind.
For data sources:
For data targets:
Scenario: A user selects one or more files in Windows Explorer and copies them to the Clipboard. Your application extracts the file names and pastes them into the document.
This scenario could be used, for instance, to allow a user to create an HTML link by cutting and pasting the file to your application. Your application can then extract the file name from the data object and process it to create an anchor tag.
When a user selects a file in Windows Explorer and copies it to the Clipboard, the Shell creates a data object. It then calls OleSetClipboard to place a pointer to the data object's IDataObject interface on the Clipboard.
When the user selects the Paste command from your application's menu or toolbar:
Note
The final two steps in this procedure are included for completeness. They are typically not necessary for simple file transfers. All data objects used for this type of data transfer should contain the CF_HDROP format, which can be used to determine the names of the files contained by the object. However, for more general data transfers, you should enumerate the formats and select the best one that your application can handle.
The next step is to extract one or more file names from the data object and paste them into your application. Note that the procedure discussed in this section for extracting a file name from a data object applies equally well to drag-and-drop transfers.
The simplest way to retrieve file names from a data object is the CF_HDROP format:
Call IDataObject::GetData. Set the cfFormat member of the FORMATETC structure to CF_HDROP and the tymed member to TYMED_HGLOBAL. The dwAspect member is normally set to DVASPECT_CONTENT. However, if you need to have the file's path in short (8.3) format, set dwAspect to DVASPECT_SHORT.
When IDataObject::GetData returns, the hGlobal member of the STGMEDIUM structure points to a global memory object that contains the data.
Create an HDROP variable and set it to the hGlobal member of the STGMEDIUM structure. The HDROP variable is now a handle to a DROPFILES structure followed by a double null-terminated string containing the fully qualified file paths of the copied files.
Determine how many file paths are in the list by calling DragQueryFile with the iFile parameter set to 0xFFFFFFFF. The function returns the number of file paths in the list. The file path's zero-based index in this list is used in the next step to identify a particular path.
Extract the file paths from the global memory object by calling DragQueryFile once for each file, with iFile set to the file's index.
Process the file paths as needed and paste them into your application.
Call ReleaseStgMedium and pass in the pointer to the STGMEDIUM structure that you passed to IDataObject::GetData in step 1. Once you have released the structure, the HDROP value that you created in step 2 is no longer valid and should not be used.
Scenario: A user drags one or more files from Windows Explorer and drops them on your application's window. Your application extracts the content of the file (s) and pastes it into the application.
This scenario uses drag-and-drop to transfer the files from Windows Explorer to the application. Prior to the operation, your application must:
After the user initiates the operation by selecting one or more files and starting to drag them:
There are several different ways to extract the contents of a Shell object from a data object. In general, use the following order:
If the data extraction process will be lengthy, you might want to do the operation asynchronously on a background thread. Your primary thread can then proceed without unnecessary delays. For a discussion of how to handle asynchronous data extraction, see Dragging and Dropping Shell Objects Asynchronously.
The CFSTR_FILECONTENTS format provides a very flexible and powerful way to transfer the contents of a file. It is not even necessary for the data to be stored as a single file. All that is required for this format is that the data object present the data to the target as if it were a file. For instance, the actual data might be a section of a text document or a block of data extracted from a database. The target can treat the data as a file and does not need to know anything about the underlying storage mechanism.
Namespace extensions normally use CFSTR_FILECONTENTS to transfer data because this format does not assume any particular storage mechanism. A namespace extension can use whatever storage mechanism is convenient, and use this format to present its objects to applications as if they were files.
The data transfer mechanism for CFSTR_FILECONTENTS is normally TYMED_ISTREAM. Transferring an IStream interface pointer requires much less memory than loading the data into a global memory object, and IStream is a simpler way to represent data than IStorage.
A CFSTR_FILECONTENTS format is always accompanied by a CFSTR_FILEDESCRIPTOR format. You must examine the contents of this format first. If more than one file is being transferred, the data object will actually contain multiple CFSTR_FILECONTENTS formats, one for each file. The CFSTR_FILEDESCRIPTOR format contains the name and attributes of each file, and provides an index value for each file that is needed to extract a particular file's CFSTR_FILECONTENTS format.
To extract a CFSTR_FILECONTENTS format:
Scenario: A file is moved from the file system to a namespace extension using an optimized move.
In a conventional move operation, the target makes a copy of the data and the source deletes the original. This procedure can be inefficient because it requires two copies of the data. With large objects such as databases, a conventional move operation might not even be practical.
With an optimized move, the target uses its understanding of how the data is stored to handle the entire move operation. There is never a second copy of the data, and there is no need for the source to delete the original data. Shell data is well suited to optimized moves because the target can handle the entire operation using the Shell API. A typical example is moving files. Once the target has the path of a file to be moved, it can use SHFileOperation to move it. There is no need for the source to delete the original file.
Note
The Shell normally uses an optimized move to move files. To handle Shell data transfer properly, your application must be capable of detecting and handling an optimized move.
Optimized moves are handled in the following way:
The source calls DoDragDrop with the dwEffect parameter set to DROPEFFECT_MOVE to indicate that the source objects can be moved.
The target receives the DROPEFFECT_MOVE value through one of its IDropTarget methods, indicating that a move is allowed.
The target either copies the object (unoptimized move) or moves the object (optimized move).
The target then tells the source whether it needs to delete the original data.
An optimized move is the default operation, with the data deleted by the target. To inform the source that an optimized move was performed:
If the target did an unoptimized move, the data must be deleted by the source. To inform the source that an unoptimized move was performed:
The source inspects the two values that can be returned by the target. If both are set to DROPEFFECT_MOVE, it completes the unoptimized move by deleting the original data. Otherwise, the target did an optimized move and the original data has been deleted.
Scenario: One or more files are cut from a folder in Windows Explorer and pasted into a namespace extension. Windows Explorer leaves the files highlighted until it receives feedback on the outcome of the paste operation.
Traditionally, when a user cuts data it immediately disappears from view. This might not be efficient, and it can lead to usability problems if the user becomes concerned about what has happened to the data. An alternative approach is to use a delete-on-paste operation.
With a delete-on-paste operation, the selected data is not immediately removed from view. Instead, the source application marks it as selected, perhaps by changing the font or background color. After the target application has pasted the data, it notifies the source about the outcome of the operation. If the target performed an optimized move, the source can simply update its display. If the target performed a normal move, the source must also delete its copy of the data. If the paste fails, the source application restores the selected data to its original appearance.
Note
The Shell normally uses delete-on-paste when a cut/paste operation is used to move files. Delete-on-paste operations with Shell objects normally use an optimized move to move the files. To handle Shell data transfer properly, your application must be capable of detecting and handling delete-on-paste operations.
The essential requirement for delete-on-paste is that the target must report the outcome of the operation to the source. However, standard Clipboard techniques cannot be used to implement delete-on-paste because they do not provide a way for the target to communicate with the source. Instead, the target application uses the data object's IDataObject::SetData method to report the outcome to the data object. The data object can then communicate with the source through a private interface.
The basic procedure for a delete-on-paste operation is as follows:
Scenario: A user drags an object from or drops it on a virtual folder.
Virtual folders contain objects that are generally not part of the file system. Some virtual folders, such as the Recycle Bin, can represent data that is stored on the hard disk but not as ordinary file system objects. Some can represent stored data that is on a remote system, such as a handheld PC, or an FTP site. Others, such as the Printers folder, contain objects that do not represent stored data at all. While some virtual folders are part of the system, developers can also create and install custom virtual folders by implementing a namespace extension.
Regardless of the type of data or how it is stored, the folder and file objects that are contained by a virtual folder are presented by the Shell as if they were normal files and folders. It is the responsibility of the virtual folder to take whatever data it contains and present it to the Shell appropriately. This requirement means that virtual folders normally support drag-and-drop and Clipboard data transfers.
There are thus two groups of developers who need to be concerned with data transfer to and from virtual folders:
Virtual folders can represent virtually any type of data and can store that data in any way they choose. Some virtual folders might actually contain normal file system files and folders. Others might, for instance, pack all their objects into a single document or database.
When a file system object is transferred to an application, the data object normally contains a CF_HDROP format with the object's fully qualified path. Your application can extract this string and use the normal file system functions to open the file and extract its data. However, because virtual folders typically do not contain normal file system objects, they generally do not use CF_HDROP.
Instead of CF_HDROP, data is normally transferred from virtual folders with the CFSTR_FILEDESCRIPTOR/CFSTR_FILECONTENTS formats. The CFSTR_FILECONTENTS format has two advantages over CF_HDROP:
Global memory objects are rarely used to transfer data to or from virtual objects because the data must be copied into memory in its entirety. Transferring an interface pointer requires almost no memory and is much more efficient. With very large files, an interface pointer might be the only practical data transfer mechanism. Typically, data is represented by an IStream pointer, because that interface is somewhat more flexible than IStorage. The target extracts the pointer from the data object and uses the interface methods to extract the data.
For further discussion of how to handle the CFSTR_FILEDESCRIPTOR/CFSTR_FILECONTENTS formats, see Using the CFSTR_FILECONTENTS Format to Extract Data from a File.
When you implement a namespace extension, you will normally want to support drag-and-drop capabilities. Follow the recommendations for drop sources and targets discussed in General Guidelines. In particular, a namespace extension must:
Scenario: The user drops a file on the Recycle Bin. Your application or namespace extension deletes the original file.
The Recycle Bin is a virtual folder that is used as a repository for files that are no longer needed. As long as the Recycle Bin has not been emptied, the user can later recover the file and return it to the file system.
For the most part, transferring Shell objects to the Recycle Bin works much like any other folder. However, when a user drops a file on the Recycle Bin, the source needs to delete the original, even if the feedback from the folder indicates a copy operation. Normally, a drop source has no way of knowing which folder its data object has been dropped on. However, for Windows 2000 and later systems, when a data object is dropped on the Recycle Bin, the Shell will call the data object's IDataObject::SetData method with a CFSTR_TARGETCLSID format set to the Recycle Bin's class identifier (CLSID) (CLSID_RecycleBin). To handle the Recycle Bin case properly, your data object should be able to recognize this format and communicate the information to the source through a private interface.
Note
When IDataObject::SetData is called with a CFSTR_TARGETCLSID format set to CLSID_RecycleBin, the data source should close any open handles to the objects that are being transferred before returning from the method. Otherwise, you might create sharing violations.
Scenario: A user drags some data from an OLE application's data file and drops it on the desktop or Windows Explorer.
Windows allows users to drag an object from an OLE application's data file and drop it on the desktop or a file system folder. This operation creates a scrap file, which contains the data or a link to the data. The file name is taken from the short name registered for the CLSID of the object and the CF_TEXT data. For the Shell to create a scrap file containing data, the application's IDataObject interface must support the CF_EMBEDSOURCE Clipboard format. To create a file containing a link, IDataObject must support the CF_LINKSOURCE format.
There are also three optional features that an application can implement to support scrap files:
A round trip involves transferring a data object to another container and then back to the original document. For instance, a user could transfer a group of cells from a spreadsheet to the desktop, creating a scrap file with the data. If the user then transfers the scrap back to the spreadsheet, the data needs to be integrated into the document as it was before the original transfer.
When the Shell creates the scrap file, it represents the data as an embedding object. When the scrap is transferred to another container, it is transferred as an embedding object, even if it is being returned to the original document. Your application is responsible for determining the data format contained in the scrap and putting the data back into its native format if necessary.
To establish the format of the embedded object, determine its CLSID by retrieving the object's CF_OBJECTDESCRIPTOR format. If the CLSID indicates a data format that belongs to the application, it should transfer the native data instead of calling OleCreateFromData.
When the Shell creates a scrap file, it checks the registry for the list of available formats. By default, there are two formats available: CF_EMBEDSOURCE and CF_LINKSOURCE. However, there are a number of scenarios where applications might need to have scrap files in different formats:
Applications can add formats to the scrap by caching them in the registry. There are two types of cached formats:
To add a priority cache or delay-rendered format, create a DataFormat subkey under the CLSID key of the application that is the source of the data. Under that subkey, create a PriorityCacheFormats or DelayRenderFormats subkey. For each priority cache or delay-rendered format, create a numbered subkey starting with zero. Set the value of this key to either a string with the registered name of the format or a #X value, where X represents the format number of a standard Clipboard format.
The following sample shows cached formats for two applications. The MyProg1 application has the rich-text format as a priority cache format, and a private format "My Format" as a delay-rendered format. The MyProg2 application has the CF_BITMAP format (#8") as a priority cache format.
HKEY_CLASSES_ROOT
CLSID
{GUID}
(Default) = MyProg1
DataFormats
PriorityCacheFormats
0
(Default) = Rich Text Format
DelayRenderFormats
0
(Default) = My Format
{GUID}
(Default) = MyProg2
DataFormats
PriorityCacheFormats
0
(Default) = #8
Additional formats can be added by creating additional numbered subkeys.
A delayed rendering format allows an application to create a scrap file but delay the expense of rendering the data until it is requested by a target. The IDataObject interface of a scrap will offer the delayed rendering formats to the target along with native and cached data. If the target requests a delayed rendering format, the Shell will run the application and provide the data to the target from the active object.
Note
Because delayed rendering is somewhat risky, it should be used with caution. It will not work if the server is not available, or on applications that are not OLE-enabled.
Scenario: A user transfers a large block of data from source to target. To avoid blocking both applications for a significant amount of time, the target extracts the data asynchronously.
Normally, drag-and-drop is a synchronous operation. In brief:
In short, synchronous data transfer can block the primary threads of both applications for a significant amount of time. In particular, both threads must wait while the target extracts the data. For small amounts of data, the time required to extract data is small and synchronous data transfer works quite well. However, synchronously extracting large amounts of data can cause lengthy delays and interfere with the UI of both target and source.
The IAsyncOperation/IDataObjectAsyncCapability interface is an optional interface that can be implemented by a data object. It gives the drop target the ability to extract data from the data object asynchronously on a background thread. Once data extraction is handed off to the background thread, the primary threads of both applications are free to proceed.
Note
The interface was originally named IAsyncOperation, but this was later changed to IDataObjectAsyncCapability. Otherwise, the two interfaces are identical.
The purpose of IAsyncOperation/IDataObjectAsyncCapability is to allow the drop source and drop target to negotiate whether data can be extracted asynchronously. The following procedure outlines how the drop source uses the interface:
The following procedure outlines how the drop target uses the IAsyncOperation/IDataObjectAsyncCapability interface to extract data asynchronously:
Events
May 19, 6 PM - May 23, 12 AM
Calling all developers, creators, and AI innovators to join us in Seattle @Microsoft Build May 19-22.
Register today