876 lines
22 KiB
C++
876 lines
22 KiB
C++
#include "private.h"
|
|
#include "subsmgrp.h"
|
|
#include "subitem.h"
|
|
|
|
// Contains implementations of IEnumSubscription and ISubscriptionMgr2
|
|
|
|
HRESULT SubscriptionItemFromCookie(BOOL fCreateNew, const SUBSCRIPTIONCOOKIE *pCookie, ISubscriptionItem **ppSubscriptionItem);
|
|
|
|
HRESULT DoGetItemFromURL(LPCTSTR pszURL, ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
HRESULT hr;
|
|
SUBSCRIPTIONCOOKIE cookie;
|
|
|
|
if ((NULL == pszURL) || (NULL == ppSubscriptionItem))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
hr = ReadCookieFromInetDB(pszURL, &cookie);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = SubscriptionItemFromCookie(FALSE, &cookie, ppSubscriptionItem);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT DoGetItemFromURLW(LPCWSTR pwszURL, ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
TCHAR szURL[INTERNET_MAX_URL_LENGTH];
|
|
|
|
if ((NULL == pwszURL) || (NULL == ppSubscriptionItem))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
StrCpyN(szURL, pwszURL, ARRAYSIZE(szURL));
|
|
#else
|
|
MyOleStrToStrN(szURL, ARRAYSIZE(szURL), pwszURL);
|
|
#endif
|
|
|
|
return DoGetItemFromURL(szURL, ppSubscriptionItem);
|
|
}
|
|
|
|
|
|
HRESULT DoAbortItems(
|
|
/* [in] */ DWORD dwNumCookies,
|
|
/* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if ((0 == dwNumCookies) || (NULL == pCookies))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
ISubscriptionThrottler *pst;
|
|
|
|
hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pst->AbortItems(dwNumCookies, pCookies);
|
|
pst->Release();
|
|
}
|
|
else
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT DoCreateSubscriptionItem(
|
|
/* [in] */ const SUBSCRIPTIONITEMINFO *pSubscriptionItemInfo,
|
|
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
|
|
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
HRESULT hr;
|
|
ISubscriptionItem *psi;
|
|
|
|
if ((NULL == pSubscriptionItemInfo) || (NULL == pNewCookie) || (NULL == ppSubscriptionItem))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*ppSubscriptionItem = NULL;
|
|
|
|
CreateCookie(pNewCookie);
|
|
|
|
hr = SubscriptionItemFromCookie(TRUE, pNewCookie, &psi);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ASSERT(NULL != psi);
|
|
|
|
hr = psi->SetSubscriptionItemInfo(pSubscriptionItemInfo);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
*ppSubscriptionItem = psi;
|
|
}
|
|
else
|
|
{
|
|
// Don't leak or leave slop hanging around
|
|
psi->Release();
|
|
DoDeleteSubscriptionItem(pNewCookie, FALSE);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT DoCloneSubscriptionItem(
|
|
/* [in] */ ISubscriptionItem *pSubscriptionItem,
|
|
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
|
|
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
HRESULT hr;
|
|
SUBSCRIPTIONCOOKIE NewCookie;
|
|
|
|
if ((NULL == pSubscriptionItem) || (NULL == ppSubscriptionItem))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
IEnumItemProperties *peip;
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
|
|
*ppSubscriptionItem = NULL;
|
|
|
|
// First get existing subscription details
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
hr = pSubscriptionItem->GetSubscriptionItemInfo(&sii);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISubscriptionItem *psi;
|
|
|
|
// Mark as temp and create a new subscription item
|
|
sii.dwFlags |= SI_TEMPORARY;
|
|
|
|
hr = DoCreateSubscriptionItem(&sii, &NewCookie, &psi);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (pNewCookie)
|
|
*pNewCookie = NewCookie;
|
|
|
|
// Get properties from existing item
|
|
hr = pSubscriptionItem->EnumProperties(&peip);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ULONG count;
|
|
|
|
ASSERT(NULL != peip);
|
|
|
|
hr = peip->GetCount(&count);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ITEMPROP *pProps = new ITEMPROP[count];
|
|
LPWSTR *pNames = new LPWSTR[count];
|
|
VARIANT *pVars = new VARIANT[count];
|
|
|
|
if ((NULL != pProps) && (NULL != pNames) && (NULL != pVars))
|
|
{
|
|
hr = peip->Next(count, pProps, &count);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ULONG i;
|
|
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
pNames[i] = pProps[i].pwszName;
|
|
pVars[i] = pProps[i].variantValue;
|
|
}
|
|
|
|
hr = psi->WriteProperties(count, pNames, pVars);
|
|
|
|
// clean up from enum
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
if (pProps[i].pwszName)
|
|
{
|
|
CoTaskMemFree(pProps[i].pwszName);
|
|
}
|
|
VariantClear(&pProps[i].variantValue);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
SAFEDELETE(pProps);
|
|
SAFEDELETE(pNames);
|
|
SAFEDELETE(pVars);
|
|
}
|
|
|
|
peip->Release();
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
*ppSubscriptionItem = psi;
|
|
}
|
|
else
|
|
{
|
|
psi->Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT DoDeleteSubscriptionItem(
|
|
/* [in] */ const SUBSCRIPTIONCOOKIE *pCookie,
|
|
/* [in] */ BOOL fAbortItem)
|
|
{
|
|
HRESULT hr;
|
|
TCHAR szKey[MAX_PATH];
|
|
|
|
if (NULL == pCookie)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (fAbortItem)
|
|
{
|
|
DoAbortItems(1, pCookie);
|
|
}
|
|
|
|
if (ItemKeyNameFromCookie(pCookie, szKey, ARRAYSIZE(szKey)))
|
|
{
|
|
// Notify the agent that it is about to get deleted.
|
|
ISubscriptionItem *pItem=NULL;
|
|
if (SUCCEEDED(SubscriptionItemFromCookie(FALSE, pCookie, &pItem)))
|
|
{
|
|
SUBSCRIPTIONITEMINFO sii = { sizeof(SUBSCRIPTIONITEMINFO) };
|
|
|
|
if (SUCCEEDED(pItem->GetSubscriptionItemInfo(&sii)))
|
|
{
|
|
ASSERT(!(sii.dwFlags & SI_TEMPORARY));
|
|
|
|
ISubscriptionAgentControl *psac=NULL;
|
|
|
|
if (SUCCEEDED(CoCreateInstance(sii.clsidAgent, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionAgentControl, (void**)&psac)))
|
|
{
|
|
psac->SubscriptionControl(pItem, SUBSCRIPTION_AGENT_DELETE);
|
|
psac->Release();
|
|
}
|
|
|
|
FireSubscriptionEvent(SUBSNOTF_DELETE, pCookie);
|
|
|
|
if (GUID_NULL != sii.ScheduleGroup)
|
|
{
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
pSyncScheduleMgr->RemoveSchedule(&sii.ScheduleGroup);
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
pItem->Release();
|
|
}
|
|
|
|
hr = (SHDeleteKey(HKEY_CURRENT_USER, szKey) == ERROR_SUCCESS) ? S_OK : E_FAIL;
|
|
}
|
|
else
|
|
{
|
|
TraceMsg(TF_ALWAYS, "Failed to delete subscription item.");
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT AddUpdateSubscription(SUBSCRIPTIONCOOKIE *pCookie,
|
|
SUBSCRIPTIONITEMINFO *psii,
|
|
LPCWSTR pwszURL,
|
|
ULONG nProps,
|
|
const LPWSTR rgwszName[],
|
|
VARIANT rgValue[])
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
ASSERT((0 == nProps) || ((NULL != rgwszName) && (NULL != rgValue)));
|
|
|
|
TCHAR szURL[INTERNET_MAX_URL_LENGTH];
|
|
ISubscriptionItem *psi = NULL;
|
|
SUBSCRIPTIONCOOKIE cookie;
|
|
|
|
#ifdef UNICODE
|
|
StrCpyNW(szURL, pwszURL, ARRAYSIZE(szURL));
|
|
#else
|
|
MyOleStrToStrN(szURL, ARRAYSIZE(szURL), pwszURL);
|
|
#endif
|
|
|
|
// Try and get the cookie from the inet db otherwise
|
|
// create a new one.
|
|
|
|
if (*pCookie == CLSID_NULL)
|
|
{
|
|
CreateCookie(&cookie);
|
|
}
|
|
else
|
|
{
|
|
cookie = *pCookie;
|
|
}
|
|
// Update the inet db
|
|
WriteCookieToInetDB(szURL, &cookie, FALSE);
|
|
|
|
hr = SubscriptionItemFromCookie(TRUE, &cookie, &psi);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = psi->SetSubscriptionItemInfo(psii);
|
|
if (SUCCEEDED(hr) && (nProps > 0))
|
|
{
|
|
ASSERT(NULL != psi);
|
|
hr = psi->WriteProperties(nProps, rgwszName, rgValue);
|
|
}
|
|
psi->Release();
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
DoDeleteSubscriptionItem(&cookie, TRUE);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT SubscriptionItemFromCookie(BOOL fCreateNew, const SUBSCRIPTIONCOOKIE *pCookie, ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
HRESULT hr;
|
|
|
|
ASSERT((NULL != pCookie) && (NULL != ppSubscriptionItem));
|
|
|
|
HKEY hkey;
|
|
|
|
if (OpenItemKey(pCookie, fCreateNew ? TRUE : FALSE, KEY_READ | KEY_WRITE, &hkey))
|
|
{
|
|
*ppSubscriptionItem = new CSubscriptionItem(pCookie, hkey);
|
|
if (NULL != *ppSubscriptionItem)
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
RegCloseKey(hkey);
|
|
}
|
|
else
|
|
{
|
|
*ppSubscriptionItem = NULL;
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
BOOL ItemKeyNameFromCookie(const SUBSCRIPTIONCOOKIE *pCookie, TCHAR *pszKeyName, DWORD cchKeyName)
|
|
{
|
|
WCHAR wszGuid[GUIDSTR_MAX];
|
|
|
|
ASSERT((NULL != pCookie) && (NULL != pszKeyName) && (cchKeyName >= ARRAYSIZE(WEBCHECK_REGKEY_STORE) + GUIDSTR_MAX));
|
|
|
|
if (StringFromGUID2(*pCookie, wszGuid, ARRAYSIZE(wszGuid)))
|
|
{
|
|
#ifdef UNICODE
|
|
wnsprintfW(pszKeyName, cchKeyName, L"%s\\%s", c_szRegKeyStore, wszGuid);
|
|
#else
|
|
wnsprintfA(pszKeyName, cchKeyName, "%s\\%S", c_szRegKeyStore, wszGuid);
|
|
#endif
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
BOOL OpenItemKey(const SUBSCRIPTIONCOOKIE *pCookie, BOOL fCreateNew, REGSAM samDesired, HKEY *phkey)
|
|
{
|
|
TCHAR szKeyName[MAX_PATH];
|
|
|
|
ASSERT((NULL != pCookie) && (NULL != phkey));
|
|
|
|
if (ItemKeyNameFromCookie(pCookie, szKeyName, ARRAYSIZE(szKeyName)))
|
|
{
|
|
if (fCreateNew)
|
|
{
|
|
DWORD dwDisposition;
|
|
return RegCreateKeyEx(HKEY_CURRENT_USER, szKeyName, 0, NULL, REG_OPTION_NON_VOLATILE, samDesired, NULL, phkey, &dwDisposition) == ERROR_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
return RegOpenKeyEx(HKEY_CURRENT_USER, szKeyName, 0, samDesired, phkey) == ERROR_SUCCESS;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
// ISubscriptionMgr2 members
|
|
|
|
STDMETHODIMP CSubscriptionMgr::GetItemFromURL(
|
|
/* [in] */ LPCWSTR pwszURL,
|
|
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
return DoGetItemFromURLW(pwszURL, ppSubscriptionItem);
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::GetItemFromCookie(
|
|
/* [in] */ const SUBSCRIPTIONCOOKIE *pSubscriptionCookie,
|
|
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
if ((NULL == pSubscriptionCookie) || (NULL == ppSubscriptionItem))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
return SubscriptionItemFromCookie(FALSE, pSubscriptionCookie, ppSubscriptionItem);
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::GetSubscriptionRunState(
|
|
/* [in] */ DWORD dwNumCookies,
|
|
/* [in] */ const SUBSCRIPTIONCOOKIE *pSubscriptionCookies,
|
|
/* [out] */ DWORD *pdwRunState)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if ((0 == dwNumCookies) || (NULL == pSubscriptionCookies) || (NULL == pdwRunState))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
ISubscriptionThrottler *pst;
|
|
|
|
hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pst->GetSubscriptionRunState(dwNumCookies, pSubscriptionCookies, pdwRunState);
|
|
pst->Release();
|
|
}
|
|
else
|
|
{
|
|
// Couldn't connect to a running throttler so assume nothing
|
|
// is running.
|
|
for (DWORD i = 0; i < dwNumCookies; i++)
|
|
{
|
|
*pdwRunState++ = 0;
|
|
}
|
|
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::EnumSubscriptions(
|
|
/* [in] */ DWORD dwFlags,
|
|
/* [out] */ IEnumSubscription **ppEnumSubscriptions)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if ((dwFlags & ~SUBSMGRENUM_MASK) || (NULL == ppEnumSubscriptions))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
CEnumSubscription *pes = new CEnumSubscription;
|
|
|
|
*ppEnumSubscriptions = NULL;
|
|
|
|
if (NULL != pes)
|
|
{
|
|
hr = pes->Initialize(dwFlags);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pes->QueryInterface(IID_IEnumSubscription, (void **)ppEnumSubscriptions);
|
|
}
|
|
pes->Release();
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::UpdateItems(
|
|
/* [in] */ DWORD dwFlags,
|
|
/* [in] */ DWORD dwNumCookies,
|
|
/* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if ((dwFlags & ~SUBSMGRUPDATE_MASK) || (0 == dwNumCookies) || (NULL == pCookies))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
// Fail if restrictions are in place.
|
|
// BUGBUG: Should we have a flag parameter to override this?
|
|
if (SHRestricted2W(REST_NoManualUpdates, NULL, 0))
|
|
{
|
|
SGMessageBox(NULL, IDS_RESTRICTED, MB_OK);
|
|
return E_ACCESSDENIED;
|
|
}
|
|
|
|
ISyncMgrSynchronizeInvoke *pSyncMgrInvoke;
|
|
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncMgrSynchronizeInvoke, (void **)&pSyncMgrInvoke);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
DWORD dwInvokeFlags = SYNCMGRINVOKE_STARTSYNC;
|
|
|
|
if (dwFlags & SUBSMGRUPDATE_MINIMIZE)
|
|
{
|
|
dwInvokeFlags |= SYNCMGRINVOKE_MINIMIZED;
|
|
}
|
|
hr = pSyncMgrInvoke->UpdateItems(dwInvokeFlags, CLSID_WebCheckOfflineSync, dwNumCookies * sizeof(SUBSCRIPTIONCOOKIE), (const BYTE *)pCookies);
|
|
pSyncMgrInvoke->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::AbortItems(
|
|
/* [in] */ DWORD dwNumCookies,
|
|
/* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies)
|
|
{
|
|
return DoAbortItems(dwNumCookies, pCookies);
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::AbortAll()
|
|
{
|
|
HRESULT hr;
|
|
|
|
ISubscriptionThrottler *pst;
|
|
|
|
hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pst->AbortAll();
|
|
pst->Release();
|
|
}
|
|
else
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// ISubscriptionMgrPriv
|
|
STDMETHODIMP CSubscriptionMgr::CreateSubscriptionItem(
|
|
/* [in] */ const SUBSCRIPTIONITEMINFO *pSubscriptionItemInfo,
|
|
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
|
|
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
return DoCreateSubscriptionItem(pSubscriptionItemInfo, pNewCookie, ppSubscriptionItem);
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::CloneSubscriptionItem(
|
|
/* [in] */ ISubscriptionItem *pSubscriptionItem,
|
|
/* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie,
|
|
/* [out] */ ISubscriptionItem **ppSubscriptionItem)
|
|
{
|
|
return DoCloneSubscriptionItem(pSubscriptionItem, pNewCookie, ppSubscriptionItem);
|
|
}
|
|
|
|
|
|
STDMETHODIMP CSubscriptionMgr::DeleteSubscriptionItem(
|
|
/* [in] */ const SUBSCRIPTIONCOOKIE *pCookie)
|
|
{
|
|
return DoDeleteSubscriptionItem(pCookie, TRUE);
|
|
}
|
|
|
|
// ** CEnumSubscription **
|
|
|
|
|
|
CEnumSubscription::CEnumSubscription()
|
|
{
|
|
ASSERT(NULL == m_pCookies);
|
|
ASSERT(0 == m_nCurrent);
|
|
ASSERT(0 == m_nCount);
|
|
|
|
m_cRef = 1;
|
|
|
|
DllAddRef();
|
|
}
|
|
|
|
|
|
CEnumSubscription::~CEnumSubscription()
|
|
{
|
|
if (NULL != m_pCookies)
|
|
{
|
|
delete [] m_pCookies;
|
|
}
|
|
|
|
DllRelease();
|
|
}
|
|
|
|
|
|
HRESULT CEnumSubscription::Initialize(DWORD dwFlags)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
HKEY hkey = NULL;
|
|
DWORD dwDisposition;
|
|
|
|
ASSERT(0 == m_nCount);
|
|
|
|
if (RegCreateKeyEx(HKEY_CURRENT_USER, c_szRegKeyStore, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_READ | KEY_WRITE, NULL, &hkey, &dwDisposition) == ERROR_SUCCESS)
|
|
{
|
|
DWORD nCount;
|
|
|
|
if (RegQueryInfoKey(hkey,
|
|
NULL, // address of buffer for class string
|
|
NULL, // address of size of class string buffer
|
|
NULL, // reserved
|
|
&nCount, // address of buffer for number of subkeys
|
|
NULL, // address of buffer for longest subkey name length
|
|
NULL, // address of buffer for longest class string length
|
|
NULL, // address of buffer for number of value entries
|
|
NULL, // address of buffer for longest value name length
|
|
NULL, // address of buffer for longest value data length
|
|
NULL, // address of buffer for security descriptor length
|
|
NULL // address of buffer for last write time
|
|
) == ERROR_SUCCESS)
|
|
{
|
|
SUBSCRIPTIONCOOKIE Cookie;
|
|
m_pCookies = new SUBSCRIPTIONCOOKIE[nCount];
|
|
|
|
if (NULL != m_pCookies)
|
|
{
|
|
TCHAR szKeyName[GUIDSTR_MAX];
|
|
|
|
hr = S_OK;
|
|
|
|
for (ULONG i = 0; (i < nCount) && (S_OK == hr); i++)
|
|
{
|
|
if (RegEnumKey(hkey, i, szKeyName, ARRAYSIZE(szKeyName)) == ERROR_SUCCESS)
|
|
{
|
|
HRESULT hrConvert;
|
|
#ifdef UNICODE
|
|
hrConvert = CLSIDFromString(szKeyName, &Cookie);
|
|
#else
|
|
WCHAR wszKeyName[GUIDSTR_MAX];
|
|
MultiByteToWideChar(CP_ACP, 0, szKeyName, -1,
|
|
wszKeyName, ARRAYSIZE(wszKeyName));
|
|
hrConvert = CLSIDFromString(wszKeyName, &Cookie);
|
|
#endif
|
|
if (SUCCEEDED(hrConvert))
|
|
{
|
|
ISubscriptionItem *psi;
|
|
|
|
m_pCookies[m_nCount] = Cookie;
|
|
|
|
hr = SubscriptionItemFromCookie(FALSE, &Cookie, &psi);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
|
|
if (SUCCEEDED(psi->GetSubscriptionItemInfo(&sii)))
|
|
{
|
|
// Only count this if it's not a temporary
|
|
// or the caller asked for temporary items.
|
|
if ((!(sii.dwFlags & SI_TEMPORARY)) || (dwFlags & SUBSMGRENUM_TEMP))
|
|
{
|
|
m_nCount++;
|
|
}
|
|
}
|
|
psi->Release();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// IUnknown members
|
|
STDMETHODIMP CEnumSubscription::QueryInterface(REFIID riid, void **ppv)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (NULL == ppv)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if ((IID_IUnknown == riid) || (IID_IEnumSubscription == riid))
|
|
{
|
|
*ppv = (IEnumSubscription *)this;
|
|
AddRef();
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
*ppv = NULL;
|
|
hr = E_NOINTERFACE;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP_(ULONG) CEnumSubscription::AddRef()
|
|
{
|
|
return ++m_cRef;
|
|
}
|
|
|
|
|
|
STDMETHODIMP_(ULONG) CEnumSubscription::Release()
|
|
{
|
|
if (--m_cRef == 0)
|
|
{
|
|
delete this;
|
|
return 0;
|
|
}
|
|
|
|
return m_cRef;
|
|
}
|
|
|
|
|
|
HRESULT CEnumSubscription::CopyRange(ULONG nStart, ULONG nCount, SUBSCRIPTIONCOOKIE *pCookies, ULONG *pnCopied)
|
|
{
|
|
ULONG nCopied = 0;
|
|
|
|
ASSERT((NULL != pCookies) && (NULL != pnCopied));
|
|
|
|
if (m_nCurrent < m_nCount)
|
|
{
|
|
ULONG nRemaining = m_nCount - m_nCurrent;
|
|
nCopied = min(nRemaining, nCount);
|
|
memcpy(pCookies, m_pCookies + m_nCurrent, nCopied * sizeof(SUBSCRIPTIONCOOKIE));
|
|
}
|
|
|
|
*pnCopied = nCopied;
|
|
|
|
return (nCopied == nCount) ? S_OK : S_FALSE;
|
|
}
|
|
|
|
|
|
// IEnumSubscription
|
|
STDMETHODIMP CEnumSubscription::Next(
|
|
/* [in] */ ULONG celt,
|
|
/* [length_is][size_is][out] */ SUBSCRIPTIONCOOKIE *rgelt,
|
|
/* [out] */ ULONG *pceltFetched)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if ((0 == celt) || ((celt > 1) && (NULL == pceltFetched)) || (NULL == rgelt))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
DWORD nFetched;
|
|
|
|
hr = CopyRange(m_nCurrent, celt, rgelt, &nFetched);
|
|
|
|
m_nCurrent += nFetched;
|
|
|
|
if (pceltFetched)
|
|
{
|
|
*pceltFetched = nFetched;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CEnumSubscription::Skip(
|
|
/* [in] */ ULONG celt)
|
|
{
|
|
HRESULT hr;
|
|
|
|
m_nCurrent += celt;
|
|
|
|
if (m_nCurrent > (m_nCount - 1))
|
|
{
|
|
m_nCurrent = m_nCount; // Passed the last one
|
|
hr = S_FALSE;
|
|
}
|
|
else
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CEnumSubscription::Reset()
|
|
{
|
|
m_nCurrent = 0;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CEnumSubscription::Clone(
|
|
/* [out] */ IEnumSubscription **ppenum)
|
|
{
|
|
HRESULT hr = E_OUTOFMEMORY;
|
|
|
|
*ppenum = NULL;
|
|
|
|
CEnumSubscription *pes = new CEnumSubscription;
|
|
|
|
if (NULL != pes)
|
|
{
|
|
pes->m_pCookies = new SUBSCRIPTIONCOOKIE[m_nCount];
|
|
if (NULL != pes->m_pCookies)
|
|
{
|
|
ULONG nFetched;
|
|
|
|
hr = E_FAIL;
|
|
|
|
pes->m_nCount = m_nCount;
|
|
CopyRange(0, m_nCount, pes->m_pCookies, &nFetched);
|
|
|
|
ASSERT(m_nCount == nFetched);
|
|
|
|
if (m_nCount == nFetched)
|
|
{
|
|
hr = pes->QueryInterface(IID_IEnumSubscription, (void **)ppenum);
|
|
}
|
|
}
|
|
pes->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CEnumSubscription::GetCount(
|
|
/* [out] */ ULONG *pnCount)
|
|
{
|
|
if (NULL == pnCount)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*pnCount = m_nCount;
|
|
|
|
return S_OK;
|
|
} |