// API to install a channel by creating a system folder in the channel directory // // Julian Jiggins (julianj), 4th May, 1997 // #include "stdinc.h" #include "resource.h" #include "cdfidl.h" #include "xmlutil.h" #include "persist.h" #include "cdfview.h" #include "chanapi.h" #include "chanmgrp.h" #include "chanmgri.h" #include "chanenum.h" #include "dll.h" #include "shguidp.h" #include "winineti.h" #define _SHDOCVW_ #include #include #ifdef UNIX #undef EVAL #define EVAL(x) x STDAPI SHAddSubscribeFavorite(HWND hwnd, LPCWSTR pwszURL, LPCWSTR pwszName, DWORD dwFlags, SUBSCRIPTIONTYPE subsType, SUBSCRIPTIONINFO* pInfo); #endif /* UNIX */ #define SHELLFOLDERS \ TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Shell Folders") // Wininet cache preload registry key in HKCU const TCHAR c_szRegKeyCachePreload[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Cache\\Preload"); BOOL PathCombineCleanPath(LPTSTR pszCleanPath, LPCTSTR pszPath); #ifndef UNICODE int MyPathCleanupSpec(LPCTSTR pszDir, LPTSTR pszSpec); #endif void Channel_OrderItem(LPCTSTR szPath); // This was copied from shdocvw\favorite.cpp static const int CREATESUBS_ACTIVATE = 0x8000; //hidden flag meaning channel is already on system // // Debugging code // #if 0 void DumpOrderList(HDPA hdpa) { int i = 0; PORDERITEM poi = (PORDERITEM)DPA_GetPtr(hdpa, i); while (poi) { TCHAR szName[MAX_PATH]; wnsprintf(szName, ARRAYSIZE(szName), "nOrder=%d, lParam=%d, pidl=", poi->nOrder, poi->lParam); OutputDebugString(szName); ASSERT(SHGetPathFromIDListA(poi->pidl, szName)); OutputDebugString(szName); OutputDebugString("\n"); i++; poi = (PORDERITEM)DPA_GetPtr(hdpa, i); } } void DumpPidl(LPITEMIDLIST pidl) { TCHAR szName[MAX_PATH]; ASSERT(SHGetPathFromIDListA(pidl, szName)); OutputDebugString(szName); OutputDebugString("\n"); } #endif // // Constructor and destructor. // //////////////////////////////////////////////////////////////////////////////// // // *** CChannelMgr::CChannelMgr *** // // Constructor. // //////////////////////////////////////////////////////////////////////////////// CChannelMgr::CChannelMgr ( void ) : m_cRef(1) { TraceMsg(TF_OBJECTS, "+ IChannelMgr"); DllAddRef(); return; } //////////////////////////////////////////////////////////////////////////////// // // *** CChannelMgr::~CChannelMgr *** // // Destructor. // //////////////////////////////////////////////////////////////////////////////// CChannelMgr::~CChannelMgr ( void ) { ASSERT(0 == m_cRef); // // Matching Release for the constructor Addref. // TraceMsg(TF_OBJECTS, "- IChannelMgr"); DllRelease(); return; } // // IUnknown methods. // //////////////////////////////////////////////////////////////////////////////// // // *** CChannelMgr::QueryInterface *** // // CChannelMgr QI. // //////////////////////////////////////////////////////////////////////////////// STDMETHODIMP CChannelMgr::QueryInterface ( REFIID riid, void **ppv ) { ASSERT(ppv); HRESULT hr; *ppv = NULL; if (IID_IUnknown == riid || IID_IChannelMgr == riid) { *ppv = (IChannelMgr*)this; } else if ((IID_IChannelMgrPriv2 == riid) || (IID_IChannelMgrPriv == riid)) { *ppv = (IChannelMgrPriv2*)this; } else if (IID_IShellCopyHook == riid) { *ppv = (ICopyHook*)this; } #ifdef UNICODE else if (IID_IShellCopyHookA == riid) { *ppv = (ICopyHookA*)this; } #endif if (*ppv) { ((IUnknown*)*ppv)->AddRef(); hr = S_OK; } else { hr = E_NOINTERFACE; } ASSERT((SUCCEEDED(hr) && *ppv) || (FAILED(hr) && NULL == *ppv)); return hr; } //////////////////////////////////////////////////////////////////////////////// // // *** CChannelMgr::AddRef *** // //////////////////////////////////////////////////////////////////////////////// STDMETHODIMP_(ULONG) CChannelMgr::AddRef ( void ) { ASSERT(m_cRef != 0); ASSERT(m_cRef < (ULONG)-1); return ++m_cRef; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** CChannelMgr::Release *** // //////////////////////////////////////////////////////////////////////////////// STDMETHODIMP_(ULONG) CChannelMgr::Release ( void ) { ASSERT (m_cRef != 0); ULONG cRef = --m_cRef; if (0 == cRef) delete this; return cRef; } //////////////////////////////////////////////////////////////////////////////// // // IChannelMgr member(s) // //////////////////////////////////////////////////////////////////////////////// STDMETHODIMP CChannelMgr::AddCategory(CHANNELCATEGORYINFO *pCategoryInfo) { ASSERT(pCategoryInfo); if (!pCategoryInfo || pCategoryInfo->cbSize < sizeof(CHANNELCATEGORYINFO)) { return E_INVALIDARG; } // // Convert all the wide str params to tstrs // LPWSTR pwszURL = pCategoryInfo->pszURL; LPWSTR pwszTitle = pCategoryInfo->pszTitle; LPWSTR pwszLogo = pCategoryInfo->pszLogo; LPWSTR pwszIcon = pCategoryInfo->pszIcon; LPWSTR pwszWideLogo = pCategoryInfo->pszWideLogo; // // REVIEW:is this too much to alloc on the stack? // TCHAR szURL[INTERNET_MAX_URL_LENGTH]; TCHAR szTitle[MAX_PATH]; TCHAR szLogo[INTERNET_MAX_URL_LENGTH]; TCHAR szIcon[INTERNET_MAX_URL_LENGTH]; TCHAR szWideLogo[INTERNET_MAX_URL_LENGTH]; LPTSTR pszURL = NULL; LPTSTR pszTitle = NULL; LPTSTR pszLogo = NULL; LPTSTR pszIcon = NULL; LPTSTR pszWideLogo = NULL; if (pwszTitle) { SHUnicodeToTChar(pwszTitle, szTitle, MAX_PATH); pszTitle = szTitle; } else { return E_INVALIDARG; // required option } if (pwszURL) { SHUnicodeToTChar(pwszURL, szURL, INTERNET_MAX_URL_LENGTH); pszURL = szURL; } if (pwszLogo) { SHUnicodeToTChar(pwszLogo, szLogo,INTERNET_MAX_URL_LENGTH); pszLogo = szLogo; } if (pwszIcon) { SHUnicodeToTChar(pwszIcon, szIcon,INTERNET_MAX_URL_LENGTH); pszIcon = szIcon; } if (pwszWideLogo) { SHUnicodeToTChar(pwszWideLogo, szWideLogo,INTERNET_MAX_URL_LENGTH); pszWideLogo = szWideLogo; } // // Find the Channel directory // Attempt to create folder if one doesn't exist // TCHAR szPath[MAX_PATH]; if (FAILED(Channel_GetFolder(szPath, DOC_CHANNEL))) { return E_FAIL; // couldn't find Channel Folder or create empty one } // Convert the title into a path component TCHAR szFileTitle[MAX_PATH]; szFileTitle[0] = TEXT('\0'); PathCombineCleanPath(szFileTitle, pszTitle); TraceMsg(TF_GENERAL, "AddCategory(): pszTitle = %s, szFileTitle = %s", pszTitle, szFileTitle); // // add title to channel folder path // PathCombine(szPath, szPath, szFileTitle); // // Create the logoized, iconized, webviewed special folder // public ChanMgr api doesn't handle iconIndex yet. Should fix! // // // REARCHITECT: this is not clean or elegant // we only work if the incoming URL is infact a UNC // and we copy the file to the Category Folder // if (pszURL) { TCHAR szTargetPath[MAX_PATH]; LPTSTR pszFilename = PathFindFileName(pszURL); // // Create folder, webview htm is just filename no path. // Channel_CreateSpecialFolder(szPath, pszFilename, pszLogo, pszWideLogo, pszIcon, 0); // // Now build target fully qualified path to use to copy html file // PathCombine(szTargetPath, szPath, pszFilename); // // Copy html into category folder and mark it hidden if (!CopyFile(pszURL, szTargetPath, FALSE)) { // If the copy fails, try again after clearing the attributes. SetFileAttributes(szTargetPath, FILE_ATTRIBUTE_NORMAL); CopyFile(pszURL, szTargetPath, FALSE); } SetFileAttributes(szTargetPath, FILE_ATTRIBUTE_HIDDEN); } else Channel_CreateSpecialFolder(szPath, NULL, pszLogo, pszWideLogo, pszIcon, 0); // // Place the channel category in the appropriate "order". // Channel_OrderItem(szPath); // // Notify the system that a new item has been added. // SHChangeNotify(SHCNE_MKDIR, SHCNF_PATH, (void*)szPath, NULL); return S_OK; } // // DeleteCategory // STDMETHODIMP CChannelMgr::DeleteCategory(LPWSTR pwzTitle) { TCHAR szTitle[INTERNET_MAX_URL_LENGTH]; SHUnicodeToTChar(pwzTitle, szTitle, INTERNET_MAX_URL_LENGTH); // // REVIEW: deletegate to just deleting the channel ok??? // return ::DeleteChannel(szTitle); } STDMETHODIMP CChannelMgr::AddChannelShortcut(CHANNELSHORTCUTINFO *pChannelInfo) { if (!pChannelInfo || pChannelInfo->cbSize < sizeof(CHANNELSHORTCUTINFO) || pChannelInfo->pszURL == NULL || pChannelInfo->pszTitle == NULL) { ASSERT(FALSE); return E_INVALIDARG; } TCHAR szURL[INTERNET_MAX_URL_LENGTH]; TCHAR szTitle[INTERNET_MAX_URL_LENGTH]; TCHAR szLogo[INTERNET_MAX_URL_LENGTH]; TCHAR szIcon[INTERNET_MAX_URL_LENGTH]; TCHAR szWideLogo[INTERNET_MAX_URL_LENGTH]; LPTSTR pszLogo = NULL; LPTSTR pszIcon = NULL; LPTSTR pszWideLogo = NULL; // // Convert BSTRs to TSTRs // SHUnicodeToTChar(pChannelInfo->pszURL, szURL, INTERNET_MAX_URL_LENGTH); SHUnicodeToTChar(pChannelInfo->pszTitle, szTitle, INTERNET_MAX_URL_LENGTH); // // Now handle optional arguments // if (pChannelInfo->pszLogo != NULL) { SHUnicodeToTChar(pChannelInfo->pszLogo, szLogo, INTERNET_MAX_URL_LENGTH); pszLogo = szLogo; } if (pChannelInfo->pszWideLogo != NULL) { SHUnicodeToTChar(pChannelInfo->pszWideLogo, szWideLogo, INTERNET_MAX_URL_LENGTH); pszWideLogo = szWideLogo; } if (pChannelInfo->pszIcon != NULL) { SHUnicodeToTChar(pChannelInfo->pszIcon, szIcon, INTERNET_MAX_URL_LENGTH); pszIcon = szIcon; } return ::AddChannel(szTitle, szURL, pszLogo, pszWideLogo, pszIcon, pChannelInfo->bIsSoftware ? DOC_SOFTWAREUPDATE : DOC_CHANNEL); } STDMETHODIMP CChannelMgr::DeleteChannelShortcut(LPWSTR pwzTitle) { TCHAR szTitle[INTERNET_MAX_URL_LENGTH]; SHUnicodeToTChar(pwzTitle, szTitle, INTERNET_MAX_URL_LENGTH); return ::DeleteChannel(szTitle); } STDMETHODIMP CChannelMgr::EnumChannels(DWORD dwEnumFlags, LPCWSTR pszURL, IEnumChannels** ppIEnumChannels) { *ppIEnumChannels = (IEnumChannels*) new CChannelEnum(dwEnumFlags, pszURL); return *ppIEnumChannels ? S_OK : E_OUTOFMEMORY; } // // IChannelMgrPriv // STDMETHODIMP CChannelMgr::GetBaseChannelPath(LPSTR pszPath, int cch) { ASSERT(pszPath || 0 == cch); HRESULT hr; #ifdef UNICODE TCHAR tszPath[MAX_PATH]; hr = Channel_GetBasePath(tszPath, MAX_PATH); SHTCharToAnsi(tszPath, pszPath, ARRAYSIZE(tszPath)); #else hr = Channel_GetBasePath(pszPath, cch); #endif return hr; } STDMETHODIMP CChannelMgr::GetChannelFolderPath (LPSTR pszPath, int cch, CHANNELFOLDERLOCATION cflChannel) { ASSERT (pszPath || 0 == cch); XMLDOCTYPE xdt; switch (cflChannel) { case CF_CHANNEL: xdt = DOC_CHANNEL; break; case CF_SOFTWAREUPDATE: xdt = DOC_SOFTWAREUPDATE; break; default: return E_INVALIDARG; } TCHAR tszPath[MAX_PATH]; HRESULT hr = Channel_GetFolder(tszPath, xdt); if (FAILED(hr) || ( SUCCEEDED(hr) && cch <= StrLen(tszPath)) ) return E_FAIL; #ifdef UNICODE SHTCharToAnsi(tszPath, pszPath, ARRAYSIZE(tszPath)); #else StrCpy(pszPath, tszPath); #endif return S_OK; } STDMETHODIMP CChannelMgr::GetChannelFolder (LPITEMIDLIST* ppidl, CHANNELFOLDERLOCATION cflChannel) { if (ppidl == NULL) return E_FAIL; char szPath[MAX_PATH]; HRESULT hr = GetChannelFolderPath (szPath, ARRAYSIZE(szPath), cflChannel); if (FAILED (hr)) return hr; #ifdef UNICODE TCHAR tszPath[MAX_PATH]; SHAnsiToTChar(szPath, tszPath, ARRAYSIZE(tszPath)); return Channel_CreateILFromPath (tszPath, ppidl); #else return Channel_CreateILFromPath (szPath, ppidl); #endif } STDMETHODIMP CChannelMgr::InvalidateCdfCache(void) { InterlockedIncrement((LONG*)&g_dwCacheCount); return S_OK; } STDMETHODIMP CChannelMgr::PreUpdateChannelImage( LPCSTR pszPath, LPSTR pszHashItem, int* piIndex, UINT* puFlags, int* piImageIndex ) { #ifdef UNICODE TCHAR tszPath[MAX_PATH]; TCHAR tszHashItem[MAX_PATH]; SHAnsiToTChar(pszPath, tszPath, ARRAYSIZE(tszPath)); SHAnsiToTChar(pszHashItem, tszHashItem, ARRAYSIZE(tszHashItem)); return ::PreUpdateChannelImage(tszPath, tszHashItem, piIndex, puFlags, piImageIndex); #else return ::PreUpdateChannelImage(pszPath, pszHashItem, piIndex, puFlags, piImageIndex); #endif } STDMETHODIMP CChannelMgr::UpdateChannelImage( LPCWSTR pszHashItem, int iIndex, UINT uFlags, int iImageIndex ) { ::UpdateChannelImage(pszHashItem, iIndex, uFlags, iImageIndex); return S_OK; } STDMETHODIMP CChannelMgr::ShowChannel( IWebBrowser2 *pIWebBrowser2, LPWSTR pwszURL, HWND hwnd ) { HRESULT hr; if (!pwszURL || !pIWebBrowser2) { hr = E_INVALIDARG; } else { hr = ::NavigateBrowser(pIWebBrowser2, pwszURL, hwnd); } return hr; } STDMETHODIMP CChannelMgr::IsChannelInstalled(LPCWSTR pwszURL) { return ::Channel_IsInstalled(pwszURL) ? S_OK : S_FALSE; } STDMETHODIMP CChannelMgr::AddAndSubscribe(HWND hwnd, LPCWSTR pwszURL, ISubscriptionMgr *pSubscriptionMgr) { return AddAndSubscribeEx2(hwnd, pwszURL, pSubscriptionMgr, FALSE); } STDMETHODIMP CChannelMgr::AddAndSubscribeEx2(HWND hwnd, LPCWSTR pwszURL, ISubscriptionMgr *pSubscriptionMgr, BOOL bAlwaysSubscribe) { HRESULT hr; BSTR bstrPreinstalled = NULL; HRESULT hrPreinstalled = IsChannelPreinstalled(pwszURL, &bstrPreinstalled); if (hrPreinstalled == S_OK) { RemovePreinstalledMapping(pwszURL); } WCHAR wszTitle[MAX_PATH]; TASK_TRIGGER tt = {0}; SUBSCRIPTIONINFO si = {0}; BOOL fIsSoftware = FALSE; si.cbSize = sizeof(SUBSCRIPTIONINFO); si.fUpdateFlags |= SUBSINFO_SCHEDULE; si.schedule = SUBSSCHED_AUTO; si.pTrigger = (LPVOID)&tt; hr = DownloadMinCDF(hwnd, pwszURL, wszTitle, ARRAYSIZE(wszTitle), &si, &fIsSoftware); if (hr == S_OK) { DWORD dwFlags = 0; BOOL bInstalled = Channel_IsInstalled(pwszURL); if (bInstalled) { dwFlags |= CREATESUBS_ACTIVATE | CREATESUBS_FROMFAVORITES; } else { dwFlags |= CREATESUBS_ADDTOFAVORITES; } if (bAlwaysSubscribe || !bInstalled || (hrPreinstalled == S_OK)) { if (!pSubscriptionMgr) { hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&pSubscriptionMgr); } else { pSubscriptionMgr->AddRef(); } if (SUCCEEDED(hr)) { hr = pSubscriptionMgr->CreateSubscription(hwnd, pwszURL, wszTitle, dwFlags, SUBSTYPE_CHANNEL, &si); // This will kill the one we may have CoCreated pSubscriptionMgr->Release(); } } } if (hr != S_OK && hrPreinstalled == S_OK && bstrPreinstalled != NULL) { SetupPreinstalledMapping(pwszURL, bstrPreinstalled); } SysFreeString(bstrPreinstalled); return hr; } STDMETHODIMP CChannelMgr::WriteScreenSaverURL(LPCWSTR pwszURL, LPCWSTR pwszScreenSaverURL) { return Channel_WriteScreenSaverURL(pwszURL, pwszScreenSaverURL); } STDMETHODIMP CChannelMgr::RefreshScreenSaverURLs() { return Channel_RefreshScreenSaverURLs(); } STDMETHODIMP CChannelMgr::DownloadMinCDF(HWND hwnd, LPCWSTR pwszURL, LPWSTR pwszTitle, DWORD cchTitle, SUBSCRIPTIONINFO *pSubInfo, BOOL *pfIsSoftware) { HRESULT hr; IXMLDocument* pIXMLDocument; IXMLElement* pIXMLElement; ASSERT(pSubInfo); ASSERT(pfIsSoftware); *pwszTitle = NULL; DLL_ForcePreloadDlls(PRELOAD_MSXML); hr = CoCreateInstance(CLSID_XMLDocument, NULL, CLSCTX_INPROC_SERVER, IID_IXMLDocument, (void**)&pIXMLDocument); if (SUCCEEDED(hr)) { BOOL fStartedOffLine = IsGlobalOffline(); BOOL fInformUserOfDownloadProblem = FALSE; SetGlobalOffline(FALSE); if (InternetAutodial(INTERNET_AUTODIAL_FORCE_ONLINE, 0)) { if (!DownloadCdfUI(hwnd, pwszURL, pIXMLDocument)) { hr = E_FAIL; fInformUserOfDownloadProblem = TRUE; } else { Channel_SendUpdateNotifications(pwszURL); LONG lDontCare; hr = XML_GetFirstChannelElement(pIXMLDocument, &pIXMLElement, &lDontCare); if (SUCCEEDED(hr)) { *pfIsSoftware = XML_GetDocType(pIXMLDocument) == DOC_SOFTWAREUPDATE; XML_GetSubscriptionInfo(pIXMLElement, pSubInfo); BSTR bstrTitle = XML_GetAttribute(pIXMLElement, XML_TITLE); if (bstrTitle) { if (bstrTitle[0]) { StrCpyNW(pwszTitle, bstrTitle, cchTitle); } else { if (StrCpyNW(pwszTitle, PathFindFileNameW(pwszURL), cchTitle)) { PathRemoveExtensionW(pwszTitle); } else { hr = S_FALSE; } } SysFreeString(bstrTitle); } else { hr = E_OUTOFMEMORY; } pIXMLElement->Release(); } } } else { hr = HRESULT_FROM_WIN32(GetLastError()); fInformUserOfDownloadProblem = TRUE; } pIXMLDocument->Release(); if (fStartedOffLine) { SetGlobalOffline(TRUE); } if (fInformUserOfDownloadProblem) { ASSERT(FAILED(hr)); CDFMessageBox(hwnd, IDS_INFO_MUST_CONNECT, IDS_INFO_DLG_TITLE, MB_OK | MB_ICONINFORMATION); // Set return val to S_FALSE so the caller can distinguish between // errors which we've informed the user of and hard failures // such as out of memory, etc. hr = S_FALSE; } } return hr; } //////////////////////////////////////////////////////////////////////////////// // // *** Channel_RemoveURLMapping *** // // Description: // Removes the cache and registry settings that wininet uses to map the // given url to a local file. // //////////////////////////////////////////////////////////////////////////////// #define PRELOAD_REG_KEY \ TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Cache\\Preload") void Channel_RemoveURLMapping(LPCTSTR pszURL) { DWORD cbcei = MAX_CACHE_ENTRY_INFO_SIZE; BYTE cei[MAX_CACHE_ENTRY_INFO_SIZE]; LPINTERNET_CACHE_ENTRY_INFO pcei = (LPINTERNET_CACHE_ENTRY_INFO)cei; // // Look up the url in the cache // if (GetUrlCacheEntryInfoEx(pszURL, pcei, &cbcei, NULL, 0, NULL, 0)) { // // see if it has a mapping because it is a preinstalled cache entry // if (pcei->CacheEntryType & INSTALLED_CACHE_ENTRY) { // // Clear the flag // pcei->CacheEntryType &= ~INSTALLED_CACHE_ENTRY; SetUrlCacheEntryInfo(pszURL, pcei, CACHE_ENTRY_ATTRIBUTE_FC); // // Now remove the mapping from the registry // HKEY hk; if (RegOpenKeyEx(HKEY_CURRENT_USER, PRELOAD_REG_KEY, 0, KEY_WRITE, &hk) == ERROR_SUCCESS) { RegDeleteValue(hk, pszURL); RegCloseKey(hk); } } } } #ifndef UNICODE // // widechar version of above routine // void Channel_RemoveURLMapping(LPCWSTR wszURL) { CHAR szURL[INTERNET_MAX_URL_LENGTH]; if (SHUnicodeToTChar(wszURL, szURL, ARRAYSIZE(szURL))) { Channel_RemoveURLMapping(szURL); } } #endif // cheap lookup function to see if we are dealing with a preinstalled URL BOOL Channel_CheckURLMapping( LPCWSTR wszURL ) { TCHAR szURL[INTERNET_MAX_URL_LENGTH]; if (SHUnicodeToTChar(wszURL, szURL, ARRAYSIZE(szURL))) { HKEY hkey; if ( RegOpenKeyEx(HKEY_CURRENT_USER, PRELOAD_REG_KEY, 0, KEY_READ, &hkey) == ERROR_SUCCESS ) { // check to see if the value exists for the URL.... TCHAR szPath[MAX_PATH]; DWORD cbSize = sizeof(szPath); LONG lRes = RegQueryValueEx( hkey, szURL, NULL, NULL, (LPBYTE) szPath, &cbSize ); RegCloseKey( hkey ); return (lRes == ERROR_SUCCESS ); } } return FALSE; } //////////////////////////////////////////////////////////////////////////////// // // *** CChannelMgr::IsChannelPreinstalled *** // // Description: // Returns S_OK if the channel url has a preinstalled cache entry // //////////////////////////////////////////////////////////////////////////////// STDMETHODIMP CChannelMgr::IsChannelPreinstalled(LPCWSTR pwszURL, BSTR * bstrFile) { HRESULT hr = S_FALSE; TCHAR szURL[INTERNET_MAX_URL_LENGTH]; if (SHUnicodeToTChar(pwszURL, szURL, ARRAYSIZE(szURL))) { DWORD cbcei = MAX_CACHE_ENTRY_INFO_SIZE; BYTE cei[MAX_CACHE_ENTRY_INFO_SIZE]; LPINTERNET_CACHE_ENTRY_INFO pcei = (LPINTERNET_CACHE_ENTRY_INFO)cei; // // Look up the url in the cache // if (GetUrlCacheEntryInfoEx(szURL, pcei, &cbcei, NULL, 0, NULL, 0)) { // // see if it has a mapping because it is a preinstalled cache entry // if (pcei->CacheEntryType & INSTALLED_CACHE_ENTRY) { // // Get a BSTR from the internet cache entry local file name // if (bstrFile) { WCHAR wszFile[MAX_PATH]; SHTCharToUnicode(pcei->lpszLocalFileName, wszFile, ARRAYSIZE(wszFile)); *bstrFile = SysAllocString(wszFile); } hr = S_OK; } } } return hr; } STDMETHODIMP CChannelMgr::RemovePreinstalledMapping(LPCWSTR pwszURL) { Channel_RemoveURLMapping(pwszURL); return S_OK; } STDMETHODIMP CChannelMgr::SetupPreinstalledMapping(LPCWSTR pwszURL, LPCWSTR pwszFile) { FILETIME ftZero = {0}; CommitUrlCacheEntryW(pwszURL, pwszFile, ftZero, ftZero, INSTALLED_CACHE_ENTRY, NULL, 0, NULL, 0); // // Make sure that there isn't an in memory cached old version of this URL // #ifdef UNICODE Cache_RemoveItem(pwszURL); #else char szURL[INTERNET_MAX_URL_LENGTH]; SHUnicodeToAnsi(pwszURL, szURL, ARRAYSIZE(szURL)); Cache_RemoveItem(szURL); #endif return S_OK; } // // Create the special folder that can have a webview associated with it, and an // icon and a logo view. // HRESULT Channel_CreateSpecialFolder( LPCTSTR pszPath, // path to folder to create LPCTSTR pszURL, // url for webview LPCTSTR pszLogo, // [optional] path to logo LPCTSTR pszWideLogo, // [optional] path to wide logo LPCTSTR pszIcon, // [optional] path to icon file int nIconIndex // index to icon in above file ) { // // First create the directory if it doesn't exist // if (!PathFileExists(pszPath)) { if (Channel_CreateDirectory(pszPath) != 0) { return E_FAIL; } } // // Mark it as a SYSTEM folder // if (!SetFileAttributes(pszPath, FILE_ATTRIBUTE_SYSTEM)) return E_FAIL; // // Make desktop.ini // TCHAR szDesktopIni[MAX_PATH]; PathCombine(szDesktopIni, pszPath, TEXT("desktop.ini")); WritePrivateProfileString(NULL, NULL, NULL, szDesktopIni); // // Write ConfirmFileOp=0 to turn off shell warnings during file operations // EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("ConfirmFileOp"), TEXT("0"), szDesktopIni)); // // Write the URL for this category folders webview // if (pszURL) { EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("URL"), pszURL, szDesktopIni)); } // // Write the Logo for this channel if present // if (pszLogo) { EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("Logo"), pszLogo, szDesktopIni)); } // // Write the WideLogo for this channel if present // if (pszWideLogo) { EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("WideLogo"), pszWideLogo, szDesktopIni)); } // // Write the Icon URL for this category folder if present // if (pszIcon) { TCHAR szIconIndex[8]; // can handle 999999 ASSERT(nIconIndex >= 0 && nIconIndex <= 999999); // sanity check wnsprintf(szIconIndex, ARRAYSIZE(szIconIndex), TEXT("%d"), nIconIndex); EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("IconIndex"), szIconIndex, szDesktopIni)); EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("IconFile"), pszIcon, szDesktopIni)); } // // Flush the buffers // WritePrivateProfileString(NULL, NULL, NULL, szDesktopIni); SetFileAttributes(szDesktopIni, FILE_ATTRIBUTE_HIDDEN); return S_OK; } // // Create the special channels folder // // // Channel folders are no longer created in IE5+ // /* HRESULT Channel_CreateChannelFolder( XMLDOCTYPE xdt ) { TCHAR szPath[MAX_PATH]; if (SUCCEEDED(Channel_GetFolder(szPath, xdt)) && szPath[0] != 0) { // // Create a special folder with an icon that lives in the cdfview.dll // return Channel_CreateSpecialFolder( szPath, NULL, NULL, NULL, g_szModuleName, -IDI_CHANNELFOLDER); } else return E_FAIL; } */ HRESULT Channel_GetBasePath(LPTSTR pszPath, int cch) { ASSERT(pszPath || 0 == cch); HRESULT hr = E_FAIL; HKEY hKey; DWORD dwLen = cch * sizeof(TCHAR); if (RegOpenKey(HKEY_CURRENT_USER, SHELLFOLDERS, &hKey) == ERROR_SUCCESS) { #ifndef UNIX if (RegQueryValueEx(hKey, TEXT("Favorites"), NULL, NULL, (LPBYTE)pszPath, &dwLen) == ERROR_SUCCESS) { hr = S_OK; } #else TCHAR szUnexpandPath[MAX_PATH+1]; dwLen = MAX_PATH; if (RegQueryValueEx(hKey, TEXT("Favorites"), NULL, NULL, (LPBYTE)szUnexpandPath, &dwLen) == ERROR_SUCCESS) { hr = S_OK; } DWORD length = ExpandEnvironmentStrings((LPTSTR)szUnexpandPath, (LPTSTR)pszPath, cch); if (length == 0 || length > cch) hr = E_FAIL; #endif /* UNIX */ RegCloseKey(hKey); } return hr; } BSTR Channel_GetFullPath(LPCWSTR pwszName) { ASSERT(pwszName); BSTR bstrRet = NULL; TCHAR szName[MAX_PATH]; if (SHUnicodeToTChar(pwszName, szName, ARRAYSIZE(szName))) { TCHAR szPath[MAX_PATH]; if (SUCCEEDED(Channel_GetFolder(szPath, DOC_CHANNEL))) { if (PathCombineCleanPath(szPath, szName)) { WCHAR wszPath[MAX_PATH]; if (SHTCharToUnicode(szPath, wszPath, ARRAYSIZE(wszPath))) { bstrRet = SysAllocString(wszPath); } } } } return bstrRet; } HRESULT Channel_GetFolder(LPTSTR pszPath, XMLDOCTYPE xdt ) { TCHAR szFavs[MAX_PATH]; TCHAR szChannel[MAX_PATH]; ULONG cbChannel = sizeof(szChannel); HRESULT hr = E_FAIL; if (SUCCEEDED(Channel_GetBasePath(szFavs, ARRAYSIZE(szFavs)))) { switch (xdt) { case DOC_CHANNEL: // // Get the potentially localized name of the Channel folder from the // registry if it is there. Otherwise just read it from the resource. // Then tack this on the favorites path. // if (ERROR_SUCCESS != SHRegGetUSValue(L"Software\\Microsoft\\Windows\\CurrentVersion", L"ChannelFolderName", NULL, (void*)szChannel, &cbChannel, TRUE, NULL, 0)) { MLLoadString(IDS_CHANNEL_FOLDER, szChannel, ARRAYSIZE(szChannel)); } break; case DOC_SOFTWAREUPDATE: MLLoadString(IDS_SOFTWAREUPDATE_FOLDER, szChannel, ARRAYSIZE(szChannel)); break; } PathCombine(pszPath, szFavs, szChannel); // // If the channels folder doesn't exist create it now // if (!PathFileExists(pszPath)) { // // In IE5+ use the favorites folder if the channels folder does not // exist. // StrCpy(pszPath, szFavs); // // Create special folder with an icon that lives in cdfview.dll // /* Channel_CreateSpecialFolder( pszPath, NULL, NULL, NULL, g_szModuleName, -IDI_CHANNELFOLDER); */ } hr = S_OK; } return hr; } // NOTE: This registry location and name is used by webcheck as well so do not // change it here without updating webcheck. const TCHAR c_szRegKeyWebcheck[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Webcheck"); const TCHAR c_szRegValueChannelGuide[] = TEXT("ChannelGuide"); HRESULT Channel_GetChannelGuide(LPTSTR pszPath, int cch) { ASSERT(pszPath || 0 == cch); HRESULT hr = E_FAIL; HKEY hKey; DWORD dwLen = cch * sizeof(TCHAR); if (RegOpenKey(HKEY_CURRENT_USER, c_szRegKeyWebcheck, &hKey) == ERROR_SUCCESS) { if (RegQueryValueEx(hKey, c_szRegValueChannelGuide, NULL, NULL, (LPBYTE)pszPath, &dwLen) == ERROR_SUCCESS) { hr = S_OK; } RegCloseKey(hKey); } return hr; } // // Channel_OrderChannel - Set the order of the new channel in the folder // void Channel_OrderItem(LPCTSTR szPath) { HRESULT hr; BOOL bRet; LPITEMIDLIST pidlParent = NULL, pidlChild = NULL; IShellFolder *psfDesktop = NULL; IShellFolder *psfParent = NULL; IPersistFolder *pPF = NULL; IOrderList *pOL = NULL; HDPA hdpa = NULL; int iChannel = -1; // Pick a negative index to see if we find it. int iInsert = 0; // Assume we don't find the channel guide. PORDERITEM poi; // Get the full pidl of the new channel (ignore the pidlParent name) hr = Channel_CreateILFromPath(szPath, &pidlParent); if (FAILED(hr)) goto cleanup; // Allocate a child pidl and make the parent pidl pidlChild = ILClone(ILFindLastID(pidlParent)); if (!pidlChild) goto cleanup; bRet = ILRemoveLastID(pidlParent); ASSERT(bRet); // Get the IShellFolder of the parent by going through the // desktop folder. hr = SHGetDesktopFolder(&psfDesktop); if (FAILED(hr)) goto cleanup; hr = psfDesktop->BindToObject(pidlParent, NULL, IID_IShellFolder, (void**)&psfParent); if (FAILED(hr)) goto cleanup; // Get the order list of the parent. hr = CoCreateInstance(CLSID_OrderListExport, NULL, CLSCTX_INPROC_SERVER, IID_IPersistFolder, (void**)&pPF); if (FAILED(hr)) goto cleanup; hr = pPF->Initialize(pidlParent); if (FAILED(hr)) goto cleanup; hr = pPF->QueryInterface(IID_IOrderList, (void**)&pOL); if (FAILED(hr)) goto cleanup; hr = pOL->GetOrderList(&hdpa); // Create a DPA list if there wasn't one already. if (!hdpa) { hdpa = DPA_Create(2); if (!hdpa) goto cleanup; } else { // First, get the channel guide pidl. TCHAR szGuide[MAX_PATH]; WCHAR wzGuide[MAX_PATH]; LPITEMIDLIST pidlGuide = NULL; if (SUCCEEDED(Channel_GetChannelGuide(szGuide, ARRAYSIZE(szGuide)))) { if (SHTCharToUnicode(szGuide, wzGuide, ARRAYSIZE(wzGuide))) { ULONG ucch; hr = psfParent->ParseDisplayName(NULL, NULL, wzGuide, &ucch, &pidlGuide, NULL); ASSERT(!pidlGuide || SUCCEEDED(hr)); } } // Now do the search. // Check to see if the channel is in the DPA list. // Check to see if the channel guide is there and first. int i = 0; poi = (PORDERITEM)DPA_GetPtr(hdpa, i); while (poi) { if (!psfParent->CompareIDs(0, pidlChild, poi->pidl)) iChannel = poi->nOrder; if (pidlGuide && !psfParent->CompareIDs(0, pidlGuide, poi->pidl) && (poi->nOrder == 0)) iInsert = 1; i++; poi = (PORDERITEM)DPA_GetPtr(hdpa, i); } } // If the channel pidl was not found, insert it at the end if (iChannel < 0) { // Allocate an order item to insert hr = pOL->AllocOrderItem(&poi, pidlChild); if (SUCCEEDED(hr)) { iChannel = DPA_InsertPtr(hdpa, 0x7fffffff, poi); if (iChannel >= 0) poi->nOrder = iChannel; } } // Reorder the channels. The new channel is in the list at // position iChannel. We're moving it to position iInsert. if (iChannel >= 0) { int i = 0; poi = (PORDERITEM)DPA_GetPtr(hdpa, i); while (poi) { if (poi->nOrder == iChannel && iChannel >= iInsert) poi->nOrder = iInsert; else if (poi->nOrder >= iInsert && poi->nOrder < iChannel) poi->nOrder++; i++; poi = (PORDERITEM)DPA_GetPtr(hdpa, i); } // Finally, set the order. hr = pOL->SetOrderList(hdpa, psfParent); ASSERT(SUCCEEDED(hr)); } cleanup: if (hdpa) pOL->FreeOrderList(hdpa); if (pOL) pOL->Release(); if (pPF) pPF->Release(); if (psfParent) psfParent->Release(); if (psfDesktop) psfDesktop->Release(); ILFree(pidlParent); // NULL is OK. ILFree(pidlChild); } // // AddChannel - Add a channel // HRESULT AddChannel( LPCTSTR pszName, LPCTSTR pszURL, LPCTSTR pszLogo, LPCTSTR pszWideLogo, LPCTSTR pszIcon, XMLDOCTYPE xdt ) { HRESULT hr = S_OK; BOOL fDirectoryAlreadyExisted = FALSE; // // Find the Channel directory // Attempt to create folder if one doesn't exist // TCHAR szPath[MAX_PATH]; if (FAILED(Channel_GetFolder(szPath, xdt))) { return E_FAIL; // couldn't find Channel Folder or create empty one } // Cleanup each of the path components independently. PathCombineCleanPath(szPath, pszName); TraceMsg(TF_GENERAL, "Channel Path = %s", szPath); // // Make the new folder if it doesn't already exist // if (!PathFileExists(szPath)) { if (Channel_CreateDirectory(szPath) != 0) { return E_FAIL; } } else { fDirectoryAlreadyExisted = TRUE; } // // Mark it as a SYSTEM folder // if (!SetFileAttributes(szPath, FILE_ATTRIBUTE_SYSTEM)) return E_FAIL; // // Add the channel to the registry database of all channels. // //if (FAILED(Reg_WriteChannel(szPath, pszURL))) // return E_FAIL; // // Place the channel folder in the appropriate "order". // Channel_OrderItem(szPath); // Build the CDFINI GUID string // TCHAR szCDFViewGUID[GUID_STR_LEN]; SHStringFromGUID(CLSID_CDFINI, szCDFViewGUID, ARRAYSIZE(szCDFViewGUID)); // // Make desktop.ini // PathCombine(szPath, szPath, TEXT("desktop.ini")); TraceMsg(TF_GENERAL, "INI path = %s", szPath); WritePrivateProfileString(NULL, NULL, NULL, szPath); // Create desktop.ini // // Write in the CDFViewer GUID // EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("CLSID"), szCDFViewGUID, szPath)); // // Write ConfirmFileOp=0 to turn off shell warnings during file operations // EVAL(WritePrivateProfileString( TEXT(".ShellClassInfo"), TEXT("ConfirmFileOp"), TEXT("0"), szPath)); // // Write the actual URL for this channel // EVAL(WritePrivateProfileString( TEXT("Channel"), TEXT("CDFURL"), pszURL, szPath)); Channel_GetAndWriteScreenSaverURL(pszURL, szPath); // // Write the default Logo URL for this channel if present // if (pszLogo) { EVAL(WritePrivateProfileString( TEXT("Channel"), TEXT("Logo"), pszLogo, szPath)); } // // Write the default WideLogo URL for this channel if present // if (pszWideLogo) { EVAL(WritePrivateProfileString( TEXT("Channel"), TEXT("WideLogo"), pszWideLogo, szPath)); } // // Write the default Icon URL for this channel if present // if (pszIcon) { EVAL(WritePrivateProfileString( TEXT("Channel"), TEXT("Icon"), pszIcon, szPath)); } // // Flush the buffers // WritePrivateProfileString(NULL, NULL, NULL, szPath); // Create desktop.ini EVAL(SetFileAttributes(szPath, FILE_ATTRIBUTE_HIDDEN)); // // Notify the system that a new item has been added, or if the item // already existed just send an UPDATEDIR notification // PathRemoveFileSpec(szPath); if (fDirectoryAlreadyExisted) { SHChangeNotify(SHCNE_UPDATEDIR, SHCNF_PATH, (void*)szPath, NULL); } else { SHChangeNotify(SHCNE_MKDIR, SHCNF_PATH, (void*)szPath, NULL); } return S_OK; } // // DeleteChannel - Deletes a channel by name // // Returns // S_OK if channel existed and successfully deleted // S_FALSE if channel didn't exist. // E_FAIL else. // HRESULT DeleteChannel(LPTSTR szName) { TCHAR szFolderPath[MAX_PATH]; TCHAR szDesktopIniPath[MAX_PATH]; if (PathIsRelative(szName)) { // // Find the Channel directory // Note don't create if doesn't exist // // FEATURE: this won't find app channels. if (FAILED(Channel_GetFolder(szFolderPath, DOC_CHANNEL))) { return S_FALSE; // couldn't find Channel Folder so channel can't exist } // Cleanup each of the path components independently. PathCombineCleanPath(szFolderPath, szName); TraceMsg(TF_GENERAL, "Delete Channel Path = %s", szFolderPath); } else { // Assume absolute paths were retrieved by enumeration and // therefore don't need to be "cleaned". StrCpyN(szFolderPath, szName, ARRAYSIZE(szFolderPath)); } // Create the desktop.ini path PathCombine(szDesktopIniPath, szFolderPath, TEXT("desktop.ini")); // Find URL to CDF from desktop.ini TCHAR szCDFURL[INTERNET_MAX_URL_LENGTH]; GetPrivateProfileString(TEXT("Channel"), TEXT("CDFURL"), TEXT(""), szCDFURL, ARRAYSIZE(szCDFURL), szDesktopIniPath); // Remove the URL from the cache preload registry key HKEY hkeyPreload; LONG lRet = RegOpenKeyEx(HKEY_CURRENT_USER, c_szRegKeyCachePreload, 0, KEY_WRITE, &hkeyPreload); if (ERROR_SUCCESS == lRet) { lRet = RegDeleteValue(hkeyPreload, szCDFURL); lRet = RegCloseKey(hkeyPreload); ASSERT(ERROR_SUCCESS == lRet); } if ( !DeleteFile(szDesktopIniPath) || !SetFileAttributes(szFolderPath, FILE_ATTRIBUTE_NORMAL) || // // REVIEW: should change this to delete all contents of channel folder // incase other files get stored there in the future. // !RemoveDirectory(szFolderPath) ) { return S_FALSE; } return S_OK; } // // CountChannels - Counts the number of channels // // Returns the count // DWORD CountChannels(void) { DWORD cChannels = 0; IEnumChannels *pEnum = (IEnumChannels *) new CChannelEnum(CHANENUM_CHANNELFOLDER, NULL); if (pEnum) { CHANNELENUMINFO cei; while (S_OK == pEnum->Next(1, &cei, NULL)) { cChannels++; } pEnum->Release(); } return cChannels; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** OpenChannel *** // // // Description: // Opens a new browser and selects the given channel // // Parameters: // [In] hwndParent - The owner hwnd. // [In] hinst - The hinstance for this process. // [In] pszCmdLine - The local path to the cdf file. This will be the path // to the file in the cache if the cdf came from the net. // [In] nShow - ShowWindow parameter. // // Return: // None. // // Comments: // This is the implementation for the context menu "Open Channel" command. // It gets invoke via RunDll32.exe. // //////////////////////////////////////////////////////////////////////////////// EXTERN_C STDAPI_(void) OpenChannel( HWND hwndParent, HINSTANCE hinst, LPSTR pszCmdLine, int nShow ) { WCHAR wszPath[INTERNET_MAX_URL_LENGTH]; if (MultiByteToWideChar(CP_ACP, 0, pszCmdLine, -1, wszPath, ARRAYSIZE(wszPath))) { OpenChannelHelper(wszPath, hwndParent); } return; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** Subscribe *** // // // Description: // Takes the given cdf through the user subscription path. // // Parameters: // [In] hwndParent - The owner hwnd. // [In] hinst - The hinstance for this process. // [In] pszCmdLine - The local path to the cdf file. This will be the path // to the file in the cache if the cdf came from the net. // [In] nShow - ShowWindow parameter. // // Return: // None. // // Comments: // This is the implementation of the context menu "Subscribe" command. It // gets invoke via rundll32.exe. // // This function handles channel and desktop component cdfs. // //////////////////////////////////////////////////////////////////////////////// EXTERN_C STDAPI_(void) Subscribe( HWND hwndParent, HINSTANCE hinst, LPSTR pszCmdLine, int nShow ) { WCHAR wszPath[INTERNET_MAX_URL_LENGTH]; if (MultiByteToWideChar(CP_ACP, 0, pszCmdLine, -1, wszPath, ARRAYSIZE(wszPath))) { SubscribeToCDF(hwndParent, wszPath, STC_ALL); } return; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** ParseDesktopComponent *** // // // Description: // Fills an information structure for a desktop component and optionally // allows the user to subscribe to the component. // // Parameters: // [In] hwndOwner - The owner hwnd. If NULL there is no attempt to // subscribe to this URL. // [In] wszURL - The URL to the desktop component cdf. // [Out] pInfo - A pointer that receives desktop component information // that is read from the cdf. // // Return: // S_OK if the desktop info could be read. // S_FALSE if the user hits cancel // E_FAIL otherwise. // // Comments: // This function is used by the desktop component property pages. It may // create a subscription to the component but unlike Subscribe, it will // not add a desktop component to the system. Creating the component is // left to the desktop component property pages. // //////////////////////////////////////////////////////////////////////////////// EXTERN_C STDAPI ParseDesktopComponent( HWND hwndOwner, LPWSTR wszURL, COMPONENT* pInfo ) { HRESULT hr; hr = CoInitialize(NULL); if (SUCCEEDED(hr)) { CCdfView* pCCdfView = new CCdfView; if (pCCdfView) { hr = pCCdfView->Load(wszURL, 0); if (SUCCEEDED(hr)) { IXMLDocument* pIXMLDocument; TraceMsg(TF_CDFPARSE, "ParseDesktopComponent"); TCHAR szFile[MAX_PATH]; TCHAR szURL[INTERNET_MAX_URL_LENGTH]; SHUnicodeToTChar(wszURL, szURL, ARRAYSIZE(szURL)); hr = URLDownloadToCacheFile(NULL, szURL, szFile, ARRAYSIZE(szFile), 0, NULL); if (SUCCEEDED(hr)) { hr = pCCdfView->ParseCdf(NULL, &pIXMLDocument, PARSE_LOCAL); if (SUCCEEDED(hr)) { ASSERT(pIXMLDocument); if (DOC_DESKTOPCOMPONENT == XML_GetDocType(pIXMLDocument)) { BOOL fOk = FALSE; if (hwndOwner) { fOk = SubscriptionHelper(pIXMLDocument, hwndOwner, SUBSTYPE_DESKTOPCHANNEL, SUBSACTION_SUBSCRIBEONLY, wszURL, DOC_DESKTOPCOMPONENT, NULL); } if (fOk) { if(SUCCEEDED(hr = XML_GetDesktopComponentInfo(pIXMLDocument, pInfo)) && !pInfo->wszSubscribedURL[0]) { // Since XML_GetDesktopComponentInfo did not fillout the SubscribedURL // field (because the CDF file didn't have a SELF tag), we need to // fill it with what was really subscribed to (the URL for CDF file itself) StrCpyNW(pInfo->wszSubscribedURL, wszURL, ARRAYSIZE(pInfo->wszSubscribedURL)); } } else hr = S_FALSE; } else { hr = E_FAIL; } pIXMLDocument->Release(); } } } pCCdfView->Release(); } } return hr; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** SubscribeToCDF *** // // // Description: // Subscribes to the given cdf. // // Parameters: // [In] hwndOwner - The owner hwnd. Used to display the subscription // wizard. // [In] wszURL - An url to the cdf. // [In] dwFlags - The type of cdf the caller wants to subscribe to. // STC_CHANNEL, STC_DESKTOPCOMPONENT or STC_ALL. // // Return: // S_OK if the subscription worked and the user subscribed. // S_FALSE if the subscription UI appeared but the user decided not to // subscribe. // E_INVALIDARG if the cdf couldn't be opened or parsed. // E_ACCESSDENIED if the cdf file doesn't match the type specified in // dwFlags. // E_FAIL on any other errors. // // Comments: // Private API. Currently called by desktop component drop handler. // //////////////////////////////////////////////////////////////////////////////// EXTERN_C STDAPI SubscribeToCDF( HWND hwndOwner, LPWSTR wszURL, DWORD dwFlags ) { HRESULT hr; hr = CoInitialize(NULL); if (SUCCEEDED(hr)) { CCdfView* pCCdfView = new CCdfView; if (pCCdfView) { hr = pCCdfView->Load(wszURL, 0); if (SUCCEEDED(hr)) { IXMLDocument* pIXMLDocument; TraceMsg(TF_CDFPARSE, "SubscribeToCDF"); hr = pCCdfView->ParseCdf(NULL, &pIXMLDocument, PARSE_LOCAL); if (SUCCEEDED(hr)) { ASSERT(pIXMLDocument); XMLDOCTYPE xdt = XML_GetDocType(pIXMLDocument); switch(xdt) { case DOC_CHANNEL: case DOC_SOFTWAREUPDATE: // Admins can disallow adding channels and limit // the number of installed channels. if ((dwFlags & STC_CHANNEL) && !SHRestricted2W(REST_NoAddingChannels, wszURL, 0) && (!SHRestricted2W(REST_MaxChannelCount, NULL, 0) || (CountChannels() < SHRestricted2W(REST_MaxChannelCount, NULL, 0)))) { if (SubscriptionHelper(pIXMLDocument, hwndOwner, SUBSTYPE_CHANNEL, SUBSACTION_ADDADDITIONALCOMPONENTS, wszURL, xdt, NULL)) { OpenChannelHelper(wszURL, hwndOwner); hr = S_OK; } else { hr = S_FALSE; } } else { hr = E_ACCESSDENIED; } break; case DOC_DESKTOPCOMPONENT: #ifndef UNIX if (hwndOwner && (WhichPlatform() != PLATFORM_INTEGRATED)) #else /* No Active Desktop on Unix */ if (0) #endif /* UNIX */ { TCHAR szText[MAX_PATH]; TCHAR szTitle[MAX_PATH]; MLLoadString(IDS_BROWSERONLY_DLG_TEXT, szText, ARRAYSIZE(szText)); MLLoadString(IDS_BROWSERONLY_DLG_TITLE, szTitle, ARRAYSIZE(szTitle)); MessageBox(hwndOwner, szText, szTitle, MB_OK); } else if (dwFlags & STC_DESKTOPCOMPONENT) { if (SubscriptionHelper(pIXMLDocument, hwndOwner, SUBSTYPE_DESKTOPCHANNEL, SUBSACTION_ADDADDITIONALCOMPONENTS, wszURL, DOC_DESKTOPCOMPONENT, NULL)) { hr = S_OK; } else { hr = S_FALSE; } } else { hr = E_ACCESSDENIED; } break; case DOC_UNKNOWN: //If it is NOT a cdfFile, then we get DOC_UNKNOWN. We must return error // here sothat the caller knows that this is NOT a cdffile. hr = E_INVALIDARG; break; default: break; } pIXMLDocument->Release(); } else { hr = E_INVALIDARG; } } pCCdfView->Release(); } CoUninitialize(); } return hr; } // // Utility functions. // //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** OpenChannelHelper *** // // // Description: // Opens the browser with the wszURL channel selected. // // Parameters: // [In] wszURL - The URL of the cdf file to display. // [In] hwndOwner - The owning hwnd for error messages. Can be NULL. // // Return: // S_OK if the browser was opened. // E_FAIL otherwise. // // Comments: // // //////////////////////////////////////////////////////////////////////////////// HRESULT OpenChannelHelper( LPWSTR wszURL, HWND hwndOwner ) { // // REVIEW: Handle non-channel cdfs. // HRESULT hr; hr = CoInitialize(NULL); if (SUCCEEDED(hr)) { IWebBrowser2* pIWebBrowser2; hr = CoCreateInstance(CLSID_InternetExplorer, NULL, CLSCTX_LOCAL_SERVER, IID_IWebBrowser2, (void**)&pIWebBrowser2); if (SUCCEEDED(hr)) { ASSERT(pIWebBrowser2); // // Navigate to the root url of the cdf ref'd in wszURL // hr = NavigateBrowser(pIWebBrowser2, wszURL, hwndOwner); pIWebBrowser2->put_Visible(VARIANT_TRUE); pIWebBrowser2->Release(); } CoUninitialize(); } return hr; } //////////////////////////////////////////////////////////////////////////////// // // *** ShowChannelPane // // Description - shows the channel pane for the given web browser object // //////////////////////////////////////////////////////////////////////////////// HRESULT ShowChannelPane( IWebBrowser2* pIWebBrowser2 ) { HRESULT hr; VARIANT guid; VARIANT empty = {0}; BSTR bstrGuid; TCHAR szGuid[GUID_STR_LEN]; WCHAR wszGuid[GUID_STR_LEN]; if (!SHRestricted2W(REST_NoChannelUI, NULL, 0)) { SHStringFromGUID(CLSID_FavBand, szGuid, ARRAYSIZE(szGuid)); SHTCharToUnicode(szGuid, wszGuid, ARRAYSIZE(wszGuid)); if ((bstrGuid = SysAllocString(wszGuid)) == NULL) return E_OUTOFMEMORY; guid.vt = VT_BSTR; guid.bstrVal = bstrGuid; hr = pIWebBrowser2->ShowBrowserBar(&guid, &empty, &empty); SysFreeString(bstrGuid); } else { hr = E_FAIL; } return hr; } // // These routines are only needed if we want to pidl a variant pidl for // navigating the ChannelPane to a specific pidl // SAFEARRAY * MakeSafeArrayFromData(const void * pData,DWORD cbData) { SAFEARRAY * psa; if (!pData || 0 == cbData) return NULL; // nothing to do // create a one-dimensional safe array psa = SafeArrayCreateVector(VT_UI1,0,cbData); ASSERT(psa); if (psa) { // copy data into the area in safe array reserved for data // Note we party directly on the pointer instead of using locking/ // unlocking functions. Since we just created this and no one // else could possibly know about it or be using it, this is OK. ASSERT(psa->pvData); memcpy(psa->pvData,pData,cbData); } return psa; } /************************************************************\ FUNCTION: InitVARIANTFromPidl PARAMETER: pvar - Allocated by caller and filled in by this function. pidl - Allocated by caller and caller needs to free. DESCRIPTION: This function will take the PIDL parameter and COPY it into the Variant data structure. This allows the pidl to be freed and the pvar to be used later, however, it is necessary to call VariantClear(pvar) to free memory that this function allocates. \************************************************************/ BOOL InitVARIANTFromPidl(VARIANT* pvar, LPCITEMIDLIST pidl) { UINT cb = ILGetSize(pidl); SAFEARRAY* psa = MakeSafeArrayFromData((const void *)pidl, cb); if (psa) { ASSERT(psa->cDims == 1); // ASSERT(psa->cbElements == cb); ASSERT(ILGetSize((LPCITEMIDLIST)psa->pvData)==cb); VariantInit(pvar); pvar->vt = VT_ARRAY|VT_UI1; pvar->parray = psa; return TRUE; } return FALSE; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** NavigateBrowser *** // // // Description: // Navigate the browser to the correct URL for the given cdf. // // Parameters: // [In] IWebBrowser2 - The browser top naviagte. // [In] szwURL - The path to the cdf file. // [In] hwnd - The wner hwnd. // // Return: // S_OK if the cdf file was parsed and the browser naviagted to teh URL. // E_FAIL otherwise. // // Comments: // Read the href of the first CHANNEL tag in the cfd file and navigate // the browser to this href. // //////////////////////////////////////////////////////////////////////////////// HRESULT NavigateBrowser( IWebBrowser2* pIWebBrowser2, LPWSTR wszURL, HWND hwnd ) { ASSERT(pIWebBrowser2); ASSERT(wszURL); ASSERT(*wszURL != 0); HRESULT hr = E_FAIL; // // Try to navigate to the title page. // CCdfView* pCCdfView = new CCdfView; if (pCCdfView) { hr = pCCdfView->Load(wszURL, 0); if (SUCCEEDED(hr)) { IXMLDocument* pIXMLDocument = NULL; TraceMsg(TF_CDFPARSE, "NavigateBrowser"); BOOL fIsURLChannelShortcut = PathIsDirectoryW(wszURL); hr = pCCdfView->ParseCdf(NULL, &pIXMLDocument, PARSE_LOCAL | PARSE_REMOVEGLEAM); if (SUCCEEDED(hr)) { ASSERT(pIXMLDocument); // // Iff the CDF parsed correctly then show the channel pane // // pIWebBrowser2->put_TheaterMode(VARIANT_TRUE); pIWebBrowser2->put_Visible(VARIANT_TRUE); IXMLElement* pIXMLElement; LONG nIndex; hr = XML_GetFirstChannelElement(pIXMLDocument, &pIXMLElement, &nIndex); if (SUCCEEDED(hr)) { ASSERT(pIXMLElement); BSTR bstrURL = XML_GetAttribute(pIXMLElement, XML_HREF); if (bstrURL && *bstrURL) { if (!fIsURLChannelShortcut) { LPOLESTR pszPath = Channel_GetChannelPanePath(wszURL); if (pszPath) { if (SUCCEEDED(ShowChannelPane(pIWebBrowser2))) NavigateChannelPane(pIWebBrowser2, pszPath); CoTaskMemFree(pszPath); } } else { TCHAR szChanDir[MAX_PATH]; if (SUCCEEDED(Channel_GetFolder(szChanDir, DOC_CHANNEL))) { WCHAR wszChanDir[MAX_PATH]; if (SHTCharToUnicode(szChanDir, wszChanDir, ARRAYSIZE(wszChanDir))) { if (PathIsPrefixW(wszChanDir, wszURL)) { if (SUCCEEDED(ShowChannelPane( pIWebBrowser2))) { NavigateChannelPane(pIWebBrowser2, wszURL); } } } } } VARIANT vNull = {0}; VARIANT vTargetURL; vTargetURL.vt = VT_BSTR; vTargetURL.bstrVal = bstrURL; // // Nav the main browser pain to the target URL // hr = pIWebBrowser2->Navigate2(&vTargetURL, &vNull, &vNull, &vNull, &vNull); } if (bstrURL) SysFreeString(bstrURL); pIXMLElement->Release(); } } else if (OLE_E_NOCACHE == hr) { VARIANT vNull = {0}; VARIANT vTargetURL; if (!fIsURLChannelShortcut) { vTargetURL.bstrVal = SysAllocString(wszURL); } else { vTargetURL.bstrVal = pCCdfView->ReadFromIni(TSTR_INI_URL); } if (vTargetURL.bstrVal) { vTargetURL.vt = VT_BSTR; // // Nav the main browser pain to the target URL // hr = pIWebBrowser2->Navigate2(&vTargetURL, &vNull, &vNull, &vNull, &vNull); SysFreeString(vTargetURL.bstrVal); } } if (pIXMLDocument) pIXMLDocument->Release(); } pCCdfView->Release(); } return hr; } // // Navigate the channel pane to the given channel. // HRESULT NavigateChannelPane( IWebBrowser2* pIWebBrowser2, LPCWSTR pwszPath ) { ASSERT(pIWebBrowser2); HRESULT hr = E_FAIL; if (pwszPath) { TCHAR szPath[MAX_PATH]; if (SHUnicodeToTChar(pwszPath, szPath, ARRAYSIZE(szPath))) { LPITEMIDLIST pidl; if (SUCCEEDED(Channel_CreateILFromPath(szPath, &pidl))) { ASSERT(pidl); VARIANT varPath; if (InitVARIANTFromPidl(&varPath, pidl)) { VARIANT varNull = {0}; VARIANT varFlags; varFlags.vt = VT_I4; varFlags.lVal = navBrowserBar; hr = pIWebBrowser2->Navigate2(&varPath, &varFlags, &varNull, &varNull, &varNull); VariantClear(&varPath); } ILFree(pidl); } } } return hr; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** SubscriptionHelper *** // // // Description: // Gets subscription information from the given document and uses that info // to create a subscription // // Parameters: // [In] pIXMLDocument - A pointer to the cdf document. // [In] hwnd - The ownser hwnd. Used to display UI. // [In] st - The type of subscription. // [In] sa - Flag used to determine if additional steps should // be taken if a user does create a subscription. // // Return: // TRUE if a subscition for this document exists when this function returns. // FALSE if the document doesn't have a subscription and one wasn't created. // // Comments: // If a subscription to a channel is created then a channel shortcut has to // be added to the favorites\channel folder. // // If a subscription to a desktop component is created then the caller // determines if the desktop component gets added to the system. // //////////////////////////////////////////////////////////////////////////////// BOOL SubscriptionHelper( IXMLDocument *pIXMLDocument, HWND hwnd, SUBSCRIPTIONTYPE st, SUBSCRIPTIONACTION sa, LPCWSTR pszwURL, XMLDOCTYPE xdt, BSTR* pbstrSubscribedURL ) { ASSERT(pIXMLDocument); BOOL bChannelInstalled = FALSE; HRESULT hr; IXMLElement* pIXMLElement; LONG nIndex; hr = XML_GetFirstChannelElement(pIXMLDocument, &pIXMLElement, &nIndex); if (SUCCEEDED(hr)) { ASSERT(pIXMLElement); BSTR bstrURL = XML_GetAttribute(pIXMLElement, XML_SELF); if ((NULL == bstrURL || 0 == *bstrURL) && pszwURL) { if (bstrURL) SysFreeString(bstrURL); bstrURL = SysAllocString(pszwURL); } if (bstrURL) { ISubscriptionMgr* pISubscriptionMgr = NULL; #ifndef UNIX hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&pISubscriptionMgr); if (SUCCEEDED(hr)) { ASSERT(pISubscriptionMgr); //hr = pISubscriptionMgr->IsSubscribed(bstrURL, &bSubscribed); bChannelInstalled = Channel_IsInstalled(bstrURL); if (SUBSTYPE_DESKTOPCHANNEL == st && hwnd) { BOOL bSubscribed; hr = pISubscriptionMgr->IsSubscribed(bstrURL, &bSubscribed); if (bSubscribed) { TCHAR szText[MAX_PATH]; TCHAR szTitle[MAX_PATH]; MLLoadString(IDS_OVERWRITE_DLG_TEXT, szText, ARRAYSIZE(szText)); MLLoadString(IDS_OVERWRITE_DLG_TITLE, szTitle, ARRAYSIZE(szTitle)); if (IDYES == MessageBox(hwnd, szText, szTitle, MB_YESNO | MB_ICONQUESTION)) { pISubscriptionMgr->DeleteSubscription(bstrURL, NULL); bChannelInstalled = FALSE; } } } #else bChannelInstalled = Channel_IsInstalled(bstrURL); #endif /* UNIX */ if (!bChannelInstalled) { BSTR bstrName; if (SUBSTYPE_DESKTOPCHANNEL != st) { bstrName = XML_GetAttribute(pIXMLElement, XML_TITLE); } else { IXMLElement* pDskCmpIXMLElement; if (SUCCEEDED(XML_GetFirstDesktopComponentElement( pIXMLDocument, &pDskCmpIXMLElement, &nIndex))) { ASSERT(pDskCmpIXMLElement); bstrName = XML_GetAttribute(pDskCmpIXMLElement, XML_TITLE); pDskCmpIXMLElement->Release(); } else { bstrName = NULL; } } if ((NULL == bstrName || 0 == *bstrName) && pszwURL) { WCHAR szwFilename[MAX_PATH]; if (StrCpyNW(szwFilename, PathFindFileNameW(pszwURL), ARRAYSIZE(szwFilename))) { PathRemoveExtensionW(szwFilename); if (bstrName) SysFreeString(bstrName); bstrName = SysAllocString(szwFilename); } } if (bstrName) { TASK_TRIGGER tt = {0}; SUBSCRIPTIONINFO si = {0}; si.cbSize = sizeof(SUBSCRIPTIONINFO); si.fUpdateFlags |= SUBSINFO_SCHEDULE; si.schedule = SUBSSCHED_AUTO; si.pTrigger = (LPVOID)&tt; XML_GetSubscriptionInfo(pIXMLElement, &si); bChannelInstalled = SubscribeToURL(pISubscriptionMgr, bstrURL, bstrName, &si, hwnd, st, (xdt==DOC_SOFTWAREUPDATE)); #ifndef UNIX if (bChannelInstalled && SUBSACTION_ADDADDITIONALCOMPONENTS == sa) { if (SUBSTYPE_CHANNEL == st) { // Update the subscription if the user has // choosen to view the screen saver item. if ( SUCCEEDED(pISubscriptionMgr->GetSubscriptionInfo(bstrURL, &si)) && (si.fChannelFlags & CHANNEL_AGENT_PRECACHE_SCRNSAVER) ) { pISubscriptionMgr->UpdateSubscription(bstrURL); } //t-mattgi: moved this to AddToFav code in shdocvw //because there, we know what folder to add it to //AddChannel(szName, szURL, NULL, NULL, NULL, xdt); } else if (SUBSTYPE_DESKTOPCHANNEL == st) { COMPONENT Info; if(SUCCEEDED(XML_GetDesktopComponentInfo( pIXMLDocument, &Info))) { if(!Info.wszSubscribedURL[0]) { // Since XML_GetDesktopComponentInfo did not fillout the SubscribedURL // field (because the CDF file didn't have a SELF tag), we need to // fill it with what was really subscribed to (the URL for CDF file itself) StrCpyNW(Info.wszSubscribedURL, bstrURL, ARRAYSIZE(Info.wszSubscribedURL)); } pISubscriptionMgr->UpdateSubscription(bstrURL); AddDesktopComponent(&Info); } } } #endif /* !UNIX */ } } #ifndef UNIX pISubscriptionMgr->Release(); } #endif /* !UNIX */ if (pbstrSubscribedURL) { *pbstrSubscribedURL = bstrURL; } else { SysFreeString(bstrURL); } } pIXMLElement->Release(); } return bChannelInstalled; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** SubscribeToURL *** // // // Description: // Takes the user to the subscription wizard for the given URL. // // Parameters: // [In] ISubscriptionMgr - The subscription manager interface // [In] bstrURL - The URL to subscribe. // [In] bstrName - The name of the subscription. // [In] psi - A subscription information structure. // [In] hwnd - The owner hwnd. // [In] st - The type of subsciption. SUBSTYPE_CHANNEL or // SUSBSTYPE_DESKTOPCOMPONENT. // [In] bIsSoftare - modifies SUBSTYPE_CHANNEL since software updates // don't have their own subscriptiontype // // Return: // TRUE if the user subscribe to the URL. // FALSE if the user didn't subscribe to the URL. // // Comments: // The subscription manager CreateSubscription function takes the user // through the subscriptiopn wizard. // //////////////////////////////////////////////////////////////////////////////// BOOL SubscribeToURL( ISubscriptionMgr* pISubscriptionMgr, BSTR bstrURL, BSTR bstrName, SUBSCRIPTIONINFO* psi, HWND hwnd, SUBSCRIPTIONTYPE st, BOOL bIsSoftware ) { #ifndef UNIX ASSERT(pISubscriptionMgr); #endif /* !UNIX */ ASSERT(bstrURL); ASSERT(bstrName); BOOL bSubscribed = FALSE; DWORD dwFlags = 0; if (Channel_IsInstalled(bstrURL)) { dwFlags |= CREATESUBS_ACTIVATE | CREATESUBS_FROMFAVORITES; } else { dwFlags |= CREATESUBS_ADDTOFAVORITES; } if (bIsSoftware) { dwFlags |= CREATESUBS_SOFTWAREUPDATE; } #ifndef UNIX HRESULT hr = pISubscriptionMgr->CreateSubscription(hwnd, bstrURL, bstrName, dwFlags, st, psi); #else /* Unix does not have subscription support */ /* But, we want to add the channel to favorites */ HRESULT hr = E_FAIL; if ((dwFlags & CREATESUBS_ADDTOFAVORITES) && (st == SUBSTYPE_CHANNEL || st == SUBSTYPE_URL)) hr = SHAddSubscribeFavorite(hwnd, bstrURL, bstrName, dwFlags, st, psi); #endif /* UNIX */ #if 0 if (SUCCEEDED(hr)) { pISubscriptionMgr->IsSubscribed(bstrURL, &bSubscribed); } #else //t-mattgi: REVIEW with edwardP //can't just check if subscribed -- they might choose to add to channel bar without //subscribing, then we still want to return true. or do we? bSubscribed = (hr == S_OK); //case where they clicked OK #endif return bSubscribed; } //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\ // // *** AddDesktopComponent *** // // // Description: // Calls the desktop component manager to add a new component. // // Parameters: // [In] pInfo - Information about the new component to add. // // Return: // S_OK if the component was added. // E_FAIL otherwise. // // Comments: // // //////////////////////////////////////////////////////////////////////////////// HRESULT AddDesktopComponent( COMPONENT* pInfo ) { ASSERT(pInfo); HRESULT hr = S_OK; #ifndef UNIX /* No Active Desktop on Unix */ IActiveDesktop* pIActiveDesktop; hr = CoCreateInstance(CLSID_ActiveDesktop, NULL, CLSCTX_INPROC_SERVER, IID_IActiveDesktop, (void**)&pIActiveDesktop); if (SUCCEEDED(hr)) { ASSERT(pIActiveDesktop); // Assign a default position to the component pInfo->cpPos.iLeft = COMPONENT_DEFAULT_LEFT; pInfo->cpPos.iTop = COMPONENT_DEFAULT_TOP; hr = pIActiveDesktop->AddDesktopItem(pInfo, 0); // // Apply all except refresh as this causes timing issues because the // desktop is in offline mode but not in silent mode // if (SUCCEEDED(hr)) { DWORD dwFlags = AD_APPLY_ALL; // If the desktop component url is already in cache, we want to // refresh right away - otherwise not if(!(CDFIDL_IsCachedURL(pInfo->wszSubscribedURL))) { //It is not in cache, we want to wait until the download // is done before refresh. So don't refresh right away dwFlags &= ~(AD_APPLY_REFRESH); } else dwFlags |= AD_APPLY_BUFFERED_REFRESH; hr = pIActiveDesktop->ApplyChanges(dwFlags); } pIActiveDesktop->Release(); } #endif /* UNIX */ return hr; } //////////////////////////////////////////////////////////////////////////////// // // *** Channel_CreateDirectory *** // // Description: // creates a directory including any intermediate // directories in the path. // // Parameters: // LPCTSTR pszPath - path of directory to create // // Returns: // 0 if function succeeded, else returns GetLastError() // // Comments: // Copied from SHCreateDirectory. Can't use SHCreateDirectory directly // because that fires off an SHChangeNotify msg immediately and need to // fire it off only after the desktop.ini has been created in AddChannel(). // Also would have to do a runtime check for NT vs Win95 as this api doesn't // have A & W versions. // //////////////////////////////////////////////////////////////////////////////// int Channel_CreateDirectory(LPCTSTR pszPath) { int ret = 0; if (!CreateDirectory(pszPath, NULL)) { TCHAR *pSlash, szTemp[MAX_PATH + 1]; // +1 for PathAddBackslash() TCHAR *pEnd; ret = GetLastError(); // There are certain error codes that we should bail out here // before going through and walking up the tree... switch (ret) { case ERROR_FILENAME_EXCED_RANGE: case ERROR_FILE_EXISTS: return(ret); } StrCpyN(szTemp, pszPath, ARRAYSIZE(szTemp) - 1); pEnd = PathAddBackslash(szTemp); // for the loop below // assume we have 'X:\' to start this should even work // on UNC names because will will ignore the first error #ifndef UNIX pSlash = szTemp + 3; #else /* absolute paths on unix start with / */ pSlash = szTemp + 1; #endif /* UNIX */ // create each part of the dir in order while (*pSlash) { while (*pSlash && *pSlash != TEXT(FILENAME_SEPARATOR)) pSlash = CharNext(pSlash); if (*pSlash) { ASSERT(*pSlash == TEXT(FILENAME_SEPARATOR)); *pSlash = 0; // terminate path at seperator if (pSlash + 1 == pEnd) ret = CreateDirectory(szTemp, NULL) ? 0 : GetLastError(); else ret = CreateDirectory(szTemp, NULL) ? 0 : GetLastError(); } *pSlash++ = TEXT(FILENAME_SEPARATOR); // put the seperator back } } return ret; } //////////////////////////////////////////////////////////////////////////////// // PathCombineCleanPath //////////////////////////////////////////////////////////////////////////////// BOOL PathCombineCleanPath ( LPTSTR pszCleanPath, LPCTSTR pszPath ) { TCHAR szComponent[MAX_PATH]; TCHAR * pszComponent = szComponent; BOOL bCleaned = FALSE; if (*pszPath == TEXT(FILENAME_SEPARATOR)) pszPath++; *pszComponent = TEXT('\0'); for (;;) { if ( (*pszPath == TEXT(FILENAME_SEPARATOR)) || (!*pszPath) ) { *pszComponent = TEXT('\0'); #ifdef UNICODE PathCleanupSpec(NULL, szComponent); #else MyPathCleanupSpec(NULL, szComponent); #endif PathCombine(pszCleanPath, pszCleanPath, szComponent); bCleaned = TRUE; if (*pszPath) { // Reset for the next component pszComponent = szComponent; *pszComponent = TEXT('\0'); pszPath = CharNext(pszPath); } else break; } else { LPTSTR pszNextChar = CharNext(pszPath); while (pszPath < pszNextChar) { *pszComponent++ = *pszPath++; } } } return bCleaned; } #ifndef UNICODE //////////////////////////////////////////////////////////////////////////////// // MyPathCleanupPath //////////////////////////////////////////////////////////////////////////////// int MyPathCleanupSpec(LPCTSTR pszDir, LPTSTR pszSpec) { OSVERSIONINFO osvi; osvi.dwOSVersionInfoSize = sizeof(osvi); GetVersionEx(&osvi); if ((VER_PLATFORM_WIN32_NT == osvi.dwPlatformId)) { WCHAR wszDir[MAX_PATH]; WCHAR wszSpec[MAX_PATH]; LPWSTR pwszDir = wszDir; int iRet; if (pszDir) EVAL(MultiByteToWideChar(CP_ACP, 0, pszDir, -1, wszDir, ARRAYSIZE(wszDir)) != 0); else pwszDir = NULL; EVAL(MultiByteToWideChar(CP_ACP, 0, pszSpec, -1, wszSpec, ARRAYSIZE(wszSpec)) != 0); iRet = PathCleanupSpec((LPTSTR)pwszDir, (LPTSTR)wszSpec); EVAL(WideCharToMultiByte(CP_ACP, 0, wszSpec, -1, pszSpec, MAX_PATH, NULL, NULL) != 0); return iRet; } else return PathCleanupSpec(pszDir, pszSpec); } #endif // // Is the given path from the recycle bin? // BOOL IsRecycleBinPath(LPCTSTR pszPath) { ASSERT(pszPath); // // "RECYCLED" is hard coded in bitbuck.c in shell32 // TCHAR* pszRootless = PathSkipRoot(pszPath); return (pszRootless ? (0 == StrNCmpI(TEXT("RECYCLED"), pszRootless, 8)) : FALSE); } //////////////////////////////////////////////////////////////////////////////// // // ICopyHook::CopyCallback // // Either allows the shell to move, copy, delete, or rename a folder or printer // object, or disallows the shell from carrying out the operation. The shell // calls each copy hook handler registered for a folder or printer object until // either all the handlers have been called or one of them returns IDCANCEL. // // RETURNS: // // IDYES - Allows the operation. // IDNO - Prevents the operation on this file, but continues with any other // operations (for example, a batch copy operation). // IDCANCEL - Prevents the current operation and cancels any pending operations // //////////////////////////////////////////////////////////////////////////////// UINT CChannelMgr::CopyCallback( HWND hwnd, // Handle of the parent window for displaying UI objects UINT wFunc, // Operation to perform. UINT wFlags, // Flags that control the operation LPCTSTR pszSrcFile, // Pointer to the source file DWORD dwSrcAttribs, // Source file attributes LPCTSTR pszDestFile, // Pointer to the destination file DWORD dwDestAttribs // Destination file attributes ) { HRESULT hr; // // Return immediately if this isn't a delete of a system folder // Redundant check as this should be made in shdocvw // if (!(dwSrcAttribs & FILE_ATTRIBUTE_SYSTEM) || !(dwSrcAttribs & FILE_ATTRIBUTE_DIRECTORY)) { return IDYES; } // // Build a string containing the guid to check for in the desktop.ini // TCHAR szFolderGUID[GUID_STR_LEN]; TCHAR szCDFViewGUID[GUID_STR_LEN]; SHStringFromGUID(CLSID_CDFINI, szCDFViewGUID, ARRAYSIZE(szCDFViewGUID)); // // Build path to desktop.ini in folder // TCHAR szPath[MAX_PATH]; PathCombine(szPath, pszSrcFile, TEXT("desktop.ini")); // // Read CLSID from desktop.ini if present // GetPrivateProfileString( TEXT(".ShellClassInfo"), TEXT("CLSID"), TEXT(""), szFolderGUID, GUID_STR_LEN, szPath); if (StrEql(szFolderGUID, szCDFViewGUID)) { // // We are deleting/renaming a folder that has the system bit set and // desktop.ini that contains the CLSID for the CDFINI handler, so it // must be a channel // // // Find URL to CDF from desktop.ini // TCHAR szCDFURL[INTERNET_MAX_URL_LENGTH]; WCHAR wszCDFURL[INTERNET_MAX_URL_LENGTH]; GetPrivateProfileString( TEXT("Channel"), TEXT("CDFURL"), TEXT(""), szCDFURL, INTERNET_MAX_URL_LENGTH, szPath); switch(wFunc) { case FO_RENAME: //Reg_RemoveChannel(pszSrcFile); //Reg_WriteChannel(pszDestFile, szCDFURL); break; case FO_COPY: //Reg_WriteChannel(pszDestFile, szCDFURL); break; case FO_MOVE: //Reg_RemoveChannel(pszSrcFile); //Reg_WriteChannel(pszDestFile, szCDFURL); break; case FO_DELETE: TraceMsg(TF_GENERAL, "Deleting a channel"); // // Check if there is a shell restriction against remove channel(s) // if (SHRestricted2(REST_NoRemovingChannels, szCDFURL, 0) || SHRestricted2(REST_NoEditingChannels, szCDFURL, 0) ) { TraceMsg(TF_GENERAL, "Channel Delete blocked by shell restriction"); return IDNO; } else if (!IsRecycleBinPath(pszSrcFile)) { // // Delete the in memory cache entry for this url. // Cache_RemoveItem(szCDFURL); // // Delete the wininet cache entry for this cdf. // DeleteUrlCacheEntry(szCDFURL); // // Remove the channel from the registry. // //Reg_RemoveChannel(pszSrcFile); // // Convert UrlToCdf to wide str, and then to bstr // SHTCharToUnicode(szCDFURL, wszCDFURL, INTERNET_MAX_URL_LENGTH); BSTR bstrCDFURL = SysAllocString(wszCDFURL); if (bstrCDFURL) { // // Create the subscription manager // ISubscriptionMgr* pISubscriptionMgr = NULL; hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&pISubscriptionMgr); if (SUCCEEDED(hr)) { // // Delete the actual subscription to the CDF // hr = pISubscriptionMgr->DeleteSubscription(bstrCDFURL,NULL); TraceMsg(TF_GENERAL, SUCCEEDED(hr) ? "DeleteSubscription Succeeded" : "DeleteSubscription Failed"); pISubscriptionMgr->Release(); } SysFreeString(bstrCDFURL); } } break; default: break; } } return IDYES; } #ifdef UNICODE UINT CChannelMgr::CopyCallback( HWND hwnd, // Handle of the parent window for displaying UI objects UINT wFunc, // Operation to perform. UINT wFlags, // Flags that control the operation LPCSTR pszSrcFile, // Pointer to the source file DWORD dwSrcAttribs, // Source file attributes LPCSTR pszDestFile, // Pointer to the destination file DWORD dwDestAttribs // Destination file attributes ) { WCHAR wszSrcFile[MAX_PATH]; WCHAR wszDestFile[MAX_PATH]; SHAnsiToUnicode(pszSrcFile, wszSrcFile, MAX_PATH); SHAnsiToUnicode(pszDestFile, wszDestFile, MAX_PATH); return CopyCallback(hwnd, wFunc, wFlags, wszSrcFile, dwSrcAttribs, wszDestFile, dwDestAttribs); } #endif HRESULT Channel_CreateILFromPath(LPCTSTR pszPath, LPITEMIDLIST* ppidl) { ASSERT(pszPath); ASSERT(ppidl); HRESULT hr; IShellFolder* pIShellFolder; hr = SHGetDesktopFolder(&pIShellFolder); if (SUCCEEDED(hr)) { ASSERT(pIShellFolder); WCHAR wszPath[MAX_PATH]; if (SHTCharToUnicode(pszPath, wszPath, ARRAYSIZE(wszPath))) { ULONG ucch; hr = pIShellFolder->ParseDisplayName(NULL, NULL, wszPath, &ucch, ppidl, NULL); } else { hr = E_FAIL; } pIShellFolder->Release(); } return hr; } HRESULT GetChannelIconInfo(LPCTSTR pszPath, LPTSTR pszHashItem, int* piIndex, UINT* puFlags) { ASSERT(pszPath); ASSERT(pszHashItem); ASSERT(piIndex); ASSERT(puFlags); HRESULT hr; IShellFolder* pdsktopIShellFolder; hr = SHGetDesktopFolder(&pdsktopIShellFolder); if (SUCCEEDED(hr)) { ASSERT(pdsktopIShellFolder); LPITEMIDLIST pidl; hr = Channel_CreateILFromPath(pszPath, &pidl); if (SUCCEEDED(hr)) { ASSERT(pidl); ASSERT(!ILIsEmpty(pidl)); ASSERT(!ILIsEmpty(_ILNext(pidl))); LPITEMIDLIST pidlLast = ILClone(ILFindLastID(pidl)); if (pidlLast) { if (ILRemoveLastID(pidl)) { IShellFolder* pIShellFolder; hr = pdsktopIShellFolder->BindToObject(pidl, NULL, IID_IShellFolder, (void**)&pIShellFolder); if (SUCCEEDED(hr)) { ASSERT(pIShellFolder); IExtractIcon* pIExtractIcon; hr = pIShellFolder->GetUIObjectOf(NULL, 1, (LPCITEMIDLIST*)&pidlLast, IID_IExtractIcon, NULL, (void**)&pIExtractIcon); if (SUCCEEDED(hr)) { ASSERT(pIExtractIcon); hr = pIExtractIcon->GetIconLocation(0, pszHashItem, MAX_PATH, piIndex, puFlags); pIExtractIcon->Release(); } pIShellFolder->Release(); } } ILFree(pidlLast); } ILFree(pidl); } pdsktopIShellFolder->Release(); } return hr; } HRESULT PreUpdateChannelImage(LPCTSTR pszPath, LPTSTR pszHashItem, int* piIndex, UINT* puFlags, int* piImageIndex) { ASSERT(pszPath); ASSERT(pszHashItem); ASSERT(piIndex); ASSERT(puFlags); ASSERT(piImageIndex); HRESULT hr; TraceMsg(TF_GLEAM, "Pre SHChangeNotify %s", pszPath); hr = GetChannelIconInfo(pszPath, pszHashItem, piIndex, puFlags); if (SUCCEEDED(hr)) { SHFILEINFO fi = {0}; if (SHGetFileInfo(pszPath, 0, &fi, sizeof(SHFILEINFO), SHGFI_SYSICONINDEX)) { *piImageIndex = fi.iIcon; } else { *piImageIndex = -1; } } return hr; } // // UpdateChannelImage is a copy of _SHUpdateImageW found in shdocvw\util.cpp! // void UpdateChannelImage(LPCWSTR pszHashItem, int iIndex, UINT uFlags, int iImageIndex) { SHChangeUpdateImageIDList rgPidl; SHChangeDWORDAsIDList rgDWord; int cLen = MAX_PATH - (lstrlenW( pszHashItem ) + 1); cLen *= sizeof( WCHAR ); if ( cLen < 0 ) cLen = 0; // make sure we send a valid index if ( iImageIndex == -1 ) iImageIndex = II_DOCUMENT; rgPidl.dwProcessID = GetCurrentProcessId(); rgPidl.iIconIndex = iIndex; rgPidl.iCurIndex = iImageIndex; rgPidl.uFlags = uFlags; StrCpyNW( rgPidl.szName, pszHashItem, ARRAYSIZE(rgPidl.szName) ); rgPidl.cb = (USHORT)(sizeof( rgPidl ) - cLen); _ILNext( (LPITEMIDLIST) &rgPidl )->mkid.cb = 0; rgDWord.cb = sizeof( rgDWord) - sizeof(USHORT); rgDWord.dwItem1 = (DWORD) iImageIndex; rgDWord.dwItem2 = 0; rgDWord.cbZero = 0; TraceMsg(TF_GLEAM, "Sending SHChangeNotify %S,%d (image index %d)", pszHashItem, iIndex, iImageIndex); // pump it as an extended event SHChangeNotify(SHCNE_UPDATEIMAGE, SHCNF_IDLIST | SHCNF_FLUSH, &rgDWord, &rgPidl); return; } HRESULT UpdateImage(LPCTSTR pszPath) { ASSERT(pszPath); HRESULT hr; TCHAR szHash[MAX_PATH]; int iIndex; UINT uFlags; int iImageIndex; hr = PreUpdateChannelImage(pszPath, szHash, &iIndex, &uFlags, &iImageIndex); if (SUCCEEDED(hr)) { WCHAR wszHash[MAX_PATH]; SHTCharToUnicode(szHash, wszHash, ARRAYSIZE(wszHash)); UpdateChannelImage(wszHash, iIndex, uFlags, iImageIndex); } return hr; } // // Determines if the channel is installed on the system. // BOOL Channel_IsInstalled( LPCWSTR pszURL ) { ASSERT(pszURL); BOOL fRet = FALSE; CChannelEnum* pCChannelEnum = new CChannelEnum(CHANENUM_CHANNELFOLDER | CHANENUM_SOFTUPDATEFOLDER, pszURL); if (pCChannelEnum) { CHANNELENUMINFO ci; fRet = (S_OK == pCChannelEnum->Next(1, &ci, NULL)); pCChannelEnum->Release(); } return fRet; } HRESULT Channel_WriteScreenSaverURL( LPCWSTR pszURL, LPCWSTR pszScreenSaverURL ) { ASSERT(pszURL); HRESULT hr = S_OK; #ifndef UNIX CChannelEnum* pCChannelEnum = new CChannelEnum(CHANENUM_CHANNELFOLDER | CHANENUM_SOFTUPDATEFOLDER | CHANENUM_PATH, pszURL); if (pCChannelEnum) { CHANNELENUMINFO ci; if (S_OK == pCChannelEnum->Next(1, &ci, NULL)) { TCHAR szDesktopINI[MAX_PATH]; TCHAR szScreenSaverURL[INTERNET_MAX_URL_LENGTH]; ASSERT(ci.pszPath); if (pszScreenSaverURL) { SHUnicodeToTChar(pszScreenSaverURL, szScreenSaverURL, ARRAYSIZE(szScreenSaverURL)); } SHUnicodeToTChar(ci.pszPath, szDesktopINI, ARRAYSIZE(szDesktopINI)); PathCombine(szDesktopINI, szDesktopINI, c_szDesktopINI); WritePrivateProfileString(c_szChannel, c_szScreenSaverURL, pszScreenSaverURL ? szScreenSaverURL : NULL, szDesktopINI); CoTaskMemFree(ci.pszPath); } else { hr = E_FAIL; } pCChannelEnum->Release(); } else { hr = E_OUTOFMEMORY; } #endif /* !UNIX */ return hr; } HRESULT Channel_GetAndWriteScreenSaverURL(LPCTSTR pszURL, LPCTSTR pszDesktopINI) { HRESULT hr = S_OK; #ifndef UNIX CCdfView* pCCdfView = new CCdfView; if (NULL != pCCdfView) { WCHAR wszURL[INTERNET_MAX_URL_LENGTH]; SHTCharToUnicode(pszURL, wszURL, ARRAYSIZE(wszURL)); if (SUCCEEDED(pCCdfView->Load(wszURL, 0))) { IXMLDocument* pIXMLDocument; if (SUCCEEDED(pCCdfView->ParseCdf(NULL, &pIXMLDocument, PARSE_LOCAL))) { BSTR bstrSSUrl; TCHAR szSSURL[INTERNET_MAX_URL_LENGTH]; TCHAR *pszScreenSaverURL = NULL; ASSERT(NULL != pIXMLDocument); if (SUCCEEDED(XML_GetScreenSaverURL(pIXMLDocument, &bstrSSUrl))) { SHUnicodeToTChar(bstrSSUrl, szSSURL, ARRAYSIZE(szSSURL)); pszScreenSaverURL = szSSURL; TraceMsg(TF_GENERAL, "CDFVIEW: %ws has screensaver URL=%ws", wszURL, bstrSSUrl); SysFreeString(bstrSSUrl); } else { TraceMsg(TF_GENERAL, "CDFVIEW: %ws has no screensaver URL", wszURL); } WritePrivateProfileString(c_szChannel, c_szScreenSaverURL, pszScreenSaverURL, pszDesktopINI); pIXMLDocument->Release(); } else { TraceMsg(TF_GENERAL, "CDFVIEW: cdf parse failed %ws", wszURL); } } else { TraceMsg(TF_GENERAL, "CDFVIEW: Couldn't load cdf %ws", wszURL); } pCCdfView->Release(); } else { hr = E_OUTOFMEMORY; } #endif /* !UNIX */ return hr; } HRESULT Channel_RefreshScreenSaverURLs() { HRESULT hr = S_OK; #ifndef UNIX CChannelEnum* pCChannelEnum = new CChannelEnum(CHANENUM_CHANNELFOLDER | CHANENUM_SOFTUPDATEFOLDER | CHANENUM_PATH | CHANENUM_URL, NULL); if (pCChannelEnum) { CHANNELENUMINFO ci; while (S_OK == pCChannelEnum->Next(1, &ci, NULL)) { TCHAR szDesktopINI[MAX_PATH]; TCHAR szURL[INTERNET_MAX_URL_LENGTH]; ASSERT(ci.pszPath); ASSERT(ci.pszURL); SHUnicodeToTChar(ci.pszPath, szDesktopINI, ARRAYSIZE(szDesktopINI)); SHUnicodeToTChar(ci.pszURL, szURL, ARRAYSIZE(szURL)); PathCombine(szDesktopINI, szDesktopINI, c_szDesktopINI); Channel_GetAndWriteScreenSaverURL(szURL, szDesktopINI); CoTaskMemFree(ci.pszPath); CoTaskMemFree(ci.pszURL); } pCChannelEnum->Release(); } else { hr = E_OUTOFMEMORY; } #endif /* !UNIX */ return hr; } // // Finds the path of the channel in the channels folder that has the given url. // LPOLESTR Channel_GetChannelPanePath( LPCWSTR pszURL ) { ASSERT(pszURL); LPOLESTR pstrRet = NULL; CChannelEnum* pCChannelEnum = new CChannelEnum(CHANENUM_CHANNELFOLDER | CHANENUM_PATH, pszURL); if (pCChannelEnum) { CHANNELENUMINFO ci; if (S_OK == pCChannelEnum->Next(1, &ci, NULL)) pstrRet = ci.pszPath; pCChannelEnum->Release(); } return pstrRet; } // // Send SHChangeNotify for agiven URL. // void Channel_SendUpdateNotifications(LPCWSTR pwszURL) { CChannelEnum* pCChannelEnum = new CChannelEnum( CHANENUM_CHANNELFOLDER | CHANENUM_PATH, pwszURL); if (pCChannelEnum) { CHANNELENUMINFO ci; while (S_OK == pCChannelEnum->Next(1, &ci, NULL)) { TCHAR szPath[MAX_PATH]; if (SHUnicodeToTChar(ci.pszPath, szPath, ARRAYSIZE(szPath))) { // // Clearing the gleam flag will result in a SHCNE_UPDATEIMAGE // notification. // TCHAR szURL[INTERNET_MAX_URL_LENGTH]; if (SHUnicodeToTChar(pwszURL, szURL, ARRAYSIZE(szURL))) ClearGleamFlag(szURL, szPath); SHChangeNotify(SHCNE_UPDATEDIR, SHCNF_PATH, (void*)szPath, NULL); } CoTaskMemFree(ci.pszPath); } pCChannelEnum->Release(); } return; }