|
|
/*****************************************************************************
* * fndcm.c - IContextMenu interface * *****************************************************************************/
#include "fnd.h"
#include <wab.h>
#include <shlwapi.h>
#ifdef _WIN64
#pragma pack(push,8)
#endif // _WIN64
/*****************************************************************************
* * The sqiffle for this file. * *****************************************************************************/
#define sqfl sqflCm
/*****************************************************************************
* * PICI * * I'm getting lazy. * *****************************************************************************/
typedef LPCMINVOKECOMMANDINFO PICI;
/*****************************************************************************
* * Declare the interfaces we will be providing. * * We must implement an IShellExtInit so the shell * will know that we are ready for action. * *****************************************************************************/
Primary_Interface(CFndCm, IContextMenu); Secondary_Interface(CFndCm, IShellExtInit);
/*****************************************************************************
* * CFndCm * * The context menu extension for "Find... &People". * *****************************************************************************/
typedef struct CFndCm {
/* Supported interfaces */ IContextMenu cm; IShellExtInit sxi;
} CFndCm, FCM, *PFCM;
typedef IContextMenu CM, *PCM; typedef IShellExtInit SXI, *PSXI; typedef IDataObject DTO, *PDTO;
/*****************************************************************************
* * CFndCm_QueryInterface (from IUnknown) * * We need to check for our additional interfaces before falling * through to Common_QueryInterface. * *****************************************************************************/
STDMETHODIMP CFndCm_QueryInterface(PCM pcm, RIID riid, PPV ppvObj) { PFCM this = IToClass(CFndCm, cm, pcm); HRESULT hres; if (IsEqualIID(riid, &IID_IShellExtInit)) { *ppvObj = &this->sxi; Common_AddRef(this); hres = NOERROR; } else { hres = Common_QueryInterface(this, riid, ppvObj); } AssertF(fLimpFF(FAILED(hres), *ppvObj == 0)); return hres; }
/*****************************************************************************
* * CFndCm_AddRef (from IUnknown) * CFndCm_Release (from IUnknown) * *****************************************************************************/
#define CFndCm_AddRef Common_AddRef
#define CFndCm_Release Common_Release
/*****************************************************************************
* * CFndCm_Finalize (from Common) * * Release the resources of an CFndCm. * *****************************************************************************/
void EXTERNAL CFndCm_Finalize(PV pv) { PFCM this = pv;
EnterProc(CFndCm_Finalize, (_ "p", pv));
ExitProc(); }
/*****************************************************************************
* * CFndCm_QueryContextMenu (From IContextMenu) * * Given an existing context menu hmenu, insert new context menu * items at location imi (imi = index to menu imi), returning the * number of menu items added. * * Our job is to add the "Find... People" menu option. * * hmenu - destination menu * imi - location at which menu items should be inserted * idcMin - first available menu identifier * idcMax - first unavailable menu identifier * *****************************************************************************/
#pragma BEGIN_CONST_DATA
TCHAR c_tszMyself[] = TEXT(".{32714800-2E5F-11d0-8B85-00AA0044F941}");
#pragma END_CONST_DATA
STDMETHODIMP CFndCm_QueryContextMenu(PCM pcm, HMENU hmenu, UINT imi, UINT idcMin, UINT idcMax, UINT uFlags) { PFCM this = IToClass(CFndCm, cm, pcm); HRESULT hres; MENUITEMINFO mii; TCHAR tsz[256]; SHFILEINFO sfi; EnterProc(CFndCm_QueryContextMenu, (_ "pu", pcm, idcMin));
LoadString(g_hinst, IDS_ONTHEINTERNET, tsz, cA(tsz));
SHGetFileInfo(c_tszMyself, FILE_ATTRIBUTE_DIRECTORY, &sfi, cbX(sfi), SHGFI_SMALLICON | SHGFI_SYSICONINDEX | SHGFI_USEFILEATTRIBUTES);
mii.cbSize = sizeof(mii); mii.fMask = MIIM_DATA | MIIM_ID | MIIM_TYPE; mii.fType = MFT_STRING; mii.fState = MFS_UNCHECKED; mii.wID = idcMin; mii.dwItemData = sfi.iIcon; mii.dwTypeData = tsz;
InsertMenuItem(hmenu, imi, TRUE, &mii);
hres = hresUs(1);
ExitOleProc(); return hres; }
const static TCHAR lpszWABRegPathKey[] = TEXT("Software\\Microsoft\\WAB\\DLLPath"); const static TCHAR lpszWABDll[] = TEXT("Wab32.dll");
// GetWABDllPath - loads the WAB DLL path from the registry
// szPath - ptr to buffer
// cb - sizeof buffer
//
void GetWABDllPath(LPTSTR szPath, ULONG cb) { DWORD dwType = 0; ULONG cbData = cb; HKEY hKey = NULL; if(szPath) { *szPath = '\0'; if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, lpszWABRegPathKey, 0, KEY_READ, &hKey)) RegQueryValueEx( hKey, "", NULL, &dwType, (LPBYTE) szPath, &cbData); } if(hKey) RegCloseKey(hKey); return; }
// LoadLibrary_WABDll() - Load the WAB library based on the WAB DLL path
//
HINSTANCE LoadLibrary_WABDll() { TCHAR szWABDllPath[MAX_PATH];
// [PaulHi] We only need one reference count on the wab32.dll. The wabfind.dll
// will unload the wab32.dll when it unloads. The wabfind.dll won't unload
// until all CMFind threads are finished through the global g_cRef variable.
if (g_hinstWABDLL) return g_hinstWABDLL;
GetWABDllPath(szWABDllPath, sizeof(szWABDllPath));
// if you want an IE4.0x WAB or later then you should fail here if the WAB
// DLL path could not be retreived from the registry
// Otherwise, if you don't care, you can just do a LoadLibrary("wab32.dll")
return(g_hinstWABDLL = LoadLibrary( (lstrlen(szWABDllPath)) ? szWABDllPath : lpszWABDll )); }
//
// Initialize the WAB and get an instance of IWABObject and IAddrBook
//
HRESULT InitWAB(LPWABOBJECT * lppWABObject, LPADRBOOK * lppAdrBook) { HRESULT hr = E_FAIL; LPWABOPEN lpfnWABOpen = NULL; // defined in WABAPI.H
HINSTANCE hinstWAB = NULL; WAB_PARAM WP = {0};
WP.ulFlags = WAB_ENABLE_PROFILES;
hinstWAB = LoadLibrary_WABDll();
if(hinstWAB) { lpfnWABOpen = (LPWABOPEN) GetProcAddress(hinstWAB, "WABOpen"); if(lpfnWABOpen) hr = lpfnWABOpen(lppAdrBook, lppWABObject, &WP, 0); }
// Remember to release the IAddrBook and IWABObject objects retrieved above
return hr; }
/*
- - WABThreadProc * * Since the WAB Find thing is a dialog, it is blocking Explorer's thread * and user's can't access the Start menu anymore .. so we put the dialog * on a seperate thread * */ DWORD WINAPI WABThreadProc( LPVOID lpParam ) { HRESULT hres; LPWABOBJECT lpWABObject = NULL; LPADRBOOK lpAdrBook = NULL;
if(!HR_FAILED(hres = InitWAB(&lpWABObject,&lpAdrBook))) { hres = lpWABObject->lpVtbl->Find(lpWABObject,lpAdrBook,NULL);
// Release the WAB and AB objects
lpAdrBook->lpVtbl->Release(lpAdrBook); lpWABObject->lpVtbl->Release(lpWABObject); }
// Our work is done, wabfind.dll can safely go away
InterlockedDecrement((LPLONG)&g_cRef);
return 0; } /*****************************************************************************
* * _CFndCm_InvokeFind * *****************************************************************************/
STDMETHODIMP _CFndCm_InvokeFind(void) {
HANDLE hThread = NULL; DWORD dwThreadID = 0;
// Increment the refcount so that when the Shell continues (the Invoke returns)
// wabfind.dll doesn't unload when the Shell releases the FindCM
InterlockedIncrement((LPLONG)&g_cRef);
hThread = CreateThread( NULL, // no security attributes
0, // use default stack size
WABThreadProc, // thread function
(LPVOID) NULL, // argument to thread function
0, // use default creation flags
&dwThreadID); // returns the thread identifier
CloseHandle(hThread); return S_OK; }
/*****************************************************************************
* * CFndCm_InvokeCommand (from IContextMenu) * * We have only one command, called "find". * *****************************************************************************/
#pragma BEGIN_CONST_DATA
TCHAR c_tszFind[] = TEXT("find");
#pragma END_CONST_DATA
STDMETHODIMP CFndCm_InvokeCommand(PCM pcm, PICI pici) { PFCM this = IToClass(CFndCm, cm, pcm); HRESULT hres; EnterProc(CFndCm_InvokeCommand, (_ HIWORD(pici->lpVerb) ? "pA" : "pu", pcm, pici->lpVerb));
if (pici->cbSize >= sizeof(*pici)) { if ( #ifdef SHELL32_IS_BUG_FREE //;Internal
(HIWORD(pici->lpVerb) && lstrcmpi(c_tszFind, pici->lpVerb) == 0) || //;Internal
pici->lpVerb == 0 //;Internal
#else //;Internal
fLimpFF(HIWORD(pici->lpVerb), lstrcmpi(c_tszFind, pici->lpVerb) == 0) #endif //;Internal
) { hres = _CFndCm_InvokeFind(); } else { hres = E_INVALIDARG; } } else { hres = E_INVALIDARG; } ExitOleProc(); return hres; }
/*****************************************************************************
* * CFndCm_GetCommandString (from IContextMenu) * * Somebody wants to convert a command id into a string of some sort. * *****************************************************************************/
STDMETHODIMP CFndCm_GetCommandString(PCM pcm, UINT_PTR idCmd, UINT uFlags, UINT *pwRsv, LPSTR pszName, UINT cchMax) { PFCM this = IToClass(CFndCm, cm, pcm); HRESULT hres; EnterProc(CFndCm_GetCommandString, (_ "uu", idCmd, uFlags));
if (idCmd == 0) { switch (uFlags) { case GCS_HELPTEXT: if (cchMax) { pszName[0] = '\0'; if (LoadString(g_hinst, IDS_FINDHELP, pszName, cchMax)) { hres = NOERROR; } else { hres = E_INVALIDARG; } } else { hres = E_INVALIDARG; } break;
case GCS_VALIDATE: hres = NOERROR; break;
case GCS_VERB: StrCpyN(pszName, c_tszFind, cchMax); hres = NOERROR; break;
default: hres = E_NOTIMPL; break; } } else { hres = E_INVALIDARG; }
ExitOleProc(); return hres; }
/*****************************************************************************
* * CFndCm_SXI_Initialize (from IShellExtension) * *****************************************************************************/
STDMETHODIMP CFndCm_SXI_Initialize(PSXI psxi, PCIDL pidlFolder, PDTO pdto, HKEY hk) { PFCM this = IToClass(CFndCm, sxi, psxi); HRESULT hres; EnterProc(CFndCm_SXI_Initialize, (_ ""));
hres = S_OK; ExitOleProc(); return hres; }
/*****************************************************************************
* * CFndCm_New (from IClassFactory) * * Note that we release the pfcm that Common_New created, because we * are done with it. The real refcount is handled by the * CFndCm_QueryInterface. * *****************************************************************************/
STDMETHODIMP CFndCm_New(RIID riid, PPV ppvObj) { HRESULT hres; EnterProc(CFndCm_New, (_ "G", riid));
*ppvObj = 0; hres = Common_New(CFndCm, ppvObj); if (SUCCEEDED(hres)) { PFCM pfcm = *ppvObj; pfcm->sxi.lpVtbl = Secondary_Vtbl(CFndCm, IShellExtInit); hres = CFndCm_QueryInterface(&pfcm->cm, riid, ppvObj); Common_Release(pfcm); }
ExitOleProcPpv(ppvObj); return hres; }
/*****************************************************************************
* * The long-awaited vtbls * *****************************************************************************/
#pragma BEGIN_CONST_DATA
Primary_Interface_Begin(CFndCm, IContextMenu) CFndCm_QueryContextMenu, CFndCm_InvokeCommand, CFndCm_GetCommandString, Primary_Interface_End(CFndCm, IContextMenu)
Secondary_Interface_Begin(CFndCm, IShellExtInit, sxi) CFndCm_SXI_Initialize, Secondary_Interface_End(CFndCm, IShellExtInit, sxi)
#ifdef _WIN64
#pragma pack(pop)
#endif //_WIN64
|