698 lines
19 KiB
C++
698 lines
19 KiB
C++
|
// MLStrW.cpp : Implementation of CMLStrW
|
||
|
#include "private.h"
|
||
|
|
||
|
#ifndef NEWMLSTR
|
||
|
|
||
|
#ifdef ASTRIMPL
|
||
|
#include "mlstr.h"
|
||
|
#include "mlsbwalk.h"
|
||
|
|
||
|
|
||
|
// CMLStrW
|
||
|
|
||
|
STDMETHODIMP CMLStrW::Sync(BOOL fNoAccess)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->Sync(fNoAccess);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetLength(long* plLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->GetLength(plLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetMLStr(long lDestPos, long lDestLen, IUnknown* pSrcMLStr, long lSrcPos, long lSrcLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->SetMLStr(lDestPos, lDestLen, pSrcMLStr, lSrcPos, lSrcLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetMLStr(long lSrcPos, long lSrcLen, IUnknown* pUnkOuter, DWORD dwClsContext, const IID* piid, IUnknown** ppDestMLStr, long* plDestPos, long* plDestLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->GetMLStr(lSrcPos, lSrcLen, pUnkOuter, dwClsContext, piid, ppDestMLStr, plDestPos, plDestLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetWStr(long lDestPos, long lDestLen, const WCHAR* pszSrc, long cchSrc, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
ASSERT_READ_BLOCK(pszSrc, cchSrc);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(pcchActual);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(plActualLen);
|
||
|
|
||
|
POWNER const pOwner = GetOwner();
|
||
|
HRESULT hr = pOwner->CheckThread();
|
||
|
CMLStr::CLock Lock(TRUE, pOwner, hr);
|
||
|
long cchDestPos;
|
||
|
long cchDestLen;
|
||
|
long cchActual;
|
||
|
long lActualLen;
|
||
|
|
||
|
if (SUCCEEDED(hr) && (pOwner->GetBufFlags() & MLSTR_WRITE))
|
||
|
hr = E_INVALIDARG; // Not writable StrBuf; TODO: Replace StrBuf in this case if allowed
|
||
|
|
||
|
if (SUCCEEDED(hr) &&
|
||
|
SUCCEEDED(hr = pOwner->PrepareMLStrBuf()) &&
|
||
|
SUCCEEDED(hr = pOwner->RegularizePosLen(&lDestPos, &lDestLen)) &&
|
||
|
SUCCEEDED(hr = pOwner->GetCCh(0, lDestPos, &cchDestPos)) &&
|
||
|
SUCCEEDED(hr = pOwner->GetCCh(cchDestPos, lDestLen, &cchDestLen)))
|
||
|
{
|
||
|
IMLangStringBufW* const pMLStrBufW = pOwner->GetMLStrBufW();
|
||
|
|
||
|
if (pMLStrBufW)
|
||
|
{
|
||
|
if (cchSrc > cchDestLen)
|
||
|
{
|
||
|
hr = pMLStrBufW->Insert(cchDestPos, cchSrc - cchDestLen, (pcchActual || plActualLen) ? &cchSrc : NULL);
|
||
|
cchSrc += cchDestLen;
|
||
|
}
|
||
|
else if (cchSrc < cchDestLen)
|
||
|
{
|
||
|
hr = pMLStrBufW->Delete(cchDestPos, cchDestLen - cchSrc);
|
||
|
}
|
||
|
|
||
|
CMLStrBufWalkW BufWalk(pMLStrBufW, cchDestPos, cchSrc, (pcchActual || plActualLen));
|
||
|
|
||
|
lActualLen = 0;
|
||
|
while (BufWalk.Lock(hr))
|
||
|
{
|
||
|
long lLen;
|
||
|
|
||
|
if (plActualLen)
|
||
|
hr = pOwner->CalcLenW(pszSrc, BufWalk.GetCCh(), &lLen);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
lActualLen += lLen;
|
||
|
::memcpy(BufWalk.GetStr(), pszSrc, sizeof(WCHAR) * BufWalk.GetCCh());
|
||
|
pszSrc += BufWalk.GetCCh();
|
||
|
}
|
||
|
|
||
|
BufWalk.Unlock(hr);
|
||
|
}
|
||
|
|
||
|
cchActual = BufWalk.GetDoneCCh();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
IMLangStringBufA* const pMLStrBufA = pOwner->GetMLStrBufA(); // Should succeed because PrepareMLStrBuf() above was succeeded
|
||
|
const UINT uCodePage = pOwner->GetCodePage();
|
||
|
long cchSrcA;
|
||
|
|
||
|
if (SUCCEEDED(hr = pOwner->ConvWStrToAStr(pcchActual || plActualLen, uCodePage, pszSrc, cchSrc, NULL, 0, &cchSrcA, NULL, NULL)))
|
||
|
{
|
||
|
if (cchSrcA > cchDestLen)
|
||
|
{
|
||
|
hr = pMLStrBufA->Insert(cchDestPos, cchSrcA - cchDestLen, (pcchActual || plActualLen) ? &cchSrcA : NULL);
|
||
|
cchSrcA += cchDestLen;
|
||
|
}
|
||
|
else if (cchSrcA < cchDestLen)
|
||
|
{
|
||
|
hr = pMLStrBufA->Delete(cchDestPos, cchDestLen - cchSrcA);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
CMLStrBufWalkA BufWalk(pMLStrBufA, cchDestPos, cchSrcA, (pcchActual || plActualLen));
|
||
|
|
||
|
cchActual = 0;
|
||
|
lActualLen = 0;
|
||
|
while (BufWalk.Lock(hr))
|
||
|
{
|
||
|
long cchWrittenA;
|
||
|
long cchWrittenW;
|
||
|
long lWrittenLen;
|
||
|
|
||
|
if (SUCCEEDED(hr = pOwner->ConvWStrToAStr(pcchActual || plActualLen, uCodePage, pszSrc, cchSrc, BufWalk.GetStr(), BufWalk.GetCCh(), &cchWrittenA, &cchWrittenW, &lWrittenLen)))
|
||
|
{
|
||
|
pszSrc += cchWrittenW;
|
||
|
cchSrc -= cchWrittenW;
|
||
|
cchActual += cchWrittenW;
|
||
|
lActualLen += lWrittenLen;
|
||
|
}
|
||
|
|
||
|
BufWalk.Unlock(hr, cchWrittenA);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (pcchActual)
|
||
|
*pcchActual = cchActual;
|
||
|
if (plActualLen)
|
||
|
*plActualLen = lActualLen;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (pcchActual)
|
||
|
*pcchActual = 0;
|
||
|
if (plActualLen)
|
||
|
*plActualLen = 0;
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetStrBufW(long lDestPos, long lDestLen, IMLangStringBufW* pSrcBuf, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->SetStrBufCommon(this, lDestPos, lDestLen, 0, pSrcBuf, NULL, pcchActual, plActualLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetWStr(long lSrcPos, long lSrcLen, WCHAR* pszDest, long cchDest, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
ASSERT_WRITE_BLOCK_OR_NULL(pszDest, cchDest);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(pcchActual);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(plActualLen);
|
||
|
|
||
|
POWNER const pOwner = GetOwner();
|
||
|
HRESULT hr = pOwner->CheckThread();
|
||
|
CMLStr::CLock Lock(FALSE, pOwner, hr);
|
||
|
long cchSrcPos;
|
||
|
long cchSrcLen;
|
||
|
long cchActual;
|
||
|
long lActualLen;
|
||
|
|
||
|
if (SUCCEEDED(hr) &&
|
||
|
SUCCEEDED(hr = pOwner->RegularizePosLen(&lSrcPos, &lSrcLen)) &&
|
||
|
SUCCEEDED(hr = pOwner->GetCCh(0, lSrcPos, &cchSrcPos)) &&
|
||
|
SUCCEEDED(hr = pOwner->GetCCh(cchSrcPos, lSrcLen, &cchSrcLen)))
|
||
|
{
|
||
|
IMLangStringBufW* const pMLStrBufW = pOwner->GetMLStrBufW();
|
||
|
IMLangStringBufA* const pMLStrBufA = pOwner->GetMLStrBufA();
|
||
|
|
||
|
if (pszDest)
|
||
|
cchActual = min(cchSrcLen, cchDest);
|
||
|
else
|
||
|
cchActual = cchSrcLen;
|
||
|
|
||
|
if (pMLStrBufW)
|
||
|
{
|
||
|
CMLStrBufWalkW BufWalk(pMLStrBufW, cchSrcPos, cchActual, (pcchActual || plActualLen));
|
||
|
|
||
|
lActualLen = 0;
|
||
|
while (BufWalk.Lock(hr))
|
||
|
{
|
||
|
long lLen;
|
||
|
|
||
|
if (plActualLen)
|
||
|
hr = pOwner->CalcLenW(BufWalk.GetStr(), BufWalk.GetCCh(), &lLen);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
lActualLen += lLen;
|
||
|
|
||
|
if (pszDest)
|
||
|
{
|
||
|
::memcpy(pszDest, BufWalk.GetStr(), sizeof(WCHAR) * BufWalk.GetCCh());
|
||
|
pszDest += BufWalk.GetCCh();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
BufWalk.Unlock(hr);
|
||
|
}
|
||
|
|
||
|
cchActual = BufWalk.GetDoneCCh();
|
||
|
}
|
||
|
else if (pMLStrBufA)
|
||
|
{
|
||
|
CMLStrBufWalkA BufWalk(pMLStrBufA, cchSrcPos, cchActual, (pcchActual || plActualLen));
|
||
|
|
||
|
cchActual = 0;
|
||
|
lActualLen = 0;
|
||
|
while ((!pszDest || cchDest > 0) && BufWalk.Lock(hr))
|
||
|
{
|
||
|
CHAR* const pszBuf = BufWalk.GetStr();
|
||
|
long cchWrittenA;
|
||
|
long cchWrittenW;
|
||
|
long lWrittenLen;
|
||
|
|
||
|
if (SUCCEEDED(hr = pOwner->ConvAStrToWStr(pOwner->GetCodePage(), pszBuf, BufWalk.GetCCh(), pszDest, cchDest, &cchWrittenA, &cchWrittenW, &lWrittenLen)))
|
||
|
{
|
||
|
lActualLen += lWrittenLen;
|
||
|
cchActual += cchWrittenW;
|
||
|
|
||
|
if (pszDest)
|
||
|
{
|
||
|
pszDest += cchWrittenW;
|
||
|
cchDest -= cchWrittenW;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
BufWalk.Unlock(hr, cchWrittenA);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ASSERT(cchActual == 0); // MLStrBuf is not available
|
||
|
lActualLen = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (pcchActual)
|
||
|
*pcchActual = cchActual;
|
||
|
if (plActualLen)
|
||
|
*plActualLen = lActualLen;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (pcchActual)
|
||
|
*pcchActual = 0;
|
||
|
if (plActualLen)
|
||
|
*plActualLen = 0;
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetStrBufW(long lSrcPos, long lSrcMaxLen, IMLangStringBufW** ppDestBuf, long* plDestLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
ASSERT_WRITE_PTR_OR_NULL(ppDestBuf);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(plDestLen);
|
||
|
|
||
|
POWNER const pOwner = GetOwner();
|
||
|
HRESULT hr = pOwner->CheckThread();
|
||
|
CMLStr::CLock Lock(FALSE, pOwner, hr);
|
||
|
IMLangStringBufW* pMLStrBufW;
|
||
|
|
||
|
if (SUCCEEDED(hr) &&
|
||
|
SUCCEEDED(hr = pOwner->RegularizePosLen(&lSrcPos, &lSrcMaxLen)) &&
|
||
|
lSrcMaxLen <= 0)
|
||
|
{
|
||
|
hr = E_INVALIDARG;
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
pMLStrBufW = pOwner->GetMLStrBufW();
|
||
|
if (!pMLStrBufW)
|
||
|
hr = MLSTR_E_STRBUFNOTAVAILABLE;
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (ppDestBuf)
|
||
|
{
|
||
|
pMLStrBufW->AddRef();
|
||
|
*ppDestBuf = pMLStrBufW;
|
||
|
}
|
||
|
if (plDestLen)
|
||
|
*plDestLen = lSrcMaxLen;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (ppDestBuf)
|
||
|
*ppDestBuf = NULL;
|
||
|
if (plDestLen)
|
||
|
*plDestLen = 0;
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::LockWStr(long lSrcPos, long lSrcLen, long lFlags, long cchRequest, WCHAR** ppszDest, long* pcchDest, long* plDestLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
ASSERT_WRITE_PTR_OR_NULL(ppszDest);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(pcchDest);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(plDestLen);
|
||
|
|
||
|
POWNER const pOwner = GetOwner();
|
||
|
HRESULT hr = pOwner->CheckThread();
|
||
|
CMLStr::CLock Lock(lFlags & MLSTR_WRITE, pOwner, hr);
|
||
|
long cchSrcPos;
|
||
|
long cchSrcLen;
|
||
|
WCHAR* pszBuf = NULL;
|
||
|
long cchBuf;
|
||
|
long lLockLen;
|
||
|
BOOL fDirectLock;
|
||
|
|
||
|
if (SUCCEEDED(hr) && (!lFlags || (lFlags & ~pOwner->GetBufFlags() & MLSTR_WRITE)))
|
||
|
hr = E_INVALIDARG; // No flags specified, or not writable StrBuf; TODO: Replace StrBuf in this case if allowed
|
||
|
|
||
|
if (!(lFlags & MLSTR_WRITE))
|
||
|
cchRequest = 0;
|
||
|
|
||
|
if (SUCCEEDED(hr) &&
|
||
|
SUCCEEDED(hr = pOwner->PrepareMLStrBuf()) &&
|
||
|
SUCCEEDED(hr = pOwner->RegularizePosLen(&lSrcPos, &lSrcLen)) &&
|
||
|
SUCCEEDED(hr = pOwner->GetCCh(0, lSrcPos, &cchSrcPos)) &&
|
||
|
SUCCEEDED(hr = pOwner->GetCCh(cchSrcPos, lSrcLen, &cchSrcLen)))
|
||
|
{
|
||
|
IMLangStringBufW* const pMLStrBufW = pOwner->GetMLStrBufW();
|
||
|
fDirectLock = (pMLStrBufW != 0);
|
||
|
|
||
|
if (fDirectLock)
|
||
|
{
|
||
|
long cchInserted;
|
||
|
long cchLockLen = cchSrcLen;
|
||
|
|
||
|
if (cchRequest > cchSrcLen &&
|
||
|
SUCCEEDED(hr = pMLStrBufW->Insert(cchSrcPos + cchSrcLen, cchRequest - cchSrcLen, &cchInserted)))
|
||
|
{
|
||
|
pOwner->SetBufCCh(pOwner->GetBufCCh() + cchInserted);
|
||
|
cchLockLen += cchInserted;
|
||
|
|
||
|
if (!pcchDest && cchLockLen < cchRequest)
|
||
|
hr = E_OUTOFMEMORY; // Can't insert in StrBuf
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr) &&
|
||
|
SUCCEEDED(hr = pMLStrBufW->LockBuf(cchSrcPos, cchLockLen, &pszBuf, &cchBuf)) &&
|
||
|
!pcchDest && cchBuf < max(cchSrcLen, cchRequest))
|
||
|
{
|
||
|
hr = E_OUTOFMEMORY; // Can't lock StrBuf
|
||
|
}
|
||
|
|
||
|
if (plDestLen && SUCCEEDED(hr))
|
||
|
hr = pOwner->CalcLenW(pszBuf, cchBuf, &lLockLen);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
long cchSize;
|
||
|
|
||
|
if (SUCCEEDED(hr = pOwner->CalcBufSizeW(lSrcLen, &cchSize)))
|
||
|
{
|
||
|
cchBuf = max(cchSize, cchRequest);
|
||
|
hr = pOwner->MemAlloc(sizeof(*pszBuf) * cchBuf, (void**)&pszBuf);
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr) && (lFlags & MLSTR_READ))
|
||
|
hr = GetWStr(lSrcPos, lSrcLen, pszBuf, cchBuf, (pcchDest) ? &cchBuf : NULL, (plDestLen) ? &lLockLen : NULL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr) &&
|
||
|
SUCCEEDED(hr = Lock.FallThrough()))
|
||
|
{
|
||
|
hr = pOwner->GetLockInfo()->Lock((fDirectLock) ? pOwner->UnlockWStrDirect : pOwner->UnlockWStrIndirect, lFlags, 0, pszBuf, lSrcPos, lSrcLen, cchSrcPos, cchBuf);
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (ppszDest)
|
||
|
*ppszDest = pszBuf;
|
||
|
if (pcchDest)
|
||
|
*pcchDest = cchBuf;
|
||
|
if (plDestLen)
|
||
|
*plDestLen = lLockLen;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (pszBuf)
|
||
|
{
|
||
|
if (fDirectLock)
|
||
|
pOwner->GetMLStrBufW()->UnlockBuf(pszBuf, 0, 0);
|
||
|
else
|
||
|
pOwner->MemFree(pszBuf);
|
||
|
}
|
||
|
|
||
|
if (ppszDest)
|
||
|
*ppszDest = NULL;
|
||
|
if (pcchDest)
|
||
|
*pcchDest = 0;
|
||
|
if (plDestLen)
|
||
|
*plDestLen = 0;
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::UnlockWStr(const WCHAR* pszSrc, long cchSrc, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
ASSERT_READ_BLOCK(pszSrc, cchSrc);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(pcchActual);
|
||
|
ASSERT_WRITE_PTR_OR_NULL(plActualLen);
|
||
|
|
||
|
return GetOwner()->UnlockStrCommon(pszSrc, cchSrc, pcchActual, plActualLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetLocale(long lDestPos, long lDestLen, LCID locale)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->SetLocale(lDestPos, lDestLen, locale);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetLocale(long lSrcPos, long lSrcMaxLen, LCID* plocale, long* plLocalePos, long* plLocaleLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->GetLocale(lSrcPos, lSrcMaxLen, plocale, plLocalePos, plLocaleLen);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#else // NEWMLSTR
|
||
|
|
||
|
#include "mlstr.h"
|
||
|
|
||
|
|
||
|
// CMLStrW
|
||
|
|
||
|
CMLStrW::CMLStrW(void) :
|
||
|
m_pAttrWStr(NULL),
|
||
|
m_pAttrLocale(NULL),
|
||
|
m_dwAttrWStrCookie(NULL),
|
||
|
m_dwAttrLocaleCookie(NULL)
|
||
|
{
|
||
|
::InitializeCriticalSection(&m_cs);
|
||
|
}
|
||
|
|
||
|
CMLStrW::~CMLStrW(void)
|
||
|
{
|
||
|
if (m_dwAttrLocaleCookie)
|
||
|
GetOwner()->UnregisterAttr(m_dwAttrLocaleCookie);
|
||
|
if (m_dwAttrWStrCookie)
|
||
|
GetOwner()->UnregisterAttr(m_dwAttrWStrCookie);
|
||
|
if (m_pAttrLocale)
|
||
|
m_pAttrLocale->Release();
|
||
|
if (m_pAttrWStr)
|
||
|
m_pAttrWStr->Release();
|
||
|
|
||
|
::DeleteCriticalSection(&m_cs);
|
||
|
}
|
||
|
|
||
|
HRESULT CMLStrW::GetAttrWStrReal(IMLStrAttrWStr** ppAttr)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
::EnterCriticalSection(&m_cs);
|
||
|
|
||
|
if (!m_pAttrWStr)
|
||
|
{
|
||
|
IMLStrAttrWStr* pAttr;
|
||
|
|
||
|
hr = ::CoCreateInstance(CLSID_CMLStrAttrWStr, NULL, CLSCTX_ALL, IID_IUnknown, (void**)&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = GetOwner()->RegisterAttr(pAttr, &m_dwAttrWStrCookie);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
pAttr->Release();
|
||
|
|
||
|
hr = GetOwner()->FindAttr(IID_IMLStrAttrWStr, 0, (IUnknown**)&pAttr);
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
m_pAttrWStr = pAttr;
|
||
|
}
|
||
|
|
||
|
if (ppAttr)
|
||
|
*ppAttr = m_pAttrWStr;
|
||
|
|
||
|
::LeaveCriticalSection(&m_cs);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CMLStrW::GetAttrLocaleReal(IMLStrAttrLocale** ppAttr)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
::EnterCriticalSection(&m_cs);
|
||
|
|
||
|
if (!m_pAttrLocale)
|
||
|
{
|
||
|
IMLStrAttrLocale* pAttr;
|
||
|
|
||
|
hr = ::CoCreateInstance(CLSID_CMLStrAttrLocale, NULL, CLSCTX_ALL, IID_IUnknown, (void**)&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = GetOwner()->RegisterAttr(pAttr, &m_dwAttrLocaleCookie);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
pAttr->Release();
|
||
|
|
||
|
hr = GetOwner()->FindAttr(IID_IMLStrAttrLocale, 0, (IUnknown**)&pAttr);
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
m_pAttrLocale = pAttr;
|
||
|
}
|
||
|
|
||
|
if (ppAttr)
|
||
|
*ppAttr = m_pAttrLocale;
|
||
|
|
||
|
::LeaveCriticalSection(&m_cs);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::LockMLStr(long lPos, long lLen, DWORD dwFlags, DWORD* pdwCookie, long* plActualPos, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->LockMLStr(lPos, lLen, dwFlags, pdwCookie, plActualPos, plActualLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::UnlockMLStr(DWORD dwCookie)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->UnlockMLStr(dwCookie);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetLength(long* plLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->GetLength(plLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetMLStr(long lDestPos, long lDestLen, IUnknown* pSrcMLStr, long lSrcPos, long lSrcLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->SetMLStr(lDestPos, lDestLen, pSrcMLStr, lSrcPos, lSrcLen);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::RegisterAttr(IUnknown* pUnk, DWORD* pdwCookie)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->RegisterAttr(pUnk, pdwCookie);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::UnregisterAttr(DWORD dwCookie)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->UnregisterAttr(dwCookie);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::EnumAttr(IEnumUnknown** ppEnumUnk)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->EnumAttr(ppEnumUnk);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::FindAttr(REFIID riid, LPARAM lParam, IUnknown** ppUnk)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
return GetOwner()->FindAttr(riid, lParam, ppUnk);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetWStr(long lDestPos, long lDestLen, const WCHAR* pszSrc, long cchSrc, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrWStr* pAttr;
|
||
|
HRESULT hr = GetAttrWStr(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->SetWStr(lDestPos, lDestLen, pszSrc, cchSrc, pcchActual, plActualLen);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetStrBufW(long lDestPos, long lDestLen, IMLangStringBufW* pSrcBuf, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrWStr* pAttr;
|
||
|
HRESULT hr = GetAttrWStr(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->SetStrBufW(lDestPos, lDestLen, pSrcBuf, pcchActual, plActualLen);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetWStr(long lSrcPos, long lSrcLen, WCHAR* pszDest, long cchDest, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrWStr* pAttr;
|
||
|
HRESULT hr = GetAttrWStr(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->GetWStr(lSrcPos, lSrcLen, pszDest, cchDest, pcchActual, plActualLen);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetStrBufW(long lSrcPos, long lSrcMaxLen, IMLangStringBufW** ppDestBuf, long* plDestLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrWStr* pAttr;
|
||
|
HRESULT hr = GetAttrWStr(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->GetStrBufW(lSrcPos, lSrcMaxLen, ppDestBuf, plDestLen);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::LockWStr(long lSrcPos, long lSrcLen, long lFlags, long cchRequest, WCHAR** ppszDest, long* pcchDest, long* plDestLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrWStr* pAttr;
|
||
|
HRESULT hr = GetAttrWStr(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->LockWStr(lSrcPos, lSrcLen, lFlags, cchRequest, ppszDest, pcchDest, plDestLen);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::UnlockWStr(const WCHAR* pszSrc, long cchSrc, long* pcchActual, long* plActualLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrWStr* pAttr;
|
||
|
HRESULT hr = GetAttrWStr(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->UnlockWStr(pszSrc, cchSrc, pcchActual, plActualLen);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::SetLocale(long lDestPos, long lDestLen, LCID locale)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrLocale* pAttr;
|
||
|
HRESULT hr = GetAttrLocale(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->SetLong(lDestPos, lDestLen, (long)locale);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CMLStrW::GetLocale(long lSrcPos, long lSrcMaxLen, LCID* plocale, long* plLocalePos, long* plLocaleLen)
|
||
|
{
|
||
|
ASSERT_THIS;
|
||
|
|
||
|
IMLStrAttrLocale* pAttr;
|
||
|
HRESULT hr = GetAttrLocale(&pAttr);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pAttr->GetLong(lSrcPos, lSrcMaxLen, (long*)plocale, plLocalePos, plLocaleLen);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
#endif // NEWMLSTR
|