You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1639 lines
46 KiB
1639 lines
46 KiB
#include "private.h"
|
|
#include "subsmgrp.h"
|
|
#include "offl_cpp.h"
|
|
#include "helper.h"
|
|
#include "propshts.h"
|
|
#include "apithk.h"
|
|
|
|
#include <mluisupp.h>
|
|
|
|
#undef TF_THISMODULE
|
|
#define TF_THISMODULE TF_WEBCHECKCORE
|
|
|
|
extern HRESULT LoadSubscription(LPCTSTR url, LPMYPIDL *);
|
|
extern TCHAR szInternetSettings[];
|
|
|
|
extern void PropagateGeneralProp(HWND, POOEBuf);
|
|
extern HRESULT CreateSubscriptionFromOOEBuf(POOEBuf, LPMYPIDL *);
|
|
|
|
extern BOOL CALLBACK _AddOnePropSheetPage(HPROPSHEETPAGE, LPARAM);
|
|
|
|
#define MAX_STR_LENGTH 200
|
|
|
|
extern DWORD aHelpIDs[];
|
|
extern TCHAR c_szHelpFile[];
|
|
|
|
typedef struct
|
|
{
|
|
CSubscriptionMgr* pMgr;
|
|
LPCWSTR pwszName;
|
|
LPCWSTR pwszUrl;
|
|
SUBSCRIPTIONINFO* pSubsInfo;
|
|
SUBSCRIPTIONTYPE subsType;
|
|
DWORD dwFlags;
|
|
} SUBSCRIBE_ADI_INFO;
|
|
|
|
static const TCHAR SUBSCRIBEADIPROP[] = TEXT("SADIP");
|
|
|
|
INT_PTR CALLBACK SummarizeDesktopSubscriptionDlgProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Private imports from shdocvw (AddToFavorites API)
|
|
//
|
|
STDAPI SHAddSubscribeFavorite (HWND hwnd, LPCWSTR pwszURL, LPCWSTR pwszName, DWORD dwFlags,
|
|
SUBSCRIPTIONTYPE subsType, SUBSCRIPTIONINFO* pInfo);
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
void UpdateSubsInfoFromOOE (SUBSCRIPTIONINFO* pInfo, POOEBuf pooe);
|
|
|
|
HRESULT CreateBSTRFromTSTR(BSTR * pBstr, LPCTSTR sz)
|
|
{
|
|
int i = lstrlen(sz) + 1;
|
|
*pBstr = SysAllocStringLen(NULL, i);
|
|
if(NULL == *pBstr)
|
|
return E_OUTOFMEMORY;
|
|
|
|
MyStrToOleStrN(*pBstr, i, sz);
|
|
return S_OK;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Subsctiption Manager
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
// constructor / destructor
|
|
//
|
|
CSubscriptionMgr::CSubscriptionMgr(void)
|
|
{
|
|
ASSERT(NULL == _pidl);
|
|
|
|
m_cRef = 1;
|
|
m_eInitSrc = _INIT_FROM_URL; // Default.
|
|
m_oldType = SUBSTYPE_URL; // Default.
|
|
DllAddRef();
|
|
}
|
|
|
|
CSubscriptionMgr::~CSubscriptionMgr()
|
|
{
|
|
if (_pidl)
|
|
{
|
|
COfflineFolderEnum::FreePidl(_pidl);
|
|
_pidl = NULL;
|
|
}
|
|
SAFELOCALFREE(m_pBuf);
|
|
SAFERELEASE(m_pUIHelper);
|
|
DllRelease();
|
|
}
|
|
|
|
//
|
|
// IUnknown members
|
|
//
|
|
STDMETHODIMP CSubscriptionMgr::QueryInterface(REFIID riid, void ** ppv)
|
|
{
|
|
*ppv=NULL;
|
|
|
|
// Validate requested interface
|
|
if ((IID_IUnknown == riid) ||
|
|
(IID_ISubscriptionMgr == riid) ||
|
|
(IID_ISubscriptionMgr2 == riid))
|
|
{
|
|
*ppv=(ISubscriptionMgr2 *)this;
|
|
}
|
|
else if(IID_ISubscriptionMgrPriv == riid)
|
|
{
|
|
*ppv=(ISubscriptionMgrPriv *)this;
|
|
}
|
|
else if(IID_IShellExtInit == riid)
|
|
{
|
|
*ppv=(IShellExtInit *)this;
|
|
}
|
|
else if(IID_IShellPropSheetExt == riid)
|
|
{
|
|
*ppv=(IShellPropSheetExt *)this;
|
|
}
|
|
else
|
|
{
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
((LPUNKNOWN)*ppv)->AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
STDMETHODIMP_(ULONG) CSubscriptionMgr::AddRef(void)
|
|
{
|
|
return ++m_cRef;
|
|
}
|
|
|
|
|
|
STDMETHODIMP_(ULONG) CSubscriptionMgr::Release(void)
|
|
{
|
|
if( 0L != --m_cRef )
|
|
return m_cRef;
|
|
|
|
delete this;
|
|
return 0L;
|
|
}
|
|
|
|
|
|
HRESULT CSubscriptionMgr::RemovePages(HWND hDlg)
|
|
{
|
|
HRESULT hr;
|
|
|
|
ASSERT(NULL != m_pUIHelper);
|
|
|
|
if (NULL == m_pUIHelper)
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
ISubscriptionAgentShellExt *psase;
|
|
|
|
hr = m_pUIHelper->QueryInterface(IID_ISubscriptionAgentShellExt, (void **)&psase);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = psase->RemovePages(hDlg);
|
|
psase->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CSubscriptionMgr::SaveSubscription()
|
|
{
|
|
HRESULT hr;
|
|
|
|
ASSERT(NULL != m_pUIHelper);
|
|
|
|
if (NULL == m_pUIHelper)
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
DWORD dwMaxCount = SHRestricted2W(REST_MaxSubscriptionCount, NULL, 0);
|
|
DWORD dwCount;
|
|
SUBSCRIPTIONTYPE subsType = (m_eInitSrc == _INIT_FROM_INTSHCUT) ?
|
|
SUBSTYPE_URL :
|
|
SUBSTYPE_CHANNEL;
|
|
|
|
if ((dwMaxCount > 0) &&
|
|
SUCCEEDED(CountSubscriptions(subsType, &dwCount)) &&
|
|
(dwCount >= dwMaxCount))
|
|
{
|
|
SGMessageBox(GetForegroundWindow(), IDS_RESTRICTED, MB_OK);
|
|
return E_ACCESSDENIED;
|
|
}
|
|
|
|
ISubscriptionAgentShellExt *psase;
|
|
|
|
hr = m_pUIHelper->QueryInterface(IID_ISubscriptionAgentShellExt, (void **)&psase);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = psase->SaveSubscription();
|
|
psase->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CSubscriptionMgr::URLChange(LPCWSTR pwszNewURL)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
HRESULT GetInfoFromDataObject(IDataObject *pido,
|
|
TCHAR *pszPath, DWORD cchPath,
|
|
TCHAR *pszFriendlyName, DWORD cchFriendlyName,
|
|
TCHAR *pszURL, DWORD cchURL,
|
|
INIT_SRC_ENUM *peInitSrc)
|
|
{
|
|
STGMEDIUM stgmed;
|
|
FORMATETC fmtetc = { CF_HDROP, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
|
|
HRESULT hr = pido->GetData(&fmtetc, &stgmed);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
TCHAR szTempURL[INTERNET_MAX_URL_LENGTH];
|
|
TCHAR *pszSection;
|
|
TCHAR *pszEntry;
|
|
TCHAR szTempPath[MAX_PATH];
|
|
TCHAR szIniPath[MAX_PATH];
|
|
|
|
if (DragQueryFile((HDROP)stgmed.hGlobal, 0, szTempPath, ARRAYSIZE(szTempPath)))
|
|
{
|
|
// save path
|
|
if (NULL != pszPath)
|
|
{
|
|
StrCpyN(pszPath, szTempPath, cchPath);
|
|
}
|
|
|
|
StrCpyN(szIniPath, szTempPath, ARRAYSIZE(szIniPath));
|
|
|
|
// make friendly name from path
|
|
if (NULL != pszFriendlyName)
|
|
{
|
|
PathStripPath(szTempPath);
|
|
PathRemoveExtension(szTempPath);
|
|
StrCpyN(pszFriendlyName, szTempPath, cchFriendlyName);
|
|
}
|
|
|
|
if ((NULL != pszURL) || (NULL != peInitSrc))
|
|
{
|
|
|
|
if (PathIsDirectory(szIniPath))
|
|
{
|
|
PathAppend(szIniPath, TEXT("desktop.ini"));
|
|
pszSection = TEXT("Channel");
|
|
pszEntry = TEXT("CDFURL");
|
|
|
|
if (NULL != peInitSrc)
|
|
*peInitSrc = _INIT_FROM_CHANNEL;
|
|
}
|
|
else
|
|
{
|
|
pszSection = TEXT("InternetShortcut");
|
|
pszEntry = TEXT("URL");
|
|
|
|
if (NULL != peInitSrc)
|
|
*peInitSrc = _INIT_FROM_INTSHCUT;
|
|
}
|
|
|
|
if (NULL != pszURL)
|
|
{
|
|
// canonicalize url
|
|
if (SHGetIniString(pszSection, pszEntry,
|
|
szTempURL,
|
|
INTERNET_MAX_URL_LENGTH,
|
|
szIniPath))
|
|
{
|
|
if(!InternetCanonicalizeUrl(szTempURL, pszURL, &cchURL, ICU_NO_ENCODE))
|
|
{
|
|
// failed - use non-canonical version
|
|
StrCpyN(pszURL, szTempURL, cchURL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = E_FAIL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
ReleaseStgMedium(&stgmed);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// IShellExtInit / IShellPropSheetExt members
|
|
//
|
|
STDMETHODIMP CSubscriptionMgr::Initialize(LPCITEMIDLIST pcidlFolder, IDataObject * pido, HKEY hkeyProgID)
|
|
{
|
|
HRESULT hr;
|
|
ISubscriptionItem *psi;
|
|
CLSID clsid;
|
|
SUBSCRIPTIONCOOKIE cookie;
|
|
|
|
hr = GetInfoFromDataObject(pido, m_pszPath, ARRAYSIZE(m_pszPath),
|
|
m_pszFriendly, ARRAYSIZE(m_pszFriendly),
|
|
m_pszURL, ARRAYSIZE(m_pszURL),
|
|
&m_eInitSrc);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = DoGetItemFromURL(m_pszURL, &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
|
|
hr = psi->GetSubscriptionItemInfo(&sii);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
clsid = sii.clsidAgent;
|
|
}
|
|
|
|
psi->GetCookie(&cookie);
|
|
|
|
psi->Release();
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
// New subscription
|
|
hr = S_OK;
|
|
CreateCookie(&cookie);
|
|
|
|
switch (m_eInitSrc)
|
|
{
|
|
case _INIT_FROM_INTSHCUT:
|
|
clsid = CLSID_WebCrawlerAgent;
|
|
break;
|
|
|
|
case _INIT_FROM_CHANNEL:
|
|
clsid = CLSID_ChannelAgent;
|
|
break;
|
|
|
|
default:
|
|
hr = E_FAIL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// HACKHACK:
|
|
// We call coinit and uninit and keep an object pointer around.
|
|
// This ain't cool but will work as long as the agents are in
|
|
// webcheck. Need to fix this for multi-cast handler.
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER,
|
|
IID_IUnknown, (void**)&m_pUIHelper);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISubscriptionAgentShellExt *psase;
|
|
|
|
hr = m_pUIHelper->QueryInterface(IID_ISubscriptionAgentShellExt, (void **)&psase);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
WCHAR wszURL[ARRAYSIZE(m_pszURL)];
|
|
WCHAR wszName[MAX_NAME + 1];
|
|
|
|
MyStrToOleStrN(wszURL, ARRAYSIZE(wszURL), m_pszURL);
|
|
MyStrToOleStrN(wszName, ARRAYSIZE(wszName), m_pszFriendly);
|
|
|
|
hr = psase->Initialize(&cookie, wszURL, wszName,
|
|
(clsid == CLSID_ChannelAgent) ?
|
|
SUBSTYPE_CHANNEL : SUBSTYPE_URL);
|
|
psase->Release();
|
|
}
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSubscriptionMgr::AddPages(LPFNADDPROPSHEETPAGE lpfnAddPage, LPARAM lParam)
|
|
{
|
|
HRESULT hr;
|
|
if (SHRestricted2W(REST_NoEditingSubscriptions, NULL, 0))
|
|
return E_FAIL;
|
|
|
|
ASSERT(NULL != m_pUIHelper);
|
|
|
|
if (NULL == m_pUIHelper)
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
IShellPropSheetExt *pspse;
|
|
|
|
hr = m_pUIHelper->QueryInterface(IID_IShellPropSheetExt, (void **)&pspse);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pspse->AddPages(lpfnAddPage, lParam);
|
|
pspse->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSubscriptionMgr::ReplacePage(UINT uPageID, LPFNADDPROPSHEETPAGE lpfnReplacePage, LPARAM lParam)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
// ISubscriptionMgr members
|
|
//
|
|
|
|
STDMETHODIMP CSubscriptionMgr::IsSubscribed(LPCWSTR pURL, BOOL * pFSub)
|
|
{
|
|
HRESULT hr;
|
|
|
|
ASSERT (pURL && pFSub);
|
|
MyOleStrToStrN(m_pszURL, INTERNET_MAX_URL_LENGTH, pURL);
|
|
|
|
* pFSub = FALSE;
|
|
|
|
if (!_pidl)
|
|
{
|
|
hr = LoadSubscription(m_pszURL, &_pidl);
|
|
if ((hr != S_OK) || (!_pidl))
|
|
return S_OK;
|
|
}
|
|
else if (UrlCompare(URL(&(_pidl->ooe)), m_pszURL, TRUE))
|
|
{
|
|
COfflineFolderEnum::FreePidl(_pidl);
|
|
_pidl = NULL;
|
|
hr = LoadSubscription(m_pszURL, &_pidl);
|
|
if ((hr != S_OK) || (!_pidl))
|
|
return S_OK;
|
|
}
|
|
|
|
* pFSub = TRUE;
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSubscriptionMgr::DeleteSubscription(LPCWSTR pURL, HWND hwnd)
|
|
{
|
|
ASSERT(pURL);
|
|
MyOleStrToStrN(m_pszURL, INTERNET_MAX_URL_LENGTH, pURL);
|
|
|
|
if (!_pidl)
|
|
{
|
|
HRESULT hr;
|
|
|
|
hr = LoadSubscription(m_pszURL, &_pidl);
|
|
if ((hr != S_OK) || (!_pidl))
|
|
return E_FAIL;
|
|
}
|
|
|
|
// This is a restricted action. The restriction
|
|
// is checked in ConfirmDelete. If you remove this call,
|
|
// you must add the restriction check here.
|
|
if (!ConfirmDelete(hwnd, 1, &_pidl))
|
|
return E_FAIL;
|
|
|
|
HRESULT hr = DoDeleteSubscription(&(_pidl->ooe));
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
TraceMsg(TF_ALWAYS, "%s(URL:%s) deleted", NAME(&(_pidl->ooe)), URL(&(_pidl->ooe)));
|
|
|
|
_GenerateEvent(SHCNE_DELETE, (LPITEMIDLIST)_pidl, NULL);
|
|
|
|
COfflineFolderEnum::FreePidl(_pidl);
|
|
_pidl = NULL;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSubscriptionMgr::ShowSubscriptionProperties(LPCWSTR pURL, HWND hwnd)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
LPMYPIDL oldPidl = NULL, newPidl = NULL;
|
|
|
|
ASSERT(pURL);
|
|
MyOleStrToStrN(m_pszURL, INTERNET_MAX_URL_LENGTH, pURL);
|
|
|
|
if (!m_pBuf)
|
|
{
|
|
m_pBuf = (OOEBuf *)MemAlloc(LPTR, sizeof(OOEBuf));
|
|
if (NULL == m_pBuf)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
GetDefaultOOEBuf(m_pBuf, SUBSTYPE_URL);
|
|
|
|
if(SUCCEEDED(LoadSubscription(m_pszURL, &oldPidl)))
|
|
{
|
|
POOEntry pooe = &(oldPidl->ooe);
|
|
|
|
StrCpyN(m_pszFriendly, NAME(&(oldPidl->ooe)), ARRAYSIZE(m_pszFriendly));
|
|
|
|
CopyToOOEBuf(&(oldPidl->ooe), m_pBuf);
|
|
|
|
m_pBuf->m_dwPropSheetFlags = PSF_IS_ALREADY_SUBSCRIBED;
|
|
}
|
|
else
|
|
{
|
|
CreateCookie(&m_pBuf->m_Cookie);
|
|
StrCpyN(m_pszFriendly, m_pszURL, ARRAYSIZE(m_pszFriendly));
|
|
StrCpyN(m_pBuf->m_URL, m_pszURL, ARRAYSIZE(m_pBuf->m_URL));
|
|
StrCpyN(m_pBuf->m_Name, m_pszURL, ARRAYSIZE(m_pBuf->m_Name));
|
|
}
|
|
|
|
hr = CoCreateInstance(*(&m_pBuf->clsidDest), NULL, CLSCTX_INPROC_SERVER,
|
|
IID_IUnknown, (void **)&m_pUIHelper);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISubscriptionAgentShellExt *psase;
|
|
|
|
hr = m_pUIHelper->QueryInterface(IID_ISubscriptionAgentShellExt, (void **)&psase);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
WCHAR wszURL[MAX_URL + 1];
|
|
WCHAR wszName[MAX_NAME + 1];
|
|
|
|
MyStrToOleStrN(wszURL, ARRAYSIZE(wszURL), m_pBuf->m_URL);
|
|
MyStrToOleStrN(wszName, ARRAYSIZE(wszName), m_pBuf->m_Name);
|
|
|
|
hr = psase->Initialize(&m_pBuf->m_Cookie, wszURL, wszName, (SUBSCRIPTIONTYPE)-1);
|
|
psase->Release();
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
PROPSHEETHEADER psh = { 0 } ;
|
|
HPROPSHEETPAGE hPropPage[MAX_PROP_PAGES];
|
|
|
|
// initialize propsheet header.
|
|
psh.dwSize = sizeof(PROPSHEETHEADER);
|
|
psh.dwFlags = PSH_PROPTITLE;
|
|
psh.hwndParent = hwnd;
|
|
psh.pszCaption = m_pszFriendly;
|
|
psh.hInstance = g_hInst;
|
|
psh.nPages = 0;
|
|
psh.nStartPage = 0;
|
|
psh.phpage = hPropPage;
|
|
|
|
PROPSHEETPAGE psp;
|
|
psp.dwSize = sizeof(PROPSHEETPAGE);
|
|
psp.dwFlags = PSP_DEFAULT;
|
|
psp.hInstance = MLGetHinst();
|
|
psp.pszIcon = NULL;
|
|
psp.pszTitle = NULL;
|
|
psp.lParam = (LPARAM)this;
|
|
|
|
psp.pszTemplate = MAKEINTRESOURCE(IDD_SUBSPROPS_SUMMARY);
|
|
psp.pfnDlgProc = SummaryPropDlgProc;
|
|
|
|
psh.phpage[psh.nPages++] = Whistler_CreatePropertySheetPageW(&psp);
|
|
|
|
if (NULL != hPropPage[0])
|
|
{
|
|
if (m_pBuf->m_dwPropSheetFlags & PSF_IS_ALREADY_SUBSCRIBED)
|
|
{
|
|
hr = AddPages(_AddOnePropSheetPage, (LPARAM)&psh);
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
INT_PTR iRet = PropertySheet(&psh);
|
|
|
|
if (iRet < 0)
|
|
{
|
|
hr = E_FAIL;
|
|
}
|
|
else
|
|
{
|
|
hr = LoadSubscription(m_pszURL, &newPidl);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (_pidl)
|
|
{
|
|
COfflineFolderEnum::FreePidl(_pidl);
|
|
}
|
|
|
|
_pidl = newPidl;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (NULL != oldPidl)
|
|
{
|
|
COfflineFolderEnum::FreePidl(oldPidl);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
//
|
|
//
|
|
|
|
void
|
|
CSubscriptionMgr::ChangeSubscriptionValues (
|
|
OOEBuf *pCurrent,
|
|
SUBSCRIPTIONINFO *pNew
|
|
)
|
|
{
|
|
//
|
|
// Channel flags
|
|
//
|
|
|
|
if (SUBSINFO_CHANNELFLAGS & pNew->fUpdateFlags)
|
|
{
|
|
pCurrent->fChannelFlags = pNew->fChannelFlags;
|
|
}
|
|
|
|
|
|
//
|
|
// The subscription schedule.
|
|
//
|
|
|
|
if (SUBSINFO_SCHEDULE & pNew->fUpdateFlags)
|
|
{
|
|
|
|
switch (pNew->schedule)
|
|
{
|
|
|
|
case SUBSSCHED_DAILY:
|
|
case SUBSSCHED_MANUAL:
|
|
case SUBSSCHED_WEEKLY:
|
|
LoadGroupCookie(&pCurrent->groupCookie, pNew->schedule);
|
|
break;
|
|
|
|
case SUBSSCHED_CUSTOM:
|
|
pCurrent->groupCookie = pNew->customGroupCookie;
|
|
break;
|
|
|
|
case SUBSSCHED_AUTO:
|
|
{
|
|
// FEATURE. We should look at subType;
|
|
memset(&pCurrent->groupCookie, 0, sizeof(pCurrent->groupCookie)); //t-mattgi so it will look at the trigger
|
|
PTASK_TRIGGER pNewTrigger = ((PTASK_TRIGGER)pNew->pTrigger);
|
|
if (pNewTrigger && pNewTrigger->cbTriggerSize == sizeof(TASK_TRIGGER))
|
|
{
|
|
pCurrent->m_Trigger = *pNewTrigger;
|
|
}
|
|
else //bad trigger; use daily as default
|
|
{
|
|
pCurrent->m_Trigger.cbTriggerSize = 0;
|
|
pCurrent->groupCookie = NOTFCOOKIE_SCHEDULE_GROUP_DAILY;
|
|
}
|
|
}
|
|
pCurrent->fChannelFlags |= CHANNEL_AGENT_DYNAMIC_SCHEDULE;
|
|
break;
|
|
|
|
default:
|
|
ASSERT(FALSE);
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Recurse levels.
|
|
//
|
|
|
|
if (SUBSINFO_RECURSE & pNew->fUpdateFlags)
|
|
pCurrent->m_RecurseLevels = pNew->dwRecurseLevels;
|
|
|
|
//
|
|
// Webcrawler flags. Note: The flags are not or'd with the current flags.
|
|
// The caller must set all of the webcrawler flags they want to use.
|
|
//
|
|
|
|
if (SUBSINFO_WEBCRAWL & pNew->fUpdateFlags)
|
|
pCurrent->m_RecurseFlags = pNew->fWebcrawlerFlags;
|
|
|
|
//
|
|
// Mail notification.
|
|
//
|
|
|
|
if (SUBSINFO_MAILNOT & pNew->fUpdateFlags)
|
|
pCurrent->bMail = pNew->bMailNotification;
|
|
else
|
|
pCurrent->bMail = FALSE;
|
|
|
|
//
|
|
// Need password.
|
|
//
|
|
|
|
if (SUBSINFO_NEEDPASSWORD & pNew->fUpdateFlags)
|
|
pCurrent->bNeedPassword = pNew->bNeedPassword;
|
|
else
|
|
pCurrent->bNeedPassword = FALSE;
|
|
|
|
//
|
|
// User name.
|
|
//
|
|
|
|
if (SUBSINFO_USER & pNew->fUpdateFlags)
|
|
{
|
|
if (pNew->bstrUserName)
|
|
{
|
|
MyOleStrToStrN(pCurrent->username, MAX_USERNAME, pNew->bstrUserName);
|
|
}
|
|
pCurrent->bNeedPassword = pNew->bNeedPassword;
|
|
}
|
|
|
|
//
|
|
// Password.
|
|
//
|
|
|
|
if (SUBSINFO_PASSWORD & pNew->fUpdateFlags)
|
|
{
|
|
if (pNew->bstrPassword)
|
|
{
|
|
MyOleStrToStrN(pCurrent->password, MAX_PASSWORD, pNew->bstrPassword);
|
|
}
|
|
pCurrent->bNeedPassword = pNew->bNeedPassword;
|
|
}
|
|
|
|
//
|
|
// Friendly Name.
|
|
//
|
|
|
|
if (SUBSINFO_FRIENDLYNAME & pNew->fUpdateFlags)
|
|
{
|
|
if (pNew->bstrFriendlyName)
|
|
{
|
|
MyOleStrToStrN(pCurrent->m_Name, MAX_NAME, pNew->bstrFriendlyName);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Gleam
|
|
//
|
|
|
|
if (SUBSINFO_GLEAM & pNew->fUpdateFlags)
|
|
{
|
|
pCurrent->bGleam = pNew->bGleam;
|
|
}
|
|
|
|
//
|
|
// Changes only (notification only)
|
|
//
|
|
|
|
if (SUBSINFO_CHANGESONLY & pNew->fUpdateFlags)
|
|
{
|
|
pCurrent->bChangesOnly = pNew->bChangesOnly;
|
|
}
|
|
|
|
//
|
|
// dwMaxSizeKB
|
|
//
|
|
if (SUBSINFO_MAXSIZEKB & pNew->fUpdateFlags)
|
|
{
|
|
pCurrent->m_SizeLimit = pNew->dwMaxSizeKB;
|
|
}
|
|
|
|
//
|
|
// Task flags
|
|
//
|
|
if (SUBSINFO_TASKFLAGS & pNew->fUpdateFlags)
|
|
{
|
|
pCurrent->grfTaskTrigger = pNew->fTaskFlags;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// CSubscriptionMgr::CountSubscriptions
|
|
// FEATURE: We could make this public if other people need it. An enumerator
|
|
// would be more useful though...
|
|
//
|
|
HRESULT CSubscriptionMgr::CountSubscriptions(SUBSCRIPTIONTYPE subType, PDWORD pdwCount)
|
|
{
|
|
HRESULT hr;
|
|
IEnumSubscription *pes;
|
|
|
|
ASSERT(NULL != pdwCount);
|
|
|
|
*pdwCount = 0;
|
|
|
|
hr = EnumSubscriptions(0, &pes);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONCOOKIE cookie;
|
|
|
|
while (S_OK == pes->Next(1, &cookie, NULL))
|
|
{
|
|
ISubscriptionItem *psi;
|
|
DWORD dwRet;
|
|
|
|
if (SUCCEEDED(SubscriptionItemFromCookie(FALSE, &cookie, &psi)))
|
|
{
|
|
if (SUCCEEDED(ReadDWORD(psi, c_szPropChannel, &dwRet)) && dwRet)
|
|
{
|
|
if (SUBSTYPE_CHANNEL == subType)
|
|
(*pdwCount)++;
|
|
}
|
|
else if (SUCCEEDED(ReadDWORD(psi, c_szPropDesktopComponent, &dwRet)) && dwRet)
|
|
{
|
|
if (SUBSTYPE_DESKTOPURL == subType || SUBSTYPE_DESKTOPCHANNEL == subType)
|
|
(*pdwCount)++;
|
|
}
|
|
else
|
|
{
|
|
if (SUBSTYPE_URL == subType)
|
|
(*pdwCount)++;
|
|
}
|
|
psi->Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// CSubscriptionMgr::IsValidSubscriptionInfo
|
|
//
|
|
|
|
#define SUBSCRIPTIONSCHEDULE_MAX 4
|
|
|
|
BOOL CSubscriptionMgr::IsValidSubscriptionInfo(SUBSCRIPTIONTYPE subType, SUBSCRIPTIONINFO *pSI)
|
|
{
|
|
if (pSI->cbSize != sizeof(SUBSCRIPTIONINFO))
|
|
{
|
|
return FALSE;
|
|
}
|
|
else if (pSI->fUpdateFlags & ~SUBSINFO_ALLFLAGS)
|
|
{
|
|
return FALSE;
|
|
}
|
|
else if (pSI->pTrigger && ((TASK_TRIGGER*)(pSI->pTrigger))->cbTriggerSize &&
|
|
(subType == SUBSTYPE_URL || subType == SUBSTYPE_DESKTOPURL)) // || pSI->schedule != SUBSSCHED_AUTO))
|
|
{
|
|
return FALSE;
|
|
}
|
|
else if (pSI->fUpdateFlags & SUBSINFO_SCHEDULE)
|
|
{
|
|
if (pSI->schedule > SUBSCRIPTIONSCHEDULE_MAX)
|
|
{
|
|
return FALSE;
|
|
}
|
|
if (pSI->schedule == SUBSSCHED_CUSTOM && pSI->customGroupCookie == CLSID_NULL)
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// *** 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 RemoveURLMapping(LPCWSTR pszURL)
|
|
{
|
|
BYTE cei[MY_MAX_CACHE_ENTRY_INFO];
|
|
DWORD cbcei = sizeof(cei);
|
|
LPINTERNET_CACHE_ENTRY_INFOW pcei = (LPINTERNET_CACHE_ENTRY_INFOW)cei;
|
|
|
|
//
|
|
// Look up the url in the cache
|
|
//
|
|
if (GetUrlCacheEntryInfoExW(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;
|
|
SetUrlCacheEntryInfoW(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)
|
|
{
|
|
RegDeleteValueW(hk, pszURL);
|
|
RegCloseKey(hk);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// CSubscriptionMgr::CreateSubscription
|
|
// entry point for creating a subscription
|
|
// flags:
|
|
// CREATESUBS_FROMFAVORITES -- already exists in favorites, use alternate summary dialog
|
|
// that doesn't do AddToFavorites. Valid only for channel or url.
|
|
// CREATESUBS_INACTIVEPLATINUM -- when creating a channel subscription, show Activate Channel dialog
|
|
// valid only for channel subscriptions with CREATESUBS_FROMFAVORITES
|
|
// CREATESUBS_ADDTOFAVORITES -- display summary dialog before wizard
|
|
// default summary: for channel or url, use AddToFavorites from shdocvw
|
|
// for desktop item, just a confirmation dialog
|
|
// for other, no summary -- straight to wizard
|
|
// CREATESUBS_NOUI -- totally silent
|
|
// CREATESUBS_NOSAVE -- update subscription in memory buffer, not on disk (pInfo must be non-NULL)
|
|
//
|
|
STDMETHODIMP
|
|
CSubscriptionMgr::CreateSubscription (
|
|
HWND hwnd,
|
|
LPCWSTR pwszURL,
|
|
LPCWSTR pwszFriendlyName,
|
|
DWORD dwFlags,
|
|
SUBSCRIPTIONTYPE subsType,
|
|
SUBSCRIPTIONINFO *pInfo
|
|
)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
BOOL bAlready;
|
|
|
|
if (IsFlagSet(dwFlags, CREATESUBS_NOUI) || !IsFlagSet (dwFlags, CREATESUBS_ADDTOFAVORITES))
|
|
{ //no UI, so skip ATF dialog
|
|
hr = CreateSubscriptionNoSummary (hwnd, pwszURL, pwszFriendlyName, dwFlags,
|
|
subsType, pInfo);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
//
|
|
// The user successfully subscribed to this URL so remove
|
|
// mappings used for preinstalled content as this URL is now
|
|
// "Activated"
|
|
//
|
|
RemoveURLMapping(pwszURL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (subsType)
|
|
{
|
|
case SUBSTYPE_URL:
|
|
case SUBSTYPE_CHANNEL:
|
|
hr = SHAddSubscribeFavorite (hwnd, pwszURL, pwszFriendlyName,
|
|
dwFlags, subsType, pInfo);
|
|
break;
|
|
|
|
case SUBSTYPE_DESKTOPCHANNEL:
|
|
case SUBSTYPE_DESKTOPURL:
|
|
hr = IsSubscribed (pwszURL, &bAlready);
|
|
if (SUCCEEDED (hr) && bAlready)
|
|
break; //don't display summary dialog since it has nothing useful for this case
|
|
hr = CreateDesktopSubscription (hwnd, pwszURL, pwszFriendlyName,
|
|
dwFlags, subsType, pInfo);
|
|
break;
|
|
|
|
default: //SUBSTYPE_EXTERNAL -- don't know what kind of summary to show
|
|
hr = CreateSubscriptionNoSummary (hwnd, pwszURL, pwszFriendlyName, dwFlags,
|
|
subsType, pInfo);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//
|
|
// CSubscriptionMgr::CreateSubscriptionNoSummary
|
|
// modify a SUBSCRIPTIONINFO interactively, using the wizard
|
|
// persists info to Subscriptions folder, unless SUBSINFO_NOSAVE passed
|
|
//
|
|
STDMETHODIMP
|
|
CSubscriptionMgr::CreateSubscriptionNoSummary (
|
|
HWND hwnd,
|
|
LPCWSTR pwszURL,
|
|
LPCWSTR pwszFriendlyName,
|
|
DWORD dwFlags,
|
|
SUBSCRIPTIONTYPE subType,
|
|
SUBSCRIPTIONINFO *pInfo
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
//
|
|
// Validate the parameters.
|
|
//
|
|
if (!IS_VALID_SUBSCRIPTIONTYPE(subType)
|
|
|| !pwszURL
|
|
|| !pwszFriendlyName
|
|
|| (!pInfo && (dwFlags & CREATESUBS_NOSAVE))
|
|
|| (pInfo && !IsValidSubscriptionInfo(subType, pInfo)))
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
//
|
|
// Fail if already subscribed and we aren't in no save or no UI mode.
|
|
// Caller is responsible for UI.
|
|
//
|
|
BOOL fAlreadySubscribed;
|
|
if ((FAILED(IsSubscribed(pwszURL, &fAlreadySubscribed)) || fAlreadySubscribed) &&
|
|
(!(dwFlags & (CREATESUBS_NOSAVE | CREATESUBS_NOUI))))
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
//
|
|
// Fail if restrictions are in place.
|
|
// FEATURE: Currently cdfview is handling channel restrictions for
|
|
// but we should probably do it here.
|
|
// Should we have a flag parameter to override this?
|
|
//
|
|
if (SUBSTYPE_URL == subType)
|
|
{
|
|
DWORD dwMaxCount = SHRestricted2W(REST_MaxSubscriptionCount, NULL, 0);
|
|
DWORD dwCount;
|
|
if (SHRestricted2W(REST_NoAddingSubscriptions, pwszURL, 0)
|
|
|| ((dwMaxCount > 0)
|
|
&& SUCCEEDED(CountSubscriptions(subType, &dwCount))
|
|
&& (dwCount >= dwMaxCount)))
|
|
{
|
|
if (!IsFlagSet(dwFlags, CREATESUBS_NOUI))
|
|
SGMessageBox(hwnd, IDS_RESTRICTED, MB_OK);
|
|
return E_ACCESSDENIED;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Get the subscription defaults and merge in the caller's info
|
|
//
|
|
OOEBuf subProps;
|
|
GetDefaultOOEBuf(&subProps, subType);
|
|
|
|
//this is (intentionally) duplicated below... it needs to be after the ChangeSubscriptionValues()
|
|
//call, but we need to grab the url first to make sure it's subscribable.
|
|
MyOleStrToStrN(subProps.m_URL, INTERNET_MAX_URL_LENGTH, pwszURL);
|
|
|
|
// Does this mean we can't support plugin protocols?
|
|
if (/*(subType != SUBSTYPE_EXTERNAL) &&*/ !IsHTTPPrefixed(subProps.m_URL))
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (pInfo)
|
|
{
|
|
ChangeSubscriptionValues(&subProps, pInfo);
|
|
if (fAlreadySubscribed)
|
|
{
|
|
ReadCookieFromInetDB(subProps.m_URL, &subProps.m_Cookie);
|
|
subProps.m_dwPropSheetFlags |= PSF_IS_ALREADY_SUBSCRIBED;
|
|
}
|
|
}
|
|
|
|
// Disallow password caching if restriction is in place. This both
|
|
// skips the wizard page and prevents the caller's password from
|
|
// being saved.
|
|
if (SHRestricted2W(REST_NoSubscriptionPasswords, NULL, 0))
|
|
{
|
|
subProps.bNeedPassword = FALSE;
|
|
subProps.username[0] = 0;
|
|
subProps.password[0] = 0;
|
|
subProps.dwFlags &= ~(PROP_WEBCRAWL_UNAME | PROP_WEBCRAWL_PSWD);
|
|
}
|
|
|
|
//the passed-in name and url override whatever's in the info buffer
|
|
MyOleStrToStrN(subProps.m_URL, INTERNET_MAX_URL_LENGTH, pwszURL);
|
|
MyOleStrToStrN(subProps.m_Name, MAX_NAME_QUICKLINK, pwszFriendlyName);
|
|
|
|
//
|
|
// If we're in UI mode, initialize the wizard
|
|
//
|
|
if (!IsFlagSet(dwFlags, CREATESUBS_NOUI))
|
|
{
|
|
hr = CreateWizard(hwnd, subType, &subProps);
|
|
|
|
} // !NOUI
|
|
|
|
//
|
|
// If we're not in NOSAVE mode, then create/save the subscription
|
|
//
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (!IsFlagSet(dwFlags, CREATESUBS_NOSAVE))
|
|
{
|
|
//
|
|
// Create a new pidl with the user specified properties.
|
|
//
|
|
if (_pidl)
|
|
{
|
|
COfflineFolderEnum::FreePidl(_pidl);
|
|
_pidl = NULL;
|
|
SAFERELEASE(m_pUIHelper);
|
|
}
|
|
hr = CreateSubscriptionFromOOEBuf(&subProps, &_pidl);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ASSERT(_pidl);
|
|
//
|
|
// Send a notification that a subscription has changed.
|
|
//
|
|
_GenerateEvent(SHCNE_CREATE, (LPITEMIDLIST)_pidl, NULL);
|
|
}
|
|
} //!NOSAVE
|
|
else if (S_OK == hr)
|
|
{
|
|
//in NOSAVE mode, so don't actually create subscription -- save it back
|
|
//to passed-in buffer
|
|
ASSERT (pInfo);
|
|
pInfo->fUpdateFlags = SUBSINFO_ALLFLAGS; //fill in all possible fields
|
|
UpdateSubsInfoFromOOE (pInfo, &subProps);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CSubscriptionMgr::CreateDesktopSubscription (HWND hwnd, LPCWSTR pwszURL, LPCWSTR pwszFriendlyName,
|
|
DWORD dwFlags, SUBSCRIPTIONTYPE subsType, SUBSCRIPTIONINFO *pInfo)
|
|
{
|
|
HRESULT hr;
|
|
SUBSCRIPTIONINFO siTemp = { sizeof(SUBSCRIPTIONINFO), 0 };
|
|
if (!pInfo)
|
|
pInfo = &siTemp; //make sure we have a valid buffer if caller doesn't give us one
|
|
|
|
//make sure adminrestrictions allow this
|
|
|
|
if (SHRestricted2W(REST_NoAddingChannels, pwszURL, 0))
|
|
return E_FAIL;
|
|
|
|
SUBSCRIBE_ADI_INFO parms = { this, pwszFriendlyName, pwszURL, pInfo, subsType, dwFlags };
|
|
|
|
//make sure this url is subscribable; if not, show error dialog
|
|
{
|
|
TCHAR sz[MAX_URL];
|
|
MyOleStrToStrN (sz, ARRAYSIZE(sz), pwszURL);
|
|
|
|
if (!IsHTTPPrefixed (sz))
|
|
{
|
|
SGMessageBox(hwnd, IDS_HTTPONLY, MB_ICONINFORMATION | MB_OK);
|
|
return E_INVALIDARG;
|
|
}
|
|
}
|
|
|
|
INT_PTR iDlgResult = DialogBoxParam (MLGetHinst(), MAKEINTRESOURCE(IDD_DESKTOP_SUBSCRIPTION_SUMMARY),
|
|
hwnd, SummarizeDesktopSubscriptionDlgProc, (LPARAM)&parms);
|
|
|
|
switch (iDlgResult)
|
|
{
|
|
case -1:
|
|
hr = E_FAIL;
|
|
break;
|
|
case IDCANCEL:
|
|
hr = S_FALSE;
|
|
break;
|
|
default:
|
|
hr = CreateSubscriptionNoSummary (hwnd, pwszURL, pwszFriendlyName,
|
|
CREATESUBS_NOUI | dwFlags, subsType, pInfo);
|
|
break;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CSubscriptionMgr::GetDefaultInfo(
|
|
SUBSCRIPTIONTYPE subType,
|
|
SUBSCRIPTIONINFO *pInfo
|
|
)
|
|
{
|
|
//
|
|
// Validate the parameters.
|
|
//
|
|
if (!IS_VALID_SUBSCRIPTIONTYPE(subType)
|
|
|| !pInfo
|
|
|| (pInfo->cbSize != sizeof(SUBSCRIPTIONINFO)))
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
memset((void *)pInfo, 0, sizeof(SUBSCRIPTIONINFO));
|
|
pInfo->cbSize = sizeof(SUBSCRIPTIONINFO);
|
|
|
|
// Fill in default structure. Note that lines are commented out
|
|
// below to indicate the field is initialized to 0 without wasting
|
|
// code (memset above already cleared structure out.)
|
|
|
|
pInfo->fUpdateFlags = SUBSINFO_RECURSE | SUBSINFO_MAILNOT
|
|
| SUBSINFO_WEBCRAWL
|
|
/*| SUBSINFO_SCHEDULE */ | SUBSINFO_CHANGESONLY
|
|
| SUBSINFO_CHANNELFLAGS;
|
|
pInfo->dwRecurseLevels = DEFAULTLEVEL;
|
|
pInfo->schedule = SUBSSCHED_AUTO;
|
|
|
|
switch (subType)
|
|
{
|
|
case SUBSTYPE_URL:
|
|
// pInfo->bChangesOnly = FALSE;
|
|
// pInfo->bMailNotification = FALSE;
|
|
// pInfo->bPasswordNeeded = FALSE;
|
|
pInfo->fWebcrawlerFlags = DEFAULTFLAGS;
|
|
break;
|
|
|
|
case SUBSTYPE_CHANNEL:
|
|
// pInfo->bChangesOnly = FALSE;
|
|
// pInfo->bMailNotification = FALSE;
|
|
pInfo->fChannelFlags = CHANNEL_AGENT_PRECACHE_ALL | CHANNEL_AGENT_DYNAMIC_SCHEDULE;
|
|
break;
|
|
|
|
case SUBSTYPE_DESKTOPCHANNEL:
|
|
// pInfo->bChangesOnly = FALSE;
|
|
// pInfo->bMailNotification = FALSE;
|
|
pInfo->fChannelFlags = CHANNEL_AGENT_PRECACHE_ALL | CHANNEL_AGENT_DYNAMIC_SCHEDULE;
|
|
break;
|
|
|
|
case SUBSTYPE_DESKTOPURL:
|
|
// pInfo->bChangesOnly = FALSE;
|
|
// pInfo->bMailNotification = FALSE;
|
|
pInfo->fWebcrawlerFlags = DEFAULTFLAGS;
|
|
break;
|
|
|
|
default:
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CSubscriptionMgr::GetSubscriptionInfo(
|
|
LPCWSTR pwszURL,
|
|
SUBSCRIPTIONINFO *pInfo
|
|
)
|
|
{
|
|
HRESULT hr;
|
|
|
|
//
|
|
// Validate the parameters.
|
|
//
|
|
if (!pInfo
|
|
|| !pwszURL
|
|
|| (pInfo->cbSize != sizeof(SUBSCRIPTIONINFO)))
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
BOOL bSubscribe;
|
|
hr = IsSubscribed(pwszURL, &bSubscribe);
|
|
|
|
RETURN_ON_FAILURE(hr);
|
|
if (!bSubscribe)
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
|
|
// We cannot rely on the caller passing us a clean SUBSCRIPTIONINFO
|
|
// structure. We need to clean it ourselves.
|
|
DWORD dwFlags = pInfo->fUpdateFlags;
|
|
ZeroMemory(pInfo, sizeof(SUBSCRIPTIONINFO));
|
|
pInfo->cbSize = sizeof(SUBSCRIPTIONINFO);
|
|
pInfo->fUpdateFlags = dwFlags;
|
|
|
|
OOEBuf ooeb; // Alas, we need the code in UpdateSubsInfoFromOOE
|
|
CopyToOOEBuf (&(_pidl->ooe), &ooeb); //to work once for a buf and once for an entry, and it's
|
|
UpdateSubsInfoFromOOE (pInfo, &ooeb); //easier to convert entry->buf so we do that here.
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
void UpdateSubsInfoFromOOE (SUBSCRIPTIONINFO* pInfo, POOEBuf pooe)
|
|
{
|
|
DWORD dwFlags = pInfo->fUpdateFlags & SUBSINFO_ALLFLAGS;
|
|
SUBSCRIPTIONTYPE subType = GetItemCategory(pooe);
|
|
|
|
if (dwFlags & SUBSINFO_USER)
|
|
{
|
|
SAFEFREEBSTR (pInfo->bstrUserName);
|
|
CreateBSTRFromTSTR(&(pInfo->bstrUserName), pooe->username);
|
|
}
|
|
if (dwFlags & SUBSINFO_PASSWORD)
|
|
{
|
|
SAFEFREEBSTR (pInfo->bstrPassword);
|
|
CreateBSTRFromTSTR(&(pInfo->bstrPassword), pooe->password);
|
|
}
|
|
if (dwFlags & SUBSINFO_FRIENDLYNAME)
|
|
{
|
|
SAFEFREEBSTR (pInfo->bstrFriendlyName);
|
|
CreateBSTRFromTSTR(&(pInfo->bstrFriendlyName), pooe->m_Name);
|
|
}
|
|
|
|
pInfo->fUpdateFlags = dwFlags;
|
|
if (dwFlags & SUBSINFO_SCHEDULE)
|
|
{
|
|
pInfo->schedule = GetGroup(pooe);
|
|
if (pInfo->schedule == SUBSSCHED_CUSTOM)
|
|
{
|
|
if (pooe->groupCookie != GUID_NULL)
|
|
{
|
|
pInfo->customGroupCookie = pooe->groupCookie;
|
|
}
|
|
else
|
|
{
|
|
GetItemSchedule(&pooe->m_Cookie, &pInfo->customGroupCookie);
|
|
if (pInfo->customGroupCookie == GUID_NULL)
|
|
{
|
|
pInfo->schedule = SUBSSCHED_MANUAL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (PTASK_TRIGGER pInfoTrigger = (PTASK_TRIGGER)pInfo->pTrigger)
|
|
{
|
|
if (pInfoTrigger->cbTriggerSize == pooe->m_Trigger.cbTriggerSize)
|
|
*(pInfoTrigger) = pooe->m_Trigger;
|
|
else
|
|
pInfoTrigger->cbTriggerSize = 0;
|
|
}
|
|
//otherwise, it's already null and we can't do anything about it... luckily, we'll never
|
|
//have a trigger that we need to write back to a SUBSCRIPTIONINFO that didn't already have
|
|
//one.
|
|
|
|
if (dwFlags & SUBSINFO_RECURSE)
|
|
pInfo->dwRecurseLevels = pooe->m_RecurseLevels;
|
|
if (dwFlags & SUBSINFO_WEBCRAWL)
|
|
pInfo->fWebcrawlerFlags = pooe->m_RecurseFlags;
|
|
if (dwFlags & SUBSINFO_MAILNOT)
|
|
pInfo->bMailNotification = pooe->bMail;
|
|
if (dwFlags & SUBSINFO_GLEAM)
|
|
pInfo->bGleam = pooe->bGleam;
|
|
if (dwFlags & SUBSINFO_CHANGESONLY)
|
|
pInfo->bChangesOnly = pooe->bChangesOnly;
|
|
if (dwFlags & SUBSINFO_NEEDPASSWORD)
|
|
pInfo->bNeedPassword = pooe->bNeedPassword;
|
|
if (dwFlags & SUBSINFO_CHANNELFLAGS)
|
|
{
|
|
if ((subType==SUBSTYPE_CHANNEL)||(subType==SUBSTYPE_DESKTOPCHANNEL))
|
|
{
|
|
pInfo->fChannelFlags = pooe->fChannelFlags;
|
|
}
|
|
else
|
|
{
|
|
pInfo->fChannelFlags = 0;
|
|
pInfo->fUpdateFlags &= (~SUBSINFO_CHANNELFLAGS);
|
|
}
|
|
}
|
|
if (dwFlags & SUBSINFO_MAXSIZEKB)
|
|
pInfo->dwMaxSizeKB = pooe->m_SizeLimit;
|
|
|
|
|
|
if (dwFlags & SUBSINFO_TYPE)
|
|
{
|
|
pInfo->subType = GetItemCategory(pooe);
|
|
ASSERT(IS_VALID_SUBSCRIPTIONTYPE(pInfo->subType));
|
|
}
|
|
|
|
if (dwFlags & SUBSINFO_TASKFLAGS)
|
|
{
|
|
pInfo->fTaskFlags = pooe->grfTaskTrigger;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CSubscriptionMgr::UpdateSubscription(LPCWSTR pwszURL)
|
|
{
|
|
ASSERT(pwszURL);
|
|
BOOL bSubscribe = FALSE;
|
|
HRESULT hr = IsSubscribed(pwszURL, &bSubscribe);
|
|
CLSID clsId;
|
|
|
|
RETURN_ON_FAILURE(hr);
|
|
if (!bSubscribe)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
//
|
|
// Fail if restrictions are in place.
|
|
// FEATURE: Should we have a flag parameter to override this?
|
|
//
|
|
if (SHRestricted2W(REST_NoManualUpdates, NULL, 0))
|
|
{
|
|
SGMessageBox(NULL, IDS_RESTRICTED, MB_OK);
|
|
return E_ACCESSDENIED;
|
|
}
|
|
|
|
clsId = _pidl->ooe.m_Cookie;
|
|
hr = SendUpdateRequests(NULL, &clsId, 1);
|
|
_pidl->ooe.m_Cookie = clsId;
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CSubscriptionMgr::UpdateAll()
|
|
{
|
|
//
|
|
// Fail if restrictions are in place.
|
|
// FEATURE: Should we have a flag parameter to override this?
|
|
//
|
|
if (SHRestricted2W(REST_NoManualUpdates, NULL, 0))
|
|
{
|
|
SGMessageBox(NULL, IDS_RESTRICTED, MB_OK);
|
|
return E_ACCESSDENIED;
|
|
}
|
|
|
|
return SendUpdateRequests(NULL, NULL, 0);
|
|
}
|
|
|
|
HRESULT MergeOOEBuf(POOEBuf p1, POOEBuf p2, DWORD fMask)
|
|
{
|
|
ASSERT(p1 && p2);
|
|
DWORD dwMask = p2->dwFlags & fMask;
|
|
|
|
if (dwMask == 0)
|
|
return S_OK;
|
|
|
|
if (p1->clsidDest != p2->clsidDest)
|
|
return E_INVALIDARG;
|
|
|
|
if (dwMask & PROP_WEBCRAWL_COOKIE)
|
|
{
|
|
// We shouldn't merge cookies.
|
|
}
|
|
|
|
if (dwMask & PROP_WEBCRAWL_SIZE)
|
|
{
|
|
p1->m_SizeLimit = p2->m_SizeLimit;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_FLAGS)
|
|
{
|
|
p1->m_RecurseFlags = p2->m_RecurseFlags;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_LEVEL)
|
|
{
|
|
p1->m_RecurseLevels = p2->m_RecurseLevels;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_URL)
|
|
{
|
|
StrCpyN(p1->m_URL, p2->m_URL, ARRAYSIZE(p1->m_URL));
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_NAME)
|
|
{
|
|
StrCpyN(p1->m_Name, p2->m_Name, ARRAYSIZE(p1->m_Name));
|
|
}
|
|
|
|
if (dwMask & PROP_WEBCRAWL_PSWD)
|
|
{
|
|
StrCpyN(p1->password, p2->password, ARRAYSIZE(p1->password));
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_UNAME)
|
|
{
|
|
StrCpyN(p1->username, p2->username, ARRAYSIZE(p1->username));
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_DESKTOP)
|
|
{
|
|
p1->bDesktop = p2->bDesktop;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_CHANNEL)
|
|
{
|
|
p1->bChannel = p2->bChannel;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_EMAILNOTF)
|
|
{
|
|
p1->bMail = p2->bMail;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_RESCH)
|
|
{
|
|
p1->grfTaskTrigger = p2->grfTaskTrigger;
|
|
p1->groupCookie = p2->groupCookie;
|
|
p1->fChannelFlags |= (p2->fChannelFlags & CHANNEL_AGENT_DYNAMIC_SCHEDULE);
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_LAST)
|
|
{
|
|
p1->m_LastUpdated = p2->m_LastUpdated;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_STATUS)
|
|
{
|
|
p1->status = p2->status;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_PRIORITY)
|
|
{
|
|
p1->m_Priority = p2->m_Priority;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_GLEAM)
|
|
{
|
|
p1->bGleam = p2->bGleam;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_CHANGESONLY)
|
|
{
|
|
p1->bChangesOnly = p2->bChangesOnly;
|
|
}
|
|
if (dwMask & PROP_WEBCRAWL_CHANNELFLAGS)
|
|
{
|
|
p1->fChannelFlags = p2->fChannelFlags;
|
|
}
|
|
|
|
p1->dwFlags |= (p2->dwFlags & fMask & (~PROP_WEBCRAWL_COOKIE));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
INT_PTR CALLBACK SummarizeDesktopSubscriptionDlgProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
SUBSCRIBE_ADI_INFO* pInfo = (SUBSCRIBE_ADI_INFO*)GetProp(hDlg,SUBSCRIBEADIPROP);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
pInfo = (SUBSCRIBE_ADI_INFO*)lParam;
|
|
ASSERT (pInfo);
|
|
SetProp (hDlg, SUBSCRIBEADIPROP, (HANDLE)pInfo);
|
|
{ //block to declare vars to update captions
|
|
TCHAR sz[MAX_URL];
|
|
|
|
if (pInfo->subsType == SUBSTYPE_DESKTOPCHANNEL)
|
|
{
|
|
if(MLLoadString(
|
|
(pInfo->pSubsInfo->bNeedPassword ? IDS_DESKTOPCHANNEL_SUMMARY_TEXT : IDS_DESKTOPCHANNEL_SUMMARY_NOPW),
|
|
sz, ARRAYSIZE(sz)))
|
|
{
|
|
SetDlgItemText(hDlg, IDC_DESKTOP_SUMMARY_TEXT, sz);
|
|
}
|
|
}
|
|
|
|
MyOleStrToStrN (sz, ARRAYSIZE(sz), pInfo->pwszName);
|
|
SetListViewToString(GetDlgItem(hDlg, IDC_SUBSCRIBE_ADI_NAME), sz);
|
|
MyOleStrToStrN (sz, ARRAYSIZE(sz), pInfo->pwszUrl);
|
|
SetListViewToString (GetDlgItem (hDlg, IDC_SUBSCRIBE_ADI_URL), sz);
|
|
}
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
ASSERT (pInfo);
|
|
switch (GET_WM_COMMAND_ID(wParam, lParam))
|
|
{
|
|
case IDCANCEL:
|
|
EndDialog(hDlg, IDCANCEL);
|
|
break;
|
|
|
|
case IDOK:
|
|
//subscription happens in calling function when we return IDOK
|
|
EndDialog(hDlg, IDOK);
|
|
break;
|
|
|
|
case IDC_SUBSCRIBE_CUSTOMIZE:
|
|
//run through wizard in NOSAVE mode
|
|
if (pInfo->pMgr &&
|
|
S_OK == pInfo->pMgr->CreateSubscriptionNoSummary (hDlg, pInfo->pwszUrl,
|
|
pInfo->pwszName, pInfo->dwFlags | CREATESUBS_NOSAVE,
|
|
pInfo->subsType, pInfo->pSubsInfo))
|
|
{
|
|
SendMessage (hDlg, WM_NEXTDLGCTL, (WPARAM)GetDlgItem(hDlg, IDOK), TRUE);
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
if (LOWORD(wParam) == IDC_SUBSCRIBE_ADI_URL)
|
|
{
|
|
NM_LISTVIEW * pnmlv = (NM_LISTVIEW *)lParam;
|
|
if (pnmlv->hdr.code == LVN_GETINFOTIP)
|
|
{
|
|
TCHAR szURL[MAX_URL];
|
|
LV_ITEM lvi = {0};
|
|
lvi.mask = LVIF_TEXT;
|
|
lvi.pszText = szURL;
|
|
lvi.cchTextMax = ARRAYSIZE(szURL);
|
|
if (!ListView_GetItem (GetDlgItem (hDlg, IDC_SUBSCRIBE_ADI_URL), &lvi))
|
|
return FALSE;
|
|
|
|
NMLVGETINFOTIP * pTip = (NMLVGETINFOTIP *)pnmlv;
|
|
ASSERT(pTip);
|
|
StrCpyN(pTip->pszText, szURL, pTip->cchTextMax);
|
|
return TRUE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_DESTROY:
|
|
RemoveProp (hDlg, SUBSCRIBEADIPROP);
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|