2554 lines
71 KiB
C++
2554 lines
71 KiB
C++
#include "private.h"
|
|
#include "offl_cpp.h"
|
|
#include "subsmgrp.h"
|
|
#include "helper.h"
|
|
|
|
#include <mluisupp.h>
|
|
|
|
#ifdef DEBUG
|
|
void DumpTaskTrigger(TASK_TRIGGER * pTaskTrigger);
|
|
#endif // DEBUG
|
|
|
|
// {D994B6F0-DA3C-11d1-857D-00C04FA35C89}
|
|
const GUID NOOP_SCHEDULE_COOKIE =
|
|
{ 0xd994b6f0, 0xda3c, 0x11d1, { 0x85, 0x7d, 0x0, 0xc0, 0x4f, 0xa3, 0x5c, 0x89
|
|
} };
|
|
|
|
#ifndef TASK_FLAG_RUN_ONLY_IF_LOGGED_ON
|
|
#define TASK_FLAG_RUN_ONLY_IF_LOGGED_ON (0x2000)
|
|
#endif
|
|
|
|
const PROPSPEC c_rgPropRead[] = {
|
|
{ PRSPEC_PROPID, PID_INTSITE_SUBSCRIPTION},
|
|
{ PRSPEC_PROPID, PID_INTSITE_FLAGS},
|
|
{ PRSPEC_PROPID, PID_INTSITE_TRACKING},
|
|
{ PRSPEC_PROPID, PID_INTSITE_CODEPAGE},
|
|
};
|
|
|
|
void UpdateTimeFormat(LPTSTR tszTimeFormat, ULONG cchTimeFormat);
|
|
|
|
HRESULT WriteProperties(POOEntry pooe);
|
|
HRESULT ReadProperties(POOEBuf pBuf);
|
|
|
|
const TCHAR c_szLoadWC[] = TEXT("LoadWC");
|
|
|
|
void FixupRandomTrigger(TASK_TRIGGER *pTrigger)
|
|
{
|
|
if (pTrigger->wRandomMinutesInterval > 0)
|
|
{
|
|
// We have a random interval so we need to add a random number of minutes to it.
|
|
// Given the fact that all of the fields need to carry over to the next, the
|
|
// simplest way to do this is to flatten the start time into FILETIME, add the
|
|
// random minutes, and then convert back to a TASK_TRIGGER. This let's us use
|
|
// Win32 APIs instead of doing all of the calendar and carry over stuff ourselves.
|
|
|
|
SYSTEMTIME st;
|
|
CFileTime ft;
|
|
|
|
memset(&st, 0, sizeof(SYSTEMTIME));
|
|
st.wYear = pTrigger->wBeginYear;
|
|
st.wMonth = pTrigger->wBeginMonth;
|
|
st.wDay = pTrigger->wBeginDay;
|
|
st.wHour = pTrigger->wStartHour;
|
|
st.wMinute = pTrigger->wStartMinute;
|
|
|
|
SystemTimeToFileTime(&st, &ft);
|
|
|
|
ft += ONE_MINUTE_IN_FILETIME * (__int64)Random(pTrigger->wRandomMinutesInterval);
|
|
|
|
FileTimeToSystemTime(&ft, &st);
|
|
|
|
pTrigger->wBeginYear = st.wYear;
|
|
pTrigger->wBeginMonth = st.wMonth;
|
|
pTrigger->wBeginDay = st.wDay;
|
|
pTrigger->wStartHour = st.wHour;
|
|
pTrigger->wStartMinute = st.wMinute;
|
|
|
|
pTrigger->wRandomMinutesInterval = 0;
|
|
}
|
|
}
|
|
|
|
// Come up with a name like "MSN Recommended Schedule"
|
|
void CreatePublisherScheduleNameW(WCHAR *pwszSchedName, int cchSchedName,
|
|
const TCHAR *pszName, const WCHAR *pwszName)
|
|
{
|
|
WCHAR wszFormat[MAX_PATH];
|
|
WCHAR wszPubName[MAX_PATH];
|
|
const WCHAR *pwszPubName;
|
|
|
|
ASSERT((NULL != pszName) || (NULL != pwszName));
|
|
ASSERT((NULL != pwszSchedName) && (cchSchedName > 0));
|
|
|
|
if (NULL == pwszName)
|
|
{
|
|
ASSERT(NULL != pszName);
|
|
MyStrToOleStrN(wszPubName, ARRAYSIZE(wszPubName), pszName);
|
|
pwszPubName = wszPubName;
|
|
}
|
|
else
|
|
{
|
|
pwszPubName = pwszName;
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
MLLoadStringW(IDS_RECOMMENDED_SCHEDULE_FORMAT, wszFormat, ARRAYSIZE(wszFormat));
|
|
#else
|
|
CHAR szFormat[MAX_PATH];
|
|
MLLoadStringA(IDS_RECOMMENDED_SCHEDULE_FORMAT, szFormat, ARRAYSIZE(szFormat));
|
|
MultiByteToWideChar(CP_ACP, 0, szFormat, -1, wszFormat, ARRAYSIZE(wszFormat));
|
|
#endif
|
|
|
|
wnsprintfW(pwszSchedName, cchSchedName, wszFormat, pwszPubName);
|
|
}
|
|
|
|
void CreatePublisherScheduleName(TCHAR *pszSchedName, int cchSchedName,
|
|
const TCHAR *pszName, const WCHAR *pwszName)
|
|
{
|
|
WCHAR wszSchedName[MAX_PATH];
|
|
|
|
CreatePublisherScheduleNameW(wszSchedName, ARRAYSIZE(wszSchedName),
|
|
pszName, pwszName);
|
|
|
|
MyOleStrToStrN(pszSchedName, cchSchedName, wszSchedName);
|
|
}
|
|
|
|
HICON LoadItemIcon(ISubscriptionItem *psi, BOOL bLarge)
|
|
{
|
|
HICON hIcon = NULL;
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
SUBSCRIPTIONCOOKIE cookie;
|
|
HRESULT hr;
|
|
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
|
|
psi->GetCookie(&cookie);
|
|
|
|
hr = psi->GetSubscriptionItemInfo(&sii);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISubscriptionAgentShellExt *pSubscriptionAgentShellExt;
|
|
|
|
hr = CoCreateInstance(sii.clsidAgent, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionAgentShellExt, (void **)&pSubscriptionAgentShellExt);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pSubscriptionAgentShellExt->Initialize(&cookie, L"", L"", (SUBSCRIPTIONTYPE)-1);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
IExtractIcon *pExtractIcon;
|
|
hr = pSubscriptionAgentShellExt->QueryInterface(IID_IExtractIcon,
|
|
(void **)&pExtractIcon);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
TCHAR szIconFile[INTERNET_MAX_URL_LENGTH];
|
|
int iIndex;
|
|
UINT wFlags;
|
|
HICON hIconScrap = NULL;
|
|
HICON *phIconLarge = bLarge ? &hIcon : &hIconScrap;
|
|
HICON *phIconSmall = bLarge ? &hIconScrap : &hIcon;
|
|
|
|
hr = pExtractIcon->GetIconLocation(0, szIconFile, ARRAYSIZE(szIconFile), &iIndex, &wFlags);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pExtractIcon->Extract(szIconFile, iIndex, phIconLarge, phIconSmall,
|
|
MAKELONG(GetSystemMetrics(SM_CXICON),
|
|
GetSystemMetrics(SM_CXSMICON)));
|
|
|
|
if (S_FALSE == hr)
|
|
{
|
|
hIcon = ExtractIcon(g_hInst, szIconFile, iIndex);
|
|
|
|
if (NULL == hIcon)
|
|
{
|
|
hr = E_FAIL;
|
|
}
|
|
}
|
|
else if ((NULL != hIconScrap) && (hIcon != hIconScrap))
|
|
{
|
|
DestroyIcon(hIconScrap);
|
|
}
|
|
}
|
|
pExtractIcon->Release();
|
|
}
|
|
}
|
|
|
|
pSubscriptionAgentShellExt->Release();
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
DWORD dwChannel = 0;
|
|
DWORD dwDesktop = 0;
|
|
int iSize = bLarge ? GetSystemMetrics(SM_CXICON) : GetSystemMetrics(SM_CXSMICON);
|
|
int id;
|
|
HINSTANCE hinstSrc;
|
|
|
|
ReadDWORD(psi, c_szPropChannel, &dwChannel);
|
|
ReadDWORD(psi, c_szPropDesktopComponent, &dwDesktop);
|
|
|
|
if (dwDesktop == 1)
|
|
{
|
|
id = IDI_DESKTOPITEM;
|
|
hinstSrc = MLGetHinst();
|
|
}
|
|
else if (dwChannel == 1)
|
|
{
|
|
id = IDI_CHANNEL;
|
|
hinstSrc = g_hInst;
|
|
}
|
|
else
|
|
{
|
|
id = IDI_WEBDOC;
|
|
hinstSrc = g_hInst;
|
|
}
|
|
|
|
hIcon = (HICON)LoadImage(hinstSrc, MAKEINTRESOURCE(id), IMAGE_ICON, iSize, iSize, LR_DEFAULTCOLOR);
|
|
}
|
|
|
|
return hIcon;
|
|
}
|
|
|
|
BOOL ScheduleCookieExists(SYNCSCHEDULECOOKIE *pSchedCookie)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
hr = CoInitialize(NULL);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISyncSchedule *pSyncSchedule = NULL;
|
|
|
|
hr = pSyncScheduleMgr->OpenSchedule(pSchedCookie, 0, &pSyncSchedule);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
pSyncSchedule->Release();
|
|
}
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr == S_OK;
|
|
}
|
|
|
|
BOOL IsScheduleNameInUse(TCHAR *pszSchedName)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
CWaitCursor waitCursor;
|
|
ISyncSchedule *pSyncSchedule = NULL;
|
|
SYNCSCHEDULECOOKIE schedCookie = GUID_NULL;
|
|
WCHAR wszSchedName[MAX_PATH];
|
|
|
|
MyStrToOleStrN(wszSchedName, ARRAYSIZE(wszSchedName), pszSchedName);
|
|
|
|
hr = pSyncScheduleMgr->CreateSchedule(wszSchedName, 0,
|
|
&schedCookie, &pSyncSchedule);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
pSyncSchedule->Release();
|
|
}
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr == SYNCMGR_E_NAME_IN_USE;
|
|
}
|
|
|
|
struct CONFLICT_DATA
|
|
{
|
|
TCHAR szSchedName[MAX_PATH];
|
|
TCHAR szFriendlyTrigger[MAX_PATH];
|
|
};
|
|
|
|
INT_PTR CALLBACK SchedConflictDlgProc(HWND hdlg, UINT message, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
|
|
switch (message)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
TCHAR szConflictFormat[MAX_PATH];
|
|
TCHAR szConflictMsg[MAX_PATH * 2];
|
|
CONFLICT_DATA *pcd = (CONFLICT_DATA *)lParam;
|
|
|
|
ASSERT(NULL != pcd);
|
|
|
|
MLLoadString(IDS_SCHED_CONFLICT_FORMAT,
|
|
szConflictFormat, ARRAYSIZE(szConflictFormat));
|
|
|
|
wnsprintf(szConflictMsg, ARRAYSIZE(szConflictMsg),
|
|
szConflictFormat, pcd->szSchedName);
|
|
|
|
SetDlgItemText(hdlg, IDC_SCHEDULE_MESSAGE, szConflictMsg);
|
|
SetDlgItemText(hdlg, IDC_FRIENDLY_SCHEDULE_TEXT, pcd->szFriendlyTrigger);
|
|
|
|
bResult = TRUE;
|
|
break;
|
|
}
|
|
|
|
case WM_COMMAND:
|
|
if (HIWORD(wParam) == BN_CLICKED)
|
|
{
|
|
EndDialog(hdlg, LOWORD(wParam));
|
|
bResult = TRUE;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
BOOL CompareTaskTrigger(TASK_TRIGGER *pTrigA, TASK_TRIGGER *pTrigB)
|
|
{
|
|
BOOL bDontMatch; // TRUE if any elements don't match
|
|
|
|
// Simple memcmp won't work since the start date will be different
|
|
// when in fact they are effectively the same schedule - at least
|
|
// from a user perspective.
|
|
|
|
// BUGBUG - this is far from complete - we only check for values
|
|
// which can be set by our wizard.
|
|
|
|
bDontMatch = pTrigA->wStartHour != pTrigB->wStartHour;
|
|
bDontMatch |= pTrigA->wStartMinute != pTrigB->wStartMinute;
|
|
bDontMatch |= pTrigA->TriggerType != pTrigB->TriggerType;
|
|
bDontMatch |= pTrigA->Type.Daily.DaysInterval != pTrigB->Type.Daily.DaysInterval;
|
|
bDontMatch |= pTrigA->MinutesDuration != pTrigB->MinutesDuration;
|
|
bDontMatch |= pTrigA->MinutesInterval != pTrigB->MinutesInterval;
|
|
bDontMatch |= pTrigA->wRandomMinutesInterval != pTrigB->wRandomMinutesInterval;
|
|
|
|
return !bDontMatch;
|
|
}
|
|
|
|
|
|
// HandleScheduleNameConflict
|
|
|
|
// Return values:
|
|
// CONFLICT_NONE - pSchedCookie will be GUID_NULL and the caller is
|
|
// free to create a new schedule
|
|
// CONFLICT_RESOLVED_USE_NEW - pSchedCookie will be the cookie of an existing
|
|
// schedule which the caller should update with
|
|
// it's new TASK_TRIGGER
|
|
// CONFLICT_RESOLVED_USE_OLD - pSchedCookie will be the cookie of an existing
|
|
// schedule which the caller should use without
|
|
// modifying anything
|
|
// CONFLICT_UNRESOLVED - pSchedCookie will be GUID_NULL and the caller
|
|
// shouldn't do anything until the user has made
|
|
// up his/her mind
|
|
|
|
int HandleScheduleNameConflict(/* in */ TCHAR *pszSchedName,
|
|
/* in */ TASK_TRIGGER *pTrigger,
|
|
/* in */ HWND hwndParent,
|
|
/* out */ SYNCSCHEDULECOOKIE *pSchedCookie)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
int iResult = CONFLICT_NONE;
|
|
|
|
*pSchedCookie = GUID_NULL;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISyncSchedule *pSyncSchedule = NULL;
|
|
SYNCSCHEDULECOOKIE schedCookie = GUID_NULL;
|
|
WCHAR wszSchedName[MAX_PATH];
|
|
|
|
MyStrToOleStrN(wszSchedName, ARRAYSIZE(wszSchedName), pszSchedName);
|
|
|
|
{
|
|
CWaitCursor waitCursor;
|
|
|
|
hr = pSyncScheduleMgr->CreateSchedule(wszSchedName, 0,
|
|
&schedCookie, &pSyncSchedule);
|
|
}
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
pSyncSchedule->Release();
|
|
}
|
|
else if (SYNCMGR_E_NAME_IN_USE == hr)
|
|
{
|
|
// schedCookie will have the cookie of the conflicting schedule
|
|
hr = pSyncScheduleMgr->OpenSchedule(&schedCookie, 0, &pSyncSchedule);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ITaskTrigger *pITaskTrigger;
|
|
|
|
hr = pSyncSchedule->GetTrigger(&pITaskTrigger);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
TASK_TRIGGER existTrigger = { sizeof(TASK_TRIGGER) };
|
|
|
|
hr = pITaskTrigger->GetTrigger(&existTrigger);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (!CompareTaskTrigger(&existTrigger, pTrigger))
|
|
{
|
|
CONFLICT_DATA cd;
|
|
LPWSTR pwszFriendlyTrigger;
|
|
|
|
StrCpyN(cd.szSchedName, pszSchedName, ARRAYSIZE(cd.szSchedName));
|
|
if (SUCCEEDED(pITaskTrigger->GetTriggerString(&pwszFriendlyTrigger)))
|
|
{
|
|
MyOleStrToStrN(cd.szFriendlyTrigger,
|
|
ARRAYSIZE(cd.szFriendlyTrigger),
|
|
pwszFriendlyTrigger);
|
|
CoTaskMemFree(pwszFriendlyTrigger);
|
|
}
|
|
else
|
|
{
|
|
cd.szFriendlyTrigger[0] = TEXT('\0');
|
|
}
|
|
|
|
INT_PTR iRet = DialogBoxParam(MLGetHinst(),
|
|
MAKEINTRESOURCE(IDD_DUPLICATE_SCHEDULE),
|
|
hwndParent,
|
|
SchedConflictDlgProc,
|
|
(LPARAM)&cd);
|
|
switch (iRet)
|
|
{
|
|
case IDC_NEW_SETTINGS:
|
|
iResult = CONFLICT_RESOLVED_USE_NEW;
|
|
*pSchedCookie = schedCookie;
|
|
break;
|
|
|
|
case IDC_OLD_SETTINGS:
|
|
iResult = CONFLICT_RESOLVED_USE_OLD;
|
|
*pSchedCookie = schedCookie;
|
|
break;
|
|
|
|
default:
|
|
iResult = CONFLICT_UNRESOLVED;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
pITaskTrigger->Release();
|
|
}
|
|
pSyncSchedule->Release();
|
|
}
|
|
}
|
|
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return iResult;
|
|
}
|
|
|
|
HRESULT UpdateScheduleTrigger(SYNCSCHEDULECOOKIE *pSchedCookie, TASK_TRIGGER *pTrigger)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
CWaitCursor waitCursor;
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISyncSchedule *pSyncSchedule;
|
|
|
|
hr = pSyncScheduleMgr->OpenSchedule(pSchedCookie, 0, &pSyncSchedule);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ITaskTrigger *pITaskTrigger;
|
|
|
|
hr = pSyncSchedule->GetTrigger(&pITaskTrigger);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
FixupRandomTrigger(pTrigger);
|
|
|
|
hr = pITaskTrigger->SetTrigger(pTrigger);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pSyncSchedule->Save();
|
|
}
|
|
|
|
pITaskTrigger->Release();
|
|
}
|
|
pSyncSchedule->Release();
|
|
}
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT RemoveItemFromAllSchedules(SUBSCRIPTIONCOOKIE *pCookie)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
CWaitCursor waitCursor;
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
IEnumSyncSchedules *pEnumSyncSchedules;
|
|
|
|
hr = pSyncScheduleMgr->EnumSyncSchedules(&pEnumSyncSchedules);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SYNCSCHEDULECOOKIE schedCookie;
|
|
ULONG ulFetched;
|
|
|
|
while (S_OK == pEnumSyncSchedules->Next(1, &schedCookie, &ulFetched) &&
|
|
(0 != ulFetched)) // BUGBUG - this shouldn't be necessary
|
|
{
|
|
ISyncSchedule *pSyncSchedule;
|
|
|
|
// If this fails, there ain't much we can do about
|
|
// it so just plod along anyhow
|
|
|
|
if (SUCCEEDED(pSyncScheduleMgr->OpenSchedule(&schedCookie, 0, &pSyncSchedule)))
|
|
{
|
|
// Don't care about the return value, it's cheaper
|
|
// for us to just delete than to ask if it's there
|
|
// and then delete.
|
|
pSyncSchedule->SetItemCheck(CLSID_WebCheckOfflineSync,
|
|
pCookie,
|
|
SYNCMGRITEMSTATE_UNCHECKED);
|
|
pSyncSchedule->Save();
|
|
pSyncSchedule->Release();
|
|
}
|
|
}
|
|
pEnumSyncSchedules->Release();
|
|
}
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT AddRemoveScheduledItem(SYNC_HANDLER_ITEM_INFO *pSyncHandlerItemInfo, // For Add
|
|
SUBSCRIPTIONCOOKIE *pCookie, // For Remove
|
|
SYNCSCHEDULECOOKIE *pSchedCookie, BOOL bAdd)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
ASSERT((bAdd && (NULL != pSyncHandlerItemInfo)) ||
|
|
(!bAdd && (NULL != pCookie)));
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISyncSchedule *pSyncSchedule;
|
|
|
|
hr = pSyncScheduleMgr->OpenSchedule(pSchedCookie, 0, &pSyncSchedule);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (bAdd)
|
|
{
|
|
|
|
hr = pSyncSchedule->AddItem(pSyncHandlerItemInfo);
|
|
hr = pSyncSchedule->SetItemCheck(CLSID_WebCheckOfflineSync,
|
|
&pSyncHandlerItemInfo->itemID,
|
|
SYNCMGRITEMSTATE_CHECKED);
|
|
}
|
|
else
|
|
{
|
|
|
|
hr = pSyncSchedule->SetItemCheck(CLSID_WebCheckOfflineSync,
|
|
pCookie,
|
|
SYNCMGRITEMSTATE_UNCHECKED);
|
|
}
|
|
hr = pSyncSchedule->Save();
|
|
pSyncSchedule->Release();
|
|
}
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT SetScheduleValues(ISyncSchedule *pSyncSchedule,
|
|
TASK_TRIGGER *pTrigger,
|
|
DWORD dwSyncScheduleFlags)
|
|
{
|
|
HRESULT hr;
|
|
ITaskTrigger *pITaskTrigger;
|
|
|
|
ASSERT(NULL != pSyncSchedule);
|
|
ASSERT(NULL != pTrigger);
|
|
|
|
hr = pSyncSchedule->GetTrigger(&pITaskTrigger);
|
|
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
FixupRandomTrigger(pTrigger);
|
|
hr = pITaskTrigger->SetTrigger(pTrigger);
|
|
pITaskTrigger->Release();
|
|
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
DWORD dwFlags;
|
|
DWORD dwConnectionType = SYNCSCHEDINFO_FLAGS_CONNECTION_LAN;
|
|
WCHAR wszConnectionName[MAX_PATH];
|
|
|
|
// Return code doesn't help us. This returns the best guess
|
|
// at connection:
|
|
// 1) LAN
|
|
// 2) currently connected connectoid
|
|
// 3) auto-dial connectoid
|
|
// This is according to darrenmi, if this changes - kill him.
|
|
InternetGetConnectedStateExW(&dwFlags, wszConnectionName,
|
|
ARRAYSIZE(wszConnectionName), 0);
|
|
|
|
if (dwFlags & INTERNET_CONNECTION_MODEM)
|
|
{
|
|
dwConnectionType = SYNCSCHEDINFO_FLAGS_CONNECTION_WAN;
|
|
}
|
|
|
|
hr = pSyncSchedule->SetConnection(
|
|
(dwConnectionType == SYNCSCHEDINFO_FLAGS_CONNECTION_WAN) ?
|
|
wszConnectionName : NULL,
|
|
dwConnectionType);
|
|
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pSyncSchedule->SetFlags(dwSyncScheduleFlags);
|
|
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pSyncSchedule->Save();
|
|
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CreateSchedule(LPWSTR pwszScheduleName, DWORD dwSyncScheduleFlags,
|
|
SYNCSCHEDULECOOKIE *pSchedCookie, TASK_TRIGGER *pTrigger,
|
|
BOOL fDupCookieOK)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
ISyncSchedule *pSyncSchedule = NULL;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
CWaitCursor waitCursor;
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pSyncScheduleMgr->CreateSchedule(pwszScheduleName, 0,
|
|
pSchedCookie, &pSyncSchedule);
|
|
|
|
ASSERT((FAILED(hr) && (NULL == pSyncSchedule)) ||
|
|
(SUCCEEDED(hr) && (NULL != pSyncSchedule)));
|
|
|
|
switch (hr)
|
|
{
|
|
case S_OK:
|
|
hr = SetScheduleValues(pSyncSchedule, pTrigger, dwSyncScheduleFlags);
|
|
|
|
#ifdef DEBUG
|
|
if (FAILED(hr))
|
|
{
|
|
TraceMsg(TF_ALWAYS, "SetScheduleValues failed - hr=0x%08x", hr);
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
|
|
case HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS):
|
|
if (fDupCookieOK)
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
break;
|
|
}
|
|
|
|
SAFERELEASE(pSyncSchedule);
|
|
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
BOOL IsCookieOnSchedule(ISyncSchedule *pSyncSchedule, SUBSCRIPTIONCOOKIE *pCookie)
|
|
{
|
|
HRESULT hr;
|
|
DWORD dwCheckState = SYNCMGRITEMSTATE_UNCHECKED;
|
|
|
|
hr = pSyncSchedule->GetItemCheck(CLSID_WebCheckOfflineSync,
|
|
pCookie,
|
|
&dwCheckState);
|
|
|
|
return SUCCEEDED(hr) && (SYNCMGRITEMSTATE_CHECKED & dwCheckState);
|
|
}
|
|
|
|
struct GIS_DATA
|
|
{
|
|
SUBSCRIPTIONCOOKIE *pSubsCookie;
|
|
SYNCSCHEDULECOOKIE *pSchedCookie;
|
|
};
|
|
|
|
BOOL GetItemScheduleCallback(ISyncSchedule *pSyncSchedule,
|
|
SYNCSCHEDULECOOKIE *pSchedCookie,
|
|
LPARAM lParam)
|
|
{
|
|
BOOL bContinue = TRUE;
|
|
GIS_DATA *pgd = (GIS_DATA *)lParam;
|
|
|
|
if (IsCookieOnSchedule(pSyncSchedule, pgd->pSubsCookie))
|
|
{
|
|
*pgd->pSchedCookie = *pSchedCookie;
|
|
bContinue = FALSE;
|
|
}
|
|
|
|
return bContinue;
|
|
}
|
|
|
|
|
|
HRESULT GetItemSchedule(SUBSCRIPTIONCOOKIE *pSubsCookie, SYNCSCHEDULECOOKIE *pSchedCookie)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
*pSchedCookie = GUID_NULL;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISubscriptionItem *psi;
|
|
|
|
// First let's chech to see if it has a custom schedule
|
|
hr = SubscriptionItemFromCookie(FALSE, pSubsCookie, &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
|
|
hr = psi->GetSubscriptionItemInfo(&sii);
|
|
|
|
psi->Release();
|
|
|
|
if (SUCCEEDED(hr) && (sii.ScheduleGroup != GUID_NULL))
|
|
{
|
|
*pSchedCookie = sii.ScheduleGroup;
|
|
}
|
|
else
|
|
{
|
|
GIS_DATA gd;
|
|
|
|
gd.pSubsCookie = pSubsCookie;
|
|
gd.pSchedCookie = pSchedCookie;
|
|
EnumSchedules(GetItemScheduleCallback, (LPARAM)&gd);
|
|
}
|
|
}
|
|
|
|
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT EnumSchedules(SCHEDULEENUMCALLBACK pCallback, LPARAM lParam)
|
|
{
|
|
HRESULT hr;
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
CWaitCursor waitCursor;
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
IEnumSyncSchedules *pEnumSyncSchedules;
|
|
|
|
hr = pSyncScheduleMgr->EnumSyncSchedules(&pEnumSyncSchedules);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SYNCSCHEDULECOOKIE schedCookie;
|
|
ULONG ulFetched;
|
|
|
|
while (S_OK == pEnumSyncSchedules->Next(1, &schedCookie, &ulFetched)&&
|
|
(0 != ulFetched)) // BUGBUG - this shouldn't be necessary
|
|
{
|
|
ISyncSchedule *pSyncSchedule;
|
|
|
|
HRESULT hrTemp = pSyncScheduleMgr->OpenSchedule(&schedCookie, 0, &pSyncSchedule);
|
|
if (SUCCEEDED(hrTemp) && pSyncSchedule)
|
|
{
|
|
BOOL bContinue = pCallback(pSyncSchedule, &schedCookie, lParam);
|
|
pSyncSchedule->Release();
|
|
|
|
if (!bContinue)
|
|
{
|
|
hr = S_FALSE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
pEnumSyncSchedules->Release();
|
|
}
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
SUBSCRIPTIONTYPE GetItemCategory(BOOL bDesktop, const CLSID& clsidDest)
|
|
{
|
|
SUBSCRIPTIONTYPE st;
|
|
|
|
if (clsidDest == CLSID_WebCrawlerAgent)
|
|
{
|
|
st = (!bDesktop) ? SUBSTYPE_URL : SUBSTYPE_DESKTOPURL;
|
|
}
|
|
else if (clsidDest == CLSID_ChannelAgent)
|
|
{
|
|
st = (!bDesktop) ? SUBSTYPE_CHANNEL : SUBSTYPE_DESKTOPCHANNEL;
|
|
}
|
|
else
|
|
{
|
|
st = SUBSTYPE_EXTERNAL;
|
|
}
|
|
|
|
return st;
|
|
}
|
|
|
|
|
|
SUBSCRIPTIONSCHEDULE GetGroup(BOOL bDesktop, const CLSID& clsidDest,
|
|
DWORD fChannelFlags, const NOTIFICATIONCOOKIE& groupCookie)
|
|
{
|
|
|
|
|
|
SUBSCRIPTIONTYPE category = GetItemCategory(bDesktop, clsidDest);
|
|
if (category == SUBSTYPE_CHANNEL || category == SUBSTYPE_DESKTOPCHANNEL) {
|
|
if ((fChannelFlags & CHANNEL_AGENT_DYNAMIC_SCHEDULE) &&
|
|
(GUID_NULL == groupCookie))
|
|
return SUBSSCHED_AUTO;
|
|
}
|
|
|
|
// We have no idea about the AUTO schedule stuff of unknown types.
|
|
|
|
if (groupCookie == NOTFCOOKIE_SCHEDULE_GROUP_DAILY)
|
|
return SUBSSCHED_DAILY;
|
|
else if (groupCookie == NOTFCOOKIE_SCHEDULE_GROUP_WEEKLY)
|
|
return SUBSSCHED_WEEKLY;
|
|
else if (groupCookie == NOTFCOOKIE_SCHEDULE_GROUP_MANUAL)
|
|
return SUBSSCHED_MANUAL;
|
|
else
|
|
return SUBSSCHED_CUSTOM;
|
|
}
|
|
|
|
HRESULT LoadGroupCookie(NOTIFICATIONCOOKIE * pCookie, SUBSCRIPTIONSCHEDULE subGroup)
|
|
{
|
|
if (pCookie) {
|
|
switch (subGroup) {
|
|
case SUBSSCHED_DAILY:
|
|
*pCookie = NOTFCOOKIE_SCHEDULE_GROUP_DAILY;
|
|
break;
|
|
case SUBSSCHED_WEEKLY:
|
|
*pCookie = NOTFCOOKIE_SCHEDULE_GROUP_WEEKLY;
|
|
break;
|
|
case SUBSSCHED_MANUAL:
|
|
*pCookie = NOTFCOOKIE_SCHEDULE_GROUP_MANUAL;
|
|
break;
|
|
default:
|
|
*pCookie = CLSID_NULL;
|
|
ASSERT(0);
|
|
break;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
|
|
|
|
// Subscription helper functions.
|
|
|
|
|
|
HRESULT TSTR2BSTR(VARIANT * pvarBSTR, LPCTSTR srcTSTR)
|
|
{
|
|
ASSERT(pvarBSTR);
|
|
ASSERT(srcTSTR);
|
|
|
|
BSTR bstrBuf = NULL;
|
|
LONG lTSTRLen = 0;
|
|
|
|
lTSTRLen = lstrlen(srcTSTR) + 1;
|
|
bstrBuf = SysAllocStringLen(NULL, lTSTRLen);
|
|
if (!bstrBuf)
|
|
return E_OUTOFMEMORY;
|
|
MyStrToOleStrN(bstrBuf, lTSTRLen, srcTSTR);
|
|
pvarBSTR->vt = VT_BSTR;
|
|
pvarBSTR->bstrVal = bstrBuf;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WriteCookieToInetDB(LPCTSTR pszURL, SUBSCRIPTIONCOOKIE *pCookie, BOOL bRemove)
|
|
{
|
|
PROPVARIANT propCookie;
|
|
LPOLESTR pclsid = NULL; // init to keep compiler happy
|
|
|
|
ASSERT(pszURL);
|
|
if (bRemove)
|
|
{
|
|
propCookie.vt = VT_EMPTY;
|
|
}
|
|
else
|
|
{
|
|
ASSERT(pCookie);
|
|
|
|
if (FAILED(StringFromCLSID(*pCookie, &pclsid)))
|
|
return E_FAIL;
|
|
|
|
propCookie.vt = VT_LPWSTR;
|
|
propCookie.pwszVal = pclsid;
|
|
}
|
|
|
|
HRESULT hr = IntSiteHelper(pszURL, &c_rgPropRead[PROP_SUBSCRIPTION], &propCookie, 1, TRUE);
|
|
|
|
if (!bRemove)
|
|
CoTaskMemFree(pclsid);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT WritePropertiesToItem(POOEntry pooe, ISubscriptionItem *psi)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
VARIANT var;
|
|
BOOL bHasUNAME = TRUE;
|
|
|
|
ASSERT(NULL != psi);
|
|
|
|
VariantInit(&var);
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_URL)
|
|
{
|
|
if (FAILED(TSTR2BSTR(&var, URL(pooe))))
|
|
return E_FAIL;
|
|
WriteVariant(psi, c_szPropURL, &var);
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_NAME)
|
|
{
|
|
if (FAILED(TSTR2BSTR(&var, NAME(pooe))))
|
|
return E_FAIL;
|
|
WriteVariant(psi, c_szPropName, &var);
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_UNAME)
|
|
{
|
|
if(*(UNAME(pooe)))
|
|
{
|
|
if (FAILED(TSTR2BSTR(&var, UNAME(pooe))))
|
|
return E_FAIL;
|
|
WriteVariant(psi, c_szPropCrawlUsername, &var);
|
|
VariantClear(&var);
|
|
}
|
|
else
|
|
{
|
|
WriteEMPTY(psi, c_szPropCrawlUsername);
|
|
bHasUNAME = FALSE;
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_DESKTOP)
|
|
{
|
|
if (pooe->bDesktop)
|
|
{
|
|
WriteDWORD(psi, c_szPropDesktopComponent, 1);
|
|
}
|
|
else
|
|
{
|
|
WriteEMPTY(psi, c_szPropDesktopComponent);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_CHANNEL)
|
|
{
|
|
if (pooe->bChannel)
|
|
{
|
|
WriteDWORD(psi, c_szPropChannel, 1);
|
|
}
|
|
else
|
|
{
|
|
WriteEMPTY(psi, c_szPropChannel);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_GLEAM)
|
|
{
|
|
if (pooe->bGleam)
|
|
{
|
|
WriteDWORD(psi, c_szPropEnableShortcutGleam, 1);
|
|
}
|
|
else
|
|
{
|
|
WriteEMPTY(psi, c_szPropEnableShortcutGleam);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_CHANGESONLY)
|
|
{
|
|
if (pooe->bChangesOnly)
|
|
{
|
|
WriteDWORD(psi, c_szPropCrawlChangesOnly, 1);
|
|
}
|
|
else
|
|
{
|
|
WriteEMPTY(psi, c_szPropCrawlChangesOnly);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_CHANNELFLAGS)
|
|
{
|
|
if (pooe->fChannelFlags)
|
|
{
|
|
WriteDWORD(psi, c_szPropChannelFlags, pooe->fChannelFlags);
|
|
}
|
|
else
|
|
{
|
|
WriteEMPTY(psi, c_szPropChannelFlags);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_EMAILNOTF)
|
|
{
|
|
if (pooe->bMail)
|
|
{
|
|
WriteDWORD(psi, c_szPropEmailNotf, 1);
|
|
}
|
|
else
|
|
{
|
|
WriteEMPTY(psi, c_szPropEmailNotf);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_PSWD)
|
|
{
|
|
if (*(PASSWD(pooe)) && bHasUNAME)
|
|
{
|
|
if (FAILED(TSTR2BSTR(&var, PASSWD(pooe))))
|
|
return E_FAIL;
|
|
WritePassword(psi, var.bstrVal);
|
|
VariantClear(&var);
|
|
}
|
|
else
|
|
{
|
|
WritePassword(psi, NULL);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_LEVEL)
|
|
{
|
|
if(pooe->m_RecurseLevels)
|
|
{
|
|
WriteDWORD(psi, c_szPropCrawlLevels, pooe->m_RecurseLevels);
|
|
}
|
|
else
|
|
{
|
|
// top page only was specified, empty out levels
|
|
WriteEMPTY(psi, c_szPropCrawlLevels);
|
|
}
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_FLAGS)
|
|
{
|
|
WriteDWORD(psi, c_szPropCrawlFlags, pooe->m_RecurseFlags);
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_ACTUALSIZE)
|
|
{
|
|
WriteDWORD(psi, c_szPropCrawlActualSize, pooe->m_ActualSize);
|
|
}
|
|
|
|
if (pooe->dwFlags & PROP_WEBCRAWL_SIZE)
|
|
{
|
|
if(pooe->m_SizeLimit)
|
|
{
|
|
// limit was specified
|
|
WriteDWORD(psi, c_szPropCrawlMaxSize, pooe->m_SizeLimit);
|
|
}
|
|
else
|
|
{
|
|
// no limit was specified, empty out limit prop
|
|
WriteEMPTY(psi, c_szPropCrawlMaxSize);
|
|
}
|
|
}
|
|
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
if (SUCCEEDED(psi->GetSubscriptionItemInfo(&sii)))
|
|
{
|
|
sii.dwFlags = pooe->grfTaskTrigger;
|
|
psi->SetSubscriptionItemInfo(&sii);
|
|
}
|
|
|
|
// We don't write Status/Last update.
|
|
|
|
// BUGBUG: We should fail the subscription if we were unable to write
|
|
// any of the properties into the notification for some reason.
|
|
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT WriteProperties(POOEntry pooe)
|
|
{
|
|
HRESULT hr;
|
|
ISubscriptionItem *psi = NULL;
|
|
|
|
ASSERT(NULL != pooe);
|
|
|
|
hr = SubscriptionItemFromCookie(FALSE, &pooe->m_Cookie, &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = WritePropertiesToItem(pooe, psi);
|
|
psi->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CreatePublisherSchedule()
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
#define RANDOM_TIME_START 0 // 12am (in minutes)
|
|
#define RANDOM_TIME_END 300 // 5am (in minutes)
|
|
#define RANDOM_TIME_INC 30 // 30min increment
|
|
|
|
DWORD GetRandomTime(DWORD StartMins, DWORD EndMins, DWORD Inc)
|
|
{
|
|
DWORD Range;
|
|
DWORD nIncrements;
|
|
|
|
if (StartMins > EndMins)
|
|
{
|
|
Range = ((1440 - StartMins) + EndMins);
|
|
}
|
|
else
|
|
{
|
|
Range = (EndMins - StartMins);
|
|
}
|
|
|
|
nIncrements = ((Range / Inc) + 1);
|
|
|
|
return (StartMins + (Random(nIncrements) * Inc));
|
|
}
|
|
|
|
HRESULT CreateDefaultSchedule(SUBSCRIPTIONSCHEDULE subsSchedule,
|
|
SYNCSCHEDULECOOKIE *pSchedCookie)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
TASK_TRIGGER trig;
|
|
int resID;
|
|
|
|
*pSchedCookie = GUID_NULL;
|
|
|
|
switch (subsSchedule)
|
|
{
|
|
case SUBSSCHED_DAILY:
|
|
trig.TriggerType = TASK_TIME_TRIGGER_DAILY;
|
|
trig.Type.Daily.DaysInterval = 1;
|
|
resID = IDS_DAILY_GRO;
|
|
*pSchedCookie = NOTFCOOKIE_SCHEDULE_GROUP_DAILY;
|
|
break;
|
|
|
|
case SUBSSCHED_WEEKLY:
|
|
trig.TriggerType = TASK_TIME_TRIGGER_WEEKLY;
|
|
trig.Type.Weekly.WeeksInterval = 1;
|
|
trig.Type.Weekly.rgfDaysOfTheWeek = TASK_MONDAY;
|
|
resID = IDS_WEEKLY_GRO;
|
|
*pSchedCookie = NOTFCOOKIE_SCHEDULE_GROUP_WEEKLY;
|
|
break;
|
|
|
|
case SUBSSCHED_AUTO:
|
|
case SUBSSCHED_CUSTOM:
|
|
case SUBSSCHED_MANUAL:
|
|
default:
|
|
resID = 0;
|
|
hr = E_FAIL;
|
|
break;
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (!ScheduleCookieExists(pSchedCookie))
|
|
{
|
|
WCHAR wszSchedName[MAX_PATH];
|
|
DWORD dwRandTime = GetRandomTime(RANDOM_TIME_START,
|
|
RANDOM_TIME_END,
|
|
RANDOM_TIME_INC);
|
|
|
|
trig.cbTriggerSize = sizeof(TASK_TRIGGER);
|
|
trig.wRandomMinutesInterval = RANDOM_TIME_INC;
|
|
trig.wStartHour = (UINT)(dwRandTime / 60);
|
|
trig.wStartMinute = (UINT)(dwRandTime % 60);
|
|
trig.rgFlags = 0;
|
|
|
|
MLLoadStringW(resID, wszSchedName, ARRAYSIZE(wszSchedName));
|
|
|
|
hr = CreateSchedule(wszSchedName, 0, pSchedCookie, &trig, TRUE);
|
|
|
|
if (hr == SYNCMGR_E_NAME_IN_USE)
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT AddIt(ISubscriptionItem *psi, POOEntry pooe, SUBSCRIPTIONSCHEDULE subGroup)
|
|
{
|
|
SYNCSCHEDULECOOKIE schedCookie = GUID_NULL;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
switch (subGroup)
|
|
{
|
|
case SUBSSCHED_DAILY:
|
|
case SUBSSCHED_WEEKLY:
|
|
hr = CreateDefaultSchedule(subGroup, &schedCookie);
|
|
break;
|
|
|
|
case SUBSSCHED_CUSTOM:
|
|
schedCookie = pooe->groupCookie;
|
|
hr = S_OK;
|
|
break;
|
|
|
|
case SUBSSCHED_MANUAL:
|
|
RemoveItemFromAllSchedules(&pooe->m_Cookie);
|
|
hr = S_FALSE;
|
|
break;
|
|
|
|
case SUBSSCHED_AUTO:
|
|
// BUGBUG - for now, until pub schedules are wired in
|
|
hr = CreateDefaultSchedule(SUBSSCHED_DAILY, &schedCookie);
|
|
break;
|
|
}
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
ASSERT(GUID_NULL != schedCookie);
|
|
|
|
if (NOOP_SCHEDULE_COOKIE == schedCookie)
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
if (GUID_NULL != schedCookie)
|
|
{
|
|
SYNC_HANDLER_ITEM_INFO shii;
|
|
|
|
shii.handlerID = CLSID_WebCheckOfflineSync;
|
|
shii.itemID = pooe->m_Cookie;
|
|
shii.hIcon = NULL;
|
|
MyStrToOleStrN(shii.wszItemName, ARRAYSIZE(shii.wszItemName), NAME(pooe));
|
|
shii.dwCheckState = SYNCMGRITEMSTATE_CHECKED;
|
|
|
|
hr = AddScheduledItem(&shii, &schedCookie);
|
|
}
|
|
else
|
|
{
|
|
hr = E_FAIL;
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
// return hr;
|
|
}
|
|
|
|
HRESULT ScheduleIt(ISubscriptionItem *psi, TCHAR *pszName, TASK_TRIGGER *pTrigger)
|
|
{
|
|
HRESULT hr;
|
|
SUBSCRIPTIONITEMINFO subscriptionItemInfo;
|
|
|
|
ASSERT(pTrigger->cbTriggerSize == sizeof(TASK_TRIGGER));
|
|
|
|
subscriptionItemInfo.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
|
|
#ifdef DEBUG
|
|
DumpTaskTrigger(pTrigger);
|
|
#endif
|
|
|
|
hr = psi->GetSubscriptionItemInfo(&subscriptionItemInfo);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (GUID_NULL != subscriptionItemInfo.ScheduleGroup)
|
|
{
|
|
hr = UpdateScheduleTrigger(&subscriptionItemInfo.ScheduleGroup, pTrigger);
|
|
}
|
|
else
|
|
{
|
|
hr = E_FAIL;
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
WCHAR wszSchedName[MAX_PATH];
|
|
|
|
CreatePublisherScheduleNameW(wszSchedName, ARRAYSIZE(wszSchedName),
|
|
pszName, NULL);
|
|
|
|
hr = CreateSchedule(wszSchedName, SYNCSCHEDINFO_FLAGS_READONLY,
|
|
&subscriptionItemInfo.ScheduleGroup, pTrigger, TRUE);
|
|
|
|
if (SUCCEEDED(hr) || (hr == SYNCMGR_E_NAME_IN_USE))
|
|
{
|
|
psi->SetSubscriptionItemInfo(&subscriptionItemInfo);
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
TraceMsg(TF_ALWAYS, "Error creating schedule - hr=0x%08x", hr);
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SYNC_HANDLER_ITEM_INFO shii;
|
|
|
|
shii.handlerID = CLSID_WebCheckOfflineSync;
|
|
psi->GetCookie(&shii.itemID);
|
|
shii.hIcon = NULL;
|
|
MyStrToOleStrN(shii.wszItemName, ARRAYSIZE(shii.wszItemName), pszName);
|
|
|
|
hr = AddScheduledItem(&shii, &subscriptionItemInfo.ScheduleGroup);
|
|
}
|
|
|
|
return S_OK;
|
|
// return hr;
|
|
}
|
|
|
|
HRESULT ReadProperties(POOEBuf pBuf)
|
|
{
|
|
VARIANT var;
|
|
HRESULT hr;
|
|
ASSERT(pBuf);
|
|
BOOL bHasUNAME = TRUE;
|
|
ISubscriptionItem *psi = NULL;
|
|
|
|
ASSERT(NULL != pBuf);
|
|
|
|
hr = SubscriptionItemFromCookie(FALSE, &pBuf->m_Cookie, &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
VariantInit(&var);
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_URL)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropURL, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_BSTR))
|
|
{
|
|
MyOleStrToStrN(pBuf->m_URL, MAX_URL, var.bstrVal);
|
|
}
|
|
else
|
|
{
|
|
pBuf->m_URL[0] = (TCHAR)0;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_NAME)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropName, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_BSTR))
|
|
{
|
|
MyOleStrToStrN(pBuf->m_Name, MAX_NAME, var.bstrVal);
|
|
}
|
|
else
|
|
{
|
|
pBuf->m_Name[0] = (TCHAR)0;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_UNAME)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropCrawlUsername, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_BSTR))
|
|
{
|
|
MyOleStrToStrN(pBuf->username, MAX_USERNAME, var.bstrVal);
|
|
}
|
|
else
|
|
{
|
|
pBuf->username[0] = (TCHAR)0;
|
|
bHasUNAME = FALSE;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_DESKTOP)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropDesktopComponent, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_I4) && (var.lVal == 1))
|
|
{
|
|
pBuf->bDesktop = TRUE;
|
|
}
|
|
else
|
|
{
|
|
pBuf->bDesktop = FALSE;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_CHANNEL)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropChannel, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_I4) && (var.lVal == 1))
|
|
{
|
|
pBuf->bChannel = TRUE;
|
|
}
|
|
else
|
|
{
|
|
pBuf->bChannel = FALSE;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_GLEAM)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropEnableShortcutGleam, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_I4) && (var.lVal == 1))
|
|
{
|
|
pBuf->bGleam = TRUE;
|
|
}
|
|
else
|
|
{
|
|
pBuf->bGleam = FALSE;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_CHANGESONLY)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropCrawlChangesOnly, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_I4) && (var.lVal == 1))
|
|
{
|
|
pBuf->bChangesOnly = TRUE;
|
|
}
|
|
else
|
|
{
|
|
pBuf->bChangesOnly = FALSE;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_CHANNELFLAGS)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropChannelFlags, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_I4))
|
|
{
|
|
pBuf->fChannelFlags = var.lVal;
|
|
}
|
|
else
|
|
{
|
|
pBuf->fChannelFlags = 0;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_EMAILNOTF)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropEmailNotf, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_I4) && (var.lVal == 1))
|
|
{
|
|
pBuf->bMail = TRUE;
|
|
}
|
|
else
|
|
{
|
|
pBuf->bMail = FALSE;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
if ((pBuf->dwFlags & PROP_WEBCRAWL_PSWD) && bHasUNAME)
|
|
{
|
|
BSTR bstrVal = NULL;
|
|
hr = ReadPassword(psi, &bstrVal);
|
|
if (SUCCEEDED(hr) && bstrVal)
|
|
{
|
|
MyOleStrToStrN(pBuf->password, MAX_PASSWORD, bstrVal);
|
|
}
|
|
else
|
|
{
|
|
pBuf->password[0] = (TCHAR)0;
|
|
}
|
|
SAFEFREEBSTR(bstrVal);
|
|
}
|
|
|
|
if ((pBuf->dwFlags & PROP_WEBCRAWL_PSWD) || (pBuf->dwFlags & PROP_WEBCRAWL_UNAME)) {
|
|
//bNeedPassword isn't stored in the property map... calculate it from the presence
|
|
//of username/password.
|
|
pBuf->bNeedPassword = pBuf->password[0] || pBuf->username[0];
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_STATUS)
|
|
{
|
|
hr = ReadVariant(psi, c_szPropStatusString, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_BSTR))
|
|
{
|
|
MyOleStrToStrN(pBuf->statusStr, MAX_STATUS, var.bstrVal);
|
|
}
|
|
else
|
|
{
|
|
pBuf->statusStr[0] = (TCHAR)0;
|
|
}
|
|
VariantClear(&var);
|
|
|
|
hr = ReadSCODE(psi, c_szPropStatusCode, &(pBuf->status));
|
|
// BUGBUG! What should we put here if we don't have last status?
|
|
if (FAILED(hr))
|
|
{
|
|
pBuf->status = S_OK;
|
|
}
|
|
VariantClear(&var);
|
|
}
|
|
|
|
|
|
// Use the CompletionTime property if present and it is greater than
|
|
// value in the NOTIFICATIONITEM structure.
|
|
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_LAST)
|
|
{
|
|
CFileTime ft;
|
|
hr = ReadVariant(psi, c_szPropCompletionTime, &var);
|
|
if (SUCCEEDED(hr) && (var.vt == VT_DATE))
|
|
{
|
|
VariantTimeToFileTime(var.date, ft);
|
|
|
|
if (ft > pBuf->m_LastUpdated)
|
|
{
|
|
pBuf->m_LastUpdated = ft;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_LEVEL)
|
|
{
|
|
hr = ReadDWORD(psi, c_szPropCrawlLevels, (DWORD *)&(pBuf->m_RecurseLevels));
|
|
if (FAILED(hr))
|
|
{
|
|
pBuf->m_RecurseLevels = 0;
|
|
}
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_FLAGS)
|
|
{
|
|
hr = ReadDWORD(psi, c_szPropCrawlFlags, (DWORD *)&(pBuf->m_RecurseFlags));
|
|
if (FAILED(hr))
|
|
{
|
|
pBuf->m_RecurseFlags = 0; // Minimal memory usage.
|
|
}
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_SIZE)
|
|
{
|
|
hr = ReadDWORD(psi, c_szPropCrawlMaxSize, (DWORD *)&(pBuf->m_SizeLimit));
|
|
if (FAILED(hr))
|
|
{
|
|
pBuf->m_SizeLimit = 0;
|
|
}
|
|
}
|
|
|
|
if (pBuf->dwFlags & PROP_WEBCRAWL_ACTUALSIZE)
|
|
{
|
|
hr = ReadDWORD(psi, c_szPropCrawlActualSize, (DWORD *)&(pBuf->m_ActualSize));
|
|
if (FAILED(hr))
|
|
{
|
|
pBuf->m_ActualSize = 0;
|
|
}
|
|
}
|
|
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
if (SUCCEEDED(psi->GetSubscriptionItemInfo(&sii)))
|
|
{
|
|
pBuf->grfTaskTrigger = sii.dwFlags;
|
|
}
|
|
|
|
psi->Release();
|
|
}
|
|
|
|
// BUGBUG: Need to support c_szPropEnableShortcutGleam here.
|
|
// BUGBUG: Need to support c_szPropEnableShortcutGleam here.
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
// If url is NULL, no need to compare.
|
|
HRESULT LoadWithCookie(LPCTSTR pszURL, POOEBuf pBuf, DWORD *pdwBufferSize, SUBSCRIPTIONCOOKIE *pCookie)
|
|
{
|
|
HRESULT hr = LoadOOEntryInfo(pBuf, pCookie, pdwBufferSize);
|
|
|
|
if (SUCCEEDED(hr) && pszURL)
|
|
{
|
|
if (UrlCompare(pBuf->m_URL, pszURL, TRUE))
|
|
{
|
|
TraceMsg(TF_ALWAYS, "Mismatched cookie/URL in LoadWithCookie");
|
|
hr = E_FAIL; // Mismatched cookie!
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT ReadCookieFromInetDB(LPCTSTR pszURL, SUBSCRIPTIONCOOKIE *pCookie)
|
|
{
|
|
ASSERT(pszURL && pCookie);
|
|
PROPVARIANT propCookie;
|
|
|
|
PropVariantInit(&propCookie);
|
|
|
|
HRESULT hr = FindURLProps(pszURL, &propCookie);
|
|
|
|
if (SUCCEEDED(hr) && (propCookie.vt == VT_LPWSTR))
|
|
{
|
|
hr = CLSIDFromString(propCookie.pwszVal, pCookie);
|
|
}
|
|
PropVariantClear(&propCookie);
|
|
|
|
// If we couldn't find it, use a brute force approach
|
|
if (S_OK != hr)
|
|
{
|
|
CEnumSubscription *pes = new CEnumSubscription;
|
|
|
|
if (NULL != pes)
|
|
{
|
|
if (SUCCEEDED(pes->Initialize(0)))
|
|
{
|
|
SUBSCRIPTIONCOOKIE cookie;
|
|
BOOL bFound = FALSE;
|
|
|
|
while (!bFound && (S_OK == pes->Next(1, &cookie, NULL)))
|
|
{
|
|
ISubscriptionItem *psi;
|
|
|
|
if (SUCCEEDED(SubscriptionItemFromCookie(FALSE, &cookie, &psi)))
|
|
{
|
|
LPTSTR pszCurURL;
|
|
|
|
if (SUCCEEDED(ReadTSTR(psi, c_szPropURL, &pszCurURL)))
|
|
{
|
|
bFound = (StrCmpI(pszCurURL, pszURL) == 0);
|
|
CoTaskMemFree(pszCurURL);
|
|
}
|
|
psi->Release();
|
|
}
|
|
}
|
|
|
|
if (bFound)
|
|
{
|
|
WriteCookieToInetDB(pszURL, &cookie, FALSE);
|
|
*pCookie = cookie;
|
|
hr = S_OK;
|
|
}
|
|
}
|
|
pes->Release();
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT LoadSubscription(LPCTSTR url, LPMYPIDL *ppidl)
|
|
{
|
|
HRESULT hr;
|
|
|
|
POOEntry pooe = NULL;
|
|
OOEBuf ooeBuf;
|
|
DWORD dwBufferSize;
|
|
SUBSCRIPTIONCOOKIE cookie;
|
|
|
|
hr = ReadCookieFromInetDB(url, &cookie);
|
|
if (S_OK == hr)
|
|
{
|
|
hr = LoadWithCookie(url, &ooeBuf, &dwBufferSize, &cookie);
|
|
if (hr == S_OK)
|
|
{
|
|
*ppidl = COfflineFolderEnum::NewPidl(dwBufferSize);
|
|
if (!(*ppidl))
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
pooe = &((*ppidl)->ooe);
|
|
CopyToMyPooe(&ooeBuf, pooe);
|
|
}
|
|
else
|
|
{
|
|
WriteCookieToInetDB(url, NULL, TRUE);
|
|
hr = E_FAIL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
// KENSY: This needs to work like GetDefaultInfo
|
|
|
|
HRESULT GetDefaultOOEBuf(OOEBuf * pBuf, SUBSCRIPTIONTYPE subType)
|
|
{
|
|
ASSERT(pBuf);
|
|
ASSERT(IS_VALID_SUBSCRIPTIONTYPE(subType));
|
|
|
|
memset((void *)pBuf, 0, sizeof(OOEBuf));
|
|
pBuf->dwFlags = PROP_WEBCRAWL_ALL;
|
|
pBuf->m_RecurseLevels = DEFAULTLEVEL;
|
|
pBuf->m_RecurseFlags = DEFAULTFLAGS;
|
|
pBuf->m_Priority = AGENT_PRIORITY_NORMAL;
|
|
if (subType == SUBSTYPE_CHANNEL || subType == SUBSTYPE_DESKTOPCHANNEL)
|
|
{
|
|
pBuf->clsidDest = CLSID_ChannelAgent;
|
|
pBuf->fChannelFlags = CHANNEL_AGENT_PRECACHE_ALL | CHANNEL_AGENT_DYNAMIC_SCHEDULE;
|
|
}
|
|
else
|
|
{
|
|
pBuf->clsidDest = CLSID_WebCrawlerAgent;
|
|
}
|
|
pBuf->bDesktop = (subType == SUBSTYPE_DESKTOPCHANNEL || subType == SUBSTYPE_DESKTOPURL);
|
|
pBuf->bChannel = (subType == SUBSTYPE_CHANNEL || subType == SUBSTYPE_DESKTOPCHANNEL);
|
|
pBuf->bGleam = !(pBuf->bDesktop);
|
|
pBuf->m_LastUpdated = 0;
|
|
pBuf->m_NextUpdate = 0;
|
|
|
|
// BUGBUG: Is this what we want? IE 4 was DAILY.
|
|
// Default to not changing the schedule settings -- if it's already subscribed
|
|
// we won't blast anything and if it's not already subscribed then it will
|
|
// just be manual.
|
|
pBuf->groupCookie = NOOP_SCHEDULE_COOKIE;
|
|
|
|
pBuf->grfTaskTrigger = TASK_FLAG_RUN_IF_CONNECTED_TO_INTERNET | // Default to not autodial
|
|
TASK_FLAG_START_ONLY_IF_IDLE; // and to idle time
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CreateSubscriptionFromOOEBuf(OOEBuf *pBuf, LPMYPIDL *ppidl)
|
|
{
|
|
HRESULT hr;
|
|
DWORD dwBufferSize = BufferSize(pBuf);
|
|
|
|
*ppidl = COfflineFolderEnum::NewPidl(dwBufferSize);
|
|
if (!(*ppidl))
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
POOEntry pooe = &((*ppidl)->ooe);
|
|
CopyToMyPooe(pBuf, pooe);
|
|
|
|
// See if the caller has already given us a cookie
|
|
if (GUID_NULL == pooe->m_Cookie)
|
|
{
|
|
// Nope, see if we have one already
|
|
ReadCookieFromInetDB(URL(pooe), &(pooe->m_Cookie));
|
|
|
|
if (GUID_NULL == pooe->m_Cookie)
|
|
{
|
|
// Nope, so create one
|
|
CreateCookie(&pooe->m_Cookie);
|
|
}
|
|
}
|
|
|
|
WriteCookieToInetDB(URL(pooe), &(pooe->m_Cookie), FALSE);
|
|
|
|
WCHAR wszURL[INTERNET_MAX_URL_LENGTH];
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
sii.dwFlags = 0;
|
|
sii.dwPriority = 0;
|
|
sii.ScheduleGroup = CLSID_NULL;
|
|
sii.clsidAgent = pooe->clsidDest;
|
|
MyStrToOleStrN(wszURL, ARRAYSIZE(wszURL), URL(pooe));
|
|
hr = AddUpdateSubscription(&(pooe->m_Cookie), &sii, wszURL, 0, NULL, NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = WriteProperties(pooe);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ISubscriptionItem *psi;
|
|
|
|
hr = SubscriptionItemFromCookie(TRUE, &pooe->m_Cookie, &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONSCHEDULE subGroup = GetGroup(pooe);
|
|
SUBSCRIPTIONTYPE subType = GetItemCategory(pooe);
|
|
|
|
if (subGroup == SUBSSCHED_AUTO)
|
|
{
|
|
if (subType != SUBSTYPE_CHANNEL && subType != SUBSTYPE_DESKTOPCHANNEL)
|
|
{
|
|
hr = AddIt(psi, pooe, SUBSSCHED_DAILY);
|
|
}
|
|
else
|
|
{
|
|
if (pooe->m_Trigger.cbTriggerSize == sizeof(TASK_TRIGGER))
|
|
{
|
|
hr = ScheduleIt(psi, NAME(pooe), &pooe->m_Trigger);
|
|
pooe->groupCookie = CLSID_NULL;
|
|
}
|
|
else
|
|
{
|
|
hr = AddIt(psi, pooe, SUBSSCHED_DAILY);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = AddIt(psi, pooe, subGroup);
|
|
}
|
|
|
|
psi->Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
FireSubscriptionEvent(SUBSNOTF_CREATE, &pooe->m_Cookie);
|
|
}
|
|
else
|
|
{
|
|
TraceMsg(TF_ALWAYS, "Failed to add new subscription");
|
|
TraceMsg(TF_ALWAYS, "\thr = 0x%x", hr);
|
|
COfflineFolderEnum::FreePidl(*ppidl);
|
|
*ppidl = NULL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT SendUpdateRequests(HWND hwnd, CLSID * arrClsid, UINT count)
|
|
{
|
|
ISubscriptionMgr2 *pSubsMgr2;
|
|
HRESULT hr;
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr2, (void**)&pSubsMgr2);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONCOOKIE *pCookies = NULL;
|
|
ULONG nItemsToRun = count;
|
|
|
|
if (NULL == arrClsid)
|
|
{
|
|
IEnumSubscription *pes;
|
|
|
|
hr = pSubsMgr2->EnumSubscriptions(0, &pes);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ASSERT(NULL != pes);
|
|
|
|
pes->GetCount(&nItemsToRun);
|
|
if (nItemsToRun > 0)
|
|
{
|
|
pCookies = new SUBSCRIPTIONCOOKIE[nItemsToRun];
|
|
|
|
if (NULL != pCookies)
|
|
{
|
|
hr = pes->Next(nItemsToRun, pCookies, &nItemsToRun);
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
pes->Release();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pCookies = arrClsid;
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pSubsMgr2->UpdateItems(0, nItemsToRun, pCookies);
|
|
}
|
|
|
|
if ((NULL == arrClsid) && (NULL != pCookies))
|
|
{
|
|
delete [] pCookies;
|
|
}
|
|
|
|
pSubsMgr2->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT DoDeleteSubscription(POOEntry pooe)
|
|
{
|
|
HRESULT hr;
|
|
ISubscriptionItem *psi;
|
|
|
|
ASSERT(NULL != pooe);
|
|
|
|
hr = SubscriptionItemFromCookie(FALSE, &pooe->m_Cookie, &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
WritePassword(psi, NULL);
|
|
|
|
hr = DoDeleteSubscriptionItem(&pooe->m_Cookie, TRUE);
|
|
|
|
if (SUCCEEDED(hr) && (GetItemCategory(pooe) != SUBSTYPE_EXTERNAL))
|
|
{
|
|
WriteCookieToInetDB(URL(pooe), NULL, TRUE);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT PersistUpdate(POOEntry pooe, BOOL bCreate)
|
|
{
|
|
HRESULT hr;
|
|
ISubscriptionItem *psi;
|
|
|
|
hr = SubscriptionItemFromCookie(bCreate, &(pooe->m_Cookie), &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONITEMINFO sii = { sizeof(SUBSCRIPTIONITEMINFO) };
|
|
|
|
hr = psi->GetSubscriptionItemInfo(&sii);
|
|
|
|
if (SUCCEEDED(hr) || bCreate)
|
|
{
|
|
sii.clsidAgent = pooe->clsidDest;
|
|
hr = psi->SetSubscriptionItemInfo(&sii);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = WritePropertiesToItem(pooe, psi);
|
|
|
|
if (SUCCEEDED(hr) && IsNativeAgent(pooe->clsidDest))
|
|
{
|
|
WriteCookieToInetDB(URL(pooe), &(pooe->m_Cookie), FALSE);
|
|
}
|
|
}
|
|
}
|
|
|
|
// REVIEW: should we delete on failure here?
|
|
psi->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
#ifdef NEWSCHED_AUTONAME
|
|
void NewSched_AutoNameHelper(HWND hDlg)
|
|
{
|
|
TCHAR szDays[16];
|
|
TCHAR szTime[128];
|
|
TCHAR szFormat[MAX_PATH];
|
|
TCHAR szSchedName[MAX_PATH];
|
|
LPTSTR lpArguments[2];
|
|
BOOL bTranslate;
|
|
int nDays = GetDlgItemInt(hDlg, IDC_SCHEDULE_DAYS, &bTranslate, FALSE);
|
|
|
|
if (MLLoadString((nDays == 1) ? IDS_SCHED_FORMAT_DAILY : IDS_SCHED_FORMAT,
|
|
szFormat, ARRAYSIZE(szFormat)))
|
|
{
|
|
TCHAR szTimeFormat[32];
|
|
SYSTEMTIME st;
|
|
|
|
DateTime_GetSystemtime(GetDlgItem(hDlg, IDC_SCHEDULE_TIME), &st);
|
|
|
|
UpdateTimeFormat(szTimeFormat, ARRAYSIZE(szTimeFormat));
|
|
GetTimeFormat(LOCALE_USER_DEFAULT, TIME_NOSECONDS, &st,
|
|
szTimeFormat, szTime, ARRAYSIZE(szTime));
|
|
GetDlgItemText(hDlg, IDC_SCHEDULE_DAYS, szDays, ARRAYSIZE(szDays));
|
|
|
|
lpArguments[0] = szDays;
|
|
lpArguments[1] = szTime;
|
|
|
|
if (FormatMessage(FORMAT_MESSAGE_FROM_STRING | FORMAT_MESSAGE_ARGUMENT_ARRAY,
|
|
szFormat, 0, 0, szSchedName, ARRAYSIZE(szSchedName),
|
|
(va_list *)&lpArguments[0]))
|
|
{
|
|
SetDlgItemText(hDlg, IDC_SCHEDULE_NAME, szSchedName);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
BOOL NewSched_ResolveNameConflictHelper(HWND hDlg, TASK_TRIGGER *pTrig,
|
|
SYNCSCHEDULECOOKIE *pSchedCookie)
|
|
{
|
|
BOOL bResult;
|
|
|
|
SYSTEMTIME st;
|
|
TCHAR szSchedName[MAX_PATH];
|
|
|
|
GetDlgItemText(hDlg, IDC_SCHEDULE_NAME, szSchedName, ARRAYSIZE(szSchedName));
|
|
|
|
TrimWhiteSpace(szSchedName);
|
|
|
|
if (szSchedName[0] != 0)
|
|
{
|
|
bResult = TRUE;
|
|
|
|
memset(pTrig, 0, sizeof(TASK_TRIGGER));
|
|
pTrig->cbTriggerSize = sizeof(TASK_TRIGGER);
|
|
|
|
GetLocalTime(&st);
|
|
pTrig->wBeginYear = st.wYear;
|
|
pTrig->wBeginMonth = st.wMonth;
|
|
pTrig->wBeginDay = st.wDay;
|
|
|
|
DateTime_GetSystemtime(GetDlgItem(hDlg, IDC_SCHEDULE_TIME), &st);
|
|
pTrig->wStartHour = st.wHour;
|
|
pTrig->wStartMinute = st.wMinute;
|
|
|
|
pTrig->TriggerType = TASK_TIME_TRIGGER_DAILY;
|
|
|
|
BOOL bTranslated;
|
|
pTrig->Type.Daily.DaysInterval = (WORD)GetDlgItemInt(hDlg, IDC_SCHEDULE_DAYS, &bTranslated, FALSE);
|
|
|
|
int iConflictResult = HandleScheduleNameConflict(szSchedName,
|
|
pTrig,
|
|
hDlg,
|
|
pSchedCookie);
|
|
switch (iConflictResult)
|
|
{
|
|
case CONFLICT_NONE:
|
|
ASSERT(GUID_NULL == *pSchedCookie);
|
|
break;
|
|
|
|
case CONFLICT_RESOLVED_USE_NEW:
|
|
ASSERT(GUID_NULL != *pSchedCookie);
|
|
break;
|
|
|
|
case CONFLICT_RESOLVED_USE_OLD:
|
|
ASSERT(GUID_NULL == *pSchedCookie);
|
|
pTrig->cbTriggerSize = 0;
|
|
break;
|
|
|
|
case CONFLICT_UNRESOLVED:
|
|
bResult = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SGMessageBox(hDlg, IDS_EMPTY_SCHEDULE_NAME, MB_OK | MB_ICONWARNING);
|
|
bResult = FALSE;
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
void NewSched_CreateScheduleHelper(HWND hDlg, TASK_TRIGGER *pTrig,
|
|
SYNCSCHEDULECOOKIE *pSchedCookie)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (GUID_NULL == *pSchedCookie)
|
|
{
|
|
// Create new schedule
|
|
TCHAR szSchedName[MAX_PATH];
|
|
WCHAR wszSchedName[MAX_PATH];
|
|
|
|
DWORD dwSyncScheduleFlags =
|
|
(IsDlgButtonChecked(hDlg, IDC_WIZ_SCHEDULE_AUTOCONNECT) == BST_CHECKED)
|
|
? SYNCSCHEDINFO_FLAGS_AUTOCONNECT : 0;
|
|
|
|
GetDlgItemText(hDlg, IDC_SCHEDULE_NAME, szSchedName, ARRAYSIZE(szSchedName));
|
|
MyStrToOleStrN(wszSchedName, ARRAYSIZE(wszSchedName), szSchedName);
|
|
hr = CreateSchedule(wszSchedName, dwSyncScheduleFlags, pSchedCookie, pTrig, FALSE);
|
|
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
else if (sizeof(TASK_TRIGGER) == pTrig->cbTriggerSize)
|
|
{
|
|
// Update existing schedule with new task trigger
|
|
hr = UpdateScheduleTrigger(pSchedCookie, pTrig);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
else
|
|
{
|
|
// Use existing schedule without munging it
|
|
}
|
|
}
|
|
|
|
void NewSched_SetDefaultScheduleName(HWND hDlg)
|
|
{
|
|
if (SUCCEEDED(CoInitialize(NULL)))
|
|
{
|
|
ISyncScheduleMgr *pSyncScheduleMgr;
|
|
|
|
if (SUCCEEDED(CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr)))
|
|
{
|
|
SYNCSCHEDULECOOKIE schedCookie;
|
|
ISyncSchedule *pSyncSchedule;
|
|
|
|
if (SUCCEEDED(pSyncScheduleMgr->CreateSchedule(L"", 0, &schedCookie, &pSyncSchedule)))
|
|
{
|
|
WCHAR wszSchedName[MAX_PATH];
|
|
DWORD cchSchedName = ARRAYSIZE(wszSchedName);
|
|
|
|
if (SUCCEEDED(pSyncSchedule->GetScheduleName(&cchSchedName, wszSchedName)))
|
|
{
|
|
TCHAR szSchedName[MAX_PATH];
|
|
|
|
MyOleStrToStrN(szSchedName, ARRAYSIZE(szSchedName), wszSchedName);
|
|
SetDlgItemText(hDlg, IDC_SCHEDULE_NAME, szSchedName);
|
|
}
|
|
pSyncSchedule->Release();
|
|
}
|
|
pSyncScheduleMgr->Release();
|
|
}
|
|
CoUninitialize();
|
|
}
|
|
}
|
|
|
|
void NewSched_OnInitDialogHelper(HWND hDlg)
|
|
{
|
|
SYSTEMTIME st;
|
|
|
|
GetLocalTime(&st);
|
|
|
|
Edit_LimitText(GetDlgItem(hDlg, IDC_SCHEDULE_NAME), MAX_PATH - 1);
|
|
Edit_LimitText(GetDlgItem(hDlg, IDC_SCHEDULE_DAYS), 2);
|
|
SendMessage(GetDlgItem(hDlg, IDC_SCHEDULE_DAYS_SPIN),
|
|
UDM_SETRANGE, 0, MAKELONG(99, 1));
|
|
SendMessage(GetDlgItem(hDlg, IDC_SCHEDULE_DAYS_SPIN), UDM_SETPOS, 0, 1);
|
|
|
|
HWND hwndTimePicker = GetDlgItem(hDlg, IDC_SCHEDULE_TIME);
|
|
TCHAR szTimeFormat[32];
|
|
|
|
UpdateTimeFormat(szTimeFormat, ARRAYSIZE(szTimeFormat));
|
|
DateTime_SetSystemtime(hwndTimePicker, GDT_VALID, &st);
|
|
DateTime_SetFormat(hwndTimePicker, szTimeFormat);
|
|
|
|
NewSched_SetDefaultScheduleName(hDlg);
|
|
}
|
|
|
|
int KeepSpinNumberInRange(HWND hdlg, int idEdit, int idSpin, int minVal, int maxVal)
|
|
{
|
|
BOOL bTranslate;
|
|
int val = GetDlgItemInt(hdlg, idEdit, &bTranslate, FALSE);
|
|
if (!bTranslate || (val < minVal) || (val > maxVal))
|
|
{
|
|
// We have a problem, query the spin control
|
|
val = LOWORD(SendDlgItemMessage(hdlg, idSpin, UDM_GETPOS, 0, 0));
|
|
val = max(minVal, min(maxVal, val));
|
|
SetDlgItemInt(hdlg, idEdit, val, FALSE);
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
|
|
void SetPropSheetFlags(POOEBuf pBuf, BOOL bSet, DWORD dwPropSheetFlags)
|
|
{
|
|
if (bSet)
|
|
{
|
|
pBuf->m_dwPropSheetFlags |= dwPropSheetFlags;
|
|
}
|
|
else
|
|
{
|
|
pBuf->m_dwPropSheetFlags &= ~dwPropSheetFlags;
|
|
}
|
|
}
|
|
|
|
|
|
HRESULT FindURLProps(LPCTSTR m_URL, PROPVARIANT * pVarInfo)
|
|
{
|
|
HRESULT hr;
|
|
|
|
hr = IntSiteHelper(m_URL, &c_rgPropRead[PROP_SUBSCRIPTION], pVarInfo, 1, FALSE);
|
|
return hr;
|
|
}
|
|
|
|
HRESULT LoadOOEntryInfo(POOEBuf pBuf, SUBSCRIPTIONCOOKIE *pCookie, DWORD *pdwSize)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (!pBuf || !pCookie || !pdwSize)
|
|
{
|
|
TraceMsg(TF_ALWAYS, "Invalid ARG (1/2/3) %x %x", pBuf, pCookie, pdwSize);
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
ISubscriptionItem *psi;
|
|
hr = SubscriptionItemFromCookie(FALSE, pCookie, &psi);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
SUBSCRIPTIONITEMINFO sii;
|
|
|
|
sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO);
|
|
|
|
hr = psi->GetSubscriptionItemInfo(&sii);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ZeroMemory((void *)pBuf, sizeof(OOEBuf));
|
|
pBuf->m_Cookie = *pCookie;
|
|
|
|
// TODO: resolve scheduling goo!
|
|
|
|
// pBuf->groupCookie = pItem->groupCookie;
|
|
// pBuf->grfTaskTrigger = pItem->TaskData.dwTaskFlags;
|
|
/* if (pItem->groupCookie == CLSID_NULL) {
|
|
pBuf->m_Trigger = pItem->TaskTrigger;
|
|
if (pBuf->m_Trigger.cbTriggerSize != sizeof(TASK_TRIGGER)) {
|
|
ASSERT(0);
|
|
return E_INVALIDARG;
|
|
}
|
|
} else {
|
|
pBuf->m_Trigger.cbTriggerSize = 0; // Invalid
|
|
}
|
|
*/
|
|
|
|
pBuf->clsidDest = sii.clsidAgent;
|
|
|
|
if (!IsNativeAgent(sii.clsidAgent))
|
|
{
|
|
pBuf->dwFlags = PROP_WEBCRAWL_EXTERNAL;
|
|
}
|
|
else
|
|
{
|
|
pBuf->dwFlags = PROP_WEBCRAWL_ALL;
|
|
}
|
|
|
|
hr = ReadProperties(pBuf);
|
|
*pdwSize = BufferSize(pBuf);
|
|
}
|
|
psi->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
// SaveBufferChange
|
|
// newBuf: [in/out]
|
|
|
|
|
|
HRESULT SaveBufferChange(POOEBuf newBuf, BOOL bCreate)
|
|
{
|
|
HRESULT hr;
|
|
DWORD dwSize;
|
|
POOEntry pooe;
|
|
LPMYPIDL newPidl;
|
|
|
|
ASSERT (newBuf);
|
|
if (newBuf->dwFlags == 0)
|
|
return S_OK;
|
|
|
|
dwSize = BufferSize(newBuf);
|
|
newPidl = COfflineFolderEnum::NewPidl(dwSize);
|
|
if (!newPidl)
|
|
return E_OUTOFMEMORY;
|
|
|
|
pooe = &(newPidl->ooe);
|
|
CopyToMyPooe(newBuf, pooe);
|
|
newBuf->dwFlags = 0;
|
|
hr = PersistUpdate(pooe, bCreate);
|
|
if (SUCCEEDED(hr)) {
|
|
DWORD dwPropSheetFlags = newBuf->m_dwPropSheetFlags; // Preserve prop sheet flags
|
|
|
|
hr = LoadWithCookie(URL(pooe), newBuf, &dwSize, &(pooe->m_Cookie));
|
|
|
|
newBuf->m_dwPropSheetFlags = dwPropSheetFlags; // restore
|
|
newBuf->dwFlags = 0;
|
|
if (hr == S_OK) {
|
|
COfflineFolderEnum::FreePidl(newPidl);
|
|
newPidl = COfflineFolderEnum::NewPidl(dwSize);
|
|
if (!(newPidl)) {
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
pooe = &(newPidl->ooe);
|
|
CopyToMyPooe(newBuf, pooe);
|
|
}
|
|
_GenerateEvent(SHCNE_UPDATEITEM, (LPITEMIDLIST)newPidl, NULL);
|
|
}
|
|
COfflineFolderEnum::FreePidl(newPidl);
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// IntSiteHelper
|
|
|
|
// pszURL url to read/write props for
|
|
// pPropspec properties to read or write
|
|
// pReadPropvar where to store or get properties
|
|
// uPropVarArraySize number of properties
|
|
// fWrite read/write flag
|
|
|
|
|
|
HRESULT IntSiteHelper(LPCTSTR pszURL, const PROPSPEC *pPropspec,
|
|
PROPVARIANT *pPropvar, UINT uPropVarArraySize, BOOL fWrite)
|
|
{
|
|
HRESULT hr;
|
|
IUniformResourceLocator * purl = NULL;
|
|
IPropertySetStorage * ppropsetstg = NULL; // init to keep compiler happy
|
|
IPropertyStorage * ppropstg = NULL; // init to keep compiler happy
|
|
|
|
hr = SHCoCreateInstance(NULL, &CLSID_InternetShortcut, NULL, IID_IUniformResourceLocator, (LPVOID*)&purl);
|
|
if(SUCCEEDED(hr)) {
|
|
hr = purl->SetURL(pszURL, 0);
|
|
}
|
|
|
|
if(SUCCEEDED(hr)) {
|
|
hr = purl->QueryInterface(IID_IPropertySetStorage,
|
|
(LPVOID *)&ppropsetstg);
|
|
}
|
|
|
|
if(SUCCEEDED(hr)) {
|
|
hr = ppropsetstg->Open(FMTID_InternetSite, STGM_READWRITE, &ppropstg);
|
|
ppropsetstg->Release();
|
|
}
|
|
|
|
if(SUCCEEDED(hr)) {
|
|
if(fWrite) {
|
|
hr = ppropstg->WriteMultiple(uPropVarArraySize, pPropspec,
|
|
pPropvar, 0);
|
|
ppropstg->Commit(STGC_DEFAULT);
|
|
} else {
|
|
hr = ppropstg->ReadMultiple(uPropVarArraySize, pPropspec,
|
|
pPropvar);
|
|
}
|
|
ppropstg->Release();
|
|
}
|
|
|
|
if(purl)
|
|
purl->Release();
|
|
|
|
return hr;
|
|
}
|
|
|
|
// CODE FROM SYNCMGR SOURCES.
|
|
|
|
|
|
// Local constants
|
|
|
|
// DEFAULT_TIME_FORMAT - what to use if there's a problem getting format
|
|
// from system.
|
|
|
|
#define ARRAYLEN(a) (sizeof(a) / sizeof((a)[0]))
|
|
#define DEFAULT_TIME_FORMAT TEXT("hh:mm tt")
|
|
#define GET_LOCALE_INFO(lcid) \
|
|
{ \
|
|
cch = GetLocaleInfo(LOCALE_USER_DEFAULT, \
|
|
(lcid), \
|
|
tszScratch, \
|
|
ARRAYLEN(tszScratch)); \
|
|
if (!cch) \
|
|
{ \
|
|
break; \
|
|
} \
|
|
}
|
|
|
|
|
|
// Function: UpdateTimeFormat
|
|
|
|
// Synopsis: Construct a time format containing hour and minute for use
|
|
// with the date picker control.
|
|
|
|
// Arguments: [tszTimeFormat] - buffer to fill with time format
|
|
// [cchTimeFormat] - size in chars of buffer
|
|
|
|
// Modifies: *[tszTimeFormat]
|
|
|
|
// History: 11-18-1996 DavidMun Created
|
|
|
|
// Notes: This is called on initialization and for wininichange
|
|
// processing.
|
|
|
|
|
|
void
|
|
UpdateTimeFormat(
|
|
LPTSTR tszTimeFormat,
|
|
ULONG cchTimeFormat)
|
|
{
|
|
ULONG cch;
|
|
TCHAR tszScratch[80];
|
|
BOOL fAmPm = FALSE;
|
|
BOOL fAmPmPrefixes = FALSE;
|
|
BOOL fLeadingZero = FALSE;
|
|
|
|
do
|
|
{
|
|
GET_LOCALE_INFO(LOCALE_ITIME);
|
|
fAmPm = (*tszScratch == TEXT('0'));
|
|
|
|
if (fAmPm)
|
|
{
|
|
GET_LOCALE_INFO(LOCALE_ITIMEMARKPOSN);
|
|
fAmPmPrefixes = (*tszScratch == TEXT('1'));
|
|
}
|
|
|
|
GET_LOCALE_INFO(LOCALE_ITLZERO);
|
|
fLeadingZero = (*tszScratch == TEXT('1'));
|
|
|
|
GET_LOCALE_INFO(LOCALE_STIME);
|
|
|
|
|
|
// See if there's enough room in destination string
|
|
|
|
|
|
cch = 1 + // terminating nul
|
|
1 + // first hour digit specifier "h"
|
|
2 + // minutes specifier "mm"
|
|
(fLeadingZero != 0) + // leading hour digit specifier "h"
|
|
lstrlen(tszScratch) + // separator string
|
|
(fAmPm ? 3 : 0); // space and "tt" for AM/PM
|
|
|
|
if (cch > cchTimeFormat)
|
|
{
|
|
cch = 0; // signal error
|
|
}
|
|
} while (0);
|
|
|
|
|
|
// If there was a problem in getting locale info for building time string
|
|
// just use the default and bail.
|
|
|
|
|
|
if (!cch)
|
|
{
|
|
StrCpy(tszTimeFormat, DEFAULT_TIME_FORMAT);
|
|
return;
|
|
}
|
|
|
|
|
|
// Build a time string that has hours and minutes but no seconds.
|
|
|
|
|
|
tszTimeFormat[0] = TEXT('\0');
|
|
|
|
if (fAmPm)
|
|
{
|
|
if (fAmPmPrefixes)
|
|
{
|
|
StrCpy(tszTimeFormat, TEXT("tt "));
|
|
}
|
|
|
|
StrCat(tszTimeFormat, TEXT("h"));
|
|
|
|
if (fLeadingZero)
|
|
{
|
|
StrCat(tszTimeFormat, TEXT("h"));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
StrCat(tszTimeFormat, TEXT("H"));
|
|
|
|
if (fLeadingZero)
|
|
{
|
|
StrCat(tszTimeFormat, TEXT("H"));
|
|
}
|
|
}
|
|
|
|
StrCat(tszTimeFormat, tszScratch); // separator
|
|
StrCat(tszTimeFormat, TEXT("mm"));
|
|
|
|
if (fAmPm && !fAmPmPrefixes)
|
|
{
|
|
StrCat(tszTimeFormat, TEXT(" tt"));
|
|
}
|
|
}
|