//---------------------------------------------------------------------------- // // Microsoft Windows // Copyright (C) Microsoft Corporation, 1997 // // File: admin.cpp // // Contents: // // Classes: // // Functions: // // History: // //---------------------------------------------------------------------------- #include "private.h" #include "shguidp.h" #include "chanmgr.h" #include "chanmgrp.h" #include "winineti.h" #include // Infodelivery Policies registry locations #define INFODELIVERY_POLICIES TEXT("Software\\Policies\\Microsoft\\Internet Explorer\\Infodelivery") // const TCHAR c_szRegKeyRestrictions[] = INFODELIVERY_POLICIES TEXT("\\Restrictions"); const TCHAR c_szRegKeyModifications[] = INFODELIVERY_POLICIES TEXT("\\Modifications"); const TCHAR c_szRegKeyCompletedMods[] = INFODELIVERY_POLICIES TEXT("\\CompletedModifications"); const TCHAR c_szRegKeyIESetup[] = TEXT("Software\\Microsoft\\IE4\\Setup"); // Wininet cache preload directory const TCHAR c_szRegKeyCachePreload[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Cache\\Preload"); // Registry key names of supported Modifications const TCHAR c_szAddChannels[] = TEXT("AddChannels"); const TCHAR c_szRemoveChannels[] = TEXT("RemoveChannels"); const TCHAR c_szRemoveAllChannels[] = TEXT("RemoveAllChannels"); const TCHAR c_szAddSubscriptions[] = TEXT("AddSubscriptions"); const TCHAR c_szRemoveSubscriptions[] = TEXT("RemoveSubscriptions"); const TCHAR c_szAddScheduleGroups[] = TEXT("AddScheduleGroups"); const TCHAR c_szRemoveScheduleGroups[] = TEXT("RemoveScheduleGroups"); const TCHAR c_szAddDesktopComponents[] = TEXT("AddDesktopComponents"); const TCHAR c_szRemoveDesktopComponents[] = TEXT("RemoveDesktopComponents"); // Registry value names of supported Modifications const TCHAR c_szURL[] = TEXT("URL"); const TCHAR c_szTitle[] = TEXT("Title"); const TCHAR c_szLogo[] = TEXT("Logo"); const TCHAR c_szWideLogo[] = TEXT("WideLogo"); const TCHAR c_szIcon[] = TEXT("Icon"); const TCHAR c_szCategory[] = TEXT("Category"); const TCHAR c_szChannelGuide[] = TEXT("ChannelGuide"); // DO NOTE CHANGE THIS STRING WITHOUT UPDATING CDFVIEW!!! const TCHAR c_szPreloadURL[] = TEXT("PreloadURL"); const TCHAR c_szLCID[] = TEXT("LangId"); // This must be an LCID despite its name const TCHAR c_szSoftware[] = TEXT("Software"); const TCHAR c_szSubscriptionType[] = TEXT("SubscriptionType"); const TCHAR c_szScheduleGroup[] = TEXT("ScheduleGroup"); const TCHAR c_szEarliestTime[] = TEXT("EarliestTime"); const TCHAR c_szIntervalTime[] = TEXT("IntervalTime"); const TCHAR c_szLatestTime[] = TEXT("LatestTime"); const TCHAR c_szComponentType[] = TEXT("DesktopComponentType"); const TCHAR c_szUsername[] = TEXT("Username"); const TCHAR c_szPassword[] = TEXT("Password"); const TCHAR c_szOldIEVersion[] = TEXT("OldIEVersion"); const TCHAR c_szNonActive[] = TEXT("NonActive"); const TCHAR c_szOffline[] = TEXT("Offline"); const TCHAR c_szSynchronize[] = TEXT("Synchronize"); // Names of reserved schedule groups that we support even in localized version const WCHAR c_szScheduleAuto[] = L"Auto"; const WCHAR c_szScheduleDaily[] = L"Daily"; const WCHAR c_szScheduleWeekly[] = L"Weekly"; const WCHAR c_szScheduleManual[] = L"Manual"; // Function prototypes for Modification handlers HRESULT ProcessAddChannels(HKEY hkey); HRESULT ProcessRemoveChannels(HKEY hkey); HRESULT ProcessRemoveAllChannels(HKEY hkey); HRESULT ProcessAddSubscriptions(HKEY hkey); HRESULT ProcessRemoveSubscriptions(HKEY hkey); HRESULT ProcessRemoveDesktopComponents(HKEY hkey); HRESULT Channel_GetBasePath(LPTSTR pszPath, int cch); // Helper functions void ShowChannelDirectories(BOOL fShow); // Table of supported Actions and corresponding functions // NOTE: The table must be ordered appropriately (RemoveAll must come before Add) typedef HRESULT (*PFNACTION)(HKEY); typedef struct { LPCTSTR szAction; PFNACTION pfnAction; } ACTIONTABLE; ACTIONTABLE rgActionTable[] = { { c_szRemoveAllChannels, &ProcessRemoveAllChannels }, { c_szRemoveSubscriptions, &ProcessRemoveSubscriptions }, { c_szRemoveChannels, &ProcessRemoveChannels }, { c_szRemoveDesktopComponents, &ProcessRemoveDesktopComponents }, { c_szAddChannels, &ProcessAddChannels }, { c_szAddSubscriptions, &ProcessAddSubscriptions } }; #define ACTIONTABLECOUNT (sizeof(rgActionTable) / sizeof(ACTIONTABLE)) #define ACTIONTABLE_ADDCHANNELS 5 // Helper class to manipulate registry keys class CRegKey { HKEY m_hkey; DWORD dwIndex; public: CRegKey(void) { m_hkey = NULL; dwIndex = 0; } ~CRegKey(void) { if (m_hkey) { LONG lRet = RegCloseKey(m_hkey); ASSERT(ERROR_SUCCESS == lRet); m_hkey = NULL; } } void SetKey(HKEY hkey) { m_hkey = hkey; } HKEY GetKey(void) { return m_hkey; } HRESULT OpenForRead(HKEY hkey, LPCTSTR szSubKey) { ASSERT(NULL == m_hkey); LONG lRet = RegOpenKeyEx(hkey, szSubKey, 0, KEY_QUERY_VALUE | KEY_ENUMERATE_SUB_KEYS, &m_hkey); ASSERT((ERROR_SUCCESS == lRet) || !m_hkey); return (ERROR_SUCCESS == lRet)?(S_OK):(E_FAIL); } HRESULT CreateForWrite(HKEY hkey, LPCTSTR szSubKey) { ASSERT(NULL == m_hkey); DWORD dwDisp; LONG lRet = RegCreateKeyEx(hkey, szSubKey, 0, TEXT(""), 0, KEY_QUERY_VALUE | KEY_SET_VALUE, NULL, &m_hkey, &dwDisp); ASSERT(ERROR_SUCCESS == lRet); return (ERROR_SUCCESS == lRet)?(S_OK):(E_FAIL); } HRESULT GetSubKeyCount(PDWORD pdwKeys) { ASSERT(NULL != m_hkey); LONG lRet = RegQueryInfoKey(m_hkey, NULL, NULL, NULL, pdwKeys, NULL, NULL, NULL, NULL, NULL, NULL, NULL); ASSERT(ERROR_SUCCESS == lRet); return (ERROR_SUCCESS == lRet)?(S_OK):(E_FAIL); } HRESULT Next(LPTSTR szSubKey) { ASSERT(NULL != m_hkey); DWORD dwLen = MAX_PATH; // Assumes size of incoming buffer. LONG lRet = RegEnumKeyEx(m_hkey, dwIndex, szSubKey, &dwLen, NULL, NULL, NULL, NULL); dwIndex++; if (ERROR_SUCCESS == lRet) return S_OK; else if (ERROR_NO_MORE_ITEMS == lRet) return S_FALSE; else { ASSERT(FALSE); return E_FAIL; } } HRESULT Reset(void) { dwIndex = 0; return S_OK; } HRESULT SetValue(LPCTSTR szValueName, DWORD dwValue) { ASSERT(m_hkey); LONG lRet = RegSetValueEx(m_hkey, szValueName, 0, REG_DWORD, (LPBYTE)&dwValue, sizeof(dwValue)); ASSERT(ERROR_SUCCESS == lRet); return (ERROR_SUCCESS == lRet)?(S_OK):(E_FAIL); } HRESULT GetValue(LPCTSTR szValueName, DWORD *pdwValue) { ASSERT(m_hkey); DWORD dwType = REG_DWORD; DWORD dwLen = sizeof(DWORD); LONG lRet = RegQueryValueEx(m_hkey, szValueName, 0, &dwType, (LPBYTE)pdwValue, &dwLen); return (ERROR_SUCCESS == lRet)?(S_OK):(E_FAIL); } HRESULT GetStringValue(LPCTSTR szValueName, LPTSTR szValue, DWORD cbValue) { ASSERT(m_hkey); return SHRegGetValue(m_hkey, NULL, szValueName, SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL, szValue, &cbValue) == ERROR_SUCCESS ? S_OK : E_FAIL; } HRESULT SetBSTRValue(LPCTSTR szValueName, BSTR bstr) { ASSERT(m_hkey); TCHAR szValue[INTERNET_MAX_URL_LENGTH]; MyOleStrToStrN(szValue, ARRAYSIZE(szValue), bstr); LONG lRet = RegSetValueEx(m_hkey, szValueName, 0, REG_SZ, (LPBYTE)szValue, lstrlen(szValue) + 1); ASSERT(ERROR_SUCCESS == lRet); return (ERROR_SUCCESS == lRet)?(S_OK):(E_FAIL); } HRESULT GetBSTRValue(LPCTSTR szValueName, BSTR *pbstr) { ASSERT(m_hkey); TCHAR szValue[INTERNET_MAX_URL_LENGTH]; DWORD cbValue = sizeof(szValue); HRESULT hr; *pbstr = NULL; LONG lRet = SHRegGetValue(m_hkey, NULL, szValueName, SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL, szValue, &cbValue); if (ERROR_SUCCESS == lRet) { DWORD cchValue = cbValue / sizeof(TCHAR); *pbstr = SysAllocStringLen(NULL, cchValue); // cchValue includes null terminator if (*pbstr) { MyStrToOleStrN(*pbstr, cchValue, szValue); hr = S_OK; } else { hr = E_OUTOFMEMORY; } } else { hr = HRESULT_FROM_WIN32(lRet); } return hr; } }; // Helper class to manage Dynamic Pointer Arrays of HKEYs. class CRegKeyDPA { HDPA m_hdpa; int m_count; public: CRegKeyDPA(void) { m_hdpa = NULL; m_count = 0; } ~CRegKeyDPA(void) { if (m_hdpa) { ASSERT(m_count); int i; for (i = 0; i < m_count; i++) RegCloseKey(GetKey(i)); DPA_Destroy(m_hdpa); } } int GetCount(void) { return m_count; } HKEY GetKey(int i) { ASSERT(i >= 0 && i < m_count); return (HKEY)DPA_GetPtr(m_hdpa, i); } HRESULT Add(HKEY hkey, LPCTSTR szSubKey) { if (!m_hdpa) { m_hdpa = DPA_CreateEx(5, NULL); // Choose arbitrary growth value if (!m_hdpa) return E_FAIL; } HKEY hkeyNew; LONG lRet = RegOpenKeyEx(hkey, szSubKey, 0, KEY_QUERY_VALUE | KEY_ENUMERATE_SUB_KEYS, &hkeyNew); if (ERROR_SUCCESS != lRet) return E_FAIL; if (-1 == DPA_InsertPtr(m_hdpa, DPA_APPEND, hkeyNew)) { RegCloseKey(hkeyNew); return E_FAIL; } m_count++; return S_OK; } }; // // 8/18/98 darrenmi // Copied (and butchered) from shdocvw\util.cpp so we don't have to load it at startup // DWORD WCRestricted2W(BROWSER_RESTRICTIONS rest, LPCWSTR pwzUrl, DWORD dwReserved) { DWORD dwType, dw = 0, dwSize = sizeof(DWORD); // we only handle NoChannelUI restriction if(rest != REST_NoChannelUI) { return 0; } // read registry setting SHGetValue(HKEY_CURRENT_USER, TEXT("Software\\Policies\\Microsoft\\Internet Explorer\\Infodelivery\\Restrictions"), TEXT("NoChannelUI"), &dwType, &dw, &dwSize); return dw; } // ProcessInfodeliveryPolicies // // This is the main Admin API for Infodelivery. It returns E_FAIL for errors, // S_FALSE for nothing to process, and S_OK for correctly processed items. // // Reg key organization [Modifications] - the key to process // [GUID1] - group of actions // [AddChannels] - sample action // [Channel1] - element of an action // HRESULT ProcessInfodeliveryPolicies(void) { HRESULT hr; CRegKey regModifications; TCHAR szGUID[MAX_PATH]; // Check if channels should be hidden. if (WCRestricted2W(REST_NoChannelUI, NULL, 0)) { ShowChannelDirectories(FALSE); } else { ShowChannelDirectories(TRUE); } // Bail out quickly if there are no Modifications to perform. (Return S_FALSE) hr = regModifications.OpenForRead(HKEY_CURRENT_USER, c_szRegKeyModifications); if (FAILED(hr)) return S_FALSE; // Prepare to use the CompletedModifications key. CRegKey regCompletedMods; hr = regCompletedMods.CreateForWrite(HKEY_CURRENT_USER, c_szRegKeyCompletedMods); if (FAILED(hr)) return hr; hr = CoInitialize(NULL); if (FAILED(hr)) return hr; // Prepare queues of registry keys to actions CRegKeyDPA rgKeyQueue[ACTIONTABLECOUNT]; // Enumerate the GUID keys, skipping the completed ones. // Enumerate the Actions beneath them and add them to queues. // ignoring errors here too. while (S_OK == regModifications.Next(szGUID)) { DWORD dwValue; if (FAILED(regCompletedMods.GetValue(szGUID, &dwValue))) { CRegKey regGUID; TCHAR szAction[MAX_PATH]; hr = regGUID.OpenForRead(regModifications.GetKey(), szGUID); while (S_OK == regGUID.Next(szAction)) { // Search the table to see if it's a key we understand. // If so, add it to the queue. int i; for (i = 0; i < ACTIONTABLECOUNT; i++) { if (!StrCmpI(rgActionTable[i].szAction, szAction)) { rgKeyQueue[i].Add(regGUID.GetKey(), szAction); break; } } } } } // Process all the keys we've accumulated. (Correct order is assumed.) int i; for (i = 0; i < ACTIONTABLECOUNT; i++) { if (rgKeyQueue[i].GetCount()) { int iKey; for (iKey = 0; iKey < rgKeyQueue[i].GetCount(); iKey++) { (rgActionTable[i].pfnAction)(rgKeyQueue[i].GetKey(iKey)); } } } // Walk the GUIDs we've processed and mark them completed with the time. // Updating ones we skipped as well will help with garbage collection. regModifications.Reset(); while (S_OK == regModifications.Next(szGUID)) { SYSTEMTIME st; FILETIME ft; GetSystemTime(&st); SystemTimeToFileTime(&st, &ft); regCompletedMods.SetValue(szGUID, ft.dwHighDateTime); } // Delete the Actions. NOTE: NT's RegDeleteKey() doesn't delete sub-keys. // This shlwapi API uses KEY_ALL_ACCESS. // We probably have to close all the keys here. SHDeleteKey(HKEY_CURRENT_USER, c_szRegKeyModifications); // If any channels were processed, tell the cache to reload. // We should only do this for default channels. if (rgKeyQueue[ACTIONTABLE_ADDCHANNELS].GetCount()) { ASSERT(!StrCmpI(rgActionTable[ACTIONTABLE_ADDCHANNELS].szAction, c_szAddChannels)); LoadUrlCacheContent(); } CoUninitialize(); return S_OK; } // // ProcessAddChannels_SortCallback - sort in reverse order // int ProcessAddChannels_SortCallback(PVOID p1, PVOID p2, LPARAM lparam) { return StrCmpI((LPTSTR)p2, (LPTSTR)p1); } // // ProcessAddChannels // HRESULT ProcessAddChannels(HKEY hkey) { // Enumerate the channels in the AddChannels key HRESULT hr; DWORD dwChannels; CRegKey regAdd; regAdd.SetKey(hkey); hr = regAdd.GetSubKeyCount(&dwChannels); if (SUCCEEDED(hr) && dwChannels) { // Check if the channels are the same code page as the system default. BOOL bCodePageMatch = TRUE; LCID lcidChannel = 0; if (SUCCEEDED(regAdd.GetValue(c_szLCID, &lcidChannel))) { TCHAR szCodePageSystem[8]; TCHAR szCodePageChannel[8]; szCodePageChannel[0] = 0; // Init in case there's no locale info GetLocaleInfo(lcidChannel, LOCALE_IDEFAULTANSICODEPAGE, szCodePageChannel, ARRAYSIZE(szCodePageChannel)); int iRet = GetLocaleInfo(GetSystemDefaultLCID(), LOCALE_IDEFAULTANSICODEPAGE, szCodePageSystem, ARRAYSIZE(szCodePageSystem)); ASSERT(iRet); if (StrCmpI(szCodePageSystem, szCodePageChannel)) bCodePageMatch = FALSE; } hr = E_FAIL; TCHAR *pch = (TCHAR *)MemAlloc(LMEM_FIXED, dwChannels * MAX_PATH * sizeof(TCHAR)); if (pch) { HDPA hdpa = DPA_Create(dwChannels); if (hdpa) { DWORD i; TCHAR *pchCur = pch; for (i = 0; i < dwChannels; i++) { if ((S_OK != regAdd.Next(pchCur)) || (-1 == DPA_InsertPtr(hdpa, DPA_APPEND, pchCur))) break; pchCur += MAX_PATH; } if (i >= dwChannels) { // Sort channels by registry key name, DPA_Sort(hdpa, ProcessAddChannels_SortCallback, 0); // Now create them. for (i = 0; i < dwChannels; i++) { BSTR bstrURL = NULL; BSTR bstrTitle = NULL; BSTR bstrLogo = NULL; BSTR bstrWideLogo = NULL; BSTR bstrIcon = NULL; BSTR bstrPreloadURL = NULL; DWORD dwCategory = 0; // default to channel DWORD dwChannelGuide = 0; // default to not a guide DWORD dwSoftware = 0; // default to non-software channel DWORD dwOffline = 0; DWORD dwSynchronize = 0; CRegKey regChannel; regChannel.OpenForRead(hkey, (LPCTSTR)DPA_GetPtr(hdpa, i)); hr = regChannel.GetBSTRValue(c_szURL, &bstrURL); hr = regChannel.GetBSTRValue(c_szTitle, &bstrTitle); hr = regChannel.GetBSTRValue(c_szLogo, &bstrLogo); hr = regChannel.GetBSTRValue(c_szWideLogo, &bstrWideLogo); hr = regChannel.GetBSTRValue(c_szIcon, &bstrIcon); hr = regChannel.GetBSTRValue(c_szPreloadURL, &bstrPreloadURL); hr = regChannel.GetValue(c_szCategory, &dwCategory); hr = regChannel.GetValue(c_szChannelGuide, &dwChannelGuide); hr = regChannel.GetValue(c_szSoftware, &dwSoftware); hr = regChannel.GetValue(c_szOffline, &dwOffline); hr = regChannel.GetValue(c_szSynchronize, &dwSynchronize); if (bstrTitle) { IChannelMgr *pChannelMgr = NULL; hr = CoCreateInstance(CLSID_ChannelMgr, NULL, CLSCTX_INPROC_SERVER, IID_IChannelMgr, (void**)&pChannelMgr); if (SUCCEEDED(hr)) { // See if channel already exists - do nothing if it does (62976) IEnumChannels *pEnumChannels = NULL; if (SUCCEEDED(pChannelMgr->EnumChannels(CHANENUM_ALLFOLDERS, bstrURL, &pEnumChannels))) { CHANNELENUMINFO Bogus={0}; ULONG cFetched=0; if ((S_OK == pEnumChannels->Next(1, &Bogus, &cFetched)) && cFetched) { // Oops. It exists. Skip all this goo. hr = E_FAIL; } } SAFERELEASE(pEnumChannels); } if (SUCCEEDED(hr)) { if (dwCategory && bCodePageMatch) { // create a category (useless if code page doesn't match) CHANNELCATEGORYINFO csi = {0}; csi.cbSize = sizeof(csi); csi.pszURL = bstrURL; csi.pszTitle = bstrTitle; csi.pszLogo = bstrLogo; csi.pszIcon = bstrIcon; csi.pszWideLogo = bstrWideLogo; hr = pChannelMgr->AddCategory(&csi); } else if (!dwCategory && bstrURL) { // update the registry if it's a channel guide if (dwChannelGuide) { CRegKey reg; hr = reg.CreateForWrite(HKEY_CURRENT_USER, c_szRegKey); if (SUCCEEDED(hr)) reg.SetBSTRValue(c_szChannelGuide, bstrTitle); } // tell wininet if there's preload content if (bstrPreloadURL) { CRegKey reg; hr = reg.CreateForWrite(HKEY_CURRENT_USER, c_szRegKeyCachePreload); if (SUCCEEDED(hr)) { TCHAR szURL[INTERNET_MAX_URL_LENGTH]; MyOleStrToStrN(szURL, ARRAYSIZE(szURL), bstrURL); reg.SetBSTRValue(szURL, bstrPreloadURL); } } // create a channel (use URL instead of Title if code page doesn't match) CHANNELSHORTCUTINFO csi = {0}; csi.cbSize = sizeof(csi); csi.pszURL = bstrURL; if (bCodePageMatch) csi.pszTitle = bstrTitle; else csi.pszTitle = bstrURL; csi.pszLogo = bstrLogo; csi.pszIcon = bstrIcon; csi.pszWideLogo = bstrWideLogo; if (dwSoftware) csi.bIsSoftware = TRUE; hr = pChannelMgr->AddChannelShortcut(&csi); } } SAFERELEASE(pChannelMgr); if (dwOffline) { ISubscriptionMgr2 *pSubMgr2 = NULL; hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr2, (void**)&pSubMgr2); if (SUCCEEDED(hr)) { hr = pSubMgr2->CreateSubscription(NULL, bstrURL, bstrTitle, CREATESUBS_NOUI, SUBSTYPE_CHANNEL, NULL); if (dwSynchronize) { BOOL bIsSubscribed; SUBSCRIPTIONCOOKIE cookie; if (SUCCEEDED(pSubMgr2->IsSubscribed(bstrURL, &bIsSubscribed)) && bIsSubscribed && SUCCEEDED(ReadCookieFromInetDB(bstrURL, &cookie))) { pSubMgr2->UpdateItems(SUBSMGRUPDATE_MINIMIZE, 1, &cookie); } } pSubMgr2->Release(); } } } SAFEFREEBSTR(bstrURL); SAFEFREEBSTR(bstrTitle); SAFEFREEBSTR(bstrLogo); SAFEFREEBSTR(bstrWideLogo); SAFEFREEBSTR(bstrIcon); SAFEFREEBSTR(bstrPreloadURL); } } DPA_Destroy(hdpa); } MemFree(pch); } } regAdd.SetKey(NULL); return S_OK; } // // ProcessRemoveChannels // HRESULT ProcessRemoveChannels(HKEY hkey) { // Enumerate the channel keys in the RemoveChannels key HRESULT hr; CRegKey reg; reg.SetKey(hkey); TCHAR szChannel[MAX_PATH]; while (S_OK == reg.Next(szChannel)) { CRegKey regChannel; DWORD dwNonActive = 0; // default to deleting Active & NonActive channels TCHAR szURL[INTERNET_MAX_URL_LENGTH]; regChannel.OpenForRead(hkey, szChannel); regChannel.GetValue(c_szNonActive, &dwNonActive); if (SUCCEEDED(regChannel.GetStringValue(c_szURL, szURL, sizeof(szURL)))) { // Check if the channel is Active to determine if we can delete it if (dwNonActive) { CRegKey regPreload; if (SUCCEEDED(regPreload.OpenForRead(HKEY_CURRENT_USER, c_szRegKeyCachePreload))) { if (SUCCEEDED(regPreload.GetStringValue(szURL, NULL, 0))) { dwNonActive = 0; } } } // Now delete the channel if appropriate if (!dwNonActive) { IChannelMgr *pChannelMgr = NULL; hr = CoCreateInstance(CLSID_ChannelMgr, NULL, CLSCTX_INPROC_SERVER, IID_IChannelMgr, (void**)&pChannelMgr); if (SUCCEEDED(hr)) { BSTR bstrURL; if (SUCCEEDED(regChannel.GetBSTRValue(c_szURL, &bstrURL))) { IEnumChannels *pEnum; hr = pChannelMgr->EnumChannels(CHANENUM_ALLFOLDERS | CHANENUM_PATH, bstrURL, &pEnum); if (SUCCEEDED(hr)) { CHANNELENUMINFO info; while (S_OK == pEnum->Next(1, &info, NULL)) { hr = pChannelMgr->DeleteChannelShortcut(info.pszPath); ASSERT(SUCCEEDED(hr)); CoTaskMemFree(info.pszPath); } pEnum->Release(); } SysFreeString(bstrURL); } pChannelMgr->Release(); } } } } reg.SetKey(NULL); return S_OK; } // // ProcessAddSubscriptions // HRESULT ProcessAddSubscriptions(HKEY hkey) { // Enumerate the subscription keys in the AddSubscriptions key HRESULT hr; CRegKey reg; reg.SetKey(hkey); TCHAR szSubscription[MAX_PATH]; while (S_OK == reg.Next(szSubscription)) { // Create the subscription // What if there is one already? CRegKey regSubscription; regSubscription.OpenForRead(hkey, szSubscription); BSTR bstrURL, bstrTitle, bstrGroup, bstrUsername, bstrPassword; DWORD dwSubType; DWORD dwSynchronize = 0; hr = regSubscription.GetBSTRValue(c_szURL, &bstrURL); hr = regSubscription.GetBSTRValue(c_szTitle, &bstrTitle); hr = regSubscription.GetBSTRValue(c_szScheduleGroup, &bstrGroup); hr = regSubscription.GetBSTRValue(c_szUsername, &bstrUsername); hr = regSubscription.GetBSTRValue(c_szPassword, &bstrPassword); hr = regSubscription.GetValue(c_szSynchronize, &dwSynchronize); if (bstrURL && bstrTitle && bstrGroup && SUCCEEDED(regSubscription.GetValue(c_szSubscriptionType, &dwSubType))) { SUBSCRIPTIONINFO si = {0}; si.cbSize = sizeof(SUBSCRIPTIONINFO); si.fUpdateFlags = SUBSINFO_SCHEDULE; if (bstrUsername && bstrPassword) { si.fUpdateFlags |= (SUBSINFO_USER | SUBSINFO_PASSWORD); si.bstrUserName = bstrUsername; si.bstrPassword = bstrPassword; } if (dwSubType == SUBSTYPE_CHANNEL || dwSubType == SUBSTYPE_DESKTOPCHANNEL) { si.fUpdateFlags |= SUBSINFO_CHANNELFLAGS; si.fChannelFlags = 0; // Notify only. } if (SUCCEEDED(hr)) { ISubscriptionMgr2 *pSubMgr2 = NULL; hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr2, (void**)&pSubMgr2); if (SUCCEEDED(hr)) { hr = pSubMgr2->CreateSubscription(NULL, bstrURL, bstrTitle, CREATESUBS_NOUI, (SUBSCRIPTIONTYPE)dwSubType, &si); if (dwSynchronize) { BOOL bIsSubscribed; SUBSCRIPTIONCOOKIE cookie; if (SUCCEEDED(pSubMgr2->IsSubscribed(bstrURL, &bIsSubscribed)) && bIsSubscribed && SUCCEEDED(ReadCookieFromInetDB(bstrURL, &cookie))) { pSubMgr2->UpdateItems(SUBSMGRUPDATE_MINIMIZE, 1, &cookie); } } pSubMgr2->Release(); } } } SAFEFREEBSTR(bstrURL); SAFEFREEBSTR(bstrTitle); SAFEFREEBSTR(bstrGroup); SAFEFREEBSTR(bstrUsername); SAFEFREEBSTR(bstrPassword); } reg.SetKey(NULL); return S_OK; } // // ProcessRemoveSubscriptions // HRESULT ProcessRemoveSubscriptions(HKEY hkey) { // Enumerate the subscription keys in the RemoveSubscriptions key HRESULT hr; CRegKey reg; reg.SetKey(hkey); TCHAR szSubscription[MAX_PATH]; while (S_OK == reg.Next(szSubscription)) { // Find the URL to delete CRegKey regSubscription; regSubscription.OpenForRead(hkey, szSubscription); BSTR bstrURL; if (SUCCEEDED(regSubscription.GetBSTRValue(c_szURL, &bstrURL))) { ISubscriptionMgr *pSubMgr = NULL; hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&pSubMgr); if (SUCCEEDED(hr)) { hr = pSubMgr->DeleteSubscription(bstrURL, NULL); pSubMgr->Release(); } SysFreeString(bstrURL); } } reg.SetKey(NULL); return S_OK; } // // PRIVATE VERSION HANDLING CODE - REVIEW THIS CODE SHOULD HAVE BEEN STOLEN // FROM SETUP // struct MYVERSION { DWORD dw1; // most sig version number DWORD dw2; DWORD dw3; DWORD dw4; // least sig version number }; int CompareDW(DWORD dw1, DWORD dw2) { if (dw1 > dw2) return 1; if (dw1 < dw2) return -1; return 0; } int CompareVersion(MYVERSION * pv1, MYVERSION * pv2) { int rv; rv = CompareDW(pv1->dw1, pv2->dw1); if (rv == 0) { rv = CompareDW(pv1->dw2, pv2->dw2); if (rv == 0) { rv = CompareDW(pv1->dw3, pv2->dw3); if (rv == 0) { rv = CompareDW(pv1->dw4, pv2->dw4); } } } return rv; } // // Returns TRUE if an INT was parsed and *pwsz is NOT NULL // if a . was found // BOOL GetDWORDFromStringAndAdvancePtr(DWORD *pdw, LPWSTR *pwsz) { if (!StrToIntExW(*pwsz, 0, (int *)pdw)) return FALSE; *pwsz = StrChrW(*pwsz, L'.'); if (*pwsz) *pwsz = *pwsz +1; return TRUE; } BOOL GetVersionFromString(MYVERSION *pver, LPWSTR pwsz) { BOOL rv; rv = GetDWORDFromStringAndAdvancePtr(&pver->dw1, &pwsz); if (!rv || pwsz == NULL) return FALSE; rv = GetDWORDFromStringAndAdvancePtr(&pver->dw2, &pwsz); if (!rv || pwsz == NULL) return FALSE; rv = GetDWORDFromStringAndAdvancePtr(&pver->dw3, &pwsz); if (!rv || pwsz == NULL) return FALSE; rv = GetDWORDFromStringAndAdvancePtr(&pver->dw4, &pwsz); if (!rv) return FALSE; return TRUE; } // // ProcessRemoveAllChannels // HRESULT ProcessRemoveAllChannels(HKEY hkey) { HRESULT hr; HINSTANCE hAdvPack = NULL; DELNODE pfDELNODE = NULL; IChannelMgrPriv *pChannelMgrPriv = NULL; CRegKey regAdd; regAdd.SetKey(hkey); TCHAR szChannelFolder[MAX_PATH]; hr = CoCreateInstance(CLSID_ChannelMgr, NULL, CLSCTX_INPROC_SERVER, IID_IChannelMgrPriv, (void**)&pChannelMgrPriv); if (FAILED(hr)) { goto Exit; } if ((hAdvPack = LoadLibrary(TEXT("advpack.dll"))) != NULL) { pfDELNODE = (DELNODE)GetProcAddress( hAdvPack, "DelNode"); if (!pfDELNODE) { hr = HRESULT_FROM_WIN32(GetLastError()); goto Exit; } } else { hr = HRESULT_FROM_WIN32(GetLastError()); goto Exit; } // Loop Through Channel Folders to delete while (S_OK == regAdd.Next(szChannelFolder)) { DWORD dwSoftware = 0, dwChannelGuide = 0; CRegKey regChannelFolder; CHAR szChannelPath[MAX_PATH]; TCHAR szChannelPathT[MAX_PATH]; TCHAR szFavsT[MAX_PATH]; //Retrieve Unicode data from registry BSTR bstrOldIEVersion = NULL; BOOL bVersion = TRUE; regChannelFolder.OpenForRead(hkey, szChannelFolder); // Check whether old IE version is correct. hr = regChannelFolder.GetBSTRValue(c_szOldIEVersion, &bstrOldIEVersion); if (SUCCEEDED(hr) && bstrOldIEVersion) { CRegKey regKeyIESetup; hr = regKeyIESetup.OpenForRead(HKEY_LOCAL_MACHINE, c_szRegKeyIESetup); if (SUCCEEDED(hr)) { BSTR bstrRealOldIEVersion = NULL; hr = regKeyIESetup.GetBSTRValue(c_szOldIEVersion, &bstrRealOldIEVersion); if (SUCCEEDED(hr) && bstrRealOldIEVersion) { MYVERSION verOldIEVersion, verRealOldIEVersion; if (GetVersionFromString(&verOldIEVersion, bstrOldIEVersion) && GetVersionFromString(&verRealOldIEVersion, bstrRealOldIEVersion)) { // // If the old version of IE that was on this machine (verRealOldIEVersion) // is infact NEWER than the old version number in the CABs that we want to // delete (verOldIEVersion) then dont blow away old channel folder. // Otherwise default to blow away channels. // if (CompareVersion(&verRealOldIEVersion, &verOldIEVersion) > 0) { bVersion = FALSE; } } SAFEFREEBSTR(bstrRealOldIEVersion); } } SAFEFREEBSTR(bstrOldIEVersion); } if (!bVersion) { continue; } hr = regChannelFolder.GetValue(c_szChannelGuide, &dwChannelGuide); if (FAILED(hr) || (SUCCEEDED(hr) && !dwChannelGuide)) { if (SUCCEEDED(pChannelMgrPriv->GetChannelFolderPath(szChannelPath, MAX_PATH, IChannelMgrPriv::CF_CHANNEL))) { // Retrieve Favorites Path from registry if (SUCCEEDED(Channel_GetBasePath((LPTSTR)szFavsT, ARRAYSIZE(szFavsT)))) { // Convert from ANSI SHAnsiToTChar(szChannelPath, szChannelPathT, ARRAYSIZE(szChannelPathT)); // If channel folder doesn't exist, then szChannelPath will contain the Favorites path. // Don't delete the entries. if (StrCmpI(szFavsT, szChannelPathT)) pfDELNODE(szChannelPath, ADN_DONT_DEL_DIR); } } } hr = regChannelFolder.GetValue(c_szSoftware, &dwSoftware); if (FAILED(hr) || (SUCCEEDED(hr) && !dwSoftware)) { if (SUCCEEDED(pChannelMgrPriv->GetChannelFolderPath(szChannelPath, MAX_PATH, IChannelMgrPriv::CF_SOFTWAREUPDATE))) { pfDELNODE(szChannelPath, ADN_DONT_DEL_DIR); } } hr = S_OK; } regAdd.SetKey(NULL); Exit: SAFERELEASE(pChannelMgrPriv); if (hAdvPack) { FreeLibrary(hAdvPack); } return hr; } // // ProcessRemoveDesktopComponents // HRESULT ProcessRemoveDesktopComponents(HKEY hkey) { // Enumerate the component keys in the ProcessRemoveDesktopComponents key // HRESULT hr; CRegKey reg; reg.SetKey(hkey); TCHAR szComponent[MAX_PATH]; while (S_OK == reg.Next(szComponent)) { // Find the URL to delete CRegKey regComponent; regComponent.OpenForRead(hkey, szComponent); BSTR bstrURL; if (SUCCEEDED(regComponent.GetBSTRValue(c_szURL, &bstrURL))) { SysFreeString(bstrURL); } } reg.SetKey(NULL); return S_OK; } // // NoChannelUI processing. // #define SHELLFOLDERS \ TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Shell Folders") typedef enum _tagXMLDOCTYPE { DOC_CHANNEL, DOC_SOFTWAREUPDATE } XMLDOCTYPE; // // Get the path to the favorites directory. // HRESULT Channel_GetBasePath(LPTSTR pszPath, int cch) { ASSERT(pszPath || 0 == cch); DWORD cbPath = cch * sizeof(TCHAR); return SHRegGetValue(HKEY_CURRENT_USER, SHELLFOLDERS, TEXT("Favorites"), SRRF_RT_REG_SZ | SRRF_RT_REG_EXPAND_SZ | SRRF_NOEXPAND, NULL, pszPath, &cbPath) == ERROR_SUCCESS ? S_OK : E_FAIL; } HRESULT Channel_GetFolder(LPTSTR pszPath, XMLDOCTYPE xdt ) { TCHAR szFavs[MAX_PATH]; TCHAR szChannel[MAX_PATH]; HRESULT hr = E_FAIL; if (SUCCEEDED(Channel_GetBasePath(szFavs, ARRAYSIZE(szFavs)))) { // // Get the potentially localized name of the Channel folder from // tack this on the Favorites path // MLLoadString( ((xdt == DOC_CHANNEL)? IDS_CHANNEL_FOLDER : IDS_SOFTWAREUPDATE_FOLDER), szChannel, MAX_PATH); PathCombine(pszPath, szFavs, szChannel); hr = S_OK; } return hr; } // // Set/Clear the "hidden" attribute of a channel directory. // void ShowChannelDirectory(BOOL fShow, XMLDOCTYPE xdt) { TCHAR szPath[MAX_PATH]; DWORD dwAttributes; if (SUCCEEDED(Channel_GetFolder(szPath, xdt))) { dwAttributes = GetFileAttributes(szPath); if (0xffffffff != dwAttributes) { if (fShow && (dwAttributes & FILE_ATTRIBUTE_HIDDEN)) { SetFileAttributes(szPath, dwAttributes & ~FILE_ATTRIBUTE_HIDDEN); } else if (!fShow && !(dwAttributes & FILE_ATTRIBUTE_HIDDEN)) { SetFileAttributes(szPath, dwAttributes | FILE_ATTRIBUTE_HIDDEN); } } } } // // Hide or show channel directories // void ShowChannelDirectories(BOOL fShow) { ShowChannelDirectory(fShow, DOC_CHANNEL); ShowChannelDirectory(fShow, DOC_SOFTWAREUPDATE); }