#ifndef __ATLDLGS_H__
#define __ATLDLGS_H__
#pragma once
#ifndef __cplusplus
        #error ATL requires C++ compilation (use a .cpp suffix)
#endif
#ifndef __ATLAPP_H__
        #error atldlgs.h requires atlapp.h to be included first
#endif
#ifndef __ATLWIN_H__
        #error atldlgs.h requires atlwin.h to be included first
#endif
#include <commdlg.h>
#include <shlobj.h>
#if (_WIN32_WINNT >= 0x0600) && !defined(_WIN32_WCE)
  #include <shobjidl.h>
#endif 
namespace WTL
{
#if (_WIN32_WINNT >= 0x0500) && !defined(OPENFILENAME_SIZE_VERSION_400)
  #ifndef CDSIZEOF_STRUCT
    #define CDSIZEOF_STRUCT(structname, member)  (((int)((LPBYTE)(&((structname*)0)->member) - ((LPBYTE)((structname*)0)))) + sizeof(((structname*)0)->member))
  #endif
  #define OPENFILENAME_SIZE_VERSION_400A  CDSIZEOF_STRUCT(OPENFILENAMEA,lpTemplateName)
  #define OPENFILENAME_SIZE_VERSION_400W  CDSIZEOF_STRUCT(OPENFILENAMEW,lpTemplateName)
  #ifdef UNICODE
    #define OPENFILENAME_SIZE_VERSION_400  OPENFILENAME_SIZE_VERSION_400W
  #else
    #define OPENFILENAME_SIZE_VERSION_400  OPENFILENAME_SIZE_VERSION_400A
  #endif 
#endif 
#if !defined(_WIN32_WCE) && !defined(CDN_INCLUDEITEM)
  #define CDN_INCLUDEITEM         (CDN_FIRST - 0x0007)
#endif
template <class T>
class ATL_NO_VTABLE CFileDialogImpl : public ATL::CDialogImplBase
{
public:
#if defined(__AYGSHELL_H__) && (_WIN32_WCE >= 0x0501)
        OPENFILENAMEEX m_ofn;
#else
        OPENFILENAME m_ofn;
#endif
        BOOL m_bOpenFileDialog;            
        TCHAR m_szFileTitle[_MAX_FNAME];   
        TCHAR m_szFileName[_MAX_PATH];     
        CFileDialogImpl(BOOL bOpenFileDialog, 
                        LPCTSTR lpszDefExt = NULL,
                        LPCTSTR lpszFileName = NULL,
                        DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
                        LPCTSTR lpszFilter = NULL,
                        HWND hWndParent = NULL)
        {
                memset(&m_ofn, 0, sizeof(m_ofn)); 
                m_szFileName[0] = _T('\0');
                m_szFileTitle[0] = _T('\0');
                m_bOpenFileDialog = bOpenFileDialog;
                m_ofn.lStructSize = sizeof(m_ofn);
#if (_WIN32_WINNT >= 0x0500)
                
                if(AtlIsOldWindows())
                {
                        ATLASSERT(sizeof(m_ofn) > OPENFILENAME_SIZE_VERSION_400);   
                        m_ofn.lStructSize = OPENFILENAME_SIZE_VERSION_400;
                }
#endif 
                m_ofn.lpstrFile = m_szFileName;
                m_ofn.nMaxFile = _MAX_PATH;
                m_ofn.lpstrDefExt = lpszDefExt;
                m_ofn.lpstrFileTitle = (LPTSTR)m_szFileTitle;
                m_ofn.nMaxFileTitle = _MAX_FNAME;
#ifndef _WIN32_WCE
                m_ofn.Flags = dwFlags | OFN_EXPLORER | OFN_ENABLEHOOK | OFN_ENABLESIZING;
#else 
                m_ofn.Flags = dwFlags | OFN_EXPLORER | OFN_ENABLEHOOK;
#endif 
                m_ofn.lpstrFilter = lpszFilter;
                m_ofn.hInstance = ModuleHelper::GetResourceInstance();
                m_ofn.lpfnHook = (LPOFNHOOKPROC)T::StartDialogProc;
                m_ofn.hwndOwner = hWndParent;
                
                if(lpszFileName != NULL)
                SecureHelper::strncpy_x(m_szFileName, _countof(m_szFileName), lpszFileName, _TRUNCATE);
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                ATLASSERT((m_ofn.Flags & OFN_ENABLEHOOK) != 0);
                ATLASSERT(m_ofn.lpfnHook != NULL);   
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                if(m_ofn.hwndOwner == NULL)   
                        m_ofn.hwndOwner = hWndParent;
                ATLASSERT(m_hWnd == NULL);
                ModuleHelper::AddCreateWndData(&m_thunk.cd, (ATL::CDialogImplBase*)this);
                BOOL bRet;
                if(m_bOpenFileDialog)
#if defined(__AYGSHELL_H__) && (_WIN32_WCE >= 0x0501)
                        bRet = ::GetOpenFileNameEx(&m_ofn);
                else
                        bRet = ::GetSaveFileName((LPOPENFILENAME)&m_ofn);
#else
                        bRet = ::GetOpenFileName(&m_ofn);
                else
                        bRet = ::GetSaveFileName(&m_ofn);
#endif
                m_hWnd = NULL;
                return bRet ? IDOK : IDCANCEL;
        }
        ATL::CWindow GetFileDialogWindow() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return ATL::CWindow(GetParent());
        }
        int GetFilePath(LPTSTR lpstrFilePath, int nLength) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                return (int)GetFileDialogWindow().SendMessage(CDM_GETFILEPATH, nLength, (LPARAM)lpstrFilePath);
        }
        int GetFolderIDList(LPVOID lpBuff, int nLength) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                return (int)GetFileDialogWindow().SendMessage(CDM_GETFOLDERIDLIST, nLength, (LPARAM)lpBuff);
        }
        int GetFolderPath(LPTSTR lpstrFolderPath, int nLength) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                return (int)GetFileDialogWindow().SendMessage(CDM_GETFOLDERPATH, nLength, (LPARAM)lpstrFolderPath);
        }
        int GetSpec(LPTSTR lpstrSpec, int nLength) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                return (int)GetFileDialogWindow().SendMessage(CDM_GETSPEC, nLength, (LPARAM)lpstrSpec);
        }
        void SetControlText(int nCtrlID, LPCTSTR lpstrText)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                GetFileDialogWindow().SendMessage(CDM_SETCONTROLTEXT, nCtrlID, (LPARAM)lpstrText);
        }
        void SetDefExt(LPCTSTR lpstrExt)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                GetFileDialogWindow().SendMessage(CDM_SETDEFEXT, 0, (LPARAM)lpstrExt);
        }
        BOOL GetReadOnlyPref() const    
        {
                return ((m_ofn.Flags & OFN_READONLY) != 0) ? TRUE : FALSE;
        }
        void HideControl(int nCtrlID)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((m_ofn.Flags & OFN_EXPLORER) != 0);
                GetFileDialogWindow().SendMessage(CDM_HIDECONTROL, nCtrlID);
        }
        BOOL EndDialog(INT_PTR  = 0)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                GetFileDialogWindow().SendMessage(WM_COMMAND, MAKEWPARAM(IDCANCEL, 0));
                return TRUE;
        }
        BEGIN_MSG_MAP(CFileDialogImpl)
                NOTIFY_CODE_HANDLER(CDN_FILEOK, _OnFileOK)
                NOTIFY_CODE_HANDLER(CDN_FOLDERCHANGE, _OnFolderChange)
                NOTIFY_CODE_HANDLER(CDN_HELP, _OnHelp)
                NOTIFY_CODE_HANDLER(CDN_INITDONE, _OnInitDone)
                NOTIFY_CODE_HANDLER(CDN_SELCHANGE, _OnSelChange)
                NOTIFY_CODE_HANDLER(CDN_SHAREVIOLATION, _OnShareViolation)
                NOTIFY_CODE_HANDLER(CDN_TYPECHANGE, _OnTypeChange)
#ifndef _WIN32_WCE
                NOTIFY_CODE_HANDLER(CDN_INCLUDEITEM, _OnIncludeItem)
#endif 
        END_MSG_MAP()
        LRESULT _OnFileOK(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                return !pT->OnFileOK((LPOFNOTIFY)pnmh);
        }
        LRESULT _OnFolderChange(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                pT->OnFolderChange((LPOFNOTIFY)pnmh);
                return 0;
        }
        LRESULT _OnHelp(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                pT->OnHelp((LPOFNOTIFY)pnmh);
                return 0;
        }
        LRESULT _OnInitDone(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                pT->OnInitDone((LPOFNOTIFY)pnmh);
                return 0;
        }
        LRESULT _OnSelChange(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                pT->OnSelChange((LPOFNOTIFY)pnmh);
                return 0;
        }
        LRESULT _OnShareViolation(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                return pT->OnShareViolation((LPOFNOTIFY)pnmh);
        }
        LRESULT _OnTypeChange(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                pT->OnTypeChange((LPOFNOTIFY)pnmh);
                return 0;
        }
#ifndef _WIN32_WCE
        LRESULT _OnIncludeItem(int , LPNMHDR pnmh, BOOL& )
        {
                ATLASSERT(::IsWindow(m_hWnd));
                T* pT = static_cast<T*>(this);
                return pT->OnIncludeItem((LPOFNOTIFYEX)pnmh);
        }
#endif 
        BOOL OnFileOK(LPOFNOTIFY )
        {
                return TRUE;
        }
        void OnFolderChange(LPOFNOTIFY )
        {
        }
        void OnHelp(LPOFNOTIFY )
        {
        }
        void OnInitDone(LPOFNOTIFY )
        {
        }
        void OnSelChange(LPOFNOTIFY )
        {
        }
        int OnShareViolation(LPOFNOTIFY )
        {
                return 0;
        }
        void OnTypeChange(LPOFNOTIFY )
        {
        }
#ifndef _WIN32_WCE
        BOOL OnIncludeItem(LPOFNOTIFYEX )
        {
                return TRUE;   
        }
#endif 
};
class CFileDialog : public CFileDialogImpl<CFileDialog>
{
public:
        CFileDialog(BOOL bOpenFileDialog, 
                LPCTSTR lpszDefExt = NULL,
                LPCTSTR lpszFileName = NULL,
                DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
                LPCTSTR lpszFilter = NULL,
                HWND hWndParent = NULL)
                : CFileDialogImpl<CFileDialog>(bOpenFileDialog, lpszDefExt, lpszFileName, dwFlags, lpszFilter, hWndParent)
        { }
        
        DECLARE_EMPTY_MSG_MAP()
};
#if defined(__AYGSHELL_H__) && (_WIN32_WCE >= 0x0501)
class CFileDialogEx : public CFileDialogImpl<CFileDialogEx>
{
public:
        CFileDialogEx( 
                LPCTSTR lpszDefExt = NULL,
                LPCTSTR lpszFileName = NULL,
                DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
                OFN_EXFLAG ExFlags = OFN_EXFLAG_THUMBNAILVIEW,
                OFN_SORTORDER dwSortOrder = OFN_SORTORDER_AUTO,         
                LPCTSTR lpszFilter = NULL,
                HWND hWndParent = NULL)
                : CFileDialogImpl<CFileDialogEx>(TRUE, lpszDefExt, lpszFileName, dwFlags, lpszFilter, hWndParent)
        {
                m_ofn.ExFlags = ExFlags;
                m_ofn.dwSortOrder = dwSortOrder;
        }
        
        DECLARE_EMPTY_MSG_MAP()
};
#endif 
#ifndef _WIN32_WCE
#ifndef _WTL_FIXED_OFN_BUFFER_LENGTH
  #define _WTL_FIXED_OFN_BUFFER_LENGTH 0x10000
#endif
template <class T>
class ATL_NO_VTABLE CMultiFileDialogImpl : public CFileDialogImpl< T >
{
public:
        mutable LPCTSTR m_pNextFile; 
#ifndef _UNICODE
        bool m_bIsNT;
#endif
        CMultiFileDialogImpl(
                LPCTSTR lpszDefExt = NULL,
                LPCTSTR lpszFileName = NULL,
                DWORD dwFlags = OFN_HIDEREADONLY,
                LPCTSTR lpszFilter = NULL,
                HWND hWndParent = NULL)
                : CFileDialogImpl<T>(TRUE, lpszDefExt, lpszFileName, dwFlags, lpszFilter, hWndParent), 
                  m_pNextFile(NULL)
        {
                m_ofn.Flags |= OFN_ALLOWMULTISELECT;   
#ifndef _UNICODE
                OSVERSIONINFO ovi = { sizeof(ovi) };
                ::GetVersionEx(&ovi);
                m_bIsNT = (ovi.dwPlatformId == VER_PLATFORM_WIN32_NT);
                if (m_bIsNT)
                {
                        
                        
                        ATLVERIFY(ResizeFilenameBuffer(_WTL_FIXED_OFN_BUFFER_LENGTH));
                }
#endif
        }
        ~CMultiFileDialogImpl()
        {
                if (m_ofn.lpstrFile != m_szFileName)   
                        delete[] m_ofn.lpstrFile;
        }
        
        
        
        
        int GetDirectory(LPTSTR pBuffer, int nBufLen) const
        {
                if (m_ofn.lpstrFile == NULL)
                        return 0;
                LPCTSTR pStr = m_ofn.lpstrFile;
                int nLength = lstrlen(pStr);
                if (pStr[nLength + 1] == 0)
                {
                        
                        LPCTSTR pSep = _strrchr(pStr, _T('\\'));
                        if (pSep != NULL)
                                nLength = (int)(DWORD_PTR)(pSep - pStr);
                }
                int nRet = 0;
                if (pBuffer == NULL)   
                {
                        nRet = nLength + 1;
                }
                else if (nBufLen > nLength)
                {
                        SecureHelper::strncpy_x(pBuffer, nBufLen, pStr, nLength);
                        nRet = nLength;
                }
                return nRet;
        }
#if defined(_WTL_USE_CSTRING) || defined(__ATLSTR_H__)
        bool GetDirectory(_CSTRING_NS::CString& strDir) const
        {
                bool bRet = false;
                int nLength = GetDirectory(NULL, 0);
                if (nLength > 0)
                {
                        bRet = (GetDirectory(strDir.GetBuffer(nLength), nLength) > 0);
                        strDir.ReleaseBuffer(nLength - 1);
                }
                return bRet;
        }
#endif 
        
        LPCTSTR GetFirstFileName() const
        {
                if (m_ofn.lpstrFile == NULL)
                        return NULL;
                m_pNextFile = NULL;   
                LPCTSTR pStr = m_ofn.lpstrFile;
                int nLength = lstrlen(pStr);
                if (pStr[nLength + 1] != 0)
                {
                        
                        
                        pStr += nLength + 1;
                        
                        m_pNextFile = pStr;
                        GetNextFileName();
                }
                else
                {
                        
                        LPCTSTR pSep = _strrchr(pStr, _T('\\'));
                        if (pSep != NULL)
                                pStr = pSep + 1;
                }
                return pStr;
        }
        
        LPCTSTR GetNextFileName() const
        {
                if (m_pNextFile == NULL)
                        return NULL;
                LPCTSTR pStr = m_pNextFile;
                
                
                int nLength = lstrlen(pStr);
                m_pNextFile = (pStr[nLength + 1] != 0) ? &pStr[nLength + 1] : NULL;
                return pStr;
        }
        
        
        
        
        int GetFirstPathName(LPTSTR pBuffer, int nBufLen) const
        {
                LPCTSTR pStr = GetFirstFileName();
                int nLengthDir = GetDirectory(NULL, 0);
                if((pStr == NULL) || (nLengthDir == 0))
                        return 0;
                
                int nLengthTotal = nLengthDir + lstrlen(pStr);
                int nRet = 0;
                if(pBuffer == NULL) 
                {
                        nRet = nLengthTotal + 1;
                }
                else if (nBufLen > nLengthTotal) 
                {               
                        GetDirectory(pBuffer, nBufLen);
                        SecureHelper::strcat_x(pBuffer, nBufLen, _T("\\"));
                        SecureHelper::strcat_x(pBuffer, nBufLen, pStr);
                        nRet = nLengthTotal;
                }
                return nRet;
        }
#if defined(_WTL_USE_CSTRING) || defined(__ATLSTR_H__)
        bool GetFirstPathName(_CSTRING_NS::CString& strPath) const
        {
                bool bRet = false;
                int nLength = GetFirstPathName(NULL, 0);
                if (nLength > 0)
                {
                        bRet = (GetFirstPathName(strPath.GetBuffer(nLength), nLength) > 0);
                        strPath.ReleaseBuffer(nLength - 1);
                }
                return bRet;
        }
#endif 
        
        
        
        
        
        int GetNextPathName(LPTSTR pBuffer, int nBufLen) const
        {
                if (m_pNextFile == NULL)
                        return 0;
                int nRet = 0;
                LPCTSTR pStr = m_pNextFile;
                
                if (_strrchr(pStr, _T('\\')) != NULL)
                {
                        
                        int nLength = lstrlen(pStr);
                        if (pBuffer == NULL) 
                        {
                                nRet = nLength + 1;
                        }
                        else if (nBufLen > nLength) 
                        {
                                SecureHelper::strcpy_x(pBuffer, nBufLen, GetNextFileName());
                                nRet = nBufLen;
                        }
                }
                else
                {
                        
                        int nLengthDir = GetDirectory(NULL, 0);
                        if (nLengthDir > 0)
                        {
                                
                                int nLengthTotal = nLengthDir + lstrlen(pStr);
                                if(pBuffer == NULL) 
                                {
                                        nRet = nLengthTotal + 1;
                                }
                                else if (nBufLen > nLengthTotal) 
                                {
                                        GetDirectory(pBuffer, nBufLen);
                                        SecureHelper::strcat_x(pBuffer, nBufLen, _T("\\"));
                                        SecureHelper::strcat_x(pBuffer, nBufLen, GetNextFileName());
                                        nRet = nLengthTotal;
                                }
                        }
                }
                return nRet;
        }
#if defined(_WTL_USE_CSTRING) || defined(__ATLSTR_H__)
        bool GetNextPathName(_CSTRING_NS::CString& strPath) const
        {
                bool bRet = false;
                int nLength = GetNextPathName(NULL, 0);
                if (nLength > 0)
                {
                        bRet = (GetNextPathName(strPath.GetBuffer(nLength), nLength) > 0);
                        strPath.ReleaseBuffer(nLength - 1);
                }
                return bRet;
        }
#endif 
        bool ResizeFilenameBuffer(DWORD dwLength)
        {
                if (dwLength > m_ofn.nMaxFile)
                {
                        
                        if (m_ofn.lpstrFile != m_szFileName)
                        {
                                delete[] m_ofn.lpstrFile;
                                m_ofn.lpstrFile = NULL;
                                m_ofn.nMaxFile = 0;
                        }
                        
                        LPTSTR lpstrBuff = NULL;
                        ATLTRY(lpstrBuff = new TCHAR[dwLength]);
                        if (lpstrBuff != NULL)
                        {
                                m_ofn.lpstrFile = lpstrBuff;
                                m_ofn.lpstrFile[0] = 0;
                                m_ofn.nMaxFile = dwLength;
                        }
                }
                return (m_ofn.lpstrFile != NULL);
        }
        void OnSelChange(LPOFNOTIFY )
        {
#ifndef _UNICODE
                
                if (m_bIsNT)
                        return;
#endif
                
                int nLength = GetSpec(NULL, 0);
                if (nLength <= 1)
                        return; 
                
                
                nLength += GetFolderPath(NULL, 0) + 1;
                if (!ResizeFilenameBuffer(nLength))
                {
                        ATLASSERT(FALSE);
                        return;
                }
                
                if ((m_ofn.Flags & OFN_NODEREFERENCELINKS) != 0)
                        return;
                
                if (GetSpec(m_ofn.lpstrFile, m_ofn.nMaxFile) <= 0)
                        return;
                
                
                int nBytes = GetFolderIDList(NULL, 0);
                CTempBuffer<ITEMIDLIST> idlist;
                idlist.AllocateBytes(nBytes);
                if ((nBytes <= 0) || (GetFolderIDList(idlist, nBytes) <= 0))
                        return;
                
                ATL::CComPtr<IShellFolder> pDesktop, pFolder;
                if (FAILED(::SHGetDesktopFolder(&pDesktop)))
                        return;
                if (FAILED(pDesktop->BindToObject(idlist, NULL, IID_IShellFolder, (void**)&pFolder)))
                        return;
                
                
                DWORD nExtraChars = 0;
                bool bInsideQuotes = false;
                LPCTSTR pAnchor = m_ofn.lpstrFile;
                LPCTSTR pChar = m_ofn.lpstrFile;
                for ( ; *pChar; ++pChar)
                {
                        
                        if (*pChar == _T('\"'))
                        {
                                
                                bInsideQuotes = !bInsideQuotes;
                                
                                if (bInsideQuotes)
                                {
                                        
                                        pAnchor = pChar + 1;
                                }
                                else 
                                {
                                        
                                        if (pChar - pAnchor >= MAX_PATH)
                                                return;
                                        
                                        USES_CONVERSION;
                                        int nFileNameLength = (int)(DWORD_PTR)(pChar - pAnchor);
                                        TCHAR szFileName[MAX_PATH];
                                        SecureHelper::strncpy_x(szFileName, MAX_PATH, pAnchor, nFileNameLength);
                                        LPITEMIDLIST pidl = NULL;
                                        DWORD dwAttrib = SFGAO_LINK;
                                        if (SUCCEEDED(pFolder->ParseDisplayName(NULL, NULL, T2W(szFileName), NULL, &pidl, &dwAttrib)))
                                        {
                                                
                                                if (dwAttrib & SFGAO_LINK)
                                                {
                                                        
                                                        ATL::CComPtr<IShellLink> pLink;
                                                        if (SUCCEEDED(pFolder->BindToObject(pidl, NULL, IID_IShellLink, (void**)&pLink)))
                                                        {
                                                                
                                                                TCHAR szPath[MAX_PATH];
                                                                if (SUCCEEDED(pLink->GetPath(szPath, MAX_PATH, NULL, 0)))
                                                                {
                                                                        
                                                                        
                                                                        int nNewLength = lstrlen(szPath);
                                                                        if (nNewLength > nFileNameLength)
                                                                                nExtraChars += nNewLength - nFileNameLength;
                                                                }
                                                        }
                                                }
                                                
                                                ::CoTaskMemFree(pidl);
                                        }
                                }
                        }
                }
                
                if (nExtraChars > 0)
                        ATLVERIFY(ResizeFilenameBuffer(m_ofn.nMaxFile + nExtraChars));
        }
        
        static const TCHAR* _strrchr(const TCHAR* p, TCHAR ch)
        {
#ifndef _ATL_MIN_CRT
                return _tcsrchr(p, ch);
#else 
                const TCHAR* lpsz = NULL;
                while (*p != 0)
                {
                        if (*p == ch)
                                lpsz = p;
                        p = ::CharNext(p);
                }
                return lpsz;
#endif 
        }
};
class CMultiFileDialog : public CMultiFileDialogImpl<CMultiFileDialog>
{
public:
        CMultiFileDialog(
                LPCTSTR lpszDefExt = NULL,
                LPCTSTR lpszFileName = NULL,
                DWORD dwFlags = OFN_HIDEREADONLY,
                LPCTSTR lpszFilter = NULL,
                HWND hWndParent = NULL)
                : CMultiFileDialogImpl<CMultiFileDialog>(lpszDefExt, lpszFileName, dwFlags, lpszFilter, hWndParent)
        { }
        BEGIN_MSG_MAP(CMultiFileDialog)
                CHAIN_MSG_MAP(CMultiFileDialogImpl<CMultiFileDialog>)
        END_MSG_MAP()
};
#endif 
#if (_WIN32_WINNT >= 0x0600) && !defined(_WIN32_WCE)
template <class T>
class ATL_NO_VTABLE CShellFileDialogImpl : public IFileDialogEvents
{
public:
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                INT_PTR nRet = -1;
                T* pT = static_cast<T*>(this);
                if(pT->m_spFileDlg == NULL)
                {
                        ATLASSERT(FALSE);
                        return nRet;
                }
                DWORD dwCookie = 0;
                pT->_Advise(dwCookie);
                HRESULT hRet = pT->m_spFileDlg->Show(hWndParent);
                if(SUCCEEDED(hRet))
                        nRet = IDOK;
                else if(hRet == HRESULT_FROM_WIN32(ERROR_CANCELLED))
                        nRet = IDCANCEL;
                else
                        ATLASSERT(FALSE);   
                pT->_Unadvise(dwCookie);
                return nRet;
        }
        bool IsNull() const
        {
                const T* pT = static_cast<const T*>(this);
                return (pT->m_spFileDlg == NULL);
        }
        HRESULT GetFilePath(LPWSTR lpstrFilePath, int cchLength)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg != NULL);
                ATL::CComPtr<IShellItem> spItem;
                HRESULT hRet = pT->m_spFileDlg->GetResult(&spItem);
                if(SUCCEEDED(hRet))
                        hRet = GetFileNameFromShellItem(spItem, SIGDN_FILESYSPATH, lpstrFilePath, cchLength);
                return hRet;
        }
        HRESULT GetFileTitle(LPWSTR lpstrFileTitle, int cchLength)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg != NULL);
                ATL::CComPtr<IShellItem> spItem;
                HRESULT hRet = pT->m_spFileDlg->GetResult(&spItem);
                if(SUCCEEDED(hRet))
                        hRet = GetFileNameFromShellItem(spItem, SIGDN_NORMALDISPLAY, lpstrFileTitle, cchLength);
                return hRet;
        }
#if defined(_WTL_USE_CSTRING) || defined(__ATLSTR_H__)
        HRESULT GetFilePath(_CSTRING_NS::CString& strFilePath)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg != NULL);
                ATL::CComPtr<IShellItem> spItem;
                HRESULT hRet = pT->m_spFileDlg->GetResult(&spItem);
                if(SUCCEEDED(hRet))
                        hRet = GetFileNameFromShellItem(spItem, SIGDN_FILESYSPATH, strFilePath);
                return hRet;
        }
        HRESULT GetFileTitle(_CSTRING_NS::CString& strFileTitle)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg != NULL);
                ATL::CComPtr<IShellItem> spItem;
                HRESULT hRet = pT->m_spFileDlg->GetResult(&spItem);
                if(SUCCEEDED(hRet))
                        hRet = GetFileNameFromShellItem(spItem, SIGDN_NORMALDISPLAY, strFileTitle);
                return hRet;
        }
#endif 
        static HRESULT GetFileNameFromShellItem(IShellItem* pShellItem, SIGDN type, LPWSTR lpstr, int cchLength)
        {
                ATLASSERT(pShellItem != NULL);
                LPWSTR lpstrName = NULL;
                HRESULT hRet = pShellItem->GetDisplayName(type, &lpstrName);
                if(SUCCEEDED(hRet))
                {
                        if(lstrlenW(lpstrName) < cchLength)
                        {
                                SecureHelper::strcpyW_x(lpstr, cchLength, lpstrName);
                        }
                        else
                        {
                                ATLASSERT(FALSE);
                                hRet = DISP_E_BUFFERTOOSMALL;
                        }
                        ::CoTaskMemFree(lpstrName);
                }
                return hRet;
        }
#if defined(_WTL_USE_CSTRING) || defined(__ATLSTR_H__)
        static HRESULT GetFileNameFromShellItem(IShellItem* pShellItem, SIGDN type, _CSTRING_NS::CString& str)
        {
                ATLASSERT(pShellItem != NULL);
                LPWSTR lpstrName = NULL;
                HRESULT hRet = pShellItem->GetDisplayName(type, &lpstrName);
                if(SUCCEEDED(hRet))
                {
                        str = lpstrName;
                        ::CoTaskMemFree(lpstrName);
                }
                return hRet;
        }
#endif 
        void _Advise(DWORD& dwCookie)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg != NULL);
                HRESULT hRet = pT->m_spFileDlg->Advise((IFileDialogEvents*)this, &dwCookie);
                ATLVERIFY(SUCCEEDED(hRet));
        }
        void _Unadvise(DWORD dwCookie)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg != NULL);
                HRESULT hRet = pT->m_spFileDlg->Unadvise(dwCookie);
                ATLVERIFY(SUCCEEDED(hRet));
        }
        void _Init(LPCWSTR lpszFileName, DWORD dwOptions, LPCWSTR lpszDefExt, const COMDLG_FILTERSPEC* arrFilterSpec, UINT uFilterSpecCount)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg != NULL);
                HRESULT hRet = E_FAIL;
                if(lpszFileName != NULL)
                {
                        hRet = pT->m_spFileDlg->SetFileName(lpszFileName);
                        ATLASSERT(SUCCEEDED(hRet));
                }
                hRet = pT->m_spFileDlg->SetOptions(dwOptions);
                ATLASSERT(SUCCEEDED(hRet));
                if(lpszDefExt != NULL)
                {
                        hRet = pT->m_spFileDlg->SetDefaultExtension(lpszDefExt);
                        ATLASSERT(SUCCEEDED(hRet));
                }
                if(arrFilterSpec != NULL && uFilterSpecCount != 0U)
                {
                        hRet = pT->m_spFileDlg->SetFileTypes(uFilterSpecCount, arrFilterSpec);
                        ATLASSERT(SUCCEEDED(hRet));
                }
        }
        STDMETHOD(QueryInterface)(REFIID riid, void** ppvObject)
        {
                if(ppvObject == NULL)
                        return E_POINTER;
                T* pT = static_cast<T*>(this);
                if(IsEqualGUID(riid, IID_IUnknown) || IsEqualGUID(riid, IID_IFileDialogEvents))
                {
                        *ppvObject = (IFileDialogEvents*)pT;
                        
                        return S_OK;
                }
                return E_NOINTERFACE;
        }
        virtual ULONG STDMETHODCALLTYPE AddRef()
        {
                return 1;
        }
        virtual ULONG STDMETHODCALLTYPE Release()
        {
                return 1;
        }
        virtual HRESULT STDMETHODCALLTYPE IFileDialogEvents::OnFileOk(IFileDialog* pfd)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg.IsEqualObject(pfd));
                pfd;   
                return pT->OnFileOk();
        }
        virtual HRESULT STDMETHODCALLTYPE IFileDialogEvents::OnFolderChanging(IFileDialog* pfd, IShellItem* psiFolder)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg.IsEqualObject(pfd));
                pfd;   
                return pT->OnFolderChanging(psiFolder);
        }
        virtual HRESULT STDMETHODCALLTYPE IFileDialogEvents::OnFolderChange(IFileDialog* pfd)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg.IsEqualObject(pfd));
                pfd;   
                return pT->OnFolderChange();
        }
        virtual HRESULT STDMETHODCALLTYPE IFileDialogEvents::OnSelectionChange(IFileDialog* pfd)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg.IsEqualObject(pfd));
                pfd;   
                return pT->OnSelectionChange();
        }
        virtual HRESULT STDMETHODCALLTYPE IFileDialogEvents::OnShareViolation(IFileDialog* pfd, IShellItem* psi, FDE_SHAREVIOLATION_RESPONSE* pResponse)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg.IsEqualObject(pfd));
                pfd;   
                return pT->OnShareViolation(psi, pResponse);
        }
        virtual HRESULT STDMETHODCALLTYPE IFileDialogEvents::OnTypeChange(IFileDialog* pfd)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg.IsEqualObject(pfd));
                pfd;   
                return pT->OnTypeChange();
        }
        virtual HRESULT STDMETHODCALLTYPE IFileDialogEvents::OnOverwrite(IFileDialog* pfd, IShellItem* psi, FDE_OVERWRITE_RESPONSE* pResponse)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_spFileDlg.IsEqualObject(pfd));
                pfd;   
                return pT->OnOverwrite(psi, pResponse);
        }
        HRESULT OnFileOk()
        {
                return E_NOTIMPL;
        }
        HRESULT OnFolderChanging(IShellItem* )
        {
                return E_NOTIMPL;
        }
        HRESULT OnFolderChange()
        {
                return E_NOTIMPL;
        }
        HRESULT OnSelectionChange()
        {
                return E_NOTIMPL;
        }
        HRESULT OnShareViolation(IShellItem* , FDE_SHAREVIOLATION_RESPONSE* )
        {
                return E_NOTIMPL;
        }
        HRESULT OnTypeChange()
        {
                return E_NOTIMPL;
        }
        HRESULT OnOverwrite(IShellItem* , FDE_OVERWRITE_RESPONSE* )
        {
                return E_NOTIMPL;
        }
};
template <class T>
class ATL_NO_VTABLE CShellFileOpenDialogImpl : public CShellFileDialogImpl< T >
{
public:
        ATL::CComPtr<IFileOpenDialog> m_spFileDlg;
        CShellFileOpenDialogImpl(LPCWSTR lpszFileName = NULL, 
                                 DWORD dwOptions = FOS_FORCEFILESYSTEM | FOS_PATHMUSTEXIST | FOS_FILEMUSTEXIST, 
                                 LPCWSTR lpszDefExt = NULL, 
                                 const COMDLG_FILTERSPEC* arrFilterSpec = NULL, 
                                 UINT uFilterSpecCount = 0U)
        {
                HRESULT hRet = m_spFileDlg.CoCreateInstance(CLSID_FileOpenDialog);
                if(SUCCEEDED(hRet))
                        _Init(lpszFileName, dwOptions, lpszDefExt, arrFilterSpec, uFilterSpecCount);
        }
        IFileOpenDialog* GetPtr()
        {
                return m_spFileDlg;
        }
};
class CShellFileOpenDialog : public CShellFileOpenDialogImpl<CShellFileOpenDialog>
{
public:
        CShellFileOpenDialog(LPCWSTR lpszFileName = NULL, 
                             DWORD dwOptions = FOS_FORCEFILESYSTEM | FOS_PATHMUSTEXIST | FOS_FILEMUSTEXIST, 
                             LPCWSTR lpszDefExt = NULL, 
                             const COMDLG_FILTERSPEC* arrFilterSpec = NULL, 
                             UINT uFilterSpecCount = 0U) : CShellFileOpenDialogImpl<CShellFileOpenDialog>(lpszFileName, dwOptions, lpszDefExt, arrFilterSpec, uFilterSpecCount)
        { }
        void _Advise(DWORD& )
        { }
        void _Unadvise(DWORD )
        { }
};
template <class T>
class ATL_NO_VTABLE CShellFileSaveDialogImpl : public CShellFileDialogImpl< T >
{
public:
        ATL::CComPtr<IFileSaveDialog> m_spFileDlg;
        CShellFileSaveDialogImpl(LPCWSTR lpszFileName = NULL, 
                                 DWORD dwOptions = FOS_FORCEFILESYSTEM | FOS_PATHMUSTEXIST | FOS_OVERWRITEPROMPT, 
                                 LPCWSTR lpszDefExt = NULL, 
                                 const COMDLG_FILTERSPEC* arrFilterSpec = NULL, 
                                 UINT uFilterSpecCount = 0U)
        {
                HRESULT hRet = m_spFileDlg.CoCreateInstance(CLSID_FileSaveDialog);
                if(SUCCEEDED(hRet))
                        _Init(lpszFileName, dwOptions, lpszDefExt, arrFilterSpec, uFilterSpecCount);
        }
        IFileSaveDialog* GetPtr()
        {
                return m_spFileDlg;
        }
};
class CShellFileSaveDialog : public CShellFileSaveDialogImpl<CShellFileSaveDialog>
{
public:
        CShellFileSaveDialog(LPCWSTR lpszFileName = NULL, 
                             DWORD dwOptions = FOS_FORCEFILESYSTEM | FOS_PATHMUSTEXIST | FOS_OVERWRITEPROMPT, 
                             LPCWSTR lpszDefExt = NULL, 
                             const COMDLG_FILTERSPEC* arrFilterSpec = NULL, 
                             UINT uFilterSpecCount = 0U) : CShellFileSaveDialogImpl<CShellFileSaveDialog>(lpszFileName, dwOptions, lpszDefExt, arrFilterSpec, uFilterSpecCount)
        { }
        void _Advise(DWORD& )
        { }
        void _Unadvise(DWORD )
        { }
};
#endif 
#ifndef _WIN32_WCE
template <class T>
class ATL_NO_VTABLE CFolderDialogImpl
{
public:
        BROWSEINFO m_bi;
        LPCTSTR m_lpstrInitialFolder;
        LPCITEMIDLIST m_pidlInitialSelection;
        bool m_bExpandInitialSelection;
        TCHAR m_szFolderDisplayName[MAX_PATH];
        TCHAR m_szFolderPath[MAX_PATH];
        LPITEMIDLIST m_pidlSelected;
        HWND m_hWnd;   
        CFolderDialogImpl(HWND hWndParent = NULL, LPCTSTR lpstrTitle = NULL, UINT uFlags = BIF_RETURNONLYFSDIRS) : 
                        m_lpstrInitialFolder(NULL), m_pidlInitialSelection(NULL), m_bExpandInitialSelection(false), m_pidlSelected(NULL), m_hWnd(NULL)
        {
                memset(&m_bi, 0, sizeof(m_bi)); 
                m_bi.hwndOwner = hWndParent;
                m_bi.pidlRoot = NULL;
                m_bi.pszDisplayName = m_szFolderDisplayName;
                m_bi.lpszTitle = lpstrTitle;
                m_bi.ulFlags = uFlags;
                m_bi.lpfn = BrowseCallbackProc;
                m_bi.lParam = (LPARAM)static_cast<T*>(this);
                m_szFolderPath[0] = 0;
                m_szFolderDisplayName[0] = 0;
        }
        ~CFolderDialogImpl()
        {
                ::CoTaskMemFree(m_pidlSelected);
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                if(m_bi.hwndOwner == NULL)   
                        m_bi.hwndOwner = hWndParent;
                
                m_szFolderPath[0] = 0;
                m_szFolderDisplayName[0] = 0;
                ::CoTaskMemFree(m_pidlSelected);
                INT_PTR nRet = IDCANCEL;
                m_pidlSelected = ::SHBrowseForFolder(&m_bi);
                if(m_pidlSelected != NULL)
                {
                        nRet = IDOK;
                        
                        
                        if((m_bi.ulFlags & BIF_RETURNONLYFSDIRS) != 0)
                        {
                                if(::SHGetPathFromIDList(m_pidlSelected, m_szFolderPath) == FALSE)
                                        nRet = IDCANCEL;
                        }
                }
                return nRet;
        }
        
        void SetInitialFolder(LPCTSTR lpstrInitialFolder, bool bExpand = true)
        {
                
                m_lpstrInitialFolder = lpstrInitialFolder;
                m_bExpandInitialSelection = bExpand;
        }
        void SetInitialSelection(LPCITEMIDLIST pidl, bool bExpand = true)
        {
                m_pidlInitialSelection = pidl;
                m_bExpandInitialSelection = bExpand;
        }
        
        LPITEMIDLIST GetSelectedItem(bool bDetach = false)
        {
                LPITEMIDLIST pidl = m_pidlSelected;
                if(bDetach)
                        m_pidlSelected = NULL;
                return pidl;
        }
        LPCTSTR GetFolderPath() const
        {
                return m_szFolderPath;
        }
        LPCTSTR GetFolderDisplayName() const
        {
                return m_szFolderDisplayName;
        }
        int GetFolderImageIndex() const
        {
                return m_bi.iImage;
        }
        static int CALLBACK BrowseCallbackProc(HWND hWnd, UINT uMsg, LPARAM lParam, LPARAM lpData)
        {
#ifndef BFFM_VALIDATEFAILED
  #ifdef UNICODE
                const int BFFM_VALIDATEFAILED = 4;
  #else
                const int BFFM_VALIDATEFAILED = 3;
  #endif
#endif 
#ifndef BFFM_IUNKNOWN
                const int BFFM_IUNKNOWN = 5;
#endif 
#ifndef BIF_NEWDIALOGSTYLE
                const UINT BIF_NEWDIALOGSTYLE = 0x0040;
#endif 
                int nRet = 0;
                T* pT = (T*)lpData;
                bool bClear = false;
                if(pT->m_hWnd == NULL)
                {
                        pT->m_hWnd = hWnd;
                        bClear = true;
                }
                else
                {
                        ATLASSERT(pT->m_hWnd == hWnd);
                }
                switch(uMsg)
                {
                case BFFM_INITIALIZED:
                        
                        
                        if(pT->m_pidlInitialSelection != NULL)
                                pT->SetSelection(pT->m_pidlInitialSelection);
                        else if(pT->m_lpstrInitialFolder != NULL)
                                pT->SetSelection(pT->m_lpstrInitialFolder);
                        
                        if(pT->m_bExpandInitialSelection && ((pT->m_bi.ulFlags & BIF_NEWDIALOGSTYLE) != 0))
                        {
                                if(pT->m_pidlInitialSelection != NULL)
                                        pT->SetExpanded(pT->m_pidlInitialSelection);
                                else if(pT->m_lpstrInitialFolder != NULL)
                                        pT->SetExpanded(pT->m_lpstrInitialFolder);
                        }
                        pT->OnInitialized();
                        break;
                case BFFM_SELCHANGED:
                        pT->OnSelChanged((LPITEMIDLIST)lParam);
                        break;
                case BFFM_VALIDATEFAILED:
                        nRet = pT->OnValidateFailed((LPCTSTR)lParam);
                        break;
                case BFFM_IUNKNOWN:
                        pT->OnIUnknown((IUnknown*)lParam);
                        break;
                default:
                        ATLTRACE2(atlTraceUI, 0, _T("Unknown message received in CFolderDialogImpl::BrowseCallbackProc\n"));
                        break;
                }
                if(bClear)
                        pT->m_hWnd = NULL;
                return nRet;
        }
        void OnInitialized()
        {
        }
        void OnSelChanged(LPITEMIDLIST )
        {
        }
        int OnValidateFailed(LPCTSTR )
        {
                return 1;   
        }
        void OnIUnknown(IUnknown* )
        {
        }
        
        void EnableOK(BOOL bEnable)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, BFFM_ENABLEOK, 0, bEnable);
        }
        void SetSelection(LPCITEMIDLIST pItemIDList)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, BFFM_SETSELECTION, FALSE, (LPARAM)pItemIDList);
        }
        void SetSelection(LPCTSTR lpstrFolderPath)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, BFFM_SETSELECTION, TRUE, (LPARAM)lpstrFolderPath);
        }
        void SetStatusText(LPCTSTR lpstrText)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, BFFM_SETSTATUSTEXT, 0, (LPARAM)lpstrText);
        }
        void SetOKText(LPCTSTR lpstrOKText)
        {
#ifndef BFFM_SETOKTEXT
                const UINT BFFM_SETOKTEXT = WM_USER + 105;
#endif
                ATLASSERT(m_hWnd != NULL);
                USES_CONVERSION;
                LPCWSTR lpstr = T2CW(lpstrOKText);
                ::SendMessage(m_hWnd, BFFM_SETOKTEXT, (WPARAM)lpstr, 0L);
        }
        void SetExpanded(LPCITEMIDLIST pItemIDList)
        {
#ifndef BFFM_SETEXPANDED
                const UINT BFFM_SETEXPANDED = WM_USER + 106;
#endif
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, BFFM_SETEXPANDED, FALSE, (LPARAM)pItemIDList);
        }
        void SetExpanded(LPCTSTR lpstrFolderPath)
        {
#ifndef BFFM_SETEXPANDED
                const UINT BFFM_SETEXPANDED = WM_USER + 106;
#endif
                ATLASSERT(m_hWnd != NULL);
                USES_CONVERSION;
                LPCWSTR lpstr = T2CW(lpstrFolderPath);
                ::SendMessage(m_hWnd, BFFM_SETEXPANDED, TRUE, (LPARAM)lpstr);
        }
};
class CFolderDialog : public CFolderDialogImpl<CFolderDialog>
{
public:
        CFolderDialog(HWND hWndParent = NULL, LPCTSTR lpstrTitle = NULL, UINT uFlags = BIF_RETURNONLYFSDIRS)
                : CFolderDialogImpl<CFolderDialog>(hWndParent, lpstrTitle, uFlags)
        { }
};
#endif 
class ATL_NO_VTABLE CCommonDialogImplBase : public ATL::CWindowImplBase
{
public:
        static UINT_PTR APIENTRY HookProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
        {
                if(uMsg != WM_INITDIALOG)
                        return 0;
                CCommonDialogImplBase* pT = (CCommonDialogImplBase*)ModuleHelper::ExtractCreateWndData();
                ATLASSERT(pT != NULL);
                ATLASSERT(pT->m_hWnd == NULL);
                ATLASSERT(::IsWindow(hWnd));
                
                if(!pT->SubclassWindow(hWnd))
                {
                        ATLTRACE2(atlTraceUI, 0, _T("Subclassing a common dialog failed\n"));
                        return 0;
                }
                
                LRESULT lRes = 0;
                if(pT->ProcessWindowMessage(pT->m_hWnd, uMsg, wParam, lParam, lRes, 0) == FALSE)
                        return 0;
                return lRes;
        }
        BOOL EndDialog(INT_PTR  = 0)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                SendMessage(WM_COMMAND, MAKEWPARAM(IDABORT, 0));
                return TRUE;
        }
        HWND Create(HWND, ATL::_U_RECT, LPCTSTR, DWORD, DWORD, ATL::_U_MENUorID, ATOM, LPVOID)
        {
                ATLASSERT(FALSE);   
                return NULL;
        }
        static LRESULT CALLBACK StartWindowProc(HWND , UINT , WPARAM , LPARAM )
        {
                ATLASSERT(FALSE);   
                return 0;
        }
};
#ifndef _WIN32_WCE
template <class T>
class ATL_NO_VTABLE CFontDialogImpl : public CCommonDialogImplBase
{
public:
        enum { _cchStyleName = 64 };
        CHOOSEFONT m_cf;
        TCHAR m_szStyleName[_cchStyleName];  
        LOGFONT m_lf;                        
        CFontDialogImpl(LPLOGFONT lplfInitial = NULL,
                        DWORD dwFlags = CF_EFFECTS | CF_SCREENFONTS,
                        HDC hDCPrinter = NULL,
                        HWND hWndParent = NULL)
        {
                memset(&m_cf, 0, sizeof(m_cf));
                memset(&m_lf, 0, sizeof(m_lf));
                memset(&m_szStyleName, 0, sizeof(m_szStyleName));
                m_cf.lStructSize = sizeof(m_cf);
                m_cf.hwndOwner = hWndParent;
                m_cf.rgbColors = RGB(0, 0, 0);
                m_cf.lpszStyle = (LPTSTR)&m_szStyleName;
                m_cf.Flags = dwFlags | CF_ENABLEHOOK;
                m_cf.lpfnHook = (LPCFHOOKPROC)T::HookProc;
                if(lplfInitial != NULL)
                {
                        m_cf.lpLogFont = lplfInitial;
                        m_cf.Flags |= CF_INITTOLOGFONTSTRUCT;
                        m_lf = *lplfInitial;
                }
                else
                {
                        m_cf.lpLogFont = &m_lf;
                }
                if(hDCPrinter != NULL)
                {
                        m_cf.hDC = hDCPrinter;
                        m_cf.Flags |= CF_PRINTERFONTS;
                }
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                ATLASSERT((m_cf.Flags & CF_ENABLEHOOK) != 0);
                ATLASSERT(m_cf.lpfnHook != NULL);   
                if(m_cf.hwndOwner == NULL)          
                        m_cf.hwndOwner = hWndParent;
                ATLASSERT(m_hWnd == NULL);
                ModuleHelper::AddCreateWndData(&m_thunk.cd, (CCommonDialogImplBase*)this);
                BOOL bRet = ::ChooseFont(&m_cf);
                m_hWnd = NULL;
                if(bRet)   
                        SecureHelper::memcpy_x(&m_lf, sizeof(m_lf), m_cf.lpLogFont, sizeof(m_lf));
                return bRet ? IDOK : IDCANCEL;
        }
        
        void GetCurrentFont(LPLOGFONT lplf) const
        {
                ATLASSERT(lplf != NULL);
                if(m_hWnd != NULL)
                        ::SendMessage(m_hWnd, WM_CHOOSEFONT_GETLOGFONT, 0, (LPARAM)lplf);
                else
                        *lplf = m_lf;
        }
        
#ifndef _WIN32_WCE
        void SetLogFont(LPLOGFONT lplf)
        {
                ATLASSERT(lplf != NULL);
#ifndef WM_CHOOSEFONT_SETLOGFONT
                const UINT WM_CHOOSEFONT_SETLOGFONT = (WM_USER + 101);
#endif
                if(m_hWnd != NULL)
                {
                        ::SendMessage(m_hWnd, WM_CHOOSEFONT_SETLOGFONT, 0, (LPARAM)lplf);
                }
                else
                {
                        m_lf = *lplf;
                        m_cf.Flags |= CF_INITTOLOGFONTSTRUCT;
                }
        }
        void SetFlags(DWORD dwFlags)
        {
#ifndef WM_CHOOSEFONT_SETFLAGS
                const UINT WM_CHOOSEFONT_SETFLAGS = (WM_USER + 102);
#endif
                if(m_hWnd != NULL)
                {
                        CHOOSEFONT cf = { sizeof(CHOOSEFONT) };
                        cf.Flags = dwFlags;
                        ::SendMessage(m_hWnd, WM_CHOOSEFONT_SETFLAGS, 0, (LPARAM)&cf);
                }
                else
                {
                        m_cf.Flags = dwFlags;
                }
        }
#endif 
        
        LPCTSTR GetFaceName() const   
        {
                return (LPCTSTR)m_cf.lpLogFont->lfFaceName;
        }
        LPCTSTR GetStyleName() const  
        {
                return m_cf.lpszStyle;
        }
        int GetSize() const           
        {
                return m_cf.iPointSize;
        }
        COLORREF GetColor() const     
        {
                return m_cf.rgbColors;
        }
        int GetWeight() const         
        {
                return (int)m_cf.lpLogFont->lfWeight;
        }
        BOOL IsStrikeOut() const      
        {
                return (m_cf.lpLogFont->lfStrikeOut) ? TRUE : FALSE;
        }
        BOOL IsUnderline() const      
        {
                return (m_cf.lpLogFont->lfUnderline) ? TRUE : FALSE;
        }
        BOOL IsBold() const           
        {
                return (m_cf.lpLogFont->lfWeight == FW_BOLD) ? TRUE : FALSE;
        }
        BOOL IsItalic() const         
        {
                return m_cf.lpLogFont->lfItalic ? TRUE : FALSE;
        }
};
class CFontDialog : public CFontDialogImpl<CFontDialog>
{
public:
        CFontDialog(LPLOGFONT lplfInitial = NULL,
                DWORD dwFlags = CF_EFFECTS | CF_SCREENFONTS,
                HDC hDCPrinter = NULL,
                HWND hWndParent = NULL)
                : CFontDialogImpl<CFontDialog>(lplfInitial, dwFlags, hDCPrinter, hWndParent)
        { }
        DECLARE_EMPTY_MSG_MAP()
};
#endif 
#if defined(_RICHEDIT_) && !defined(_WIN32_WCE)
template <class T>
class ATL_NO_VTABLE CRichEditFontDialogImpl : public CFontDialogImpl< T >
{
public:
        CRichEditFontDialogImpl(const CHARFORMAT& charformat,
                        DWORD dwFlags = CF_SCREENFONTS,
                        HDC hDCPrinter = NULL,
                        HWND hWndParent = NULL)
                        : CFontDialogImpl< T >(NULL, dwFlags, hDCPrinter, hWndParent)
        {
                m_cf.Flags |= CF_INITTOLOGFONTSTRUCT;
                m_cf.Flags |= FillInLogFont(charformat);
                m_cf.lpLogFont = &m_lf;
                if((charformat.dwMask & CFM_COLOR) != 0)
                        m_cf.rgbColors = charformat.crTextColor;
        }
        void GetCharFormat(CHARFORMAT& cf) const
        {
                USES_CONVERSION;
                cf.dwEffects = 0;
                cf.dwMask = 0;
                if((m_cf.Flags & CF_NOSTYLESEL) == 0)
                {
                        cf.dwMask |= CFM_BOLD | CFM_ITALIC;
                        cf.dwEffects |= IsBold() ? CFE_BOLD : 0;
                        cf.dwEffects |= IsItalic() ? CFE_ITALIC : 0;
                }
                if((m_cf.Flags & CF_NOSIZESEL) == 0)
                {
                        cf.dwMask |= CFM_SIZE;
                        
                        cf.yHeight = GetSize() * 2;
                }
                if((m_cf.Flags & CF_NOFACESEL) == 0)
                {
                        cf.dwMask |= CFM_FACE;
                        cf.bPitchAndFamily = m_cf.lpLogFont->lfPitchAndFamily;
#if (_RICHEDIT_VER >= 0x0200)
                        SecureHelper::strcpy_x(cf.szFaceName, _countof(cf.szFaceName), GetFaceName());
#else 
                        SecureHelper::strcpyA_x(cf.szFaceName, _countof(cf.szFaceName), T2A((LPTSTR)(LPCTSTR)GetFaceName()));
#endif 
                }
                if((m_cf.Flags & CF_EFFECTS) != 0)
                {
                        cf.dwMask |= CFM_UNDERLINE | CFM_STRIKEOUT | CFM_COLOR;
                        cf.dwEffects |= IsUnderline() ? CFE_UNDERLINE : 0;
                        cf.dwEffects |= IsStrikeOut() ? CFE_STRIKEOUT : 0;
                        cf.crTextColor = GetColor();
                }
                if((m_cf.Flags & CF_NOSCRIPTSEL) == 0)
                {
                        cf.bCharSet = m_cf.lpLogFont->lfCharSet;
                        cf.dwMask |= CFM_CHARSET;
                }
                cf.yOffset = 0;
        }
        DWORD FillInLogFont(const CHARFORMAT& cf)
        {
                USES_CONVERSION;
                DWORD dwFlags = 0;
                if((cf.dwMask & CFM_SIZE) != 0)
                {
                        HDC hDC = ::CreateDC(_T("DISPLAY"), NULL, NULL, NULL);
                        LONG yPerInch = ::GetDeviceCaps(hDC, LOGPIXELSY);
                        m_lf.lfHeight = -(int)((cf.yHeight * yPerInch) / 1440);
                }
                else
                        m_lf.lfHeight = 0;
                m_lf.lfWidth = 0;
                m_lf.lfEscapement = 0;
                m_lf.lfOrientation = 0;
                if((cf.dwMask & (CFM_ITALIC | CFM_BOLD)) == (CFM_ITALIC | CFM_BOLD))
                {
                        m_lf.lfWeight = ((cf.dwEffects & CFE_BOLD) != 0) ? FW_BOLD : FW_NORMAL;
                        m_lf.lfItalic = (BYTE)(((cf.dwEffects & CFE_ITALIC) != 0) ? TRUE : FALSE);
                }
                else
                {
                        dwFlags |= CF_NOSTYLESEL;
                        m_lf.lfWeight = FW_DONTCARE;
                        m_lf.lfItalic = FALSE;
                }
                if((cf.dwMask & (CFM_UNDERLINE | CFM_STRIKEOUT | CFM_COLOR)) == (CFM_UNDERLINE|CFM_STRIKEOUT|CFM_COLOR))
                {
                        dwFlags |= CF_EFFECTS;
                        m_lf.lfUnderline = (BYTE)(((cf.dwEffects & CFE_UNDERLINE) != 0) ? TRUE : FALSE);
                        m_lf.lfStrikeOut = (BYTE)(((cf.dwEffects & CFE_STRIKEOUT) != 0) ? TRUE : FALSE);
                }
                else
                {
                        m_lf.lfUnderline = (BYTE)FALSE;
                        m_lf.lfStrikeOut = (BYTE)FALSE;
                }
                if((cf.dwMask & CFM_CHARSET) != 0)
                        m_lf.lfCharSet = cf.bCharSet;
                else
                        dwFlags |= CF_NOSCRIPTSEL;
                m_lf.lfOutPrecision = OUT_DEFAULT_PRECIS;
                m_lf.lfClipPrecision = CLIP_DEFAULT_PRECIS;
                m_lf.lfQuality = DEFAULT_QUALITY;
                if((cf.dwMask & CFM_FACE) != 0)
                {
                        m_lf.lfPitchAndFamily = cf.bPitchAndFamily;
#if (_RICHEDIT_VER >= 0x0200)
                        SecureHelper::strcpy_x(m_lf.lfFaceName, _countof(m_lf.lfFaceName), cf.szFaceName);
#else 
                        SecureHelper::strcpy_x(m_lf.lfFaceName, _countof(m_lf.lfFaceName), A2T((LPSTR)cf.szFaceName));
#endif 
                }
                else
                {
                        m_lf.lfPitchAndFamily = DEFAULT_PITCH|FF_DONTCARE;
                        m_lf.lfFaceName[0] = (TCHAR)0;
                }
                return dwFlags;
        }
};
class CRichEditFontDialog : public CRichEditFontDialogImpl<CRichEditFontDialog>
{
public:
        CRichEditFontDialog(const CHARFORMAT& charformat,
                DWORD dwFlags = CF_SCREENFONTS,
                HDC hDCPrinter = NULL,
                HWND hWndParent = NULL)
                : CRichEditFontDialogImpl<CRichEditFontDialog>(charformat, dwFlags, hDCPrinter, hWndParent)
        { }
        DECLARE_EMPTY_MSG_MAP()
};
#endif 
#if !defined(_WIN32_WCE) || ((_WIN32_WCE > 420) && !(defined(WIN32_PLATFORM_WFSP) && (_WIN32_WCE > 0x0500)))
#ifdef _WIN32_WCE
  #pragma comment(lib, "commdlg.lib")
  #ifndef SETRGBSTRING
    #define SETRGBSTRING _T("commdlg_SetRGBColor")
  #endif
  #ifndef COLOROKSTRING
    #define COLOROKSTRING _T("commdlg_ColorOK")
  #endif
#endif
template <class T>
class ATL_NO_VTABLE CColorDialogImpl : public CCommonDialogImplBase
{
public:
        CHOOSECOLOR m_cc;
        CColorDialogImpl(COLORREF clrInit = 0, DWORD dwFlags = 0, HWND hWndParent = NULL)
        {
                memset(&m_cc, 0, sizeof(m_cc));
                m_cc.lStructSize = sizeof(m_cc);
                m_cc.lpCustColors = GetCustomColors();
                m_cc.hwndOwner = hWndParent;
                m_cc.Flags = dwFlags | CC_ENABLEHOOK;
                m_cc.lpfnHook = (LPCCHOOKPROC)T::HookProc;
                if(clrInit != 0)
                {
                        m_cc.rgbResult = clrInit;
                        m_cc.Flags |= CC_RGBINIT;
                }
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                ATLASSERT((m_cc.Flags & CC_ENABLEHOOK) != 0);
                ATLASSERT(m_cc.lpfnHook != NULL);   
                if(m_cc.hwndOwner == NULL)          
                        m_cc.hwndOwner = hWndParent;
                ATLASSERT(m_hWnd == NULL);
                ModuleHelper::AddCreateWndData(&m_thunk.cd, (CCommonDialogImplBase*)this);
                BOOL bRet = ::ChooseColor(&m_cc);
                m_hWnd = NULL;
                return bRet ? IDOK : IDCANCEL;
        }
        
        void SetCurrentColor(COLORREF clr)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                SendMessage(_GetSetRGBMessage(), 0, (LPARAM)clr);
        }
        
        COLORREF GetColor() const
        {
                return m_cc.rgbResult;
        }
        static UINT_PTR APIENTRY HookProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
        {
                if(uMsg != WM_INITDIALOG && uMsg != _GetColorOKMessage())
                        return 0;
                LPCHOOSECOLOR lpCC = (LPCHOOSECOLOR)lParam;
                CCommonDialogImplBase* pT = NULL;
                if(uMsg == WM_INITDIALOG)
                {
                        pT = (CCommonDialogImplBase*)ModuleHelper::ExtractCreateWndData();
                        lpCC->lCustData = (LPARAM)pT;
                        ATLASSERT(pT != NULL);
                        ATLASSERT(pT->m_hWnd == NULL);
                        ATLASSERT(::IsWindow(hWnd));
                        
                        if(!pT->SubclassWindow(hWnd))
                        {
                                ATLTRACE2(atlTraceUI, 0, _T("Subclassing a Color common dialog failed\n"));
                                return 0;
                        }
                }
                else if(uMsg == _GetColorOKMessage())
                {
                        pT = (CCommonDialogImplBase*)lpCC->lCustData;
                        ATLASSERT(pT != NULL);
                        ATLASSERT(::IsWindow(pT->m_hWnd));
                }
                
                LRESULT lRes;
                if(pT->ProcessWindowMessage(pT->m_hWnd, uMsg, wParam, lParam, lRes, 0) == FALSE)
                        return 0;
                return lRes;
        }
        static COLORREF* GetCustomColors()
        {
                static COLORREF rgbCustomColors[16] =
                {
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                        RGB(255, 255, 255), RGB(255, 255, 255), 
                };
                return rgbCustomColors;
        }
        static UINT _GetSetRGBMessage()
        {
                static UINT uSetRGBMessage = 0;
                if(uSetRGBMessage == 0)
                {
                        CStaticDataInitCriticalSectionLock lock;
                        if(FAILED(lock.Lock()))
                        {
                                ATLTRACE2(atlTraceUI, 0, _T("ERROR : Unable to lock critical section in CColorDialogImpl::_GetSetRGBMessage.\n"));
                                ATLASSERT(FALSE);
                                return 0;
                        }
                        if(uSetRGBMessage == 0)
                                uSetRGBMessage = ::RegisterWindowMessage(SETRGBSTRING);
                        lock.Unlock();
                }
                ATLASSERT(uSetRGBMessage != 0);
                return uSetRGBMessage;
        }
        static UINT _GetColorOKMessage()
        {
                static UINT uColorOKMessage = 0;
                if(uColorOKMessage == 0)
                {
                        CStaticDataInitCriticalSectionLock lock;
                        if(FAILED(lock.Lock()))
                        {
                                ATLTRACE2(atlTraceUI, 0, _T("ERROR : Unable to lock critical section in CColorDialogImpl::_GetColorOKMessage.\n"));
                                ATLASSERT(FALSE);
                                return 0;
                        }
                        if(uColorOKMessage == 0)
                                uColorOKMessage = ::RegisterWindowMessage(COLOROKSTRING);
                        lock.Unlock();
                }
                ATLASSERT(uColorOKMessage != 0);
                return uColorOKMessage;
        }
        BEGIN_MSG_MAP(CColorDialogImpl)
                MESSAGE_HANDLER(_GetColorOKMessage(), _OnColorOK)
        END_MSG_MAP()
        LRESULT _OnColorOK(UINT, WPARAM, LPARAM, BOOL&)
        {
                T* pT = static_cast<T*>(this);
                return pT->OnColorOK();
        }
        BOOL OnColorOK()        
        {
                return FALSE;
        }
};
class CColorDialog : public CColorDialogImpl<CColorDialog>
{
public:
        CColorDialog(COLORREF clrInit = 0, DWORD dwFlags = 0, HWND hWndParent = NULL)
                : CColorDialogImpl<CColorDialog>(clrInit, dwFlags, hWndParent)
        { }
        
        DECLARE_EMPTY_MSG_MAP()
};
#endif 
#ifndef _WIN32_WCE
static HDC _AtlCreateDC(HGLOBAL hDevNames, HGLOBAL hDevMode)
{
        if(hDevNames == NULL)
                return NULL;
        LPDEVNAMES lpDevNames = (LPDEVNAMES)::GlobalLock(hDevNames);
        LPDEVMODE  lpDevMode = (hDevMode != NULL) ? (LPDEVMODE)::GlobalLock(hDevMode) : NULL;
        if(lpDevNames == NULL)
                return NULL;
        HDC hDC = ::CreateDC((LPCTSTR)lpDevNames + lpDevNames->wDriverOffset,
                                          (LPCTSTR)lpDevNames + lpDevNames->wDeviceOffset,
                                          (LPCTSTR)lpDevNames + lpDevNames->wOutputOffset,
                                          lpDevMode);
        ::GlobalUnlock(hDevNames);
        if(hDevMode != NULL)
                ::GlobalUnlock(hDevMode);
        return hDC;
}
template <class T>
class ATL_NO_VTABLE CPrintDialogImpl : public CCommonDialogImplBase
{
public:
        
        PRINTDLG& m_pd;
        CPrintDialogImpl(BOOL bPrintSetupOnly = FALSE,  
                        DWORD dwFlags = PD_ALLPAGES | PD_USEDEVMODECOPIES | PD_NOPAGENUMS | PD_NOSELECTION,
                        HWND hWndParent = NULL)
                        : m_pd(m_pdActual)
        {
                memset(&m_pdActual, 0, sizeof(m_pdActual));
                m_pd.lStructSize = sizeof(m_pdActual);
                m_pd.hwndOwner = hWndParent;
                m_pd.Flags = (dwFlags | PD_ENABLEPRINTHOOK | PD_ENABLESETUPHOOK);
                m_pd.lpfnPrintHook = (LPPRINTHOOKPROC)T::HookProc;
                m_pd.lpfnSetupHook = (LPSETUPHOOKPROC)T::HookProc;
                if(bPrintSetupOnly)
                        m_pd.Flags |= PD_PRINTSETUP;
                else
                        m_pd.Flags |= PD_RETURNDC;
                m_pd.Flags &= ~PD_RETURNIC; 
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                ATLASSERT((m_pd.Flags & PD_ENABLEPRINTHOOK) != 0);
                ATLASSERT((m_pd.Flags & PD_ENABLESETUPHOOK) != 0);
                ATLASSERT(m_pd.lpfnPrintHook != NULL);   
                ATLASSERT(m_pd.lpfnSetupHook != NULL);   
                ATLASSERT((m_pd.Flags & PD_RETURNDEFAULT) == 0);   
                if(m_pd.hwndOwner == NULL)   
                        m_pd.hwndOwner = hWndParent;
                ATLASSERT(m_hWnd == NULL);
                ModuleHelper::AddCreateWndData(&m_thunk.cd, (CCommonDialogImplBase*)this);
                BOOL bRet = ::PrintDlg(&m_pd);
                m_hWnd = NULL;
                return bRet ? IDOK : IDCANCEL;
        }
        
        BOOL GetDefaults()
        {
                m_pd.Flags |= PD_RETURNDEFAULT;
                ATLASSERT(m_pd.hDevMode == NULL);    
                ATLASSERT(m_pd.hDevNames == NULL);   
                return ::PrintDlg(&m_pd);
        }
        
        int GetCopies() const
        {
                if((m_pd.Flags & PD_USEDEVMODECOPIES) != 0)
                {
                        LPDEVMODE lpDevMode = GetDevMode();
                        return (lpDevMode != NULL) ? lpDevMode->dmCopies : -1;
                }
                return m_pd.nCopies;
        }
        BOOL PrintCollate() const       
        {
                return ((m_pd.Flags & PD_COLLATE) != 0) ? TRUE : FALSE;
        }
        BOOL PrintSelection() const     
        {
                return ((m_pd.Flags & PD_SELECTION) != 0) ? TRUE : FALSE;
        }
        BOOL PrintAll() const           
        {
                return (!PrintRange() && !PrintSelection()) ? TRUE : FALSE;
        }
        BOOL PrintRange() const         
        {
                return ((m_pd.Flags & PD_PAGENUMS) != 0) ? TRUE : FALSE;
        }
        BOOL PrintToFile() const        
        {
                return ((m_pd.Flags & PD_PRINTTOFILE) != 0) ? TRUE : FALSE;
        }
        int GetFromPage() const         
        {
                return PrintRange() ? m_pd.nFromPage : -1;
        }
        int GetToPage() const           
        {
                return PrintRange() ? m_pd.nToPage : -1;
        }
        LPDEVMODE GetDevMode() const    
        {
                if(m_pd.hDevMode == NULL)
                        return NULL;
                return (LPDEVMODE)::GlobalLock(m_pd.hDevMode);
        }
        LPCTSTR GetDriverName() const   
        {
                if(m_pd.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_pd.hDevNames);
                if(lpDev == NULL)
                        return NULL;
                return (LPCTSTR)lpDev + lpDev->wDriverOffset;
        }
        LPCTSTR GetDeviceName() const   
        {
                if(m_pd.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_pd.hDevNames);
                if(lpDev == NULL)
                        return NULL;
                return (LPCTSTR)lpDev + lpDev->wDeviceOffset;
        }
        LPCTSTR GetPortName() const     
        {
                if(m_pd.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_pd.hDevNames);
                if(lpDev == NULL)
                        return NULL;
                return (LPCTSTR)lpDev + lpDev->wOutputOffset;
        }
        HDC GetPrinterDC() const        
        {
                ATLASSERT((m_pd.Flags & PD_RETURNDC) != 0);
                return m_pd.hDC;
        }
        
        
        
        
        
        HDC CreatePrinterDC()
        {
                m_pd.hDC = _AtlCreateDC(m_pd.hDevNames, m_pd.hDevMode);
                return m_pd.hDC;
        }
        PRINTDLG m_pdActual; 
        
        CPrintDialogImpl(PRINTDLG& pdInit) : m_pd(pdInit)
        { }
        BEGIN_MSG_MAP(CPrintDialogImpl)
#ifdef psh1
                COMMAND_ID_HANDLER(psh1, OnPrintSetup) 
#else 
                COMMAND_ID_HANDLER(0x0400, OnPrintSetup) 
#endif 
        END_MSG_MAP()
        LRESULT OnPrintSetup(WORD wNotifyCode, WORD wID, HWND hWndCtl, BOOL& )
        {
                T dlgSetup(m_pd);
                ModuleHelper::AddCreateWndData(&dlgSetup.m_thunk.cd, (CCommonDialogImplBase*)&dlgSetup);
                return DefWindowProc(WM_COMMAND, MAKEWPARAM(wID, wNotifyCode), (LPARAM)hWndCtl);
        }
};
class CPrintDialog : public CPrintDialogImpl<CPrintDialog>
{
public:
        CPrintDialog(BOOL bPrintSetupOnly = FALSE,
                DWORD dwFlags = PD_ALLPAGES | PD_USEDEVMODECOPIES | PD_NOPAGENUMS | PD_NOSELECTION,
                HWND hWndParent = NULL)
                : CPrintDialogImpl<CPrintDialog>(bPrintSetupOnly, dwFlags, hWndParent)
        { }
        CPrintDialog(PRINTDLG& pdInit) : CPrintDialogImpl<CPrintDialog>(pdInit)
        { }
};
#endif 
#if (WINVER >= 0x0500) && !defined(_WIN32_WCE)
}; 
#include <atlcom.h>
extern "C" const __declspec(selectany) IID IID_IPrintDialogCallback = {0x5852a2c3, 0x6530, 0x11d1, {0xb6, 0xa3, 0x0, 0x0, 0xf8, 0x75, 0x7b, 0xf9}};
extern "C" const __declspec(selectany) IID IID_IPrintDialogServices = {0x509aaeda, 0x5639, 0x11d1, {0xb6, 0xa1, 0x0, 0x0, 0xf8, 0x75, 0x7b, 0xf9}};
namespace WTL
{
template <class T>
class ATL_NO_VTABLE CPrintDialogExImpl : 
                                public ATL::CWindow,
                                public ATL::CMessageMap,
                                public IPrintDialogCallback,
                                public ATL::IObjectWithSiteImpl< T >
{
public:
        PRINTDLGEX m_pdex;
        CPrintDialogExImpl(DWORD dwFlags = PD_ALLPAGES | PD_USEDEVMODECOPIES | PD_NOPAGENUMS | PD_NOSELECTION | PD_NOCURRENTPAGE,
                                HWND hWndParent = NULL)
        {
                memset(&m_pdex, 0, sizeof(m_pdex));
                m_pdex.lStructSize = sizeof(PRINTDLGEX);
                m_pdex.hwndOwner = hWndParent;
                m_pdex.Flags = dwFlags;
                m_pdex.nStartPage = START_PAGE_GENERAL;
                
                m_pdex.Flags &= ~PD_RETURNIC; 
        }
        HRESULT DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                ATLASSERT(m_hWnd == NULL);
                ATLASSERT((m_pdex.Flags & PD_RETURNDEFAULT) == 0);   
                if(m_pdex.hwndOwner == NULL)   
                        m_pdex.hwndOwner = hWndParent;
                T* pT = static_cast<T*>(this);
                m_pdex.lpCallback = (IUnknown*)(IPrintDialogCallback*)pT;
                HRESULT hResult = ::PrintDlgEx(&m_pdex);
                m_hWnd = NULL;
                return hResult;
        }
        BOOL EndDialog(INT_PTR  = 0)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                SendMessage(WM_COMMAND, MAKEWPARAM(IDABORT, 0));
                return TRUE;
        }
        
        HRESULT GetDefaults()
        {
                m_pdex.Flags |= PD_RETURNDEFAULT;
                ATLASSERT(m_pdex.hDevMode == NULL);    
                ATLASSERT(m_pdex.hDevNames == NULL);   
                return ::PrintDlgEx(&m_pdex);
        }
        
        int GetCopies() const
        {
                if((m_pdex.Flags & PD_USEDEVMODECOPIES) != 0)
                {
                        LPDEVMODE lpDevMode = GetDevMode();
                        return (lpDevMode != NULL) ? lpDevMode->dmCopies : -1;
                }
                return m_pdex.nCopies;
        }
        BOOL PrintCollate() const       
        {
                return ((m_pdex.Flags & PD_COLLATE) != 0) ? TRUE : FALSE;
        }
        BOOL PrintSelection() const     
        {
                return ((m_pdex.Flags & PD_SELECTION) != 0) ? TRUE : FALSE;
        }
        BOOL PrintAll() const           
        {
                return (!PrintRange() && !PrintSelection()) ? TRUE : FALSE;
        }
        BOOL PrintRange() const         
        {
                return ((m_pdex.Flags & PD_PAGENUMS) != 0) ? TRUE : FALSE;
        }
        BOOL PrintToFile() const        
        {
                return ((m_pdex.Flags & PD_PRINTTOFILE) != 0) ? TRUE : FALSE;
        }
        LPDEVMODE GetDevMode() const    
        {
                if(m_pdex.hDevMode == NULL)
                        return NULL;
                return (LPDEVMODE)::GlobalLock(m_pdex.hDevMode);
        }
        LPCTSTR GetDriverName() const   
        {
                if(m_pdex.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_pdex.hDevNames);
                if(lpDev == NULL)
                        return NULL;
                return (LPCTSTR)lpDev + lpDev->wDriverOffset;
        }
        LPCTSTR GetDeviceName() const   
        {
                if(m_pdex.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_pdex.hDevNames);
                if(lpDev == NULL)
                        return NULL;
                return (LPCTSTR)lpDev + lpDev->wDeviceOffset;
        }
        LPCTSTR GetPortName() const     
        {
                if(m_pdex.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_pdex.hDevNames);
                if(lpDev == NULL)
                        return NULL;
                return (LPCTSTR)lpDev + lpDev->wOutputOffset;
        }
        HDC GetPrinterDC() const        
        {
                ATLASSERT((m_pdex.Flags & PD_RETURNDC) != 0);
                return m_pdex.hDC;
        }
        
        
        
        
        
        HDC CreatePrinterDC()
        {
                m_pdex.hDC = _AtlCreateDC(m_pdex.hDevNames, m_pdex.hDevMode);
                return m_pdex.hDC;
        }
        STDMETHOD(QueryInterface)(REFIID riid, void** ppvObject)
        {
                if(ppvObject == NULL)
                        return E_POINTER;
                T* pT = static_cast<T*>(this);
                if(IsEqualGUID(riid, IID_IUnknown) || IsEqualGUID(riid, IID_IPrintDialogCallback))
                {
                        *ppvObject = (IPrintDialogCallback*)pT;
                        
                        return S_OK;
                }
                else if(IsEqualGUID(riid, IID_IObjectWithSite))
                {
                        *ppvObject = (IObjectWithSite*)pT;
                        
                        return S_OK;
                }
                return E_NOINTERFACE;
        }
        virtual ULONG STDMETHODCALLTYPE AddRef()
        {
                return 1;
        }
        virtual ULONG STDMETHODCALLTYPE Release()
        {
                return 1;
        }
        STDMETHOD(InitDone)()
        {
                return S_FALSE;
        }
        STDMETHOD(SelectionChange)()
        {
                return S_FALSE;
        }
        STDMETHOD(HandleMessage)(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT* plResult)
        {
                
                if(m_hWnd == NULL)
                        Attach(hWnd);
                
                HRESULT hRet = ProcessWindowMessage(hWnd, uMsg, wParam, lParam, *plResult, 0) ? S_OK : S_FALSE;
                if(hRet == S_OK && uMsg == WM_NOTIFY)   
                        ::SetWindowLongPtr(GetParent(), DWLP_MSGRESULT, (LONG_PTR)*plResult);
                if(uMsg == WM_INITDIALOG && hRet == S_OK && (BOOL)*plResult != FALSE)
                        hRet = S_FALSE;
                return hRet;
        }
};
class CPrintDialogEx : public CPrintDialogExImpl<CPrintDialogEx>
{
public:
        CPrintDialogEx(
                DWORD dwFlags = PD_ALLPAGES | PD_USEDEVMODECOPIES | PD_NOPAGENUMS | PD_NOSELECTION | PD_NOCURRENTPAGE,
                HWND hWndParent = NULL)
                : CPrintDialogExImpl<CPrintDialogEx>(dwFlags, hWndParent)
        { }
        DECLARE_EMPTY_MSG_MAP()
};
#endif 
#ifndef _WIN32_WCE
template <class T>
class ATL_NO_VTABLE CPageSetupDialogImpl : public CCommonDialogImplBase
{
public:
        PAGESETUPDLG m_psd;
        ATL::CWndProcThunk m_thunkPaint;
        CPageSetupDialogImpl(DWORD dwFlags = PSD_MARGINS | PSD_INWININIINTLMEASURE, HWND hWndParent = NULL)
        {
                memset(&m_psd, 0, sizeof(m_psd));
                m_psd.lStructSize = sizeof(m_psd);
                m_psd.hwndOwner = hWndParent;
                m_psd.Flags = (dwFlags | PSD_ENABLEPAGESETUPHOOK | PSD_ENABLEPAGEPAINTHOOK);
                m_psd.lpfnPageSetupHook = (LPPAGESETUPHOOK)T::HookProc;
                m_thunkPaint.Init((WNDPROC)T::PaintHookProc, this);
#if (_ATL_VER >= 0x0700)
                m_psd.lpfnPagePaintHook = (LPPAGEPAINTHOOK)m_thunkPaint.GetWNDPROC();
#else
                m_psd.lpfnPagePaintHook = (LPPAGEPAINTHOOK)&(m_thunkPaint.thunk);
#endif
        }
        DECLARE_EMPTY_MSG_MAP()
        LPDEVMODE GetDevMode() const    
        {
                if(m_psd.hDevMode == NULL)
                        return NULL;
                return (LPDEVMODE)::GlobalLock(m_psd.hDevMode);
        }
        LPCTSTR GetDriverName() const   
        {
                if(m_psd.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_psd.hDevNames);
                return (LPCTSTR)lpDev + lpDev->wDriverOffset;
        }
        LPCTSTR GetDeviceName() const   
        {
                if(m_psd.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_psd.hDevNames);
                return (LPCTSTR)lpDev + lpDev->wDeviceOffset;
        }
        LPCTSTR GetPortName() const     
        {
                if(m_psd.hDevNames == NULL)
                        return NULL;
                LPDEVNAMES lpDev = (LPDEVNAMES)::GlobalLock(m_psd.hDevNames);
                return (LPCTSTR)lpDev + lpDev->wOutputOffset;
        }
        HDC CreatePrinterDC()
        {
                return _AtlCreateDC(m_psd.hDevNames, m_psd.hDevMode);
        }
        SIZE GetPaperSize() const
        {
                SIZE size;
                size.cx = m_psd.ptPaperSize.x;
                size.cy = m_psd.ptPaperSize.y;
                return size;
        }
        void GetMargins(LPRECT lpRectMargins, LPRECT lpRectMinMargins) const
        {
                if(lpRectMargins != NULL)
                        *lpRectMargins = m_psd.rtMargin;
                if(lpRectMinMargins != NULL)
                        *lpRectMinMargins = m_psd.rtMinMargin;
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                ATLASSERT((m_psd.Flags & PSD_ENABLEPAGESETUPHOOK) != 0);
                ATLASSERT((m_psd.Flags & PSD_ENABLEPAGEPAINTHOOK) != 0);
                ATLASSERT(m_psd.lpfnPageSetupHook != NULL);   
                ATLASSERT(m_psd.lpfnPagePaintHook != NULL);   
                if(m_psd.hwndOwner == NULL)   
                        m_psd.hwndOwner = hWndParent;
                ATLASSERT(m_hWnd == NULL);
                ModuleHelper::AddCreateWndData(&m_thunk.cd, (CCommonDialogImplBase*)this);
                BOOL bRet = ::PageSetupDlg(&m_psd);
                m_hWnd = NULL;
                return bRet ? IDOK : IDCANCEL;
        }
        static UINT_PTR CALLBACK PaintHookProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
        {
                T* pT = (T*)hWnd;
                UINT_PTR uRet = 0;
                switch(uMsg)
                {
                case WM_PSD_PAGESETUPDLG:
                        uRet = pT->PreDrawPage(LOWORD(wParam), HIWORD(wParam), (LPPAGESETUPDLG)lParam);
                        break;
                case WM_PSD_FULLPAGERECT:
                case WM_PSD_MINMARGINRECT:
                case WM_PSD_MARGINRECT:
                case WM_PSD_GREEKTEXTRECT:
                case WM_PSD_ENVSTAMPRECT:
                case WM_PSD_YAFULLPAGERECT:
                        uRet = pT->OnDrawPage(uMsg, (HDC)wParam, (LPRECT)lParam);
                        break;
                default:
                        ATLTRACE2(atlTraceUI, 0, _T("CPageSetupDialogImpl::PaintHookProc - unknown message received\n"));
                        break;
                }
                return uRet;
        }
        UINT_PTR PreDrawPage(WORD , WORD , LPPAGESETUPDLG )
        {
                
                return 0;
        }
        UINT_PTR OnDrawPage(UINT , HDC , LPRECT )
        {
                return 0; 
        }
};
class CPageSetupDialog : public CPageSetupDialogImpl<CPageSetupDialog>
{
public:
        CPageSetupDialog(DWORD dwFlags = PSD_MARGINS | PSD_INWININIINTLMEASURE, HWND hWndParent = NULL)
                : CPageSetupDialogImpl<CPageSetupDialog>(dwFlags, hWndParent)
        { }
        
        static UINT_PTR CALLBACK PaintHookProc(HWND, UINT, WPARAM, LPARAM)
        {
                return 0;
        }
};
#endif 
#ifndef _WIN32_WCE
template <class T>
class ATL_NO_VTABLE CFindReplaceDialogImpl : public CCommonDialogImplBase
{
public:
        enum { _cchFindReplaceBuffer = 128 };
        FINDREPLACE m_fr;
        TCHAR m_szFindWhat[_cchFindReplaceBuffer];
        TCHAR m_szReplaceWith[_cchFindReplaceBuffer];
        CFindReplaceDialogImpl()
        {
                memset(&m_fr, 0, sizeof(m_fr));
                m_szFindWhat[0] = _T('\0');
                m_szReplaceWith[0] = _T('\0');
                m_fr.lStructSize = sizeof(m_fr);
                m_fr.Flags = FR_ENABLEHOOK;
                m_fr.lpfnHook = (LPFRHOOKPROC)T::HookProc;
                m_fr.lpstrFindWhat = (LPTSTR)m_szFindWhat;
                m_fr.wFindWhatLen = _cchFindReplaceBuffer;
                m_fr.lpstrReplaceWith = (LPTSTR)m_szReplaceWith;
                m_fr.wReplaceWithLen = _cchFindReplaceBuffer;
        }
        
        
        virtual void OnFinalMessage(HWND )
        {
                delete this;
        }
        HWND Create(BOOL bFindDialogOnly, 
                        LPCTSTR lpszFindWhat,
                        LPCTSTR lpszReplaceWith = NULL,
                        DWORD dwFlags = FR_DOWN,
                        HWND hWndParent = NULL)
        {
                ATLASSERT((m_fr.Flags & FR_ENABLEHOOK) != 0);
                ATLASSERT(m_fr.lpfnHook != NULL);
                m_fr.Flags |= dwFlags;
                if(hWndParent == NULL)
                        m_fr.hwndOwner = ::GetActiveWindow();
                else
                        m_fr.hwndOwner = hWndParent;
                ATLASSERT(m_fr.hwndOwner != NULL); 
                if(lpszFindWhat != NULL)
                        SecureHelper::strncpy_x(m_szFindWhat, _countof(m_szFindWhat), lpszFindWhat, _TRUNCATE);
                if(lpszReplaceWith != NULL)
                        SecureHelper::strncpy_x(m_szReplaceWith, _countof(m_szReplaceWith), lpszReplaceWith, _TRUNCATE);
                ATLASSERT(m_hWnd == NULL);
                ModuleHelper::AddCreateWndData(&m_thunk.cd, (CCommonDialogImplBase*)this);
                HWND hWnd = NULL;
                if(bFindDialogOnly)
                        hWnd = ::FindText(&m_fr);
                else
                        hWnd = ::ReplaceText(&m_fr);
                ATLASSERT(m_hWnd == hWnd);
                return hWnd;
        }
        static const UINT GetFindReplaceMsg()
        {
                static const UINT nMsgFindReplace = ::RegisterWindowMessage(FINDMSGSTRING);
                return nMsgFindReplace;
        }
        
        
        static T* PASCAL GetNotifier(LPARAM lParam)
        {
                ATLASSERT(lParam != NULL);
                T* pDlg = (T*)(lParam - offsetof(T, m_fr));
                return pDlg;
        }
        
        LPCTSTR GetFindString() const    
        {
                return (LPCTSTR)m_fr.lpstrFindWhat;
        }
        LPCTSTR GetReplaceString() const 
        {
                return (LPCTSTR)m_fr.lpstrReplaceWith;
        }
        BOOL SearchDown() const          
        {
                return ((m_fr.Flags & FR_DOWN) != 0) ? TRUE : FALSE;
        }
        BOOL FindNext() const            
        {
                return ((m_fr.Flags & FR_FINDNEXT) != 0) ? TRUE : FALSE;
        }
        BOOL MatchCase() const           
        {
                return ((m_fr.Flags & FR_MATCHCASE) != 0) ? TRUE : FALSE;
        }
        BOOL MatchWholeWord() const      
        {
                return ((m_fr.Flags & FR_WHOLEWORD) != 0) ? TRUE : FALSE;
        }
        BOOL ReplaceCurrent() const      
        {
                return ((m_fr. Flags & FR_REPLACE) != 0) ? TRUE : FALSE;
        }
        BOOL ReplaceAll() const          
        {
                return ((m_fr.Flags & FR_REPLACEALL) != 0) ? TRUE : FALSE;
        }
        BOOL IsTerminating() const       
        {
                return ((m_fr.Flags & FR_DIALOGTERM) != 0) ? TRUE : FALSE ;
        }
};
class CFindReplaceDialog : public CFindReplaceDialogImpl<CFindReplaceDialog>
{
public:
        DECLARE_EMPTY_MSG_MAP()
};
#endif 
#if (_ATL_VER >= 0x800)
typedef ATL::_DialogSplitHelper::DLGTEMPLATEEX DLGTEMPLATEEX;
typedef ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX DLGITEMTEMPLATEEX;
#else 
typedef ATL::_DialogSizeHelper::_ATL_DLGTEMPLATEEX DLGTEMPLATEEX;
#pragma pack(push, 4)
struct DLGITEMTEMPLATEEX
{
        DWORD helpID;
        DWORD exStyle;
        DWORD style;
        short x;
        short y;
        short cx;
        short cy;
        WORD id;
};
#pragma pack(pop)
#endif 
class CMemDlgTemplate
{
public:
        enum StdCtrlType
        {
                CTRL_BUTTON    = 0x0080,
                CTRL_EDIT      = 0x0081,
                CTRL_STATIC    = 0x0082,
                CTRL_LISTBOX   = 0x0083,
                CTRL_SCROLLBAR = 0x0084,
                CTRL_COMBOBOX  = 0x0085
        };
        CMemDlgTemplate() : m_pData(NULL), m_pPtr(NULL), m_cAllocated(0)
        { }
        ~CMemDlgTemplate()
        {
                Reset();
        }
        bool IsValid() const
        {
                return (m_pData != NULL);
        }
        bool IsTemplateEx() const
        {
                return (IsValid() && ((DLGTEMPLATEEX*)m_pData)->signature == 0xFFFF);
        }
        LPDLGTEMPLATE GetTemplatePtr()
        {
                return reinterpret_cast<LPDLGTEMPLATE>(m_pData);
        }
        DLGTEMPLATEEX* GetTemplateExPtr()
        {
                return reinterpret_cast<DLGTEMPLATEEX*>(m_pData);
        }
        void Reset()
        {
                if (IsValid())
                        ATLVERIFY(::GlobalFree(m_pData) == NULL);
                m_pData = NULL;
                m_pPtr = NULL;
                m_cAllocated = 0;
        }
        void Create(bool bDlgEx, LPCTSTR lpszCaption, short nX, short nY, short nWidth, short nHeight, DWORD dwStyle = 0, DWORD dwExStyle = 0, 
                    LPCTSTR lpstrFontName = NULL, WORD wFontSize = 0, WORD wWeight = 0, BYTE bItalic = 0, BYTE bCharset = 0, DWORD dwHelpID = 0,
                                ATL::_U_STRINGorID ClassName = 0U, ATL::_U_STRINGorID Menu = 0U)
        {
                
                if (lpstrFontName != NULL)
                {
                        dwStyle |= DS_SETFONT;
                }
                else
                {
                        dwStyle &= ~DS_SETFONT;
                }
                if (bDlgEx)
                {
                        DLGTEMPLATEEX dlg = {1, 0xFFFF, dwHelpID, dwExStyle, dwStyle, 0, nX, nY, nWidth, nHeight};
                        AddData(&dlg, sizeof(dlg));
                }
                else
                {
                        DLGTEMPLATE dlg = {dwStyle, dwExStyle, 0, nX, nY, nWidth, nHeight};
                        AddData(&dlg, sizeof(dlg));
                }
#ifndef _WIN32_WCE
                if (Menu.m_lpstr == NULL)
                {
                        WORD menuData = 0;
                        AddData(&menuData, sizeof(WORD));
                }
                else if (IS_INTRESOURCE(Menu.m_lpstr))
                {
                        WORD menuData[] = {0xFFFF, (WORD)Menu.m_lpstr};
                        AddData(menuData, sizeof(menuData));
                }
                else
                {
                        AddString(Menu.m_lpstr);
                }
#else 
                
                ATLASSERT(Menu.m_lpstr == NULL);
                Menu.m_lpstr;   
                WORD menuData = 0;
                AddData(&menuData, sizeof(WORD));
#endif 
                if (ClassName.m_lpstr == NULL)
                {
                        WORD classData = 0;
                        AddData(&classData, sizeof(WORD));
                }
                else if (IS_INTRESOURCE(ClassName.m_lpstr))
                {
                        WORD classData[] = {0xFFFF, (WORD)ClassName.m_lpstr};
                        AddData(classData, sizeof(classData));
                }
                else
                {
                        AddString(ClassName.m_lpstr);
                }
                
                AddString(lpszCaption);
                if (lpstrFontName != NULL)
                {
                        AddData(&wFontSize, sizeof(wFontSize));
                        if (bDlgEx)
                        {
                                AddData(&wWeight, sizeof(wWeight));
                                AddData(&bItalic, sizeof(bItalic));
                                AddData(&bCharset, sizeof(bCharset));
                        }
                        AddString(lpstrFontName);
                }
        }
        void AddControl(ATL::_U_STRINGorID ClassName, WORD wId, short nX, short nY, short nWidth, short nHeight, DWORD dwStyle, DWORD dwExStyle,
                        ATL::_U_STRINGorID Text, const WORD* pCreationData = NULL, WORD nCreationData = 0, DWORD dwHelpID = 0)
        {
                ATLASSERT(IsValid());
                
                m_pPtr = (LPBYTE)(DWORD_PTR)((DWORD)(DWORD_PTR)(m_pPtr + 3) & (~3));
                if (IsTemplateEx())
                {
                        DLGTEMPLATEEX* dlg = (DLGTEMPLATEEX*)m_pData;
                        dlg->cDlgItems++;
                        DLGITEMTEMPLATEEX item = {dwHelpID, ATL::CControlWinTraits::GetWndExStyle(0) | dwExStyle, ATL::CControlWinTraits::GetWndStyle(0) | dwStyle, nX, nY, nWidth, nHeight, wId};
                        AddData(&item, sizeof(item));
                }
                else
                {
                        LPDLGTEMPLATE dlg = (LPDLGTEMPLATE)m_pData;
                        dlg->cdit++;
                        DLGITEMTEMPLATE item = {ATL::CControlWinTraits::GetWndStyle(0) | dwStyle, ATL::CControlWinTraits::GetWndExStyle(0) | dwExStyle, nX, nY, nWidth, nHeight, wId};
                        AddData(&item, sizeof(item));
                }
                ATLASSERT(ClassName.m_lpstr != NULL);
                if (IS_INTRESOURCE(ClassName.m_lpstr))
                {
                        WORD wData[] = {0xFFFF, (WORD)ClassName.m_lpstr};
                        AddData(wData, sizeof(wData));
                }
                else
                {
                        AddString(ClassName.m_lpstr);
                }
                if (Text.m_lpstr == NULL)
                {
                        WORD classData = 0;
                        AddData(&classData, sizeof(WORD));
                }
                else if (IS_INTRESOURCE(Text.m_lpstr))
                {
                        WORD wData[] = {0xFFFF, (WORD)Text.m_lpstr};
                        AddData(wData, sizeof(wData));
                }
                else
                {
                        AddString(Text.m_lpstr);
                }
                AddData(&nCreationData, sizeof(nCreationData));
                if ((nCreationData != 0))
                {
                        ATLASSERT(pCreationData != NULL);
                        AddData(pCreationData, nCreationData * sizeof(WORD));
                }
        }
        void AddStdControl(StdCtrlType CtrlType, WORD wId, short nX, short nY, short nWidth, short nHeight,
                           DWORD dwStyle, DWORD dwExStyle, ATL::_U_STRINGorID Text, const WORD* pCreationData = NULL, WORD nCreationData = 0, DWORD dwHelpID = 0)
        {
                AddControl(CtrlType, wId, nX, nY, nWidth, nHeight, dwStyle, dwExStyle, Text, pCreationData, nCreationData, dwHelpID);
        }
protected:
        void AddData(LPCVOID pData, size_t nData)
        {
                ATLASSERT(pData != NULL);
                const size_t ALLOCATION_INCREMENT = 1024;
                if (m_pData == NULL)
                {
                        m_cAllocated = ((nData / ALLOCATION_INCREMENT) + 1) * ALLOCATION_INCREMENT;
                        m_pPtr = m_pData = static_cast<LPBYTE>(::GlobalAlloc(GPTR, m_cAllocated));
                        ATLASSERT(m_pData != NULL);
                }
                else if (((m_pPtr - m_pData) + nData) > m_cAllocated)
                {
                        size_t ptrPos = (m_pPtr - m_pData);
                        m_cAllocated += ((nData / ALLOCATION_INCREMENT) + 1) * ALLOCATION_INCREMENT;
                        m_pData = static_cast<LPBYTE>(::GlobalReAlloc(m_pData, m_cAllocated, 0));
                        ATLASSERT(m_pData != NULL);
                        m_pPtr = m_pData + ptrPos;
                }
                SecureHelper::memcpy_x(m_pPtr, m_cAllocated - (m_pPtr - m_pData), pData, nData);
                m_pPtr += nData;
        }
        void AddString(LPCTSTR lpszStr)
        {
                if (lpszStr == NULL)
                {
                        WCHAR szEmpty = 0;
                        AddData(&szEmpty, sizeof(szEmpty));
                }
                else
                {
                        USES_CONVERSION;
                        LPCWSTR lpstr = T2CW(lpszStr);
                        int nSize = lstrlenW(lpstr) + 1;
                        AddData(lpstr, nSize * sizeof(WCHAR));
                }
        }
        LPBYTE m_pData;
        LPBYTE m_pPtr;
        SIZE_T m_cAllocated;
};
#define BEGIN_DIALOG(x, y, width, height) \
        void DoInitTemplate() \
        { \
                bool bExTemplate = false; \
                short nX = x, nY = y, nWidth = width, nHeight = height; \
                LPCTSTR szCaption = NULL; \
                DWORD dwStyle = WS_POPUP | WS_BORDER | WS_SYSMENU; \
                DWORD dwExStyle = 0; \
                LPCTSTR szFontName = NULL; \
                WORD wFontSize = 0; \
                WORD wWeight = 0; \
                BYTE bItalic = 0; \
                BYTE bCharset = 0; \
                DWORD dwHelpID = 0; \
                ATL::_U_STRINGorID Menu = 0U; \
                ATL::_U_STRINGorID ClassName = 0U;
#define BEGIN_DIALOG_EX(x, y, width, height, helpID) \
        void DoInitTemplate() \
        { \
                bool bExTemplate = true; \
                short nX = x, nY = y, nWidth = width, nHeight = height; \
                LPCTSTR szCaption = NULL; \
                DWORD dwStyle = WS_POPUP | WS_BORDER | WS_SYSMENU; \
                DWORD dwExStyle = 0; \
                LPCTSTR szFontName = NULL; \
                WORD wFontSize = 0; \
                WORD wWeight = 0; \
                BYTE bItalic = 0; \
                BYTE bCharset = 0; \
                DWORD dwHelpID = helpID; \
                ATL::_U_STRINGorID Menu = 0U; \
                ATL::_U_STRINGorID ClassName = 0U;
#define END_DIALOG() \
                m_Template.Create(bExTemplate, szCaption, nX, nY, nWidth, nHeight, dwStyle, dwExStyle, szFontName, wFontSize, wWeight, bItalic, bCharset, dwHelpID, ClassName, Menu); \
        };
#define DIALOG_CAPTION(caption) \
                szCaption = caption;
#define DIALOG_STYLE(style) \
                dwStyle = style;
#define DIALOG_EXSTYLE(exStyle) \
                dwExStyle = exStyle;
#define DIALOG_FONT(pointSize, typeFace) \
                wFontSize = pointSize; \
                szFontName = typeFace;
#define DIALOG_FONT_EX(pointsize, typeface, weight, italic, charset) \
                ATLASSERT(bExTemplate); \
                wFontSize = pointsize; \
                szFontName = typeface; \
                wWeight = weight; \
                bItalic = italic; \
                bCharset = charset;
#define DIALOG_MENU(menuName) \
                Menu = menuName;
#define DIALOG_CLASS(className) \
                ClassName = className;
#define BEGIN_CONTROLS_MAP() \
        void DoInitControls() \
        {
#define END_CONTROLS_MAP() \
        };
#define CONTROL_LTEXT(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_STATIC, (WORD)id, x, y, width, height, style | SS_LEFT | WS_GROUP, exStyle, text, NULL, 0);
#define CONTROL_CTEXT(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_STATIC, (WORD)id, x, y, width, height, style | SS_CENTER | WS_GROUP, exStyle, text, NULL, 0);
#define CONTROL_RTEXT(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_STATIC, (WORD)id, x, y, width, height, style | SS_RIGHT | WS_GROUP, exStyle, text, NULL, 0);
#define CONTROL_PUSHBUTTON(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_PUSHBUTTON | WS_TABSTOP, exStyle, text, NULL, 0);
#define CONTROL_DEFPUSHBUTTON(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_DEFPUSHBUTTON | WS_TABSTOP, exStyle, text, NULL, 0);
#ifndef _WIN32_WCE
#define CONTROL_PUSHBOX(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_PUSHBOX | WS_TABSTOP, exStyle, text, NULL, 0);
#endif 
#define CONTROL_STATE3(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_3STATE | WS_TABSTOP, exStyle, text, NULL, 0);
#define CONTROL_AUTO3STATE(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_AUTO3STATE | WS_TABSTOP, exStyle, text, NULL, 0);
#define CONTROL_CHECKBOX(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_CHECKBOX | WS_TABSTOP, exStyle, text, NULL, 0);
#define CONTROL_AUTOCHECKBOX(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_AUTOCHECKBOX | WS_TABSTOP, exStyle, text, NULL, 0);
#define CONTROL_RADIOBUTTON(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_RADIOBUTTON | WS_TABSTOP, exStyle, text, NULL, 0);
#define CONTROL_AUTORADIOBUTTON(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_AUTORADIOBUTTON | WS_TABSTOP, exStyle, text, NULL, 0);
#define CONTROL_COMBOBOX(id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_COMBOBOX, (WORD)id, x, y, width, height, style | CBS_DROPDOWN | WS_TABSTOP, exStyle, (LPCTSTR)NULL, NULL, 0);
#define CONTROL_EDITTEXT(id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_EDIT, (WORD)id, x, y, width, height, style | ES_LEFT | WS_BORDER | WS_TABSTOP, exStyle, (LPCTSTR)NULL, NULL, 0);
#define CONTROL_GROUPBOX(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_BUTTON, (WORD)id, x, y, width, height, style | BS_GROUPBOX, exStyle, text, NULL, 0);
#define CONTROL_LISTBOX(id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_LISTBOX, (WORD)id, x, y, width, height, style | LBS_NOTIFY | WS_BORDER, exStyle, (LPCTSTR)NULL, NULL, 0);
#define CONTROL_SCROLLBAR(id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_SCROLLBAR, (WORD)id, x, y, width, height, style | SBS_HORZ, exStyle, (LPCTSTR)NULL, NULL, 0);
#define CONTROL_ICON(text, id, x, y, width, height, style, exStyle) \
        m_Template.AddStdControl(WTL::CMemDlgTemplate::CTRL_STATIC, (WORD)id, x, y, width, height, style | SS_ICON, exStyle, text, NULL, 0);
#define CONTROL_CONTROL(text, id, className, style, x, y, width, height, exStyle) \
        m_Template.AddControl(className, (WORD)id, x, y, width, height, style, exStyle, text, NULL, 0);
template <class T, class TDlgTemplate = CMemDlgTemplate, class TBase = ATL::CDialogImpl<T, ATL::CWindow> >
class ATL_NO_VTABLE CIndirectDialogImpl : public TBase
{
public:
        enum { IDD = 0 };   
        TDlgTemplate m_Template;
        void CreateTemplate()
        {
                T* pT = static_cast<T*>(this);
                pT->DoInitTemplate();
                pT->DoInitControls();
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow(), LPARAM dwInitParam = NULL)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_hWnd == NULL);
                if (!m_Template.IsValid())
                        CreateTemplate();
#if (_ATL_VER >= 0x0800)
                
                BOOL result = m_thunk.Init(NULL, NULL);
                if (result == FALSE)
                {
                        SetLastError(ERROR_OUTOFMEMORY);
                        return -1;
                }
#endif 
                ModuleHelper::AddCreateWndData(&m_thunk.cd, pT);
#ifdef _DEBUG
                m_bModal = true;
#endif 
                return ::DialogBoxIndirectParam(ModuleHelper::GetResourceInstance(), m_Template.GetTemplatePtr(), hWndParent, (DLGPROC)T::StartDialogProc, dwInitParam);
        }
        HWND Create(HWND hWndParent, LPARAM dwInitParam = NULL)
        {
                T* pT = static_cast<T*>(this);
                ATLASSERT(pT->m_hWnd == NULL);
                if (!m_Template.IsValid())
                        CreateTemplate();
#if (_ATL_VER >= 0x0800)
                
                BOOL result = m_thunk.Init(NULL, NULL);
                if (result == FALSE) 
                {
                        SetLastError(ERROR_OUTOFMEMORY);
                        return NULL;
                }
#endif 
                ModuleHelper::AddCreateWndData(&m_thunk.cd, pT);
#ifdef _DEBUG
                m_bModal = false;
#endif 
                HWND hWnd = ::CreateDialogIndirectParam(ModuleHelper::GetResourceInstance(), (LPCDLGTEMPLATE)m_Template.GetTemplatePtr(), hWndParent, (DLGPROC)T::StartDialogProc, dwInitParam);
                ATLASSERT(m_hWnd == hWnd);
                return hWnd;
        }
        
        HWND Create(HWND hWndParent, RECT&, LPARAM dwInitParam = NULL)
        {
                return Create(hWndParent, dwInitParam);
        }
        void DoInitTemplate() 
        {
                ATLASSERT(FALSE);   
        }
        void DoInitControls() 
        {
                ATLASSERT(FALSE);   
        }
};
class CPropertySheetWindow : public ATL::CWindow
{
public:
        CPropertySheetWindow(HWND hWnd = NULL) : ATL::CWindow(hWnd)
        { }
        CPropertySheetWindow& operator =(HWND hWnd)
        {
                m_hWnd = hWnd;
                return *this;
        }
        int GetPageCount() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                HWND hWndTabCtrl = GetTabControl();
                ATLASSERT(hWndTabCtrl != NULL);
                return (int)::SendMessage(hWndTabCtrl, TCM_GETITEMCOUNT, 0, 0L);
        }
        HWND GetActivePage() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (HWND)::SendMessage(m_hWnd, PSM_GETCURRENTPAGEHWND, 0, 0L);
        }
        int GetActiveIndex() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                HWND hWndTabCtrl = GetTabControl();
                ATLASSERT(hWndTabCtrl != NULL);
                return (int)::SendMessage(hWndTabCtrl, TCM_GETCURSEL, 0, 0L);
        }
        BOOL SetActivePage(int nPageIndex)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (BOOL)::SendMessage(m_hWnd, PSM_SETCURSEL, nPageIndex, 0L);
        }
        BOOL SetActivePage(HPROPSHEETPAGE hPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(hPage != NULL);
                return (BOOL)::SendMessage(m_hWnd, PSM_SETCURSEL, 0, (LPARAM)hPage);
        }
        BOOL SetActivePageByID(int nPageID)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (BOOL)::SendMessage(m_hWnd, PSM_SETCURSELID, 0, nPageID);
        }
        void SetTitle(LPCTSTR lpszText, UINT nStyle = 0)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT((nStyle & ~PSH_PROPTITLE) == 0); 
                ATLASSERT(lpszText != NULL);
                ::SendMessage(m_hWnd, PSM_SETTITLE, nStyle, (LPARAM)lpszText);
        }
        HWND GetTabControl() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (HWND)::SendMessage(m_hWnd, PSM_GETTABCONTROL, 0, 0L);
        }
        void SetFinishText(LPCTSTR lpszText)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_SETFINISHTEXT, 0, (LPARAM)lpszText);
        }
        void SetWizardButtons(DWORD dwFlags)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::PostMessage(m_hWnd, PSM_SETWIZBUTTONS, 0, dwFlags);
        }
        BOOL AddPage(HPROPSHEETPAGE hPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(hPage != NULL);
                return (BOOL)::SendMessage(m_hWnd, PSM_ADDPAGE, 0, (LPARAM)hPage);
        }
        BOOL AddPage(LPCPROPSHEETPAGE pPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(pPage != NULL);
                HPROPSHEETPAGE hPage = ::CreatePropertySheetPage(pPage);
                if(hPage == NULL)
                        return FALSE;
                return (BOOL)::SendMessage(m_hWnd, PSM_ADDPAGE, 0, (LPARAM)hPage);
        }
#ifndef _WIN32_WCE
        BOOL InsertPage(int nNewPageIndex, HPROPSHEETPAGE hPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(hPage != NULL);
                return (BOOL)::SendMessage(m_hWnd, PSM_INSERTPAGE, nNewPageIndex, (LPARAM)hPage);
        }
        BOOL InsertPage(int nNewPageIndex, LPCPROPSHEETPAGE pPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(pPage != NULL);
                HPROPSHEETPAGE hPage = ::CreatePropertySheetPage(pPage);
                if(hPage == NULL)
                        return FALSE;
                return (BOOL)::SendMessage(m_hWnd, PSM_INSERTPAGE, nNewPageIndex, (LPARAM)hPage);
        }
        BOOL InsertPage(HPROPSHEETPAGE hPageInsertAfter, HPROPSHEETPAGE hPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(hPage != NULL);
                return (BOOL)::SendMessage(m_hWnd, PSM_INSERTPAGE, (WPARAM)hPageInsertAfter, (LPARAM)hPage);
        }
        BOOL InsertPage(HPROPSHEETPAGE hPageInsertAfter, LPCPROPSHEETPAGE pPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(pPage != NULL);
                HPROPSHEETPAGE hPage = ::CreatePropertySheetPage(pPage);
                if(hPage == NULL)
                        return FALSE;
                return (BOOL)::SendMessage(m_hWnd, PSM_INSERTPAGE, (WPARAM)hPageInsertAfter, (LPARAM)hPage);
        }
#endif 
        void RemovePage(int nPageIndex)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_REMOVEPAGE, nPageIndex, 0L);
        }
        void RemovePage(HPROPSHEETPAGE hPage)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(hPage != NULL);
                ::SendMessage(m_hWnd, PSM_REMOVEPAGE, 0, (LPARAM)hPage);
        }
        BOOL PressButton(int nButton)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (BOOL)::SendMessage(m_hWnd, PSM_PRESSBUTTON, nButton, 0L);
        }
        BOOL Apply()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (BOOL)::SendMessage(m_hWnd, PSM_APPLY, 0, 0L);
        }
        void CancelToClose()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_CANCELTOCLOSE, 0, 0L);
        }
        void SetModified(HWND hWndPage, BOOL bChanged = TRUE)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(::IsWindow(hWndPage));
                UINT uMsg = bChanged ? PSM_CHANGED : PSM_UNCHANGED;
                ::SendMessage(m_hWnd, uMsg, (WPARAM)hWndPage, 0L);
        }
        LRESULT QuerySiblings(WPARAM wParam, LPARAM lParam)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return ::SendMessage(m_hWnd, PSM_QUERYSIBLINGS, wParam, lParam);
        }
        void RebootSystem()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_REBOOTSYSTEM, 0, 0L);
        }
        void RestartWindows()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_RESTARTWINDOWS, 0, 0L);
        }
        BOOL IsDialogMessage(LPMSG lpMsg)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (BOOL)::SendMessage(m_hWnd, PSM_ISDIALOGMESSAGE, 0, (LPARAM)lpMsg);
        }
#if (_WIN32_IE >= 0x0500) && !defined(_WIN32_WCE)
        int HwndToIndex(HWND hWnd) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (int)::SendMessage(m_hWnd, PSM_HWNDTOINDEX, (WPARAM)hWnd, 0L);
        }
        HWND IndexToHwnd(int nIndex) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (HWND)::SendMessage(m_hWnd, PSM_INDEXTOHWND, nIndex, 0L);
        }
        int PageToIndex(HPROPSHEETPAGE hPage) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (int)::SendMessage(m_hWnd, PSM_PAGETOINDEX, 0, (LPARAM)hPage);
        }
        HPROPSHEETPAGE IndexToPage(int nIndex) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (HPROPSHEETPAGE)::SendMessage(m_hWnd, PSM_INDEXTOPAGE, nIndex, 0L);
        }
        int IdToIndex(int nID) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (int)::SendMessage(m_hWnd, PSM_IDTOINDEX, 0, nID);
        }
        int IndexToId(int nIndex) const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (int)::SendMessage(m_hWnd, PSM_INDEXTOID, nIndex, 0L);
        }
        int GetResult() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (int)::SendMessage(m_hWnd, PSM_GETRESULT, 0, 0L);
        }
        BOOL RecalcPageSizes()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (BOOL)::SendMessage(m_hWnd, PSM_RECALCPAGESIZES, 0, 0L);
        }
        void SetHeaderTitle(int nIndex, LPCTSTR lpstrHeaderTitle)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_SETHEADERTITLE, nIndex, (LPARAM)lpstrHeaderTitle);
        }
        void SetHeaderSubTitle(int nIndex, LPCTSTR lpstrHeaderSubTitle)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_SETHEADERSUBTITLE, nIndex, (LPARAM)lpstrHeaderSubTitle);
        }
#endif 
        HWND Create(LPCTSTR, HWND, ATL::_U_RECT = NULL, LPCTSTR = NULL, DWORD = 0, DWORD = 0, ATL::_U_MENUorID = 0U, LPVOID = NULL)
        {
                ATLASSERT(FALSE);
                return NULL;
        }
};
template <class T, class TBase = CPropertySheetWindow>
class ATL_NO_VTABLE CPropertySheetImpl : public ATL::CWindowImplBaseT< TBase >
{
public:
        PROPSHEETHEADER m_psh;
        ATL::CSimpleArray<HPROPSHEETPAGE> m_arrPages;
#if defined(_AYGSHELL_H_) || defined(__AYGSHELL_H__) 
  #ifndef PROPSHEET_LINK_SIZE
        #define PROPSHEET_LINK_SIZE 128
  #endif 
        TCHAR m_szLink[PROPSHEET_LINK_SIZE];
        static LPCTSTR m_pszTitle;
        static LPCTSTR m_pszLink;
#endif 
        CPropertySheetImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL, UINT uStartPage = 0, HWND hWndParent = NULL)
        {
                memset(&m_psh, 0, sizeof(PROPSHEETHEADER));
                m_psh.dwSize = sizeof(PROPSHEETHEADER);
                m_psh.dwFlags = PSH_USECALLBACK;
                m_psh.hInstance = ModuleHelper::GetResourceInstance();
                m_psh.phpage = NULL;   
                m_psh.nPages = 0;      
                m_psh.pszCaption = title.m_lpstr;
                m_psh.nStartPage = uStartPage;
                m_psh.hwndParent = hWndParent;   
                m_psh.pfnCallback = T::PropSheetCallback;
#if defined(_AYGSHELL_H_) || defined(__AYGSHELL_H__) 
                m_psh.dwFlags |= PSH_MAXIMIZE;
                m_szLink[0] = 0;
#endif 
        }
        ~CPropertySheetImpl()
        {
                if(m_arrPages.GetSize() > 0)   
                {
                        for(int i = 0; i < m_arrPages.GetSize(); i++)
                                ::DestroyPropertySheetPage((HPROPSHEETPAGE)m_arrPages[i]);
                }
        }
        static int CALLBACK PropSheetCallback(HWND hWnd, UINT uMsg, LPARAM lParam)
        {
                lParam;   
                int nRet = 0;
                if(uMsg == PSCB_INITIALIZED)
                {
                        ATLASSERT(hWnd != NULL);
                        T* pT = (T*)ModuleHelper::ExtractCreateWndData();
                        
                        pT->SubclassWindow(hWnd);
                        
                        pT->_CleanUpPages();
#if defined(_AYGSHELL_H_) || defined(__AYGSHELL_H__) 
                        m_pszTitle = pT->m_psh.pszCaption;
                        if(*pT->m_szLink != 0)
                                m_pszLink = pT->m_szLink;
#endif  
                        pT->OnSheetInitialized();
                }
#if defined(_AYGSHELL_H_) || defined(__AYGSHELL_H__) 
                else
                {
                        switch(uMsg)
                        {
                        case PSCB_GETVERSION :
                                nRet = COMCTL32_VERSION;
                                break;
                        case PSCB_GETTITLE :
                                if(m_pszTitle != NULL)
                                {
                                        lstrcpy((LPTSTR)lParam, m_pszTitle);
                                        m_pszTitle = NULL;
                                }
                                break;
                        case PSCB_GETLINKTEXT:
                                if(m_pszLink != NULL)
                                {
                                        lstrcpy((LPTSTR)lParam, m_pszLink);
                                        m_pszLink = NULL;
                                }
                                break;
                        default:
                                break;
                        }
                }
#endif 
                return nRet;
        }
        void OnSheetInitialized()
        {
        }
        HWND Create(HWND hWndParent = NULL)
        {
                ATLASSERT(m_hWnd == NULL);
                m_psh.dwFlags |= PSH_MODELESS;
                if(m_psh.hwndParent == NULL)
                        m_psh.hwndParent = hWndParent;
                m_psh.phpage = (HPROPSHEETPAGE*)m_arrPages.GetData();
                m_psh.nPages = m_arrPages.GetSize();
                T* pT = static_cast<T*>(this);
                ModuleHelper::AddCreateWndData(&pT->m_thunk.cd, pT);
                HWND hWnd = (HWND)::PropertySheet(&m_psh);
                _CleanUpPages();   
                ATLASSERT(m_hWnd == hWnd);
                return hWnd;
        }
        INT_PTR DoModal(HWND hWndParent = ::GetActiveWindow())
        {
                ATLASSERT(m_hWnd == NULL);
                m_psh.dwFlags &= ~PSH_MODELESS;
                if(m_psh.hwndParent == NULL)
                        m_psh.hwndParent = hWndParent;
                m_psh.phpage = (HPROPSHEETPAGE*)m_arrPages.GetData();
                m_psh.nPages = m_arrPages.GetSize();
                T* pT = static_cast<T*>(this);
                ModuleHelper::AddCreateWndData(&pT->m_thunk.cd, pT);
                INT_PTR nRet = ::PropertySheet(&m_psh);
                _CleanUpPages();   
                return nRet;
        }
        
        void _CleanUpPages()
        {
                m_psh.nPages = 0;
                m_psh.phpage = NULL;
                m_arrPages.RemoveAll();
        }
        int GetPageCount() const
        {
                if(m_hWnd == NULL)   
                        return m_arrPages.GetSize();
                return TBase::GetPageCount();
        }
        int GetActiveIndex() const
        {
                if(m_hWnd == NULL)   
                        return m_psh.nStartPage;
                return TBase::GetActiveIndex();
        }
        HPROPSHEETPAGE GetPage(int nPageIndex) const
        {
                ATLASSERT(m_hWnd == NULL);   
                return (HPROPSHEETPAGE)m_arrPages[nPageIndex];
        }
        int GetPageIndex(HPROPSHEETPAGE hPage) const
        {
                ATLASSERT(m_hWnd == NULL);   
                return m_arrPages.Find((HPROPSHEETPAGE&)hPage);
        }
        BOOL SetActivePage(int nPageIndex)
        {
                if(m_hWnd == NULL)   
                {
                        ATLASSERT(nPageIndex >= 0 && nPageIndex < m_arrPages.GetSize());
                        m_psh.nStartPage = nPageIndex;
                        return TRUE;
                }
                return TBase::SetActivePage(nPageIndex);
        }
        BOOL SetActivePage(HPROPSHEETPAGE hPage)
        {
                ATLASSERT(hPage != NULL);
                if (m_hWnd == NULL)   
                {
                        int nPageIndex = GetPageIndex(hPage);
                        if(nPageIndex == -1)
                                return FALSE;
                        return SetActivePage(nPageIndex);
                }
                return TBase::SetActivePage(hPage);
        }
        void SetTitle(LPCTSTR lpszText, UINT nStyle = 0)
        {
                ATLASSERT((nStyle & ~PSH_PROPTITLE) == 0);   
                ATLASSERT(lpszText != NULL);
                if(m_hWnd == NULL)
                {
                        
                        m_psh.pszCaption = lpszText;   
                        m_psh.dwFlags &= ~PSH_PROPTITLE;
                        m_psh.dwFlags |= nStyle;
                }
                else
                {
                        
                        TBase::SetTitle(lpszText, nStyle);
                }
        }
#if defined(_AYGSHELL_H_) || defined(__AYGSHELL_H__) 
        void SetLinkText(LPCTSTR lpszText)
        {
                ATLASSERT(lpszText != NULL);
                ATLASSERT(lstrlen(lpszText) < PROPSHEET_LINK_SIZE);
                lstrcpy(m_szLink, lpszText);
        }
#endif 
        void SetWizardMode()
        {
                m_psh.dwFlags |= PSH_WIZARD;
        }
        void EnableHelp()
        {
                m_psh.dwFlags |= PSH_HASHELP;
        }
        BOOL AddPage(HPROPSHEETPAGE hPage)
        {
                ATLASSERT(hPage != NULL);
                BOOL bRet = FALSE;
                if(m_hWnd != NULL)
                        bRet = TBase::AddPage(hPage);
                else    
                        bRet = m_arrPages.Add((HPROPSHEETPAGE&)hPage);
                return bRet;
        }
        BOOL AddPage(LPCPROPSHEETPAGE pPage)
        {
                ATLASSERT(pPage != NULL);
                HPROPSHEETPAGE hPage = ::CreatePropertySheetPage(pPage);
                if(hPage == NULL)
                        return FALSE;
                BOOL bRet = AddPage(hPage);
                if(!bRet)
                        ::DestroyPropertySheetPage(hPage);
                return bRet;
        }
        BOOL RemovePage(HPROPSHEETPAGE hPage)
        {
                ATLASSERT(hPage != NULL);
                if (m_hWnd == NULL)   
                {
                        int nPage = GetPageIndex(hPage);
                        if(nPage == -1)
                                return FALSE;
                        return RemovePage(nPage);
                }
                TBase::RemovePage(hPage);
                return TRUE;
        }
        BOOL RemovePage(int nPageIndex)
        {
                BOOL bRet = TRUE;
                if(m_hWnd != NULL)
                        TBase::RemovePage(nPageIndex);
                else    
                        bRet = m_arrPages.RemoveAt(nPageIndex);
                return bRet;
        }
#if (_WIN32_IE >= 0x0400) && !defined(_WIN32_WCE)
        void SetHeader(LPCTSTR szbmHeader)
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psh.dwFlags &= ~PSH_WIZARD;
                m_psh.dwFlags |= (PSH_HEADER | PSH_WIZARD97);
                m_psh.pszbmHeader = szbmHeader;
        }
        void SetHeader(HBITMAP hbmHeader)
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psh.dwFlags &= ~PSH_WIZARD;
                m_psh.dwFlags |= (PSH_HEADER | PSH_USEHBMHEADER | PSH_WIZARD97);
                m_psh.hbmHeader = hbmHeader;
        }
        void SetWatermark(LPCTSTR szbmWatermark, HPALETTE hplWatermark = NULL)
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psh.dwFlags &= ~PSH_WIZARD;
                m_psh.dwFlags |= PSH_WATERMARK | PSH_WIZARD97;
                m_psh.pszbmWatermark = szbmWatermark;
                if (hplWatermark != NULL)
                {
                        m_psh.dwFlags |= PSH_USEHPLWATERMARK;
                        m_psh.hplWatermark = hplWatermark;
                }
        }
        void SetWatermark(HBITMAP hbmWatermark, HPALETTE hplWatermark = NULL)
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psh.dwFlags &= ~PSH_WIZARD;
                m_psh.dwFlags |= (PSH_WATERMARK | PSH_USEHBMWATERMARK | PSH_WIZARD97);
                m_psh.hbmWatermark = hbmWatermark;
                if (hplWatermark != NULL)
                {
                        m_psh.dwFlags |= PSH_USEHPLWATERMARK;
                        m_psh.hplWatermark = hplWatermark;
                }
        }
        void StretchWatermark(bool bStretchWatermark)
        {
                ATLASSERT(m_hWnd == NULL);   
                if (bStretchWatermark)
                        m_psh.dwFlags |= PSH_STRETCHWATERMARK;
                else
                        m_psh.dwFlags &= ~PSH_STRETCHWATERMARK;
        }
#endif 
        BEGIN_MSG_MAP(CPropertySheetImpl)
                MESSAGE_HANDLER(WM_COMMAND, OnCommand)
                MESSAGE_HANDLER(WM_SYSCOMMAND, OnSysCommand)
        END_MSG_MAP()
        LRESULT OnCommand(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& )
        {
                LRESULT lRet = DefWindowProc(uMsg, wParam, lParam);
                if(HIWORD(wParam) == BN_CLICKED && (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) &&
                   ((m_psh.dwFlags & PSH_MODELESS) != 0) && (GetActivePage() == NULL))
                        DestroyWindow();
                return lRet;
        }
        LRESULT OnSysCommand(UINT , WPARAM wParam, LPARAM , BOOL& bHandled)
        {
                if(((m_psh.dwFlags & PSH_MODELESS) == PSH_MODELESS) && ((wParam & 0xFFF0) == SC_CLOSE))
                        SendMessage(WM_CLOSE);
                else
                        bHandled = FALSE;
                return 0;
        }
};
#if defined(_AYGSHELL_H_) || defined(__AYGSHELL_H__) 
template < class T, class TBase >
LPCWSTR CPropertySheetImpl<T,TBase>::m_pszTitle = NULL;
template < class T, class TBase>
LPCWSTR CPropertySheetImpl<T,TBase>::m_pszLink = NULL;
#endif 
class CPropertySheet : public CPropertySheetImpl<CPropertySheet>
{
public:
        CPropertySheet(ATL::_U_STRINGorID title = (LPCTSTR)NULL, UINT uStartPage = 0, HWND hWndParent = NULL)
                : CPropertySheetImpl<CPropertySheet>(title, uStartPage, hWndParent)
        { }
};
class CPropertyPageWindow : public ATL::CWindow
{
public:
        CPropertyPageWindow(HWND hWnd = NULL) : ATL::CWindow(hWnd)
        { }
        CPropertyPageWindow& operator =(HWND hWnd)
        {
                m_hWnd = hWnd;
                return *this;
        }
        CPropertySheetWindow GetPropertySheet() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return CPropertySheetWindow(GetParent());
        }
        BOOL Apply()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                return GetPropertySheet().Apply();
        }
        void CancelToClose()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetPropertySheet().CancelToClose();
        }
        void SetModified(BOOL bChanged = TRUE)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetPropertySheet().SetModified(m_hWnd, bChanged);
        }
        LRESULT QuerySiblings(WPARAM wParam, LPARAM lParam)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                return GetPropertySheet().QuerySiblings(wParam, lParam);
        }
        void RebootSystem()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetPropertySheet().RebootSystem();
        }
        void RestartWindows()
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetPropertySheet().RestartWindows();
        }
        void SetWizardButtons(DWORD dwFlags)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetPropertySheet().SetWizardButtons(dwFlags);
        }
        HWND Create(LPCTSTR, HWND, ATL::_U_RECT = NULL, LPCTSTR = NULL, DWORD = 0, DWORD = 0, ATL::_U_MENUorID = 0U, LPVOID = NULL)
        {
                ATLASSERT(FALSE);
                return NULL;
        }
};
template <class T, class TBase = CPropertyPageWindow>
class ATL_NO_VTABLE CPropertyPageImpl : public ATL::CDialogImplBaseT< TBase >
{
public:
        PROPSHEETPAGE m_psp;
        operator PROPSHEETPAGE*() { return &m_psp; }
        CPropertyPageImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL)
        {
                
                memset(&m_psp, 0, sizeof(PROPSHEETPAGE));
                m_psp.dwSize = sizeof(PROPSHEETPAGE);
                m_psp.dwFlags = PSP_USECALLBACK;
                m_psp.hInstance = ModuleHelper::GetResourceInstance();
                T* pT = static_cast<T*>(this);
                m_psp.pszTemplate = MAKEINTRESOURCE(pT->IDD);
                m_psp.pfnDlgProc = (DLGPROC)T::StartDialogProc;
                m_psp.pfnCallback = T::PropPageCallback;
                m_psp.lParam = (LPARAM)pT;
                if(title.m_lpstr != NULL)
                        SetTitle(title);
        }
        static UINT CALLBACK PropPageCallback(HWND hWnd, UINT uMsg, LPPROPSHEETPAGE ppsp)
        {
                hWnd;   
                ATLASSERT(hWnd == NULL);
                T* pT = (T*)ppsp->lParam;
                UINT uRet = 0;
                switch(uMsg)
                {
                case PSPCB_CREATE:
                        {
                                ATL::CDialogImplBaseT< TBase >* pPage = (ATL::CDialogImplBaseT< TBase >*)pT;
                                ModuleHelper::AddCreateWndData(&pPage->m_thunk.cd, pPage);
                                uRet = pT->OnPageCreate() ? 1 : 0;
                        }
                        break;
#if (_WIN32_IE >= 0x0500)
                case PSPCB_ADDREF:
                        pT->OnPageAddRef();
                        break;
#endif 
                case PSPCB_RELEASE:
                        pT->OnPageRelease();
                        break;
                default:
                        break;
                }
                return uRet;
        }
        bool OnPageCreate()
        {
                return true;   
        }
#if (_WIN32_IE >= 0x0500)
        void OnPageAddRef()
        {
        }
#endif 
        void OnPageRelease()
        {
        }
        HPROPSHEETPAGE Create()
        {
                return ::CreatePropertySheetPage(&m_psp);
        }
        void SetTitle(ATL::_U_STRINGorID title)
        {
                m_psp.pszTitle = title.m_lpstr;
                m_psp.dwFlags |= PSP_USETITLE;
        }
#if (_WIN32_IE >= 0x0500) && !defined(_WIN32_WCE)
        void SetHeaderTitle(LPCTSTR lpstrHeaderTitle)
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psp.dwFlags |= PSP_USEHEADERTITLE;
                m_psp.pszHeaderTitle = lpstrHeaderTitle;
        }
        void SetHeaderSubTitle(LPCTSTR lpstrHeaderSubTitle)
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psp.dwFlags |= PSP_USEHEADERSUBTITLE;
                m_psp.pszHeaderSubTitle = lpstrHeaderSubTitle;
        }
#endif 
        void EnableHelp()
        {
                m_psp.dwFlags |= PSP_HASHELP;
        }
        BEGIN_MSG_MAP(CPropertyPageImpl)
                MESSAGE_HANDLER(WM_NOTIFY, OnNotify)
        END_MSG_MAP()
        
        
        LRESULT OnNotify(UINT , WPARAM , LPARAM lParam, BOOL& bHandled)
        {
#ifndef _WIN32_WCE
                
                ATLASSERT(::IsWindow(m_hWnd));
#endif
                NMHDR* pNMHDR = (NMHDR*)lParam;
                
                if(pNMHDR->hwndFrom != m_hWnd && pNMHDR->hwndFrom != ::GetParent(m_hWnd))
                {
                        bHandled = FALSE;
                        return 1;
                }
#ifdef _WIN32_WCE
                ATLASSERT(::IsWindow(m_hWnd));
#endif
                T* pT = static_cast<T*>(this);
                LRESULT lResult = 0;
                switch(pNMHDR->code)
                {
#ifdef _WTL_NEW_PAGE_NOTIFY_HANDLERS
                case PSN_SETACTIVE:
                        lResult = pT->OnSetActive();
                        break;
                case PSN_KILLACTIVE:
                        lResult = pT->OnKillActive();
                        break;
                case PSN_APPLY:
                        lResult = pT->OnApply();
                        break;
                case PSN_RESET:
                        pT->OnReset();
                        break;
                case PSN_QUERYCANCEL:
                        lResult = pT->OnQueryCancel();
                        break;
                case PSN_WIZNEXT:
                        lResult = pT->OnWizardNext();
                        break;
                case PSN_WIZBACK:
                        lResult = pT->OnWizardBack();
                        break;
                case PSN_WIZFINISH:
                        lResult = pT->OnWizardFinish();
                        break;
                case PSN_HELP:
                        pT->OnHelp();
                        break;
#ifndef _WIN32_WCE
#if (_WIN32_IE >= 0x0400)
                case PSN_GETOBJECT:
                        if(!pT->OnGetObject((LPNMOBJECTNOTIFY)lParam))
                                bHandled = FALSE;
                        break;
#endif 
#if (_WIN32_IE >= 0x0500)
                case PSN_TRANSLATEACCELERATOR:
                        {
                                LPPSHNOTIFY lpPSHNotify = (LPPSHNOTIFY)lParam;
                                lResult = pT->OnTranslateAccelerator((LPMSG)lpPSHNotify->lParam);
                        }
                        break;
                case PSN_QUERYINITIALFOCUS:
                        {
                                LPPSHNOTIFY lpPSHNotify = (LPPSHNOTIFY)lParam;
                                lResult = (LRESULT)pT->OnQueryInitialFocus((HWND)lpPSHNotify->lParam);
                        }
                        break;
#endif 
#endif 
#else 
                case PSN_SETACTIVE:
                        lResult = pT->OnSetActive() ? 0 : -1;
                        break;
                case PSN_KILLACTIVE:
                        lResult = !pT->OnKillActive();
                        break;
                case PSN_APPLY:
                        lResult = pT->OnApply() ? PSNRET_NOERROR : PSNRET_INVALID_NOCHANGEPAGE;
                        break;
                case PSN_RESET:
                        pT->OnReset();
                        break;
                case PSN_QUERYCANCEL:
                        lResult = !pT->OnQueryCancel();
                        break;
                case PSN_WIZNEXT:
                        lResult = pT->OnWizardNext();
                        break;
                case PSN_WIZBACK:
                        lResult = pT->OnWizardBack();
                        break;
                case PSN_WIZFINISH:
                        lResult = !pT->OnWizardFinish();
                        break;
                case PSN_HELP:
                        pT->OnHelp();
                        break;
#ifndef _WIN32_WCE
#if (_WIN32_IE >= 0x0400)
                case PSN_GETOBJECT:
                        if(!pT->OnGetObject((LPNMOBJECTNOTIFY)lParam))
                                bHandled = FALSE;
                        break;
#endif 
#if (_WIN32_IE >= 0x0500)
                case PSN_TRANSLATEACCELERATOR:
                        {
                                LPPSHNOTIFY lpPSHNotify = (LPPSHNOTIFY)lParam;
                                lResult = pT->OnTranslateAccelerator((LPMSG)lpPSHNotify->lParam) ? PSNRET_MESSAGEHANDLED : PSNRET_NOERROR;
                        }
                        break;
                case PSN_QUERYINITIALFOCUS:
                        {
                                LPPSHNOTIFY lpPSHNotify = (LPPSHNOTIFY)lParam;
                                lResult = (LRESULT)pT->OnQueryInitialFocus((HWND)lpPSHNotify->lParam);
                        }
                        break;
#endif 
#endif 
#endif 
                default:
                        bHandled = FALSE;   
                }
                return lResult;
        }
        
        
#ifdef _WTL_NEW_PAGE_NOTIFY_HANDLERS
        int OnSetActive()
        {
                
                
                
                return 0;
        }
        BOOL OnKillActive()
        {
                
                
                return FALSE;
        }
        int OnApply()
        {
                
                
                
                return PSNRET_NOERROR;
        }
        void OnReset()
        {
        }
        BOOL OnQueryCancel()
        {
                
                
                return FALSE;
        }
        int OnWizardBack()
        {
                
                
                
                return 0;
        }
        int OnWizardNext()
        {
                
                
                
                return 0;
        }
        INT_PTR OnWizardFinish()
        {
                
                
                
                return FALSE;
        }
        void OnHelp()
        {
        }
#ifndef _WIN32_WCE
#if (_WIN32_IE >= 0x0400)
        BOOL OnGetObject(LPNMOBJECTNOTIFY )
        {
                return FALSE;   
        }
#endif 
#if (_WIN32_IE >= 0x0500)
        int OnTranslateAccelerator(LPMSG )
        {
                
                
                return PSNRET_NOERROR;
        }
        HWND OnQueryInitialFocus(HWND )
        {
                
                
                return NULL;
        }
#endif 
#endif 
#else 
        BOOL OnSetActive()
        {
                return TRUE;
        }
        BOOL OnKillActive()
        {
                return TRUE;
        }
        BOOL OnApply()
        {
                return TRUE;
        }
        void OnReset()
        {
        }
        BOOL OnQueryCancel()
        {
                return TRUE;    
        }
        int OnWizardBack()
        {
                
                
                
                return 0;
        }
        int OnWizardNext()
        {
                
                
                
                return 0;
        }
        BOOL OnWizardFinish()
        {
                return TRUE;
        }
        void OnHelp()
        {
        }
#ifndef _WIN32_WCE
#if (_WIN32_IE >= 0x0400)
        BOOL OnGetObject(LPNMOBJECTNOTIFY )
        {
                return FALSE;   
        }
#endif 
#if (_WIN32_IE >= 0x0500)
        BOOL OnTranslateAccelerator(LPMSG )
        {
                return FALSE;   
        }
        HWND OnQueryInitialFocus(HWND )
        {
                return NULL;   
        }
#endif 
#endif 
#endif 
};
template <WORD t_wDlgTemplateID>
class CPropertyPage : public CPropertyPageImpl<CPropertyPage<t_wDlgTemplateID> >
{
public:
        enum { IDD = t_wDlgTemplateID };
        CPropertyPage(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : CPropertyPageImpl<CPropertyPage>(title)
        { }
        DECLARE_EMPTY_MSG_MAP()
};
#ifndef _ATL_NO_HOSTING
template <class T, class TBase = CPropertyPageWindow>
class ATL_NO_VTABLE CAxPropertyPageImpl : public CPropertyPageImpl< T, TBase >
{
public:
        HGLOBAL m_hInitData;
        HGLOBAL m_hDlgRes;
        HGLOBAL m_hDlgResSplit;
        CAxPropertyPageImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : 
                        CPropertyPageImpl< T, TBase >(title),
                        m_hInitData(NULL), m_hDlgRes(NULL), m_hDlgResSplit(NULL)
        {
                T* pT = static_cast<T*>(this);
                pT;   
                
                ATL::AtlAxWinInit();
                HINSTANCE hInstance = ModuleHelper::GetResourceInstance();
                LPCTSTR lpTemplateName = MAKEINTRESOURCE(pT->IDD);
                HRSRC hDlg = ::FindResource(hInstance, lpTemplateName, (LPTSTR)RT_DIALOG);
                if(hDlg != NULL)
                {
                        HRSRC hDlgInit = ::FindResource(hInstance, lpTemplateName, (LPTSTR)_ATL_RT_DLGINIT);
                        BYTE* pInitData = NULL;
                        if(hDlgInit != NULL)
                        {
                                m_hInitData = ::LoadResource(hInstance, hDlgInit);
                                pInitData = (BYTE*)::LockResource(m_hInitData);
                        }
                        m_hDlgRes = ::LoadResource(hInstance, hDlg);
                        DLGTEMPLATE* pDlg = (DLGTEMPLATE*)::LockResource(m_hDlgRes);
                        LPCDLGTEMPLATE lpDialogTemplate = ATL::_DialogSplitHelper::SplitDialogTemplate(pDlg, pInitData);
                        if(lpDialogTemplate != pDlg)
                                m_hDlgResSplit = GlobalHandle(lpDialogTemplate);
                        
                        if(lpDialogTemplate != NULL)
                        {
                                m_psp.dwFlags |= PSP_DLGINDIRECT;
                                m_psp.pResource = lpDialogTemplate;
                        }
                        else
                        {
                                ATLASSERT(FALSE && _T("CAxPropertyPageImpl - ActiveX initializtion failed!"));
                        }
                }
                else
                {
                        ATLASSERT(FALSE && _T("CAxPropertyPageImpl - Cannot find dialog template!"));
                }
        }
        ~CAxPropertyPageImpl()
        {
                if(m_hInitData != NULL)
                {
                        UnlockResource(m_hInitData);
                        FreeResource(m_hInitData);
                }
                if(m_hDlgRes != NULL)
                {
                        UnlockResource(m_hDlgRes);
                        FreeResource(m_hDlgRes);
                }
                if(m_hDlgResSplit != NULL)
                {
                        ::GlobalFree(m_hDlgResSplit);
                }
        }
        
        BOOL PreTranslateMessage(LPMSG pMsg)
        {
                if ((pMsg->message < WM_KEYFIRST || pMsg->message > WM_KEYLAST) &&
                   (pMsg->message < WM_MOUSEFIRST || pMsg->message > WM_MOUSELAST))
                        return FALSE;
                
                HWND hWndCtl = ::GetFocus();
                if (IsChild(hWndCtl) && ::GetParent(hWndCtl) != m_hWnd)
                {
                        do
                        {
                                hWndCtl = ::GetParent(hWndCtl);
                        }
                        while (::GetParent(hWndCtl) != m_hWnd);
                }
                
                return (BOOL)::SendMessage(hWndCtl, WM_FORWARDMSG, 0, (LPARAM)pMsg);
        }
#if (_WIN32_IE >= 0x0500)
        
#ifdef _WTL_NEW_PAGE_NOTIFY_HANDLERS
        int OnTranslateAccelerator(LPMSG lpMsg)
        {
                T* pT = static_cast<T*>(this);
                return (pT->PreTranslateMessage(lpMsg) != FALSE) ? PSNRET_MESSAGEHANDLED : PSNRET_NOERROR;
        }
#else 
        BOOL OnTranslateAccelerator(LPMSG lpMsg)
        {
                T* pT = static_cast<T*>(this);
                return pT->PreTranslateMessage(lpMsg);
        }
#endif 
#endif 
#if (_ATL_VER >= 0x0700)
        int GetIDD()
        {
                return( static_cast<T*>(this)->IDD );
        }
        virtual DLGPROC GetDialogProc()
        {
                return DialogProc;
        }
        static INT_PTR CALLBACK DialogProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
        {
                CAxPropertyPageImpl< T, TBase >* pThis = (CAxPropertyPageImpl< T, TBase >*)hWnd;
                if (uMsg == WM_INITDIALOG)
                {
                        HRESULT hr;
                        if (FAILED(hr = pThis->CreateActiveXControls(pThis->GetIDD())))
                        {
                                ATLASSERT(FALSE);
                                return FALSE;
                        }
                }
                return CPropertyPageImpl< T, TBase >::DialogProc(hWnd, uMsg, wParam, lParam);
        }
        virtual HRESULT CreateActiveXControls(UINT nID)
        {
                
                HRSRC hDlgInit = ::FindResource(ATL::_AtlBaseModule.GetResourceInstance(), MAKEINTRESOURCE(nID), (LPTSTR)_ATL_RT_DLGINIT);
                BYTE* pInitData = NULL;
                HGLOBAL hData = NULL;
                HRESULT hr = S_OK;
                if (hDlgInit != NULL)
                {
                        hData = ::LoadResource(ATL::_AtlBaseModule.GetResourceInstance(), hDlgInit);
                        if (hData != NULL)
                                pInitData = (BYTE*) ::LockResource(hData);
                }
                HRSRC hDlg = ::FindResource(ATL::_AtlBaseModule.GetResourceInstance(), MAKEINTRESOURCE(nID), (LPTSTR)RT_DIALOG);
                if (hDlg != NULL)
                {
                        HGLOBAL hResource = ::LoadResource(ATL::_AtlBaseModule.GetResourceInstance(), hDlg);
                        DLGTEMPLATE* pDlg = NULL;
                        if (hResource != NULL)
                        {
                                pDlg = (DLGTEMPLATE*) ::LockResource(hResource);
                                if (pDlg != NULL)
                                {
                                        
                                        BOOL bDialogEx = ATL::_DialogSplitHelper::IsDialogEx(pDlg);
                                        WORD nItems = ATL::_DialogSplitHelper::DlgTemplateItemCount(pDlg);
                                        
                                        DLGITEMTEMPLATE* pItem = ATL::_DialogSplitHelper::FindFirstDlgItem(pDlg);
                                        HWND hWndPrev = GetWindow(GW_CHILD);
                                        
                                        for (WORD nItem = 0; nItem < nItems; nItem++)
                                        {
                                                DWORD wID = bDialogEx ? ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->id : pItem->id;
                                                if (ATL::_DialogSplitHelper::IsActiveXControl(pItem, bDialogEx))
                                                {
                                                        BYTE* pData = NULL;
                                                        DWORD dwLen = ATL::_DialogSplitHelper::FindCreateData(wID, pInitData, &pData);
                                                        ATL::CComPtr<IStream> spStream;
                                                        if (dwLen != 0)
                                                        {
                                                                HGLOBAL h = GlobalAlloc(GHND, dwLen);
                                                                if (h != NULL)
                                                                {
                                                                        BYTE* pBytes = (BYTE*) GlobalLock(h);
                                                                        BYTE* pSource = pData; 
                                                                        SecureHelper::memcpy_x(pBytes, dwLen, pSource, dwLen);
                                                                        GlobalUnlock(h);
                                                                        CreateStreamOnHGlobal(h, TRUE, &spStream);
                                                                }
                                                                else
                                                                {
                                                                        hr = E_OUTOFMEMORY;
                                                                        break;
                                                                }
                                                        }
                                                        ATL::CComBSTR bstrLicKey;
                                                        hr = ATL::_DialogSplitHelper::ParseInitData(spStream, &bstrLicKey.m_str);
                                                        if (SUCCEEDED(hr))
                                                        {
                                                                ATL::CAxWindow2 wnd;
                                                                
                                                                LPWSTR pszClassName = 
                                                                        bDialogEx ? 
                                                                                (LPWSTR)(((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem) + 1) :
                                                                                (LPWSTR)(pItem + 1);
                                                                
                                                                RECT rect;
                                                                rect.left = 
                                                                        bDialogEx ? 
                                                                                ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->x : 
                                                                                pItem->x;
                                                                rect.top = 
                                                                        bDialogEx ? 
                                                                                ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->y : 
                                                                                pItem->y;
                                                                rect.right = rect.left + 
                                                                        (bDialogEx ? 
                                                                                ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->cx : 
                                                                                pItem->cx);
                                                                rect.bottom = rect.top + 
                                                                        (bDialogEx ? 
                                                                                ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->cy : 
                                                                                pItem->cy);
                                                                
                                                                MapDialogRect(&rect);
                                                                
                                                                wnd.Create(m_hWnd, 
                                                                        &rect, 
                                                                        NULL, 
                                                                        (bDialogEx ? 
                                                                                ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->style : 
                                                                                pItem->style) | WS_TABSTOP, 
                                                                        bDialogEx ? 
                                                                                ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->exStyle : 
                                                                                0,
                                                                        bDialogEx ? 
                                                                                ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->id : 
                                                                                pItem->id,
                                                                        NULL);
                                                                if (wnd != NULL)
                                                                {
#ifndef _WIN32_WCE
                                                                        
                                                                        if (bDialogEx && ((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->helpID != 0)
                                                                                wnd.SetWindowContextHelpId(((ATL::_DialogSplitHelper::DLGITEMTEMPLATEEX*)pItem)->helpID);
#endif 
                                                                        
                                                                        hr = wnd.CreateControlLic(pszClassName, spStream, NULL, bstrLicKey);
                                                                        if (FAILED(hr))
                                                                                break;
                                                                        
                                                                        if (nItem == 0)
                                                                                hWndPrev = HWND_TOP;
                                                                        wnd.SetWindowPos(hWndPrev, 0,0,0,0,SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE);
                                                                        hWndPrev = wnd;
                                                                }
                                                                else
                                                                {
                                                                        hr = ATL::AtlHresultFromLastError();
                                                                }
                                                        }
                                                }
                                                else
                                                {
                                                        if (nItem != 0)
                                                                hWndPrev = ::GetWindow(hWndPrev, GW_HWNDNEXT);
                                                }
                                                pItem = ATL::_DialogSplitHelper::FindNextDlgItem(pItem, bDialogEx);
                                        }
                                }
                                else
                                        hr = ATL::AtlHresultFromLastError();
                        }
                        else
                                hr = ATL::AtlHresultFromLastError();
                }
                return hr;
        }
        HRESULT AdviseSinkMap(bool bAdvise)
        {
                if(!bAdvise && m_hWnd == NULL)
                {
                        
                        ATLTRACE2(atlTraceUI, 0, _T("CAxPropertyPageImpl::AdviseSinkMap called after the window was destroyed\n"));
                        return S_OK;
                }
                HRESULT hRet = E_NOTIMPL;
                __if_exists(T::_GetSinkMapFinder)
                {
                        T* pT = static_cast<T*>(this);
                        hRet = AtlAdviseSinkMap(pT, bAdvise);
                }
                return hRet;
        }
        typedef CPropertyPageImpl< T, TBase>   _baseClass;
        BEGIN_MSG_MAP(CAxPropertyPageImpl)
                MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog)
                MESSAGE_HANDLER(WM_DESTROY, OnDestroy)
                CHAIN_MSG_MAP(_baseClass)
        END_MSG_MAP()
        LRESULT OnInitDialog(UINT , WPARAM , LPARAM , BOOL& bHandled)
        {
                
                ExecuteDlgInit(static_cast<T*>(this)->IDD);
                AdviseSinkMap(true);
                bHandled = FALSE;
                return 1;
        }
        LRESULT OnDestroy(UINT , WPARAM , LPARAM , BOOL& bHandled)
        {
                AdviseSinkMap(false);
                bHandled = FALSE;
                return 1;
        }
#endif 
};
template <WORD t_wDlgTemplateID>
class CAxPropertyPage : public CAxPropertyPageImpl<CAxPropertyPage<t_wDlgTemplateID> >
{
public:
        enum { IDD = t_wDlgTemplateID };
        CAxPropertyPage(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : CAxPropertyPageImpl<CAxPropertyPage>(title)
        { }
#if (_WIN32_IE >= 0x0500) || (_ATL_VER >= 0x0700)
        
        BEGIN_MSG_MAP(CAxPropertyPage)
                CHAIN_MSG_MAP(CAxPropertyPageImpl<CAxPropertyPage<t_wDlgTemplateID> >)
        END_MSG_MAP()
#else 
        DECLARE_EMPTY_MSG_MAP()
#endif 
};
#endif 
#if (_WIN32_IE >= 0x0500) && !defined(_WIN32_WCE)
class CWizard97SheetWindow : public CPropertySheetWindow
{
public:
        CWizard97SheetWindow(HWND hWnd = NULL) : CPropertySheetWindow(hWnd)
        { }
        CWizard97SheetWindow& operator =(HWND hWnd)
        {
                m_hWnd = hWnd;
                return *this;
        }
        HFONT GetExteriorPageTitleFont(void)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (HFONT)::SendMessage(m_hWnd, GetMessage_GetExteriorPageTitleFont(), 0, 0L);
        }
        HFONT GetBulletFont(void)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return (HFONT)::SendMessage(m_hWnd, GetMessage_GetBulletFont(), 0, 0L);
        }
        static UINT GetMessage_GetExteriorPageTitleFont()
        {
                static UINT uGetExteriorPageTitleFont = 0;
                if(uGetExteriorPageTitleFont == 0)
                {
                        CStaticDataInitCriticalSectionLock lock;
                        if(FAILED(lock.Lock()))
                        {
                                ATLTRACE2(atlTraceUI, 0, _T("ERROR : Unable to lock critical section in CWizard97SheetWindow::GetMessage_GetExteriorPageTitleFont().\n"));
                                ATLASSERT(FALSE);
                                return 0;
                        }
                        if(uGetExteriorPageTitleFont == 0)
                                uGetExteriorPageTitleFont = ::RegisterWindowMessage(_T("GetExteriorPageTitleFont_531AF056-B8BE-4c4c-B786-AC608DF0DF12"));
                        lock.Unlock();
                }
                ATLASSERT(uGetExteriorPageTitleFont != 0);
                return uGetExteriorPageTitleFont;
        }
        static UINT GetMessage_GetBulletFont()
        {
                static UINT uGetBulletFont = 0;
                if(uGetBulletFont == 0)
                {
                        CStaticDataInitCriticalSectionLock lock;
                        if(FAILED(lock.Lock()))
                        {
                                ATLTRACE2(atlTraceUI, 0, _T("ERROR : Unable to lock critical section in CWizard97SheetWindow::GetMessage_GetBulletFont().\n"));
                                ATLASSERT(FALSE);
                                return 0;
                        }
                        if(uGetBulletFont == 0)
                                uGetBulletFont = ::RegisterWindowMessage(_T("GetBulletFont_AD347D08-8F65-45ef-982E-6352E8218AD5"));
                        lock.Unlock();
                }
                ATLASSERT(uGetBulletFont != 0);
                return uGetBulletFont;
        }
        HWND Create(LPCTSTR, HWND, ATL::_U_RECT = NULL, LPCTSTR = NULL, DWORD = 0, DWORD = 0, ATL::_U_MENUorID = 0U, LPVOID = NULL)
        {
                ATLASSERT(FALSE);
                return NULL;
        }
};
template <class T, class TBase = CWizard97SheetWindow>
class ATL_NO_VTABLE CWizard97SheetImpl : public CPropertySheetImpl< T, TBase >
{
protected:
        typedef CWizard97SheetImpl< T, TBase > thisClass;
        typedef CPropertySheetImpl< T, TBase > baseClass;
        CFont m_fontExteriorPageTitle;   
        CFont m_fontBullet;              
        bool m_bReceivedFirstSizeMessage;   
public:
        CWizard97SheetImpl(ATL::_U_STRINGorID title, ATL::_U_STRINGorID headerBitmap, ATL::_U_STRINGorID watermarkBitmap, UINT uStartPage = 0, HWND hWndParent = NULL) :
                        baseClass(title, uStartPage, hWndParent),
                        m_bReceivedFirstSizeMessage(false)
        {
                m_psh.dwFlags &= ~(PSH_NOCONTEXTHELP);
                m_psh.dwFlags &= ~(PSH_WIZARD | PSH_WIZARD_LITE);
                m_psh.dwFlags |= (PSH_HASHELP | PSH_WIZARDCONTEXTHELP);
                m_psh.dwFlags |= PSH_WIZARD97;
                baseClass::SetHeader(headerBitmap.m_lpstr);
                baseClass::SetWatermark(watermarkBitmap.m_lpstr);
        }
        void OnSheetInitialized()
        {
                T* pT = static_cast<T*>(this);
                pT->_InitializeFonts();
                
                
        }
        void _InitializeFonts()
        {
                
                
                
                CFontHandle fontThisDialog = this->GetFont();
                CClientDC dcScreen(NULL);
                LOGFONT titleLogFont = {0};
                LOGFONT bulletLogFont = {0};
                fontThisDialog.GetLogFont(&titleLogFont);
                fontThisDialog.GetLogFont(&bulletLogFont);
                
                
                titleLogFont.lfCharSet = DEFAULT_CHARSET;
                titleLogFont.lfWeight = FW_BOLD;
                SecureHelper::strcpy_x(titleLogFont.lfFaceName, _countof(titleLogFont.lfFaceName), _T("Verdana Bold"));
                INT titleFontPointSize = 12;
                titleLogFont.lfHeight = -::MulDiv(titleFontPointSize, dcScreen.GetDeviceCaps(LOGPIXELSY), 72);
                m_fontExteriorPageTitle.CreateFontIndirect(&titleLogFont);
                
                
                bulletLogFont.lfCharSet = DEFAULT_CHARSET;
                bulletLogFont.lfWeight = FW_NORMAL;
                SecureHelper::strcpy_x(bulletLogFont.lfFaceName, _countof(bulletLogFont.lfFaceName), _T("Marlett"));
                INT bulletFontSize = 8;
                bulletLogFont.lfHeight = -::MulDiv(bulletFontSize, dcScreen.GetDeviceCaps(LOGPIXELSY), 72);
                m_fontBullet.CreateFontIndirect(&bulletLogFont);
        }
        BEGIN_MSG_MAP(thisClass)
                MESSAGE_HANDLER(CWizard97SheetWindow::GetMessage_GetExteriorPageTitleFont(), OnGetExteriorPageTitleFont)
                MESSAGE_HANDLER(CWizard97SheetWindow::GetMessage_GetBulletFont(), OnGetBulletFont)
                MESSAGE_HANDLER(WM_SIZE, OnSize)
                CHAIN_MSG_MAP(baseClass)
        END_MSG_MAP()
        LRESULT OnGetExteriorPageTitleFont(UINT , WPARAM , LPARAM , BOOL& )
        {
                return (LRESULT)(HFONT)m_fontExteriorPageTitle;
        }
        LRESULT OnGetBulletFont(UINT , WPARAM , LPARAM , BOOL& )
        {
                return (LRESULT)(HFONT)m_fontBullet;
        }
        LRESULT OnSize(UINT , WPARAM , LPARAM , BOOL& bHandled)
        {
                if(!m_bReceivedFirstSizeMessage)
                {
                        m_bReceivedFirstSizeMessage = true;
                        this->CenterWindow();
                }
                bHandled = FALSE;
                return 0;
        }
};
class CWizard97Sheet : public CWizard97SheetImpl<CWizard97Sheet>
{
protected:
        typedef CWizard97Sheet thisClass;
        typedef CWizard97SheetImpl<CWizard97Sheet> baseClass;
public:
        CWizard97Sheet(ATL::_U_STRINGorID title, ATL::_U_STRINGorID headerBitmap, ATL::_U_STRINGorID watermarkBitmap, UINT uStartPage = 0, HWND hWndParent = NULL) :
                baseClass(title, headerBitmap, watermarkBitmap, uStartPage, hWndParent)
        { }
        BEGIN_MSG_MAP(thisClass)
                CHAIN_MSG_MAP(baseClass)
        END_MSG_MAP()
};
#define WIZARD97_EXTERIOR_CXDLG 317
#define WIZARD97_EXTERIOR_CYDLG 193
#define WIZARD97_INTERIOR_CXDLG 317
#define WIZARD97_INTERIOR_CYDLG 143
class CWizard97PageWindow : public CPropertyPageWindow
{
public:
        CWizard97PageWindow(HWND hWnd = NULL) : CPropertyPageWindow(hWnd)
        { }
        CWizard97PageWindow& operator =(HWND hWnd)
        {
                m_hWnd = hWnd;
                return *this;
        }
        CWizard97SheetWindow GetPropertySheet() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return CWizard97SheetWindow(GetParent());
        }
        HFONT GetExteriorPageTitleFont(void)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return GetPropertySheet().GetExteriorPageTitleFont();
        }
        HFONT GetBulletFont(void)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                return GetPropertySheet().GetBulletFont();
        }
        HWND Create(LPCTSTR, HWND, ATL::_U_RECT = NULL, LPCTSTR = NULL, DWORD = 0, DWORD = 0, ATL::_U_MENUorID = 0U, LPVOID = NULL)
        {
                ATLASSERT(FALSE);
                return NULL;
        }
};
template <class T, class TBase = CWizard97PageWindow>
class ATL_NO_VTABLE CWizard97PageImpl : public CPropertyPageImpl< T, TBase >
{
protected:
        typedef CWizard97PageImpl< T, TBase > thisClass;
        typedef CPropertyPageImpl< T, TBase > baseClass;
public:
        CWizard97PageImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : baseClass(title)
        { }
        BEGIN_MSG_MAP(thisClass)
                CHAIN_MSG_MAP(baseClass)
        END_MSG_MAP()
};
template <class T, class TBase = CWizard97PageWindow>
class ATL_NO_VTABLE CWizard97ExteriorPageImpl : public CPropertyPageImpl< T, TBase >
{
protected:
        typedef CWizard97ExteriorPageImpl< T, TBase > thisClass;
        typedef CPropertyPageImpl< T, TBase > baseClass;
public:
        CWizard97ExteriorPageImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : baseClass(title)
        {
                m_psp.dwFlags |= PSP_HASHELP;
                m_psp.dwFlags |= PSP_HIDEHEADER;
        }
        BEGIN_MSG_MAP(thisClass)
                CHAIN_MSG_MAP(baseClass)
        END_MSG_MAP()
};
template <class T, class TBase = CWizard97PageWindow>
class ATL_NO_VTABLE CWizard97InteriorPageImpl : public CPropertyPageImpl< T, TBase >
{
protected:
        typedef CWizard97InteriorPageImpl< T, TBase > thisClass;
        typedef CPropertyPageImpl< T, TBase > baseClass;
public:
        CWizard97InteriorPageImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : baseClass(title)
        {
                m_psp.dwFlags |= PSP_HASHELP;
                m_psp.dwFlags &= ~PSP_HIDEHEADER;
                m_psp.dwFlags |= PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
                
                
                baseClass::SetHeaderTitle(_T("Call SetHeaderTitle in Derived Class"));
                baseClass::SetHeaderSubTitle(_T("Call SetHeaderSubTitle in the constructor of the Derived Class."));
        }
        BEGIN_MSG_MAP(thisClass)
                CHAIN_MSG_MAP(baseClass)
        END_MSG_MAP()
};
#endif 
#if (_WIN32_WINNT >= 0x0600) && !defined(_WIN32_WCE)
class CAeroWizardFrameWindow : public CPropertySheetWindow
{
public:
        CAeroWizardFrameWindow(HWND hWnd = NULL) : CPropertySheetWindow(hWnd)
        { }
        CAeroWizardFrameWindow& operator =(HWND hWnd)
        {
                m_hWnd = hWnd;
                return *this;
        }
        void SetNextText(LPCWSTR lpszText)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_SETNEXTTEXT, 0, (LPARAM)lpszText);
        }
        void ShowWizardButtons(DWORD dwButtons, DWORD dwStates)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::PostMessage(m_hWnd, PSM_SHOWWIZBUTTONS, (WPARAM)dwStates, (LPARAM)dwButtons);
        }
        void EnableWizardButtons(DWORD dwButtons, DWORD dwStates)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::PostMessage(m_hWnd, PSM_ENABLEWIZBUTTONS, (WPARAM)dwStates, (LPARAM)dwButtons);
        }
        void SetButtonText(DWORD dwButton, LPCWSTR lpszText)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ::SendMessage(m_hWnd, PSM_SETBUTTONTEXT, (WPARAM)dwButton, (LPARAM)lpszText);
        }
};
template <class T, class TBase = CAeroWizardFrameWindow>
class ATL_NO_VTABLE CAeroWizardFrameImpl : public CPropertySheetImpl<T, TBase >
{
public:
        CAeroWizardFrameImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL, UINT uStartPage = 0, HWND hWndParent = NULL) :
                CPropertySheetImpl<T, TBase >(title, uStartPage, hWndParent)
        {
                m_psh.dwFlags |= PSH_WIZARD | PSH_AEROWIZARD;
        }
        void EnableResizing()
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psh.dwFlags |= PSH_RESIZABLE;
        }
        void UseHeaderBitmap()
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psh.dwFlags |= PSH_HEADERBITMAP;
        }
        void SetNoMargin()
        {
                ATLASSERT(m_hWnd == NULL);   
                m_psh.dwFlags |= PSH_NOMARGIN;
        }
        HWND Create(HWND  = NULL)
        {
                ATLASSERT(FALSE);   
                return NULL;
        }
};
class CAeroWizardFrame : public CAeroWizardFrameImpl<CAeroWizardFrame>
{
public:
        CAeroWizardFrame(ATL::_U_STRINGorID title = (LPCTSTR)NULL, UINT uStartPage = 0, HWND hWndParent = NULL)
                : CAeroWizardFrameImpl<CAeroWizardFrame>(title, uStartPage, hWndParent)
        { }
        BEGIN_MSG_MAP(CAeroWizardFrame)
                MESSAGE_HANDLER(WM_COMMAND, CAeroWizardFrameImpl<CAeroWizardFrame>::OnCommand)
        END_MSG_MAP()
};
class CAeroWizardPageWindow : public CPropertyPageWindow
{
public:
        CAeroWizardPageWindow(HWND hWnd = NULL) : CPropertyPageWindow(hWnd)
        { }
        CAeroWizardPageWindow& operator =(HWND hWnd)
        {
                m_hWnd = hWnd;
                return *this;
        }
        CAeroWizardFrameWindow GetAeroWizardFrame() const
        {
                ATLASSERT(::IsWindow(m_hWnd));
                
                return CAeroWizardFrameWindow(GetParent());
        }
        void SetNextText(LPCWSTR lpszText)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetAeroWizardFrame().SetNextText(lpszText);
        }
        void ShowWizardButtons(DWORD dwButtons, DWORD dwStates)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetAeroWizardFrame().ShowWizardButtons(dwButtons, dwStates);
        }
        void EnableWizardButtons(DWORD dwButtons, DWORD dwStates)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetAeroWizardFrame().EnableWizardButtons(dwButtons, dwStates);
        }
        void SetButtonText(DWORD dwButton, LPCWSTR lpszText)
        {
                ATLASSERT(::IsWindow(m_hWnd));
                ATLASSERT(GetParent() != NULL);
                GetAeroWizardFrame().SetButtonText(dwButton, lpszText);
        }
};
template <class T, class TBase = CAeroWizardPageWindow>
class ATL_NO_VTABLE CAeroWizardPageImpl : public CPropertyPageImpl<T, TBase >
{
public:
        CAeroWizardPageImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : CPropertyPageImpl<T, TBase >(title)
        { }
};
template <WORD t_wDlgTemplateID>
class CAeroWizardPage : public CAeroWizardPageImpl<CAeroWizardPage<t_wDlgTemplateID> >
{
public:
        enum { IDD = t_wDlgTemplateID };
        CAeroWizardPage(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : CAeroWizardPageImpl<CAeroWizardPage>(title)
        { }
        DECLARE_EMPTY_MSG_MAP()
};
#ifndef _ATL_NO_HOSTING
template <class T, class TBase = CAeroWizardPageWindow>
class ATL_NO_VTABLE CAeroWizardAxPageImpl : public CAxPropertyPageImpl< T, TBase >
{
public:
        CAeroWizardAxPageImpl(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : CAxPropertyPageImpl< T, TBase >(title)
        { }
};
template <WORD t_wDlgTemplateID>
class CAeroWizardAxPage : public CAeroWizardAxPageImpl<CAeroWizardAxPage<t_wDlgTemplateID> >
{
public:
        enum { IDD = t_wDlgTemplateID };
        CAeroWizardAxPage(ATL::_U_STRINGorID title = (LPCTSTR)NULL) : CAeroWizardAxPageImpl<CAeroWizardAxPage>(title)
        { }
#if (_WIN32_IE >= 0x0500) || (_ATL_VER >= 0x0700)
        
        BEGIN_MSG_MAP(CAeroWizardAxPage)
                CHAIN_MSG_MAP(CAeroWizardAxPageImpl<CAeroWizardAxPage<t_wDlgTemplateID> >)
        END_MSG_MAP()
#else 
        DECLARE_EMPTY_MSG_MAP()
#endif 
};
#endif 
#endif 
#if ((_WIN32_WINNT >= 0x0600) || defined(_WTL_TASKDIALOG)) && !defined(_WIN32_WCE)
inline int AtlTaskDialog(HWND hWndParent, 
                         ATL::_U_STRINGorID WindowTitle, ATL::_U_STRINGorID MainInstructionText, ATL::_U_STRINGorID ContentText, 
                         TASKDIALOG_COMMON_BUTTON_FLAGS dwCommonButtons = 0U, ATL::_U_STRINGorID Icon = (LPCTSTR)NULL)
{
        int nRet = -1;
#ifdef _WTL_TASKDIALOG_DIRECT
        USES_CONVERSION;
        HRESULT hRet = ::TaskDialog(hWndParent, ModuleHelper::GetResourceInstance(), T2CW(WindowTitle.m_lpstr), T2CW(MainInstructionText.m_lpstr), T2CW(ContentText.m_lpstr), dwCommonButtons, T2CW(Icon.m_lpstr), &nRet);
        ATLVERIFY(SUCCEEDED(hRet));
#else
        
        typedef HRESULT (STDAPICALLTYPE *PFN_TaskDialog)(HWND hwndParent, HINSTANCE hInstance, PCWSTR pszWindowTitle, PCWSTR pszMainInstruction, PCWSTR pszContent, TASKDIALOG_COMMON_BUTTON_FLAGS dwCommonButtons, PCWSTR pszIcon, int* pnButton);
        HMODULE m_hCommCtrlDLL = ::LoadLibrary(_T("comctl32.dll"));
        if(m_hCommCtrlDLL != NULL)
        {
                PFN_TaskDialog pfnTaskDialog = (PFN_TaskDialog)::GetProcAddress(m_hCommCtrlDLL, "TaskDialog");
                if(pfnTaskDialog != NULL)
                {
                        USES_CONVERSION;
                        HRESULT hRet = pfnTaskDialog(hWndParent, ModuleHelper::GetResourceInstance(), T2CW(WindowTitle.m_lpstr), T2CW(MainInstructionText.m_lpstr), T2CW(ContentText.m_lpstr), dwCommonButtons, T2CW(Icon.m_lpstr), &nRet);
                        ATLVERIFY(SUCCEEDED(hRet));
                }
                ::FreeLibrary(m_hCommCtrlDLL);
        }
#endif
        return nRet;
}
class CTaskDialogConfig : public TASKDIALOGCONFIG
{
public:
        CTaskDialogConfig()
        {
                Init();
        }
        void Init()
        {
                memset(this, 0, sizeof(TASKDIALOGCONFIG));   
                this->cbSize = sizeof(TASKDIALOGCONFIG);
                this->hInstance = ModuleHelper::GetResourceInstance();
        }
        
        void SetCommonButtons(TASKDIALOG_COMMON_BUTTON_FLAGS dwCommonButtons)
        {
                this->dwCommonButtons = dwCommonButtons;
        }
        
        void SetWindowTitle(UINT nID)
        {
                this->pszWindowTitle = MAKEINTRESOURCEW(nID);
        }
        void SetWindowTitle(LPCWSTR lpstrWindowTitle)
        {
                this->pszWindowTitle = lpstrWindowTitle;
        }
        
        void SetMainIcon(HICON hIcon)
        {
                this->dwFlags |= TDF_USE_HICON_MAIN;
                this->hMainIcon = hIcon;
        }
        void SetMainIcon(UINT nID)
        {
                this->dwFlags &= ~TDF_USE_HICON_MAIN;
                this->pszMainIcon = MAKEINTRESOURCEW(nID);
        }
        void SetMainIcon(LPCWSTR lpstrMainIcon)
        {
                this->dwFlags &= ~TDF_USE_HICON_MAIN;
                this->pszMainIcon = lpstrMainIcon;
        }
        
        void SetMainInstructionText(UINT nID)
        {
                this->pszMainInstruction = MAKEINTRESOURCEW(nID);
        }
        void SetMainInstructionText(LPCWSTR lpstrMainInstruction)
        {
                this->pszMainInstruction = lpstrMainInstruction;
        }
        
        void SetContentText(UINT nID)
        {
                this->pszContent = MAKEINTRESOURCEW(nID);
        }
        void SetContentText(LPCWSTR lpstrContent)
        {
                this->pszContent = lpstrContent;
        }
        
        void SetButtons(const TASKDIALOG_BUTTON* pButtons, UINT cButtons, int nDefaultButton = 0)
        {
                this->pButtons = pButtons;
                this->cButtons = cButtons;
                if(nDefaultButton != 0)
                        this->nDefaultButton = nDefaultButton;
        }
        void SetDefaultButton(int nDefaultButton)
        {
                this->nDefaultButton = nDefaultButton;
        }
        
        void SetRadioButtons(const TASKDIALOG_BUTTON* pRadioButtons, UINT cRadioButtons, int nDefaultRadioButton = 0)
        {
                this->pRadioButtons = pRadioButtons;
                this->cRadioButtons = cRadioButtons;
                if(nDefaultRadioButton != 0)
                        this->nDefaultRadioButton = nDefaultRadioButton;
        }
        void SetDefaultRadioButton(int nDefaultRadioButton)
        {
                this->nDefaultRadioButton = nDefaultRadioButton;
        }
        
        void SetVerificationText(UINT nID)
        {
                this->pszVerificationText = MAKEINTRESOURCEW(nID);
        }
        void SetVerificationText(LPCWSTR lpstrVerificationText)
        {
                this->pszVerificationText = lpstrVerificationText;
        }
        
        void SetExpandedInformationText(UINT nID)
        {
                this->pszExpandedInformation = MAKEINTRESOURCEW(nID);
        }
        void SetExpandedInformationText(LPCWSTR lpstrExpandedInformation)
        {
                this->pszExpandedInformation = lpstrExpandedInformation;
        }
        
        void SetExpandedControlText(UINT nID)
        {
                this->pszExpandedControlText = MAKEINTRESOURCEW(nID);
        }
        void SetExpandedControlText(LPCWSTR lpstrExpandedControlText)
        {
                this->pszExpandedControlText = lpstrExpandedControlText;
        }
        
        void SetCollapsedControlText(UINT nID)
        {
                this->pszCollapsedControlText = MAKEINTRESOURCEW(nID);
        }
        void SetCollapsedControlText(LPCWSTR lpstrCollapsedControlText)
        {
                this->pszCollapsedControlText = lpstrCollapsedControlText;
        }
        
        void SetFooterIcon(HICON hIcon)
        {
                this->dwFlags |= TDF_USE_HICON_FOOTER;
                this->hFooterIcon = hIcon;
        }
        void SetFooterIcon(UINT nID)
        {
                this->dwFlags &= ~TDF_USE_HICON_FOOTER;
                this->pszFooterIcon = MAKEINTRESOURCEW(nID);
        }
        void SetFooterIcon(LPCWSTR lpstrFooterIcon)
        {
                this->dwFlags &= ~TDF_USE_HICON_FOOTER;
                this->pszFooterIcon = lpstrFooterIcon;
        }
        
        void SetFooterText(UINT nID)
        {
                this->pszFooter = MAKEINTRESOURCEW(nID);
        }
        void SetFooterText(LPCWSTR lpstrFooterText)
        {
                this->pszFooter = lpstrFooterText;
        }
        
        void SetWidth(UINT cxWidth)
        {
                this->cxWidth = cxWidth;
        }
        
        void ModifyFlags(DWORD dwRemove, DWORD dwAdd)
        {
                this->dwFlags = (this->dwFlags & ~dwRemove) | dwAdd;
        }
};
template <class T>
class ATL_NO_VTABLE CTaskDialogImpl
{
public:
        CTaskDialogConfig m_tdc;
        HWND m_hWnd;   
        CTaskDialogImpl(HWND hWndParent = NULL) : m_hWnd(NULL)
        {
                m_tdc.hwndParent = hWndParent;
                m_tdc.pfCallback = T::TaskDialogCallback;
                m_tdc.lpCallbackData = (LONG_PTR)static_cast<T*>(this);
        }
        HRESULT DoModal(HWND hWndParent = ::GetActiveWindow(), int* pnButton = NULL, int* pnRadioButton = NULL, BOOL* pfVerificationFlagChecked = NULL)
        {
                if(m_tdc.hwndParent == NULL)
                        m_tdc.hwndParent = hWndParent;
#ifdef _WTL_TASKDIALOG_DIRECT
                return ::TaskDialogIndirect(&m_tdc, pnButton, pnRadioButton, pfVerificationFlagChecked);
#else
                
                typedef HRESULT (STDAPICALLTYPE *PFN_TaskDialogIndirect)(const TASKDIALOGCONFIG* pTaskConfig, int* pnButton, int* pnRadioButton, BOOL* pfVerificationFlagChecked);
                HRESULT hRet = E_UNEXPECTED;
                HMODULE m_hCommCtrlDLL = ::LoadLibrary(_T("comctl32.dll"));
                if(m_hCommCtrlDLL != NULL)
                {
                        PFN_TaskDialogIndirect pfnTaskDialogIndirect = (PFN_TaskDialogIndirect)::GetProcAddress(m_hCommCtrlDLL, "TaskDialogIndirect");
                        if(pfnTaskDialogIndirect != NULL)
                                hRet = pfnTaskDialogIndirect(&m_tdc, pnButton, pnRadioButton, pfVerificationFlagChecked);
                        ::FreeLibrary(m_hCommCtrlDLL);
                }
                return hRet;
#endif
        }
        
        void SetCommonButtons(TASKDIALOG_COMMON_BUTTON_FLAGS dwCommonButtons)
        {       m_tdc.SetCommonButtons(dwCommonButtons); }
        
        void SetWindowTitle(UINT nID)
        {       m_tdc.SetWindowTitle(nID); }
        void SetWindowTitle(LPCWSTR lpstrWindowTitle)
        {       m_tdc.SetWindowTitle(lpstrWindowTitle); }
        
        void SetMainIcon(HICON hIcon)
        {       m_tdc.SetMainIcon(hIcon); }
        void SetMainIcon(UINT nID)
        {       m_tdc.SetMainIcon(nID); }
        void SetMainIcon(LPCWSTR lpstrMainIcon)
        {       m_tdc.SetMainIcon(lpstrMainIcon); }
        
        void SetMainInstructionText(UINT nID)
        {       m_tdc.SetMainInstructionText(nID); }
        void SetMainInstructionText(LPCWSTR lpstrMainInstruction)
        {       m_tdc.SetMainInstructionText(lpstrMainInstruction); }
        
        void SetContentText(UINT nID)
        {       m_tdc.SetContentText(nID); }
        void SetContentText(LPCWSTR lpstrContent)
        {       m_tdc.SetContentText(lpstrContent); }
        
        void SetButtons(const TASKDIALOG_BUTTON* pButtons, UINT cButtons, int nDefaultButton = 0)
        {       m_tdc.SetButtons(pButtons, cButtons, nDefaultButton); }
        void SetDefaultButton(int nDefaultButton)
        {       m_tdc.SetDefaultButton(nDefaultButton); }
        
        void SetRadioButtons(const TASKDIALOG_BUTTON* pRadioButtons, UINT cRadioButtons, int nDefaultRadioButton = 0)
        {       m_tdc.SetRadioButtons(pRadioButtons, cRadioButtons, nDefaultRadioButton); }
        void SetDefaultRadioButton(int nDefaultRadioButton)
        {       m_tdc.SetDefaultRadioButton(nDefaultRadioButton); }
        
        void SetVerificationText(UINT nID)
        {       m_tdc.SetVerificationText(nID); }
        void SetVerificationText(LPCWSTR lpstrVerificationText)
        {       m_tdc.SetVerificationText(lpstrVerificationText); }
        
        void SetExpandedInformationText(UINT nID)
        {       m_tdc.SetExpandedInformationText(nID); }
        void SetExpandedInformationText(LPCWSTR lpstrExpandedInformation)
        {       m_tdc.SetExpandedInformationText(lpstrExpandedInformation); }
        
        void SetExpandedControlText(UINT nID)
        {       m_tdc.SetExpandedControlText(nID); }
        void SetExpandedControlText(LPCWSTR lpstrExpandedControlText)
        {       m_tdc.SetExpandedControlText(lpstrExpandedControlText); }
        
        void SetCollapsedControlText(UINT nID)
        {       m_tdc.SetCollapsedControlText(nID); }
        void SetCollapsedControlText(LPCWSTR lpstrCollapsedControlText)
        {       m_tdc.SetCollapsedControlText(lpstrCollapsedControlText); }
        
        void SetFooterIcon(HICON hIcon)
        {       m_tdc.SetFooterIcon(hIcon); }
        void SetFooterIcon(UINT nID)
        {       m_tdc.SetFooterIcon(nID); }
        void SetFooterIcon(LPCWSTR lpstrFooterIcon)
        {       m_tdc.SetFooterIcon(lpstrFooterIcon); }
        
        void SetFooterText(UINT nID)
        {       m_tdc.SetFooterText(nID); }
        void SetFooterText(LPCWSTR lpstrFooterText)
        {       m_tdc.SetFooterText(lpstrFooterText); }
        
        void SetWidth(UINT cxWidth)
        {       m_tdc.SetWidth(cxWidth); }
        
        void ModifyFlags(DWORD dwRemove, DWORD dwAdd)
        {       m_tdc.ModifyFlags(dwRemove, dwAdd); }
        static HRESULT CALLBACK TaskDialogCallback(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam, LONG_PTR lpRefData)
        {
                T* pT = (T*)lpRefData;
                ATLASSERT(pT->m_hWnd == NULL || pT->m_hWnd == hWnd);
                BOOL bRet = FALSE;
                switch(uMsg)
                {
                case TDN_DIALOG_CONSTRUCTED:
                        pT->m_hWnd = hWnd;
                        pT->OnDialogConstructed();
                        break;
                case TDN_CREATED:
                        pT->OnCreated();
                        break;
                case TDN_BUTTON_CLICKED:
                        bRet = pT->OnButtonClicked((int)wParam);
                        break;
                case TDN_RADIO_BUTTON_CLICKED:
                        pT->OnRadioButtonClicked((int)wParam);
                        break;
                case TDN_HYPERLINK_CLICKED:
                        pT->OnHyperlinkClicked((LPCWSTR)lParam);
                        break;
                case TDN_EXPANDO_BUTTON_CLICKED:
                        pT->OnExpandoButtonClicked((wParam != 0));
                        break;
                case TDN_VERIFICATION_CLICKED:
                        pT->OnVerificationClicked((wParam != 0));
                        break;
                case TDN_HELP:
                        pT->OnHelp();
                        break;
                case TDN_TIMER:
                        bRet = pT->OnTimer((DWORD)wParam);
                        break;
                case TDN_NAVIGATED:
                        pT->OnNavigated();
                        break;
                case TDN_DESTROYED:
                        pT->OnDestroyed();
                        pT->m_hWnd = NULL;
                        break;
                default:
                        ATLTRACE2(atlTraceUI, 0, _T("Unknown notification received in CTaskDialogImpl::TaskDialogCallback\n"));
                        break;
                }
                return (HRESULT)bRet;
        }
        void OnDialogConstructed()
        {
        }
        void OnCreated()
        {
        }
        BOOL OnButtonClicked(int )
        {
                return FALSE;   
        }
        void OnRadioButtonClicked(int )
        {
        }
        void OnHyperlinkClicked(LPCWSTR )
        {
        }
        void OnExpandoButtonClicked(bool )
        {
        }
        void OnVerificationClicked(bool )
        {
        }
        void OnHelp()
        {
        }
        BOOL OnTimer(DWORD )
        {
                return FALSE;   
        }
        void OnNavigated()
        {
        }
        void OnDestroyed()
        {
        }
        void NavigatePage(TASKDIALOGCONFIG& tdc)
        {
                ATLASSERT(m_hWnd != NULL);
                tdc.cbSize = sizeof(TASKDIALOGCONFIG);
                if(tdc.hwndParent == NULL)
                        tdc.hwndParent = m_tdc.hwndParent;
                tdc.pfCallback = m_tdc.pfCallback;
                tdc.lpCallbackData = m_tdc.lpCallbackData;
                (TASKDIALOGCONFIG)m_tdc = tdc;
                ::SendMessage(m_hWnd, TDM_NAVIGATE_PAGE, 0, (LPARAM)&tdc);
        }
        
        void NavigatePage()
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_NAVIGATE_PAGE, 0, (LPARAM)&m_tdc);
        }
        void ClickButton(int nButton)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_CLICK_BUTTON, nButton, 0L);
        }
        void SetMarqueeProgressBar(BOOL bMarquee)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_SET_MARQUEE_PROGRESS_BAR, bMarquee, 0L);
        }
        BOOL SetProgressBarState(int nNewState)
        {
                ATLASSERT(m_hWnd != NULL);
                return (BOOL)::SendMessage(m_hWnd, TDM_SET_PROGRESS_BAR_STATE, nNewState, 0L);
        }
        DWORD SetProgressBarRange(int nMinRange, int nMaxRange)
        {
                ATLASSERT(m_hWnd != NULL);
                return (DWORD)::SendMessage(m_hWnd, TDM_SET_PROGRESS_BAR_RANGE, 0, MAKELPARAM(nMinRange, nMaxRange));
        }
        int SetProgressBarPos(int nNewPos)
        {
                ATLASSERT(m_hWnd != NULL);
                return (int)::SendMessage(m_hWnd, TDM_SET_PROGRESS_BAR_POS, nNewPos, 0L);
        }
        BOOL SetProgressBarMarquee(BOOL bMarquee, UINT uSpeed)
        {
                ATLASSERT(m_hWnd != NULL);
                return (BOOL)::SendMessage(m_hWnd, TDM_SET_PROGRESS_BAR_MARQUEE, bMarquee, uSpeed);
        }
        void SetElementText(TASKDIALOG_ELEMENTS element, LPCWSTR lpstrText)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_SET_ELEMENT_TEXT, element, (LPARAM)lpstrText);
        }
        void ClickRadioButton(int nRadioButton)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_CLICK_RADIO_BUTTON, nRadioButton, 0L);
        }
        void EnableButton(int nButton, BOOL bEnable)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_ENABLE_BUTTON, nButton, bEnable);
        }
        void EnableRadioButton(int nButton, BOOL bEnable)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_ENABLE_RADIO_BUTTON, nButton, bEnable);
        }
        void ClickVerification(BOOL bCheck, BOOL bFocus)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_CLICK_VERIFICATION, bCheck, bFocus);
        }
        void UpdateElementText(TASKDIALOG_ELEMENTS element, LPCWSTR lpstrText)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_UPDATE_ELEMENT_TEXT, element, (LPARAM)lpstrText);
        }
        void SetButtonElevationRequiredState(int nButton, BOOL bElevation)
        {
                ATLASSERT(m_hWnd != NULL);
                ::SendMessage(m_hWnd, TDM_SET_BUTTON_ELEVATION_REQUIRED_STATE, nButton, bElevation);
        }
        void UpdateIcon(TASKDIALOG_ICON_ELEMENTS element, HICON hIcon)
        {
                ATLASSERT(m_hWnd != NULL);
#ifdef _DEBUG
                if(element == TDIE_ICON_MAIN)
                        ATLASSERT((m_tdc.dwFlags & TDF_USE_HICON_MAIN) != 0);
                else if(element == TDIE_ICON_FOOTER)
                        ATLASSERT((m_tdc.dwFlags & TDF_USE_HICON_FOOTER) != 0);
#endif 
                ::SendMessage(m_hWnd, TDM_UPDATE_ICON, element, (LPARAM)hIcon);
        }
        void UpdateIcon(TASKDIALOG_ICON_ELEMENTS element, LPCWSTR lpstrIcon)
        {
                ATLASSERT(m_hWnd != NULL);
#ifdef _DEBUG
                if(element == TDIE_ICON_MAIN)
                        ATLASSERT((m_tdc.dwFlags & TDF_USE_HICON_MAIN) == 0);
                else if(element == TDIE_ICON_FOOTER)
                        ATLASSERT((m_tdc.dwFlags & TDF_USE_HICON_FOOTER) == 0);
#endif 
                ::SendMessage(m_hWnd, TDM_UPDATE_ICON, element, (LPARAM)lpstrIcon);
        }
};
class CTaskDialog : public CTaskDialogImpl<CTaskDialog>
{
public:
        CTaskDialog(HWND hWndParent = NULL) : CTaskDialogImpl<CTaskDialog>(hWndParent)
        {
                m_tdc.pfCallback = NULL;
        }
};
#endif 
}; 
#endif