896 lines
22 KiB
C
896 lines
22 KiB
C
|
// WTL Version 3.1
|
||
|
// Copyright (C) 1997-2000 Microsoft Corporation
|
||
|
// All rights reserved.
|
||
|
//
|
||
|
// This file is a part of Windows Template Library.
|
||
|
// The code and information is provided "as-is" without
|
||
|
// warranty of any kind, either expressed or implied.
|
||
|
|
||
|
#ifndef __ATLAPP_H__
|
||
|
#define __ATLAPP_H__
|
||
|
|
||
|
#pragma once
|
||
|
|
||
|
#ifndef __cplusplus
|
||
|
#error ATL requires C++ compilation (use a .cpp suffix)
|
||
|
#endif
|
||
|
|
||
|
#ifndef __ATLBASE_H__
|
||
|
#error atlapp.h requires atlbase.h to be included first
|
||
|
#endif
|
||
|
|
||
|
#if (WINVER < 0x0400)
|
||
|
#error WTL requires Windows version 4.0 or higher
|
||
|
#endif
|
||
|
|
||
|
|
||
|
#include <limits.h>
|
||
|
#if !defined(_ATL_MIN_CRT) & defined(_MT)
|
||
|
#include <process.h> // for _beginthreadex, _endthreadex
|
||
|
#endif
|
||
|
|
||
|
#include <commctrl.h>
|
||
|
#pragma comment(lib, "comctl32.lib")
|
||
|
|
||
|
#include <atlres.h>
|
||
|
|
||
|
|
||
|
// WTL version number
|
||
|
#define _WTL_VER 0x0310
|
||
|
|
||
|
|
||
|
// This is to support using original VC++ 6.0 headers with WTL
|
||
|
#ifndef _ATL_NO_OLD_HEADERS_WIN64
|
||
|
#if !defined(_WIN64) && !defined(_ATL_TMP_IMPL2)
|
||
|
|
||
|
#ifndef GetWindowLongPtr
|
||
|
#define GetWindowLongPtrA GetWindowLongA
|
||
|
#define GetWindowLongPtrW GetWindowLongW
|
||
|
#ifdef UNICODE
|
||
|
#define GetWindowLongPtr GetWindowLongPtrW
|
||
|
#else
|
||
|
#define GetWindowLongPtr GetWindowLongPtrA
|
||
|
#endif // !UNICODE
|
||
|
#endif // !GetWindowLongPtr
|
||
|
|
||
|
#ifndef SetWindowLongPtr
|
||
|
#define SetWindowLongPtrA SetWindowLongA
|
||
|
#define SetWindowLongPtrW SetWindowLongW
|
||
|
#ifdef UNICODE
|
||
|
#define SetWindowLongPtr SetWindowLongPtrW
|
||
|
#else
|
||
|
#define SetWindowLongPtr SetWindowLongPtrA
|
||
|
#endif // !UNICODE
|
||
|
#endif // !SetWindowLongPtr
|
||
|
|
||
|
#ifndef GWLP_WNDPROC
|
||
|
#define GWLP_WNDPROC (-4)
|
||
|
#endif
|
||
|
#ifndef GWLP_HINSTANCE
|
||
|
#define GWLP_HINSTANCE (-6)
|
||
|
#endif
|
||
|
#ifndef GWLP_HWNDPARENT
|
||
|
#define GWLP_HWNDPARENT (-8)
|
||
|
#endif
|
||
|
#ifndef GWLP_USERDATA
|
||
|
#define GWLP_USERDATA (-21)
|
||
|
#endif
|
||
|
#ifndef GWLP_ID
|
||
|
#define GWLP_ID (-12)
|
||
|
#endif
|
||
|
|
||
|
#ifndef DWLP_MSGRESULT
|
||
|
#define DWLP_MSGRESULT 0
|
||
|
#endif
|
||
|
|
||
|
typedef long LONG_PTR;
|
||
|
typedef unsigned long ULONG_PTR;
|
||
|
typedef ULONG_PTR DWORD_PTR;
|
||
|
|
||
|
#ifndef HandleToUlong
|
||
|
#define HandleToUlong( h ) ((ULONG)(ULONG_PTR)(h) )
|
||
|
#endif
|
||
|
#ifndef HandleToLong
|
||
|
#define HandleToLong( h ) ((LONG)(LONG_PTR) (h) )
|
||
|
#endif
|
||
|
#ifndef LongToHandle
|
||
|
#define LongToHandle( h) ((HANDLE)(LONG_PTR) (h))
|
||
|
#endif
|
||
|
#ifndef PtrToUlong
|
||
|
#define PtrToUlong( p ) ((ULONG)(ULONG_PTR) (p) )
|
||
|
#endif
|
||
|
#ifndef PtrToLong
|
||
|
#define PtrToLong( p ) ((LONG)(LONG_PTR) (p) )
|
||
|
#endif
|
||
|
#ifndef PtrToUint
|
||
|
#define PtrToUint( p ) ((UINT)(UINT_PTR) (p) )
|
||
|
#endif
|
||
|
#ifndef PtrToInt
|
||
|
#define PtrToInt( p ) ((INT)(INT_PTR) (p) )
|
||
|
#endif
|
||
|
#ifndef PtrToUshort
|
||
|
#define PtrToUshort( p ) ((unsigned short)(ULONG_PTR)(p) )
|
||
|
#endif
|
||
|
#ifndef PtrToShort
|
||
|
#define PtrToShort( p ) ((short)(LONG_PTR)(p) )
|
||
|
#endif
|
||
|
#ifndef IntToPtr
|
||
|
#define IntToPtr( i ) ((VOID *)(INT_PTR)((int)i))
|
||
|
#endif
|
||
|
#ifndef UIntToPtr
|
||
|
#define UIntToPtr( ui ) ((VOID *)(UINT_PTR)((unsigned int)ui))
|
||
|
#endif
|
||
|
#ifndef LongToPtr
|
||
|
#define LongToPtr( l ) ((VOID *)(LONG_PTR)((long)l))
|
||
|
#endif
|
||
|
#ifndef ULongToPtr
|
||
|
#define ULongToPtr( ul ) ((VOID *)(ULONG_PTR)((unsigned long)ul))
|
||
|
#endif
|
||
|
|
||
|
#endif //!defined(_WIN64) && !defined(_ATL_TMP_IMPL2)
|
||
|
#endif //!_ATL_NO_OLD_HEADERS_WIN64
|
||
|
|
||
|
|
||
|
namespace WTL
|
||
|
{
|
||
|
|
||
|
enum wtlTraceFlags
|
||
|
{
|
||
|
atlTraceUI = 0x10000000
|
||
|
};
|
||
|
|
||
|
// Windows version helper
|
||
|
inline bool AtlIsOldWindows()
|
||
|
{
|
||
|
OSVERSIONINFO ovi;
|
||
|
ovi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
||
|
BOOL bRet = ::GetVersionEx(&ovi);
|
||
|
return (!bRet || !((ovi.dwMajorVersion >= 5) || (ovi.dwMajorVersion == 4 && ovi.dwMinorVersion >= 90)));
|
||
|
}
|
||
|
|
||
|
// default GUI font helper
|
||
|
inline HFONT AtlGetDefaultGuiFont()
|
||
|
{
|
||
|
return (HFONT)::GetStockObject(DEFAULT_GUI_FONT);
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// Forward declarations
|
||
|
|
||
|
class CMessageFilter;
|
||
|
class CIdleHandler;
|
||
|
class CMessageLoop;
|
||
|
class CAppModule;
|
||
|
class CServerAppModule;
|
||
|
|
||
|
#ifndef _ATL_TMP_IMPL2
|
||
|
class _U_RECT;
|
||
|
class _U_MENUorID;
|
||
|
class _U_STRINGorID;
|
||
|
#endif //!_ATL_TMP_IMPL2
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CMessageFilter - Interface for message filter support
|
||
|
|
||
|
class CMessageFilter
|
||
|
{
|
||
|
public:
|
||
|
virtual BOOL PreTranslateMessage(MSG* pMsg) = 0;
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CIdleHandler - Interface for idle processing
|
||
|
|
||
|
class CIdleHandler
|
||
|
{
|
||
|
public:
|
||
|
virtual BOOL OnIdle() = 0;
|
||
|
};
|
||
|
|
||
|
#ifndef _ATL_NO_OLD_NAMES
|
||
|
// for compatilibility with old names only
|
||
|
typedef CIdleHandler CUpdateUIObject;
|
||
|
#define DoUpdate OnIdle
|
||
|
#endif //!_ATL_NO_OLD_NAMES
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CMessageLoop - message loop implementation
|
||
|
|
||
|
class CMessageLoop
|
||
|
{
|
||
|
public:
|
||
|
CSimpleArray<CMessageFilter*> m_aMsgFilter;
|
||
|
CSimpleArray<CIdleHandler*> m_aIdleHandler;
|
||
|
MSG m_msg;
|
||
|
|
||
|
// Message filter operations
|
||
|
BOOL AddMessageFilter(CMessageFilter* pMessageFilter)
|
||
|
{
|
||
|
return m_aMsgFilter.Add(pMessageFilter);
|
||
|
}
|
||
|
BOOL RemoveMessageFilter(CMessageFilter* pMessageFilter)
|
||
|
{
|
||
|
return m_aMsgFilter.Remove(pMessageFilter);
|
||
|
}
|
||
|
// Idle handler operations
|
||
|
BOOL AddIdleHandler(CIdleHandler* pIdleHandler)
|
||
|
{
|
||
|
return m_aIdleHandler.Add(pIdleHandler);
|
||
|
}
|
||
|
BOOL RemoveIdleHandler(CIdleHandler* pIdleHandler)
|
||
|
{
|
||
|
return m_aIdleHandler.Remove(pIdleHandler);
|
||
|
}
|
||
|
#ifndef _ATL_NO_OLD_NAMES
|
||
|
// for compatilibility with old names only
|
||
|
BOOL AddUpdateUI(CIdleHandler* pIdleHandler)
|
||
|
{
|
||
|
ATLTRACE2(atlTraceUI, 0, "CUpdateUIObject and AddUpdateUI are deprecated. Please change your code to use CIdleHandler and OnIdle\n");
|
||
|
return AddIdleHandler(pIdleHandler);
|
||
|
}
|
||
|
BOOL RemoveUpdateUI(CIdleHandler* pIdleHandler)
|
||
|
{
|
||
|
ATLTRACE2(atlTraceUI, 0, "CUpdateUIObject and RemoveUpdateUI are deprecated. Please change your code to use CIdleHandler and OnIdle\n");
|
||
|
return RemoveIdleHandler(pIdleHandler);
|
||
|
}
|
||
|
#endif //!_ATL_NO_OLD_NAMES
|
||
|
// message loop
|
||
|
int Run()
|
||
|
{
|
||
|
BOOL bDoIdle = TRUE;
|
||
|
int nIdleCount = 0;
|
||
|
BOOL bRet;
|
||
|
|
||
|
for(;;)
|
||
|
{
|
||
|
while(!::PeekMessage(&m_msg, NULL, 0, 0, PM_NOREMOVE) && bDoIdle)
|
||
|
{
|
||
|
if(!OnIdle(nIdleCount++))
|
||
|
bDoIdle = FALSE;
|
||
|
}
|
||
|
|
||
|
bRet = ::GetMessage(&m_msg, NULL, 0, 0);
|
||
|
|
||
|
if(bRet == -1)
|
||
|
{
|
||
|
ATLTRACE2(atlTraceUI, 0, _T("::GetMessage returned -1 (error)\n"));
|
||
|
continue; // error, don't process
|
||
|
}
|
||
|
else if(!bRet)
|
||
|
{
|
||
|
ATLTRACE2(atlTraceUI, 0, _T("CMessageLoop::Run - exiting\n"));
|
||
|
break; // WM_QUIT, exit message loop
|
||
|
}
|
||
|
|
||
|
if(!PreTranslateMessage(&m_msg))
|
||
|
{
|
||
|
::TranslateMessage(&m_msg);
|
||
|
::DispatchMessage(&m_msg);
|
||
|
}
|
||
|
|
||
|
if(IsIdleMessage(&m_msg))
|
||
|
{
|
||
|
bDoIdle = TRUE;
|
||
|
nIdleCount = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (int)m_msg.wParam;
|
||
|
}
|
||
|
|
||
|
static BOOL IsIdleMessage(MSG* pMsg)
|
||
|
{
|
||
|
// These messages should NOT cause idle processing
|
||
|
switch(pMsg->message)
|
||
|
{
|
||
|
case WM_MOUSEMOVE:
|
||
|
case WM_NCMOUSEMOVE:
|
||
|
case WM_PAINT:
|
||
|
case 0x0118: // WM_SYSTIMER (caret blink)
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
// Overrideables
|
||
|
// Override to change message filtering
|
||
|
virtual BOOL PreTranslateMessage(MSG* pMsg)
|
||
|
{
|
||
|
// loop backwards
|
||
|
for(int i = m_aMsgFilter.GetSize() - 1; i >= 0; i--)
|
||
|
{
|
||
|
CMessageFilter* pMessageFilter = m_aMsgFilter[i];
|
||
|
if(pMessageFilter != NULL && pMessageFilter->PreTranslateMessage(pMsg))
|
||
|
return TRUE;
|
||
|
}
|
||
|
return FALSE; // not translated
|
||
|
}
|
||
|
// override to change idle processing
|
||
|
virtual BOOL OnIdle(int /*nIdleCount*/)
|
||
|
{
|
||
|
for(int i = 0; i < m_aIdleHandler.GetSize(); i++)
|
||
|
{
|
||
|
CIdleHandler* pIdleHandler = m_aIdleHandler[i];
|
||
|
if(pIdleHandler != NULL)
|
||
|
pIdleHandler->OnIdle();
|
||
|
}
|
||
|
return FALSE; // don't continue
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CAppModule - module class for an application
|
||
|
|
||
|
class CAppModule : public CComModule
|
||
|
{
|
||
|
public:
|
||
|
DWORD m_dwMainThreadID;
|
||
|
CSimpleMap<DWORD, CMessageLoop*>* m_pMsgLoopMap;
|
||
|
CSimpleArray<HWND>* m_pSettingChangeNotify;
|
||
|
|
||
|
// Overrides of CComModule::Init and Term
|
||
|
HRESULT Init(_ATL_OBJMAP_ENTRY* pObjMap, HINSTANCE hInstance, const GUID* pLibID = NULL)
|
||
|
{
|
||
|
HRESULT hRet = CComModule::Init(pObjMap, hInstance, pLibID);
|
||
|
if(FAILED(hRet))
|
||
|
return hRet;
|
||
|
|
||
|
m_dwMainThreadID = ::GetCurrentThreadId();
|
||
|
typedef CSimpleMap<DWORD, CMessageLoop*> mapClass;
|
||
|
m_pMsgLoopMap = NULL;
|
||
|
ATLTRY(m_pMsgLoopMap = new mapClass);
|
||
|
if(m_pMsgLoopMap == NULL)
|
||
|
return E_OUTOFMEMORY;
|
||
|
m_pSettingChangeNotify = NULL;
|
||
|
|
||
|
return hRet;
|
||
|
}
|
||
|
void Term()
|
||
|
{
|
||
|
if(m_pSettingChangeNotify != NULL && m_pSettingChangeNotify->GetSize() > 0)
|
||
|
{
|
||
|
::DestroyWindow((*m_pSettingChangeNotify)[0]);
|
||
|
}
|
||
|
delete m_pSettingChangeNotify;
|
||
|
delete m_pMsgLoopMap;
|
||
|
CComModule::Term();
|
||
|
}
|
||
|
|
||
|
// Message loop map methods
|
||
|
BOOL AddMessageLoop(CMessageLoop* pMsgLoop)
|
||
|
{
|
||
|
ATLASSERT(pMsgLoop != NULL);
|
||
|
ATLASSERT(m_pMsgLoopMap->Lookup(::GetCurrentThreadId()) == NULL); // not in map yet
|
||
|
return m_pMsgLoopMap->Add(::GetCurrentThreadId(), pMsgLoop);
|
||
|
}
|
||
|
BOOL RemoveMessageLoop()
|
||
|
{
|
||
|
return m_pMsgLoopMap->Remove(::GetCurrentThreadId());
|
||
|
}
|
||
|
CMessageLoop* GetMessageLoop(DWORD dwThreadID = ::GetCurrentThreadId()) const
|
||
|
{
|
||
|
return m_pMsgLoopMap->Lookup(dwThreadID);
|
||
|
}
|
||
|
|
||
|
// Setting change notify methods
|
||
|
BOOL AddSettingChangeNotify(HWND hWnd)
|
||
|
{
|
||
|
ATLASSERT(::IsWindow(hWnd));
|
||
|
if(m_pSettingChangeNotify == NULL)
|
||
|
{
|
||
|
typedef CSimpleArray<HWND> notifyClass;
|
||
|
ATLTRY(m_pSettingChangeNotify = new notifyClass);
|
||
|
ATLASSERT(m_pSettingChangeNotify != NULL);
|
||
|
if(m_pSettingChangeNotify == NULL)
|
||
|
return FALSE;
|
||
|
}
|
||
|
if(m_pSettingChangeNotify->GetSize() == 0)
|
||
|
{
|
||
|
// init everything
|
||
|
_ATL_EMPTY_DLGTEMPLATE templ;
|
||
|
HWND hNtfWnd = ::CreateDialogIndirect(GetModuleInstance(), &templ, NULL, _SettingChangeDlgProc);
|
||
|
ATLASSERT(::IsWindow(hNtfWnd));
|
||
|
if(::IsWindow(hNtfWnd))
|
||
|
{
|
||
|
::SetWindowLongPtr(hNtfWnd, GWLP_USERDATA, (LONG_PTR)this);
|
||
|
m_pSettingChangeNotify->Add(hNtfWnd);
|
||
|
}
|
||
|
}
|
||
|
return m_pSettingChangeNotify->Add(hWnd);
|
||
|
}
|
||
|
|
||
|
BOOL RemoveSettingChangeNotify(HWND hWnd)
|
||
|
{
|
||
|
if(m_pSettingChangeNotify == NULL)
|
||
|
return FALSE;
|
||
|
return m_pSettingChangeNotify->Remove(hWnd);
|
||
|
}
|
||
|
|
||
|
// Implementation - setting change notify dialog template and dialog procedure
|
||
|
struct _ATL_EMPTY_DLGTEMPLATE : DLGTEMPLATE
|
||
|
{
|
||
|
_ATL_EMPTY_DLGTEMPLATE()
|
||
|
{
|
||
|
memset(this, 0, sizeof(_ATL_EMPTY_DLGTEMPLATE));
|
||
|
style = WS_POPUP;
|
||
|
}
|
||
|
WORD wMenu, wClass, wTitle;
|
||
|
};
|
||
|
|
||
|
#ifdef _WIN64
|
||
|
static INT_PTR CALLBACK _SettingChangeDlgProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
#else
|
||
|
static BOOL CALLBACK _SettingChangeDlgProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
#endif
|
||
|
{
|
||
|
if(uMsg == WM_SETTINGCHANGE)
|
||
|
{
|
||
|
CAppModule* pModule = (CAppModule*)::GetWindowLongPtr(hWnd, GWLP_USERDATA);
|
||
|
ATLASSERT(pModule != NULL);
|
||
|
ATLASSERT(pModule->m_pSettingChangeNotify != NULL);
|
||
|
for(int i = 1; i < pModule->m_pSettingChangeNotify->GetSize(); i++)
|
||
|
::SendMessageTimeout((*pModule->m_pSettingChangeNotify)[i], uMsg, wParam, lParam, SMTO_ABORTIFHUNG, 1500, NULL);
|
||
|
return TRUE;
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CServerAppModule - module class for a COM server application
|
||
|
|
||
|
class CServerAppModule : public CAppModule
|
||
|
{
|
||
|
public:
|
||
|
HANDLE m_hEventShutdown;
|
||
|
bool m_bActivity;
|
||
|
DWORD m_dwTimeOut;
|
||
|
DWORD m_dwPause;
|
||
|
|
||
|
// Override of CAppModule::Init
|
||
|
HRESULT Init(_ATL_OBJMAP_ENTRY* pObjMap, HINSTANCE hInstance, const GUID* pLibID = NULL)
|
||
|
{
|
||
|
m_dwTimeOut = 5000;
|
||
|
m_dwPause = 1000;
|
||
|
return CAppModule::Init(pObjMap, hInstance, pLibID);
|
||
|
}
|
||
|
void Term()
|
||
|
{
|
||
|
if(m_hEventShutdown != NULL && ::CloseHandle(m_hEventShutdown))
|
||
|
m_hEventShutdown = NULL;
|
||
|
CAppModule::Term();
|
||
|
}
|
||
|
|
||
|
// COM Server methods
|
||
|
LONG Unlock()
|
||
|
{
|
||
|
LONG lRet = CComModule::Unlock();
|
||
|
if(lRet == 0)
|
||
|
{
|
||
|
m_bActivity = true;
|
||
|
::SetEvent(m_hEventShutdown); // tell monitor that we transitioned to zero
|
||
|
}
|
||
|
return lRet;
|
||
|
}
|
||
|
|
||
|
void MonitorShutdown()
|
||
|
{
|
||
|
while(1)
|
||
|
{
|
||
|
::WaitForSingleObject(m_hEventShutdown, INFINITE);
|
||
|
DWORD dwWait = 0;
|
||
|
do
|
||
|
{
|
||
|
m_bActivity = false;
|
||
|
dwWait = ::WaitForSingleObject(m_hEventShutdown, m_dwTimeOut);
|
||
|
}
|
||
|
while(dwWait == WAIT_OBJECT_0);
|
||
|
// timed out
|
||
|
if(!m_bActivity && m_nLockCnt == 0) // if no activity let's really bail
|
||
|
{
|
||
|
#if ((_WIN32_WINNT >= 0x0400 ) || defined(_WIN32_DCOM)) & defined(_ATL_FREE_THREADED)
|
||
|
::CoSuspendClassObjects();
|
||
|
if(!m_bActivity && m_nLockCnt == 0)
|
||
|
#endif
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
// This handle should be valid now. If it isn't,
|
||
|
// check if _Module.Term was called first (it shouldn't)
|
||
|
if(::CloseHandle(m_hEventShutdown))
|
||
|
m_hEventShutdown = NULL;
|
||
|
::PostThreadMessage(m_dwMainThreadID, WM_QUIT, 0, 0);
|
||
|
}
|
||
|
|
||
|
bool StartMonitor()
|
||
|
{
|
||
|
m_hEventShutdown = ::CreateEvent(NULL, false, false, NULL);
|
||
|
if(m_hEventShutdown == NULL)
|
||
|
return false;
|
||
|
DWORD dwThreadID;
|
||
|
#if !defined(_ATL_MIN_CRT) & defined(_MT)
|
||
|
HANDLE hThread = (HANDLE)_beginthreadex(NULL, 0, (UINT (WINAPI*)(void*))MonitorProc, this, 0, (UINT*)&dwThreadID);
|
||
|
#else
|
||
|
HANDLE hThread = ::CreateThread(NULL, 0, MonitorProc, this, 0, &dwThreadID);
|
||
|
#endif
|
||
|
bool bRet = (hThread != NULL);
|
||
|
if(bRet)
|
||
|
::CloseHandle(hThread);
|
||
|
return bRet;
|
||
|
}
|
||
|
|
||
|
static DWORD WINAPI MonitorProc(void* pv)
|
||
|
{
|
||
|
CServerAppModule* p = (CServerAppModule*)pv;
|
||
|
p->MonitorShutdown();
|
||
|
#if !defined(_ATL_MIN_CRT) & defined(_MT)
|
||
|
_endthreadex(0);
|
||
|
#endif
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
// Scan command line and perform registration
|
||
|
// Return value specifies if server should run
|
||
|
|
||
|
// Parses the command line and registers/unregisters the rgs file if necessary
|
||
|
bool ParseCommandLine(LPCTSTR lpCmdLine, UINT nResId, HRESULT* pnRetCode)
|
||
|
{
|
||
|
TCHAR szTokens[] = _T("-/");
|
||
|
*pnRetCode = S_OK;
|
||
|
|
||
|
LPCTSTR lpszToken = FindOneOf(lpCmdLine, szTokens);
|
||
|
while(lpszToken != NULL)
|
||
|
{
|
||
|
if(lstrcmpi(lpszToken, _T("UnregServer"))==0)
|
||
|
{
|
||
|
*pnRetCode = UnregisterServer(TRUE);
|
||
|
ATLASSERT(SUCCEEDED(*pnRetCode));
|
||
|
if(FAILED(*pnRetCode))
|
||
|
return false;
|
||
|
*pnRetCode = UpdateRegistryFromResource(nResId, FALSE);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Register as Local Server
|
||
|
if(lstrcmpi(lpszToken, _T("RegServer"))==0)
|
||
|
{
|
||
|
*pnRetCode = UpdateRegistryFromResource(nResId, TRUE);
|
||
|
ATLASSERT(SUCCEEDED(*pnRetCode));
|
||
|
if(FAILED(*pnRetCode))
|
||
|
return false;
|
||
|
*pnRetCode = RegisterServer(TRUE);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
lpszToken = FindOneOf(lpszToken, szTokens);
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// Parses the command line and registers/unregisters the appid if necessary
|
||
|
bool ParseCommandLine(LPCTSTR lpCmdLine, LPCTSTR pAppId, HRESULT* pnRetCode)
|
||
|
{
|
||
|
TCHAR szTokens[] = _T("-/");
|
||
|
*pnRetCode = S_OK;
|
||
|
|
||
|
LPCTSTR lpszToken = FindOneOf(lpCmdLine, szTokens);
|
||
|
while(lpszToken != NULL)
|
||
|
{
|
||
|
if(lstrcmpi(lpszToken, _T("UnregServer"))==0)
|
||
|
{
|
||
|
*pnRetCode = UnregisterAppId(pAppId);
|
||
|
ATLASSERT(SUCCEEDED(*pnRetCode));
|
||
|
if(FAILED(*pnRetCode))
|
||
|
return false;
|
||
|
*pnRetCode = UnregisterServer(TRUE);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Register as Local Server
|
||
|
if(lstrcmpi(lpszToken, _T("RegServer"))==0)
|
||
|
{
|
||
|
*pnRetCode = RegisterAppId(pAppId);
|
||
|
ATLASSERT(SUCCEEDED(*pnRetCode));
|
||
|
if(FAILED(*pnRetCode))
|
||
|
return false;
|
||
|
*pnRetCode = RegisterServer(TRUE);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
lpszToken = FindOneOf(lpszToken, szTokens);
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#if !defined(_ATL_TMP_IMPL1) && !defined(_ATL_TMP_IMPL2)
|
||
|
// search for an occurence of string p2 in string p1
|
||
|
static LPCTSTR FindOneOf(LPCTSTR p1, LPCTSTR p2)
|
||
|
{
|
||
|
while(p1 != NULL && *p1 != NULL)
|
||
|
{
|
||
|
LPCTSTR p = p2;
|
||
|
while(p != NULL && *p != NULL)
|
||
|
{
|
||
|
if(*p1 == *p)
|
||
|
return ::CharNext(p1);
|
||
|
p = ::CharNext(p);
|
||
|
}
|
||
|
p1 = ::CharNext(p1);
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
HRESULT RegisterAppId(LPCTSTR pAppId)
|
||
|
{
|
||
|
CRegKey keyAppID;
|
||
|
if(keyAppID.Open(HKEY_CLASSES_ROOT, _T("AppID"), KEY_READ) == ERROR_SUCCESS)
|
||
|
{
|
||
|
TCHAR szModule1[_MAX_PATH];
|
||
|
TCHAR szModule2[_MAX_PATH];
|
||
|
TCHAR* pszFileName;
|
||
|
::GetModuleFileName(GetModuleInstance(), szModule1, _MAX_PATH);
|
||
|
::GetFullPathName(szModule1, _MAX_PATH, szModule2, &pszFileName);
|
||
|
CRegKey keyAppIDEXE;
|
||
|
if(keyAppIDEXE.Create(keyAppID, pszFileName, REG_NONE, REG_OPTION_NON_VOLATILE, KEY_READ | KEY_WRITE) == ERROR_SUCCESS)
|
||
|
keyAppIDEXE.SetValue(pAppId, _T("AppID"));
|
||
|
if(keyAppIDEXE.Create(keyAppID, pAppId, REG_NONE, REG_OPTION_NON_VOLATILE, KEY_READ | KEY_WRITE) == ERROR_SUCCESS)
|
||
|
keyAppIDEXE.SetValue(pszFileName);
|
||
|
}
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT UnregisterAppId(LPCTSTR pAppId)
|
||
|
{
|
||
|
CRegKey keyAppID;
|
||
|
if(keyAppID.Open(HKEY_CLASSES_ROOT, _T("AppID"), KEY_READ) == ERROR_SUCCESS)
|
||
|
{
|
||
|
TCHAR szModule1[_MAX_PATH];
|
||
|
TCHAR szModule2[_MAX_PATH];
|
||
|
TCHAR* pszFileName;
|
||
|
::GetModuleFileName(GetModuleInstance(), szModule1, _MAX_PATH);
|
||
|
::GetFullPathName(szModule1, _MAX_PATH, szModule2, &pszFileName);
|
||
|
keyAppID.RecurseDeleteKey(pszFileName);
|
||
|
keyAppID.RecurseDeleteKey(pAppId);
|
||
|
}
|
||
|
return S_OK;
|
||
|
}
|
||
|
#endif //!defined(_ATL_TMP_IMPL1) && !defined(_ATL_TMP_IMPL2)
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// ATL 3.0 Add-ons
|
||
|
|
||
|
// protect template members from windowsx.h macros
|
||
|
#ifdef _INC_WINDOWSX
|
||
|
#undef SubclassWindow
|
||
|
#endif //_INC_WINDOWSX
|
||
|
|
||
|
// define useful macros from windowsx.h
|
||
|
#ifndef GET_X_LPARAM
|
||
|
#define GET_X_LPARAM(lParam) ((int)(short)LOWORD(lParam))
|
||
|
#endif
|
||
|
#ifndef GET_Y_LPARAM
|
||
|
#define GET_Y_LPARAM(lParam) ((int)(short)HIWORD(lParam))
|
||
|
#endif
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// Dual argument helper classes
|
||
|
|
||
|
#ifndef _ATL_TMP_IMPL2
|
||
|
|
||
|
class _U_RECT
|
||
|
{
|
||
|
public:
|
||
|
_U_RECT(LPRECT lpRect) : m_lpRect(lpRect)
|
||
|
{ }
|
||
|
_U_RECT(RECT& rc) : m_lpRect(&rc)
|
||
|
{ }
|
||
|
LPRECT m_lpRect;
|
||
|
};
|
||
|
|
||
|
class _U_MENUorID
|
||
|
{
|
||
|
public:
|
||
|
_U_MENUorID(HMENU hMenu) : m_hMenu(hMenu)
|
||
|
{ }
|
||
|
_U_MENUorID(UINT nID) : m_hMenu((HMENU)LongToHandle(nID))
|
||
|
{ }
|
||
|
HMENU m_hMenu;
|
||
|
};
|
||
|
|
||
|
class _U_STRINGorID
|
||
|
{
|
||
|
public:
|
||
|
_U_STRINGorID(LPCTSTR lpString) : m_lpstr(lpString)
|
||
|
{ }
|
||
|
_U_STRINGorID(UINT nID) : m_lpstr(MAKEINTRESOURCE(nID))
|
||
|
{ }
|
||
|
LPCTSTR m_lpstr;
|
||
|
};
|
||
|
|
||
|
#endif //!_ATL_TMP_IMPL2
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// Forward notifications support for message maps
|
||
|
|
||
|
#if !defined(_ATL_TMP_IMPL1) && !defined(_ATL_TMP_IMPL2)
|
||
|
|
||
|
// forward notifications support
|
||
|
#define FORWARD_NOTIFICATIONS() \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = Atl3ForwardNotifications(m_hWnd, uMsg, wParam, lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
static LRESULT Atl3ForwardNotifications(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
||
|
{
|
||
|
LRESULT lResult = 0;
|
||
|
switch(uMsg)
|
||
|
{
|
||
|
case WM_COMMAND:
|
||
|
case WM_NOTIFY:
|
||
|
case WM_PARENTNOTIFY:
|
||
|
case WM_DRAWITEM:
|
||
|
case WM_MEASUREITEM:
|
||
|
case WM_COMPAREITEM:
|
||
|
case WM_DELETEITEM:
|
||
|
case WM_VKEYTOITEM:
|
||
|
case WM_CHARTOITEM:
|
||
|
case WM_HSCROLL:
|
||
|
case WM_VSCROLL:
|
||
|
case WM_CTLCOLORBTN:
|
||
|
case WM_CTLCOLORDLG:
|
||
|
case WM_CTLCOLOREDIT:
|
||
|
case WM_CTLCOLORLISTBOX:
|
||
|
case WM_CTLCOLORMSGBOX:
|
||
|
case WM_CTLCOLORSCROLLBAR:
|
||
|
case WM_CTLCOLORSTATIC:
|
||
|
lResult = ::SendMessage(::GetParent(hWnd), uMsg, wParam, lParam);
|
||
|
break;
|
||
|
default:
|
||
|
bHandled = FALSE;
|
||
|
break;
|
||
|
}
|
||
|
return lResult;
|
||
|
}
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// Reflected message handler macros for message maps
|
||
|
|
||
|
#define REFLECTED_COMMAND_HANDLER(id, code, func) \
|
||
|
if(uMsg == OCM_COMMAND && id == LOWORD(wParam) && code == HIWORD(wParam)) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func(HIWORD(wParam), LOWORD(wParam), (HWND)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_COMMAND_ID_HANDLER(id, func) \
|
||
|
if(uMsg == OCM_COMMAND && id == LOWORD(wParam)) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func(HIWORD(wParam), LOWORD(wParam), (HWND)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_COMMAND_CODE_HANDLER(code, func) \
|
||
|
if(uMsg == OCM_COMMAND && code == HIWORD(wParam)) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func(HIWORD(wParam), LOWORD(wParam), (HWND)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_COMMAND_RANGE_HANDLER(idFirst, idLast, func) \
|
||
|
if(uMsg == OCM_COMMAND && LOWORD(wParam) >= idFirst && LOWORD(wParam) <= idLast) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func(HIWORD(wParam), LOWORD(wParam), (HWND)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_COMMAND_RANGE_CODE_HANDLER(idFirst, idLast, code, func) \
|
||
|
if(uMsg == OCM_COMMAND && code == HIWORD(wParam) && LOWORD(wParam) >= idFirst && LOWORD(wParam) <= idLast) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func(HIWORD(wParam), LOWORD(wParam), (HWND)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_NOTIFY_HANDLER(id, cd, func) \
|
||
|
if(uMsg == OCM_NOTIFY && id == ((LPNMHDR)lParam)->idFrom && cd == ((LPNMHDR)lParam)->code) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_NOTIFY_ID_HANDLER(id, func) \
|
||
|
if(uMsg == OCM_NOTIFY && id == ((LPNMHDR)lParam)->idFrom) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_NOTIFY_CODE_HANDLER(cd, func) \
|
||
|
if(uMsg == OCM_NOTIFY && cd == ((LPNMHDR)lParam)->code) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_NOTIFY_RANGE_HANDLER(idFirst, idLast, func) \
|
||
|
if(uMsg == OCM_NOTIFY && ((LPNMHDR)lParam)->idFrom >= idFirst && ((LPNMHDR)lParam)->idFrom <= idLast) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
#define REFLECTED_NOTIFY_RANGE_CODE_HANDLER(idFirst, idLast, cd, func) \
|
||
|
if(uMsg == OCM_NOTIFY && cd == ((LPNMHDR)lParam)->code && ((LPNMHDR)lParam)->idFrom >= idFirst && ((LPNMHDR)lParam)->idFrom <= idLast) \
|
||
|
{ \
|
||
|
bHandled = TRUE; \
|
||
|
lResult = func((int)wParam, (LPNMHDR)lParam, bHandled); \
|
||
|
if(bHandled) \
|
||
|
return TRUE; \
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CString forward reference (to be used with atluser.h and atlgdi.h)
|
||
|
|
||
|
#ifdef _WTL_FORWARD_DECLARE_CSTRING
|
||
|
|
||
|
#define __ATLSTR_H__
|
||
|
class CString; // forward declaration (include atlmisc.h for the whole class)
|
||
|
|
||
|
#endif //_WTL_FORWARD_DECLARE_CSTRING
|
||
|
|
||
|
#endif //!defined(_ATL_TMP_IMPL1) && !defined(_ATL_TMP_IMPL2)
|
||
|
|
||
|
}; //namespace WTL
|
||
|
|
||
|
// These are always included
|
||
|
#include <atluser.h>
|
||
|
#include <atlgdi.h>
|
||
|
|
||
|
// old names compatibility
|
||
|
#ifdef _ATL_TMP_IMPL1
|
||
|
namespace WTL
|
||
|
{
|
||
|
typedef CFontHandle CWinFont;
|
||
|
typedef CMenuHandle CWinMenu;
|
||
|
#define CWinPropertyPageImpl CPropertyPageImpl
|
||
|
typedef CDCHandle CWinDC;
|
||
|
}; //namespace WTL
|
||
|
#endif //_ATL_TMP_IMPL1
|
||
|
|
||
|
#ifndef _WTL_NO_AUTOMATIC_NAMESPACE
|
||
|
using namespace WTL;
|
||
|
#endif //!_WTL_NO_AUTOMATIC_NAMESPACE
|
||
|
|
||
|
#endif // __ATLAPP_H__
|