2025-11-27 16:46:48 +09:00

419 lines
11 KiB
C++

// =============================================================================
// Class Implementation : COXResourceFile
// =============================================================================
//
// Source file : OXResourceFile.cpp
// Version: 9.3
// This software along with its related components, documentation and files ("The Libraries")
// is © 1994-2007 The Code Project (1612916 Ontario Limited) and use of The Libraries is
// governed by a software license agreement ("Agreement"). Copies of the Agreement are
// available at The Code Project (www.codeproject.com), as part of the package you downloaded
// to obtain this file, or directly from our office. For a copy of the license governing
// this software, you may contact us at legalaffairs@codeproject.com, or by calling 416-849-8900.
// //////////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "OXResourceFile.h"
#include "OXResourceLibrary.h"
#include "UTB64Bit.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
#define OXRES_MUN_STRING_ATTEMPT _T("Resource");
IMPLEMENT_DYNAMIC(COXResourceFile, CSharedFile)
// Data members -------------------------------------------------------------
// protected:
// COXResourceLibrary* m_pResLib; the target library if updating
// BOOL m_bAutoDeleteByLib; whether deleted by COXResourceLibrary when
// it is closed
// BOOL m_bFlushOnClose; whether flush on close
// CString m_sResType; the type of the resource
// CString m_sResName; the name of the resource
// WORD m_nResLanguage; the language of the resource
// Member functions ---------------------------------------------------------
// public:
COXResourceFile::COXResourceFile(UINT nAllocFlags /* = GMEM_DDESHARE | GMEM_MOVEABLE */,
UINT nGrowBytes /* = 4096 */)
: CSharedFile(nAllocFlags, nGrowBytes)
{
m_pResLib = NULL;
m_bAutoDeleteByLib = FALSE;
m_bFlushOnClose = TRUE;
// set default values for an empty resource
SetResType(RT_RCDATA);
SetResLanguage(OXRESOURCE_DEFLANGID);
}
COXResourceFile::~COXResourceFile()
{
if (m_pResLib || m_lpBuffer)
Close();
}
BOOL COXResourceFile::Open(UINT nOpenFlags, COXResourceLibrary* pResLib, BOOL bAutoDeleteByLib,
LPCTSTR pszType, LPCTSTR pszName, WORD nLanguage /* = OXRESOURCE_DEFLANGID */,
BOOL bMakeUniqueName /* = FALSE */)
{
// ... Cannot open for write only (use either modeRead or modeReadWrite).
// (Note that modeRead = 0x0000)
ASSERT((nOpenFlags & modeWrite) != modeWrite);
// ... Can only create a resource if opened for read/write
ASSERT( ((nOpenFlags & modeCreate) != modeCreate) ||
((nOpenFlags & modeReadWrite) == modeReadWrite) );
// ... Can only disable truncating if the resource is to be created
ASSERT( ((nOpenFlags & modeNoTruncate) != modeNoTruncate) ||
((nOpenFlags & modeCreate) == modeCreate) );
ASSERT(pResLib && pszType && pszName);
// open once only
ASSERT(m_pResLib == NULL && m_lpBuffer == NULL);
if (m_pResLib || m_lpBuffer)
return FALSE;
BOOL bSuccess = TRUE;
BOOL bFoundExisting = FALSE;
// We search for the resource if (!modeCreate) or (modeCreate & modeNoTruncate)
if (((nOpenFlags & modeCreate) != modeCreate) ||
((nOpenFlags & modeNoTruncate) == modeNoTruncate) )
{
HRSRC hRes = pResLib->FindResource(pszType, pszName, nLanguage);
if (hRes)
{
bFoundExisting = TRUE;
HGLOBAL hResData = pResLib->LoadResource(hRes);
DWORD nSize = pResLib->GetResourceSize(hRes);
if (hResData && nSize)
{
BYTE* lpBuffer = (BYTE*)::LockResource(hResData);
if (lpBuffer)
{
Write(lpBuffer, nSize);
SeekToBegin();
}
else
{
bSuccess = FALSE;
TRACE1("COXResourceFile::Open(): failed to lock the resource \"%s\".\n", pszName);
}
}
else
{
bSuccess = FALSE;
TRACE1("COXResourceFile::Open(): failed to load the resource \"%s\".\n", pszName);
}
}
else
{
// The resource was not found, fail if no modeCreate is specified
bSuccess = ((nOpenFlags & modeCreate) == modeCreate);
if (!bSuccess)
TRACE1("COXResourceFile::Open(): failed to find the resource \"%s\".\n", pszName);
}
}
if (bSuccess)
{
SetResType(pszType);
SetResName(pszName);
SetResLanguage(nLanguage);
SetResourceLibrary(pResLib, bAutoDeleteByLib,
(bFoundExisting ? FALSE : bMakeUniqueName));
m_bFlushOnClose = ((nOpenFlags & modeReadWrite) == modeReadWrite);
}
return bSuccess;
}
void COXResourceFile::Close()
{
if (m_bFlushOnClose)
Flush();
Abort();
}
void COXResourceFile::Abort()
{
SetResourceLibrary(NULL);
if (m_lpBuffer)
CSharedFile::Close();
}
void COXResourceFile::Flush()
{
if (m_pResLib == NULL || !m_bFlushOnClose)
return;
if(m_pResLib->GetUpdateHandle())
{ // BeginUpdate() has been called already
VERIFY(m_pResLib->Update(m_lpBuffer, PtrToUint(m_nFileSize), m_sResType,
m_sResName, m_nResLanguage));
}
else
{ // single resource instant update
if (m_pResLib->BeginUpdate())
{
VERIFY(m_pResLib->Update(m_lpBuffer, PtrToUint(m_nFileSize), m_sResType,
m_sResName, m_nResLanguage));
m_pResLib->EndUpdate();
}
else
TRACE0("COXResourceFile::Flush(): failed to begin an update.\n");
}
}
HGLOBAL COXResourceFile::DetachEx()
// Even though COXResourceFile::Detach() can return an HGLOBAL that is ready
// for clipboard operations, it does NOT contain the type, name and language
// info we need (it only contains the raw binary data block). To solve
// this problem, we modify the binary block a little bit:
//
// Original data: XX XX XX XX xx xx xx xx
// <1> we duplicate the first 4 bytes (1 DWORD size) at the end:
// XX XX XX XX xx xx xx xx XX XX XX XX
// <2> we write down the file size into the first 1 DWORD:
// 00 00 00 08 xx xx xx xx XX XX XX XX
// <3> we append type, name, lang info at the end
// 00 00 00 08 xx xx xx xx XX XX XX XX (type)(name)(lang)
// Now, when we decode it using SetHandleEx(), we know where to fetch the type,
// name and lang, and we'll put the original 4 bytes back, then set file size back.
{
ASSERT(m_hGlobalMemory && m_bAllowGrow);
if (m_hGlobalMemory == NULL || !m_bAllowGrow)
return NULL;
if (m_nBufferSize < sizeof(DWORD))
GrowFile(sizeof(DWORD));
DWORD_PTR nFileSize0 = m_nFileSize;
DWORD nHeadBlock0 = *((DWORD*)m_lpBuffer);
SeekToBegin();
Write((BYTE*)&nFileSize0, sizeof(DWORD));
SeekToEnd();
CArchive ar(this, CArchive::store);
ar << nHeadBlock0 << m_sResType << m_sResName << m_nResLanguage;
ar.Close();
m_bFlushOnClose = FALSE;
return CSharedFile::Detach();
}
BOOL COXResourceFile::SetHandleEx(HGLOBAL hTaggedResData, BOOL bAllowGrow /* = TRUE */)
{
ASSERT(hTaggedResData);
ASSERT(m_hGlobalMemory == NULL);
if (m_hGlobalMemory)
return FALSE;
SetHandle(hTaggedResData, bAllowGrow);
if (m_nBufferSize < sizeof(DWORD))
{
TRACE0("COXResourceFile::SetHandleEx(): 1st parameter invalid.\n");
return FALSE;
}
DWORD nFileSize0, nHeadBlock0;
SeekToBegin();
Read((BYTE*)&nFileSize0, sizeof(DWORD));
if (nFileSize0 >= m_nFileSize)
{
TRACE0("COXResourceFile::SetHandleEx(): invalid memory block.\n");
return FALSE;
}
Seek(nFileSize0, begin);
CArchive ar(this, CArchive::load);
try
{
ar >> nHeadBlock0 >> m_sResType >> m_sResName >> m_nResLanguage;
}
catch (CArchiveException* e)
{
e->Delete();
TRACE0("COXResourceFile::SetHandleEx(): wrong tag format.\n");
return FALSE;
}
catch (CFileException* e)
{
e->Delete();
AfxThrowMemoryException();
}
ar.Close();
SeekToBegin();
Write((BYTE*)&nHeadBlock0, sizeof(DWORD));
SetLength(nFileSize0);
SeekToBegin();
ASSERT_VALID(this);
return TRUE;
}
BOOL COXResourceFile::SetResourceLibrary(COXResourceLibrary* pResLib,
BOOL bAutoDeleteByLib /* = FALSE */, BOOL bMakeUniqueName /* = FALSE */)
{
if (m_pResLib)
m_pResLib->Unregister(this);
if (pResLib)
{
if (bMakeUniqueName)
VERIFY(MakeUniqueName(pResLib));
pResLib->Register(this);
}
m_pResLib = pResLib;
m_bAutoDeleteByLib = bAutoDeleteByLib;
return TRUE;
}
// protected:
BOOL COXResourceFile::MakeUniqueName(COXResourceLibrary* pSearchLibrary /* = NULL */)
// --- In : pSearchLibrary, the library in which to search for duplicate
// --- Out :
// --- Returns : TRUE if successful; FALSE otherwise
// --- Effect : rename this resource to make sure it has a unique type + name + language
// identifiers
{
if (pSearchLibrary == NULL)
pSearchLibrary = m_pResLib;
ASSERT(pSearchLibrary);
if (pSearchLibrary == NULL)
return FALSE;
WORD nAttempt;
// number attack
nAttempt = OXResCStringToInt(m_sResName);
if (nAttempt)
{
for (WORD nCount = 0; nCount < 0xFFFF; nCount++)
{
if (pSearchLibrary->FindResource(m_sResType, m_sResName, m_nResLanguage) == NULL &&
pSearchLibrary->GetOpenedResFile(m_sResType, m_sResName, m_nResLanguage) == NULL )
return TRUE;
if (nAttempt == 0xFFFF)
nAttempt = 1;
else
nAttempt++;
m_sResName = OXResIntToCString(nAttempt);
}
}
// when number attack fails, try string attack
CString sAttempt = OXResCStringToString(m_sResName);
if (sAttempt.IsEmpty())
sAttempt = OXRES_MUN_STRING_ATTEMPT;
for (int i = sAttempt.GetLength(); i > 0; i--)
{
sAttempt = sAttempt.Left(i);
nAttempt = 1;
CString sNumber;
for (DWORD dwCount = 0; dwCount < 0xFFFFFFFF; dwCount++)
{
sNumber.Format(_T("%d"), ++nAttempt);
m_sResName = sAttempt + sNumber;
if (pSearchLibrary->FindResource(m_sResType, m_sResName, m_nResLanguage) == NULL &&
pSearchLibrary->GetOpenedResFile(m_sResType, m_sResName, m_nResLanguage) == NULL )
return TRUE;
}
}
TRACE0("COXResourceFile::MakeUniqueName(): all attempts failed.\n");
return FALSE;
}
CString COXResourceFile::ValidateResString(LPCTSTR pszTypeOrName)
// --- In : pszTypeOrName, the type or name string that needs to be validated
// --- Out :
// --- Returns : the result type or name in CString format
// --- Effect : this function will check the validity of a type or name parameter,
// and format it to make sure the resulted string will work with resource
// functions. Specifically:
// (1) all lower letters will be made upper;
// (2) all leading '#' will be replaced by '_', if no numbers follow it;
// (3) all white space will be replaced by '_'.
{
WORD nID = OXResToInt(pszTypeOrName);
if (nID)
return OXResIntToCString(nID);
CString sResult = pszTypeOrName;
int nLen = sResult.GetLength();
sResult.MakeUpper();
int i = 0;
while (i < nLen && sResult[i] == _T('#'))
sResult.SetAt(i++, _T('_'));
while (i < nLen)
{
switch (sResult[i])
{
case _T(' '):
case _T('\t'):
case _T('\n'):
sResult.SetAt(i, _T('_'));
}
i++;
}
return sResult;
}
// type/name format conversion functions
CString COXResourceFile::OXResIntToCString(WORD nID)
{
CString sResult;
sResult.Format(_T("#%u"), nID);
return sResult;
}
WORD COXResourceFile::OXResToInt(LPCTSTR lpszTypeOrName)
{
WORD nID = OXResItemToInt(lpszTypeOrName);
if (nID)
return nID;
return OXResCStringToInt(lpszTypeOrName);
}
CString COXResourceFile::OXResToCString(LPCTSTR lpszTypeOrName)
{
WORD nID = OXResToInt(lpszTypeOrName);
if (nID)
return OXResIntToCString(nID);
else
return lpszTypeOrName;
}
// end of OXResourceFile.cpp