home *** CD-ROM | disk | FTP | other *** search
- // This is a part of the Microsoft Foundation Classes C++ library.
- // Copyright (C) 1992-1998 Microsoft Corporation
- // All rights reserved.
- //
- // This source code is only intended as a supplement to the
- // Microsoft Foundation Classes Reference and related
- // electronic documentation provided with the library.
- // See these sources for detailed information regarding the
- // Microsoft Foundation Classes product.
-
- #include "stdafx.h"
- #include <afxtempl.h>
- #include <math.h>
-
- #ifdef _DEBUG
- #undef THIS_FILE
- static char THIS_FILE[] = __FILE__;
- #endif
-
- /////////////////////////////////////////////////////////////////////////////
- // helpers
-
- void AFXAPI AfxCheckError(SCODE sc)
- {
- if (FAILED(sc))
- {
- if (sc == E_OUTOFMEMORY)
- AfxThrowMemoryException();
- else
- AfxThrowOleException(sc);
- }
- }
-
- AFX_STATIC BOOL AFXAPI _AfxCompareSafeArrays(SAFEARRAY* parray1, SAFEARRAY* parray2)
- {
- BOOL bCompare = FALSE;
-
- // If one is NULL they must both be NULL to compare
- if (parray1 == NULL || parray2 == NULL)
- {
- return parray1 == parray2;
- }
-
- // Dimension must match and if 0, then arrays compare
- DWORD dwDim1 = ::SafeArrayGetDim(parray1);
- DWORD dwDim2 = ::SafeArrayGetDim(parray2);
- if (dwDim1 != dwDim2)
- return FALSE;
- else if (dwDim1 == 0)
- return TRUE;
-
- // Element size must match
- DWORD dwSize1 = ::SafeArrayGetElemsize(parray1);
- DWORD dwSize2 = ::SafeArrayGetElemsize(parray2);
- if (dwSize1 != dwSize2)
- return FALSE;
-
- long* pLBound1 = NULL;
- long* pLBound2 = NULL;
- long* pUBound1 = NULL;
- long* pUBound2 = NULL;
-
- void* pData1 = NULL;
- void* pData2 = NULL;
-
- TRY
- {
- // Bounds must match
- pLBound1 = new long[dwDim1];
- pLBound2 = new long[dwDim2];
- pUBound1 = new long[dwDim1];
- pUBound2 = new long[dwDim2];
-
- size_t nTotalElements = 1;
-
- // Get and compare bounds
- for (DWORD dwIndex = 0; dwIndex < dwDim1; dwIndex++)
- {
- AfxCheckError(::SafeArrayGetLBound(
- parray1, dwIndex+1, &pLBound1[dwIndex]));
- AfxCheckError(::SafeArrayGetLBound(
- parray2, dwIndex+1, &pLBound2[dwIndex]));
- AfxCheckError(::SafeArrayGetUBound(
- parray1, dwIndex+1, &pUBound1[dwIndex]));
- AfxCheckError(::SafeArrayGetUBound(
- parray2, dwIndex+1, &pUBound2[dwIndex]));
-
- // Check the magnitude of each bound
- if (pUBound1[dwIndex] - pLBound1[dwIndex] !=
- pUBound2[dwIndex] - pLBound2[dwIndex])
- {
- delete[] pLBound1;
- delete[] pLBound2;
- delete[] pUBound1;
- delete[] pUBound2;
-
- return FALSE;
- }
-
- // Increment the element count
- nTotalElements *= pUBound1[dwIndex] - pLBound1[dwIndex] + 1;
- }
-
- // Access the data
- AfxCheckError(::SafeArrayAccessData(parray1, &pData1));
- AfxCheckError(::SafeArrayAccessData(parray2, &pData2));
-
- // Calculate the number of bytes of data and compare
- size_t nSize = nTotalElements * dwSize1;
- int nOffset = memcmp(pData1, pData2, nSize);
- bCompare = nOffset == 0;
-
- // Release the array locks
- AfxCheckError(::SafeArrayUnaccessData(parray1));
- AfxCheckError(::SafeArrayUnaccessData(parray2));
- }
- CATCH_ALL(e)
- {
- // Clean up bounds arrays
- delete[] pLBound1;
- delete[] pLBound2;
- delete[] pUBound1;
- delete[] pUBound2;
-
- // Release the array locks
- if (pData1 != NULL)
- AfxCheckError(::SafeArrayUnaccessData(parray1));
- if (pData2 != NULL)
- AfxCheckError(::SafeArrayUnaccessData(parray2));
-
- THROW_LAST();
- }
- END_CATCH_ALL
-
- // Clean up bounds arrays
- delete[] pLBound1;
- delete[] pLBound2;
- delete[] pUBound1;
- delete[] pUBound2;
-
- return bCompare;
- }
-
- AFX_STATIC void AFXAPI _AfxCreateOneDimArray(VARIANT& varSrc, DWORD dwSize)
- {
- UINT nDim;
-
- // Clear VARIANT and re-create SafeArray if necessary
- if (varSrc.vt != (VT_UI1 | VT_ARRAY) ||
- (nDim = ::SafeArrayGetDim(varSrc.parray)) != 1)
- {
- VERIFY(::VariantClear(&varSrc) == NOERROR);
- varSrc.vt = VT_UI1 | VT_ARRAY;
-
- SAFEARRAYBOUND bound;
- bound.cElements = dwSize;
- bound.lLbound = 0;
- varSrc.parray = ::SafeArrayCreate(VT_UI1, 1, &bound);
- if (varSrc.parray == NULL)
- AfxThrowMemoryException();
- }
- else
- {
- // Must redimension array if necessary
- long lLower, lUpper;
- AfxCheckError(::SafeArrayGetLBound(varSrc.parray, 1, &lLower));
- AfxCheckError(::SafeArrayGetUBound(varSrc.parray, 1, &lUpper));
-
- // Upper bound should always be greater than lower bound
- long lSize = lUpper - lLower;
- if (lSize < 0)
- {
- ASSERT(FALSE);
- lSize = 0;
-
- }
-
- if ((DWORD)lSize != dwSize)
- {
- SAFEARRAYBOUND bound;
- bound.cElements = dwSize;
- bound.lLbound = lLower;
- AfxCheckError(::SafeArrayRedim(varSrc.parray, &bound));
- }
- }
- }
-
- AFX_STATIC void AFXAPI _AfxCopyBinaryData(SAFEARRAY* parray, const void* pvSrc, DWORD dwSize)
- {
- // Access the data, copy it and unaccess it.
- void* pDest;
- AfxCheckError(::SafeArrayAccessData(parray, &pDest));
- memcpy(pDest, pvSrc, dwSize);
- AfxCheckError(::SafeArrayUnaccessData(parray));
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // COleVariant class
-
- COleVariant::COleVariant(const VARIANT& varSrc)
- {
- AfxVariantInit(this);
- AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
- }
-
- COleVariant::COleVariant(LPCVARIANT pSrc)
- {
- AfxVariantInit(this);
- AfxCheckError(::VariantCopy(this, (LPVARIANT)pSrc));
- }
-
- COleVariant::COleVariant(const COleVariant& varSrc)
- {
- AfxVariantInit(this);
- AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
- }
-
- COleVariant::COleVariant(LPCTSTR lpszSrc, VARTYPE vtSrc)
- {
- USES_CONVERSION;
- ASSERT(vtSrc == VT_BSTR || vtSrc == VT_BSTRT);
- UNUSED(vtSrc);
-
- vt = VT_BSTR;
- bstrVal = NULL;
-
- if (lpszSrc != NULL)
- {
- #ifndef _UNICODE
- if (vtSrc == VT_BSTRT)
- {
- int nLen = lstrlen(lpszSrc);
- bstrVal = ::SysAllocStringByteLen(lpszSrc, nLen);
- }
- else
- #endif
- {
- bstrVal = ::SysAllocString(T2COLE(lpszSrc));
- }
-
- if (bstrVal == NULL)
- AfxThrowMemoryException();
- }
- }
-
-
-
- void COleVariant::SetString(LPCTSTR lpszSrc, VARTYPE vtSrc)
- {
- USES_CONVERSION;
- ASSERT(vtSrc == VT_BSTR || vtSrc == VT_BSTRT);
- UNUSED(vtSrc);
-
- // Free up previous VARIANT
- Clear();
-
- vt = VT_BSTR;
- bstrVal = NULL;
-
- if (lpszSrc != NULL)
- {
- #ifndef _UNICODE
- if (vtSrc == VT_BSTRT)
- {
- int nLen = lstrlen(lpszSrc);
- bstrVal = ::SysAllocStringByteLen(lpszSrc, nLen);
- }
- else
- #endif
- {
- bstrVal = ::SysAllocString(T2COLE(lpszSrc));
- }
-
- if (bstrVal == NULL)
- AfxThrowMemoryException();
- }
- }
-
- COleVariant::COleVariant(short nSrc, VARTYPE vtSrc)
- {
- ASSERT(vtSrc == VT_I2 || vtSrc == VT_BOOL);
-
- if (vtSrc == VT_BOOL)
- {
- vt = VT_BOOL;
- if (!nSrc)
- V_BOOL(this) = AFX_OLE_FALSE;
- else
- V_BOOL(this) = AFX_OLE_TRUE;
- }
- else
- {
- vt = VT_I2;
- iVal = nSrc;
- }
- }
-
- COleVariant::COleVariant(long lSrc, VARTYPE vtSrc)
- {
- ASSERT(vtSrc == VT_I4 || vtSrc == VT_ERROR || vtSrc == VT_BOOL);
-
- if (vtSrc == VT_ERROR)
- {
- vt = VT_ERROR;
- scode = lSrc;
- }
- else if (vtSrc == VT_BOOL)
- {
- vt = VT_BOOL;
- if (!lSrc)
- V_BOOL(this) = AFX_OLE_FALSE;
- else
- V_BOOL(this) = AFX_OLE_TRUE;
- }
- else
- {
- vt = VT_I4;
- lVal = lSrc;
- }
- }
-
- // Operations
- void COleVariant::_ClearCompat()
- {
- Clear();
- }
-
- void COleVariant::ChangeType(VARTYPE vartype, LPVARIANT pSrc)
- {
- // If pSrc is NULL, convert type in place
- if (pSrc == NULL)
- pSrc = this;
- if (pSrc != this || vartype != vt)
- AfxCheckError(::VariantChangeType(this, pSrc, 0, vartype));
- }
-
- void COleVariant::Attach(VARIANT& varSrc)
- {
- // Free up previous VARIANT
- Clear();
-
- // give control of data to COleVariant
- memcpy(this, &varSrc, sizeof(varSrc));
- varSrc.vt = VT_EMPTY;
- }
-
- VARIANT COleVariant::Detach()
- {
- VARIANT varResult = *this;
- vt = VT_EMPTY;
- return varResult;
- }
-
- // Literal comparison. Types and values must match.
- BOOL COleVariant::operator==(const VARIANT& var) const
- {
- if (&var == this)
- return TRUE;
-
- // Variants not equal if types don't match
- if (var.vt != vt)
- return FALSE;
-
- // Check type specific values
- switch (vt)
- {
- case VT_EMPTY:
- case VT_NULL:
- return TRUE;
-
- case VT_BOOL:
- return V_BOOL(&var) == V_BOOL(this);
-
- case VT_UI1:
- return var.bVal == bVal;
-
- case VT_I2:
- return var.iVal == iVal;
-
- case VT_I4:
- return var.lVal == lVal;
-
- case VT_CY:
- return (var.cyVal.Hi == cyVal.Hi && var.cyVal.Lo == cyVal.Lo);
-
- case VT_R4:
- return var.fltVal == fltVal;
-
- case VT_R8:
- return var.dblVal == dblVal;
-
- case VT_DATE:
- return var.date == date;
-
- case VT_BSTR:
- return SysStringByteLen(var.bstrVal) == SysStringByteLen(bstrVal) &&
- memcmp(var.bstrVal, bstrVal, SysStringByteLen(bstrVal)) == 0;
-
- case VT_ERROR:
- return var.scode == scode;
-
- case VT_DISPATCH:
- case VT_UNKNOWN:
- return var.punkVal == punkVal;
-
- default:
- if (vt & VT_ARRAY && !(vt & VT_BYREF))
- return _AfxCompareSafeArrays(var.parray, parray);
- else
- ASSERT(FALSE); // VT_BYREF not supported
- // fall through
- }
-
- return FALSE;
- }
-
- const COleVariant& COleVariant::operator=(const VARIANT& varSrc)
- {
- AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(LPCVARIANT pSrc)
- {
- AfxCheckError(::VariantCopy(this, (LPVARIANT)pSrc));
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(const COleVariant& varSrc)
- {
- AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(const LPCTSTR lpszSrc)
- {
- USES_CONVERSION;
- // Free up previous VARIANT
- Clear();
-
- vt = VT_BSTR;
- if (lpszSrc == NULL)
- bstrVal = NULL;
- else
- {
- bstrVal = ::SysAllocString(T2COLE(lpszSrc));
- if (bstrVal == NULL)
- AfxThrowMemoryException();
- }
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(const CString& strSrc)
- {
- USES_CONVERSION;
- // Free up previous VARIANT
- Clear();
-
- vt = VT_BSTR;
- bstrVal = ::SysAllocString(T2COLE(strSrc));
- if (bstrVal == NULL)
- AfxThrowMemoryException();
-
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(BYTE nSrc)
- {
- // Free up previous VARIANT if necessary
- if (vt != VT_UI1)
- {
- Clear();
- vt = VT_UI1;
- }
-
- bVal = nSrc;
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(short nSrc)
- {
- if (vt == VT_I2)
- iVal = nSrc;
- else if (vt == VT_BOOL)
- {
- if (!nSrc)
- V_BOOL(this) = AFX_OLE_FALSE;
- else
- V_BOOL(this) = AFX_OLE_TRUE;
- }
- else
- {
- // Free up previous VARIANT
- Clear();
- vt = VT_I2;
- iVal = nSrc;
- }
-
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(long lSrc)
- {
- if (vt == VT_I4)
- lVal = lSrc;
- else if (vt == VT_ERROR)
- scode = lSrc;
- else if (vt == VT_BOOL)
- {
- if (!lSrc)
- V_BOOL(this) = AFX_OLE_FALSE;
- else
- V_BOOL(this) = AFX_OLE_TRUE;
- }
- else
- {
- // Free up previous VARIANT
- Clear();
- vt = VT_I4;
- lVal = lSrc;
- }
-
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(const COleCurrency& curSrc)
- {
- // Free up previous VARIANT if necessary
- if (vt != VT_CY)
- {
- Clear();
- vt = VT_CY;
- }
-
- cyVal = curSrc.m_cur;
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(float fltSrc)
- {
- // Free up previous VARIANT if necessary
- if (vt != VT_R4)
- {
- Clear();
- vt = VT_R4;
- }
-
- fltVal = fltSrc;
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(double dblSrc)
- {
- // Free up previous VARIANT if necessary
- if (vt != VT_R8)
- {
- Clear();
- vt = VT_R8;
- }
-
- dblVal = dblSrc;
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(const COleDateTime& dateSrc)
- {
- // Free up previous VARIANT if necessary
- if (vt != VT_DATE)
- {
- Clear();
- vt = VT_DATE;
- }
-
- date = dateSrc.m_dt;
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(const CByteArray& arrSrc)
- {
- int nSize = arrSrc.GetSize();
-
- // Set the correct type and make sure SafeArray can hold data
- _AfxCreateOneDimArray(*this, (DWORD)nSize);
-
- // Copy the data into the SafeArray
- _AfxCopyBinaryData(parray, arrSrc.GetData(), (DWORD)nSize);
-
- return *this;
- }
-
- const COleVariant& COleVariant::operator=(const CLongBinary& lbSrc)
- {
- // Set the correct type and make sure SafeArray can hold data
- _AfxCreateOneDimArray(*this, lbSrc.m_dwDataLength);
-
- // Copy the data into the SafeArray
- BYTE* pData = (BYTE*)::GlobalLock(lbSrc.m_hData);
- _AfxCopyBinaryData(parray, pData, lbSrc.m_dwDataLength);
- ::GlobalUnlock(lbSrc.m_hData);
-
- return *this;
- }
-
- void AFXAPI AfxVariantInit(LPVARIANT pVar)
- {
- memset(pVar, 0, sizeof(*pVar));
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // Diagnostics
-
- #ifdef _DEBUG
- CDumpContext& AFXAPI operator <<(CDumpContext& dc, COleVariant varSrc)
- {
- LPCVARIANT pSrc = (LPCVARIANT)varSrc;
-
- dc << "\nCOleVariant Object:";
- dc << "\n\t vt = " << pSrc->vt;
-
- // No support for VT_BYREF & VT_ARRAY
- if (pSrc->vt & VT_BYREF || pSrc->vt & VT_ARRAY)
- return dc;
-
- switch (pSrc->vt)
- {
- case VT_BOOL:
- return dc << "\n\t VT_BOOL = " << V_BOOL(pSrc);
-
- case VT_UI1:
- return dc << "\n\t bVal = " << pSrc->bVal;
-
- case VT_I2:
- return dc << "\n\t iVal = " << pSrc->iVal;
-
- case VT_I4:
- return dc << "\n\t lVal = " << pSrc->lVal;
-
- case VT_CY:
- {
- COleVariant var(varSrc);
- var.ChangeType(VT_BSTR);
- return dc << "\n\t cyVal = " << var.bstrVal;
- }
-
- case VT_R4:
- return dc << "\n\t fltVal = " << pSrc->fltVal;
-
- case VT_R8:
- return dc << "\n\t dblVal = " << pSrc->dblVal;
-
- case VT_DATE:
- {
- COleVariant var(varSrc);
- var.ChangeType(VT_BSTR);
- return dc << "\n\t date = " << var.bstrVal;
- }
-
- case VT_BSTR:
- return dc << "\n\t bstrVal = " << pSrc->bstrVal;
-
- case VT_ERROR:
- return dc << "\n\t scode = " << pSrc->scode;
-
- case VT_DISPATCH:
- case VT_UNKNOWN:
- return dc << "\n\t punkVal = " << pSrc->punkVal;
-
- case VT_EMPTY:
- case VT_NULL:
- return dc;
-
- default:
- ASSERT(FALSE);
- return dc;
- }
- }
- #endif // _DEBUG
-
- CArchive& AFXAPI operator<<(CArchive& ar, COleVariant varSrc)
- {
- LPCVARIANT pSrc = (LPCVARIANT)varSrc;
-
- ar << pSrc->vt;
-
- // No support for VT_BYREF & VT_ARRAY
- if (pSrc->vt & VT_BYREF || pSrc->vt & VT_ARRAY)
- return ar;
-
- switch (pSrc->vt)
- {
- case VT_BOOL:
- return ar << (WORD)V_BOOL(pSrc);
-
- case VT_UI1:
- return ar << pSrc->bVal;
-
- case VT_I2:
- return ar << (WORD)pSrc->iVal;
-
- case VT_I4:
- return ar << pSrc->lVal;
-
- case VT_CY:
- ar << pSrc->cyVal.Lo;
- return ar << pSrc->cyVal.Hi;
-
- case VT_R4:
- return ar << pSrc->fltVal;
-
- case VT_R8:
- return ar << pSrc->dblVal;
-
- case VT_DATE:
- return ar << pSrc->date;
-
- case VT_BSTR:
- {
- DWORD nLen = SysStringByteLen(pSrc->bstrVal);
- ar << nLen;
- if (nLen > 0)
- ar.Write(pSrc->bstrVal, nLen * sizeof(BYTE));
-
- return ar;
- }
-
- case VT_ERROR:
- return ar << pSrc->scode;
-
- case VT_DISPATCH:
- case VT_UNKNOWN:
- {
- LPPERSISTSTREAM pPersistStream;
- CArchiveStream stm(&ar);
-
- // QI for IPersistStream or IPeristStreamInit
- SCODE sc = pSrc->punkVal->QueryInterface(
- IID_IPersistStream, (void**)&pPersistStream);
- #ifndef _AFX_NO_OCC_SUPPORT
- if (FAILED(sc))
- sc = pSrc->punkVal->QueryInterface(
- IID_IPersistStreamInit, (void**)&pPersistStream);
- #endif
- AfxCheckError(sc);
-
- TRY
- {
- // Get and archive the CLSID (GUID)
- CLSID clsid;
- AfxCheckError(pPersistStream->GetClassID(&clsid));
- ar << clsid.Data1;
- ar << clsid.Data2;
- ar << clsid.Data3;
- ar.Write(&clsid.Data4[0], sizeof clsid.Data4);
-
- // Always assume object is dirty
- AfxCheckError(pPersistStream->Save(&stm, TRUE));
- }
- CATCH_ALL(e)
- {
- pPersistStream->Release();
- THROW_LAST();
- }
- END_CATCH_ALL
- pPersistStream->Release();
- }
- return ar;
-
- case VT_EMPTY:
- case VT_NULL:
- // do nothing
- return ar;
-
- default:
- ASSERT(FALSE);
- return ar;
- }
- }
-
- CArchive& AFXAPI operator>>(CArchive& ar, COleVariant& varSrc)
- {
- LPVARIANT pSrc = &varSrc;
-
- // Free up current data if necessary
- if (pSrc->vt != VT_EMPTY)
- VariantClear(pSrc);
- ar >> pSrc->vt;
-
- // No support for VT_BYREF & VT_ARRAY
- if (pSrc->vt & VT_BYREF || pSrc->vt & VT_ARRAY)
- return ar;
-
- switch (pSrc->vt)
- {
- case VT_BOOL:
- return ar >> (WORD&)V_BOOL(pSrc);
-
- case VT_UI1:
- return ar >> pSrc->bVal;
-
- case VT_I2:
- return ar >> (WORD&)pSrc->iVal;
-
- case VT_I4:
- return ar >> pSrc->lVal;
-
- case VT_CY:
- ar >> pSrc->cyVal.Lo;
- return ar >> pSrc->cyVal.Hi;
-
- case VT_R4:
- return ar >> pSrc->fltVal;
-
- case VT_R8:
- return ar >> pSrc->dblVal;
-
- case VT_DATE:
- return ar >> pSrc->date;
-
- case VT_BSTR:
- {
- DWORD nLen;
- ar >> nLen;
- if (nLen > 0)
- {
- pSrc->bstrVal = SysAllocStringByteLen(NULL, nLen);
- if (pSrc->bstrVal == NULL)
- AfxThrowMemoryException();
- ar.Read(pSrc->bstrVal, nLen * sizeof(BYTE));
- }
- else
- pSrc->bstrVal = NULL;
-
- return ar;
- }
- break;
-
- case VT_ERROR:
- return ar >> pSrc->scode;
-
- case VT_DISPATCH:
- case VT_UNKNOWN:
- {
- LPPERSISTSTREAM pPersistStream = NULL;
- CArchiveStream stm(&ar);
-
- // Retrieve the CLSID (GUID) and create an instance
- CLSID clsid;
- ar >> clsid.Data1;
- ar >> clsid.Data2;
- ar >> clsid.Data3;
- ar.Read(&clsid.Data4[0], sizeof clsid.Data4);
-
- // Create the object
- SCODE sc = CoCreateInstance(clsid, NULL, CLSCTX_ALL | CLSCTX_REMOTE_SERVER,
- pSrc->vt == VT_UNKNOWN ? IID_IUnknown : IID_IDispatch,
- (void**)&pSrc->punkVal);
- if (sc == E_INVALIDARG)
- {
- // may not support CLSCTX_REMOTE_SERVER, so try without
- sc = CoCreateInstance(clsid, NULL,
- CLSCTX_ALL & ~CLSCTX_REMOTE_SERVER,
- pSrc->vt == VT_UNKNOWN ? IID_IUnknown : IID_IDispatch,
- (void**)&pSrc->punkVal);
- }
- AfxCheckError(sc);
-
- TRY
- {
- // QI for IPersistStream or IPeristStreamInit
- sc = pSrc->punkVal->QueryInterface(
- IID_IPersistStream, (void**)&pPersistStream);
- #ifndef _AFX_NO_OCC_SUPPORT
- if (FAILED(sc))
- sc = pSrc->punkVal->QueryInterface(
- IID_IPersistStreamInit, (void**)&pPersistStream);
- #endif
- AfxCheckError(sc);
-
- // Always assumes object is dirty
- AfxCheckError(pPersistStream->Load(&stm));
- }
- CATCH_ALL(e)
- {
- // Clean up
- if (pPersistStream != NULL)
- pPersistStream->Release();
-
- pSrc->punkVal->Release();
- THROW_LAST();
- }
- END_CATCH_ALL
-
- pPersistStream->Release();
- }
- return ar;
-
- case VT_EMPTY:
- case VT_NULL:
- // do nothing
- return ar;
-
- default:
- ASSERT(FALSE);
- return ar;
- }
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // COleVariant Helpers
-
- #if _MSC_VER >= 1100
- template <> void AFXAPI ConstructElements<COleVariant> (COleVariant* pElements, int nCount)
- #else
- void AFXAPI ConstructElements(COleVariant* pElements, int nCount)
- #endif
- {
- ASSERT(nCount == 0 ||
- AfxIsValidAddress(pElements, nCount * sizeof(COleVariant)));
-
- for (; nCount--; ++pElements)
- new(pElements) COleVariant;
- }
-
- #if _MSC_VER >= 1100
- template <> void AFXAPI DestructElements<COleVariant> (COleVariant* pElements, int nCount)
- #else
- void AFXAPI DestructElements(COleVariant* pElements, int nCount)
- #endif
- {
- ASSERT(nCount == 0 ||
- AfxIsValidAddress(pElements, nCount * sizeof(COleVariant)));
-
- for (; nCount--; ++pElements)
- pElements->~COleVariant();
- }
-
- #if _MSC_VER >= 1100
- template <> void AFXAPI CopyElements<COleVariant> (COleVariant* pDest, const COleVariant* pSrc, int nCount)
- #else
- void AFXAPI CopyElements(COleVariant* pDest, const COleVariant* pSrc, int nCount)
- #endif
- {
- ASSERT(nCount == 0 ||
- AfxIsValidAddress(pDest, nCount * sizeof(COleVariant)));
- ASSERT(nCount == 0 ||
- AfxIsValidAddress(pSrc, nCount * sizeof(COleVariant)));
-
- for (; nCount--; ++pDest, ++pSrc)
- *pDest = *pSrc;
- }
-
- #if _MSC_VER >= 1100
- template <> void AFXAPI SerializeElements<COleVariant> (CArchive& ar, COleVariant* pElements, int nCount)
- #else
- void AFXAPI SerializeElements(CArchive& ar, COleVariant* pElements, int nCount)
- #endif
- {
- ASSERT(nCount == 0 ||
- AfxIsValidAddress(pElements, nCount * sizeof(COleVariant)));
-
- if (ar.IsStoring())
- {
- for (; nCount--; ++pElements)
- ar << *pElements;
- }
- else
- {
- for (; nCount--; ++pElements)
- ar >> *pElements;
- }
- }
-
- #ifdef _DEBUG
- #if _MSC_VER >= 1100
- template <> void AFXAPI DumpElements<COleVariant> (CDumpContext& dc, const COleVariant* pElements, int nCount)
- #else
- void AFXAPI DumpElements(CDumpContext& dc, const COleVariant* pElements, int nCount)
- #endif
- {
- for (; nCount--; ++pElements)
- dc << *pElements;
- }
- #endif // _DEBUG
-
- #if _MSC_VER >= 1100
- template<> UINT AFXAPI HashKey<const struct tagVARIANT&> (const struct tagVARIANT& var)
- #else
- UINT AFXAPI HashKey(const struct tagVARIANT& var)
- #endif
- {
- switch (var.vt)
- {
- case VT_EMPTY:
- case VT_NULL:
- return 0;
- case VT_I2:
- #if _MSC_VER >= 1100
- return HashKey<DWORD>((DWORD)var.iVal);
- #else
- return HashKey((DWORD)var.iVal);
- #endif
- case VT_I4:
- #if _MSC_VER >= 1100
- return HashKey<DWORD>((DWORD)var.lVal);
- #else
- return HashKey((DWORD)var.lVal);
- #endif
- case VT_R4:
- return (UINT)(var.fltVal / 16);
- case VT_R8:
- case VT_CY:
- return (UINT)(var.dblVal / 16);
- case VT_BOOL:
- #if _MSC_VER >= 1100
- return HashKey<DWORD>((DWORD)V_BOOL(&var));
- #else
- return HashKey((DWORD)V_BOOL(&var));
- #endif
- case VT_ERROR:
- #if _MSC_VER >= 1100
- return HashKey<DWORD>((DWORD)var.scode);
- #else
- return HashKey((DWORD)var.scode);
- #endif
- case VT_DATE:
- return (UINT)(var.date / 16);
- case VT_BSTR:
- #if _MSC_VER >= 1100
- return HashKey<LPCOLESTR>(var.bstrVal);
- #else
- return HashKey((LPCOLESTR)var.bstrVal);
- #endif
- case VT_DISPATCH:
- case VT_UNKNOWN:
- #if _MSC_VER >= 1100
- return HashKey<DWORD>((DWORD)var.punkVal);
- #else
- return HashKey((DWORD)var.punkVal);
- #endif
-
- default:
- // No support for VT_BYREF, VT_ARRAY, VT_VARIANT, VT_DECIMAL, & VT_UI1
- ASSERT(FALSE);
-
- // Fall through
- }
-
- return 0;
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // COleCurrency class helpers
-
- // Return the highest order bit composing dwTarget in wBit
- #define HI_BIT(dwTarget, wBit) \
- do \
- { \
- if (dwTarget != 0) \
- for (wBit = 32; (dwTarget & (0x00000001 << (wBit-1))) == 0; wBit--);\
- else \
- wBit = 0; \
- } while (0)
-
- // Left shift an (assumed unsigned) currency by wBits
- #define LSHIFT_UCUR(cur, wBits) \
- do \
- { \
- for (WORD wTempBits = wBits; wTempBits > 0; wTempBits--) \
- { \
- cur.m_cur.Hi = ((DWORD)cur.m_cur.Hi << 1); \
- cur.m_cur.Hi |= (cur.m_cur.Lo & 0x80000000) >> 31; \
- cur.m_cur.Lo = cur.m_cur.Lo << 1; \
- } \
- } while (0)
-
- // Right shift an (assumed unsigned) currency by wBits
- #define RSHIFT_UCUR(cur, wBits) \
- do \
- { \
- for (WORD wTempBits = wBits; wTempBits > 0; wTempBits--) \
- { \
- cur.m_cur.Lo = cur.m_cur.Lo >> 1; \
- cur.m_cur.Lo |= (cur.m_cur.Hi & 0x00000001) << 31; \
- cur.m_cur.Hi = ((DWORD)cur.m_cur.Hi >> 1); \
- } \
- } while (0)
-
- /////////////////////////////////////////////////////////////////////////////
- // COleCurrency class (internally currency is 8-byte int scaled by 10,000)
-
- COleCurrency::COleCurrency(long nUnits, long nFractionalUnits)
- {
- SetCurrency(nUnits, nFractionalUnits);
- SetStatus(valid);
- }
-
- const COleCurrency& COleCurrency::operator=(CURRENCY cySrc)
- {
- m_cur = cySrc;
- SetStatus(valid);
- return *this;
- }
-
- const COleCurrency& COleCurrency::operator=(const COleCurrency& curSrc)
- {
- m_cur = curSrc.m_cur;
- m_status = curSrc.m_status;
- return *this;
- }
-
- const COleCurrency& COleCurrency::operator=(const VARIANT& varSrc)
- {
- if (varSrc.vt != VT_CY)
- {
- TRY
- {
- COleVariant varTemp(varSrc);
- varTemp.ChangeType(VT_CY);
- m_cur = varTemp.cyVal;
- SetStatus(valid);
- }
- // Catch COleException from ChangeType, but not CMemoryException
- CATCH(COleException, e)
- {
- // Not able to convert VARIANT to CURRENCY
- m_cur.Hi = 0;
- m_cur.Lo = 0;
- SetStatus(invalid);
- DELETE_EXCEPTION(e);
- }
- END_CATCH
- }
- else
- {
- m_cur = varSrc.cyVal;
- SetStatus(valid);
- }
-
- return *this;
- }
-
- BOOL COleCurrency::operator<(const COleCurrency& cur) const
- {
- ASSERT(GetStatus() == valid);
- ASSERT(cur.GetStatus() == valid);
-
- return((m_cur.Hi == cur.m_cur.Hi) ?
- (m_cur.Lo < cur.m_cur.Lo) : (m_cur.Hi < cur.m_cur.Hi));
- }
-
- BOOL COleCurrency::operator>(const COleCurrency& cur) const
- {
- ASSERT(GetStatus() == valid);
- ASSERT(cur.GetStatus() == valid);
-
- return((m_cur.Hi == cur.m_cur.Hi) ?
- (m_cur.Lo > cur.m_cur.Lo) : (m_cur.Hi > cur.m_cur.Hi));
- }
-
- BOOL COleCurrency::operator<=(const COleCurrency& cur) const
- {
- ASSERT(GetStatus() == valid);
- ASSERT(cur.GetStatus() == valid);
-
- return((m_cur.Hi == cur.m_cur.Hi) ?
- (m_cur.Lo <= cur.m_cur.Lo) : (m_cur.Hi < cur.m_cur.Hi));
- }
-
- BOOL COleCurrency::operator>=(const COleCurrency& cur) const
- {
- ASSERT(GetStatus() == valid);
- ASSERT(cur.GetStatus() == valid);
-
- return((m_cur.Hi == cur.m_cur.Hi) ?
- (m_cur.Lo >= cur.m_cur.Lo) : (m_cur.Hi > cur.m_cur.Hi));
- }
-
- COleCurrency COleCurrency::operator+(const COleCurrency& cur) const
- {
- COleCurrency curResult;
-
- // If either operand Null, result Null
- if (GetStatus() == null || cur.GetStatus() == null)
- {
- curResult.SetStatus(null);
- return curResult;
- }
-
- // If either operand Invalid, result Invalid
- if (GetStatus() == invalid || cur.GetStatus() == invalid)
- {
- curResult.SetStatus(invalid);
- return curResult;
- }
-
- // Add separate CURRENCY components
- curResult.m_cur.Hi = m_cur.Hi + cur.m_cur.Hi;
- curResult.m_cur.Lo = m_cur.Lo + cur.m_cur.Lo;
-
- // Increment Hi if Lo overflows
- if (m_cur.Lo > curResult.m_cur.Lo)
- curResult.m_cur.Hi++;
-
- // Overflow if operands same sign and result sign different
- if (!((m_cur.Hi ^ cur.m_cur.Hi) & 0x80000000) &&
- ((m_cur.Hi ^ curResult.m_cur.Hi) & 0x80000000))
- {
- curResult.SetStatus(invalid);
- }
-
- return curResult;
- }
-
- COleCurrency COleCurrency::operator-(const COleCurrency& cur) const
- {
- COleCurrency curResult;
-
- // If either operand Null, result Null
- if (GetStatus() == null || cur.GetStatus() == null)
- {
- curResult.SetStatus(null);
- return curResult;
- }
-
- // If either operand Invalid, result Invalid
- if (GetStatus() == invalid || cur.GetStatus() == invalid)
- {
- curResult.SetStatus(invalid);
- return curResult;
- }
-
- // Subtract separate CURRENCY components
- curResult.m_cur.Hi = m_cur.Hi - cur.m_cur.Hi;
- curResult.m_cur.Lo = m_cur.Lo - cur.m_cur.Lo;
-
- // Decrement Hi if Lo overflows
- if (m_cur.Lo < curResult.m_cur.Lo)
- curResult.m_cur.Hi--;
-
- // Overflow if operands not same sign and result not same sign
- if (((m_cur.Hi ^ cur.m_cur.Hi) & 0x80000000) &&
- ((m_cur.Hi ^ curResult.m_cur.Hi) & 0x80000000))
- {
- curResult.SetStatus(invalid);
- }
-
- return curResult;
- }
-
- COleCurrency COleCurrency::operator-() const
- {
- // If operand not Valid, just return
- if (!GetStatus() == valid)
- return *this;
-
- COleCurrency curResult;
-
- // Negating MIN_CURRENCY,will set invalid
- if (m_cur.Hi == 0x80000000 && m_cur.Lo == 0x00000000)
- {
- curResult.SetStatus(invalid);
- }
-
- curResult.m_cur.Hi = ~m_cur.Hi;
- curResult.m_cur.Lo = -(long)m_cur.Lo;
-
- // If cy was -1 make sure Hi correctly set
- if (curResult.m_cur.Lo == 0)
- curResult.m_cur.Hi++;
-
- return curResult;
- }
-
- COleCurrency COleCurrency::operator*(long nOperand) const
- {
- // If operand not Valid, just return
- if (!GetStatus() == valid)
- return *this;
-
- COleCurrency curResult(m_cur);
- DWORD nTempOp;
-
- // Return now if one operand is 0 (optimization)
- if ((m_cur.Hi == 0x00000000 && m_cur.Lo == 0x00000000) || nOperand == 0)
- {
- curResult.m_cur.Hi = 0;
- curResult.m_cur.Lo = 0;
- return curResult;
- }
-
- // Handle only valid case of multiplying MIN_CURRENCY
- if (m_cur.Hi == 0x80000000 && m_cur.Lo == 0x00000000 && nOperand == 1)
- return curResult;
-
- // Compute absolute values.
- if (m_cur.Hi < 0)
- curResult = -curResult;
-
- nTempOp = labs(nOperand);
-
- // Check for overflow
- if (curResult.m_cur.Hi != 0)
- {
- WORD wHiBitCur, wHiBitOp;
- HI_BIT(curResult.m_cur.Hi, wHiBitCur);
- HI_BIT(nTempOp, wHiBitOp);
-
- // 63-bit limit on result. (n bits)*(m bits) = (n+m-1) bits.
- if (wHiBitCur + wHiBitOp - 1 > 63)
- {
- // Overflow!
- curResult.SetStatus(invalid);
-
- // Set to maximum negative value
- curResult.m_cur.Hi = 0x80000000;
- curResult.m_cur.Lo = 0x00000000;
-
- return curResult;
- }
- }
-
- // Break up into WORDs
- WORD wCy4, wCy3, wCy2, wCy1, wL2, wL1;
-
- wCy4 = HIWORD(curResult.m_cur.Hi);
- wCy3 = LOWORD(curResult.m_cur.Hi);
- wCy2 = HIWORD(curResult.m_cur.Lo);
- wCy1 = LOWORD(curResult.m_cur.Lo);
-
- wL2 = HIWORD(nTempOp);
- wL1 = LOWORD(nTempOp);
-
- // Multiply each set of WORDs
- DWORD dwRes11, dwRes12, dwRes21, dwRes22;
- DWORD dwRes31, dwRes32, dwRes41; // Don't need dwRes42
-
- dwRes11 = wCy1 * wL1;
- dwRes12 = wCy1 * wL2;
- dwRes21 = wCy2 * wL1;
- dwRes22 = wCy2 * wL2;
-
- dwRes31 = wCy3 * wL1;
- dwRes32 = wCy3 * wL2;
- dwRes41 = wCy4 * wL1;
-
- // Add up low order pieces
- dwRes11 += dwRes12<<16;
- curResult.m_cur.Lo = dwRes11 + (dwRes21<<16);
- curResult.m_cur.Hi = 0;
-
- // Check if carry required
- if (dwRes11 < dwRes12<<16)
- curResult.m_cur.Hi++;
- if ((DWORD)curResult.m_cur.Lo < dwRes11)
- curResult.m_cur.Hi++;
-
- // Add up the high order pieces
- curResult.m_cur.Hi += dwRes31 + (dwRes32<<16) + (dwRes41<<16) +
- dwRes22 + (dwRes12>>16) + (dwRes21>>16);
-
- // Compute result sign
- if ((m_cur.Hi ^ nOperand) & 0x80000000)
- curResult = -curResult;
-
- return curResult;
- }
-
- COleCurrency COleCurrency::operator/(long nOperand) const
- {
- // If operand not Valid, just return
- if (!GetStatus() == valid)
- return *this;
-
- COleCurrency curTemp(m_cur);
- DWORD nTempOp;
-
- // Check for divide by 0
- if (nOperand == 0)
- {
- curTemp.SetStatus(invalid);
-
- // Set to maximum negative value
- curTemp.m_cur.Hi = 0x80000000;
- curTemp.m_cur.Lo = 0x00000000;
-
- return curTemp;
- }
-
- // Compute absolute values
- if (curTemp.m_cur.Hi < 0)
- curTemp = -curTemp;
-
- nTempOp = labs(nOperand);
-
- // Optimization - division is simple if Hi == 0
- if (curTemp.m_cur.Hi == 0x0000)
- {
- curTemp.m_cur.Lo = m_cur.Lo / nTempOp;
-
- // Compute result sign
- if ((m_cur.Hi ^ nOperand) & 0x80000000)
- curTemp = -curTemp;
-
- return curTemp;
- }
-
- // Now curTemp represents remainder
- COleCurrency curResult; // Initializes to zero
- COleCurrency curTempResult;
- COleCurrency curOperand;
-
- curOperand.m_cur.Lo = nTempOp;
-
- WORD wHiBitRem;
- WORD wScaleOp;
-
- // Quit if remainder can be truncated
- while (curTemp >= curOperand)
- {
- // Scale up and divide Hi portion
- HI_BIT(curTemp.m_cur.Hi, wHiBitRem);
-
- if (wHiBitRem != 0)
- wHiBitRem += 32;
- else
- HI_BIT(curTemp.m_cur.Lo, wHiBitRem);
-
- WORD wShift = (WORD)(64 - wHiBitRem);
- LSHIFT_UCUR(curTemp, wShift);
-
- // If Operand bigger than Hi it must be scaled
- wScaleOp = (WORD)((nTempOp > (DWORD)curTemp.m_cur.Hi) ? 1 : 0);
-
- // Perform synthetic division
- curTempResult.m_cur.Hi =
- (DWORD)curTemp.m_cur.Hi / (nTempOp >> wScaleOp);
-
- // Scale back to get correct result and remainder
- RSHIFT_UCUR(curTemp, wShift);
- wShift = (WORD)(wShift - wScaleOp);
- RSHIFT_UCUR(curTempResult, wShift);
-
- // Now calculate result and remainder
- curResult += curTempResult;
- curTemp -= curTempResult * nTempOp;
- }
-
- // Compute result sign
- if ((m_cur.Hi ^ nOperand) & 0x80000000)
- curResult = -curResult;
-
- return curResult;
- }
-
- void COleCurrency::SetCurrency(long nUnits, long nFractionalUnits)
- {
- COleCurrency curUnits; // Initializes to 0
- COleCurrency curFractionalUnits; // Initializes to 0
-
- // Set temp currency value to Units (need to multiply by 10,000)
- curUnits.m_cur.Lo = (DWORD)labs(nUnits);
- curUnits = curUnits * 10000;
- if (nUnits < 0)
- curUnits = -curUnits;
-
- curFractionalUnits.m_cur.Lo = (DWORD)labs(nFractionalUnits);
- if (nFractionalUnits < 0)
- curFractionalUnits = -curFractionalUnits;
-
- // Now add together Units and FractionalUnits
- *this = curUnits + curFractionalUnits;
-
- SetStatus(valid);
- }
-
- BOOL COleCurrency::ParseCurrency(LPCTSTR lpszCurrency,
- DWORD dwFlags, LCID lcid)
- {
- USES_CONVERSION;
- CString strCurrency = lpszCurrency;
-
- SCODE sc;
- if ( FAILED(sc = VarCyFromStr((LPOLESTR)T2COLE(strCurrency),
- lcid, dwFlags, &m_cur)))
- {
- if (sc == DISP_E_TYPEMISMATCH)
- {
- // Can't convert string to CURRENCY, set 0 & invalid
- m_cur.Hi = 0x00000000;
- m_cur.Lo = 0x00000000;
- SetStatus(invalid);
- return FALSE;
- }
- else if (sc == DISP_E_OVERFLOW)
- {
- // Can't convert string to CURRENCY, set max neg & invalid
- m_cur.Hi = 0x80000000;
- m_cur.Lo = 0x00000000;
- SetStatus(invalid);
- return FALSE;
- }
- else
- {
- TRACE0("\nCOleCurrency VarCyFromStr call failed.\n\t");
- if (sc == E_OUTOFMEMORY)
- AfxThrowMemoryException();
- else
- AfxThrowOleException(sc);
- }
- }
-
- SetStatus(valid);
- return TRUE;
- }
-
- CString COleCurrency::Format(DWORD dwFlags, LCID lcid) const
- {
- USES_CONVERSION;
- CString strCur;
-
- // If null, return empty string
- if (GetStatus() == null)
- return strCur;
-
- // If invalid, return Currency resource string
- if (GetStatus() == invalid)
- {
- VERIFY(strCur.LoadString(AFX_IDS_INVALID_CURRENCY));
- return strCur;
- }
-
- COleVariant var;
- // Don't need to trap error. Should not fail due to type mismatch
- AfxCheckError(VarBstrFromCy(m_cur, lcid, dwFlags, &V_BSTR(&var)));
- var.vt = VT_BSTR;
- return OLE2CT(V_BSTR(&var));
- }
-
-
- // serialization
- #ifdef _DEBUG
- CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleCurrency curSrc)
- {
- dc << "\nCOleCurrency Object:";
- dc << "\n\tm_status = " << (long)curSrc.m_status;
-
- COleVariant var(curSrc);
- var.ChangeType(VT_CY);
- return dc << "\n\tCurrency = " << var.bstrVal;
- }
- #endif // _DEBUG
-
- CArchive& AFXAPI operator<<(CArchive& ar, COleCurrency curSrc)
- {
- ar << (long)curSrc.m_status;
- ar << curSrc.m_cur.Hi;
- return ar << curSrc.m_cur.Lo;
- }
-
- CArchive& AFXAPI operator>>(CArchive& ar, COleCurrency& curSrc)
- {
- ar >> (long&)curSrc.m_status;
- ar >> curSrc.m_cur.Hi;
- return ar >> curSrc.m_cur.Lo;
- }
-
-
- /////////////////////////////////////////////////////////////////////////////
- // COleDateTime class HELPER definitions
-
- // Verifies will fail if the needed buffer size is too large
- #define MAX_TIME_BUFFER_SIZE 128 // matches that in timecore.cpp
- #define MIN_DATE (-657434L) // about year 100
- #define MAX_DATE 2958465L // about year 9999
-
- // Half a second, expressed in days
- #define HALF_SECOND (1.0/172800.0)
-
- // One-based array of days in year at month start
- AFX_STATIC_DATA int _afxMonthDays[13] =
- {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365};
-
- /////////////////////////////////////////////////////////////////////////////
- // COleDateTime class HELPERS - implementation
-
- AFX_STATIC BOOL AFXAPI _AfxOleDateFromTm(WORD wYear, WORD wMonth, WORD wDay,
- WORD wHour, WORD wMinute, WORD wSecond, DATE& dtDest)
- {
- // Validate year and month (ignore day of week and milliseconds)
- if (wYear > 9999 || wMonth < 1 || wMonth > 12)
- return FALSE;
-
- // Check for leap year and set the number of days in the month
- BOOL bLeapYear = ((wYear & 3) == 0) &&
- ((wYear % 100) != 0 || (wYear % 400) == 0);
-
- int nDaysInMonth =
- _afxMonthDays[wMonth] - _afxMonthDays[wMonth-1] +
- ((bLeapYear && wDay == 29 && wMonth == 2) ? 1 : 0);
-
- // Finish validating the date
- if (wDay < 1 || wDay > nDaysInMonth ||
- wHour > 23 || wMinute > 59 ||
- wSecond > 59)
- {
- return FALSE;
- }
-
- // Cache the date in days and time in fractional days
- long nDate;
- double dblTime;
-
- //It is a valid date; make Jan 1, 1AD be 1
- nDate = wYear*365L + wYear/4 - wYear/100 + wYear/400 +
- _afxMonthDays[wMonth-1] + wDay;
-
- // If leap year and it's before March, subtract 1:
- if (wMonth <= 2 && bLeapYear)
- --nDate;
-
- // Offset so that 12/30/1899 is 0
- nDate -= 693959L;
-
- dblTime = (((long)wHour * 3600L) + // hrs in seconds
- ((long)wMinute * 60L) + // mins in seconds
- ((long)wSecond)) / 86400.;
-
- dtDest = (double) nDate + ((nDate >= 0) ? dblTime : -dblTime);
-
- return TRUE;
- }
-
- AFX_STATIC BOOL AFXAPI _AfxTmFromOleDate(DATE dtSrc, struct tm& tmDest)
- {
- // The legal range does not actually span year 0 to 9999.
- if (dtSrc > MAX_DATE || dtSrc < MIN_DATE) // about year 100 to about 9999
- return FALSE;
-
- long nDays; // Number of days since Dec. 30, 1899
- long nDaysAbsolute; // Number of days since 1/1/0
- long nSecsInDay; // Time in seconds since midnight
- long nMinutesInDay; // Minutes in day
-
- long n400Years; // Number of 400 year increments since 1/1/0
- long n400Century; // Century within 400 year block (0,1,2 or 3)
- long n4Years; // Number of 4 year increments since 1/1/0
- long n4Day; // Day within 4 year block
- // (0 is 1/1/yr1, 1460 is 12/31/yr4)
- long n4Yr; // Year within 4 year block (0,1,2 or 3)
- BOOL bLeap4 = TRUE; // TRUE if 4 year block includes leap year
-
- double dblDate = dtSrc; // tempory serial date
-
- // If a valid date, then this conversion should not overflow
- nDays = (long)dblDate;
-
- // Round to the second
- dblDate += ((dtSrc > 0.0) ? HALF_SECOND : -HALF_SECOND);
-
- nDaysAbsolute = (long)dblDate + 693959L; // Add days from 1/1/0 to 12/30/1899
-
- dblDate = fabs(dblDate);
- nSecsInDay = (long)((dblDate - floor(dblDate)) * 86400.);
-
- // Calculate the day of week (sun=1, mon=2...)
- // -1 because 1/1/0 is Sat. +1 because we want 1-based
- tmDest.tm_wday = (int)((nDaysAbsolute - 1) % 7L) + 1;
-
- // Leap years every 4 yrs except centuries not multiples of 400.
- n400Years = (long)(nDaysAbsolute / 146097L);
-
- // Set nDaysAbsolute to day within 400-year block
- nDaysAbsolute %= 146097L;
-
- // -1 because first century has extra day
- n400Century = (long)((nDaysAbsolute - 1) / 36524L);
-
- // Non-leap century
- if (n400Century != 0)
- {
- // Set nDaysAbsolute to day within century
- nDaysAbsolute = (nDaysAbsolute - 1) % 36524L;
-
- // +1 because 1st 4 year increment has 1460 days
- n4Years = (long)((nDaysAbsolute + 1) / 1461L);
-
- if (n4Years != 0)
- n4Day = (long)((nDaysAbsolute + 1) % 1461L);
- else
- {
- bLeap4 = FALSE;
- n4Day = (long)nDaysAbsolute;
- }
- }
- else
- {
- // Leap century - not special case!
- n4Years = (long)(nDaysAbsolute / 1461L);
- n4Day = (long)(nDaysAbsolute % 1461L);
- }
-
- if (bLeap4)
- {
- // -1 because first year has 366 days
- n4Yr = (n4Day - 1) / 365;
-
- if (n4Yr != 0)
- n4Day = (n4Day - 1) % 365;
- }
- else
- {
- n4Yr = n4Day / 365;
- n4Day %= 365;
- }
-
- // n4Day is now 0-based day of year. Save 1-based day of year, year number
- tmDest.tm_yday = (int)n4Day + 1;
- tmDest.tm_year = n400Years * 400 + n400Century * 100 + n4Years * 4 + n4Yr;
-
- // Handle leap year: before, on, and after Feb. 29.
- if (n4Yr == 0 && bLeap4)
- {
- // Leap Year
- if (n4Day == 59)
- {
- /* Feb. 29 */
- tmDest.tm_mon = 2;
- tmDest.tm_mday = 29;
- goto DoTime;
- }
-
- // Pretend it's not a leap year for month/day comp.
- if (n4Day >= 60)
- --n4Day;
- }
-
- // Make n4DaY a 1-based day of non-leap year and compute
- // month/day for everything but Feb. 29.
- ++n4Day;
-
- // Month number always >= n/32, so save some loop time */
- for (tmDest.tm_mon = (n4Day >> 5) + 1;
- n4Day > _afxMonthDays[tmDest.tm_mon]; tmDest.tm_mon++);
-
- tmDest.tm_mday = (int)(n4Day - _afxMonthDays[tmDest.tm_mon-1]);
-
- DoTime:
- if (nSecsInDay == 0)
- tmDest.tm_hour = tmDest.tm_min = tmDest.tm_sec = 0;
- else
- {
- tmDest.tm_sec = (int)nSecsInDay % 60L;
- nMinutesInDay = nSecsInDay / 60L;
- tmDest.tm_min = (int)nMinutesInDay % 60;
- tmDest.tm_hour = (int)nMinutesInDay / 60;
- }
-
- return TRUE;
- }
-
- AFX_STATIC void AFXAPI _AfxTmConvertToStandardFormat(struct tm& tmSrc)
- {
- // Convert afx internal tm to format expected by runtimes (_tcsftime, etc)
- tmSrc.tm_year -= 1900; // year is based on 1900
- tmSrc.tm_mon -= 1; // month of year is 0-based
- tmSrc.tm_wday -= 1; // day of week is 0-based
- tmSrc.tm_yday -= 1; // day of year is 0-based
- }
-
- AFX_STATIC double AFXAPI _AfxDoubleFromDate(DATE dt)
- {
- // No problem if positive
- if (dt >= 0)
- return dt;
-
- // If negative, must convert since negative dates not continuous
- // (examples: -1.25 to -.75, -1.50 to -.50, -1.75 to -.25)
- double temp = ceil(dt);
- return temp - (dt - temp);
- }
-
- AFX_STATIC DATE AFXAPI _AfxDateFromDouble(double dbl)
- {
- // No problem if positive
- if (dbl >= 0)
- return dbl;
-
- // If negative, must convert since negative dates not continuous
- // (examples: -.75 to -1.25, -.50 to -1.50, -.25 to -1.75)
- double temp = floor(dbl); // dbl is now whole part
- return temp + (temp - dbl);
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // COleDateTime class
-
- COleDateTime PASCAL COleDateTime::GetCurrentTime()
- {
- return COleDateTime(::time(NULL));
- }
-
- BOOL COleDateTime::GetAsSystemTime(SYSTEMTIME& sysTime) const
- {
- BOOL bRetVal = FALSE;
- if (GetStatus() == valid)
- {
- struct tm tmTemp;
- if (_AfxTmFromOleDate(m_dt, tmTemp))
- {
- sysTime.wYear = (WORD) tmTemp.tm_year;
- sysTime.wMonth = (WORD) tmTemp.tm_mon;
- sysTime.wDayOfWeek = (WORD) (tmTemp.tm_wday - 1);
- sysTime.wDay = (WORD) tmTemp.tm_mday;
- sysTime.wHour = (WORD) tmTemp.tm_hour;
- sysTime.wMinute = (WORD) tmTemp.tm_min;
- sysTime.wSecond = (WORD) tmTemp.tm_sec;
- sysTime.wMilliseconds = 0;
-
- bRetVal = TRUE;
- }
- }
-
- return bRetVal;
- }
-
- int COleDateTime::GetYear() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_year;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- int COleDateTime::GetMonth() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_mon;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- int COleDateTime::GetDay() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_mday;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- int COleDateTime::GetHour() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_hour;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- int COleDateTime::GetMinute() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_min;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- int COleDateTime::GetSecond() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_sec;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- int COleDateTime::GetDayOfWeek() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_wday;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- int COleDateTime::GetDayOfYear() const
- {
- struct tm tmTemp;
-
- if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
- return tmTemp.tm_yday;
- else
- return AFX_OLE_DATETIME_ERROR;
- }
-
- const COleDateTime& COleDateTime::operator=(const VARIANT& varSrc)
- {
- if (varSrc.vt != VT_DATE)
- {
- TRY
- {
- COleVariant varTemp(varSrc);
- varTemp.ChangeType(VT_DATE);
- m_dt = varTemp.date;
- SetStatus(valid);
- }
- // Catch COleException from ChangeType, but not CMemoryException
- CATCH(COleException, e)
- {
- // Not able to convert VARIANT to DATE
- DELETE_EXCEPTION(e);
- m_dt = 0;
- SetStatus(invalid);
- }
- END_CATCH
- }
- else
- {
- m_dt = varSrc.date;
- SetStatus(valid);
- }
-
- return *this;
- }
-
- const COleDateTime& COleDateTime::operator=(DATE dtSrc)
- {
- m_dt = dtSrc;
- SetStatus(valid);
-
- return *this;
- }
-
- const COleDateTime& COleDateTime::operator=(const time_t& timeSrc)
- {
- // Convert time_t to struct tm
- tm *ptm = localtime(&timeSrc);
-
- if (ptm != NULL)
- {
- m_status = _AfxOleDateFromTm((WORD)(ptm->tm_year + 1900),
- (WORD)(ptm->tm_mon + 1), (WORD)ptm->tm_mday,
- (WORD)ptm->tm_hour, (WORD)ptm->tm_min,
- (WORD)ptm->tm_sec, m_dt) ? valid : invalid;
- }
- else
- {
- // Local time must have failed (timsSrc before 1/1/70 12am)
- SetStatus(invalid);
- ASSERT(FALSE);
- }
-
- return *this;
- }
-
- const COleDateTime& COleDateTime::operator=(const SYSTEMTIME& systimeSrc)
- {
- m_status = _AfxOleDateFromTm(systimeSrc.wYear, systimeSrc.wMonth,
- systimeSrc.wDay, systimeSrc.wHour, systimeSrc.wMinute,
- systimeSrc.wSecond, m_dt) ? valid : invalid;
-
- return *this;
- }
-
- const COleDateTime& COleDateTime::operator=(const FILETIME& filetimeSrc)
- {
- // Assume UTC FILETIME, so convert to LOCALTIME
- FILETIME filetimeLocal;
- if (!FileTimeToLocalFileTime( &filetimeSrc, &filetimeLocal))
- {
- #ifdef _DEBUG
- DWORD dwError = GetLastError();
- TRACE1("\nFileTimeToLocalFileTime failed. Error = %lu.\n\t", dwError);
- #endif // _DEBUG
- m_status = invalid;
- }
- else
- {
- // Take advantage of SYSTEMTIME -> FILETIME conversion
- SYSTEMTIME systime;
- m_status = FileTimeToSystemTime(&filetimeLocal, &systime) ?
- valid : invalid;
-
- // At this point systime should always be valid, but...
- if (GetStatus() == valid)
- {
- m_status = _AfxOleDateFromTm(systime.wYear, systime.wMonth,
- systime.wDay, systime.wHour, systime.wMinute,
- systime.wSecond, m_dt) ? valid : invalid;
- }
- }
-
- return *this;
- }
-
- BOOL COleDateTime::operator<(const COleDateTime& date) const
- {
- ASSERT(GetStatus() == valid);
- ASSERT(date.GetStatus() == valid);
-
- // Handle negative dates
- return _AfxDoubleFromDate(m_dt) < _AfxDoubleFromDate(date.m_dt);
- }
-
- BOOL COleDateTime::operator>(const COleDateTime& date) const
- { ASSERT(GetStatus() == valid);
- ASSERT(date.GetStatus() == valid);
-
- // Handle negative dates
- return _AfxDoubleFromDate(m_dt) > _AfxDoubleFromDate(date.m_dt);
- }
-
- BOOL COleDateTime::operator<=(const COleDateTime& date) const
- {
- ASSERT(GetStatus() == valid);
- ASSERT(date.GetStatus() == valid);
-
- // Handle negative dates
- return _AfxDoubleFromDate(m_dt) <= _AfxDoubleFromDate(date.m_dt);
- }
-
- BOOL COleDateTime::operator>=(const COleDateTime& date) const
- {
- ASSERT(GetStatus() == valid);
- ASSERT(date.GetStatus() == valid);
-
- // Handle negative dates
- return _AfxDoubleFromDate(m_dt) >= _AfxDoubleFromDate(date.m_dt);
- }
-
- COleDateTime COleDateTime::operator+(const COleDateTimeSpan& dateSpan) const
- {
- COleDateTime dateResult; // Initializes m_status to valid
-
- // If either operand NULL, result NULL
- if (GetStatus() == null || dateSpan.GetStatus() == null)
- {
- dateResult.SetStatus(null);
- return dateResult;
- }
-
- // If either operand invalid, result invalid
- if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
- {
- dateResult.SetStatus(invalid);
- return dateResult;
- }
-
- // Compute the actual date difference by adding underlying dates
- dateResult = _AfxDateFromDouble(_AfxDoubleFromDate(m_dt) + dateSpan.m_span);
-
- // Validate within range
- dateResult.CheckRange();
-
- return dateResult;
- }
-
- COleDateTime COleDateTime::operator-(const COleDateTimeSpan& dateSpan) const
- {
- COleDateTime dateResult; // Initializes m_status to valid
-
- // If either operand NULL, result NULL
- if (GetStatus() == null || dateSpan.GetStatus() == null)
- {
- dateResult.SetStatus(null);
- return dateResult;
- }
-
- // If either operand invalid, result invalid
- if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
- {
- dateResult.SetStatus(invalid);
- return dateResult;
- }
-
- // Compute the actual date difference by subtracting underlying dates
- dateResult = _AfxDateFromDouble(_AfxDoubleFromDate(m_dt) - dateSpan.m_span);
-
- // Validate within range
- dateResult.CheckRange();
-
- return dateResult;
- }
-
- COleDateTimeSpan COleDateTime::operator-(const COleDateTime& date) const
- {
- COleDateTimeSpan spanResult;
-
- // If either operand NULL, result NULL
- if (GetStatus() == null || date.GetStatus() == null)
- {
- spanResult.SetStatus(COleDateTimeSpan::null);
- return spanResult;
- }
-
- // If either operand invalid, result invalid
- if (GetStatus() == invalid || date.GetStatus() == invalid)
- {
- spanResult.SetStatus(COleDateTimeSpan::invalid);
- return spanResult;
- }
-
- // Return result (span can't be invalid, so don't check range)
- return _AfxDoubleFromDate(m_dt) - _AfxDoubleFromDate(date.m_dt);
- }
-
- int COleDateTime::SetDateTime(int nYear, int nMonth, int nDay,
- int nHour, int nMin, int nSec)
- {
- return m_status = _AfxOleDateFromTm((WORD)nYear, (WORD)nMonth,
- (WORD)nDay, (WORD)nHour, (WORD)nMin, (WORD)nSec, m_dt) ?
- valid : invalid;
- }
-
- BOOL COleDateTime::ParseDateTime(LPCTSTR lpszDate, DWORD dwFlags, LCID lcid)
- {
- USES_CONVERSION;
- CString strDate = lpszDate;
-
- SCODE sc;
- if (FAILED(sc = VarDateFromStr((LPOLESTR)T2COLE(strDate), lcid,
- dwFlags, &m_dt)))
- {
- if (sc == DISP_E_TYPEMISMATCH)
- {
- // Can't convert string to date, set 0 and invalidate
- m_dt = 0;
- SetStatus(invalid);
- return FALSE;
- }
- else if (sc == DISP_E_OVERFLOW)
- {
- // Can't convert string to date, set -1 and invalidate
- m_dt = -1;
- SetStatus(invalid);
- return FALSE;
- }
- else
- {
- TRACE0("\nCOleDateTime VarDateFromStr call failed.\n\t");
- if (sc == E_OUTOFMEMORY)
- AfxThrowMemoryException();
- else
- AfxThrowOleException(sc);
- }
- }
-
- SetStatus(valid);
- return TRUE;
- }
-
- CString COleDateTime::Format(DWORD dwFlags, LCID lcid) const
- {
- USES_CONVERSION;
- CString strDate;
-
- // If null, return empty string
- if (GetStatus() == null)
- return strDate;
-
- // If invalid, return DateTime resource string
- if (GetStatus() == invalid)
- {
- VERIFY(strDate.LoadString(AFX_IDS_INVALID_DATETIME));
- return strDate;
- }
-
- COleVariant var;
- // Don't need to trap error. Should not fail due to type mismatch
- AfxCheckError(VarBstrFromDate(m_dt, lcid, dwFlags, &V_BSTR(&var)));
- var.vt = VT_BSTR;
- return OLE2CT(V_BSTR(&var));
- }
-
- CString COleDateTime::Format(LPCTSTR pFormat) const
- {
- CString strDate;
- struct tm tmTemp;
-
- // If null, return empty string
- if (GetStatus() == null)
- return strDate;
-
- // If invalid, return DateTime resource string
- if (GetStatus() == invalid || !_AfxTmFromOleDate(m_dt, tmTemp))
- {
- VERIFY(strDate.LoadString(AFX_IDS_INVALID_DATETIME));
- return strDate;
- }
-
- // Convert tm from afx internal format to standard format
- _AfxTmConvertToStandardFormat(tmTemp);
-
- // Fill in the buffer, disregard return value as it's not necessary
- LPTSTR lpszTemp = strDate.GetBufferSetLength(MAX_TIME_BUFFER_SIZE);
- _tcsftime(lpszTemp, strDate.GetLength(), pFormat, &tmTemp);
- strDate.ReleaseBuffer();
-
- return strDate;
- }
-
- CString COleDateTime::Format(UINT nFormatID) const
- {
- CString strFormat;
- VERIFY(strFormat.LoadString(nFormatID) != 0);
- return Format(strFormat);
- }
-
- void COleDateTime::CheckRange()
- {
- if (m_dt > MAX_DATE || m_dt < MIN_DATE) // about year 100 to about 9999
- SetStatus(invalid);
- }
-
- // serialization
- #ifdef _DEBUG
- CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleDateTime dateSrc)
- {
- dc << "\nCOleDateTime Object:";
- dc << "\n\tm_status = " << (long)dateSrc.m_status;
-
- COleVariant var(dateSrc);
- var.ChangeType(VT_BSTR);
-
- return dc << "\n\tdate = " << var.bstrVal;
- }
- #endif // _DEBUG
-
- CArchive& AFXAPI operator<<(CArchive& ar, COleDateTime dateSrc)
- {
- ar << (long)dateSrc.m_status;
- return ar << dateSrc.m_dt;
- }
-
- CArchive& AFXAPI operator>>(CArchive& ar, COleDateTime& dateSrc)
- {
- ar >> (long&)dateSrc.m_status;
- return ar >> dateSrc.m_dt;
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // COleDateTimeSpan class helpers
-
- #define MAX_DAYS_IN_SPAN 3615897L
-
- /////////////////////////////////////////////////////////////////////////////
- // COleDateTimeSpan class
- long COleDateTimeSpan::GetHours() const
- {
- ASSERT(GetStatus() == valid);
-
- double dblTemp;
-
- // Truncate days and scale up
- dblTemp = modf(m_span, &dblTemp);
-
- long lReturns = (long)((dblTemp + AFX_OLE_DATETIME_HALFSECOND) * 24);
- if (lReturns >= 24)
- lReturns -= 24;
-
- return lReturns;
- }
-
- long COleDateTimeSpan::GetMinutes() const
- {
- ASSERT(GetStatus() == valid);
-
- double dblTemp;
-
- // Truncate hours and scale up
- dblTemp = modf(m_span * 24, &dblTemp);
-
- long lReturns = (long) ((dblTemp + AFX_OLE_DATETIME_HALFSECOND) * 60);
- if (lReturns >= 60)
- lReturns -= 60;
-
- return lReturns;
- }
-
- long COleDateTimeSpan::GetSeconds() const
- {
- ASSERT(GetStatus() == valid);
-
- double dblTemp;
-
- // Truncate minutes and scale up
- dblTemp = modf(m_span * 24 * 60, &dblTemp);
-
- long lReturns = (long) ((dblTemp + AFX_OLE_DATETIME_HALFSECOND) * 60);
- if (lReturns >= 60)
- lReturns -= 60;
-
- return lReturns;
- }
-
- const COleDateTimeSpan& COleDateTimeSpan::operator=(double dblSpanSrc)
- {
- m_span = dblSpanSrc;
- SetStatus(valid);
- return *this;
- }
-
- const COleDateTimeSpan& COleDateTimeSpan::operator=(const COleDateTimeSpan& dateSpanSrc)
- {
- m_span = dateSpanSrc.m_span;
- m_status = dateSpanSrc.m_status;
- return *this;
- }
-
- COleDateTimeSpan COleDateTimeSpan::operator+(const COleDateTimeSpan& dateSpan) const
- {
- COleDateTimeSpan dateSpanTemp;
-
- // If either operand Null, result Null
- if (GetStatus() == null || dateSpan.GetStatus() == null)
- {
- dateSpanTemp.SetStatus(null);
- return dateSpanTemp;
- }
-
- // If either operand Invalid, result Invalid
- if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
- {
- dateSpanTemp.SetStatus(invalid);
- return dateSpanTemp;
- }
-
- // Add spans and validate within legal range
- dateSpanTemp.m_span = m_span + dateSpan.m_span;
- dateSpanTemp.CheckRange();
-
- return dateSpanTemp;
- }
-
- COleDateTimeSpan COleDateTimeSpan::operator-(const COleDateTimeSpan& dateSpan) const
- {
- COleDateTimeSpan dateSpanTemp;
-
- // If either operand Null, result Null
- if (GetStatus() == null || dateSpan.GetStatus() == null)
- {
- dateSpanTemp.SetStatus(null);
- return dateSpanTemp;
- }
-
- // If either operand Invalid, result Invalid
- if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
- {
- dateSpanTemp.SetStatus(invalid);
- return dateSpanTemp;
- }
-
- // Subtract spans and validate within legal range
- dateSpanTemp.m_span = m_span - dateSpan.m_span;
- dateSpanTemp.CheckRange();
-
- return dateSpanTemp;
- }
-
- void COleDateTimeSpan::SetDateTimeSpan(
- long lDays, int nHours, int nMins, int nSecs)
- {
- // Set date span by breaking into fractional days (all input ranges valid)
- m_span = lDays + ((double)nHours)/24 + ((double)nMins)/(24*60) +
- ((double)nSecs)/(24*60*60);
-
- SetStatus(valid);
- }
-
- CString COleDateTimeSpan::Format(LPCTSTR pFormat) const
- {
- CString strSpan;
- struct tm tmTemp;
-
- // If null, return empty string
- if (GetStatus() == null)
- return strSpan;
-
- // If invalid, return DateTimeSpan resource string
- if (GetStatus() == invalid || !_AfxTmFromOleDate(m_span, tmTemp))
- {
- VERIFY(strSpan.LoadString(AFX_IDS_INVALID_DATETIMESPAN));
- return strSpan;
- }
-
- // Convert tm from afx internal format to standard format
- _AfxTmConvertToStandardFormat(tmTemp);
-
- // _tcsftime() doesn't handle %D, so do it here
-
- CString strPreParsed;
- LPCTSTR pstrSource = pFormat;
- int nTargetChar = 0;
- int nAccumulatedLength = lstrlen(pFormat);
- LPTSTR pstrTarget = strPreParsed.GetBuffer(nAccumulatedLength);
-
- while (*pstrSource)
- {
- if (*pstrSource == '%' && pstrSource[1] == 'D')
- {
- TCHAR szDay[12];
- _itot(GetDays(), szDay, 10);
- strPreParsed.ReleaseBuffer(nTargetChar);
- strPreParsed += szDay;
- int nTemp = lstrlen(szDay);
- nAccumulatedLength += nTemp;
- nTargetChar += nTemp;
- pstrTarget = strPreParsed.GetBuffer(nAccumulatedLength)
- + nTargetChar;
- pstrSource = _tcsinc(pstrSource);
- pstrSource = _tcsinc(pstrSource);
- }
- *pstrTarget = *pstrSource;
- nTargetChar++;
- pstrSource = _tcsinc(pstrSource);
- pstrTarget = _tcsinc(pstrTarget);
- }
- strPreParsed.ReleaseBuffer(nTargetChar);
-
- // Fill in the buffer, disregard return value as it's not necessary
- LPTSTR lpszTemp = strSpan.GetBufferSetLength(MAX_TIME_BUFFER_SIZE);
- _tcsftime(lpszTemp, strSpan.GetLength(), (LPCTSTR) strPreParsed, &tmTemp);
- strSpan.ReleaseBuffer();
-
- return strSpan;
- }
-
- CString COleDateTimeSpan::Format(UINT nFormatID) const
- {
- CString strFormat;
- VERIFY(strFormat.LoadString(nFormatID) != 0);
- return Format(strFormat);
- }
-
- void COleDateTimeSpan::CheckRange()
- {
- if(m_span < -MAX_DAYS_IN_SPAN || m_span > MAX_DAYS_IN_SPAN)
- SetStatus(invalid);
- }
-
- // serialization
- #ifdef _DEBUG
- CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleDateTimeSpan dateSpanSrc)
- {
- dc << "\nCOleDateTimeSpan Object:";
- dc << "\n\tm_status = " << (long)dateSpanSrc.m_status;
-
- COleVariant var(dateSpanSrc.m_span);
- var.ChangeType(VT_BSTR);
-
- return dc << "\n\tdateSpan = " << var.bstrVal;
- }
- #endif // _DEBUG
-
- CArchive& AFXAPI operator<<(CArchive& ar, COleDateTimeSpan dateSpanSrc)
- {
- ar << (long)dateSpanSrc.m_status;
- return ar << dateSpanSrc.m_span;
- }
-
- CArchive& AFXAPI operator>>(CArchive& ar, COleDateTimeSpan& dateSpanSrc)
- {
- ar >> (long&)dateSpanSrc.m_status;
- return ar >> dateSpanSrc.m_span;
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // COleSafeArray class
-
- COleSafeArray::COleSafeArray(const SAFEARRAY& saSrc, VARTYPE vtSrc)
- {
- AfxSafeArrayInit(this);
- vt = (VARTYPE)(vtSrc | VT_ARRAY);
- AfxCheckError(::SafeArrayCopy((LPSAFEARRAY)&saSrc, &parray));
- m_dwDims = GetDim();
- m_dwElementSize = GetElemSize();
- }
-
- COleSafeArray::COleSafeArray(LPCSAFEARRAY pSrc, VARTYPE vtSrc)
- {
- AfxSafeArrayInit(this);
- vt = (VARTYPE)(vtSrc | VT_ARRAY);
- AfxCheckError(::SafeArrayCopy((LPSAFEARRAY)pSrc, &parray));
- m_dwDims = GetDim();
- m_dwElementSize = GetElemSize();
- }
-
- COleSafeArray::COleSafeArray(const COleSafeArray& saSrc)
- {
- AfxSafeArrayInit(this);
- *this = saSrc;
- m_dwDims = GetDim();
- m_dwElementSize = GetElemSize();
- }
-
- COleSafeArray::COleSafeArray(const VARIANT& varSrc)
- {
- AfxSafeArrayInit(this);
- *this = varSrc;
- m_dwDims = GetDim();
- m_dwElementSize = GetElemSize();
- }
-
- COleSafeArray::COleSafeArray(LPCVARIANT pSrc)
- {
- AfxSafeArrayInit(this);
- *this = pSrc;
- m_dwDims = GetDim();
- m_dwElementSize = GetElemSize();
- }
-
- // Operations
- void COleSafeArray::Attach(VARIANT& varSrc)
- {
- ASSERT(varSrc.vt & VT_ARRAY);
-
- // Free up previous safe array if necessary
- Clear();
-
- // give control of data to COleSafeArray
- memcpy(this, &varSrc, sizeof(varSrc));
- varSrc.vt = VT_EMPTY;
- }
-
- VARIANT COleSafeArray::Detach()
- {
- VARIANT varResult = *this;
- vt = VT_EMPTY;
- return varResult;
- }
-
- // Assignment operators
- COleSafeArray& COleSafeArray::operator=(const COleSafeArray& saSrc)
- {
- ASSERT(saSrc.vt & VT_ARRAY);
-
- AfxCheckError(::VariantCopy(this, (LPVARIANT)&saSrc));
- return *this;
- }
-
- COleSafeArray& COleSafeArray::operator=(const VARIANT& varSrc)
- {
- ASSERT(varSrc.vt & VT_ARRAY);
-
- AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
- return *this;
- }
-
- COleSafeArray& COleSafeArray::operator=(LPCVARIANT pSrc)
- {
- ASSERT(pSrc->vt & VT_ARRAY);
-
- AfxCheckError(::VariantCopy(this, (LPVARIANT)pSrc));
- return *this;
- }
-
- COleSafeArray& COleSafeArray::operator=(const COleVariant& varSrc)
- {
- ASSERT(varSrc.vt & VT_ARRAY);
-
- AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
- return *this;
- }
-
- // Comparison operators
- BOOL COleSafeArray::operator==(const SAFEARRAY& saSrc) const
- {
- return _AfxCompareSafeArrays(parray, (LPSAFEARRAY)&saSrc);
- }
-
- BOOL COleSafeArray::operator==(LPCSAFEARRAY pSrc) const
- {
- return _AfxCompareSafeArrays(parray, (LPSAFEARRAY)pSrc);
- }
-
- BOOL COleSafeArray::operator==(const COleSafeArray& saSrc) const
- {
- if (vt != saSrc.vt)
- return FALSE;
-
- return _AfxCompareSafeArrays(parray, saSrc.parray);
- }
-
- BOOL COleSafeArray::operator==(const VARIANT& varSrc) const
- {
- if (vt != varSrc.vt)
- return FALSE;
-
- return _AfxCompareSafeArrays(parray, varSrc.parray);
- }
-
- BOOL COleSafeArray::operator==(LPCVARIANT pSrc) const
- {
- if (vt != pSrc->vt)
- return FALSE;
-
- return _AfxCompareSafeArrays(parray, pSrc->parray);
- }
-
- BOOL COleSafeArray::operator==(const COleVariant& varSrc) const
- {
- if (vt != varSrc.vt)
- return FALSE;
-
- return _AfxCompareSafeArrays(parray, varSrc.parray);
- }
-
- #ifdef _DEBUG
- void _AfxDumpSafeArrayElement(CDumpContext& dc, COleSafeArray& saSrc,
- long* piIndices)
- {
- BYTE* pbData;
-
- pbData = (BYTE*)_alloca( saSrc.GetElemSize() );
-
- saSrc.GetElement(piIndices, pbData);
- switch(saSrc.vt&(~VT_ARRAY))
- {
- case VT_BOOL:
- dc << *((VARIANT_BOOL*)pbData);
- break;
-
- case VT_I1:
- dc << *((char*)pbData);
- break;
-
- case VT_I2:
- dc << *((short*)pbData);
- break;
-
- case VT_I4:
- dc << *((long*)pbData);
- break;
-
- case VT_UI1:
- dc << *((BYTE*)pbData);
- break;
-
- case VT_UI2:
- dc << *((WORD*)pbData);
- break;
-
- case VT_UI4:
- dc << *((DWORD*)pbData);
- break;
-
- case VT_R4:
- dc << *((float*)pbData);
- break;
-
- case VT_R8:
- dc << *((double*)pbData);
- break;
-
- case VT_CY:
- {
- COleVariant var;
-
- var.vt = VT_CY;
- var.cyVal = *((CY*)pbData);
- var.ChangeType(VT_BSTR);
- dc << var.bstrVal;
- }
- break;
-
- case VT_DATE:
- {
- COleVariant var;
- var.vt = VT_DATE;
- var.date = *((DATE*)pbData);
- var.ChangeType(VT_BSTR);
- dc << var.bstrVal;
- }
-
- case VT_BSTR:
- dc << *((BSTR*)pbData);
- break;
-
- case VT_ERROR:
- dc << *((SCODE*)pbData);
- break;
-
- case VT_DISPATCH:
- case VT_UNKNOWN:
- dc << *((IUnknown**)pbData);
- break;
-
- case VT_VARIANT:
- {
- COleVariant var;
-
- var = *((VARIANT*)pbData);
- dc << var;
- }
- break;
-
- default:
- ASSERT(FALSE);
- break;
- }
- }
-
- CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleSafeArray& saSrc)
- {
- long iDimension;
- long nDimensions;
- long* piLBounds;
- long* piUBounds;
- long* piIndices;
- BOOL bWrapped;
-
- dc << "\nCOleSafeArray Object:";
- dc << "\n\tvt = " << saSrc.vt;
- dc << "\n\tbounds:";
-
- nDimensions = saSrc.GetDim();
- piLBounds = (long*)_alloca( nDimensions*sizeof( long ) );
- piUBounds = (long*)_alloca( nDimensions*sizeof( long ) );
- piIndices = (long*)_alloca( nDimensions*sizeof( long ) );
- // Dump the bounds
- for( iDimension = 0; iDimension < nDimensions; iDimension++ )
- {
- saSrc.GetLBound( iDimension+1, &piLBounds[iDimension] );
- saSrc.GetUBound( iDimension+1, &piUBounds[iDimension] );
- dc << "\n\t(" << piLBounds[iDimension] << ", " <<
- piUBounds[iDimension] << ")";
- }
-
- if( dc.GetDepth() > 0 )
- {
- // Dump the contents of the array.
- for( iDimension = 0; iDimension < nDimensions; iDimension++ )
- {
- piIndices[iDimension] = piLBounds[iDimension];
- }
-
- while( piIndices[0] <= piUBounds[0] )
- {
- dc << "\n\t";
- for( iDimension = 0; iDimension < nDimensions; iDimension++ )
- {
- dc << "[" << piIndices[iDimension] << "]";
- }
- dc << " = ";
-
- // Dump the value of the element
- _AfxDumpSafeArrayElement(dc, saSrc, piIndices);
-
- // Increment the rightmost index, with wraparound and carry logic
- iDimension = nDimensions-1;
- bWrapped = TRUE;
- do
- {
- bWrapped = FALSE;
- piIndices[iDimension]++;
- if( piIndices[iDimension] > piUBounds[iDimension] )
- {
- if( iDimension > 0 )
- {
- // We've overstepped the bounds of this dimension, so wrap
- // around to the lower bound and make sure to increment the
- // next dimension to the left.
- bWrapped = TRUE;
- piIndices[iDimension] = piLBounds[iDimension];
- iDimension--;
- }
- }
- } while( bWrapped && (iDimension >= 0) );
- }
- }
-
- return dc;
- }
- #endif // _DEBUG
-
- void COleSafeArray::CreateOneDim(VARTYPE vtSrc, DWORD dwElements,
- const void* pvSrcData, long nLBound)
- {
- ASSERT(dwElements > 0);
-
- // Setup the bounds and create the array
- SAFEARRAYBOUND rgsabound;
- rgsabound.cElements = dwElements;
- rgsabound.lLbound = nLBound;
- Create(vtSrc, 1, &rgsabound);
-
- // Copy over the data if neccessary
- if (pvSrcData != NULL)
- {
- void* pvDestData;
- AccessData(&pvDestData);
- memcpy(pvDestData, pvSrcData, GetElemSize() * dwElements);
- UnaccessData();
- }
- }
-
- DWORD COleSafeArray::GetOneDimSize()
- {
- ASSERT(GetDim() == 1);
-
- long nUBound, nLBound;
-
- GetUBound(1, &nUBound);
- GetLBound(1, &nLBound);
-
- return nUBound + 1 - nLBound;
- }
-
- void COleSafeArray::ResizeOneDim(DWORD dwElements)
- {
- ASSERT(GetDim() == 1);
-
- SAFEARRAYBOUND rgsabound;
-
- rgsabound.cElements = dwElements;
- rgsabound.lLbound = 0;
-
- Redim(&rgsabound);
- }
-
- void COleSafeArray::Create(VARTYPE vtSrc, DWORD dwDims, DWORD* rgElements)
- {
- ASSERT(rgElements != NULL);
-
- // Allocate and fill proxy array of bounds (with lower bound of zero)
- SAFEARRAYBOUND* rgsaBounds = new SAFEARRAYBOUND[dwDims];
-
- for (DWORD dwIndex = 0; dwIndex < dwDims; dwIndex++)
- {
- // Assume lower bound is 0 and fill in element count
- rgsaBounds[dwIndex].lLbound = 0;
- rgsaBounds[dwIndex].cElements = rgElements[dwIndex];
- }
-
- TRY
- {
- Create(vtSrc, dwDims, rgsaBounds);
- }
- CATCH_ALL(e)
- {
- // Must free up memory
- delete[] rgsaBounds;
- THROW_LAST();
- }
- END_CATCH_ALL
-
- delete[] rgsaBounds;
- }
-
- void COleSafeArray::Create(VARTYPE vtSrc, DWORD dwDims, SAFEARRAYBOUND* rgsabound)
- {
- ASSERT(dwDims > 0);
- ASSERT(rgsabound != NULL);
-
- // Validate the VARTYPE for SafeArrayCreate call
- ASSERT(!(vtSrc & VT_ARRAY));
- ASSERT(!(vtSrc & VT_BYREF));
- ASSERT(!(vtSrc & VT_VECTOR));
- ASSERT(vtSrc != VT_EMPTY);
- ASSERT(vtSrc != VT_NULL);
-
- // Free up old safe array if necessary
- Clear();
-
- parray = ::SafeArrayCreate(vtSrc, dwDims, rgsabound);
-
- if (parray == NULL)
- AfxThrowMemoryException();
-
- vt = unsigned short(vtSrc | VT_ARRAY);
- m_dwDims = dwDims;
- m_dwElementSize = GetElemSize();
- }
-
- void COleSafeArray::AccessData(void** ppvData)
- {
- AfxCheckError(::SafeArrayAccessData(parray, ppvData));
- }
-
- void COleSafeArray::UnaccessData()
- {
- AfxCheckError(::SafeArrayUnaccessData(parray));
- }
-
- void COleSafeArray::AllocData()
- {
- AfxCheckError(::SafeArrayAllocData(parray));
- }
-
- void COleSafeArray::AllocDescriptor(DWORD dwDims)
- {
- AfxCheckError(::SafeArrayAllocDescriptor(dwDims, &parray));
- }
-
- void COleSafeArray::Copy(LPSAFEARRAY* ppsa)
- {
- AfxCheckError(::SafeArrayCopy(parray, ppsa));
- }
-
- void COleSafeArray::GetLBound(DWORD dwDim, long* pLbound)
- {
- AfxCheckError(::SafeArrayGetLBound(parray, dwDim, pLbound));
- }
-
- void COleSafeArray::GetUBound(DWORD dwDim, long* pUbound)
- {
- AfxCheckError(::SafeArrayGetUBound(parray, dwDim, pUbound));
- }
-
- void COleSafeArray::GetElement(long* rgIndices, void* pvData)
- {
- AfxCheckError(::SafeArrayGetElement(parray, rgIndices, pvData));
- }
-
- void COleSafeArray::PtrOfIndex(long* rgIndices, void** ppvData)
- {
- AfxCheckError(::SafeArrayPtrOfIndex(parray, rgIndices, ppvData));
- }
-
- void COleSafeArray::PutElement(long* rgIndices, void* pvData)
- {
- AfxCheckError(::SafeArrayPutElement(parray, rgIndices, pvData));
- }
-
- void COleSafeArray::Redim(SAFEARRAYBOUND* psaboundNew)
- {
- AfxCheckError(::SafeArrayRedim(parray, psaboundNew));
- }
-
- void COleSafeArray::Lock()
- {
- AfxCheckError(::SafeArrayLock(parray));
- }
-
- void COleSafeArray::Unlock()
- {
- AfxCheckError(::SafeArrayUnlock(parray));
- }
-
- void COleSafeArray::Destroy()
- {
- AfxCheckError(::SafeArrayDestroy(parray));
- }
-
- void COleSafeArray::DestroyData()
- {
- AfxCheckError(::SafeArrayDestroyData(parray));
- }
-
- void COleSafeArray::DestroyDescriptor()
- {
- AfxCheckError(::SafeArrayDestroyDescriptor(parray));
- }
-
- ///////////////////////////////////////////////////////////////////////////////
- // COleSafeArray Helpers
- void AFXAPI AfxSafeArrayInit(COleSafeArray* psa)
- {
- memset(psa, 0, sizeof(*psa));
- }
-
- /////////////////////////////////////////////////////////////////////////////
- // Simple field formatting to text item - see dlgdata.cpp for base types
-
- void AFXAPI DDX_Text(CDataExchange* pDX, int nIDC, COleDateTime& value)
- {
- HWND hWndCtrl = pDX->PrepareEditCtrl(nIDC);
- if (pDX->m_bSaveAndValidate)
- {
- int nLen = ::GetWindowTextLength(hWndCtrl);
- CString strTemp;
-
- ::GetWindowText(hWndCtrl, strTemp.GetBufferSetLength(nLen), nLen+1);
- strTemp.ReleaseBuffer();
-
- if (!value.ParseDateTime(strTemp)) // throws exception
- {
- // Can't convert string to datetime
- AfxMessageBox(AFX_IDP_PARSE_DATETIME);
- pDX->Fail(); // throws exception
- }
- }
- else
- {
- CString strTemp = value.Format();
- AfxSetWindowText(hWndCtrl, strTemp);
- }
- }
-
- void AFXAPI DDX_Text(CDataExchange* pDX, int nIDC, COleCurrency& value)
- {
- HWND hWndCtrl = pDX->PrepareEditCtrl(nIDC);
- if (pDX->m_bSaveAndValidate)
- {
- int nLen = ::GetWindowTextLength(hWndCtrl);
- CString strTemp;
-
- ::GetWindowText(hWndCtrl, strTemp.GetBufferSetLength(nLen), nLen+1);
- strTemp.ReleaseBuffer();
-
- if (!value.ParseCurrency(strTemp)) // throws exception
- {
- // Can't convert string to currency
- AfxMessageBox(AFX_IDP_PARSE_CURRENCY);
- pDX->Fail(); // throws exception
- }
- }
- else
- {
- CString strTemp = value.Format();
- AfxSetWindowText(hWndCtrl, strTemp);
- }
- }
-
- /////////////////////////////////////////////////////////////////////////////
-