/////////////////////////////////////////////////////////////////////////////// /* File: strclass.cpp Description: Typical class to handle strings. Revision History: Date Description Programmer -------- --------------------------------------------------- ---------- 07/01/97 Initial creation. BrianAu */ /////////////////////////////////////////////////////////////////////////////// #include "pch.h" #pragma hdrstop #include "strclass.h" #ifdef StrCpy # undef StrCpy #endif #ifdef StrCpyN # undef StrCpyN #endif #ifdef StrLen # undef StrLen #endif #ifdef UNICODE # define StrCpy StrCpyW # define StrCpyN StrCpyNW # define StrLen StrLenW #else # define StrCpy StrCpyA # define StrCpyN StrCpyNA # define StrLen StrLenA #endif // UNICODE const INT MAX_RESOURCE_STR_LEN = 4097; // // Disable "'operator ->' is not a UDT or reference to a UDT" warning. // This is caused when we create an autoptr to a non-UDT. It's meaningless // since there's no reason to call operator-> on a non-UDT autoptr. // #pragma warning (disable : 4284) CString::CString( VOID ) : m_pValue(new StringValue()) { } CString::CString( INT cch ) : m_pValue(new StringValue(cch)) { } CString::CString( LPCSTR pszA ) : m_pValue(new StringValue(pszA)) { } CString::CString( LPCWSTR pszW ) : m_pValue(new StringValue(pszW)) { } CString::CString( const CString& rhs ) : m_pValue(rhs.m_pValue) { InterlockedIncrement(&(m_pValue->m_cRef)); } CString::CString( HINSTANCE hInst, INT idMsg, ... ) : m_pValue(NULL) { LPTSTR pszMsg = NULL; va_list args; va_start(args, idMsg); Format(hInst, idMsg, &args); va_end(args); } CString::~CString( VOID ) { if (NULL != m_pValue) { ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } } } // // Length of string, excluding nul terminator. // INT CString::Length( VOID ) const { return m_pValue->Length(); } VOID CString::Empty( VOID ) { ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } m_pValue = NULL; m_pValue = new StringValue(); } BOOL CString::IsEmpty( VOID ) const { return (NULL != m_pValue && 0 == m_pValue->Length()); } CString& CString::operator = ( const CString& rhs ) { if (m_pValue != rhs.m_pValue) // Chk for assignment to *this. { ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } m_pValue = rhs.m_pValue; InterlockedIncrement(&(m_pValue->m_cRef)); } return *this; } CString& CString::operator = ( LPCWSTR rhsW ) { ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } // // The StringValue ctor can throw an allocation exception. // Make sure the m_pValue variable is in a consistent state // beforehand. // m_pValue = NULL; m_pValue = new StringValue(rhsW); return *this; } CString& CString::operator = ( LPCSTR rhsA ) { ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } // // The StringValue ctor can throw an allocation exception. // Make sure the m_pValue variable is in a consistent state // beforehand. // m_pValue = NULL; m_pValue = new StringValue(rhsA); return *this; } CString CString::operator + ( const CString& rhs ) const { CString strNew; LPTSTR pszTemp = NULL; try { pszTemp = StringValue::Concat(m_pValue, rhs.m_pValue); strNew = pszTemp; } catch(...) { delete[] pszTemp; throw; } delete[] pszTemp; return strNew; } CString& CString::operator += ( const CString& rhs ) { LPTSTR pszTemp = NULL; try { pszTemp = StringValue::Concat(m_pValue, rhs.m_pValue); *this = pszTemp; } catch(...) { delete[] pszTemp; throw; } delete[] pszTemp; return *this; } BOOL CString::operator == ( const CString& rhs ) const { return (0 == lstrcmp(m_pValue->m_psz, rhs.m_pValue->m_psz)); } INT CString::Compare( LPCWSTR rhsW ) const { StringValue Value(rhsW); return lstrcmp(m_pValue->m_psz, Value.m_psz); } INT CString::Compare( LPCSTR rhsA ) const { StringValue Value(rhsA); return lstrcmp(m_pValue->m_psz, Value.m_psz); } INT CString::CompareNoCase( LPCWSTR rhsW ) const { StringValue Value(rhsW); return lstrcmpi(m_pValue->m_psz, Value.m_psz); } INT CString::CompareNoCase( LPCSTR rhsA ) const { StringValue Value(rhsA); return lstrcmpi(m_pValue->m_psz, Value.m_psz); } BOOL CString::operator < ( const CString& rhs ) const { return (0 > lstrcmp(m_pValue->m_psz, rhs.m_pValue->m_psz)); } TCHAR CString::operator[]( INT index ) const { if (!ValidIndex(index)) throw CMemoryException(CMemoryException::index); return m_pValue->m_psz[index]; } TCHAR& CString::operator[]( INT index ) { if (!ValidIndex(index)) throw CMemoryException(CMemoryException::index); CopyOnWrite(); return m_pValue->m_psz[index]; } INT CString::First( TCHAR ch ) const { LPCTSTR psz = m_pValue->m_psz; LPCTSTR pszLast = psz; INT i = 0; while(psz && *psz) { if (ch == *psz) return i; psz = CharNext(psz); i += (INT)(psz - pszLast); pszLast = psz; } return -1; } INT CString::Last( TCHAR ch ) const { INT iLast = -1; INT i = 0; LPCTSTR psz = m_pValue->m_psz; LPCTSTR pszPrev = psz; while(psz && *psz) { if (ch == *psz) iLast = i; psz = CharNext(psz); i += (INT)(psz - pszPrev); pszPrev = psz; } return iLast; } CString CString::SubString( INT iFirst, INT cch ) { if (!ValidIndex(iFirst)) throw CMemoryException(CMemoryException::index); INT cchToEnd = Length() - iFirst; if (-1 == cch || cch > cchToEnd) return CString(m_pValue->m_psz + iFirst); LPTSTR pszTemp = new TCHAR[cch + 1]; if (NULL == pszTemp) throw CAllocException(); CString::StrCpyN(pszTemp, m_pValue->m_psz + iFirst, cch + 1); CString strTemp(pszTemp); delete[] pszTemp; return strTemp; } VOID CString::ToUpper( INT iFirst, INT cch ) { if (!ValidIndex(iFirst)) throw CMemoryException(CMemoryException::index); CopyOnWrite(); INT cchToEnd = Length() - iFirst; if (-1 == cch || cch > cchToEnd) cch = cchToEnd; CharUpperBuff(m_pValue->m_psz + iFirst, cch); } VOID CString::ToLower( INT iFirst, INT cch ) { if (!ValidIndex(iFirst)) throw CMemoryException(CMemoryException::index); CopyOnWrite(); INT cchToEnd = Length() - iFirst; if (-1 == cch || cch > cchToEnd) cch = cchToEnd; CharLowerBuff(m_pValue->m_psz + iFirst, cch); } VOID CString::Size( INT cch ) { StringValue *m_psv = new StringValue(cch + 1); CString::StrCpyN(m_psv->m_psz, m_pValue->m_psz, cch); ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&m_pValue->m_cRef)) { delete m_pValue; } m_pValue = m_psv; } VOID CString::CopyOnWrite( VOID ) { // // Only need to copy if ref cnt > 1. // if (m_pValue->m_cRef > 1) { StringValue * pValue = new StringValue(m_pValue->m_psz); ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } m_pValue = pValue; } } BOOL CString::Format( LPCTSTR pszFmt, ... ) { BOOL bResult; va_list args; va_start(args, pszFmt); bResult = Format(pszFmt, &args); va_end(args); return bResult; } BOOL CString::Format( LPCTSTR pszFmt, va_list *pargs ) { BOOL bResult = FALSE; TCHAR szBuffer[MAX_RESOURCE_STR_LEN]; INT cchLoaded; cchLoaded = ::FormatMessage(FORMAT_MESSAGE_FROM_STRING, pszFmt, 0, 0, szBuffer, ARRAYSIZE(szBuffer), pargs); if (0 < cchLoaded) { ASSERT( NULL == m_pValue || 0 != m_pValue->m_cRef ); if (NULL != m_pValue && 0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } m_pValue = NULL; m_pValue = new StringValue(szBuffer); bResult = TRUE; } else { DWORD dwLastError = GetLastError(); if (ERROR_SUCCESS != dwLastError) { DBGERROR((TEXT("CString::Format failed with error 0x%08X"), dwLastError)); throw CResourceException(CResourceException::string, NULL, 0); } } return bResult; } BOOL CString::Format( HINSTANCE hInst, UINT idFmt, ... ) { BOOL bResult; va_list args; va_start(args, idFmt); bResult = Format(hInst, idFmt, &args); va_end(args); return bResult; } BOOL CString::Format( HINSTANCE hInst, UINT idFmt, va_list *pargs ) { BOOL bResult = FALSE; TCHAR szFmtStr[MAX_RESOURCE_STR_LEN]; // Buffer for format string (if needed). INT cchLoaded; // // Try to load the format string as a string resource. // cchLoaded = ::LoadString(hInst, idFmt, szFmtStr, ARRAYSIZE(szFmtStr)); if (0 < cchLoaded) { // // The format string was in a string resource. // Now format it with the arg list. // bResult = Format(szFmtStr, pargs); } else { TCHAR szBuffer[MAX_RESOURCE_STR_LEN]; // // The format string may be in a message resource. // Note that if it is, the resulting formatted string will // be automatically attached to m_psz by ::FormatMessage. // cchLoaded = ::FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_FROM_HMODULE, hInst, (DWORD)idFmt, LANGIDFROMLCID(GetThreadLocale()), (LPTSTR)szBuffer, ARRAYSIZE(szBuffer), pargs); if (0 < cchLoaded) { ASSERT( NULL == m_pValue || 0 != m_pValue->m_cRef ); if (NULL != m_pValue && 0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } m_pValue = NULL; m_pValue = new StringValue(szBuffer); bResult = TRUE; } else { DWORD dwLastError = GetLastError(); if (ERROR_SUCCESS != dwLastError) { DBGERROR((TEXT("CString::Format failed with error 0x%08X"), dwLastError)); throw CResourceException(CResourceException::string, hInst, idFmt); } } } return bResult; } LPTSTR CString::GetBuffer( INT cchMax ) { if (-1 == cchMax) cchMax = m_pValue->m_cchAlloc; CopyOnWrite(); if (cchMax > m_pValue->m_cchAlloc) { // // Extend the buffer, copying original contents to dest. // StringValue *pv = new StringValue(cchMax); StrCpyN(pv->m_psz, m_pValue->m_psz, cchMax); ASSERT( 0 != m_pValue->m_cRef ); if (0 == InterlockedDecrement(&(m_pValue->m_cRef))) { delete m_pValue; } m_pValue = pv; LPTSTR pszEnd = m_pValue->m_psz + m_pValue->m_cchAlloc - 1; if (pszEnd >= m_pValue->m_psz && TEXT('\0') != *(pszEnd)) { // // Ensure it's nul terminated. // *(pszEnd) = TEXT('\0'); } } return m_pValue->m_psz; } VOID CString::ReleaseBuffer( void ) { // // Update the string length member after client has had free access // to internal buffer. // m_pValue->m_cch = StrLen(m_pValue->m_psz); } void CString::Rtrim( void ) { LPTSTR s = GetBuffer(); int len = Length(); while(0 < --len && IsWhiteSpace(s[len])) s[len] = TEXT('\0'); ReleaseBuffer(); } void CString::Ltrim( void ) { LPTSTR s0; LPTSTR s = s0 = GetBuffer(); while(*s && IsWhiteSpace(*s)) s++; while(*s) *s0++ = *s++; *s0 = TEXT('\0'); ReleaseBuffer(); } VOID CString::ExpandEnvironmentStrings( VOID ) { DWORD cchBuffer = 0; // Size of expansion buffer. DWORD cchPath = 0; // Count of actual chars in expanded buffer. CString strExpanded; // Expansion buffer. // // If necessary, keep increasing expansion buffer size until entire // expanded string fits. // do { cchBuffer += MAX_PATH; cchPath = ::ExpandEnvironmentStrings(*this, strExpanded.GetBuffer(cchBuffer), cchBuffer); } while(0 != cchPath && cchPath > cchBuffer); ReleaseBuffer(); *this = strExpanded; } bool CString::GetDisplayRect( HDC hdc, LPRECT prc ) const { return (0 != DrawText(hdc, Cstr(), Length(), prc, DT_CALCRECT)); } VOID CString::DebugOut( BOOL bNewline ) const { OutputDebugString(m_pValue->m_psz); if (bNewline) OutputDebugString(TEXT("\n\r")); } CString::StringValue::StringValue( VOID ) : m_psz(new TCHAR[1]), m_cchAlloc(1), m_cch(0), m_cRef(1) { *m_psz = TEXT('\0'); } CString::StringValue::StringValue( LPCSTR pszA ) : m_psz(NULL), m_cchAlloc(0), m_cch(0), m_cRef(1) { #ifdef UNICODE m_psz = AnsiToWide(pszA, &m_cchAlloc); m_cch = StrLenW(m_psz); #else m_cch = CString::StrLenA(pszA); m_psz = Dup(pszA, m_cch + 1); m_cchAlloc = m_cch + 1; #endif } CString::StringValue::StringValue( LPCWSTR pszW ) : m_psz(NULL), m_cchAlloc(0), m_cch(0), m_cRef(1) { #ifdef UNICODE m_cch = CString::StrLenW(pszW); m_psz = Dup(pszW, m_cch + 1); m_cchAlloc = m_cch + 1; #else m_psz = WideToAnsi(pszW, &m_cchAlloc); m_cch = StrLenA(m_psz); #endif } CString::StringValue::StringValue( INT cch ) : m_psz(NULL), m_cchAlloc(0), m_cch(0), m_cRef(0) { m_psz = Dup(TEXT(""), cch); m_cRef = 1; m_cchAlloc = cch; } CString::StringValue::~StringValue( VOID ) { delete[] m_psz; } LPWSTR CString::StringValue::AnsiToWide( LPCSTR pszA, INT *pcch ) { INT cchW = 0; LPWSTR pszW = NULL; cchW = MultiByteToWideChar(CP_ACP, 0, pszA, -1, NULL, 0); pszW = new WCHAR[cchW]; if (NULL == pszW) throw CAllocException(); MultiByteToWideChar(CP_ACP, 0, pszA, -1, pszW, cchW); if (NULL != pcch) *pcch = cchW; return pszW; } LPSTR CString::StringValue::WideToAnsi( LPCWSTR pszW, INT *pcch ) { INT cchA = 0; LPSTR pszA = NULL; cchA = WideCharToMultiByte(CP_ACP, 0, pszW, -1, NULL, 0, NULL, NULL); pszA = new CHAR[cchA]; if (NULL == pszA) throw CAllocException(); WideCharToMultiByte(CP_ACP, 0, pszW, -1, pszA, cchA, NULL, NULL); if (NULL != pcch) *pcch = cchA; return pszA; } INT CString::StringValue::Length( VOID ) const { if (0 == m_cch && NULL != m_psz) { m_cch = StrLen(m_psz); } return m_cch; } LPWSTR CString::StringValue::Dup( LPCWSTR pszW, INT cch ) { if (0 == cch) cch = CString::StrLenW(pszW) + 1; LPWSTR pszNew = new WCHAR[cch]; if (NULL == pszNew) throw CAllocException(); lstrcpynW(pszNew, pszW, cch); return pszNew; } LPSTR CString::StringValue::Dup( LPCSTR pszA, INT cch ) { if (0 == cch) cch = CString::StrLenA(pszA) + 1; LPSTR pszNew = new CHAR[cch]; if (NULL == pszNew) throw CAllocException(); lstrcpynA(pszNew, pszA, cch); return pszNew; } LPTSTR CString::StringValue::Concat( CString::StringValue *psv1, CString::StringValue *psv2 ) { LPTSTR pszTemp = NULL; INT len1 = psv1->Length(); INT len2 = psv2->Length(); const INT cch = len1 + len2 + 1; pszTemp = new TCHAR[cch]; if (NULL == pszTemp) throw CAllocException(); wnsprintf(pszTemp, cch, TEXT("%s%s"), psv1->m_psz, psv2->m_psz); return pszTemp; } #pragma warning (default : 4284)