/* Miranda NG: the free IM client for Microsoft* Windows* Copyright (c) 2012-14 Miranda NG project (http://miranda-ng.org) all portions of this codebase are copyrighted to the people listed in contributors.txt. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #pragma once #ifndef M_STRING_H__ #include <stdio.h> #include <string.h> #include <mbstring.h> #include <wchar.h> #include <m_core.h> #ifdef __MINGW32__ #include <limits.h> __inline size_t strnlen(const char *string, size_t maxlen) { const char *end = (const char *)memchr ((const void *)string, '\0', maxlen); return end ? (size_t) (end - string) : maxlen; } __inline size_t wcsnlen(const wchar_t *string, size_t maxlen) { const wchar_t *end = wmemchr (string, L'\0', maxlen); return end ? (size_t) (end - string) : maxlen; } /* FIXME: This may be wrong assumption about Langpack_GetDefaultCodePage */ #define Langpack_GetDefaultCodePage() CP_THREAD_ACP /* FIXME: This is unsafe */ #define memcpy_s(dest,size,src,count) memcpy(dest,src,count) /* FIXME: This is quite silly implementation of _mbsstr */ #define _mbsstr(str,search) strstr((const char *)str,(const char *)search) #define __max(x,y) (((x)<(y))?(y):(x)) #endif /* __MINGW32__ */ ///////////////////////////////////////////////////////////////////////////////////////// struct CMStringData; MIR_CORE_DLL(CMStringData*) mirstr_allocate(int nChars, int nCharSize); MIR_CORE_DLL(void) mirstr_free(CMStringData *pData); MIR_CORE_DLL(CMStringData*) mirstr_realloc(CMStringData* pData, int nChars, int nCharSize); MIR_CORE_DLL(CMStringData*) mirstr_getNil(); MIR_CORE_DLL(void) mirstr_lock(CMStringData* pThis); MIR_CORE_DLL(void) mirstr_release(CMStringData* pThis); MIR_CORE_DLL(void) mirstr_unlock(CMStringData* pThis); ///////////////////////////////////////////////////////////////////////////////////////// enum CMStringDataFormat { FORMAT }; struct CMStringData { int nDataLength; // Length of currently used data in XCHARs (not including terminating null) int nAllocLength; // Length of allocated data in XCHARs (not including terminating null) long nRefs; // Reference count: negative == locked __forceinline void* data() { return (this + 1); } __forceinline void AddRef() { InterlockedIncrement(&nRefs); } __forceinline bool IsLocked() const { return nRefs < 0; } __forceinline bool IsShared() const { return nRefs > 1; } __forceinline void Lock() { mirstr_lock(this); } __forceinline void Release() { mirstr_release(this); } __forceinline void Unlock() { mirstr_unlock(this); } }; template< typename BaseType = char > class ChTraitsBase { public: typedef char XCHAR; typedef LPSTR PXSTR; typedef LPCSTR PCXSTR; typedef wchar_t YCHAR; typedef LPWSTR PYSTR; typedef LPCWSTR PCYSTR; }; template<> class ChTraitsBase< wchar_t > { public: typedef wchar_t XCHAR; typedef LPWSTR PXSTR; typedef LPCWSTR PCXSTR; typedef char YCHAR; typedef LPSTR PYSTR; typedef LPCSTR PCYSTR; }; template< typename BaseType > class CMSimpleStringT { public: typedef typename ChTraitsBase< BaseType >::XCHAR XCHAR; typedef typename ChTraitsBase< BaseType >::PXSTR PXSTR; typedef typename ChTraitsBase< BaseType >::PCXSTR PCXSTR; typedef typename ChTraitsBase< BaseType >::YCHAR YCHAR; typedef typename ChTraitsBase< BaseType >::PYSTR PYSTR; typedef typename ChTraitsBase< BaseType >::PCYSTR PCYSTR; public: explicit CMSimpleStringT() { CMStringData* pData = mirstr_getNil(); Attach(pData); } CMSimpleStringT(const CMSimpleStringT& strSrc) { CMStringData* pSrcData = strSrc.GetData(); CMStringData* pNewData = CloneData(pSrcData); Attach(pNewData); } CMSimpleStringT(PCXSTR pszSrc) { int nLength = StringLength(pszSrc); CMStringData* pData = mirstr_allocate(nLength, sizeof(XCHAR)); if (pData != NULL) { Attach(pData); SetLength(nLength); CopyChars(m_pszData, nLength, pszSrc, nLength); } } CMSimpleStringT(const XCHAR* pchSrc, int nLength) { CMStringData* pData = mirstr_allocate(nLength, sizeof(XCHAR)); if (pData != NULL) { Attach(pData); SetLength(nLength); CopyChars(m_pszData, nLength, pchSrc, nLength); } } ~CMSimpleStringT() { CMStringData* pData = GetData(); pData->Release(); } operator CMSimpleStringT<BaseType>&() { return *(CMSimpleStringT<BaseType>*)this; } CMSimpleStringT& operator=(const CMSimpleStringT& strSrc) { CMStringData* pSrcData = strSrc.GetData(); CMStringData* pOldData = GetData(); if (pSrcData != pOldData) { if (pOldData->IsLocked()) SetString(strSrc.GetString(), strSrc.GetLength()); else { CMStringData* pNewData = CloneData(pSrcData); pOldData->Release(); Attach(pNewData); } } return *this; } CMSimpleStringT& operator=(PCXSTR pszSrc) { SetString(pszSrc); return *this; } CMSimpleStringT& operator+=(const CMSimpleStringT& strSrc) { Append(strSrc); return *this; } CMSimpleStringT& operator+=(PCXSTR pszSrc) { Append(pszSrc); return *this; } CMSimpleStringT& operator+=(char ch) { AppendChar(XCHAR(ch)); return *this; } CMSimpleStringT& operator+=(unsigned char ch) { AppendChar(XCHAR(ch)); return *this; } CMSimpleStringT& operator+=(wchar_t ch) { AppendChar(XCHAR(ch)); return *this; } XCHAR operator[](int iChar) const { return m_pszData[iChar]; } operator PCXSTR() const { return m_pszData; } PCXSTR c_str() const { return m_pszData; } void Append(PCXSTR pszSrc) { Append(pszSrc, StringLength(pszSrc)); } void Append(PCXSTR pszSrc, int nLength) { // See comment in SetString() about why we do this UINT_PTR nOffset = UINT_PTR(pszSrc - GetString()); int nOldLength = GetLength(); if (nOldLength < 0) { // protects from underflow nOldLength = 0; } //Make sure we don't read pass end of the terminating NULL int nSrcLength = StringLength(pszSrc); nLength = nLength > nSrcLength ? nSrcLength : nLength; int nNewLength = nOldLength + nLength; PXSTR pszBuffer = GetBuffer(nNewLength); if (nOffset <= UINT_PTR(nOldLength)) { pszSrc = pszBuffer + nOffset; // No need to call CopyCharsOverlapped, since the destination is // beyond the end of the original buffer } CopyChars(pszBuffer + nOldLength, nLength, pszSrc, nLength); ReleaseBufferSetLength(nNewLength); } void AppendChar(XCHAR ch) { UINT nOldLength = GetLength(); int nNewLength = nOldLength + 1; PXSTR pszBuffer = GetBuffer(nNewLength); pszBuffer[nOldLength] = ch; ReleaseBufferSetLength(nNewLength); } void Append(const CMSimpleStringT& strSrc) { Append(strSrc.GetString(), strSrc.GetLength()); } void Empty() { CMStringData* pOldData = GetData(); if (pOldData->nDataLength == 0) return; if (pOldData->IsLocked()) { // Don't reallocate a locked buffer that's shrinking SetLength(0); } else { pOldData->Release(); CMStringData* pNewData = mirstr_getNil(); Attach(pNewData); } } void FreeExtra() { CMStringData* pOldData = GetData(); int nLength = pOldData->nDataLength; if (pOldData->nAllocLength == nLength) return; if (!pOldData->IsLocked()) { // Don't reallocate a locked buffer that's shrinking CMStringData* pNewData = mirstr_allocate(nLength, sizeof(XCHAR)); if (pNewData == NULL) { SetLength(nLength); return; } CopyChars(PXSTR(pNewData->data()), nLength, PCXSTR(pOldData->data()), nLength); pOldData->Release(); Attach(pNewData); SetLength(nLength); } } int GetAllocLength() const { return GetData()->nAllocLength; } XCHAR GetAt(int iChar) const { return m_pszData[iChar]; } PXSTR GetBuffer() { CMStringData* pData = GetData(); if (pData->IsShared()) Fork(pData->nDataLength); return m_pszData; } PXSTR GetBuffer(int nMinBufferLength) { return PrepareWrite(nMinBufferLength); } PXSTR GetBufferSetLength(int nLength) { PXSTR pszBuffer = GetBuffer(nLength); SetLength(nLength); return pszBuffer; } int GetLength() const { return GetData()->nDataLength; } PCXSTR GetString() const { return m_pszData; } PCXSTR GetTail() const { return m_pszData + GetData()->nDataLength; } bool IsEmpty() const { return GetLength() == 0; } PXSTR LockBuffer() { CMStringData* pData = GetData(); if (pData->IsShared()) { Fork(pData->nDataLength); pData = GetData(); // Do it again, because the fork might have changed it } pData->Lock(); return m_pszData; } void UnlockBuffer() { CMStringData* pData = GetData(); pData->Unlock(); } void Preallocate(int nLength) { PrepareWrite(nLength); } void ReleaseBuffer(int nNewLength = -1) { if (nNewLength == -1) { int nAlloc = GetData()->nAllocLength; nNewLength = StringLengthN(m_pszData, nAlloc); } SetLength(nNewLength); } void ReleaseBufferSetLength(int nNewLength) { SetLength(nNewLength); } void Truncate(int nNewLength) { GetBuffer(nNewLength); ReleaseBufferSetLength(nNewLength); } void SetAt(int iChar, XCHAR ch) { int nLength = GetLength(); PXSTR pszBuffer = GetBuffer(); pszBuffer[iChar] = ch; ReleaseBufferSetLength(nLength); } void SetString(PCXSTR pszSrc) { SetString(pszSrc, StringLength(pszSrc)); } void SetString(PCXSTR pszSrc, int nLength) { if (nLength == 0) Empty(); else { UINT nOldLength = GetLength(); UINT_PTR nOffset = pszSrc - GetString(); PXSTR pszBuffer = GetBuffer(nLength); if (nOffset <= nOldLength) CopyCharsOverlapped(pszBuffer, GetAllocLength(), pszBuffer + nOffset, nLength); else CopyChars(pszBuffer, GetAllocLength(), pszSrc, nLength); ReleaseBufferSetLength(nLength); } } public: friend CMSimpleStringT __stdcall operator+(const CMSimpleStringT& str1, const CMSimpleStringT& str2) { CMSimpleStringT s; Concatenate(s, str1, str1.GetLength(), str2, str2.GetLength()); return s; } friend CMSimpleStringT __stdcall operator+(const CMSimpleStringT& str1, PCXSTR psz2) { CMSimpleStringT s; Concatenate(s, str1, str1.GetLength(), psz2, StringLength(psz2)); return s; } friend CMSimpleStringT __stdcall operator+(PCXSTR psz1, const CMSimpleStringT& str2) { CMSimpleStringT s; Concatenate(s, psz1, StringLength(psz1), str2, str2.GetLength()); return s; } static void __stdcall CopyChars(XCHAR* pchDest, const XCHAR* pchSrc, int nChars) { #pragma warning (push) #pragma warning(disable : 4996) memcpy(pchDest, pchSrc, nChars * sizeof(XCHAR)); #pragma warning (pop) } static void __stdcall CopyChars(XCHAR* pchDest, size_t nDestLen, const XCHAR* pchSrc, int nChars) { memcpy_s(pchDest, nDestLen * sizeof(XCHAR), pchSrc, nChars * sizeof(XCHAR)); } static void __stdcall CopyCharsOverlapped(XCHAR* pchDest, const XCHAR* pchSrc, int nChars) { #pragma warning (push) #pragma warning(disable : 4996) memmove(pchDest, pchSrc, nChars * sizeof(XCHAR)); #pragma warning (pop) } static void __stdcall CopyCharsOverlapped(XCHAR* pchDest, size_t nDestLen, const XCHAR* pchSrc, int nChars) { memmove_s(pchDest, nDestLen * sizeof(XCHAR), pchSrc, nChars * sizeof(XCHAR)); } static int __stdcall StringLength(const char* psz) { if (psz == NULL) return(0); return (int(strlen(psz))); } static int __stdcall StringLength(const wchar_t* psz) { if (psz == NULL) return 0; return int(wcslen(psz)); } static int __stdcall StringLengthN(const char* psz, size_t sizeInXChar) { if (psz == NULL) return 0; return int(strnlen(psz, sizeInXChar)); } static int __stdcall StringLengthN(const wchar_t* psz, size_t sizeInXChar) { if (psz == NULL) return 0; return int(wcsnlen(psz, sizeInXChar)); } protected: static void __stdcall Concatenate(CMSimpleStringT& strResult, PCXSTR psz1, int nLength1, PCXSTR psz2, int nLength2) { int nNewLength = nLength1+nLength2; PXSTR pszBuffer = strResult.GetBuffer(nNewLength); CopyChars(pszBuffer, nLength1, psz1, nLength1); CopyChars(pszBuffer + nLength1, nLength2, psz2, nLength2); strResult.ReleaseBufferSetLength(nNewLength); } // Implementation private: void Attach(CMStringData* pData) { m_pszData = static_cast<PXSTR>(pData->data()); } void Fork(int nLength) { CMStringData* pOldData = GetData(); int nOldLength = pOldData->nDataLength; CMStringData* pNewData = mirstr_allocate(nLength, sizeof(XCHAR)); if (pNewData != NULL) { int nCharsToCopy = ((nOldLength < nLength) ? nOldLength : nLength)+1; // Copy '\0' CopyChars(PXSTR(pNewData->data()), nCharsToCopy, PCXSTR(pOldData->data()), nCharsToCopy); pNewData->nDataLength = nOldLength; pOldData->Release(); Attach(pNewData); } } CMStringData* GetData() const { return (reinterpret_cast<CMStringData *>(m_pszData) - 1); } PXSTR PrepareWrite(int nLength) { CMStringData* pOldData = GetData(); int nShared = 1 - pOldData->nRefs; // nShared < 0 means true, >= 0 means false int nTooShort = pOldData->nAllocLength-nLength; // nTooShort < 0 means true, >= 0 means false if ((nShared | nTooShort) < 0) // If either sign bit is set (i.e. either is less than zero), we need to copy data PrepareWrite2(nLength); return m_pszData; } void PrepareWrite2(int nLength) { CMStringData* pOldData = GetData(); if (pOldData->nDataLength > nLength) nLength = pOldData->nDataLength; if (pOldData->IsShared()) { Fork(nLength); } else if (pOldData->nAllocLength < nLength) { // Grow exponentially, until we hit 1K. int nNewLength = pOldData->nAllocLength; if (nNewLength > 1024) nNewLength += 1024; else nNewLength *= 2; if (nNewLength < nLength) nNewLength = nLength; Reallocate(nNewLength); } } void Reallocate(int nLength) { CMStringData* pOldData = GetData(); if (pOldData->nAllocLength >= nLength || nLength <= 0) return; CMStringData* pNewData = mirstr_realloc(pOldData, nLength, sizeof(XCHAR)); if (pNewData != NULL) Attach(pNewData); } void SetLength(int nLength) { GetData()->nDataLength = nLength; m_pszData[nLength] = 0; } static CMStringData* __stdcall CloneData(CMStringData* pData) { CMStringData* pNewData = NULL; if (!pData->IsLocked()) { pNewData = pData; pNewData->AddRef(); } return pNewData; } public : // typedef CStrBufT<BaseType> CStrBuf; private: PXSTR m_pszData; }; template< typename _CharType = char > class ChTraitsCRT : public ChTraitsBase< _CharType > { public: static char* __stdcall CharNext(const char* p) { return reinterpret_cast< char* >(_mbsinc(reinterpret_cast< const unsigned char* >(p))); } static int __stdcall IsDigit(char ch) { return _ismbcdigit(ch); } static int __stdcall IsSpace(char ch) { return _ismbcspace(ch); } static int __stdcall StringCompare(LPCSTR pszA, LPCSTR pszB) { return _mbscmp(reinterpret_cast< const unsigned char* >(pszA), reinterpret_cast< const unsigned char* >(pszB)); } static int __stdcall StringCompareIgnore(LPCSTR pszA, LPCSTR pszB) { return _mbsicmp(reinterpret_cast< const unsigned char* >(pszA), reinterpret_cast< const unsigned char* >(pszB)); } static int __stdcall StringCollate(LPCSTR pszA, LPCSTR pszB) { return _mbscoll(reinterpret_cast< const unsigned char* >(pszA), reinterpret_cast< const unsigned char* >(pszB)); } static int __stdcall StringCollateIgnore(LPCSTR pszA, LPCSTR pszB) { return _mbsicoll(reinterpret_cast< const unsigned char* >(pszA), reinterpret_cast< const unsigned char* >(pszB)); } static LPCSTR __stdcall StringFindString(LPCSTR pszBlock, LPCSTR pszMatch) { return reinterpret_cast< LPCSTR >(_mbsstr(reinterpret_cast< const unsigned char* >(pszBlock), reinterpret_cast< const unsigned char* >(pszMatch))); } static LPSTR __stdcall StringFindString(LPSTR pszBlock, LPCSTR pszMatch) { return const_cast< LPSTR >(StringFindString(const_cast< LPCSTR >(pszBlock), pszMatch)); } static LPCSTR __stdcall StringFindChar(LPCSTR pszBlock, char chMatch) { return reinterpret_cast< LPCSTR >(_mbschr(reinterpret_cast< const unsigned char* >(pszBlock), (unsigned char)chMatch)); } static LPCSTR __stdcall StringFindCharRev(LPCSTR psz, char ch) { return reinterpret_cast< LPCSTR >(_mbsrchr(reinterpret_cast< const unsigned char* >(psz), (unsigned char)ch)); } static LPCSTR __stdcall StringScanSet(LPCSTR pszBlock, LPCSTR pszMatch) { return reinterpret_cast< LPCSTR >(_mbspbrk(reinterpret_cast< const unsigned char* >(pszBlock), reinterpret_cast< const unsigned char* >(pszMatch))); } static int __stdcall StringSpanIncluding(LPCSTR pszBlock, LPCSTR pszSet) { return (int)_mbsspn(reinterpret_cast< const unsigned char* >(pszBlock), reinterpret_cast< const unsigned char* >(pszSet)); } static int __stdcall StringSpanExcluding(LPCSTR pszBlock, LPCSTR pszSet) { return (int)_mbscspn(reinterpret_cast< const unsigned char* >(pszBlock), reinterpret_cast< const unsigned char* >(pszSet)); } static LPSTR __stdcall StringUppercase(LPSTR psz) { CharUpperBuffA(psz, (DWORD)strlen(psz)); return psz; } static LPSTR __stdcall StringLowercase(LPSTR psz) { CharLowerBuffA(psz, (DWORD)strlen(psz)); return psz; } static LPSTR __stdcall StringUppercase(LPSTR psz, size_t size) { CharUpperBuffA(psz, (DWORD)size); return psz; } static LPSTR __stdcall StringLowercase(LPSTR psz, size_t size) { CharLowerBuffA(psz, (DWORD)size); return psz; } static LPSTR __stdcall StringReverse(LPSTR psz) { return reinterpret_cast< LPSTR >(_mbsrev(reinterpret_cast< unsigned char* >(psz))); } static int __stdcall GetFormattedLength(LPCSTR pszFormat, va_list args); static int __stdcall Format(LPSTR pszBuffer, size_t nlength, LPCSTR pszFormat, va_list args); static int __stdcall GetBaseTypeLength(LPCSTR pszSrc) { // Returns required buffer length in XCHARs return int(strlen(pszSrc)); } static int __stdcall GetBaseTypeLength(LPCSTR pszSrc, int nLength) { (void)pszSrc; // Returns required buffer length in XCHARs return nLength; } static int __stdcall GetBaseTypeLength(LPCWSTR pszSource) { // Returns required buffer length in XCHARs return ::WideCharToMultiByte(Langpack_GetDefaultCodePage(), 0, pszSource, -1, NULL, 0, NULL, NULL)-1; } static int __stdcall GetBaseTypeLength(LPCWSTR pszSource, int nLength) { // Returns required buffer length in XCHARs return ::WideCharToMultiByte(Langpack_GetDefaultCodePage(), 0, pszSource, nLength, NULL, 0, NULL, NULL); } static void __stdcall ConvertToBaseType(LPSTR pszDest, int nDestLength, LPCSTR pszSrc, int nSrcLength = -1) { if (nSrcLength == -1) { nSrcLength=1 + GetBaseTypeLength(pszSrc); } // nLen is in XCHARs memcpy_s(pszDest, nDestLength*sizeof(char), pszSrc, nSrcLength*sizeof(char)); } static void __stdcall ConvertToBaseType(LPSTR pszDest, int nDestLength, LPCWSTR pszSrc, int nSrcLength = -1) { // nLen is in XCHARs ::WideCharToMultiByte(Langpack_GetDefaultCodePage(), 0, pszSrc, nSrcLength, pszDest, nDestLength, NULL, NULL); } static void ConvertToOem(_CharType* pstrString) { BOOL fSuccess=::CharToOemA(pstrString, pstrString); } static void ConvertToAnsi(_CharType* pstrString) { BOOL fSuccess=::OemToCharA(pstrString, pstrString); } static void ConvertToOem(_CharType* pstrString, size_t size) { if (size>UINT_MAX) { return; } DWORD dwSize=static_cast<DWORD>(size); BOOL fSuccess=::CharToOemBuffA(pstrString, pstrString, dwSize); } static void ConvertToAnsi(_CharType* pstrString, size_t size) { if (size>UINT_MAX) return; DWORD dwSize=static_cast<DWORD>(size); BOOL fSuccess=::OemToCharBuffA(pstrString, pstrString, dwSize); } static void __stdcall FloodCharacters(char ch, int nLength, char* pch) { // nLength is in XCHARs memset(pch, ch, nLength); } static BSTR __stdcall AllocSysString(const char* pchData, int nDataLength) { int nLen = ::MultiByteToWideChar(Langpack_GetDefaultCodePage(), 0, pchData, nDataLength, NULL, NULL); BSTR bstr = ::SysAllocStringLen(NULL, nLen); if (bstr != NULL) ::MultiByteToWideChar(Langpack_GetDefaultCodePage(), 0, pchData, nDataLength, bstr, nLen); return bstr; } static BOOL __stdcall ReAllocSysString(const char* pchData, BSTR* pbstr, int nDataLength) { int nLen = ::MultiByteToWideChar(Langpack_GetDefaultCodePage(), 0, pchData, nDataLength, NULL, NULL); BOOL bSuccess = ::SysReAllocStringLen(pbstr, NULL, nLen); if (bSuccess) ::MultiByteToWideChar(Langpack_GetDefaultCodePage(), 0, pchData, nDataLength, *pbstr, nLen); return bSuccess; } static int __stdcall SafeStringLen(LPCSTR psz) { // returns length in bytes return (psz != NULL) ? int(strlen(psz)) : 0; } static int __stdcall SafeStringLen(LPCWSTR psz) { // returns length in wchar_ts return (psz != NULL) ? int(wcslen(psz)) : 0; } static int __stdcall GetCharLen(const wchar_t* pch) { // returns char length return 1; } static int __stdcall GetCharLen(const char* pch) { // returns char length return int(_mbclen(reinterpret_cast< const unsigned char* >(pch))); } static DWORD __stdcall GetEnvironmentVariable(LPCSTR pszVar, LPSTR pszBuffer, DWORD dwSize) { return ::GetEnvironmentVariableA(pszVar, pszBuffer, dwSize); } }; // specialization for wchar_t template<> class ChTraitsCRT< wchar_t > : public ChTraitsBase< wchar_t > { static DWORD __stdcall _GetEnvironmentVariableW(LPCWSTR pszName, LPWSTR pszBuffer, DWORD nSize) { return ::GetEnvironmentVariableW(pszName, pszBuffer, nSize); } public: static LPWSTR __stdcall CharNext(LPCWSTR psz) { return const_cast< LPWSTR >(psz+1); } static int __stdcall IsDigit(wchar_t ch) { return iswdigit(static_cast<unsigned short>(ch)); } static int __stdcall IsSpace(wchar_t ch) { return iswspace(static_cast<unsigned short>(ch)); } static int __stdcall StringCompare(LPCWSTR pszA, LPCWSTR pszB) { return wcscmp(pszA, pszB); } static int __stdcall StringCompareIgnore(LPCWSTR pszA, LPCWSTR pszB) { return _wcsicmp(pszA, pszB); } static int __stdcall StringCollate(LPCWSTR pszA, LPCWSTR pszB) { return wcscoll(pszA, pszB); } static int __stdcall StringCollateIgnore(LPCWSTR pszA, LPCWSTR pszB) { return _wcsicoll(pszA, pszB); } static LPCWSTR __stdcall StringFindString(LPCWSTR pszBlock, LPCWSTR pszMatch) { return wcsstr(pszBlock, pszMatch); } static LPWSTR __stdcall StringFindString(LPWSTR pszBlock, LPCWSTR pszMatch) { return const_cast< LPWSTR >(StringFindString(const_cast< LPCWSTR >(pszBlock), pszMatch)); } static LPCWSTR __stdcall StringFindChar(LPCWSTR pszBlock, wchar_t chMatch) { return wcschr(pszBlock, chMatch); } static LPCWSTR __stdcall StringFindCharRev(LPCWSTR psz, wchar_t ch) { return wcsrchr(psz, ch); } static LPCWSTR __stdcall StringScanSet(LPCWSTR pszBlock, LPCWSTR pszMatch) { return wcspbrk(pszBlock, pszMatch); } static int __stdcall StringSpanIncluding(LPCWSTR pszBlock, LPCWSTR pszSet) { return (int)wcsspn(pszBlock, pszSet); } static int __stdcall StringSpanExcluding(LPCWSTR pszBlock, LPCWSTR pszSet) { return (int)wcscspn(pszBlock, pszSet); } static LPWSTR __stdcall StringUppercase(LPWSTR psz) { CharUpperBuffW(psz, (DWORD)wcslen(psz)); return psz; } static LPWSTR __stdcall StringLowercase(LPWSTR psz) { CharLowerBuffW(psz, (DWORD)wcslen(psz)); return psz; } static LPWSTR __stdcall StringUppercase(LPWSTR psz, size_t len) { CharUpperBuffW(psz, (DWORD)len); return psz; } static LPWSTR __stdcall StringLowercase(LPWSTR psz, size_t len) { CharLowerBuffW(psz, (DWORD)len); return psz; } static LPWSTR __stdcall StringReverse(LPWSTR psz) { return _wcsrev(psz); } static int __stdcall GetFormattedLength(LPCWSTR pszFormat, va_list args); static int __stdcall Format(LPWSTR pszBuffer, LPCWSTR pszFormat, va_list args) { #pragma warning (push) #pragma warning(disable : 4996) return vswprintf(pszBuffer, pszFormat, args); //!!!!!!!!! #pragma warning (pop) } static int __stdcall Format(LPWSTR pszBuffer, size_t nLength, LPCWSTR pszFormat, va_list args); static int __stdcall GetBaseTypeLength(LPCSTR pszSrc) { // Returns required buffer size in wchar_ts return ::MultiByteToWideChar(CP_ACP, 0, pszSrc, -1, NULL, 0)-1; } static int __stdcall GetBaseTypeLength(LPCSTR pszSrc, int nLength) { // Returns required buffer size in wchar_ts return ::MultiByteToWideChar(CP_ACP, 0, pszSrc, nLength, NULL, 0); } static int __stdcall GetBaseTypeLength(LPCWSTR pszSrc) { // Returns required buffer size in wchar_ts return (int)wcslen(pszSrc); } static int __stdcall GetBaseTypeLength(LPCWSTR pszSrc, int nLength) { (void)pszSrc; // Returns required buffer size in wchar_ts return nLength; } static void __stdcall ConvertToBaseType(LPWSTR pszDest, int nDestLength, LPCSTR pszSrc, int nSrcLength = -1) { // nLen is in wchar_ts ::MultiByteToWideChar(CP_ACP, 0, pszSrc, nSrcLength, pszDest, nDestLength); } static void __stdcall ConvertToBaseType(LPWSTR pszDest, int nDestLength, LPCWSTR pszSrc, int nSrcLength = -1) { if (nSrcLength == -1) { nSrcLength=1 + GetBaseTypeLength(pszSrc); } // nLen is in wchar_ts #if _MSC_VER >= 1400 wmemcpy_s(pszDest, nDestLength, pszSrc, nSrcLength); #else wmemcpy(pszDest, pszSrc, nDestLength); #endif } static void __stdcall FloodCharacters(wchar_t ch, int nLength, LPWSTR psz) { // nLength is in XCHARs for (int i = 0; i < nLength; i++) { psz[i] = ch; } } static BSTR __stdcall AllocSysString(const wchar_t* pchData, int nDataLength) { return ::SysAllocStringLen(pchData, nDataLength); } static BOOL __stdcall ReAllocSysString(const wchar_t* pchData, BSTR* pbstr, int nDataLength) { return ::SysReAllocStringLen(pbstr, pchData, nDataLength); } static int __stdcall SafeStringLen(LPCSTR psz) { // returns length in bytes return (psz != NULL) ? (int)strlen(psz) : 0; } static int __stdcall SafeStringLen(LPCWSTR psz) { // returns length in wchar_ts return (psz != NULL) ? (int)wcslen(psz) : 0; } static int __stdcall GetCharLen(const wchar_t* pch) { (void)pch; // returns char length return 1; } static int __stdcall GetCharLen(const char* pch) { // returns char length return (int)(_mbclen(reinterpret_cast< const unsigned char* >(pch))); } static DWORD __stdcall GetEnvironmentVariable(LPCWSTR pszVar, LPWSTR pszBuffer, DWORD dwSize) { return _GetEnvironmentVariableW(pszVar, pszBuffer, dwSize); } static void __stdcall ConvertToOem(LPWSTR /*psz*/) { } static void __stdcall ConvertToAnsi(LPWSTR /*psz*/) { } static void __stdcall ConvertToOem(LPWSTR /*psz*/, size_t) { } static void __stdcall ConvertToAnsi(LPWSTR /*psz*/, size_t) { } }; template< typename BaseType, class StringTraits > class CMStringT : public CMSimpleStringT< BaseType > { public: typedef CMSimpleStringT< BaseType> CThisSimpleString; typedef typename CThisSimpleString::XCHAR XCHAR; typedef typename CThisSimpleString::PXSTR PXSTR; typedef typename CThisSimpleString::PCXSTR PCXSTR; typedef typename CThisSimpleString::YCHAR YCHAR; typedef typename CThisSimpleString::PYSTR PYSTR; typedef typename CThisSimpleString::PCYSTR PCYSTR; public: CMStringT() : CThisSimpleString() { } static void __stdcall Construct(CMStringT* pString) { new(pString) CMStringT; } // Copy constructor CMStringT(const CMStringT& strSrc) : CThisSimpleString(strSrc) { } CMStringT(const XCHAR* pszSrc) : CThisSimpleString() { *this = pszSrc; } CMStringT(CMStringDataFormat, const XCHAR* pszFormat, ...) : CThisSimpleString() { va_list args; va_start(args, pszFormat); FormatV(pszFormat, args); } CMStringT(const YCHAR* pszSrc) : CThisSimpleString() { *this = pszSrc; } CMStringT(const unsigned char* pszSrc) : CThisSimpleString() { *this = reinterpret_cast<const char*>(pszSrc); } CMStringT(char ch, int nLength = 1) : CThisSimpleString() { if (nLength > 0) { PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::FloodCharacters(XCHAR(ch), nLength, pszBuffer); this->ReleaseBufferSetLength(nLength); } } CMStringT(wchar_t ch, int nLength = 1) : CThisSimpleString() { if (nLength > 0) { //Convert ch to the BaseType wchar_t pszCh[2] = { ch, 0 }; int nBaseTypeCharLen = 1; if (ch != L'\0') nBaseTypeCharLen = StringTraits::GetBaseTypeLength(pszCh); XCHAR *buffBaseTypeChar = new XCHAR[nBaseTypeCharLen + 1]; StringTraits::ConvertToBaseType(buffBaseTypeChar, nBaseTypeCharLen + 1, pszCh, 1); //allocate enough characters in String and flood (replicate) with the (converted character)*nLength PXSTR pszBuffer = this->GetBuffer(nLength*nBaseTypeCharLen); if (nBaseTypeCharLen == 1) //Optimization for a common case - wide char translates to 1 ansi/wide char. StringTraits::FloodCharacters(buffBaseTypeChar[0], nLength, pszBuffer); else { XCHAR* p = pszBuffer; for (int i = 0; i < nLength; i++) { for (int j = 0; j < nBaseTypeCharLen; ++j) { *p = buffBaseTypeChar[j]; ++p; } } } this->ReleaseBufferSetLength(nLength*nBaseTypeCharLen); delete[] buffBaseTypeChar; } } CMStringT(const XCHAR* pch, int nLength) : CThisSimpleString(pch, nLength) {} CMStringT(const YCHAR* pch, int nLength) : CThisSimpleString() { if (nLength > 0) { int nDestLength = StringTraits::GetBaseTypeLength(pch, nLength); PXSTR pszBuffer = this->GetBuffer(nDestLength); StringTraits::ConvertToBaseType(pszBuffer, nDestLength, pch, nLength); this->ReleaseBufferSetLength(nDestLength); } } // Destructor ~CMStringT() {} // Assignment operators CMStringT& operator=(const CMStringT& strSrc) { CThisSimpleString::operator=(strSrc); return *this; } CMStringT& operator=(PCXSTR pszSrc) { CThisSimpleString::operator=(pszSrc); return *this; } CMStringT& operator=(PCYSTR pszSrc) { // nDestLength is in XCHARs int nDestLength = (pszSrc != NULL) ? StringTraits::GetBaseTypeLength(pszSrc) : 0; if (nDestLength > 0) { PXSTR pszBuffer = this->GetBuffer(nDestLength); StringTraits::ConvertToBaseType(pszBuffer, nDestLength, pszSrc); this->ReleaseBufferSetLength(nDestLength); } else this->Empty(); return *this; } CMStringT& operator=(const unsigned char* pszSrc) { return operator=(reinterpret_cast<const char*>(pszSrc)); } CMStringT& operator=(char ch) { char ach[2] = { ch, 0 }; return operator=(ach); } CMStringT& operator=(wchar_t ch) { wchar_t ach[2] = { ch, 0 }; return operator=(ach); } // CMStringT& operator=(const VARIANT& var); CMStringT& operator+=(const CMStringT& str) { CThisSimpleString::operator+=(str); return *this; } CMStringT& operator+=(const CThisSimpleString& str) { CThisSimpleString::operator+=(str); return *this; } CMStringT& operator+=(PCXSTR pszSrc) { CThisSimpleString::operator+=(pszSrc); return *this; } CMStringT& operator+=(PCYSTR pszSrc) { CMStringT str(pszSrc); return operator+=(str); } CMStringT& operator+=(char ch) { CThisSimpleString::operator+=(ch); return *this; } CMStringT& operator+=(unsigned char ch) { CThisSimpleString::operator+=(ch); return *this; } CMStringT& operator+=(wchar_t ch) { CThisSimpleString::operator+=(ch); return *this; } // Comparison int Compare(PCXSTR psz) const { return StringTraits::StringCompare(this->GetString(), psz); } int CompareNoCase(PCXSTR psz) const { return StringTraits::StringCompareIgnore(this->GetString(), psz); } int Collate(PCXSTR psz) const { return StringTraits::StringCollate(this->GetString(), psz); } int CollateNoCase(PCXSTR psz) const { return StringTraits::StringCollateIgnore(this->GetString(), psz); } // Advanced manipulation // Delete 'nCount' characters, starting at index 'iIndex' int Delete(int iIndex, int nCount = 1) { if (iIndex < 0) iIndex = 0; if (nCount < 0) nCount = 0; int nLength = this->GetLength(); if (nCount + iIndex > nLength) nCount = nLength-iIndex; if (nCount > 0) { int nNewLength = nLength-nCount; int nXCHARsToCopy = nLength-(iIndex+nCount)+1; PXSTR pszBuffer = this->GetBuffer(); #if _MSC_VER >= 1400 memmove_s(pszBuffer+iIndex, nXCHARsToCopy*sizeof(XCHAR), pszBuffer+iIndex+nCount, nXCHARsToCopy*sizeof(XCHAR)); #else memmove(pszBuffer+iIndex, pszBuffer+iIndex+nCount, nXCHARsToCopy*sizeof(XCHAR)); #endif this->ReleaseBufferSetLength(nNewLength); } return this->GetLength(); } // Insert character 'ch' before index 'iIndex' int Insert(int iIndex, XCHAR ch) { if (iIndex < 0) iIndex = 0; if (iIndex > this->GetLength()) iIndex = this->GetLength(); int nNewLength = this->GetLength()+1; PXSTR pszBuffer = this->GetBuffer(nNewLength); // move existing bytes down #if _MSC_VER >= 1400 memmove_s(pszBuffer+iIndex+1, (nNewLength-iIndex)*sizeof(XCHAR), pszBuffer+iIndex, (nNewLength-iIndex)*sizeof(XCHAR)); #else memmove(pszBuffer+iIndex+1, pszBuffer+iIndex, (nNewLength-iIndex)*sizeof(XCHAR)); #endif pszBuffer[iIndex] = ch; this->ReleaseBufferSetLength(nNewLength); return nNewLength; } // Insert string 'psz' before index 'iIndex' int Insert(int iIndex, PCXSTR psz) { if (iIndex < 0) iIndex = 0; if (iIndex > this->GetLength()) iIndex = this->GetLength(); // nInsertLength and nNewLength are in XCHARs int nInsertLength = StringTraits::SafeStringLen(psz); int nNewLength = this->GetLength(); if (nInsertLength > 0) { nNewLength += nInsertLength; PXSTR pszBuffer = this->GetBuffer(nNewLength); // move existing bytes down #if _MSC_VER >= 1400 memmove_s(pszBuffer+iIndex+nInsertLength, (nNewLength-iIndex-nInsertLength+1)*sizeof(XCHAR), pszBuffer+iIndex, (nNewLength-iIndex-nInsertLength+1)*sizeof(XCHAR)); memcpy_s(pszBuffer+iIndex, nInsertLength*sizeof(XCHAR), psz, nInsertLength*sizeof(XCHAR)); #else memmove(pszBuffer+iIndex+nInsertLength, pszBuffer+iIndex, (nNewLength-iIndex-nInsertLength+1)*sizeof(XCHAR)); memcpy(pszBuffer+iIndex, psz, nInsertLength*sizeof(XCHAR)); #endif this->ReleaseBufferSetLength(nNewLength); } return nNewLength; } // Replace all occurrences of character 'chOld' with character 'chNew' int Replace(XCHAR chOld, XCHAR chNew) { int nCount = 0; // short-circuit the nop case if (chOld != chNew) { // otherwise modify each character that matches in the string bool bCopied = false; PXSTR pszBuffer = const_cast< PXSTR >(this->GetString()); // We don't actually write to pszBuffer until we've called GetBuffer(). int nLength = this->GetLength(); int iChar = 0; while (iChar < nLength) { // replace instances of the specified character only if (pszBuffer[iChar] == chOld) { if (!bCopied) { bCopied = true; pszBuffer = this->GetBuffer(nLength); } pszBuffer[iChar] = chNew; nCount++; } iChar = int(StringTraits::CharNext(pszBuffer + iChar) - pszBuffer); } if (bCopied) this->ReleaseBufferSetLength(nLength); } return nCount; } // Replace all occurrences of string 'pszOld' with string 'pszNew' int Replace(PCXSTR pszOld, PCXSTR pszNew) { // can't have empty or NULL lpszOld // nSourceLen is in XCHARs int nSourceLen = StringTraits::SafeStringLen(pszOld); if (nSourceLen == 0) return 0; // nReplacementLen is in XCHARs int nReplacementLen = StringTraits::SafeStringLen(pszNew); // loop once to figure out the size of the result string int nCount = 0; { PCXSTR pszStart = this->GetString(); PCXSTR pszEnd = pszStart + this->GetLength(); while (pszStart < pszEnd) { PCXSTR pszTarget; while ((pszTarget = StringTraits::StringFindString(pszStart, pszOld)) != NULL) { nCount++; pszStart = pszTarget + nSourceLen; } pszStart += StringTraits::SafeStringLen(pszStart) + 1; } } // if any changes were made, make them if (nCount > 0) { // if the buffer is too small, just // allocate a new buffer (slow but sure) int nOldLength = this->GetLength(); int nNewLength = nOldLength+(nReplacementLen-nSourceLen)*nCount; PXSTR pszBuffer = this->GetBuffer(__max(nNewLength, nOldLength)); PXSTR pszStart = pszBuffer; PXSTR pszEnd = pszStart+nOldLength; // loop again to actually do the work while (pszStart < pszEnd) { PXSTR pszTarget; while ((pszTarget = StringTraits::StringFindString(pszStart, pszOld)) != NULL) { int nBalance = nOldLength - int(pszTarget - pszBuffer + nSourceLen); memmove_s(pszTarget + nReplacementLen, nBalance*sizeof(XCHAR), pszTarget + nSourceLen, nBalance*sizeof(XCHAR)); memcpy_s(pszTarget, nReplacementLen*sizeof(XCHAR), pszNew, nReplacementLen*sizeof(XCHAR)); pszStart = pszTarget + nReplacementLen; pszTarget[nReplacementLen + nBalance] = 0; nOldLength += (nReplacementLen - nSourceLen); } pszStart += StringTraits::SafeStringLen(pszStart) + 1; } this->ReleaseBufferSetLength(nNewLength); } return nCount; } // Remove all occurrences of character 'chRemove' int Remove(XCHAR chRemove) { int nLength = this->GetLength(); PXSTR pszBuffer = this->GetBuffer(nLength); PXSTR pszSource = pszBuffer; PXSTR pszDest = pszBuffer; PXSTR pszEnd = pszBuffer+nLength; while (pszSource < pszEnd) { PXSTR pszNewSource = StringTraits::CharNext(pszSource); if (*pszSource != chRemove) { // Copy the source to the destination. Remember to copy all bytes of an MBCS character // Copy the source to the destination. Remember to copy all bytes of an MBCS character size_t NewSourceGap = (pszNewSource - pszSource); PXSTR pszNewDest = pszDest + NewSourceGap; for (size_t i = 0; pszDest != pszNewDest && i < NewSourceGap; i++) { *pszDest = *pszSource; pszSource++; pszDest++; } } pszSource = pszNewSource; } *pszDest = 0; int nCount = int(pszSource - pszDest); this->ReleaseBufferSetLength(nLength - nCount); return nCount; } CMStringT Tokenize(PCXSTR pszTokens, int& iStart) const { if ((pszTokens == NULL) || (*pszTokens == (XCHAR)0)) { if (iStart < this->GetLength()) return CMStringT(this->GetString() + iStart); } else { PCXSTR pszPlace = this->GetString() + iStart; PCXSTR pszEnd = this->GetString() + this->GetLength(); if (pszPlace < pszEnd) { int nIncluding = StringTraits::StringSpanIncluding(pszPlace, pszTokens); if ((pszPlace + nIncluding) < pszEnd) { pszPlace += nIncluding; int nExcluding = StringTraits::StringSpanExcluding(pszPlace, pszTokens); int iFrom = iStart + nIncluding; int nUntil = nExcluding; iStart = iFrom + nUntil + 1; return Mid(iFrom, nUntil); } } } // return empty string, done tokenizing iStart = -1; return CMStringT(); } // find routines // Find the first occurrence of character 'ch', starting at index 'iStart' int Find(XCHAR ch, int iStart = 0) const { // nLength is in XCHARs int nLength = this->GetLength(); if (iStart < 0 || iStart >= nLength) return -1; // find first single character PCXSTR psz = StringTraits::StringFindChar(this->GetString() + iStart, ch); // return -1 if not found and index otherwise return (psz == NULL) ? -1 : int(psz - this->GetString()); } // look for a specific sub-string // Find the first occurrence of string 'pszSub', starting at index 'iStart' int Find(PCXSTR pszSub, int iStart = 0) const { // iStart is in XCHARs if (pszSub == NULL) return -1; // nLength is in XCHARs int nLength = this->GetLength(); if (iStart < 0 || iStart > nLength) return -1; // find first matching substring PCXSTR psz = StringTraits::StringFindString(this->GetString() + iStart, pszSub); // return -1 for not found, distance from beginning otherwise return (psz == NULL) ? -1 : int(psz - this->GetString()); } // Find the first occurrence of any of the characters in string 'pszCharSet' int FindOneOf(PCXSTR pszCharSet) const { PCXSTR psz = StringTraits::StringScanSet(this->GetString(), pszCharSet); return (psz == NULL) ? -1 : int(psz - this->GetString()); } // Find the last occurrence of character 'ch' int ReverseFind(XCHAR ch) const { // find last single character PCXSTR psz = StringTraits::StringFindCharRev(this->GetString(), ch); // return -1 if not found, distance from beginning otherwise return (psz == NULL) ? -1 : int(psz - this->GetString()); } // manipulation // Convert the string to uppercase CMStringT& MakeUpper() { int nLength = this->GetLength(); PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::StringUppercase(pszBuffer, nLength + 1); this->ReleaseBufferSetLength(nLength); return *this; } // Convert the string to lowercase CMStringT& MakeLower() { int nLength = this->GetLength(); PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::StringLowercase(pszBuffer, nLength + 1); this->ReleaseBufferSetLength(nLength); return *this; } // Reverse the string CMStringT& MakeReverse() { int nLength = this->GetLength(); PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::StringReverse(pszBuffer); this->ReleaseBufferSetLength(nLength); return *this; } // trimming // Remove all trailing whitespace CMStringT& TrimRight() { // find beginning of trailing spaces by starting // at beginning (DBCS aware) PCXSTR psz = this->GetString(); PCXSTR pszLast = NULL; while (*psz != 0) { if (StringTraits::IsSpace(*psz)) { if (pszLast == NULL) pszLast = psz; } else pszLast = NULL; psz = StringTraits::CharNext(psz); } if (pszLast != NULL) { // truncate at trailing space start int iLast = int(pszLast - this->GetString()); this->Truncate(iLast); } return *this; } // Remove all leading whitespace CMStringT& TrimLeft() { // find first non-space character PCXSTR psz = this->GetString(); while (StringTraits::IsSpace(*psz)) psz = StringTraits::CharNext(psz); if (psz != this->GetString()) { // fix up data and length int iFirst = int(psz - this->GetString()); PXSTR pszBuffer = this->GetBuffer(this->GetLength()); psz = pszBuffer + iFirst; int nDataLength = this->GetLength() - iFirst; memmove_s(pszBuffer, (this->GetLength() + 1)*sizeof(XCHAR), psz, (nDataLength + 1)*sizeof(XCHAR)); this->ReleaseBufferSetLength(nDataLength); } return *this; } // Remove all leading and trailing whitespace CMStringT& Trim() { return TrimRight().TrimLeft(); } // Remove all leading and trailing occurrences of character 'chTarget' CMStringT& Trim(XCHAR chTarget) { return TrimRight(chTarget).TrimLeft(chTarget); } // Remove all leading and trailing occurrences of any of the characters in the string 'pszTargets' CMStringT& Trim(PCXSTR pszTargets) { return TrimRight(pszTargets).TrimLeft(pszTargets); } // trimming anything (either side) // Remove all trailing occurrences of character 'chTarget' CMStringT& TrimRight(XCHAR chTarget) { // find beginning of trailing matches // by starting at beginning (DBCS aware) PCXSTR psz = this->GetString(); PCXSTR pszLast = NULL; while (*psz != 0) { if (*psz == chTarget) { if (pszLast == NULL) pszLast = psz; } else pszLast = NULL; psz = StringTraits::CharNext(psz); } if (pszLast != NULL) { // truncate at left-most matching character int iLast = int(pszLast - this->GetString()); this->Truncate(iLast); } return *this; } // Remove all trailing occurrences of any of the characters in string 'pszTargets' CMStringT& TrimRight(PCXSTR pszTargets) { // if we're not trimming anything, we're not doing any work if ((pszTargets == NULL) || (*pszTargets == 0)) { return *this; } // find beginning of trailing matches // by starting at beginning (DBCS aware) PCXSTR psz = this->GetString(); PCXSTR pszLast = NULL; while (*psz != 0) { if (StringTraits::StringFindChar(pszTargets, *psz) != NULL) { if (pszLast == NULL) { pszLast = psz; } } else { pszLast = NULL; } psz = StringTraits::CharNext(psz); } if (pszLast != NULL) { // truncate at left-most matching character int iLast = int(pszLast - this->GetString()); this->Truncate(iLast); } return *this; } // Remove all leading occurrences of character 'chTarget' CMStringT& TrimLeft(XCHAR chTarget) { // find first non-matching character PCXSTR psz = this->GetString(); while (chTarget == *psz) { psz = StringTraits::CharNext(psz); } if (psz != this->GetString()) { // fix up data and length int iFirst = int(psz - this->GetString()); PXSTR pszBuffer = this->GetBuffer(this->GetLength()); psz = pszBuffer + iFirst; int nDataLength = this->GetLength() - iFirst; memmove_s(pszBuffer, (this->GetLength() + 1)*sizeof(XCHAR), psz, (nDataLength + 1)*sizeof(XCHAR)); this->ReleaseBufferSetLength(nDataLength); } return *this; } // Remove all leading occurrences of any of the characters in string 'pszTargets' CMStringT& TrimLeft(PCXSTR pszTargets) { // if we're not trimming anything, we're not doing any work if ((pszTargets == NULL) || (*pszTargets == 0)) { return *this; } PCXSTR psz = this->GetString(); while ((*psz != 0) && (StringTraits::StringFindChar(pszTargets, *psz) != NULL)) { psz = StringTraits::CharNext(psz); } if (psz != this->GetString()) { // fix up data and length int iFirst = int(psz - this->GetString()); PXSTR pszBuffer = this->GetBuffer(this->GetLength()); psz = pszBuffer + iFirst; int nDataLength = this->GetLength() - iFirst; memmove_s(pszBuffer, (this->GetLength() + 1)*sizeof(XCHAR), psz, (nDataLength + 1)*sizeof(XCHAR)); this->ReleaseBufferSetLength(nDataLength); } return *this; } // Convert the string to the OEM character set void AnsiToOem() { int nLength = this->GetLength(); PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::ConvertToOem(pszBuffer, nLength + 1); this->ReleaseBufferSetLength(nLength); } // Convert the string to the ANSI character set void OemToAnsi() { int nLength = this->GetLength(); PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::ConvertToAnsi(pszBuffer, nLength + 1); this->ReleaseBufferSetLength(nLength); } // Very simple sub-string extraction // Return the substring starting at index 'iFirst' CMStringT Mid(int iFirst) const { return Mid(iFirst, this->GetLength() - iFirst); } // Return the substring starting at index 'iFirst', with length 'nCount' CMStringT Mid(int iFirst, int nCount) const { // nCount is in XCHARs // out-of-bounds requests return sensible things if (iFirst < 0) iFirst = 0; if (nCount < 0) nCount = 0; if ((iFirst + nCount) > this->GetLength()) nCount = this->GetLength() - iFirst; if (iFirst > this->GetLength()) nCount = 0; // optimize case of returning entire string if ((iFirst == 0) && ((iFirst + nCount) == this->GetLength())) return *this; return CMStringT(this->GetString() + iFirst, nCount); } // Return the substring consisting of the rightmost 'nCount' characters CMStringT Right(int nCount) const { // nCount is in XCHARs if (nCount < 0) nCount = 0; int nLength = this->GetLength(); if (nCount >= nLength) { return *this; } return CMStringT(this->GetString() + nLength - nCount, nCount); } // Return the substring consisting of the leftmost 'nCount' characters CMStringT Left(int nCount) const { // nCount is in XCHARs if (nCount < 0) nCount = 0; int nLength = this->GetLength(); if (nCount >= nLength) return *this; return CMStringT(this->GetString(), nCount); } // Return the substring consisting of the leftmost characters in the set 'pszCharSet' CMStringT SpanIncluding(PCXSTR pszCharSet) const { return Left(StringTraits::StringSpanIncluding(this->GetString(), pszCharSet)); } // Return the substring consisting of the leftmost characters not in the set 'pszCharSet' CMStringT SpanExcluding(PCXSTR pszCharSet) const { return Left(StringTraits::StringSpanExcluding(this->GetString(), pszCharSet)); } // Format data using format string 'pszFormat' PCXSTR Format(PCXSTR pszFormat, ...) { va_list argList; va_start(argList, pszFormat); FormatV(pszFormat, argList); va_end(argList); return GetString(); } // Append formatted data using format string 'pszFormat' PCXSTR AppendFormat(PCXSTR pszFormat, ...) { va_list argList; va_start(argList, pszFormat); AppendFormatV(pszFormat, argList); va_end(argList); return GetString(); } void AppendFormatV(PCXSTR pszFormat, va_list args) { int nCurrentLength = this->GetLength(); int nAppendLength = StringTraits::GetFormattedLength(pszFormat, args); PXSTR pszBuffer = this->GetBuffer(nCurrentLength + nAppendLength); StringTraits::Format(pszBuffer + nCurrentLength, nAppendLength + 1, pszFormat, args); this->ReleaseBufferSetLength(nCurrentLength + nAppendLength); } PCXSTR FormatV(PCXSTR pszFormat, va_list args) { int nLength = StringTraits::GetFormattedLength(pszFormat, args); PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::Format(pszBuffer, nLength + 1, pszFormat, args); this->ReleaseBufferSetLength(nLength); return GetString(); } // OLE BSTR support // allocate a BSTR containing a copy of the string BSTR AllocSysString() const { BSTR bstrResult = StringTraits::AllocSysString(this->GetString(), this->GetLength()); return bstrResult; } BSTR SetSysString(BSTR* pbstr) const { StringTraits::ReAllocSysString(this->GetString(), pbstr, this->GetLength()); return *pbstr; } // Set the string to the value of environment variable 'pszVar' BOOL GetEnvironmentVariable(PCXSTR pszVar) { ULONG nLength = StringTraits::GetEnvironmentVariable(pszVar, NULL, 0); BOOL bRetVal = FALSE; if (nLength == 0) this->Empty(); else { PXSTR pszBuffer = this->GetBuffer(nLength); StringTraits::GetEnvironmentVariable(pszVar, pszBuffer, nLength); this->ReleaseBuffer(); bRetVal = TRUE; } return bRetVal; } // Load the string from resource 'nID' BOOL LoadString(UINT nID) { HINSTANCE hInst = StringTraits::FindStringResourceInstance(nID); if (hInst == NULL) return FALSE; return LoadString(hInst, nID); } friend CMStringT __stdcall operator+(const CMStringT& str1, const CMStringT& str2) { CMStringT strResult; Concatenate(strResult, str1, str1.GetLength(), str2, str2.GetLength()); return strResult; } friend CMStringT __stdcall operator+(const CMStringT& str1, PCXSTR psz2) { CMStringT strResult; Concatenate(strResult, str1, str1.GetLength(), psz2, StringLength(psz2)); return strResult; } friend CMStringT __stdcall operator+(PCXSTR psz1, const CMStringT& str2) { CMStringT strResult; Concatenate(strResult, psz1, StringLength(psz1), str2, str2.GetLength()); return strResult; } friend CMStringT __stdcall operator+(const CMStringT& str1, wchar_t ch2) { CMStringT strResult; XCHAR chTemp = XCHAR(ch2); Concatenate(strResult, str1, str1.GetLength(), &chTemp, 1); return strResult; } friend CMStringT __stdcall operator+(const CMStringT& str1, char ch2) { CMStringT strResult; XCHAR chTemp = XCHAR(ch2); Concatenate(strResult, str1, str1.GetLength(), &chTemp, 1); return strResult; } friend CMStringT __stdcall operator+(wchar_t ch1, const CMStringT& str2) { CMStringT strResult; XCHAR chTemp = XCHAR(ch1); Concatenate(strResult, &chTemp, 1, str2, str2.GetLength()); return strResult; } friend CMStringT __stdcall operator+(char ch1, const CMStringT& str2) { CMStringT strResult; XCHAR chTemp = XCHAR(ch1); Concatenate(strResult, &chTemp, 1, str2, str2.GetLength()); return strResult; } friend bool __stdcall operator==(const CMStringT& str1, const CMStringT& str2) { return str1.Compare(str2) == 0; } friend bool __stdcall operator==(const CMStringT& str1, PCXSTR psz2) { return str1.Compare(psz2) == 0; } friend bool __stdcall operator==(PCXSTR psz1, const CMStringT& str2) { return str2.Compare(psz1) == 0; } friend bool __stdcall operator==(const CMStringT& str1, PCYSTR psz2) { CMStringT str2(psz2); return str1 == str2; } friend bool __stdcall operator==(PCYSTR psz1, const CMStringT& str2) { CMStringT str1(psz1); return str1 == str2; } friend bool __stdcall operator!=(const CMStringT& str1, const CMStringT& str2) { return str1.Compare(str2) != 0; } friend bool __stdcall operator!=(const CMStringT& str1, PCXSTR psz2) { return str1.Compare(psz2) != 0; } friend bool __stdcall operator!=(PCXSTR psz1, const CMStringT& str2) { return str2.Compare(psz1) != 0; } friend bool __stdcall operator!=(const CMStringT& str1, PCYSTR psz2) { CMStringT str2(psz2); return str1 != str2; } friend bool __stdcall operator!=(PCYSTR psz1, const CMStringT& str2) { CMStringT str1(psz1); return str1 != str2; } friend bool __stdcall operator<(const CMStringT& str1, const CMStringT& str2) { return str1.Compare(str2) < 0; } friend bool __stdcall operator<(const CMStringT& str1, PCXSTR psz2) { return str1.Compare(psz2) < 0; } friend bool __stdcall operator<(PCXSTR psz1, const CMStringT& str2) { return str2.Compare(psz1) > 0; } friend bool __stdcall operator>(const CMStringT& str1, const CMStringT& str2) { return str1.Compare(str2) > 0; } friend bool __stdcall operator>(const CMStringT& str1, PCXSTR psz2) { return str1.Compare(psz2) > 0; } friend bool __stdcall operator>(PCXSTR psz1, const CMStringT& str2) { return str2.Compare(psz1) < 0; } friend bool __stdcall operator<=(const CMStringT& str1, const CMStringT& str2) { return str1.Compare(str2) <= 0; } friend bool __stdcall operator<=(const CMStringT& str1, PCXSTR psz2) { return str1.Compare(psz2) <= 0; } friend bool __stdcall operator<=(PCXSTR psz1, const CMStringT& str2) { return str2.Compare(psz1) >= 0; } friend bool __stdcall operator>=(const CMStringT& str1, const CMStringT& str2) { return str1.Compare(str2) >= 0; } friend bool __stdcall operator>=(const CMStringT& str1, PCXSTR psz2) { return str1.Compare(psz2) >= 0; } friend bool __stdcall operator>=(PCXSTR psz1, const CMStringT& str2) { return str2.Compare(psz1) <= 0; } friend bool __stdcall operator==(XCHAR ch1, const CMStringT& str2) { return (str2.GetLength() == 1) && (str2[0] == ch1); } friend bool __stdcall operator==(const CMStringT& str1, XCHAR ch2) { return (str1.GetLength() == 1) && (str1[0] == ch2); } friend bool __stdcall operator!=(XCHAR ch1, const CMStringT& str2) { return (str2.GetLength() != 1) || (str2[0] != ch1); } friend bool __stdcall operator!=(const CMStringT& str1, XCHAR ch2) { return (str1.GetLength() != 1) || (str1[0] != ch2); } }; typedef CMStringT< wchar_t, ChTraitsCRT< wchar_t > > CMStringW; typedef CMStringT< char, ChTraitsCRT< char > > CMStringA; typedef CMStringT< TCHAR, ChTraitsCRT< TCHAR > > CMString; ///////////////////////////////////////////////////////////////////////////////////////// // ChTraitsCRT<wchar_t> __forceinline int ChTraitsCRT<wchar_t>::GetFormattedLength(LPCWSTR pszFormat, va_list args) { return _vscwprintf(pszFormat, args); } __forceinline int ChTraitsCRT<wchar_t>::Format(LPWSTR pszBuffer, size_t nLength, LPCWSTR pszFormat, va_list args) { return _vsnwprintf(pszBuffer, nLength, pszFormat, args); } ///////////////////////////////////////////////////////////////////////////////////////// // ChTraitsCRT<char> __forceinline int ChTraitsCRT<char>::GetFormattedLength(LPCSTR pszFormat, va_list args) { return _vscprintf(pszFormat, args); } __forceinline int ChTraitsCRT<char>::Format(LPSTR pszBuffer, size_t nlength, LPCSTR pszFormat, va_list args) { return vsprintf_s(pszBuffer, nlength, pszFormat, args); } #endif // M_STRING_H__