| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352 | // Windows Template Library - WTL version 10.0// Copyright (C) Microsoft Corporation, WTL Team. All rights reserved.//// This file is a part of the Windows Template Library.// The use and distribution terms for this software are covered by the// Microsoft Public License (http://opensource.org/licenses/MS-PL)// which can be found in the file MS-PL.txt at the root folder.#ifndef __ATLMISC_H__#define __ATLMISC_H__#pragma once#ifndef __ATLAPP_H__	#error atlmisc.h requires atlapp.h to be included first#endif#ifndef _WTL_NO_COMPATIBILITY_INCLUDES  #include <atlstr.h>  #include <atltypes.h>#endif // _WTL_NO_COMPATIBILITY_INCLUDES///////////////////////////////////////////////////////////////////////////////// Classes in this file://// CRecentDocumentListBase<T, t_cchItemLen, t_nFirstID, t_nLastID>// CRecentDocumentList// CFindFile// CRegProperty// CRegPropertyImpl<T>//// Global functions://   AtlGetStockPen()//   AtlGetStockBrush()//   AtlGetStockFont()//   AtlGetStockPalette()////   AtlCompactPath()namespace WTL{///////////////////////////////////////////////////////////////////////////////// CSize scalar operators #if !defined(_WTL_NO_SIZE_SCALAR) && defined(__ATLTYPES_H__)template <class Num>inline CSize operator *(SIZE s, Num n) {	return CSize((int)(s.cx * n), (int)(s.cy * n));};template <class Num>inline void operator *=(SIZE & s, Num n){	s = s * n;};	template <class Num>inline CSize operator /(SIZE s, Num n) {	return CSize((int)(s.cx / n), (int)(s.cy / n));};template <class Num>inline void operator /=(SIZE & s, Num n){	s = s / n;};	#endif // !defined(_WTL_NO_SIZE_SCALAR) && defined(__ATLTYPES_H__)///////////////////////////////////////////////////////////////////////////////// CRecentDocumentList - MRU List Support#ifndef _WTL_MRUEMPTY_TEXT  #define _WTL_MRUEMPTY_TEXT	_T("(empty)")#endif// forward declarationinline bool AtlCompactPath(LPTSTR lpstrOut, LPCTSTR lpstrIn, int cchLen);template <class T, int t_cchItemLen = MAX_PATH, int t_nFirstID = ID_FILE_MRU_FIRST, int t_nLastID = ID_FILE_MRU_LAST>class CRecentDocumentListBase{public:// Declarations	struct _DocEntry	{		TCHAR szDocName[t_cchItemLen];		bool operator ==(const _DocEntry& de) const		{ return (lstrcmpi(szDocName, de.szDocName) == 0); }	};	enum	{		m_nMaxEntries_Min = 2,		m_nMaxEntries_Max = t_nLastID - t_nFirstID + 1,		m_cchMaxItemLen_Min = 6,		m_cchMaxItemLen_Max = t_cchItemLen,		m_cchItemNameLen = 11	};// Data members	ATL::CSimpleArray<_DocEntry> m_arrDocs;	int m_nMaxEntries;   // default is 4	HMENU m_hMenu;	TCHAR m_szNoEntries[t_cchItemLen];	int m_cchMaxItemLen;// Constructor	CRecentDocumentListBase() : m_nMaxEntries(4), m_hMenu(NULL), m_cchMaxItemLen(-1)	{		m_szNoEntries[0] = 0;		// These ASSERTs verify values of the template arguments		ATLASSERT(t_cchItemLen > m_cchMaxItemLen_Min);		ATLASSERT(m_nMaxEntries_Max > m_nMaxEntries_Min);	}// Attributes	HMENU GetMenuHandle() const	{		return m_hMenu;	}	void SetMenuHandle(HMENU hMenu)	{		ATLASSERT((hMenu == NULL) || ::IsMenu(hMenu));		m_hMenu = hMenu;		if((m_hMenu == NULL) || (::GetMenuString(m_hMenu, t_nFirstID, m_szNoEntries, t_cchItemLen, MF_BYCOMMAND) == 0))		{			T* pT = static_cast<T*>(this);			(void)pT;   // avoid level 4 warning			ATL::Checked::tcsncpy_s(m_szNoEntries, _countof(m_szNoEntries), pT->GetMRUEmptyText(), _TRUNCATE);		}	}	int GetMaxEntries() const	{		return m_nMaxEntries;	}	void SetMaxEntries(int nMaxEntries)	{		ATLASSERT((nMaxEntries >= m_nMaxEntries_Min) && (nMaxEntries <= m_nMaxEntries_Max));		if(nMaxEntries < m_nMaxEntries_Min)			nMaxEntries = m_nMaxEntries_Min;		else if(nMaxEntries > m_nMaxEntries_Max)			nMaxEntries = m_nMaxEntries_Max;		m_nMaxEntries = nMaxEntries;	}	int GetMaxItemLength() const	{		return m_cchMaxItemLen;	}	void SetMaxItemLength(int cchMaxLen)	{		ATLASSERT(((cchMaxLen >= m_cchMaxItemLen_Min) && (cchMaxLen <= m_cchMaxItemLen_Max)) || (cchMaxLen == -1));		if(cchMaxLen != -1)		{			if(cchMaxLen < m_cchMaxItemLen_Min)				cchMaxLen = m_cchMaxItemLen_Min;			else if(cchMaxLen > m_cchMaxItemLen_Max)				cchMaxLen = m_cchMaxItemLen_Max;		}		m_cchMaxItemLen = cchMaxLen;		T* pT = static_cast<T*>(this);		pT->UpdateMenu();	}// Operations	BOOL AddToList(LPCTSTR lpstrDocName)	{		_DocEntry de;		errno_t nRet = ATL::Checked::tcsncpy_s(de.szDocName, _countof(de.szDocName), lpstrDocName, _TRUNCATE);		if((nRet != 0) && (nRet != STRUNCATE))			return FALSE;		for(int i = 0; i < m_arrDocs.GetSize(); i++)		{			if(lstrcmpi(m_arrDocs[i].szDocName, lpstrDocName) == 0)			{				m_arrDocs.RemoveAt(i);				break;			}		}		if(m_arrDocs.GetSize() == m_nMaxEntries)			m_arrDocs.RemoveAt(0);		BOOL bRet = m_arrDocs.Add(de);		if(bRet)		{			T* pT = static_cast<T*>(this);			bRet = pT->UpdateMenu();		}		return bRet;	}	// This function is deprecated because it is not safe. 	// Use the version below that accepts the buffer length.	__declspec(deprecated)	BOOL GetFromList(int /*nItemID*/, LPTSTR /*lpstrDocName*/)	{		ATLASSERT(FALSE);		return FALSE;	}	BOOL GetFromList(int nItemID, LPTSTR lpstrDocName, int cchLength)	{		int nIndex = m_arrDocs.GetSize() - (nItemID - t_nFirstID) - 1;		if((nIndex < 0) || (nIndex >= m_arrDocs.GetSize()))			return FALSE;		if(lstrlen(m_arrDocs[nIndex].szDocName) >= cchLength)			return FALSE;		ATL::Checked::tcscpy_s(lpstrDocName, cchLength, m_arrDocs[nIndex].szDocName);		return TRUE;	}#ifdef __ATLSTR_H__	BOOL GetFromList(int nItemID, ATL::CString& strDocName)	{		int nIndex = m_arrDocs.GetSize() - (nItemID - t_nFirstID) - 1;		if((nIndex < 0) || (nIndex >= m_arrDocs.GetSize()))			return FALSE;		strDocName = m_arrDocs[nIndex].szDocName;		return TRUE;	}#endif // __ATLSTR_H__	BOOL RemoveFromList(int nItemID)	{		int nIndex = m_arrDocs.GetSize() - (nItemID - t_nFirstID) - 1;		BOOL bRet = m_arrDocs.RemoveAt(nIndex);		if(bRet)		{			T* pT = static_cast<T*>(this);			bRet = pT->UpdateMenu();		}		return bRet;	}	BOOL MoveToTop(int nItemID)	{		int nIndex = m_arrDocs.GetSize() - (nItemID - t_nFirstID) - 1;		if((nIndex < 0) || (nIndex >= m_arrDocs.GetSize()))			return FALSE;		_DocEntry de;		de = m_arrDocs[nIndex];		m_arrDocs.RemoveAt(nIndex);		BOOL bRet = m_arrDocs.Add(de);		if(bRet)		{			T* pT = static_cast<T*>(this);			bRet = pT->UpdateMenu();		}		return bRet;	}	BOOL ReadFromRegistry(LPCTSTR lpstrRegKey)	{		T* pT = static_cast<T*>(this);		ATL::CRegKey rkParent;		ATL::CRegKey rk;		LONG lRet = rkParent.Open(HKEY_CURRENT_USER, lpstrRegKey);		if(lRet != ERROR_SUCCESS)			return FALSE;		lRet = rk.Open(rkParent, pT->GetRegKeyName());		if(lRet != ERROR_SUCCESS)			return FALSE;		DWORD dwRet = 0;		lRet = rk.QueryDWORDValue(pT->GetRegCountName(), dwRet);		if(lRet != ERROR_SUCCESS)			return FALSE;		SetMaxEntries(dwRet);		m_arrDocs.RemoveAll();		TCHAR szRetString[t_cchItemLen] = {};		_DocEntry de;		for(int nItem = m_nMaxEntries; nItem > 0; nItem--)		{			TCHAR szBuff[m_cchItemNameLen] = {};			_stprintf_s(szBuff, m_cchItemNameLen, pT->GetRegItemName(), nItem);			ULONG ulCount = t_cchItemLen;			lRet = rk.QueryStringValue(szBuff, szRetString, &ulCount);			if(lRet == ERROR_SUCCESS)			{				ATL::Checked::tcscpy_s(de.szDocName, _countof(de.szDocName), szRetString);				m_arrDocs.Add(de);			}		}		rk.Close();		rkParent.Close();		return pT->UpdateMenu();	}	BOOL WriteToRegistry(LPCTSTR lpstrRegKey)	{		T* pT = static_cast<T*>(this);		(void)pT;   // avoid level 4 warning		ATL::CRegKey rkParent;		ATL::CRegKey rk;		LONG lRet = rkParent.Create(HKEY_CURRENT_USER, lpstrRegKey);		if(lRet != ERROR_SUCCESS)			return FALSE;		lRet = rk.Create(rkParent, pT->GetRegKeyName());		if(lRet != ERROR_SUCCESS)			return FALSE;		lRet = rk.SetDWORDValue(pT->GetRegCountName(), m_nMaxEntries);		ATLASSERT(lRet == ERROR_SUCCESS);		// set new values		int nItem;		for(nItem = m_arrDocs.GetSize(); nItem > 0; nItem--)		{			TCHAR szBuff[m_cchItemNameLen] = {};			_stprintf_s(szBuff, m_cchItemNameLen, pT->GetRegItemName(), nItem);			TCHAR szDocName[t_cchItemLen] = {};			GetFromList(t_nFirstID + nItem - 1, szDocName, t_cchItemLen);			lRet = rk.SetStringValue(szBuff, szDocName);			ATLASSERT(lRet == ERROR_SUCCESS);		}		// delete unused keys		for(nItem = m_arrDocs.GetSize() + 1; nItem <= m_nMaxEntries_Max; nItem++)		{			TCHAR szBuff[m_cchItemNameLen] = {};			_stprintf_s(szBuff, m_cchItemNameLen, pT->GetRegItemName(), nItem);			rk.DeleteValue(szBuff);		}		rk.Close();		rkParent.Close();		return TRUE;	}// Implementation	BOOL UpdateMenu()	{		if(m_hMenu == NULL)			return FALSE;		ATLASSERT(::IsMenu(m_hMenu));		int nItems = ::GetMenuItemCount(m_hMenu);		int nInsertPoint = 0;		for(int i = 0; i < nItems; i++)		{			CMenuItemInfo mi;			mi.fMask = MIIM_ID;			::GetMenuItemInfo(m_hMenu, i, TRUE, &mi);			if (mi.wID == t_nFirstID)			{				nInsertPoint = i;				break;			}		}		ATLASSERT((nInsertPoint < nItems) && "You need a menu item with an ID = t_nFirstID");		for(int j = t_nFirstID; j < (t_nFirstID + m_nMaxEntries); j++)		{			// keep the first one as an insertion point			if (j != t_nFirstID)				::DeleteMenu(m_hMenu, j, MF_BYCOMMAND);		}		TCHAR szItemText[t_cchItemLen + 6] = {};   // add space for &, 2 digits, and a space		int nSize = m_arrDocs.GetSize();		int nItem = 0;		if(nSize > 0)		{			for(nItem = 0; nItem < nSize; nItem++)			{				if(m_cchMaxItemLen == -1)				{					_stprintf_s(szItemText, t_cchItemLen + 6, _T("&%i %s"), nItem + 1, m_arrDocs[nSize - 1 - nItem].szDocName);				}				else				{					TCHAR szBuff[t_cchItemLen] = {};					T* pT = static_cast<T*>(this);					(void)pT;   // avoid level 4 warning					bool bRet = pT->CompactDocumentName(szBuff, m_arrDocs[nSize - 1 - nItem].szDocName, m_cchMaxItemLen);					(void)bRet;   // avoid level 4 warning					ATLASSERT(bRet);					_stprintf_s(szItemText, t_cchItemLen + 6, _T("&%i %s"), nItem + 1, szBuff);				}				::InsertMenu(m_hMenu, nInsertPoint + nItem, MF_BYPOSITION | MF_STRING, t_nFirstID + nItem, szItemText);			}		}		else	// empty		{			::InsertMenu(m_hMenu, nInsertPoint, MF_BYPOSITION | MF_STRING, t_nFirstID, m_szNoEntries);			::EnableMenuItem(m_hMenu, t_nFirstID, MF_GRAYED);			nItem++;		}		::DeleteMenu(m_hMenu, nInsertPoint + nItem, MF_BYPOSITION);		return TRUE;	}// Overrideables	// override to provide a different method of compacting document names	static bool CompactDocumentName(LPTSTR lpstrOut, LPCTSTR lpstrIn, int cchLen)	{		return AtlCompactPath(lpstrOut, lpstrIn, cchLen);	}	static LPCTSTR GetRegKeyName()	{		return _T("Recent Document List");	}	static LPCTSTR GetRegCountName()	{		return _T("DocumentCount");	}	static LPCTSTR GetRegItemName()	{		// Note: This string is a format string used with wsprintf().		// Resulting formatted string must be m_cchItemNameLen or less 		// characters long, including the terminating null character.		return _T("Document%i");	}	static LPCTSTR GetMRUEmptyText()	{		return _WTL_MRUEMPTY_TEXT;	}};class CRecentDocumentList : public CRecentDocumentListBase<CRecentDocumentList>{public:// nothing here};///////////////////////////////////////////////////////////////////////////////// CFindFile - file search helper classclass CFindFile{public:// Data members	HANDLE m_hFind;	WIN32_FIND_DATA m_fd;	LPTSTR m_lpszRoot;	const TCHAR m_chDirSeparator;	BOOL m_bFound;// Constructor/destructor	CFindFile() : m_hFind(NULL), m_lpszRoot(NULL), m_chDirSeparator(_T('\\')), m_bFound(FALSE)	{		memset(&m_fd, 0, sizeof(m_fd));	}	~CFindFile()	{		Close();	}// Attributes	ULONGLONG GetFileSize() const	{		ATLASSERT(m_hFind != NULL);		ULARGE_INTEGER nFileSize = {};		if(m_bFound)		{			nFileSize.LowPart = m_fd.nFileSizeLow;			nFileSize.HighPart = m_fd.nFileSizeHigh;		}		else		{			nFileSize.QuadPart = 0;		}		return nFileSize.QuadPart;	}	BOOL GetFileName(LPTSTR lpstrFileName, int cchLength) const	{		ATLASSERT(m_hFind != NULL);		if(lstrlen(m_fd.cFileName) >= cchLength)			return FALSE;		if(m_bFound)			ATL::Checked::tcscpy_s(lpstrFileName, cchLength, m_fd.cFileName);		return m_bFound;	}	BOOL GetFilePath(LPTSTR lpstrFilePath, int cchLength) const	{		ATLASSERT(m_hFind != NULL);		int nLen = lstrlen(m_lpszRoot);		ATLASSERT(nLen > 0);		if(nLen == 0)			return FALSE;		bool bAddSep = (m_lpszRoot[nLen - 1] != m_chDirSeparator);		if((lstrlen(m_lpszRoot) + (bAddSep ?  1 : 0)) >= cchLength)			return FALSE;		ATL::Checked::tcscpy_s(lpstrFilePath, cchLength, m_lpszRoot);		if(bAddSep)		{			TCHAR szSeparator[2] = { m_chDirSeparator, 0 };			ATL::Checked::tcscat_s(lpstrFilePath, cchLength, szSeparator);		}		ATL::Checked::tcscat_s(lpstrFilePath, cchLength, m_fd.cFileName);		return TRUE;	}	BOOL GetFileTitle(LPTSTR lpstrFileTitle, int cchLength) const	{		ATLASSERT(m_hFind != NULL);		TCHAR szBuff[MAX_PATH] = {};		if(!GetFileName(szBuff, MAX_PATH))			return FALSE;		if(lstrlen(szBuff) >= cchLength)			return FALSE;		// find the last dot		LPTSTR pstrDot  = _tcsrchr(szBuff, _T('.'));		if(pstrDot != NULL)			*pstrDot = 0;		ATL::Checked::tcscpy_s(lpstrFileTitle, cchLength, szBuff);		return TRUE;	}	BOOL GetFileURL(LPTSTR lpstrFileURL, int cchLength) const	{		ATLASSERT(m_hFind != NULL);		LPCTSTR lpstrFileURLPrefix = _T("file://");		const int cchPrefix = lstrlen(lpstrFileURLPrefix);		if(cchPrefix >= cchLength)			return FALSE;		ATL::Checked::tcscpy_s(lpstrFileURL, cchLength, lpstrFileURLPrefix);		return GetFilePath(&lpstrFileURL[cchPrefix], cchLength - cchPrefix);	}	BOOL GetRoot(LPTSTR lpstrRoot, int cchLength) const	{		ATLASSERT(m_hFind != NULL);		if(lstrlen(m_lpszRoot) >= cchLength)			return FALSE;		ATL::Checked::tcscpy_s(lpstrRoot, cchLength, m_lpszRoot);		return TRUE;	}#ifdef __ATLSTR_H__	ATL::CString GetFileName() const	{		ATLASSERT(m_hFind != NULL);		ATL::CString ret;		if(m_bFound)			ret = m_fd.cFileName;		return ret;	}	ATL::CString GetFilePath() const	{		ATLASSERT(m_hFind != NULL);		ATL::CString strResult = m_lpszRoot;		int nLen = strResult.GetLength();		ATLASSERT(nLen > 0);		if(nLen == 0)			return strResult;		if(strResult[nLen - 1] != m_chDirSeparator)			strResult += m_chDirSeparator;		strResult += GetFileName();		return strResult;	}	ATL::CString GetFileTitle() const	{		ATLASSERT(m_hFind != NULL);		ATL::CString strResult;		GetFileTitle(strResult.GetBuffer(MAX_PATH), MAX_PATH);		strResult.ReleaseBuffer();		return strResult;	}	ATL::CString GetFileURL() const	{		ATLASSERT(m_hFind != NULL);		ATL::CString strResult("file://");		strResult += GetFilePath();		return strResult;	}	ATL::CString GetRoot() const	{		ATLASSERT(m_hFind != NULL);		ATL::CString str = m_lpszRoot;		return str;	}#endif // __ATLSTR_H__	BOOL GetLastWriteTime(FILETIME* pTimeStamp) const	{		ATLASSERT(m_hFind != NULL);		ATLASSERT(pTimeStamp != NULL);		if(m_bFound && (pTimeStamp != NULL))		{			*pTimeStamp = m_fd.ftLastWriteTime;			return TRUE;		}		return FALSE;	}	BOOL GetLastAccessTime(FILETIME* pTimeStamp) const	{		ATLASSERT(m_hFind != NULL);		ATLASSERT(pTimeStamp != NULL);		if(m_bFound && (pTimeStamp != NULL))		{			*pTimeStamp = m_fd.ftLastAccessTime;			return TRUE;		}		return FALSE;	}	BOOL GetCreationTime(FILETIME* pTimeStamp) const	{		ATLASSERT(m_hFind != NULL);		if(m_bFound && (pTimeStamp != NULL))		{			*pTimeStamp = m_fd.ftCreationTime;			return TRUE;		}		return FALSE;	}	BOOL MatchesMask(DWORD dwMask) const	{		ATLASSERT(m_hFind != NULL);		if(m_bFound)			return ((m_fd.dwFileAttributes & dwMask) != 0);		return FALSE;	}	BOOL IsDots() const	{		ATLASSERT(m_hFind != NULL);		// return TRUE if the file name is "." or ".." and		// the file is a directory		BOOL bResult = FALSE;		if(m_bFound && IsDirectory())		{			if((m_fd.cFileName[0] == _T('.')) && ((m_fd.cFileName[1] == _T('\0')) || ((m_fd.cFileName[1] == _T('.')) && (m_fd.cFileName[2] == _T('\0')))))				bResult = TRUE;		}		return bResult;	}	BOOL IsReadOnly() const	{		return MatchesMask(FILE_ATTRIBUTE_READONLY);	}	BOOL IsDirectory() const	{		return MatchesMask(FILE_ATTRIBUTE_DIRECTORY);	}	BOOL IsCompressed() const	{		return MatchesMask(FILE_ATTRIBUTE_COMPRESSED);	}	BOOL IsSystem() const	{		return MatchesMask(FILE_ATTRIBUTE_SYSTEM);	}	BOOL IsHidden() const	{		return MatchesMask(FILE_ATTRIBUTE_HIDDEN);	}	BOOL IsTemporary() const	{		return MatchesMask(FILE_ATTRIBUTE_TEMPORARY);	}	BOOL IsNormal() const	{		return MatchesMask(FILE_ATTRIBUTE_NORMAL);	}	BOOL IsArchived() const	{		return MatchesMask(FILE_ATTRIBUTE_ARCHIVE);	}// Operations	BOOL FindFile(LPCTSTR pstrName = NULL, bool bAutoLongPath = false)	{		Close();		if(pstrName == NULL)			pstrName = _T("*.*");		if(bAutoLongPath && (lstrlen(pstrName) >= MAX_PATH))		{			LPCTSTR lpstrPrefix = _T("\\\\?\\");			int cchLongPath = lstrlen(lpstrPrefix) + lstrlen(pstrName) + 1;			ATL::CTempBuffer<TCHAR, _WTL_STACK_ALLOC_THRESHOLD> buff;			LPTSTR lpstrLongPath = buff.Allocate(cchLongPath);			if(lpstrLongPath != NULL)			{				ATL::Checked::tcscpy_s(lpstrLongPath, cchLongPath, lpstrPrefix);				ATL::Checked::tcscat_s(lpstrLongPath, cchLongPath, pstrName);				m_hFind = ::FindFirstFile(lpstrLongPath, &m_fd);			}		}		else		{			m_hFind = ::FindFirstFile(pstrName, &m_fd);		}		if(m_hFind == INVALID_HANDLE_VALUE)			return FALSE;		int cchRoot = ::GetFullPathName(pstrName, 0, NULL, NULL);		if(cchRoot > 0)			ATLTRY(m_lpszRoot = new TCHAR[cchRoot]);		if(m_lpszRoot == NULL)			return FALSE;		bool bFullPath = (::GetFullPathName(pstrName, cchRoot, m_lpszRoot, NULL) != 0);		// passed name isn't a valid path but was found by the API		ATLASSERT(bFullPath);		if(!bFullPath)		{			Close();			::SetLastError(ERROR_INVALID_NAME);			return FALSE;		}		else		{			// find the last separator			LPTSTR pstrSep  = _tcsrchr(m_lpszRoot, m_chDirSeparator);			if(pstrSep != NULL)				*pstrSep = _T('\0');		}		m_bFound = TRUE;		return TRUE;	}	BOOL FindNextFile()	{		ATLASSERT(m_hFind != NULL);		if(m_hFind == NULL)			return FALSE;		if(!m_bFound)			return FALSE;		m_bFound = ::FindNextFile(m_hFind, &m_fd);		return m_bFound;	}	void Close()	{		m_bFound = FALSE;		delete [] m_lpszRoot;		m_lpszRoot = NULL;		if((m_hFind != NULL) && (m_hFind != INVALID_HANDLE_VALUE))		{			::FindClose(m_hFind);			m_hFind = NULL;		}	}};///////////////////////////////////////////////////////////////////////////////// CRegProperty and CRegPropertyImpl<> - properties stored in registry// How to use: Derive a class from CRegPropertyImpl, add data members // for properties, and add REGPROP map to map properties to registry value names.// You can then call Read() and Write() methods to read and write properties to/from registry.// You can also use CRegProperty class directly, for one time read/write, or for custom stuff.#define REGPROP_CURRENTUSER   0x0000#define REGPROP_LOCALMACHINE  0x0001#define REGPROP_READONLY      0x0002#define REGPROP_WRITEONLY     0x0004class CRegProperty{public:// Type declarations	struct BinaryProp	{		void* pBinary;		ULONG uSize;   // buffer size in bytes, used size after read		BinaryProp() : pBinary(NULL), uSize(0U)		{ }	};	struct CharArrayProp	{		LPTSTR lpstrText;		ULONG uSize;   // buffer size in chars		CharArrayProp() : lpstrText(NULL), uSize(0U)		{ }	};// Data members	ATL::CRegKey m_regkey;	WORD m_wFlags;// Constructor	CRegProperty() : m_wFlags(REGPROP_CURRENTUSER)	{ }// Registry key methods	LSTATUS OpenRegKey(LPCTSTR lpstrRegKey, bool bWrite)	{		ATLASSERT(m_regkey.m_hKey == NULL);		HKEY hKey = ((m_wFlags & REGPROP_LOCALMACHINE) != 0) ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;		REGSAM sam = KEY_READ | KEY_WRITE;		LSTATUS lRet = -1;		if(bWrite)			lRet = m_regkey.Create(hKey, lpstrRegKey, NULL, 0, ((m_wFlags & REGPROP_WRITEONLY) != 0) ? KEY_WRITE : sam);		else			lRet = m_regkey.Open(hKey, lpstrRegKey, ((m_wFlags & REGPROP_READONLY) != 0) ? KEY_READ : sam);		return lRet;	}	void CloseRegKey()	{		LSTATUS lRet = m_regkey.Close();		(void)lRet;   // avoid level 4 warning		ATLASSERT(lRet == ERROR_SUCCESS);	}// Flag methods	WORD GetFlags() const	{		return m_wFlags;	}	WORD SetFlags(WORD wFlags, WORD wMask = 0)	{		WORD wPrevFlags = m_wFlags;		if(wMask == 0)			m_wFlags = wFlags;		else			m_wFlags = (m_wFlags & ~wMask) | (wFlags & wMask);		return wPrevFlags;	}// Generic read/write methods	template <class TProp>	LSTATUS ReadProp(LPCTSTR lpstrRegValue, TProp& prop)	{		ATLASSERT(m_regkey.m_hKey != NULL);		DWORD dwRet = 0;		LSTATUS lRet = m_regkey.QueryDWORDValue(lpstrRegValue, dwRet);		if(lRet == ERROR_SUCCESS)			prop = static_cast<TProp>(dwRet);		return lRet;	}	template <class TProp>	LSTATUS WriteProp(LPCTSTR lpstrRegValue, TProp& prop)	{		ATLASSERT(m_regkey.m_hKey != NULL);		return m_regkey.SetDWORDValue(lpstrRegValue, (DWORD)prop);	}// Specialization for bool	template <>	LSTATUS ReadProp(LPCTSTR lpstrRegValue, bool& bProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		DWORD dwRet = 0;		LSTATUS lRet = m_regkey.QueryDWORDValue(lpstrRegValue, dwRet);		if(lRet == ERROR_SUCCESS)			bProp = (dwRet != 0);		return lRet;	}	template <>	LSTATUS WriteProp(LPCTSTR lpstrRegValue, bool& bProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		return m_regkey.SetDWORDValue(lpstrRegValue, bProp ? 1 : 0);	}// Specialization for HFONT	template <>	LSTATUS ReadProp(LPCTSTR lpstrRegValue, HFONT& hFont)	{		ATLASSERT(m_regkey.m_hKey != NULL);		LOGFONT lf = {};		ULONG uSize = sizeof(lf);		LSTATUS lRet = m_regkey.QueryBinaryValue(lpstrRegValue, &lf, &uSize);		if(lRet == ERROR_SUCCESS)		{			if(hFont != NULL)				::DeleteObject(hFont);			hFont = ::CreateFontIndirect(&lf);			if(hFont == NULL)				lRet = ERROR_INVALID_DATA;		}		return lRet;	}	template <>	LSTATUS WriteProp(LPCTSTR lpstrRegValue, HFONT& hFont)	{		ATLASSERT(m_regkey.m_hKey != NULL);		CLogFont lf(hFont);		return m_regkey.SetBinaryValue(lpstrRegValue, &lf, sizeof(lf));	}// Specialization for BinaryProp	template <>	LSTATUS ReadProp(LPCTSTR lpstrRegValue, BinaryProp& binProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		ULONG uSize = 0U;		LSTATUS lRet = m_regkey.QueryBinaryValue(lpstrRegValue, NULL, &uSize);		if(lRet == ERROR_SUCCESS)		{			if(uSize <= binProp.uSize)				lRet = m_regkey.QueryBinaryValue(lpstrRegValue, binProp.pBinary, &binProp.uSize);			else				lRet = ERROR_OUTOFMEMORY;		}		return lRet;	}	template <>	LSTATUS WriteProp(LPCTSTR lpstrRegValue, BinaryProp& binProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		return m_regkey.SetBinaryValue(lpstrRegValue, binProp.pBinary, binProp.uSize);	}// Specialization for CharArrayProp	template <>	LSTATUS ReadProp(LPCTSTR lpstrRegValue, CharArrayProp& caProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		ULONG uSize = 0U;		LSTATUS lRet = m_regkey.QueryStringValue(lpstrRegValue, NULL, &uSize);		if(lRet == ERROR_SUCCESS)		{			if(uSize <= caProp.uSize)				lRet = m_regkey.QueryStringValue(lpstrRegValue, caProp.lpstrText, &caProp.uSize);			else				lRet = ERROR_OUTOFMEMORY;		}		return lRet;	}	template <>	LSTATUS WriteProp(LPCTSTR lpstrRegValue, CharArrayProp& caProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		return m_regkey.SetStringValue(lpstrRegValue, caProp.lpstrText);	}// Specialization for CString#ifdef __ATLSTR_H__	template <>	LSTATUS ReadProp(LPCTSTR lpstrRegValue, ATL::CString& strProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		ULONG uSize = 0U;		LSTATUS lRet = m_regkey.QueryStringValue(lpstrRegValue, NULL, &uSize);		if(lRet == ERROR_SUCCESS)		{			lRet = m_regkey.QueryStringValue(lpstrRegValue, strProp.GetBufferSetLength(uSize), &uSize);			strProp.ReleaseBuffer();		}		return lRet;	}	template <>	LSTATUS WriteProp(LPCTSTR lpstrRegValue, ATL::CString& strProp)	{		ATLASSERT(m_regkey.m_hKey != NULL);		return m_regkey.SetStringValue(lpstrRegValue, (LPCTSTR)strProp);	}#endif // __ATLSTR_H__// Static methods for one time read/write	template <class TProp>	static bool ReadOne(LPCTSTR lpstrRegKey, LPCTSTR lpstrRegValue, TProp& prop, WORD wFlags = REGPROP_CURRENTUSER)	{		CRegProperty rp;		rp.SetFlags(wFlags);		LSTATUS lRet = rp.OpenRegKey(lpstrRegKey, false);		if(lRet == ERROR_SUCCESS)		{			lRet = rp.ReadProp(lpstrRegValue, prop);			rp.CloseRegKey();		}		return (lRet == ERROR_SUCCESS) || (lRet == ERROR_FILE_NOT_FOUND);	}	template <class TProp>	static bool WriteOne(LPCTSTR lpstrRegKey, LPCTSTR lpstrRegValue, TProp& prop, WORD wFlags = REGPROP_CURRENTUSER)	{		CRegProperty rp;		rp.SetFlags(wFlags);		LSTATUS lRet = rp.OpenRegKey(lpstrRegKey, true);		if(lRet == ERROR_SUCCESS)		{			lRet = rp.WriteProp(lpstrRegValue, prop);			rp.CloseRegKey();		}		return (lRet == ERROR_SUCCESS);	}};#define BEGIN_REGPROP_MAP(class) \	void ReadWriteAll(bool bWrite) \	{#define REG_PROPERTY(name, prop) \		this->ReadWriteProp(name, prop, bWrite);#define END_REGPROP_MAP() \	}template <class T>class CRegPropertyImpl : public CRegProperty{public:// Methods	void Read(LPCTSTR lpstrRegKey)	{		T* pT = static_cast<T*>(this);		LSTATUS lRet = pT->OpenRegKey(lpstrRegKey, false);		if(lRet == ERROR_SUCCESS)		{			pT->ReadWriteAll(false);			pT->OnRead(lpstrRegKey);			pT->CloseRegKey();		}		else if(lRet != ERROR_FILE_NOT_FOUND)		{			pT->OnReadError(NULL, lRet);		}	}	void Write(LPCTSTR lpstrRegKey)	{		T* pT = static_cast<T*>(this);		LSTATUS lRet = pT->OpenRegKey(lpstrRegKey, true);		if(lRet == ERROR_SUCCESS)		{			pT->ReadWriteAll(true);			pT->OnWrite(lpstrRegKey);			pT->CloseRegKey();		}		else		{			pT->OnWriteError(NULL, lRet);		}	}// Implementation	template <class TProp>	void ReadWriteProp(LPCTSTR lpstrRegValue, TProp& prop, bool bWrite)	{		T* pT = static_cast<T*>(this);		if(bWrite)		{			LSTATUS lRet = pT->WriteProp(lpstrRegValue, prop);			if(lRet != ERROR_SUCCESS)				pT->OnWriteError(lpstrRegValue, lRet);		}		else		{			LSTATUS lRet = pT->ReadProp(lpstrRegValue, prop);			if((lRet != ERROR_SUCCESS) && (lRet != ERROR_FILE_NOT_FOUND))				pT->OnReadError(lpstrRegValue, lRet);		}	}// Overrideable handlers	void OnRead(LPCTSTR /*lpstrRegKey*/)	{ }	void OnWrite(LPCTSTR /*lpstrRegKey*/)	{ }	void OnReadError(LPCTSTR /*lpstrRegValue*/, LSTATUS /*lError*/)	{		ATLASSERT(FALSE);	}	void OnWriteError(LPCTSTR /*lpstrRegValue*/, LSTATUS /*lError*/)	{		ATLASSERT(FALSE);	}};///////////////////////////////////////////////////////////////////////////////// Global functions for stock GDI objectsinline HPEN AtlGetStockPen(int nPen){	ATLASSERT((nPen == WHITE_PEN) || (nPen == BLACK_PEN) || (nPen == NULL_PEN) || (nPen == DC_PEN));	return (HPEN)::GetStockObject(nPen);}inline HBRUSH AtlGetStockBrush(int nBrush){	ATLASSERT(((nBrush >= WHITE_BRUSH) && (nBrush <= HOLLOW_BRUSH)) || (nBrush == DC_BRUSH));	return (HBRUSH)::GetStockObject(nBrush);}inline HFONT AtlGetStockFont(int nFont){	ATLASSERT(((nFont >= OEM_FIXED_FONT) && (nFont <= SYSTEM_FIXED_FONT)) || (nFont == DEFAULT_GUI_FONT));	return (HFONT)::GetStockObject(nFont);}inline HPALETTE AtlGetStockPalette(int nPalette){	ATLASSERT(nPalette == DEFAULT_PALETTE); // the only one supported	return (HPALETTE)::GetStockObject(nPalette);}///////////////////////////////////////////////////////////////////////////////// Global function for compacting a path by replacing parts with ellipsis// helper for multi-byte character setsinline bool _IsDBCSTrailByte(LPCTSTR lpstr, int nChar){#ifndef _UNICODE	int i = nChar;	for( ; i > 0; i--)	{		if(!::IsDBCSLeadByte(lpstr[i - 1]))			break;	}	return ((nChar > 0) && (((nChar - i) & 1) != 0));#else // _UNICODE	(void)lpstr;   // avoid level 4 warning	(void)nChar;   // avoid level 4 warning	return false;#endif // _UNICODE}inline bool AtlCompactPath(LPTSTR lpstrOut, LPCTSTR lpstrIn, int cchLen){	ATLASSERT(lpstrOut != NULL);	ATLASSERT(lpstrIn != NULL);	ATLASSERT(cchLen > 0);	LPCTSTR szEllipsis = _T("...");	const int cchEndEllipsis = 3;	const int cchMidEllipsis = 4;	if(lstrlen(lpstrIn) < cchLen)	{		ATL::Checked::tcscpy_s(lpstrOut, cchLen, lpstrIn);		return true;	}	lpstrOut[0] = 0;	// check if the separator is a slash or a backslash	TCHAR chSlash = _T('\\');	for(LPTSTR lpstr = (LPTSTR)lpstrIn; *lpstr != 0; lpstr = ::CharNext(lpstr))	{		if((*lpstr == _T('/')) || (*lpstr == _T('\\')))			chSlash = *lpstr;	}	// find the filename portion of the path	LPCTSTR lpstrFileName = lpstrIn;	for(LPCTSTR pPath = lpstrIn; *pPath; pPath = ::CharNext(pPath))	{		if(((pPath[0] == _T('\\')) || (pPath[0] == _T(':')) || (pPath[0] == _T('/')))				&& pPath[1] && (pPath[1] != _T('\\')) && (pPath[1] != _T('/')))			lpstrFileName = pPath + 1;	}	int cchFileName = lstrlen(lpstrFileName);	// handle just the filename without a path	if((lpstrFileName == lpstrIn) && (cchLen > cchEndEllipsis))	{		bool bRet = (ATL::Checked::tcsncpy_s(lpstrOut, cchLen, lpstrIn, cchLen - cchEndEllipsis - 1) == 0);		if(bRet)		{#ifndef _UNICODE			if(_IsDBCSTrailByte(lpstrIn, cchLen - cchEndEllipsis))				lpstrOut[cchLen - cchEndEllipsis - 1] = 0;#endif // _UNICODE			ATL::Checked::tcscat_s(lpstrOut, cchLen, szEllipsis);		}		return bRet;	}	// handle just ellipsis	if((cchLen < (cchMidEllipsis + cchEndEllipsis)))	{		for(int i = 0; i < cchLen - 1; i++)			lpstrOut[i] = ((i + 1) == cchMidEllipsis) ? chSlash : _T('.');		lpstrOut[cchLen - 1] = 0;		return true;	}	// calc how much we have to copy	int cchToCopy = cchLen - (cchMidEllipsis + cchFileName) - 1;	if(cchToCopy < 0)		cchToCopy = 0;#ifndef _UNICODE	if((cchToCopy > 0) && _IsDBCSTrailByte(lpstrIn, cchToCopy))		cchToCopy--;#endif // _UNICODE	bool bRet = (ATL::Checked::tcsncpy_s(lpstrOut, cchLen, lpstrIn, cchToCopy) == 0);	if(!bRet)		return false;	// add ellipsis	ATL::Checked::tcscat_s(lpstrOut, cchLen, szEllipsis);	TCHAR szSlash[2] = { chSlash, 0 };	ATL::Checked::tcscat_s(lpstrOut, cchLen, szSlash);	// add filename (and ellipsis, if needed)	if(cchLen > (cchMidEllipsis + cchFileName))	{		ATL::Checked::tcscat_s(lpstrOut, cchLen, lpstrFileName);	}	else	{		cchToCopy = cchLen - cchMidEllipsis - cchEndEllipsis - 1;#ifndef _UNICODE		if((cchToCopy > 0) && _IsDBCSTrailByte(lpstrFileName, cchToCopy))			cchToCopy--;#endif // _UNICODE		bRet = (ATL::Checked::tcsncpy_s(&lpstrOut[cchMidEllipsis], cchLen - cchMidEllipsis, lpstrFileName, cchToCopy) == 0);		if(bRet)			ATL::Checked::tcscat_s(lpstrOut, cchLen, szEllipsis);	}	return bRet;}} // namespace WTL#endif // __ATLMISC_H__
 |