Using the Shell to Receive Notification of Removable Media Being Inserted or Removed





5.00/5 (12 votes)
Nov 21, 2002
1 min read

193395
Uses the poorly documented SHChangeNotifyRegister function to receive notification upon shell events
Introduction
Recently, I needed to determine when removable media, such as a zip disk or media card, had been inserted by the user.
I initially looked into the WM_DEVICECHANGE
broadcast message only to realise this supports CDROMs and little else. Fortunately, there is another way via the shell, through the scarcely documented SHChangeNotifyRegister
function.
We can register to be informed by the shell when certain events of interest occur. The events are numerous - covering when a drive is added or removed, files are renamed, etc. However of particular interest here are the notifications for media being inserted or removed.
It's scarcely rocket science but piecing the various bits of information together to utilise this functionality took far longer than it should have. I was amazed there was no article here on CodeProject covering this, so figured I should rectify that!
Implementation
You will need the following include
s and definition:
#include <shlobj.h>
#define WM_USER_MEDIACHANGED WM_USER+88
Together with a member variable to release the request later:
ULONG m_ulSHChangeNotifyRegister;
Nullify the member variable in your constructor, then register to be notified by the shell (perhaps in OnInitDialog
for example):
CMyWnd::CMyWnd()
{
ulSHChangeNotifyRegister = NULL;
}
BOOL CMyWnd::OnInitDialog()
{
CWnd::OnInitDialog();
// Request notification from shell of media insertion -
// allows us to detect removable media or a multimedia card
HWND hWnd = GetSafeHwnd();
LPITEMIDLIST ppidl;
if(SHGetSpecialFolderLocation(hWnd, CSIDL_DESKTOP, &ppidl) == NOERROR)
{
SHChangeNotifyEntry shCNE;
shCNE.pidl = ppidl;
shCNE.fRecursive = TRUE;
// Returns a positive integer registration identifier (ID).
// Returns zero if out of memory or in response to invalid parameters.
m_ulSHChangeNotifyRegister = SHChangeNotifyRegister(hWnd,
// Hwnd to receive notification
SHCNE_DISKEVENTS,
// Event types of interest (sources)
SHCNE_MEDIAINSERTED|SHCNE_MEDIAREMOVED,
// Events of interest - use
// SHCNE_ALLEVENTS for all events
WM_USER_MEDIACHANGED,
// Notification message to be sent
// upon the event
1,
// Number of entries in the pfsne array
&shCNE); // Array of SHChangeNotifyEntry structures that
// contain the notifications. This array should
// always be set to one when calling SHChnageNotifyRegister
// or SHChangeNotifyDeregister will not work properly.
ASSERT(m_ulSHChangeNotifyRegister != 0); // Shell notification failed
}
else
ASSERT(FALSE); // Failed to get desktop location
}
Add a message handler to your message map for the notification message:
ON_MESSAGE(WM_USER_MEDIACHANGED, OnMediaChanged)
Together with the corresponding declaration:
afx_msg LRESULT OnMediaChanged(WPARAM, LPARAM);
Then deal with the message as desired:
// The lParam value contains the event SHCNE_xxxx
// The wParam value supplies the SHNOTIFYSTRUCT
typedef struct {
DWORD dwItem1; // dwItem1 contains the previous PIDL or name of the folder.
DWORD dwItem2; // dwItem2 contains the new PIDL or name of the folder.
} SHNOTIFYSTRUCT;
LRESULT CMyWnd::OnMediaChanged(WPARAM wParam, LPARAM lParam)
{
SHNOTIFYSTRUCT *shns = (SHNOTIFYSTRUCT *)wParam;
CString strPath, strMsg;
switch(lParam)
{
case SHCNE_MEDIAINSERTED: // media inserted event
{
strPath = GetPathFromPIDL(shns->dwItem1);
if(!strPath.IsEmpty())
{
// Process strPath as required, for now a simple message box
strMsg.Format("Media inserted into %s", strPath);
AfxMessageBox(strMsg);
}
}
case SHCNE_MEDIAREMOVED: // media removed event
{
strPath = GetPathFromPIDL(shns->dwItem1);
if(!strPath.IsEmpty())
{
// Process strPath as required, for now a simple message box
strMsg.Format("Media removed from %s", strPath);
AfxMessageBox(strMsg);
}
}
//case SHCNE_xxxx: Add other events processing here
}
return NULL;
}
CString CMyWnd::GetPathFromPIDL(DWORD pidl)
{
char sPath[MAX_PATH];
CString strTemp = _T("");
if(SHGetPathFromIDList((struct _ITEMIDLIST *)pidl, sPath))
strTemp = sPath;
return strTemp;
}
Finally, deregister your request when no longer required:
void CMyWnd::OnDestroy()
{
if(m_ulSHChangeNotifyRegister)
VERIFY(SHChangeNotifyDeregister(m_ulSHChangeNotifyRegister));
CWnd::OnDestroy();
}
References
- MSDN: http://msdn.microsoft.com/library/default.asp?url=/library/en-us/shellcc/platform/shell/reference/Functions/SHChangeNotifyRegister.asp[^]
- Sameer Maggon: http://pcquest.ciol.com/content/coding/102070301.asp[^]
License
This article has no explicit license attached to it, but may contain usage terms in the article text or the download files themselves. If in doubt, please contact the author via the discussion board below.
A list of licenses authors might use can be found here.