#include "shellprv.h"
#include "mshtml.h"
#include "mshtmhst.h"
#include "mshtmdid.h"
#include "htiframe.h"
#include "exdisp.h"
#include "exdispid.h"
#include "dspsprt.h"
#include "cowsite.h"
#include "ids.h"
#include "inetsmgr.h"
#pragma hdrstop
// helper functions
typedef BOOL (*pfnDllRegisterWindowClasses)(const SHDRC * pshdrc);
BOOL SHDOCVW_DllRegisterWindowClasses(const SHDRC * pshdrc)
{
static HINSTANCE _hinstShdocvw = NULL;
static pfnDllRegisterWindowClasses _regfunc = NULL;
BOOL fSuccess = FALSE;
if (!_hinstShdocvw)
{
_hinstShdocvw = LoadLibrary(TEXT("shdocvw.dll"));
_regfunc = (pfnDllRegisterWindowClasses) GetProcAddress(_hinstShdocvw, "DllRegisterWindowClasses");
}
if (_regfunc)
fSuccess = _regfunc(pshdrc);
return fSuccess;
}
// Advise point for DIID_DWebBrowserEvents2
// Just an IDispatch implementation that delegates back to the main class. Allows us to have a separate "Invoke".
class CWebWizardPage;
class CWebEventHandler : public IServiceProvider, DWebBrowserEvents2
{
public:
CWebEventHandler(CWebWizardPage *pswp);
~CWebEventHandler(); // TODO: Make this virtual or it will never execute.
// IUnknown
STDMETHODIMP QueryInterface(REFIID riid, void **ppv);
STDMETHODIMP_(ULONG) AddRef() {return 2;}
STDMETHODIMP_(ULONG) Release() {return 1;}
// (DwebBrowserEvents)IDispatch
STDMETHODIMP GetTypeInfoCount(/* [out] */ UINT *pctinfo) { return E_NOTIMPL; }
STDMETHODIMP GetTypeInfo(
/* [in] */ UINT iTInfo,
/* [in] */ LCID lcid,
/* [out] */ ITypeInfo **ppTInfo)
{ return E_NOTIMPL; }
STDMETHODIMP GetIDsOfNames(
/* [in] */ REFIID riid,
/* [size_is][in] */ LPOLESTR *rgszNames,
/* [in] */ UINT cNames,
/* [in] */ LCID lcid,
/* [size_is][out] */ DISPID *rgDispId)
{ return E_NOTIMPL; }
/* [local] */ STDMETHODIMP Invoke(
/* [in] */ DISPID dispIdMember,
/* [in] */ REFIID riid,
/* [in] */ LCID lcid,
/* [in] */ WORD wFlags,
/* [out][in] */ DISPPARAMS *pDispParams,
/* [out] */ VARIANT *pVarResult,
/* [out] */ EXCEPINFO *pExcepInfo,
/* [out] */ UINT *puArgErr);
HRESULT _Advise(BOOL fConnect);
// IServiceProvider
STDMETHODIMP QueryService(REFGUID guidService, REFIID riid, void **ppv);
private:
CWebWizardPage* _pwizPage;
DWORD _dwCPCookie;
IConnectionPoint* _pcpCurrentConnection;
};
#define SHOW_PROGRESS_TIMER 1
#define SHOW_PROGRESS_TIMEOUT 1000 // Start showing the progress indicator after 1 second of dead time.
class CWebWizardPage : public CImpIDispatch,
CObjectWithSite,
IDocHostUIHandler,
IServiceProvider,
IWebWizardExtension,
INewWDEvents
{
public:
CWebWizardPage();
~CWebWizardPage();
// IUnknown
STDMETHODIMP QueryInterface(REFIID riid, void **ppv);
STDMETHODIMP_(ULONG) AddRef();
STDMETHODIMP_(ULONG) Release();
// IDocHostUIHandler
STDMETHODIMP ShowContextMenu(
/* [in] */ DWORD dwID,
/* [in] */ POINT *ppt,
/* [in] */ IUnknown *pcmdtReserved,
/* [in] */ IDispatch *pdispReserved)
{ return E_NOTIMPL; }
STDMETHODIMP GetHostInfo(
/* [out][in] */ DOCHOSTUIINFO *pInfo);
STDMETHODIMP ShowUI(
/* [in] */ DWORD dwID,
/* [in] */ IOleInPlaceActiveObject *pActiveObject,
/* [in] */ IOleCommandTarget *pCommandTarget,
/* [in] */ IOleInPlaceFrame *pFrame,
/* [in] */ IOleInPlaceUIWindow *pDoc)
{ return E_NOTIMPL; }
STDMETHODIMP HideUI(void)
{ return E_NOTIMPL; }
STDMETHODIMP UpdateUI(void)
{ return E_NOTIMPL; }
STDMETHODIMP EnableModeless(
/* [in] */ BOOL fEnable)
{ return E_NOTIMPL; }
STDMETHODIMP OnDocWindowActivate(
/* [in] */ BOOL fActivate)
{ return E_NOTIMPL; }
STDMETHODIMP OnFrameWindowActivate(
/* [in] */ BOOL fActivate)
{ return E_NOTIMPL; }
STDMETHODIMP ResizeBorder(
/* [in] */ LPCRECT prcBorder,
/* [in] */ IOleInPlaceUIWindow *pUIWindow,
/* [in] */ BOOL fRameWindow)
{ return E_NOTIMPL; }
STDMETHODIMP TranslateAccelerator(
/* [in] */ LPMSG lpMsg,
/* [in] */ const GUID *pguidCmdGroup,
/* [in] */ DWORD nCmdID)
{ return E_NOTIMPL; }
STDMETHODIMP GetOptionKeyPath(
/* [out] */ LPOLESTR *pchKey,
/* [in] */ DWORD dw)
{ return E_NOTIMPL; }
STDMETHODIMP GetDropTarget(
/* [in] */ IDropTarget *pDropTarget,
/* [out] */ IDropTarget **ppDropTarget)
{ return E_NOTIMPL; }
STDMETHODIMP GetExternal(
/* [out] */ IDispatch **ppDispatch);
STDMETHODIMP TranslateUrl(
/* [in] */ DWORD dwTranslate,
/* [in] */ OLECHAR *pchURLIn,
/* [out] */ OLECHAR **ppchURLOut)
{ return E_NOTIMPL; }
STDMETHODIMP FilterDataObject(
/* [in] */ IDataObject *pDO,
/* [out] */ IDataObject **ppDORet)
{ return E_NOTIMPL; }
// IServiceProvider
STDMETHODIMP QueryService(
/*[in]*/ REFGUID guidService,
/*[in]*/ REFIID riid,
/*[out]*/ void **ppv);
// INewWDEvents
// (IDispatch)
STDMETHODIMP GetTypeInfoCount(
/* [out] */ UINT *pctinfo)
{ return E_NOTIMPL; }
STDMETHODIMP GetTypeInfo(
/* [in] */ UINT iTInfo,
/* [in] */ LCID lcid,
/* [out] */ ITypeInfo **ppTInfo)
{
return CImpIDispatch::GetTypeInfo(iTInfo, lcid, ppTInfo);
}
STDMETHODIMP GetIDsOfNames(
/* [in] */ REFIID riid,
/* [size_is][in] */ LPOLESTR *rgszNames,
/* [in] */ UINT cNames,
/* [in] */ LCID lcid,
/* [size_is][out] */ DISPID *rgDispId)
{
return CImpIDispatch::GetIDsOfNames(riid, rgszNames, cNames, lcid, rgDispId);
}
STDMETHODIMP Invoke(
/* [in] */ DISPID dispIdMember,
/* [in] */ REFIID riid,
/* [in] */ LCID lcid,
/* [in] */ WORD wFlags,
/* [out][in] */ DISPPARAMS *pDispParams,
/* [out] */ VARIANT *pVarResult,
/* [out] */ EXCEPINFO *pExcepInfo,
/* [out] */ UINT *puArgErr)
{
return CImpIDispatch::Invoke(dispIdMember, riid, lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
}
STDMETHODIMP FinalBack(void);
STDMETHODIMP FinalNext(void);
STDMETHODIMP Cancel(void);
STDMETHODIMP put_Caption(
/* [in] */ BSTR bstrCaption);
STDMETHODIMP get_Caption(
/* [retval][out] */ BSTR *pbstrCaption);
STDMETHODIMP put_Property(
/* [in] */ BSTR bstrPropertyName,
/* [in] */ VARIANT *pvProperty);
STDMETHODIMP get_Property(
/* [in] */ BSTR bstrPropertyName,
/* [retval][out] */ VARIANT *pvProperty);
STDMETHODIMP SetWizardButtons(
/* [in] */ VARIANT_BOOL vfEnableBack,
/* [in] */ VARIANT_BOOL vfEnableNext,
/* [in] */ VARIANT_BOOL vfLastPage);
STDMETHODIMP SetHeaderText(
/* [in] */ BSTR bstrHeaderTitle,
/* [in] */ BSTR bstrHeaderSubtitle);
STDMETHODIMP PassportAuthenticate(
/* [in] */ BSTR bstrSignInUrl,
/* [retval][out] */ VARIANT_BOOL * pvfAuthenticated);
// IWizardExtension
STDMETHODIMP AddPages(HPROPSHEETPAGE* aPages, UINT cPages, UINT *pnPages);
STDMETHODIMP GetFirstPage(HPROPSHEETPAGE *phPage);
STDMETHODIMP GetLastPage(HPROPSHEETPAGE *phPage)
{ return GetFirstPage(phPage); }
// IWebWizardExtension
STDMETHODIMP SetInitialURL(LPCWSTR pszDefaultURL);
STDMETHODIMP SetErrorURL(LPCWSTR pszErrorURL);
protected:
friend class CWebEventHandler;
void _OnDownloadBegin();
void _OnDocumentComplete();
private:
void _InitBrowser();
HRESULT _NavigateBrowser(LPCWSTR pszUrl);
HRESULT _CallScript(IWebBrowser2* pbrowser, LPCWSTR pszFunction);
BOOL _IsScriptFunctionOnPage(IWebBrowser2* pbrowser, LPCWSTR pszFunction);
BOOL _IsBrowserVisible();
void _ShowBrowser(BOOL fShow);
void _SizeProgress();
void _ShowProgress(BOOL fShow);
void _StartShowProgressTimer();
void _SetHeaderText(LPCWSTR pszHeader, LPCWSTR pszSubHeader);
virtual INT_PTR DialogProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
static INT_PTR StaticProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
static UINT PropPageProc(HWND hwndDlg, UINT uMsg, PROPSHEETPAGE *ppsp);
BOOL OnInitDialog(HWND hwnd, HWND hwndFocus, LPARAM lParam);
BOOL OnDestroy(HWND hwnd);
BOOL OnNotify(HWND hwnd, int idCtrl, LPNMHDR pnmh);
BOOL OnTimer(HWND hwnd, UINT nIDEvent);
LONG _cRef;
CWebEventHandler *_pwebEventHandler;
IWebBrowser2 *_pwebbrowser;
IOleInPlaceActiveObject *_poipao;
HWND _hwndOCHost; // Web browser control window
HWND _hwndFrame; // Wizard frame window
HWND _hwnd; // Dialog window
HPROPSHEETPAGE _hPage;
LPWSTR _pszInitialURL;
LPWSTR _pszErrorURL;
};
INT_PTR CWebWizardPage::StaticProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
CWebWizardPage* pthis = (CWebWizardPage*) GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
PROPSHEETPAGE* ppage;
INT_PTR fProcessed;
if (uMsg == WM_INITDIALOG)
{
ppage = (PROPSHEETPAGE*) lParam;
pthis = (CWebWizardPage*) ppage->lParam;
SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR) pthis);
}
if (pthis != NULL)
{
fProcessed = pthis->DialogProc(hwndDlg, uMsg, wParam, lParam);
}
else
{
fProcessed = FALSE;
}
return fProcessed;
}
// construction and IUnknown
CWebEventHandler::CWebEventHandler(CWebWizardPage *pwswp) :
_pcpCurrentConnection(NULL),
_pwizPage(pwswp)
{
}
CWebEventHandler::~CWebEventHandler()
{
_Advise(FALSE);
}
HRESULT CWebEventHandler::QueryInterface(REFIID riid, void **ppv)
{
static const QITAB qit[] = {
QITABENTMULTI(CWebEventHandler, IDispatch, DWebBrowserEvents2),
QITABENTMULTI2(CWebEventHandler, DIID_DWebBrowserEvents2, DWebBrowserEvents2),
// QITABENTMULTI2(CWebEventHandler, DIID_DWebBrowserEvents, DWebBrowserEvents),
QITABENT(CWebEventHandler, IServiceProvider),
{ 0 },
};
return QISearch(this, qit, riid, ppv);
}
STDMETHODIMP CWebEventHandler::QueryService(REFGUID guidService, REFIID riid, void **ppv)
{
HRESULT hr = E_NOINTERFACE;
*ppv = NULL; // no result yet
// we are a site for the OleControlSite interfaces only
if (guidService == SID_OleControlSite)
{
if (riid == IID_IDispatch)
{
hr = this->QueryInterface(riid, ppv);
}
}
return hr;
}
HRESULT CWebEventHandler_CreateInstance(CWebWizardPage *pwswp, CWebEventHandler **ppweh)
{
*ppweh = new CWebEventHandler(pwswp);
if (!*ppweh)
return E_OUTOFMEMORY;
return S_OK;
}
HRESULT CWebEventHandler::_Advise(BOOL fConnect)
{
HRESULT hr = S_OK;
// If we're already connected, disconnect, since we either want to disconnect or reconnect to
// a different webbrowser.
if (_pcpCurrentConnection)
{
hr = _pcpCurrentConnection->Unadvise(_dwCPCookie);
if (SUCCEEDED(hr))
{
ATOMICRELEASE(_pcpCurrentConnection);
}
}
else
{
// We expect that if _pcpCurrentConnection is NULL, no code earlier would have changed hr, and that it is still S_OK
// The code below expects that if !SUCCEEDED(hr), Unadvise failed above.
ASSERT(SUCCEEDED(hr));
}
if (_pwizPage && _pwizPage->_pwebbrowser)
{
if (SUCCEEDED(hr) && fConnect)
{
IConnectionPointContainer* pcontainer;
hr = _pwizPage->_pwebbrowser->QueryInterface(IID_PPV_ARG(IConnectionPointContainer, &pcontainer));
if (SUCCEEDED(hr))
{
IConnectionPoint* pconnpoint;
hr = pcontainer->FindConnectionPoint(DIID_DWebBrowserEvents2, &pconnpoint);
if (SUCCEEDED(hr))
{
IDispatch* pDisp;
hr = QueryInterface(IID_PPV_ARG(IDispatch, &pDisp));
if (SUCCEEDED(hr))
{
hr = pconnpoint->Advise(pDisp, &_dwCPCookie);
pDisp->Release();
}
if (SUCCEEDED(hr))
{
// TODO: Enable ATOMICRELEASE() to verify we won't leak anything
// ATOMICRELEASE(_pcpCurrentConnection);
_pcpCurrentConnection = pconnpoint;
}
else
{
pconnpoint->Release();
}
}
pcontainer->Release();
}
}
}
return hr;
}
HRESULT CWebEventHandler::Invoke(
/* [in] */ DISPID dispIdMember,
/* [in] */ REFIID riid,
/* [in] */ LCID lcid,
/* [in] */ WORD wFlags,
/* [out][in] */ DISPPARAMS *pDispParams,
/* [out] */ VARIANT *pVarResult,
/* [out] */ EXCEPINFO *pExcepInfo,
/* [out] */ UINT *puArgErr)
{
HRESULT hr = S_OK;
switch (dispIdMember)
{
case DISPID_BEFORENAVIGATE2:
_pwizPage->_OnDownloadBegin();
break;
case DISPID_DOCUMENTCOMPLETE:
_pwizPage->_OnDocumentComplete();
break;
default:
hr = DISP_E_MEMBERNOTFOUND;
break;
}
return hr;
}
// Object for hosting HTML wizard pages
CWebWizardPage::CWebWizardPage() :
CImpIDispatch(LIBID_Shell32, 0, 0, IID_INewWDEvents),
_cRef(1)
{
// Ensure zero-init happened
ASSERT(NULL == _pwebbrowser);
ASSERT(NULL == _pwebEventHandler);
ASSERT(NULL == _pszInitialURL);
ASSERT(NULL == _pszErrorURL);
}
CWebWizardPage::~CWebWizardPage()
{
ATOMICRELEASE(_pwebbrowser);
ATOMICRELEASE(_pwebEventHandler);
ATOMICRELEASE(_punkSite);
ATOMICRELEASE(_poipao);
Str_SetPtr(&_pszInitialURL, NULL);
Str_SetPtr(&_pszErrorURL, NULL);
}
HRESULT CWebWizardPage::QueryInterface(REFIID riid, void **ppv)
{
static const QITAB qit[] =
{
QITABENTMULTI(CWebWizardPage, IWizardExtension, IWebWizardExtension),
QITABENT(CWebWizardPage, IWebWizardExtension),
QITABENT(CWebWizardPage, IDocHostUIHandler),
QITABENT(CWebWizardPage, IServiceProvider),
QITABENT(CWebWizardPage, INewWDEvents),
QITABENT(CWebWizardPage, IDispatch),
QITABENT(CWebWizardPage, IWebWizardExtension),
QITABENT(CWebWizardPage, IObjectWithSite),
{ 0 },
};
return QISearch(this, qit, riid, ppv);
}
ULONG CWebWizardPage::AddRef()
{
return InterlockedIncrement(&_cRef);
}
ULONG CWebWizardPage::Release()
{
if (InterlockedDecrement(&_cRef))
return _cRef;
delete this;
return 0;
}
HRESULT CWebWizardPage::QueryService(REFGUID guidService, REFIID riid, void **ppv)
{
HRESULT hr = E_NOINTERFACE;
*ppv = NULL;
if (_punkSite)
hr = IUnknown_QueryService(_punkSite, guidService, riid, ppv);
return hr;
}
void CWebWizardPage::_OnDownloadBegin()
{
_ShowBrowser(FALSE);
_StartShowProgressTimer();
SetWizardButtons(VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE);
}
void CWebWizardPage::_OnDocumentComplete()
{
if (!_IsScriptFunctionOnPage(_pwebbrowser, L"OnBack"))
{
// This is an invalid page; navigate to our private error page
BSTR bstrOldUrl;
if (_pwebbrowser && SUCCEEDED(_pwebbrowser->get_LocationURL(&bstrOldUrl)))
{
#ifdef DEBUG
if (IDYES == ::MessageBox(_hwnd, L"A Web Service Error has occured.\n\nDo you want to load the HTML page anyway so you can debug it?\n\n(This only appears in debug builds)", bstrOldUrl, MB_ICONERROR | MB_YESNO))
{
_ShowBrowser(TRUE);
SysFreeString(bstrOldUrl);
return;
}
#endif
BSTR bstrUrl = NULL;
BOOL fUsingCustomError = FALSE;
// If we have a custom error URL and we haven't already failed trying
// to navigate to this custom URL...
if ((NULL != _pszErrorURL) &&
(0 != StrCmpI(_pszErrorURL, bstrOldUrl)))
{
// then use the custom URL.
bstrUrl = SysAllocString(_pszErrorURL);
fUsingCustomError = TRUE;
}
else
{
bstrUrl = SysAllocString(L"res://shell32.dll/WebServiceError.htm");
}
if (bstrUrl)
{
_pwebbrowser->Navigate(bstrUrl, NULL, NULL, NULL, NULL);
SysFreeString(bstrUrl);
// Custom error URL will provide its own header and subheader
if (!fUsingCustomError)
{
WCHAR szTitle[256];
LoadString(g_hinst, IDS_WEBDLG_ERRTITLE, szTitle, ARRAYSIZE(szTitle));
#ifdef DEBUG
_SetHeaderText(szTitle, bstrOldUrl);
#else
_SetHeaderText(szTitle, L"");
#endif
}
}
SysFreeString(bstrOldUrl);
}
// else out of memory - oops.
}
else
{
_ShowBrowser(TRUE);
}
}
HRESULT CWebWizardPage::GetHostInfo(DOCHOSTUIINFO *pInfo)
{
ZeroMemory(pInfo, sizeof(*pInfo));
pInfo->cbSize = sizeof(*pInfo);
pInfo->dwDoubleClick = DOCHOSTUIDBLCLK_DEFAULT;
pInfo->dwFlags = DOCHOSTUIFLAG_DIALOG | DOCHOSTUIFLAG_NO3DBORDER |
DOCHOSTUIFLAG_ENABLE_FORMS_AUTOCOMPLETE | DOCHOSTUIFLAG_THEME |
DOCHOSTUIFLAG_FLAT_SCROLLBAR;
return S_OK;
}
HRESULT CWebWizardPage::GetExternal(IDispatch** ppDispatch)
{
return QueryInterface(IID_PPV_ARG(IDispatch, ppDispatch));
}
INT_PTR CWebWizardPage::DialogProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
HANDLE_MSG(hwndDlg, WM_INITDIALOG, OnInitDialog);
HANDLE_MSG(hwndDlg, WM_DESTROY, OnDestroy);
HANDLE_MSG(hwndDlg, WM_NOTIFY, OnNotify);
HANDLE_MSG(hwndDlg, WM_TIMER, OnTimer);
}
return FALSE;
}
HRESULT CWebWizardPage::_CallScript(IWebBrowser2* pbrowser, LPCWSTR pszFunction)
{
HRESULT hr = E_INVALIDARG;
if (pbrowser)
{
IDispatch* pdocDispatch;
hr = pbrowser->get_Document(&pdocDispatch);
if ((S_OK == hr) && pdocDispatch)
{
IHTMLDocument* pdoc;
hr = pdocDispatch->QueryInterface(IID_PPV_ARG(IHTMLDocument, &pdoc));
if (SUCCEEDED(hr))
{
IDispatch* pdispScript;
hr = pdoc->get_Script(&pdispScript);
if (S_OK == hr)
{
DISPID dispid;
hr = pdispScript->GetIDsOfNames(IID_NULL, const_cast(&pszFunction), 1, LOCALE_SYSTEM_DEFAULT, &dispid);
if (SUCCEEDED(hr))
{
unsigned int uArgErr;
DISPPARAMS dispparams = {0};
hr = pdispScript->Invoke(dispid, IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_METHOD, &dispparams, NULL, NULL, &uArgErr);
}
pdispScript->Release();
}
else
{
hr = E_FAIL;
}
pdoc->Release();
}
else
{
hr = E_FAIL;
}
pdocDispatch->Release();
}
}
return hr;
}
BOOL CWebWizardPage::_IsScriptFunctionOnPage(IWebBrowser2* pbrowser, LPCWSTR pszFunction)
{
HRESULT hr = E_INVALIDARG;
if (pbrowser)
{
IDispatch* pdocDispatch;
hr = pbrowser->get_Document(&pdocDispatch);
if (S_OK == hr && pdocDispatch)
{
IHTMLDocument* pdoc;
hr = pdocDispatch->QueryInterface(IID_PPV_ARG(IHTMLDocument, &pdoc));
if (SUCCEEDED(hr))
{
IDispatch* pdispScript;
hr = pdoc->get_Script(&pdispScript);
if (S_OK == hr)
{
DISPID dispid;
hr = pdispScript->GetIDsOfNames(IID_NULL, const_cast(&pszFunction), 1, LOCALE_SYSTEM_DEFAULT, &dispid);
pdispScript->Release();
}
else
{
hr = E_FAIL;
}
pdoc->Release();
}
else
{
hr = E_FAIL;
}
pdocDispatch->Release();
}
}
return (S_OK == hr) ? TRUE : FALSE;
}
// Uncomment this to NOT pass the LCID on the URL query string - for testing only.
BOOL CWebWizardPage::OnNotify(HWND hwnd, int idCtrl, LPNMHDR pnmh)
{
switch (pnmh->code)
{
case PSN_SETACTIVE:
{
_SizeProgress();
_ShowProgress(FALSE);
_ShowBrowser(FALSE);
// fetch the high contrast flag, and set accordingly for the HTML to read
// its OK for us to fail setting this into the property bag.
HIGHCONTRAST hc = {sizeof(hc)};
if (SystemParametersInfo(SPI_GETHIGHCONTRAST, sizeof(hc), &hc, 0))
{
VARIANT var = {VT_BOOL};
var.boolVal = (hc.dwFlags & HCF_HIGHCONTRASTON) ? VARIANT_TRUE:VARIANT_FALSE;
put_Property(L"HighContrast", &var);
}
// Position the OCHost window
RECT rectClient;
GetClientRect(hwnd, &rectClient);
SetWindowPos(_hwndOCHost, NULL, 0, 0, rectClient.right, rectClient.bottom, SWP_NOMOVE | SWP_NOOWNERZORDER);
// set the initial URL
if (_pszInitialURL)
{
WCHAR szURLWithLCID[INTERNET_MAX_URL_LENGTH];
LPCWSTR pszFormat = StrChr(_pszInitialURL, L'?') ? L"%s&lcid=%d&langid=%d":L"%s?lcid=%d&langid=%d";
if (wnsprintf(szURLWithLCID, ARRAYSIZE(szURLWithLCID), pszFormat, _pszInitialURL, GetUserDefaultLCID(), GetUserDefaultUILanguage()) > 0)
{
_NavigateBrowser(szURLWithLCID);
}
}
}
break;
// WIZNEXT and WIZBACK don't actually cause a navigation to occur - they instead forward on the message to the
// hosted web page. Real wizard navigations occur when the hosted web page calls our FinalBack() and FinalNext() methods.
case PSN_WIZNEXT:
_CallScript(_pwebbrowser, L"OnNext");
SetWindowLongPtr(hwnd, DWLP_MSGRESULT, (LONG_PTR) -1);
return TRUE;
case PSN_WIZBACK:
_CallScript(_pwebbrowser, L"OnBack");
SetWindowLongPtr(hwnd, DWLP_MSGRESULT, (LONG_PTR) -1);
return TRUE;
// query cancel results in a call to the site to determine if we are going
// to cancel out and if the site wants to provide a page for us to navigate
// to - in some cases, eg. the web publishing wizard this is important
// so that we can cancel the order being processed etc.
case PSN_QUERYCANCEL:
if (_punkSite)
{
IWizardSite *pws;
if (SUCCEEDED(_punkSite->QueryInterface(IID_PPV_ARG(IWizardSite, &pws))))
{
HPROPSHEETPAGE hpage;
if (S_OK == pws->GetCancelledPage(&hpage))
{
PropSheet_SetCurSel(GetParent(hwnd), hpage, -1);
SetWindowLongPtr(hwnd, DWLP_MSGRESULT, (LPARAM)TRUE);
}
pws->Release();
}
}
return TRUE;
case PSN_TRANSLATEACCELERATOR:
{
LPPSHNOTIFY ppsn = (LPPSHNOTIFY)pnmh;
MSG *pmsg = (MSG *)ppsn->lParam;
LONG_PTR lres = PSNRET_NOERROR;
if (_poipao && S_OK == _poipao->TranslateAccelerator(pmsg))
{
lres = PSNRET_MESSAGEHANDLED;
}
SetWindowLongPtr(hwnd, DWLP_MSGRESULT, lres);
}
break;
}
return TRUE;
}
BOOL CWebWizardPage::OnInitDialog(HWND hwnd, HWND hwndFocus, LPARAM lParam)
{
_hwnd = hwnd;
_hwndFrame = GetParent(hwnd);
// lets remap some of the text in the dialog if we need to
IResourceMap *prm;
HRESULT hr = IUnknown_QueryService(_punkSite, SID_ResourceMap, IID_PPV_ARG(IResourceMap, &prm));
if (SUCCEEDED(hr))
{
IXMLDOMNode *pdn;
hr = prm->SelectResourceScope(TEXT("dialog"), TEXT("ws:downloading"), &pdn);
if (SUCCEEDED(hr))
{
TCHAR szBuffer[512];
if (SUCCEEDED(prm->LoadString(pdn, TEXT("header"), szBuffer, ARRAYSIZE(szBuffer))))
{
SetDlgItemText(hwnd, IDC_PROGTEXT1, szBuffer);
}
if (SUCCEEDED(prm->LoadString(pdn, TEXT("footer"), szBuffer, ARRAYSIZE(szBuffer))))
{
SetDlgItemText(hwnd, IDC_PROGTEXT2, szBuffer);
}
pdn->Release();
}
prm->Release();
}
// create the web view browser that we will show the providers HTML in
SHDRC shdrc = {0};
shdrc.cbSize = sizeof(shdrc);
shdrc.dwFlags = SHDRCF_OCHOST;
if (SHDOCVW_DllRegisterWindowClasses(&shdrc))
{
RECT rectClient;
GetClientRect(hwnd, &rectClient);
_hwndOCHost = CreateWindow(OCHOST_CLASS, NULL,
WS_CHILD|WS_CLIPCHILDREN|WS_CLIPSIBLINGS|WS_TABSTOP,
0, 0, rectClient.right, rectClient.bottom,
hwnd, NULL, g_hinst, NULL);
if (_hwndOCHost)
{
OCHINITSTRUCT ocs = {0};
ocs.cbSize = sizeof(ocs);
ocs.clsidOC = CLSID_WebBrowser;
ocs.punkOwner = SAFECAST(this, IDocHostUIHandler*);
hr = OCHost_InitOC(_hwndOCHost, (LPARAM)&ocs);
if (SUCCEEDED(hr))
{
_InitBrowser();
OCHost_DoVerb(_hwndOCHost, OLEIVERB_INPLACEACTIVATE, TRUE);
ShowWindow(_hwndOCHost, TRUE);
IServiceProvider* pSP;
hr = _pwebEventHandler->QueryInterface(IID_PPV_ARG(IServiceProvider, &pSP));
if (SUCCEEDED(hr))
{
OCHost_SetServiceProvider(_hwndOCHost, pSP);
pSP->Release();
}
}
}
}
if (FAILED(hr))
EndDialog(hwnd, IDCANCEL);
return TRUE;
}
BOOL CWebWizardPage::OnTimer(HWND hwnd, UINT nIDEvent)
{
if (nIDEvent == SHOW_PROGRESS_TIMER)
{
_ShowProgress(TRUE);
}
return TRUE;
}
BOOL CWebWizardPage::OnDestroy(HWND hwnd)
{
ATOMICRELEASE(_pwebbrowser);
return TRUE;
}
void CWebWizardPage::_InitBrowser(void)
{
ASSERT(IsWindow(_hwndOCHost));
ASSERT(!_pwebbrowser);
HRESULT hr = OCHost_QueryInterface(_hwndOCHost, IID_PPV_ARG(IWebBrowser2, &_pwebbrowser));
if (SUCCEEDED(hr) && _pwebbrowser)
{
ITargetFrame2* ptgf;
if (SUCCEEDED(_pwebbrowser->QueryInterface(IID_PPV_ARG(ITargetFrame2, &ptgf))))
{
DWORD dwOptions;
if (SUCCEEDED(ptgf->GetFrameOptions(&dwOptions)))
{
dwOptions |= FRAMEOPTIONS_BROWSERBAND | FRAMEOPTIONS_SCROLL_AUTO;
ptgf->SetFrameOptions(dwOptions);
}
ptgf->Release();
}
_pwebbrowser->put_RegisterAsDropTarget(VARIANT_FALSE);
// Set up the connection point (including creating the object
if (!_pwebEventHandler)
CWebEventHandler_CreateInstance(this, &_pwebEventHandler);
if (_pwebEventHandler)
_pwebEventHandler->_Advise(TRUE);
OCHost_QueryInterface(_hwndOCHost, IID_PPV_ARG(IOleInPlaceActiveObject, &_poipao));
}
}
HRESULT CWebWizardPage::_NavigateBrowser(LPCWSTR pszUrl)
{
HRESULT hr = E_FAIL;
if (_hwndOCHost && _pwebbrowser)
{
BSTR bstrUrl = SysAllocString(pszUrl);
if (bstrUrl)
{
hr = _pwebbrowser->Navigate(bstrUrl, NULL, NULL, NULL, NULL);
SysFreeString(bstrUrl);
}
}
return hr;
}
HRESULT CWebWizardPage::FinalBack(void)
{
if (_punkSite)
{
IWizardSite *pws;
if (SUCCEEDED(_punkSite->QueryInterface(IID_PPV_ARG(IWizardSite, &pws))))
{
HPROPSHEETPAGE hpage;
HRESULT hr = pws->GetPreviousPage(&hpage);
if (SUCCEEDED(hr))
{
PropSheet_SetCurSel(_hwndFrame, hpage, -1);
}
pws->Release();
}
}
return S_OK;
}
HRESULT CWebWizardPage::FinalNext(void)
{
if (_punkSite)
{
IWizardSite *pws;
if (SUCCEEDED(_punkSite->QueryInterface(IID_PPV_ARG(IWizardSite, &pws))))
{
HPROPSHEETPAGE hpage;
HRESULT hr = pws->GetNextPage(&hpage);
if (SUCCEEDED(hr))
{
PropSheet_SetCurSel(_hwndFrame, hpage, -1);
}
pws->Release();
}
}
return S_OK;
}
HRESULT CWebWizardPage::Cancel(void)
{
PropSheet_PressButton(_hwndFrame, PSBTN_CANCEL); // simulate cancel...
return S_OK;
}
HRESULT CWebWizardPage::put_Caption(
/* [in] */ BSTR bstrCaption)
{
return S_OK;
}
HRESULT CWebWizardPage::get_Caption(
/* [retval][out] */ BSTR *pbstrCaption)
{
WCHAR szCaption[MAX_PATH];
GetWindowText(_hwndFrame, szCaption, ARRAYSIZE(szCaption));
*pbstrCaption = SysAllocString(szCaption);
return S_OK;
}
// fetch and put properties into the frames property bag. this we do
// by a QueryService call and then we can modify the properties accordingly.
HRESULT CWebWizardPage::put_Property(
/* [in] */ BSTR bstrPropertyName,
/* [in] */ VARIANT *pvProperty)
{
IPropertyBag *ppb;
HRESULT hr = IUnknown_QueryService(_punkSite, SID_WebWizardHost, IID_PPV_ARG(IPropertyBag, &ppb));
if (SUCCEEDED(hr))
{
hr = ppb->Write(bstrPropertyName, pvProperty);
ppb->Release();
}
return hr;
}
HRESULT CWebWizardPage::get_Property(
/* [in] */ BSTR bstrPropertyName,
/* [retval][out] */ VARIANT *pvProperty)
{
IPropertyBag *ppb;
HRESULT hr = IUnknown_QueryService(_punkSite, SID_WebWizardHost, IID_PPV_ARG(IPropertyBag, &ppb));
if (SUCCEEDED(hr))
{
hr = ppb->Read(bstrPropertyName, pvProperty, NULL);
if (FAILED(hr))
{
// Return a NULL-variant
VariantInit(pvProperty);
pvProperty->vt = VT_NULL;
hr = S_FALSE;
}
ppb->Release();
}
return hr;
}
HRESULT CWebWizardPage::SetWizardButtons(
/* [in] */ VARIANT_BOOL vfEnableBack,
/* [in] */ VARIANT_BOOL vfEnableNext,
/* [in] */ VARIANT_BOOL vfLastPage)
{
// We ignore vfLastPage because it isn't the last page for us!
DWORD dwButtons = 0;
if (vfEnableBack)
dwButtons |= PSWIZB_BACK;
if (vfEnableNext)
dwButtons |= PSWIZB_NEXT;
PropSheet_SetWizButtons(_hwndFrame, dwButtons);
return S_OK;
}
void CWebWizardPage::_SetHeaderText(LPCWSTR pszHeader, LPCWSTR pszSubHeader)
{
int iPageNumber = PropSheet_HwndToIndex(_hwndFrame, _hwnd);
if (-1 != iPageNumber)
{
PropSheet_SetHeaderTitle(_hwndFrame, iPageNumber, pszHeader);
PropSheet_SetHeaderSubTitle(_hwndFrame, iPageNumber, pszSubHeader);
}
}
HRESULT CWebWizardPage::SetHeaderText(
/* [in] */ BSTR bstrHeaderTitle,
/* [in] */ BSTR bstrHeaderSubtitle)
{
_SetHeaderText(bstrHeaderTitle, bstrHeaderSubtitle);
return S_OK;
}
HRESULT CWebWizardPage::PassportAuthenticate(
/* [in] */ BSTR bstrURL,
/* [retval][out] */ VARIANT_BOOL * pfAuthenticated)
{
*pfAuthenticated = VARIANT_FALSE; // the user isn't authenticated.
IXMLHttpRequest *preq;
HRESULT hr = CoCreateInstance(CLSID_XMLHTTPRequest, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IXMLHttpRequest, &preq));
if (SUCCEEDED(hr))
{
VARIANT varNULL = {0};
VARIANT varAsync = {VT_BOOL};
varAsync.boolVal = VARIANT_FALSE;
// open a post request to the destination that we have
hr = preq->open(L"GET", bstrURL, varAsync, varNULL, varNULL);
if (SUCCEEDED(hr))
{
VARIANT varBody = {0};
hr = preq->send(varBody);
if (SUCCEEDED(hr))
{
long lStatus;
hr = preq->get_status(&lStatus);
if (SUCCEEDED(hr) && (lStatus == HTTP_STATUS_OK))
{
*pfAuthenticated = VARIANT_TRUE;
}
}
}
}
return S_OK;
}
BOOL CWebWizardPage::_IsBrowserVisible()
{
return IsWindowVisible(_hwndOCHost);
}
void CWebWizardPage::_ShowBrowser(BOOL fShow)
{
ShowWindow(_hwndOCHost, fShow ? SW_SHOW : SW_HIDE);
if (fShow)
{
// Can't have these windows overlapping.
_ShowProgress(FALSE);
}
}
void CWebWizardPage::_StartShowProgressTimer()
{
_ShowProgress(FALSE);
if (!SetTimer(_hwnd, SHOW_PROGRESS_TIMER, SHOW_PROGRESS_TIMEOUT, NULL))
{
// Timer failed to set; show progress now;
_ShowProgress(TRUE);
}
}
// Size the progress bar to fit the client area it is in.
void CWebWizardPage::_SizeProgress()
{
HWND hwndProgress = GetDlgItem(_hwnd, IDC_PROGRESS);
RECT rcPage;
GetClientRect(_hwnd, &rcPage);
RECT rcProgress;
GetClientRect(hwndProgress, &rcProgress);
MapWindowPoints(hwndProgress, _hwnd, (LPPOINT) &rcProgress, 2);
rcProgress.right = rcPage.right - rcProgress.left;
SetWindowPos(hwndProgress, NULL, 0, 0, rcProgress.right - rcProgress.left, rcProgress.bottom - rcProgress.top, SWP_NOMOVE | SWP_NOZORDER | SWP_NOREPOSITION);
}
void CWebWizardPage::_ShowProgress(BOOL fShow)
{
HWND hwndProgress = GetDlgItem(_hwnd, IDC_PROGRESS);
ShowWindow(hwndProgress, fShow ? SW_SHOW : SW_HIDE);
ShowWindow(GetDlgItem(_hwnd, IDC_PROGTEXT1), fShow ? SW_SHOW : SW_HIDE);
ShowWindow(GetDlgItem(_hwnd, IDC_PROGTEXT2), fShow ? SW_SHOW : SW_HIDE);
KillTimer(_hwnd, SHOW_PROGRESS_TIMER);
if (fShow)
{
SendMessage(hwndProgress, PBM_SETMARQUEE, (WPARAM) TRUE, 0);
// Set the header/subheader to a "Connecting to Internet" message.
WCHAR szTitle[256];
WCHAR szSubtitle[256];
LoadString(g_hinst, IDS_WEBDLG_TITLE, szTitle, ARRAYSIZE(szTitle));
LoadString(g_hinst, IDS_WEBDLG_SUBTITLE, szSubtitle, ARRAYSIZE(szSubtitle));
_SetHeaderText(szTitle, szSubtitle);
}
else
{
SendMessage(hwndProgress, PBM_SETMARQUEE, (WPARAM) FALSE, 0);
}
}
// IWizardExtn
UINT CWebWizardPage::PropPageProc(HWND hwndDlg, UINT uMsg, PROPSHEETPAGE *ppsp)
{
CWebWizardPage *pwwp = (CWebWizardPage*)ppsp->lParam;
switch (uMsg)
{
case PSPCB_CREATE:
return TRUE;
// we need to release our site in this scenario, we know that we won't be using it
// anymore, and to ensure that clients down have a circular refernce to us we
// release it before they call us for our final destruction.
case PSPCB_RELEASE:
ATOMICRELEASE(pwwp->_punkSite);
break;
}
return FALSE;
}
HRESULT CWebWizardPage::AddPages(HPROPSHEETPAGE* aPages, UINT cPages, UINT *pnPages)
{
PROPSHEETPAGE psp = { 0 };
psp.dwSize = sizeof(psp);
psp.hInstance = g_hinst;
psp.dwFlags = PSP_DEFAULT|PSP_USECALLBACK ;
psp.pszTemplate = MAKEINTRESOURCE(DLG_WEBWIZARD);
psp.lParam = (LPARAM) this;
psp.pfnDlgProc = CWebWizardPage::StaticProc;
psp.pfnCallback = PropPageProc;
_hPage = CreatePropertySheetPage(&psp);
if (!_hPage)
return E_FAIL;
// return the page we created.
*aPages = _hPage;
*pnPages = 1;
return S_OK;
}
STDMETHODIMP CWebWizardPage::GetFirstPage(HPROPSHEETPAGE *phPage)
{
*phPage = _hPage;
return S_OK;
}
STDMETHODIMP CWebWizardPage::SetInitialURL(LPCWSTR pszDefaultURL)
{
HRESULT hr = E_INVALIDARG;
if (pszDefaultURL)
{
hr = Str_SetPtr(&_pszInitialURL, pszDefaultURL) ? S_OK:E_OUTOFMEMORY;
}
return hr;
}
STDMETHODIMP CWebWizardPage::SetErrorURL(LPCWSTR pszErrorURL)
{
HRESULT hr = E_INVALIDARG;
if (pszErrorURL)
{
hr = Str_SetPtr(&_pszErrorURL, pszErrorURL) ? S_OK:E_OUTOFMEMORY;
}
return hr;
}
STDAPI CWebWizardPage_CreateInstance(IUnknown* pUnkOuter, REFIID riid, void **ppv)
{
if (NULL != pUnkOuter)
{
return CLASS_E_NOAGGREGATION;
}
CWebWizardPage *pwwp = new CWebWizardPage();
if (!pwwp)
{
return E_OUTOFMEMORY;
}
HRESULT hr = pwwp->QueryInterface(riid, ppv);
pwwp->Release();
return hr;
}