Introduction
Applications such as
GetRight and
ClipMate seem to
magically know when something is copied to the Clipboard. How is it done? This
example shows you how to use the Win32 API function
SetClipboardViewer
to create a Clipboard Viewer application.
Using SetClipboardViewer
The Win32 API contains a function SetClipboardViewer
which allows an application to register itself as a viewer of
Clipboard events. When an item is copied to the Clipboard, windows notifies
the Clipboard Viewer by sending the WM_DRAWCLIPBOARD message. The
application can then use the appropriate function to get the contents of the
Clipboard and perform processing.
Example Application
I had a great idea one day (after copying and pasting internet links a few hundred times) to create a small application that would read the Clipboard when something was copied to it and extract any Internet URLs, UNC paths or filenames that it found and put them on a menu in the system tray.
I spent a little time on it and it is in a working state but it is not
finished. One day I will get around to finishing it off but for the moment it
is a good sample to learn how SetClipboardViewer
works.
![]() |
C# Source Code: RAD_ClipboardMonitorExampleSrc.zip (~50 KB) |
![]() |
VB.NET Example: ClipboardMonitor_VB.txt |
The SetClipboardViewer Function
The SetClipboardViewer
function adds the specified window to the
chain of Clipboard Viewers. Clipboard Viewer windows receive a
WM_DRAWCLIPBOARD
message whenever the content of the Clipboard changes.
The SetClipboardViewer
API declaration has the following syntax
in C#
[DllImport("User32.dll", CharSet=CharSet.Auto)] public static extern IntPtr SetClipboardViewer(IntPtr hWndNewViewer);
The hWndNewViewer
parameter is the handle to the window to be
added to the Clipboard chain.
When SetClipboardViewer
is successful, the return value
identifies the next window in the Clipboard Viewer chain. This is important as
a Clipboard Viewer is required to pass Clipboard messages to the next window
in the Clipboard chain.
Setting up the Clipboard Viewer
I need to override the WndProc
method of the
underlying Form so that the main form can respond to window messages from the
operating system. Normally .NET applications do not need to use
WndProc
as the .NET Framework takes care of these events.
protected override void WndProc(ref Message m) {
Next, call SetClipboardViewer
to register the application
as a Clipboard Viewer. The window will now receive the
WM_DRAWCLIPBOARD
message. Applications do not normally receive
the WM_DRAWCLIPBOARD
message, only those that call
SetClipboardViewer
.
_ClipboardViewerNext = SetClipboardViewer(this.Handle);
Responding to the WM_DRAWCLIPBOARD message
After calling SetWindowLong the WindowProc method will receive the
WM_DRAWCLIPBOARD
message when something is copied to the Clipboard.
protected override void WndProc(ref Message m) { switch ((Win32.Msgs)m.Msg) { case Win32.Msgs.WM_DRAWCLIPBOARD:
This is the place that I do my processing of the Clipboard data. The .NET Framework has the Clipboard, DataObject and DataFormats classes for working with the Clipboard. Clipboard.GetDataObject gets the current contents of the Clipboard as an IDataObject.
IDataObject iData = new DataObject(); iData = Clipboard.GetDataObject();
As the Clipboard can contain different types of data, use the DataObject.GetDataPresent method to check for a certain type of data. It returns true if the contents of the Clipboard can be converted to the specified type. The DataFormats class contains constants that define the names of some of the more common Clipboard formats.
if (iData.GetDataPresent(DataFormats.Rtf)) { // ...
Once I have determined that the format is available then the DataObject.GetData method can be used to get the Clipboard content in the required format. The following example gets the contents in Rich Text Format (RTF).
ctlClipboardText.Rtf = (string)iData.GetData(DataFormats.Rtf);
After performing any processing that the application requires the next window
in the Clipboard Viewer chain must be notified of the Clipboard change. This
is done by calling the Windows API function SendMessage
. The same parameters that were passed to the WndProc
method are
passed to SendMessage
to be used by the next Clipboard Viewer in
the chain.
SendMessage(_ClipboardViewerNext, m.Msg, m.WParam, m.LParam);
When overriding WndProc
I must pass any unhandled messages to the
base class by calling base.WndProc
protected override void WndProc(ref Message m) { switch ((Win32.Msgs)m.Msg) { case Win32.Msgs.WM_DRAWCLIPBOARD: // ... process Clipboard default: // unhandled window message base.WndProc(ref m); break;
The WM_CHANGECBCHAIN Message
The WM_CHANGECBCHAIN
message is sent when a window is being
removed from the chain of Clipboard Viewers. Note that the message is only
sent to the first window in the Clipboard Viewer chain. The first window and
subsequent windows in the chain must use SendMessage
to pass the
message along the chain.
If wParam
is equal to the handle of the next Clipboard Viewer
then the next Clipboard Viewer is being removed and I need to update the
pointer that refers to the next window in the chain. m.LParam
is
a pointer to the Clipboard Viewer in the chain after the one being removed.
if (m.WParam == _ClipboardViewerNext) { _ClipboardViewerNext = m.LParam; } else { SendMessage(_ClipboardViewerNext, m.Msg, m.WParam, m.LParam); }
This message can be seen by starting multiple copies of the Clipboard Viewer and watching the Debug output.
Unregistering the Application as a Clipboard Viewer
When the application closes I make sure I call ChangeClipboardChain to remove the application from the Clipboard chain.
ChangeClipboardChain(this.Handle, _ClipboardViewerNext);
And Finally...
There are a few tricks to using the SetClipboardViewer
function but it allows your application to be notified of Clipboard events.