home *** CD-ROM | disk | FTP | other *** search
/ Tricks of the Windows Gam…ming Gurus (2nd Edition) / Disc2.iso / vc98 / mfc / src / olevar.cpp < prev    next >
C/C++ Source or Header  |  1998-06-16  |  70KB  |  3,026 lines

  1. // This is a part of the Microsoft Foundation Classes C++ library.
  2. // Copyright (C) 1992-1998 Microsoft Corporation
  3. // All rights reserved.
  4. //
  5. // This source code is only intended as a supplement to the
  6. // Microsoft Foundation Classes Reference and related
  7. // electronic documentation provided with the library.
  8. // See these sources for detailed information regarding the
  9. // Microsoft Foundation Classes product.
  10.  
  11. #include "stdafx.h"
  12. #include <afxtempl.h>
  13. #include <math.h>
  14.  
  15. #ifdef _DEBUG
  16. #undef THIS_FILE
  17. static char THIS_FILE[] = __FILE__;
  18. #endif
  19.  
  20. /////////////////////////////////////////////////////////////////////////////
  21. // helpers
  22.  
  23. void AFXAPI AfxCheckError(SCODE sc)
  24. {
  25.     if (FAILED(sc))
  26.     {
  27.         if (sc == E_OUTOFMEMORY)
  28.             AfxThrowMemoryException();
  29.         else
  30.             AfxThrowOleException(sc);
  31.     }
  32. }
  33.  
  34. AFX_STATIC BOOL AFXAPI _AfxCompareSafeArrays(SAFEARRAY* parray1, SAFEARRAY* parray2)
  35. {
  36.     BOOL bCompare = FALSE;
  37.  
  38.     // If one is NULL they must both be NULL to compare
  39.     if (parray1 == NULL || parray2 == NULL)
  40.     {
  41.         return parray1 == parray2;
  42.     }
  43.  
  44.     // Dimension must match and if 0, then arrays compare
  45.     DWORD dwDim1 = ::SafeArrayGetDim(parray1);
  46.     DWORD dwDim2 = ::SafeArrayGetDim(parray2);
  47.     if (dwDim1 != dwDim2)
  48.         return FALSE;
  49.     else if (dwDim1 == 0)
  50.         return TRUE;
  51.  
  52.     // Element size must match
  53.     DWORD dwSize1 = ::SafeArrayGetElemsize(parray1);
  54.     DWORD dwSize2 = ::SafeArrayGetElemsize(parray2);
  55.     if (dwSize1 != dwSize2)
  56.         return FALSE;
  57.  
  58.     long* pLBound1 = NULL;
  59.     long* pLBound2 = NULL;
  60.     long* pUBound1 = NULL;
  61.     long* pUBound2 = NULL;
  62.  
  63.     void* pData1 = NULL;
  64.     void* pData2 = NULL;
  65.  
  66.     TRY
  67.     {
  68.         // Bounds must match
  69.         pLBound1 = new long[dwDim1];
  70.         pLBound2 = new long[dwDim2];
  71.         pUBound1 = new long[dwDim1];
  72.         pUBound2 = new long[dwDim2];
  73.  
  74.         size_t nTotalElements = 1;
  75.  
  76.         // Get and compare bounds
  77.         for (DWORD dwIndex = 0; dwIndex < dwDim1; dwIndex++)
  78.         {
  79.             AfxCheckError(::SafeArrayGetLBound(
  80.                 parray1, dwIndex+1, &pLBound1[dwIndex]));
  81.             AfxCheckError(::SafeArrayGetLBound(
  82.                 parray2, dwIndex+1, &pLBound2[dwIndex]));
  83.             AfxCheckError(::SafeArrayGetUBound(
  84.                 parray1, dwIndex+1, &pUBound1[dwIndex]));
  85.             AfxCheckError(::SafeArrayGetUBound(
  86.                 parray2, dwIndex+1, &pUBound2[dwIndex]));
  87.  
  88.             // Check the magnitude of each bound
  89.             if (pUBound1[dwIndex] - pLBound1[dwIndex] !=
  90.                 pUBound2[dwIndex] - pLBound2[dwIndex])
  91.             {
  92.                 delete[] pLBound1;
  93.                 delete[] pLBound2;
  94.                 delete[] pUBound1;
  95.                 delete[] pUBound2;
  96.  
  97.                 return FALSE;
  98.             }
  99.  
  100.             // Increment the element count
  101.             nTotalElements *= pUBound1[dwIndex] - pLBound1[dwIndex] + 1;
  102.         }
  103.  
  104.         // Access the data
  105.         AfxCheckError(::SafeArrayAccessData(parray1, &pData1));
  106.         AfxCheckError(::SafeArrayAccessData(parray2, &pData2));
  107.  
  108.         // Calculate the number of bytes of data and compare
  109.         size_t nSize = nTotalElements * dwSize1;
  110.         int nOffset = memcmp(pData1, pData2, nSize);
  111.         bCompare = nOffset == 0;
  112.  
  113.         // Release the array locks
  114.         AfxCheckError(::SafeArrayUnaccessData(parray1));
  115.         AfxCheckError(::SafeArrayUnaccessData(parray2));
  116.     }
  117.     CATCH_ALL(e)
  118.     {
  119.         // Clean up bounds arrays
  120.         delete[] pLBound1;
  121.         delete[] pLBound2;
  122.         delete[] pUBound1;
  123.         delete[] pUBound2;
  124.  
  125.         // Release the array locks
  126.         if (pData1 != NULL)
  127.             AfxCheckError(::SafeArrayUnaccessData(parray1));
  128.         if (pData2 != NULL)
  129.             AfxCheckError(::SafeArrayUnaccessData(parray2));
  130.  
  131.         THROW_LAST();
  132.     }
  133.     END_CATCH_ALL
  134.  
  135.     // Clean up bounds arrays
  136.     delete[] pLBound1;
  137.     delete[] pLBound2;
  138.     delete[] pUBound1;
  139.     delete[] pUBound2;
  140.  
  141.     return bCompare;
  142. }
  143.  
  144. AFX_STATIC void AFXAPI _AfxCreateOneDimArray(VARIANT& varSrc, DWORD dwSize)
  145. {
  146.     UINT nDim;
  147.  
  148.     // Clear VARIANT and re-create SafeArray if necessary
  149.     if (varSrc.vt != (VT_UI1 | VT_ARRAY) ||
  150.         (nDim = ::SafeArrayGetDim(varSrc.parray)) != 1)
  151.     {
  152.         VERIFY(::VariantClear(&varSrc) == NOERROR);
  153.         varSrc.vt = VT_UI1 | VT_ARRAY;
  154.  
  155.         SAFEARRAYBOUND bound;
  156.         bound.cElements = dwSize;
  157.         bound.lLbound = 0;
  158.         varSrc.parray = ::SafeArrayCreate(VT_UI1, 1, &bound);
  159.         if (varSrc.parray == NULL)
  160.             AfxThrowMemoryException();
  161.     }
  162.     else
  163.     {
  164.         // Must redimension array if necessary
  165.         long lLower, lUpper;
  166.         AfxCheckError(::SafeArrayGetLBound(varSrc.parray, 1, &lLower));
  167.         AfxCheckError(::SafeArrayGetUBound(varSrc.parray, 1, &lUpper));
  168.  
  169.         // Upper bound should always be greater than lower bound
  170.         long lSize = lUpper - lLower;
  171.         if (lSize < 0)
  172.         {
  173.             ASSERT(FALSE);
  174.             lSize = 0;
  175.  
  176.         }
  177.  
  178.         if ((DWORD)lSize != dwSize)
  179.         {
  180.             SAFEARRAYBOUND bound;
  181.             bound.cElements = dwSize;
  182.             bound.lLbound = lLower;
  183.             AfxCheckError(::SafeArrayRedim(varSrc.parray, &bound));
  184.         }
  185.     }
  186. }
  187.  
  188. AFX_STATIC void AFXAPI _AfxCopyBinaryData(SAFEARRAY* parray, const void* pvSrc, DWORD dwSize)
  189. {
  190.     // Access the data, copy it and unaccess it.
  191.     void* pDest;
  192.     AfxCheckError(::SafeArrayAccessData(parray, &pDest));
  193.     memcpy(pDest, pvSrc, dwSize);
  194.     AfxCheckError(::SafeArrayUnaccessData(parray));
  195. }
  196.  
  197. /////////////////////////////////////////////////////////////////////////////
  198. // COleVariant class
  199.  
  200. COleVariant::COleVariant(const VARIANT& varSrc)
  201. {
  202.     AfxVariantInit(this);
  203.     AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
  204. }
  205.  
  206. COleVariant::COleVariant(LPCVARIANT pSrc)
  207. {
  208.     AfxVariantInit(this);
  209.     AfxCheckError(::VariantCopy(this, (LPVARIANT)pSrc));
  210. }
  211.  
  212. COleVariant::COleVariant(const COleVariant& varSrc)
  213. {
  214.     AfxVariantInit(this);
  215.     AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
  216. }
  217.  
  218. COleVariant::COleVariant(LPCTSTR lpszSrc, VARTYPE vtSrc)
  219. {
  220.     USES_CONVERSION;
  221.     ASSERT(vtSrc == VT_BSTR || vtSrc == VT_BSTRT);
  222.     UNUSED(vtSrc);
  223.  
  224.     vt = VT_BSTR;
  225.     bstrVal = NULL;
  226.  
  227.     if (lpszSrc != NULL)
  228.     {
  229. #ifndef _UNICODE
  230.         if (vtSrc == VT_BSTRT)
  231.         {
  232.             int nLen = lstrlen(lpszSrc);
  233.             bstrVal = ::SysAllocStringByteLen(lpszSrc, nLen);
  234.         }
  235.         else
  236. #endif
  237.         {
  238.             bstrVal = ::SysAllocString(T2COLE(lpszSrc));
  239.         }
  240.  
  241.         if (bstrVal == NULL)
  242.             AfxThrowMemoryException();
  243.     }
  244. }
  245.  
  246.  
  247.  
  248. void COleVariant::SetString(LPCTSTR lpszSrc, VARTYPE vtSrc)
  249. {
  250.     USES_CONVERSION;
  251.     ASSERT(vtSrc == VT_BSTR || vtSrc == VT_BSTRT);
  252.     UNUSED(vtSrc);
  253.  
  254.     // Free up previous VARIANT
  255.     Clear();
  256.  
  257.     vt = VT_BSTR;
  258.     bstrVal = NULL;
  259.  
  260.     if (lpszSrc != NULL)
  261.     {
  262. #ifndef _UNICODE
  263.         if (vtSrc == VT_BSTRT)
  264.         {
  265.             int nLen = lstrlen(lpszSrc);
  266.             bstrVal = ::SysAllocStringByteLen(lpszSrc, nLen);
  267.         }
  268.         else
  269. #endif
  270.         {
  271.             bstrVal = ::SysAllocString(T2COLE(lpszSrc));
  272.         }
  273.  
  274.         if (bstrVal == NULL)
  275.             AfxThrowMemoryException();
  276.     }
  277. }
  278.  
  279. COleVariant::COleVariant(short nSrc, VARTYPE vtSrc)
  280. {
  281.     ASSERT(vtSrc == VT_I2 || vtSrc == VT_BOOL);
  282.  
  283.     if (vtSrc == VT_BOOL)
  284.     {
  285.         vt = VT_BOOL;
  286.         if (!nSrc)
  287.             V_BOOL(this) = AFX_OLE_FALSE;
  288.         else
  289.             V_BOOL(this) = AFX_OLE_TRUE;
  290.     }
  291.     else
  292.     {
  293.         vt = VT_I2;
  294.         iVal = nSrc;
  295.     }
  296. }
  297.  
  298. COleVariant::COleVariant(long lSrc, VARTYPE vtSrc)
  299. {
  300.     ASSERT(vtSrc == VT_I4 || vtSrc == VT_ERROR || vtSrc == VT_BOOL);
  301.  
  302.     if (vtSrc == VT_ERROR)
  303.     {
  304.         vt = VT_ERROR;
  305.         scode = lSrc;
  306.     }
  307.     else if (vtSrc == VT_BOOL)
  308.     {
  309.         vt = VT_BOOL;
  310.         if (!lSrc)
  311.             V_BOOL(this) = AFX_OLE_FALSE;
  312.         else
  313.             V_BOOL(this) = AFX_OLE_TRUE;
  314.     }
  315.     else
  316.     {
  317.         vt = VT_I4;
  318.         lVal = lSrc;
  319.     }
  320. }
  321.  
  322. // Operations
  323. void COleVariant::_ClearCompat()
  324. {
  325.     Clear();
  326. }
  327.  
  328. void COleVariant::ChangeType(VARTYPE vartype, LPVARIANT pSrc)
  329. {
  330.     // If pSrc is NULL, convert type in place
  331.     if (pSrc == NULL)
  332.         pSrc = this;
  333.     if (pSrc != this || vartype != vt)
  334.         AfxCheckError(::VariantChangeType(this, pSrc, 0, vartype));
  335. }
  336.  
  337. void COleVariant::Attach(VARIANT& varSrc)
  338. {
  339.     // Free up previous VARIANT
  340.     Clear();
  341.  
  342.     // give control of data to COleVariant
  343.     memcpy(this, &varSrc, sizeof(varSrc));
  344.     varSrc.vt = VT_EMPTY;
  345. }
  346.  
  347. VARIANT COleVariant::Detach()
  348. {
  349.     VARIANT varResult = *this;
  350.     vt = VT_EMPTY;
  351.     return varResult;
  352. }
  353.  
  354. // Literal comparison. Types and values must match.
  355. BOOL COleVariant::operator==(const VARIANT& var) const
  356. {
  357.     if (&var == this)
  358.         return TRUE;
  359.  
  360.     // Variants not equal if types don't match
  361.     if (var.vt != vt)
  362.         return FALSE;
  363.  
  364.     // Check type specific values
  365.     switch (vt)
  366.     {
  367.     case VT_EMPTY:
  368.     case VT_NULL:
  369.         return TRUE;
  370.  
  371.     case VT_BOOL:
  372.         return V_BOOL(&var) == V_BOOL(this);
  373.  
  374.     case VT_UI1:
  375.         return var.bVal == bVal;
  376.  
  377.     case VT_I2:
  378.         return var.iVal == iVal;
  379.  
  380.     case VT_I4:
  381.         return var.lVal == lVal;
  382.  
  383.     case VT_CY:
  384.         return (var.cyVal.Hi == cyVal.Hi && var.cyVal.Lo == cyVal.Lo);
  385.  
  386.     case VT_R4:
  387.         return var.fltVal == fltVal;
  388.  
  389.     case VT_R8:
  390.         return var.dblVal == dblVal;
  391.  
  392.     case VT_DATE:
  393.         return var.date == date;
  394.  
  395.     case VT_BSTR:
  396.         return SysStringByteLen(var.bstrVal) == SysStringByteLen(bstrVal) &&
  397.             memcmp(var.bstrVal, bstrVal, SysStringByteLen(bstrVal)) == 0;
  398.  
  399.     case VT_ERROR:
  400.         return var.scode == scode;
  401.  
  402.     case VT_DISPATCH:
  403.     case VT_UNKNOWN:
  404.         return var.punkVal == punkVal;
  405.  
  406.     default:
  407.         if (vt & VT_ARRAY && !(vt & VT_BYREF))
  408.             return _AfxCompareSafeArrays(var.parray, parray);
  409.         else
  410.             ASSERT(FALSE);  // VT_BYREF not supported
  411.         // fall through
  412.     }
  413.  
  414.     return FALSE;
  415. }
  416.  
  417. const COleVariant& COleVariant::operator=(const VARIANT& varSrc)
  418. {
  419.     AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
  420.     return *this;
  421. }
  422.  
  423. const COleVariant& COleVariant::operator=(LPCVARIANT pSrc)
  424. {
  425.     AfxCheckError(::VariantCopy(this, (LPVARIANT)pSrc));
  426.     return *this;
  427. }
  428.  
  429. const COleVariant& COleVariant::operator=(const COleVariant& varSrc)
  430. {
  431.     AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
  432.     return *this;
  433. }
  434.  
  435. const COleVariant& COleVariant::operator=(const LPCTSTR lpszSrc)
  436. {
  437.     USES_CONVERSION;
  438.     // Free up previous VARIANT
  439.     Clear();
  440.  
  441.     vt = VT_BSTR;
  442.     if (lpszSrc == NULL)
  443.         bstrVal = NULL;
  444.     else
  445.     {
  446.         bstrVal = ::SysAllocString(T2COLE(lpszSrc));
  447.         if (bstrVal == NULL)
  448.             AfxThrowMemoryException();
  449.     }
  450.     return *this;
  451. }
  452.  
  453. const COleVariant& COleVariant::operator=(const CString& strSrc)
  454. {
  455.     USES_CONVERSION;
  456.     // Free up previous VARIANT
  457.     Clear();
  458.  
  459.     vt = VT_BSTR;
  460.     bstrVal = ::SysAllocString(T2COLE(strSrc));
  461.     if (bstrVal == NULL)
  462.         AfxThrowMemoryException();
  463.  
  464.     return *this;
  465. }
  466.  
  467. const COleVariant& COleVariant::operator=(BYTE nSrc)
  468. {
  469.     // Free up previous VARIANT if necessary
  470.     if (vt != VT_UI1)
  471.     {
  472.         Clear();
  473.         vt = VT_UI1;
  474.     }
  475.  
  476.     bVal = nSrc;
  477.     return *this;
  478. }
  479.  
  480. const COleVariant& COleVariant::operator=(short nSrc)
  481. {
  482.     if (vt == VT_I2)
  483.         iVal = nSrc;
  484.     else if (vt == VT_BOOL)
  485.     {
  486.         if (!nSrc)
  487.             V_BOOL(this) = AFX_OLE_FALSE;
  488.         else
  489.             V_BOOL(this) = AFX_OLE_TRUE;
  490.     }
  491.     else
  492.     {
  493.         // Free up previous VARIANT
  494.         Clear();
  495.         vt = VT_I2;
  496.         iVal = nSrc;
  497.     }
  498.  
  499.     return *this;
  500. }
  501.  
  502. const COleVariant& COleVariant::operator=(long lSrc)
  503. {
  504.     if (vt == VT_I4)
  505.         lVal = lSrc;
  506.     else if (vt == VT_ERROR)
  507.         scode = lSrc;
  508.     else if (vt == VT_BOOL)
  509.     {
  510.         if (!lSrc)
  511.             V_BOOL(this) = AFX_OLE_FALSE;
  512.         else
  513.             V_BOOL(this) = AFX_OLE_TRUE;
  514.     }
  515.     else
  516.     {
  517.         // Free up previous VARIANT
  518.         Clear();
  519.         vt = VT_I4;
  520.         lVal = lSrc;
  521.     }
  522.  
  523.     return *this;
  524. }
  525.  
  526. const COleVariant& COleVariant::operator=(const COleCurrency& curSrc)
  527. {
  528.     // Free up previous VARIANT if necessary
  529.     if (vt != VT_CY)
  530.     {
  531.         Clear();
  532.         vt = VT_CY;
  533.     }
  534.  
  535.     cyVal = curSrc.m_cur;
  536.     return *this;
  537. }
  538.  
  539. const COleVariant& COleVariant::operator=(float fltSrc)
  540. {
  541.     // Free up previous VARIANT if necessary
  542.     if (vt != VT_R4)
  543.     {
  544.         Clear();
  545.         vt = VT_R4;
  546.     }
  547.  
  548.     fltVal = fltSrc;
  549.     return *this;
  550. }
  551.  
  552. const COleVariant& COleVariant::operator=(double dblSrc)
  553. {
  554.     // Free up previous VARIANT if necessary
  555.     if (vt != VT_R8)
  556.     {
  557.         Clear();
  558.         vt = VT_R8;
  559.     }
  560.  
  561.     dblVal = dblSrc;
  562.     return *this;
  563. }
  564.  
  565. const COleVariant& COleVariant::operator=(const COleDateTime& dateSrc)
  566. {
  567.     // Free up previous VARIANT if necessary
  568.     if (vt != VT_DATE)
  569.     {
  570.         Clear();
  571.         vt = VT_DATE;
  572.     }
  573.  
  574.     date = dateSrc.m_dt;
  575.     return *this;
  576. }
  577.  
  578. const COleVariant& COleVariant::operator=(const CByteArray& arrSrc)
  579. {
  580.     int nSize = arrSrc.GetSize();
  581.  
  582.     // Set the correct type and make sure SafeArray can hold data
  583.     _AfxCreateOneDimArray(*this, (DWORD)nSize);
  584.  
  585.     // Copy the data into the SafeArray
  586.     _AfxCopyBinaryData(parray, arrSrc.GetData(), (DWORD)nSize);
  587.  
  588.     return *this;
  589. }
  590.  
  591. const COleVariant& COleVariant::operator=(const CLongBinary& lbSrc)
  592. {
  593.     // Set the correct type and make sure SafeArray can hold data
  594.     _AfxCreateOneDimArray(*this, lbSrc.m_dwDataLength);
  595.  
  596.     // Copy the data into the SafeArray
  597.     BYTE* pData = (BYTE*)::GlobalLock(lbSrc.m_hData);
  598.     _AfxCopyBinaryData(parray, pData, lbSrc.m_dwDataLength);
  599.     ::GlobalUnlock(lbSrc.m_hData);
  600.  
  601.     return *this;
  602. }
  603.  
  604. void AFXAPI AfxVariantInit(LPVARIANT pVar)
  605. {
  606.     memset(pVar, 0, sizeof(*pVar));
  607. }
  608.  
  609. /////////////////////////////////////////////////////////////////////////////
  610. // Diagnostics
  611.  
  612. #ifdef _DEBUG
  613. CDumpContext& AFXAPI operator <<(CDumpContext& dc, COleVariant varSrc)
  614. {
  615.     LPCVARIANT pSrc = (LPCVARIANT)varSrc;
  616.  
  617.     dc << "\nCOleVariant Object:";
  618.     dc << "\n\t vt = " << pSrc->vt;
  619.  
  620.     // No support for VT_BYREF & VT_ARRAY
  621.     if (pSrc->vt & VT_BYREF || pSrc->vt & VT_ARRAY)
  622.         return dc;
  623.  
  624.     switch (pSrc->vt)
  625.     {
  626.     case VT_BOOL:
  627.         return dc << "\n\t VT_BOOL = " << V_BOOL(pSrc);
  628.  
  629.     case VT_UI1:
  630.         return dc << "\n\t bVal = " << pSrc->bVal;
  631.  
  632.     case VT_I2:
  633.         return dc << "\n\t iVal = " << pSrc->iVal;
  634.  
  635.     case VT_I4:
  636.         return dc << "\n\t lVal = " << pSrc->lVal;
  637.  
  638.     case VT_CY:
  639.         {
  640.             COleVariant var(varSrc);
  641.             var.ChangeType(VT_BSTR);
  642.             return dc << "\n\t cyVal = " << var.bstrVal;
  643.         }
  644.  
  645.     case VT_R4:
  646.         return dc << "\n\t fltVal = " << pSrc->fltVal;
  647.  
  648.     case VT_R8:
  649.         return dc << "\n\t dblVal = " << pSrc->dblVal;
  650.  
  651.     case VT_DATE:
  652.         {
  653.             COleVariant var(varSrc);
  654.             var.ChangeType(VT_BSTR);
  655.             return dc << "\n\t date = " << var.bstrVal;
  656.         }
  657.  
  658.     case VT_BSTR:
  659.         return dc << "\n\t bstrVal = " << pSrc->bstrVal;
  660.  
  661.     case VT_ERROR:
  662.         return dc << "\n\t scode = " << pSrc->scode;
  663.  
  664.     case VT_DISPATCH:
  665.     case VT_UNKNOWN:
  666.         return dc << "\n\t punkVal = " << pSrc->punkVal;
  667.  
  668.     case VT_EMPTY:
  669.     case VT_NULL:
  670.         return dc;
  671.  
  672.     default:
  673.         ASSERT(FALSE);
  674.         return dc;
  675.     }
  676. }
  677. #endif // _DEBUG
  678.  
  679. CArchive& AFXAPI operator<<(CArchive& ar, COleVariant varSrc)
  680. {
  681.     LPCVARIANT pSrc = (LPCVARIANT)varSrc;
  682.  
  683.     ar << pSrc->vt;
  684.  
  685.     // No support for VT_BYREF & VT_ARRAY
  686.     if (pSrc->vt & VT_BYREF || pSrc->vt & VT_ARRAY)
  687.         return ar;
  688.  
  689.     switch (pSrc->vt)
  690.     {
  691.     case VT_BOOL:
  692.         return ar << (WORD)V_BOOL(pSrc);
  693.  
  694.     case VT_UI1:
  695.         return ar << pSrc->bVal;
  696.  
  697.     case VT_I2:
  698.         return ar << (WORD)pSrc->iVal;
  699.  
  700.     case VT_I4:
  701.         return ar << pSrc->lVal;
  702.  
  703.     case VT_CY:
  704.         ar << pSrc->cyVal.Lo;
  705.         return ar << pSrc->cyVal.Hi;
  706.  
  707.     case VT_R4:
  708.         return ar << pSrc->fltVal;
  709.  
  710.     case VT_R8:
  711.         return ar << pSrc->dblVal;
  712.  
  713.     case VT_DATE:
  714.         return ar << pSrc->date;
  715.  
  716.     case VT_BSTR:
  717.         {
  718.             DWORD nLen = SysStringByteLen(pSrc->bstrVal);
  719.             ar << nLen;
  720.             if (nLen > 0)
  721.                 ar.Write(pSrc->bstrVal, nLen * sizeof(BYTE));
  722.  
  723.             return ar;
  724.         }
  725.  
  726.     case VT_ERROR:
  727.         return ar << pSrc->scode;
  728.  
  729.     case VT_DISPATCH:
  730.     case VT_UNKNOWN:
  731.         {
  732.             LPPERSISTSTREAM pPersistStream;
  733.             CArchiveStream stm(&ar);
  734.  
  735.             // QI for IPersistStream or IPeristStreamInit
  736.             SCODE sc = pSrc->punkVal->QueryInterface(
  737.                 IID_IPersistStream, (void**)&pPersistStream);
  738. #ifndef _AFX_NO_OCC_SUPPORT
  739.             if (FAILED(sc))
  740.                 sc = pSrc->punkVal->QueryInterface(
  741.                     IID_IPersistStreamInit, (void**)&pPersistStream);
  742. #endif
  743.             AfxCheckError(sc);
  744.  
  745.             TRY
  746.             {
  747.                 // Get and archive the CLSID (GUID)
  748.                 CLSID clsid;
  749.                 AfxCheckError(pPersistStream->GetClassID(&clsid));
  750.                 ar << clsid.Data1;
  751.                 ar << clsid.Data2;
  752.                 ar << clsid.Data3;
  753.                 ar.Write(&clsid.Data4[0], sizeof clsid.Data4);
  754.  
  755.                 // Always assume object is dirty
  756.                 AfxCheckError(pPersistStream->Save(&stm, TRUE));
  757.             }
  758.             CATCH_ALL(e)
  759.             {
  760.                 pPersistStream->Release();
  761.                 THROW_LAST();
  762.             }
  763.             END_CATCH_ALL
  764.             pPersistStream->Release();
  765.         }
  766.         return ar;
  767.  
  768.     case VT_EMPTY:
  769.     case VT_NULL:
  770.         // do nothing
  771.         return ar;
  772.  
  773.     default:
  774.         ASSERT(FALSE);
  775.         return ar;
  776.     }
  777. }
  778.  
  779. CArchive& AFXAPI operator>>(CArchive& ar, COleVariant& varSrc)
  780. {
  781.     LPVARIANT pSrc = &varSrc;
  782.  
  783.     // Free up current data if necessary
  784.     if (pSrc->vt != VT_EMPTY)
  785.         VariantClear(pSrc);
  786.     ar >> pSrc->vt;
  787.  
  788.     // No support for VT_BYREF & VT_ARRAY
  789.     if (pSrc->vt & VT_BYREF || pSrc->vt & VT_ARRAY)
  790.         return ar;
  791.  
  792.     switch (pSrc->vt)
  793.     {
  794.     case VT_BOOL:
  795.         return ar >> (WORD&)V_BOOL(pSrc);
  796.  
  797.     case VT_UI1:
  798.         return ar >> pSrc->bVal;
  799.  
  800.     case VT_I2:
  801.         return ar >> (WORD&)pSrc->iVal;
  802.  
  803.     case VT_I4:
  804.         return ar >> pSrc->lVal;
  805.  
  806.     case VT_CY:
  807.         ar >> pSrc->cyVal.Lo;
  808.         return ar >> pSrc->cyVal.Hi;
  809.  
  810.     case VT_R4:
  811.         return ar >> pSrc->fltVal;
  812.  
  813.     case VT_R8:
  814.         return ar >> pSrc->dblVal;
  815.  
  816.     case VT_DATE:
  817.         return ar >> pSrc->date;
  818.  
  819.     case VT_BSTR:
  820.         {
  821.             DWORD nLen;
  822.             ar >> nLen;
  823.             if (nLen > 0)
  824.             {
  825.                 pSrc->bstrVal = SysAllocStringByteLen(NULL, nLen);
  826.                 if (pSrc->bstrVal == NULL)
  827.                     AfxThrowMemoryException();
  828.                 ar.Read(pSrc->bstrVal, nLen * sizeof(BYTE));
  829.             }
  830.             else
  831.                 pSrc->bstrVal = NULL;
  832.  
  833.             return ar;
  834.         }
  835.         break;
  836.  
  837.     case VT_ERROR:
  838.         return ar >> pSrc->scode;
  839.  
  840.     case VT_DISPATCH:
  841.     case VT_UNKNOWN:
  842.         {
  843.             LPPERSISTSTREAM pPersistStream = NULL;
  844.             CArchiveStream stm(&ar);
  845.  
  846.             // Retrieve the CLSID (GUID) and create an instance
  847.             CLSID clsid;
  848.             ar >> clsid.Data1;
  849.             ar >> clsid.Data2;
  850.             ar >> clsid.Data3;
  851.             ar.Read(&clsid.Data4[0], sizeof clsid.Data4);
  852.  
  853.             // Create the object
  854.             SCODE sc = CoCreateInstance(clsid, NULL, CLSCTX_ALL | CLSCTX_REMOTE_SERVER,
  855.                 pSrc->vt == VT_UNKNOWN ? IID_IUnknown : IID_IDispatch,
  856.                 (void**)&pSrc->punkVal);
  857.             if (sc == E_INVALIDARG)
  858.             {
  859.                 // may not support CLSCTX_REMOTE_SERVER, so try without
  860.                 sc = CoCreateInstance(clsid, NULL,
  861.                     CLSCTX_ALL & ~CLSCTX_REMOTE_SERVER,
  862.                     pSrc->vt == VT_UNKNOWN ? IID_IUnknown : IID_IDispatch,
  863.                     (void**)&pSrc->punkVal);
  864.             }
  865.             AfxCheckError(sc);
  866.  
  867.             TRY
  868.             {
  869.                 // QI for IPersistStream or IPeristStreamInit
  870.                 sc = pSrc->punkVal->QueryInterface(
  871.                     IID_IPersistStream, (void**)&pPersistStream);
  872. #ifndef _AFX_NO_OCC_SUPPORT
  873.                 if (FAILED(sc))
  874.                     sc = pSrc->punkVal->QueryInterface(
  875.                         IID_IPersistStreamInit, (void**)&pPersistStream);
  876. #endif
  877.                 AfxCheckError(sc);
  878.  
  879.                 // Always assumes object is dirty
  880.                 AfxCheckError(pPersistStream->Load(&stm));
  881.             }
  882.             CATCH_ALL(e)
  883.             {
  884.                 // Clean up
  885.                 if (pPersistStream != NULL)
  886.                     pPersistStream->Release();
  887.  
  888.                 pSrc->punkVal->Release();
  889.                 THROW_LAST();
  890.             }
  891.             END_CATCH_ALL
  892.  
  893.             pPersistStream->Release();
  894.         }
  895.         return ar;
  896.  
  897.     case VT_EMPTY:
  898.     case VT_NULL:
  899.         // do nothing
  900.         return ar;
  901.  
  902.     default:
  903.         ASSERT(FALSE);
  904.         return ar;
  905.     }
  906. }
  907.  
  908. /////////////////////////////////////////////////////////////////////////////
  909. // COleVariant Helpers
  910.  
  911. #if _MSC_VER >= 1100
  912. template <> void AFXAPI ConstructElements<COleVariant> (COleVariant* pElements, int nCount)
  913. #else
  914. void AFXAPI ConstructElements(COleVariant* pElements, int nCount)
  915. #endif
  916. {
  917.     ASSERT(nCount == 0 ||
  918.         AfxIsValidAddress(pElements, nCount * sizeof(COleVariant)));
  919.  
  920.     for (; nCount--; ++pElements)
  921.         new(pElements) COleVariant;
  922. }
  923.  
  924. #if _MSC_VER >= 1100
  925. template <> void AFXAPI DestructElements<COleVariant> (COleVariant* pElements, int nCount)
  926. #else
  927. void AFXAPI DestructElements(COleVariant* pElements, int nCount)
  928. #endif
  929. {
  930.     ASSERT(nCount == 0 ||
  931.         AfxIsValidAddress(pElements, nCount * sizeof(COleVariant)));
  932.  
  933.     for (; nCount--; ++pElements)
  934.         pElements->~COleVariant();
  935. }
  936.  
  937. #if _MSC_VER >= 1100
  938. template <> void AFXAPI CopyElements<COleVariant> (COleVariant* pDest, const COleVariant* pSrc, int nCount)
  939. #else
  940. void AFXAPI CopyElements(COleVariant* pDest, const COleVariant* pSrc, int nCount)
  941. #endif
  942. {
  943.     ASSERT(nCount == 0 ||
  944.         AfxIsValidAddress(pDest, nCount * sizeof(COleVariant)));
  945.     ASSERT(nCount == 0 ||
  946.         AfxIsValidAddress(pSrc, nCount * sizeof(COleVariant)));
  947.  
  948.     for (; nCount--; ++pDest, ++pSrc)
  949.         *pDest = *pSrc;
  950. }
  951.  
  952. #if _MSC_VER >= 1100
  953. template <> void AFXAPI SerializeElements<COleVariant> (CArchive& ar, COleVariant* pElements, int nCount)
  954. #else
  955. void AFXAPI SerializeElements(CArchive& ar, COleVariant* pElements, int nCount)
  956. #endif
  957. {
  958.     ASSERT(nCount == 0 ||
  959.         AfxIsValidAddress(pElements, nCount * sizeof(COleVariant)));
  960.  
  961.     if (ar.IsStoring())
  962.     {
  963.         for (; nCount--; ++pElements)
  964.             ar << *pElements;
  965.     }
  966.     else
  967.     {
  968.         for (; nCount--; ++pElements)
  969.             ar >> *pElements;
  970.     }
  971. }
  972.  
  973. #ifdef _DEBUG
  974. #if _MSC_VER >= 1100
  975. template <> void AFXAPI DumpElements<COleVariant> (CDumpContext& dc, const COleVariant* pElements, int nCount)
  976. #else
  977. void AFXAPI DumpElements(CDumpContext& dc, const COleVariant* pElements, int nCount)
  978. #endif
  979. {
  980.     for (; nCount--; ++pElements)
  981.         dc << *pElements;
  982. }
  983. #endif // _DEBUG
  984.  
  985. #if _MSC_VER >= 1100
  986. template<> UINT AFXAPI HashKey<const struct tagVARIANT&> (const struct tagVARIANT& var)
  987. #else
  988. UINT AFXAPI HashKey(const struct tagVARIANT& var)
  989. #endif
  990. {
  991.     switch (var.vt)
  992.     {
  993.     case VT_EMPTY:
  994.     case VT_NULL:
  995.         return 0;
  996.     case VT_I2:
  997. #if _MSC_VER >= 1100
  998.         return HashKey<DWORD>((DWORD)var.iVal);
  999. #else
  1000.         return HashKey((DWORD)var.iVal);
  1001. #endif
  1002.     case VT_I4:
  1003. #if _MSC_VER >= 1100
  1004.         return HashKey<DWORD>((DWORD)var.lVal);
  1005. #else
  1006.         return HashKey((DWORD)var.lVal);
  1007. #endif
  1008.     case VT_R4:
  1009.         return (UINT)(var.fltVal / 16);
  1010.     case VT_R8:
  1011.     case VT_CY:
  1012.         return (UINT)(var.dblVal / 16);
  1013.     case VT_BOOL:
  1014. #if _MSC_VER >= 1100
  1015.         return HashKey<DWORD>((DWORD)V_BOOL(&var));
  1016. #else
  1017.         return HashKey((DWORD)V_BOOL(&var));
  1018. #endif
  1019.     case VT_ERROR:
  1020. #if _MSC_VER >= 1100
  1021.         return HashKey<DWORD>((DWORD)var.scode);
  1022. #else
  1023.         return HashKey((DWORD)var.scode);
  1024. #endif
  1025.     case VT_DATE:
  1026.         return (UINT)(var.date / 16);
  1027.     case VT_BSTR:
  1028. #if _MSC_VER >= 1100
  1029.         return HashKey<LPCOLESTR>(var.bstrVal);
  1030. #else
  1031.         return HashKey((LPCOLESTR)var.bstrVal);
  1032. #endif
  1033.     case VT_DISPATCH:
  1034.     case VT_UNKNOWN:
  1035. #if _MSC_VER >= 1100
  1036.         return HashKey<DWORD>((DWORD)var.punkVal);
  1037. #else
  1038.         return HashKey((DWORD)var.punkVal);
  1039. #endif
  1040.  
  1041.     default:
  1042.         // No support for VT_BYREF, VT_ARRAY, VT_VARIANT, VT_DECIMAL, & VT_UI1
  1043.         ASSERT(FALSE);
  1044.  
  1045.         // Fall through
  1046.     }
  1047.  
  1048.     return 0;
  1049. }
  1050.  
  1051. /////////////////////////////////////////////////////////////////////////////
  1052. // COleCurrency class helpers
  1053.  
  1054. // Return the highest order bit composing dwTarget in wBit
  1055. #define HI_BIT(dwTarget, wBit) \
  1056.     do \
  1057.     { \
  1058.         if (dwTarget != 0) \
  1059.             for (wBit = 32; (dwTarget & (0x00000001 << (wBit-1))) == 0; wBit--);\
  1060.         else \
  1061.             wBit = 0; \
  1062.     } while (0)
  1063.  
  1064. // Left shift an (assumed unsigned) currency by wBits
  1065. #define LSHIFT_UCUR(cur, wBits) \
  1066.     do \
  1067.     { \
  1068.         for (WORD wTempBits = wBits; wTempBits > 0; wTempBits--) \
  1069.         { \
  1070.             cur.m_cur.Hi = ((DWORD)cur.m_cur.Hi << 1); \
  1071.             cur.m_cur.Hi |= (cur.m_cur.Lo & 0x80000000) >> 31; \
  1072.             cur.m_cur.Lo = cur.m_cur.Lo << 1; \
  1073.         } \
  1074.     } while (0)
  1075.  
  1076. // Right shift an (assumed unsigned) currency by wBits
  1077. #define RSHIFT_UCUR(cur, wBits) \
  1078.     do \
  1079.     { \
  1080.         for (WORD wTempBits = wBits; wTempBits > 0; wTempBits--) \
  1081.         { \
  1082.             cur.m_cur.Lo = cur.m_cur.Lo >> 1; \
  1083.             cur.m_cur.Lo |= (cur.m_cur.Hi & 0x00000001) << 31; \
  1084.             cur.m_cur.Hi = ((DWORD)cur.m_cur.Hi >> 1); \
  1085.         } \
  1086.     } while (0)
  1087.  
  1088. /////////////////////////////////////////////////////////////////////////////
  1089. // COleCurrency class (internally currency is 8-byte int scaled by 10,000)
  1090.  
  1091. COleCurrency::COleCurrency(long nUnits, long nFractionalUnits)
  1092. {
  1093.     SetCurrency(nUnits, nFractionalUnits);
  1094.     SetStatus(valid);
  1095. }
  1096.  
  1097. const COleCurrency& COleCurrency::operator=(CURRENCY cySrc)
  1098. {
  1099.     m_cur = cySrc;
  1100.     SetStatus(valid);
  1101.     return *this;
  1102. }
  1103.  
  1104. const COleCurrency& COleCurrency::operator=(const COleCurrency& curSrc)
  1105. {
  1106.     m_cur = curSrc.m_cur;
  1107.     m_status = curSrc.m_status;
  1108.     return *this;
  1109. }
  1110.  
  1111. const COleCurrency& COleCurrency::operator=(const VARIANT& varSrc)
  1112. {
  1113.     if (varSrc.vt != VT_CY)
  1114.     {
  1115.         TRY
  1116.         {
  1117.             COleVariant varTemp(varSrc);
  1118.             varTemp.ChangeType(VT_CY);
  1119.             m_cur = varTemp.cyVal;
  1120.             SetStatus(valid);
  1121.         }
  1122.         // Catch COleException from ChangeType, but not CMemoryException
  1123.         CATCH(COleException, e)
  1124.         {
  1125.             // Not able to convert VARIANT to CURRENCY
  1126.             m_cur.Hi = 0;
  1127.             m_cur.Lo = 0;
  1128.             SetStatus(invalid);
  1129.             DELETE_EXCEPTION(e);
  1130.         }
  1131.         END_CATCH
  1132.     }
  1133.     else
  1134.     {
  1135.         m_cur = varSrc.cyVal;
  1136.         SetStatus(valid);
  1137.     }
  1138.  
  1139.     return *this;
  1140. }
  1141.  
  1142. BOOL COleCurrency::operator<(const COleCurrency& cur) const
  1143. {
  1144.     ASSERT(GetStatus() == valid);
  1145.     ASSERT(cur.GetStatus() == valid);
  1146.  
  1147.     return((m_cur.Hi == cur.m_cur.Hi) ?
  1148.         (m_cur.Lo < cur.m_cur.Lo) : (m_cur.Hi < cur.m_cur.Hi));
  1149. }
  1150.  
  1151. BOOL COleCurrency::operator>(const COleCurrency& cur) const
  1152. {
  1153.     ASSERT(GetStatus() == valid);
  1154.     ASSERT(cur.GetStatus() == valid);
  1155.  
  1156.     return((m_cur.Hi == cur.m_cur.Hi) ?
  1157.         (m_cur.Lo > cur.m_cur.Lo) : (m_cur.Hi > cur.m_cur.Hi));
  1158. }
  1159.  
  1160. BOOL COleCurrency::operator<=(const COleCurrency& cur) const
  1161. {
  1162.     ASSERT(GetStatus() == valid);
  1163.     ASSERT(cur.GetStatus() == valid);
  1164.  
  1165.     return((m_cur.Hi == cur.m_cur.Hi) ?
  1166.         (m_cur.Lo <= cur.m_cur.Lo) : (m_cur.Hi < cur.m_cur.Hi));
  1167. }
  1168.  
  1169. BOOL COleCurrency::operator>=(const COleCurrency& cur) const
  1170. {
  1171.     ASSERT(GetStatus() == valid);
  1172.     ASSERT(cur.GetStatus() == valid);
  1173.  
  1174.     return((m_cur.Hi == cur.m_cur.Hi) ?
  1175.         (m_cur.Lo >= cur.m_cur.Lo) : (m_cur.Hi > cur.m_cur.Hi));
  1176. }
  1177.  
  1178. COleCurrency COleCurrency::operator+(const COleCurrency& cur) const
  1179. {
  1180.     COleCurrency curResult;
  1181.  
  1182.     // If either operand Null, result Null
  1183.     if (GetStatus() == null || cur.GetStatus() == null)
  1184.     {
  1185.         curResult.SetStatus(null);
  1186.         return curResult;
  1187.     }
  1188.  
  1189.     // If either operand Invalid, result Invalid
  1190.     if (GetStatus() == invalid || cur.GetStatus() == invalid)
  1191.     {
  1192.         curResult.SetStatus(invalid);
  1193.         return curResult;
  1194.     }
  1195.  
  1196.     // Add separate CURRENCY components
  1197.     curResult.m_cur.Hi = m_cur.Hi + cur.m_cur.Hi;
  1198.     curResult.m_cur.Lo = m_cur.Lo + cur.m_cur.Lo;
  1199.  
  1200.     // Increment Hi if Lo overflows
  1201.     if (m_cur.Lo > curResult.m_cur.Lo)
  1202.         curResult.m_cur.Hi++;
  1203.  
  1204.     // Overflow if operands same sign and result sign different
  1205.     if (!((m_cur.Hi ^ cur.m_cur.Hi) & 0x80000000) &&
  1206.         ((m_cur.Hi ^ curResult.m_cur.Hi) & 0x80000000))
  1207.     {
  1208.         curResult.SetStatus(invalid);
  1209.     }
  1210.  
  1211.     return curResult;
  1212. }
  1213.  
  1214. COleCurrency COleCurrency::operator-(const COleCurrency& cur) const
  1215. {
  1216.     COleCurrency curResult;
  1217.  
  1218.     // If either operand Null, result Null
  1219.     if (GetStatus() == null || cur.GetStatus() == null)
  1220.     {
  1221.         curResult.SetStatus(null);
  1222.         return curResult;
  1223.     }
  1224.  
  1225.     // If either operand Invalid, result Invalid
  1226.     if (GetStatus() == invalid || cur.GetStatus() == invalid)
  1227.     {
  1228.         curResult.SetStatus(invalid);
  1229.         return curResult;
  1230.     }
  1231.  
  1232.     // Subtract separate CURRENCY components
  1233.     curResult.m_cur.Hi = m_cur.Hi - cur.m_cur.Hi;
  1234.     curResult.m_cur.Lo = m_cur.Lo - cur.m_cur.Lo;
  1235.  
  1236.     // Decrement Hi if Lo overflows
  1237.     if (m_cur.Lo < curResult.m_cur.Lo)
  1238.         curResult.m_cur.Hi--;
  1239.  
  1240.     // Overflow if operands not same sign and result not same sign
  1241.     if (((m_cur.Hi ^ cur.m_cur.Hi) & 0x80000000) &&
  1242.         ((m_cur.Hi ^ curResult.m_cur.Hi) & 0x80000000))
  1243.     {
  1244.         curResult.SetStatus(invalid);
  1245.     }
  1246.  
  1247.     return curResult;
  1248. }
  1249.  
  1250. COleCurrency COleCurrency::operator-() const
  1251. {
  1252.     // If operand not Valid, just return
  1253.     if (!GetStatus() == valid)
  1254.         return *this;
  1255.  
  1256.     COleCurrency curResult;
  1257.  
  1258.     // Negating MIN_CURRENCY,will set invalid
  1259.     if (m_cur.Hi == 0x80000000 && m_cur.Lo == 0x00000000)
  1260.     {
  1261.         curResult.SetStatus(invalid);
  1262.     }
  1263.  
  1264.     curResult.m_cur.Hi = ~m_cur.Hi;
  1265.     curResult.m_cur.Lo = -(long)m_cur.Lo;
  1266.  
  1267.     // If cy was -1 make sure Hi correctly set
  1268.     if (curResult.m_cur.Lo == 0)
  1269.         curResult.m_cur.Hi++;
  1270.  
  1271.     return curResult;
  1272. }
  1273.  
  1274. COleCurrency COleCurrency::operator*(long nOperand) const
  1275. {
  1276.     // If operand not Valid, just return
  1277.     if (!GetStatus() == valid)
  1278.         return *this;
  1279.  
  1280.     COleCurrency curResult(m_cur);
  1281.     DWORD nTempOp;
  1282.  
  1283.     // Return now if one operand is 0 (optimization)
  1284.     if ((m_cur.Hi == 0x00000000 && m_cur.Lo == 0x00000000) || nOperand == 0)
  1285.     {
  1286.         curResult.m_cur.Hi = 0;
  1287.         curResult.m_cur.Lo = 0;
  1288.         return curResult;
  1289.     }
  1290.  
  1291.     // Handle only valid case of multiplying MIN_CURRENCY
  1292.     if (m_cur.Hi == 0x80000000 && m_cur.Lo == 0x00000000 && nOperand == 1)
  1293.         return curResult;
  1294.  
  1295.     // Compute absolute values.
  1296.     if (m_cur.Hi < 0)
  1297.         curResult = -curResult;
  1298.  
  1299.     nTempOp = labs(nOperand);
  1300.  
  1301.     // Check for overflow
  1302.     if (curResult.m_cur.Hi != 0)
  1303.     {
  1304.         WORD wHiBitCur, wHiBitOp;
  1305.         HI_BIT(curResult.m_cur.Hi, wHiBitCur);
  1306.         HI_BIT(nTempOp, wHiBitOp);
  1307.  
  1308.         // 63-bit limit on result. (n bits)*(m bits) = (n+m-1) bits.
  1309.         if (wHiBitCur + wHiBitOp - 1 > 63)
  1310.         {
  1311.             // Overflow!
  1312.             curResult.SetStatus(invalid);
  1313.  
  1314.             // Set to maximum negative value
  1315.             curResult.m_cur.Hi = 0x80000000;
  1316.             curResult.m_cur.Lo = 0x00000000;
  1317.  
  1318.             return curResult;
  1319.         }
  1320.     }
  1321.  
  1322.     // Break up into WORDs
  1323.     WORD wCy4, wCy3, wCy2, wCy1, wL2, wL1;
  1324.  
  1325.     wCy4 = HIWORD(curResult.m_cur.Hi);
  1326.     wCy3 = LOWORD(curResult.m_cur.Hi);
  1327.     wCy2 = HIWORD(curResult.m_cur.Lo);
  1328.     wCy1 = LOWORD(curResult.m_cur.Lo);
  1329.  
  1330.     wL2 = HIWORD(nTempOp);
  1331.     wL1 = LOWORD(nTempOp);
  1332.  
  1333.     // Multiply each set of WORDs
  1334.     DWORD dwRes11, dwRes12, dwRes21, dwRes22;
  1335.     DWORD dwRes31, dwRes32, dwRes41;  // Don't need dwRes42
  1336.  
  1337.     dwRes11 = wCy1 * wL1;
  1338.     dwRes12 = wCy1 * wL2;
  1339.     dwRes21 = wCy2 * wL1;
  1340.     dwRes22 = wCy2 * wL2;
  1341.  
  1342.     dwRes31 = wCy3 * wL1;
  1343.     dwRes32 = wCy3 * wL2;
  1344.     dwRes41 = wCy4 * wL1;
  1345.  
  1346.     // Add up low order pieces
  1347.     dwRes11 += dwRes12<<16;
  1348.     curResult.m_cur.Lo = dwRes11 + (dwRes21<<16);
  1349.     curResult.m_cur.Hi = 0;
  1350.  
  1351.     // Check if carry required
  1352.     if (dwRes11 < dwRes12<<16)
  1353.         curResult.m_cur.Hi++;
  1354.     if ((DWORD)curResult.m_cur.Lo < dwRes11)
  1355.         curResult.m_cur.Hi++;
  1356.  
  1357.     // Add up the high order pieces
  1358.     curResult.m_cur.Hi += dwRes31 + (dwRes32<<16) + (dwRes41<<16) +
  1359.         dwRes22 + (dwRes12>>16) + (dwRes21>>16);
  1360.  
  1361.     // Compute result sign
  1362.     if ((m_cur.Hi ^ nOperand) & 0x80000000)
  1363.         curResult = -curResult;
  1364.  
  1365.     return curResult;
  1366. }
  1367.  
  1368. COleCurrency COleCurrency::operator/(long nOperand) const
  1369. {
  1370.     // If operand not Valid, just return
  1371.     if (!GetStatus() == valid)
  1372.         return *this;
  1373.  
  1374.     COleCurrency curTemp(m_cur);
  1375.     DWORD nTempOp;
  1376.  
  1377.     // Check for divide by 0
  1378.     if (nOperand == 0)
  1379.     {
  1380.         curTemp.SetStatus(invalid);
  1381.  
  1382.         // Set to maximum negative value
  1383.         curTemp.m_cur.Hi = 0x80000000;
  1384.         curTemp.m_cur.Lo = 0x00000000;
  1385.  
  1386.         return curTemp;
  1387.     }
  1388.  
  1389.     // Compute absolute values
  1390.     if (curTemp.m_cur.Hi < 0)
  1391.         curTemp = -curTemp;
  1392.  
  1393.     nTempOp = labs(nOperand);
  1394.  
  1395.     // Optimization - division is simple if Hi == 0
  1396.     if (curTemp.m_cur.Hi == 0x0000)
  1397.     {
  1398.         curTemp.m_cur.Lo = m_cur.Lo / nTempOp;
  1399.  
  1400.         // Compute result sign
  1401.         if ((m_cur.Hi ^ nOperand) & 0x80000000)
  1402.             curTemp = -curTemp;
  1403.  
  1404.         return curTemp;
  1405.     }
  1406.  
  1407.     // Now curTemp represents remainder
  1408.     COleCurrency curResult; // Initializes to zero
  1409.     COleCurrency curTempResult;
  1410.     COleCurrency curOperand;
  1411.  
  1412.     curOperand.m_cur.Lo = nTempOp;
  1413.  
  1414.     WORD wHiBitRem;
  1415.     WORD wScaleOp;
  1416.  
  1417.     // Quit if remainder can be truncated
  1418.     while (curTemp >= curOperand)
  1419.     {
  1420.         // Scale up and divide Hi portion
  1421.         HI_BIT(curTemp.m_cur.Hi, wHiBitRem);
  1422.  
  1423.         if (wHiBitRem != 0)
  1424.             wHiBitRem += 32;
  1425.         else
  1426.             HI_BIT(curTemp.m_cur.Lo, wHiBitRem);
  1427.  
  1428.         WORD wShift = (WORD)(64 - wHiBitRem);
  1429.         LSHIFT_UCUR(curTemp, wShift);
  1430.  
  1431.         // If Operand bigger than Hi it must be scaled
  1432.         wScaleOp = (WORD)((nTempOp > (DWORD)curTemp.m_cur.Hi) ? 1 : 0);
  1433.  
  1434.         // Perform synthetic division
  1435.         curTempResult.m_cur.Hi =
  1436.             (DWORD)curTemp.m_cur.Hi / (nTempOp >> wScaleOp);
  1437.  
  1438.         // Scale back to get correct result and remainder
  1439.         RSHIFT_UCUR(curTemp, wShift);
  1440.         wShift = (WORD)(wShift - wScaleOp);
  1441.         RSHIFT_UCUR(curTempResult, wShift);
  1442.  
  1443.         // Now calculate result and remainder
  1444.         curResult += curTempResult;
  1445.         curTemp -= curTempResult * nTempOp;
  1446.     }
  1447.  
  1448.     // Compute result sign
  1449.     if ((m_cur.Hi ^ nOperand) & 0x80000000)
  1450.         curResult = -curResult;
  1451.  
  1452.     return curResult;
  1453. }
  1454.  
  1455. void COleCurrency::SetCurrency(long nUnits, long nFractionalUnits)
  1456. {
  1457.     COleCurrency curUnits;              // Initializes to 0
  1458.     COleCurrency curFractionalUnits;    // Initializes to 0
  1459.  
  1460.     // Set temp currency value to Units (need to multiply by 10,000)
  1461.     curUnits.m_cur.Lo = (DWORD)labs(nUnits);
  1462.     curUnits = curUnits * 10000;
  1463.     if (nUnits < 0)
  1464.         curUnits = -curUnits;
  1465.  
  1466.     curFractionalUnits.m_cur.Lo = (DWORD)labs(nFractionalUnits);
  1467.     if (nFractionalUnits < 0)
  1468.         curFractionalUnits = -curFractionalUnits;
  1469.  
  1470.     // Now add together Units and FractionalUnits
  1471.     *this = curUnits + curFractionalUnits;
  1472.  
  1473.     SetStatus(valid);
  1474. }
  1475.  
  1476. BOOL COleCurrency::ParseCurrency(LPCTSTR lpszCurrency,
  1477.     DWORD dwFlags,  LCID lcid)
  1478. {
  1479.     USES_CONVERSION;
  1480.     CString strCurrency = lpszCurrency;
  1481.  
  1482.     SCODE sc;
  1483.     if ( FAILED(sc = VarCyFromStr((LPOLESTR)T2COLE(strCurrency),
  1484.         lcid, dwFlags, &m_cur)))
  1485.     {
  1486.         if (sc == DISP_E_TYPEMISMATCH)
  1487.         {
  1488.             // Can't convert string to CURRENCY, set 0 & invalid
  1489.             m_cur.Hi = 0x00000000;
  1490.             m_cur.Lo = 0x00000000;
  1491.             SetStatus(invalid);
  1492.             return FALSE;
  1493.         }
  1494.         else if (sc == DISP_E_OVERFLOW)
  1495.         {
  1496.             // Can't convert string to CURRENCY, set max neg & invalid
  1497.             m_cur.Hi = 0x80000000;
  1498.             m_cur.Lo = 0x00000000;
  1499.             SetStatus(invalid);
  1500.             return FALSE;
  1501.         }
  1502.         else
  1503.         {
  1504.             TRACE0("\nCOleCurrency VarCyFromStr call failed.\n\t");
  1505.             if (sc == E_OUTOFMEMORY)
  1506.                 AfxThrowMemoryException();
  1507.             else
  1508.                 AfxThrowOleException(sc);
  1509.         }
  1510.     }
  1511.  
  1512.     SetStatus(valid);
  1513.     return TRUE;
  1514. }
  1515.  
  1516. CString COleCurrency::Format(DWORD dwFlags, LCID lcid) const
  1517. {
  1518.     USES_CONVERSION;
  1519.     CString strCur;
  1520.  
  1521.     // If null, return empty string
  1522.     if (GetStatus() == null)
  1523.         return strCur;
  1524.  
  1525.     // If invalid, return Currency resource string
  1526.     if (GetStatus() == invalid)
  1527.     {
  1528.         VERIFY(strCur.LoadString(AFX_IDS_INVALID_CURRENCY));
  1529.         return strCur;
  1530.     }
  1531.  
  1532.     COleVariant var;
  1533.     // Don't need to trap error. Should not fail due to type mismatch
  1534.     AfxCheckError(VarBstrFromCy(m_cur, lcid, dwFlags, &V_BSTR(&var)));
  1535.     var.vt = VT_BSTR;
  1536.     return OLE2CT(V_BSTR(&var));
  1537. }
  1538.  
  1539.  
  1540. // serialization
  1541. #ifdef _DEBUG
  1542. CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleCurrency curSrc)
  1543. {
  1544.     dc << "\nCOleCurrency Object:";
  1545.     dc << "\n\tm_status = " << (long)curSrc.m_status;
  1546.  
  1547.     COleVariant var(curSrc);
  1548.     var.ChangeType(VT_CY);
  1549.     return dc << "\n\tCurrency = " << var.bstrVal;
  1550. }
  1551. #endif // _DEBUG
  1552.  
  1553. CArchive& AFXAPI operator<<(CArchive& ar, COleCurrency curSrc)
  1554. {
  1555.     ar << (long)curSrc.m_status;
  1556.     ar << curSrc.m_cur.Hi;
  1557.     return ar << curSrc.m_cur.Lo;
  1558. }
  1559.  
  1560. CArchive& AFXAPI operator>>(CArchive& ar, COleCurrency& curSrc)
  1561. {
  1562.     ar >> (long&)curSrc.m_status;
  1563.     ar >> curSrc.m_cur.Hi;
  1564.     return ar >> curSrc.m_cur.Lo;
  1565. }
  1566.  
  1567.  
  1568. /////////////////////////////////////////////////////////////////////////////
  1569. // COleDateTime class HELPER definitions
  1570.  
  1571. // Verifies will fail if the needed buffer size is too large
  1572. #define MAX_TIME_BUFFER_SIZE    128         // matches that in timecore.cpp
  1573. #define MIN_DATE                (-657434L)  // about year 100
  1574. #define MAX_DATE                2958465L    // about year 9999
  1575.  
  1576. // Half a second, expressed in days
  1577. #define HALF_SECOND  (1.0/172800.0)
  1578.  
  1579. // One-based array of days in year at month start
  1580. AFX_STATIC_DATA int _afxMonthDays[13] =
  1581.     {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365};
  1582.  
  1583. /////////////////////////////////////////////////////////////////////////////
  1584. // COleDateTime class HELPERS - implementation
  1585.  
  1586. AFX_STATIC BOOL AFXAPI _AfxOleDateFromTm(WORD wYear, WORD wMonth, WORD wDay,
  1587.     WORD wHour, WORD wMinute, WORD wSecond, DATE& dtDest)
  1588. {
  1589.     // Validate year and month (ignore day of week and milliseconds)
  1590.     if (wYear > 9999 || wMonth < 1 || wMonth > 12)
  1591.         return FALSE;
  1592.  
  1593.     //  Check for leap year and set the number of days in the month
  1594.     BOOL bLeapYear = ((wYear & 3) == 0) &&
  1595.         ((wYear % 100) != 0 || (wYear % 400) == 0);
  1596.  
  1597.     int nDaysInMonth =
  1598.         _afxMonthDays[wMonth] - _afxMonthDays[wMonth-1] +
  1599.         ((bLeapYear && wDay == 29 && wMonth == 2) ? 1 : 0);
  1600.  
  1601.     // Finish validating the date
  1602.     if (wDay < 1 || wDay > nDaysInMonth ||
  1603.         wHour > 23 || wMinute > 59 ||
  1604.         wSecond > 59)
  1605.     {
  1606.         return FALSE;
  1607.     }
  1608.  
  1609.     // Cache the date in days and time in fractional days
  1610.     long nDate;
  1611.     double dblTime;
  1612.  
  1613.     //It is a valid date; make Jan 1, 1AD be 1
  1614.     nDate = wYear*365L + wYear/4 - wYear/100 + wYear/400 +
  1615.         _afxMonthDays[wMonth-1] + wDay;
  1616.  
  1617.     //  If leap year and it's before March, subtract 1:
  1618.     if (wMonth <= 2 && bLeapYear)
  1619.         --nDate;
  1620.  
  1621.     //  Offset so that 12/30/1899 is 0
  1622.     nDate -= 693959L;
  1623.  
  1624.     dblTime = (((long)wHour * 3600L) +  // hrs in seconds
  1625.         ((long)wMinute * 60L) +  // mins in seconds
  1626.         ((long)wSecond)) / 86400.;
  1627.  
  1628.     dtDest = (double) nDate + ((nDate >= 0) ? dblTime : -dblTime);
  1629.  
  1630.     return TRUE;
  1631. }
  1632.  
  1633. AFX_STATIC BOOL AFXAPI _AfxTmFromOleDate(DATE dtSrc, struct tm& tmDest)
  1634. {
  1635.     // The legal range does not actually span year 0 to 9999.
  1636.     if (dtSrc > MAX_DATE || dtSrc < MIN_DATE) // about year 100 to about 9999
  1637.         return FALSE;
  1638.  
  1639.     long nDays;             // Number of days since Dec. 30, 1899
  1640.     long nDaysAbsolute;     // Number of days since 1/1/0
  1641.     long nSecsInDay;        // Time in seconds since midnight
  1642.     long nMinutesInDay;     // Minutes in day
  1643.  
  1644.     long n400Years;         // Number of 400 year increments since 1/1/0
  1645.     long n400Century;       // Century within 400 year block (0,1,2 or 3)
  1646.     long n4Years;           // Number of 4 year increments since 1/1/0
  1647.     long n4Day;             // Day within 4 year block
  1648.                             //  (0 is 1/1/yr1, 1460 is 12/31/yr4)
  1649.     long n4Yr;              // Year within 4 year block (0,1,2 or 3)
  1650.     BOOL bLeap4 = TRUE;     // TRUE if 4 year block includes leap year
  1651.  
  1652.     double dblDate = dtSrc; // tempory serial date
  1653.  
  1654.     // If a valid date, then this conversion should not overflow
  1655.     nDays = (long)dblDate;
  1656.  
  1657.     // Round to the second
  1658.     dblDate += ((dtSrc > 0.0) ? HALF_SECOND : -HALF_SECOND);
  1659.  
  1660.     nDaysAbsolute = (long)dblDate + 693959L; // Add days from 1/1/0 to 12/30/1899
  1661.  
  1662.     dblDate = fabs(dblDate);
  1663.     nSecsInDay = (long)((dblDate - floor(dblDate)) * 86400.);
  1664.  
  1665.     // Calculate the day of week (sun=1, mon=2...)
  1666.     //   -1 because 1/1/0 is Sat.  +1 because we want 1-based
  1667.     tmDest.tm_wday = (int)((nDaysAbsolute - 1) % 7L) + 1;
  1668.  
  1669.     // Leap years every 4 yrs except centuries not multiples of 400.
  1670.     n400Years = (long)(nDaysAbsolute / 146097L);
  1671.  
  1672.     // Set nDaysAbsolute to day within 400-year block
  1673.     nDaysAbsolute %= 146097L;
  1674.  
  1675.     // -1 because first century has extra day
  1676.     n400Century = (long)((nDaysAbsolute - 1) / 36524L);
  1677.  
  1678.     // Non-leap century
  1679.     if (n400Century != 0)
  1680.     {
  1681.         // Set nDaysAbsolute to day within century
  1682.         nDaysAbsolute = (nDaysAbsolute - 1) % 36524L;
  1683.  
  1684.         // +1 because 1st 4 year increment has 1460 days
  1685.         n4Years = (long)((nDaysAbsolute + 1) / 1461L);
  1686.  
  1687.         if (n4Years != 0)
  1688.             n4Day = (long)((nDaysAbsolute + 1) % 1461L);
  1689.         else
  1690.         {
  1691.             bLeap4 = FALSE;
  1692.             n4Day = (long)nDaysAbsolute;
  1693.         }
  1694.     }
  1695.     else
  1696.     {
  1697.         // Leap century - not special case!
  1698.         n4Years = (long)(nDaysAbsolute / 1461L);
  1699.         n4Day = (long)(nDaysAbsolute % 1461L);
  1700.     }
  1701.  
  1702.     if (bLeap4)
  1703.     {
  1704.         // -1 because first year has 366 days
  1705.         n4Yr = (n4Day - 1) / 365;
  1706.  
  1707.         if (n4Yr != 0)
  1708.             n4Day = (n4Day - 1) % 365;
  1709.     }
  1710.     else
  1711.     {
  1712.         n4Yr = n4Day / 365;
  1713.         n4Day %= 365;
  1714.     }
  1715.  
  1716.     // n4Day is now 0-based day of year. Save 1-based day of year, year number
  1717.     tmDest.tm_yday = (int)n4Day + 1;
  1718.     tmDest.tm_year = n400Years * 400 + n400Century * 100 + n4Years * 4 + n4Yr;
  1719.  
  1720.     // Handle leap year: before, on, and after Feb. 29.
  1721.     if (n4Yr == 0 && bLeap4)
  1722.     {
  1723.         // Leap Year
  1724.         if (n4Day == 59)
  1725.         {
  1726.             /* Feb. 29 */
  1727.             tmDest.tm_mon = 2;
  1728.             tmDest.tm_mday = 29;
  1729.             goto DoTime;
  1730.         }
  1731.  
  1732.         // Pretend it's not a leap year for month/day comp.
  1733.         if (n4Day >= 60)
  1734.             --n4Day;
  1735.     }
  1736.  
  1737.     // Make n4DaY a 1-based day of non-leap year and compute
  1738.     //  month/day for everything but Feb. 29.
  1739.     ++n4Day;
  1740.  
  1741.     // Month number always >= n/32, so save some loop time */
  1742.     for (tmDest.tm_mon = (n4Day >> 5) + 1;
  1743.         n4Day > _afxMonthDays[tmDest.tm_mon]; tmDest.tm_mon++);
  1744.  
  1745.     tmDest.tm_mday = (int)(n4Day - _afxMonthDays[tmDest.tm_mon-1]);
  1746.  
  1747. DoTime:
  1748.     if (nSecsInDay == 0)
  1749.         tmDest.tm_hour = tmDest.tm_min = tmDest.tm_sec = 0;
  1750.     else
  1751.     {
  1752.         tmDest.tm_sec = (int)nSecsInDay % 60L;
  1753.         nMinutesInDay = nSecsInDay / 60L;
  1754.         tmDest.tm_min = (int)nMinutesInDay % 60;
  1755.         tmDest.tm_hour = (int)nMinutesInDay / 60;
  1756.     }
  1757.  
  1758.     return TRUE;
  1759. }
  1760.  
  1761. AFX_STATIC void AFXAPI _AfxTmConvertToStandardFormat(struct tm& tmSrc)
  1762. {
  1763.     // Convert afx internal tm to format expected by runtimes (_tcsftime, etc)
  1764.     tmSrc.tm_year -= 1900;  // year is based on 1900
  1765.     tmSrc.tm_mon -= 1;      // month of year is 0-based
  1766.     tmSrc.tm_wday -= 1;     // day of week is 0-based
  1767.     tmSrc.tm_yday -= 1;     // day of year is 0-based
  1768. }
  1769.  
  1770. AFX_STATIC double AFXAPI _AfxDoubleFromDate(DATE dt)
  1771. {
  1772.     // No problem if positive
  1773.     if (dt >= 0)
  1774.         return dt;
  1775.  
  1776.     // If negative, must convert since negative dates not continuous
  1777.     // (examples: -1.25 to -.75, -1.50 to -.50, -1.75 to -.25)
  1778.     double temp = ceil(dt);
  1779.     return temp - (dt - temp);
  1780. }
  1781.  
  1782. AFX_STATIC DATE AFXAPI _AfxDateFromDouble(double dbl)
  1783. {
  1784.     // No problem if positive
  1785.     if (dbl >= 0)
  1786.         return dbl;
  1787.  
  1788.     // If negative, must convert since negative dates not continuous
  1789.     // (examples: -.75 to -1.25, -.50 to -1.50, -.25 to -1.75)
  1790.     double temp = floor(dbl); // dbl is now whole part
  1791.     return temp + (temp - dbl);
  1792. }
  1793.  
  1794. /////////////////////////////////////////////////////////////////////////////
  1795. // COleDateTime class
  1796.  
  1797. COleDateTime PASCAL COleDateTime::GetCurrentTime()
  1798. {
  1799.     return COleDateTime(::time(NULL));
  1800. }
  1801.  
  1802. BOOL COleDateTime::GetAsSystemTime(SYSTEMTIME& sysTime) const
  1803. {
  1804.     BOOL bRetVal = FALSE;
  1805.     if (GetStatus() == valid)
  1806.     {
  1807.         struct tm tmTemp;
  1808.         if (_AfxTmFromOleDate(m_dt, tmTemp))
  1809.         {
  1810.             sysTime.wYear = (WORD) tmTemp.tm_year;
  1811.             sysTime.wMonth = (WORD) tmTemp.tm_mon;
  1812.             sysTime.wDayOfWeek = (WORD) (tmTemp.tm_wday - 1);
  1813.             sysTime.wDay = (WORD) tmTemp.tm_mday;
  1814.             sysTime.wHour = (WORD) tmTemp.tm_hour;
  1815.             sysTime.wMinute = (WORD) tmTemp.tm_min;
  1816.             sysTime.wSecond = (WORD) tmTemp.tm_sec;
  1817.             sysTime.wMilliseconds = 0;
  1818.  
  1819.             bRetVal = TRUE;
  1820.         }
  1821.     }
  1822.  
  1823.     return bRetVal;
  1824. }
  1825.  
  1826. int COleDateTime::GetYear() const
  1827. {
  1828.     struct tm tmTemp;
  1829.  
  1830.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1831.         return tmTemp.tm_year;
  1832.     else
  1833.         return AFX_OLE_DATETIME_ERROR;
  1834. }
  1835.  
  1836. int COleDateTime::GetMonth() const
  1837. {
  1838.     struct tm tmTemp;
  1839.  
  1840.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1841.         return tmTemp.tm_mon;
  1842.     else
  1843.         return AFX_OLE_DATETIME_ERROR;
  1844. }
  1845.  
  1846. int COleDateTime::GetDay() const
  1847. {
  1848.     struct tm tmTemp;
  1849.  
  1850.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1851.         return tmTemp.tm_mday;
  1852.     else
  1853.         return AFX_OLE_DATETIME_ERROR;
  1854. }
  1855.  
  1856. int COleDateTime::GetHour() const
  1857. {
  1858.     struct tm tmTemp;
  1859.  
  1860.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1861.         return tmTemp.tm_hour;
  1862.     else
  1863.         return AFX_OLE_DATETIME_ERROR;
  1864. }
  1865.  
  1866. int COleDateTime::GetMinute() const
  1867. {
  1868.     struct tm tmTemp;
  1869.  
  1870.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1871.         return tmTemp.tm_min;
  1872.     else
  1873.         return AFX_OLE_DATETIME_ERROR;
  1874. }
  1875.  
  1876. int COleDateTime::GetSecond() const
  1877. {
  1878.     struct tm tmTemp;
  1879.  
  1880.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1881.         return tmTemp.tm_sec;
  1882.     else
  1883.         return AFX_OLE_DATETIME_ERROR;
  1884. }
  1885.  
  1886. int COleDateTime::GetDayOfWeek() const
  1887. {
  1888.     struct tm tmTemp;
  1889.  
  1890.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1891.         return tmTemp.tm_wday;
  1892.     else
  1893.         return AFX_OLE_DATETIME_ERROR;
  1894. }
  1895.  
  1896. int COleDateTime::GetDayOfYear() const
  1897. {
  1898.     struct tm tmTemp;
  1899.  
  1900.     if (GetStatus() == valid && _AfxTmFromOleDate(m_dt, tmTemp))
  1901.         return tmTemp.tm_yday;
  1902.     else
  1903.         return AFX_OLE_DATETIME_ERROR;
  1904. }
  1905.  
  1906. const COleDateTime& COleDateTime::operator=(const VARIANT& varSrc)
  1907. {
  1908.     if (varSrc.vt != VT_DATE)
  1909.     {
  1910.         TRY
  1911.         {
  1912.             COleVariant varTemp(varSrc);
  1913.             varTemp.ChangeType(VT_DATE);
  1914.             m_dt = varTemp.date;
  1915.             SetStatus(valid);
  1916.         }
  1917.         // Catch COleException from ChangeType, but not CMemoryException
  1918.         CATCH(COleException, e)
  1919.         {
  1920.             // Not able to convert VARIANT to DATE
  1921.             DELETE_EXCEPTION(e);
  1922.             m_dt = 0;
  1923.             SetStatus(invalid);
  1924.         }
  1925.         END_CATCH
  1926.     }
  1927.     else
  1928.     {
  1929.         m_dt = varSrc.date;
  1930.         SetStatus(valid);
  1931.     }
  1932.  
  1933.     return *this;
  1934. }
  1935.  
  1936. const COleDateTime& COleDateTime::operator=(DATE dtSrc)
  1937. {
  1938.     m_dt = dtSrc;
  1939.     SetStatus(valid);
  1940.  
  1941.     return *this;
  1942. }
  1943.  
  1944. const COleDateTime& COleDateTime::operator=(const time_t& timeSrc)
  1945. {
  1946.     // Convert time_t to struct tm
  1947.     tm *ptm = localtime(&timeSrc);
  1948.  
  1949.     if (ptm != NULL)
  1950.     {
  1951.         m_status = _AfxOleDateFromTm((WORD)(ptm->tm_year + 1900),
  1952.             (WORD)(ptm->tm_mon + 1), (WORD)ptm->tm_mday,
  1953.             (WORD)ptm->tm_hour, (WORD)ptm->tm_min,
  1954.             (WORD)ptm->tm_sec, m_dt) ? valid : invalid;
  1955.     }
  1956.     else
  1957.     {
  1958.         // Local time must have failed (timsSrc before 1/1/70 12am)
  1959.         SetStatus(invalid);
  1960.         ASSERT(FALSE);
  1961.     }
  1962.  
  1963.     return *this;
  1964. }
  1965.  
  1966. const COleDateTime& COleDateTime::operator=(const SYSTEMTIME& systimeSrc)
  1967. {
  1968.     m_status = _AfxOleDateFromTm(systimeSrc.wYear, systimeSrc.wMonth,
  1969.         systimeSrc.wDay, systimeSrc.wHour, systimeSrc.wMinute,
  1970.         systimeSrc.wSecond, m_dt) ? valid : invalid;
  1971.  
  1972.     return *this;
  1973. }
  1974.  
  1975. const COleDateTime& COleDateTime::operator=(const FILETIME& filetimeSrc)
  1976. {
  1977.     // Assume UTC FILETIME, so convert to LOCALTIME
  1978.     FILETIME filetimeLocal;
  1979.     if (!FileTimeToLocalFileTime( &filetimeSrc, &filetimeLocal))
  1980.     {
  1981. #ifdef _DEBUG
  1982.         DWORD dwError = GetLastError();
  1983.         TRACE1("\nFileTimeToLocalFileTime failed. Error = %lu.\n\t", dwError);
  1984. #endif // _DEBUG
  1985.         m_status = invalid;
  1986.     }
  1987.     else
  1988.     {
  1989.         // Take advantage of SYSTEMTIME -> FILETIME conversion
  1990.         SYSTEMTIME systime;
  1991.         m_status = FileTimeToSystemTime(&filetimeLocal, &systime) ?
  1992.             valid : invalid;
  1993.  
  1994.         // At this point systime should always be valid, but...
  1995.         if (GetStatus() == valid)
  1996.         {
  1997.             m_status = _AfxOleDateFromTm(systime.wYear, systime.wMonth,
  1998.                 systime.wDay, systime.wHour, systime.wMinute,
  1999.                 systime.wSecond, m_dt) ? valid : invalid;
  2000.         }
  2001.     }
  2002.  
  2003.     return *this;
  2004. }
  2005.  
  2006. BOOL COleDateTime::operator<(const COleDateTime& date) const
  2007. {
  2008.     ASSERT(GetStatus() == valid);
  2009.     ASSERT(date.GetStatus() == valid);
  2010.  
  2011.     // Handle negative dates
  2012.     return _AfxDoubleFromDate(m_dt) < _AfxDoubleFromDate(date.m_dt);
  2013. }
  2014.  
  2015. BOOL COleDateTime::operator>(const COleDateTime& date) const
  2016. {   ASSERT(GetStatus() == valid);
  2017.     ASSERT(date.GetStatus() == valid);
  2018.  
  2019.     // Handle negative dates
  2020.     return _AfxDoubleFromDate(m_dt) > _AfxDoubleFromDate(date.m_dt);
  2021. }
  2022.  
  2023. BOOL COleDateTime::operator<=(const COleDateTime& date) const
  2024. {
  2025.     ASSERT(GetStatus() == valid);
  2026.     ASSERT(date.GetStatus() == valid);
  2027.  
  2028.     // Handle negative dates
  2029.     return _AfxDoubleFromDate(m_dt) <= _AfxDoubleFromDate(date.m_dt);
  2030. }
  2031.  
  2032. BOOL COleDateTime::operator>=(const COleDateTime& date) const
  2033. {
  2034.     ASSERT(GetStatus() == valid);
  2035.     ASSERT(date.GetStatus() == valid);
  2036.  
  2037.     // Handle negative dates
  2038.     return _AfxDoubleFromDate(m_dt) >= _AfxDoubleFromDate(date.m_dt);
  2039. }
  2040.  
  2041. COleDateTime COleDateTime::operator+(const COleDateTimeSpan& dateSpan) const
  2042. {
  2043.     COleDateTime dateResult;    // Initializes m_status to valid
  2044.  
  2045.     // If either operand NULL, result NULL
  2046.     if (GetStatus() == null || dateSpan.GetStatus() == null)
  2047.     {
  2048.         dateResult.SetStatus(null);
  2049.         return dateResult;
  2050.     }
  2051.  
  2052.     // If either operand invalid, result invalid
  2053.     if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
  2054.     {
  2055.         dateResult.SetStatus(invalid);
  2056.         return dateResult;
  2057.     }
  2058.  
  2059.     // Compute the actual date difference by adding underlying dates
  2060.     dateResult = _AfxDateFromDouble(_AfxDoubleFromDate(m_dt) + dateSpan.m_span);
  2061.  
  2062.     // Validate within range
  2063.     dateResult.CheckRange();
  2064.  
  2065.     return dateResult;
  2066. }
  2067.  
  2068. COleDateTime COleDateTime::operator-(const COleDateTimeSpan& dateSpan) const
  2069. {
  2070.     COleDateTime dateResult;    // Initializes m_status to valid
  2071.  
  2072.     // If either operand NULL, result NULL
  2073.     if (GetStatus() == null || dateSpan.GetStatus() == null)
  2074.     {
  2075.         dateResult.SetStatus(null);
  2076.         return dateResult;
  2077.     }
  2078.  
  2079.     // If either operand invalid, result invalid
  2080.     if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
  2081.     {
  2082.         dateResult.SetStatus(invalid);
  2083.         return dateResult;
  2084.     }
  2085.  
  2086.     // Compute the actual date difference by subtracting underlying dates
  2087.     dateResult = _AfxDateFromDouble(_AfxDoubleFromDate(m_dt) - dateSpan.m_span);
  2088.  
  2089.     // Validate within range
  2090.     dateResult.CheckRange();
  2091.  
  2092.     return dateResult;
  2093. }
  2094.  
  2095. COleDateTimeSpan COleDateTime::operator-(const COleDateTime& date) const
  2096. {
  2097.     COleDateTimeSpan spanResult;
  2098.  
  2099.     // If either operand NULL, result NULL
  2100.     if (GetStatus() == null || date.GetStatus() == null)
  2101.     {
  2102.         spanResult.SetStatus(COleDateTimeSpan::null);
  2103.         return spanResult;
  2104.     }
  2105.  
  2106.     // If either operand invalid, result invalid
  2107.     if (GetStatus() == invalid || date.GetStatus() == invalid)
  2108.     {
  2109.         spanResult.SetStatus(COleDateTimeSpan::invalid);
  2110.         return spanResult;
  2111.     }
  2112.  
  2113.     // Return result (span can't be invalid, so don't check range)
  2114.     return _AfxDoubleFromDate(m_dt) - _AfxDoubleFromDate(date.m_dt);
  2115. }
  2116.  
  2117. int COleDateTime::SetDateTime(int nYear, int nMonth, int nDay,
  2118.     int nHour, int nMin, int nSec)
  2119. {
  2120.     return m_status = _AfxOleDateFromTm((WORD)nYear, (WORD)nMonth,
  2121.         (WORD)nDay, (WORD)nHour, (WORD)nMin, (WORD)nSec, m_dt) ?
  2122.         valid : invalid;
  2123. }
  2124.  
  2125. BOOL COleDateTime::ParseDateTime(LPCTSTR lpszDate, DWORD dwFlags, LCID lcid)
  2126. {
  2127.     USES_CONVERSION;
  2128.     CString strDate = lpszDate;
  2129.  
  2130.     SCODE sc;
  2131.     if (FAILED(sc = VarDateFromStr((LPOLESTR)T2COLE(strDate), lcid,
  2132.         dwFlags, &m_dt)))
  2133.     {
  2134.         if (sc == DISP_E_TYPEMISMATCH)
  2135.         {
  2136.             // Can't convert string to date, set 0 and invalidate
  2137.             m_dt = 0;
  2138.             SetStatus(invalid);
  2139.             return FALSE;
  2140.         }
  2141.         else if (sc == DISP_E_OVERFLOW)
  2142.         {
  2143.             // Can't convert string to date, set -1 and invalidate
  2144.             m_dt = -1;
  2145.             SetStatus(invalid);
  2146.             return FALSE;
  2147.         }
  2148.         else
  2149.         {
  2150.             TRACE0("\nCOleDateTime VarDateFromStr call failed.\n\t");
  2151.             if (sc == E_OUTOFMEMORY)
  2152.                 AfxThrowMemoryException();
  2153.             else
  2154.                 AfxThrowOleException(sc);
  2155.         }
  2156.     }
  2157.  
  2158.     SetStatus(valid);
  2159.     return TRUE;
  2160. }
  2161.  
  2162. CString COleDateTime::Format(DWORD dwFlags, LCID lcid) const
  2163. {
  2164.     USES_CONVERSION;
  2165.     CString strDate;
  2166.  
  2167.     // If null, return empty string
  2168.     if (GetStatus() == null)
  2169.         return strDate;
  2170.  
  2171.     // If invalid, return DateTime resource string
  2172.     if (GetStatus() == invalid)
  2173.     {
  2174.         VERIFY(strDate.LoadString(AFX_IDS_INVALID_DATETIME));
  2175.         return strDate;
  2176.     }
  2177.  
  2178.     COleVariant var;
  2179.     // Don't need to trap error. Should not fail due to type mismatch
  2180.     AfxCheckError(VarBstrFromDate(m_dt, lcid, dwFlags, &V_BSTR(&var)));
  2181.     var.vt = VT_BSTR;
  2182.     return OLE2CT(V_BSTR(&var));
  2183. }
  2184.  
  2185. CString COleDateTime::Format(LPCTSTR pFormat) const
  2186. {
  2187.     CString strDate;
  2188.     struct tm tmTemp;
  2189.  
  2190.     // If null, return empty string
  2191.     if (GetStatus() == null)
  2192.         return strDate;
  2193.  
  2194.     // If invalid, return DateTime resource string
  2195.     if (GetStatus() == invalid || !_AfxTmFromOleDate(m_dt, tmTemp))
  2196.     {
  2197.         VERIFY(strDate.LoadString(AFX_IDS_INVALID_DATETIME));
  2198.         return strDate;
  2199.     }
  2200.  
  2201.     // Convert tm from afx internal format to standard format
  2202.     _AfxTmConvertToStandardFormat(tmTemp);
  2203.  
  2204.     // Fill in the buffer, disregard return value as it's not necessary
  2205.     LPTSTR lpszTemp = strDate.GetBufferSetLength(MAX_TIME_BUFFER_SIZE);
  2206.     _tcsftime(lpszTemp, strDate.GetLength(), pFormat, &tmTemp);
  2207.     strDate.ReleaseBuffer();
  2208.  
  2209.     return strDate;
  2210. }
  2211.  
  2212. CString COleDateTime::Format(UINT nFormatID) const
  2213. {
  2214.     CString strFormat;
  2215.     VERIFY(strFormat.LoadString(nFormatID) != 0);
  2216.     return Format(strFormat);
  2217. }
  2218.  
  2219. void COleDateTime::CheckRange()
  2220. {
  2221.     if (m_dt > MAX_DATE || m_dt < MIN_DATE) // about year 100 to about 9999
  2222.         SetStatus(invalid);
  2223. }
  2224.  
  2225. // serialization
  2226. #ifdef _DEBUG
  2227. CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleDateTime dateSrc)
  2228. {
  2229.     dc << "\nCOleDateTime Object:";
  2230.     dc << "\n\tm_status = " << (long)dateSrc.m_status;
  2231.  
  2232.     COleVariant var(dateSrc);
  2233.     var.ChangeType(VT_BSTR);
  2234.  
  2235.     return dc << "\n\tdate = " << var.bstrVal;
  2236. }
  2237. #endif // _DEBUG
  2238.  
  2239. CArchive& AFXAPI operator<<(CArchive& ar, COleDateTime dateSrc)
  2240. {
  2241.     ar << (long)dateSrc.m_status;
  2242.     return ar << dateSrc.m_dt;
  2243. }
  2244.  
  2245. CArchive& AFXAPI operator>>(CArchive& ar, COleDateTime& dateSrc)
  2246. {
  2247.     ar >> (long&)dateSrc.m_status;
  2248.     return ar >> dateSrc.m_dt;
  2249. }
  2250.  
  2251. /////////////////////////////////////////////////////////////////////////////
  2252. // COleDateTimeSpan class helpers
  2253.  
  2254. #define MAX_DAYS_IN_SPAN    3615897L
  2255.  
  2256. /////////////////////////////////////////////////////////////////////////////
  2257. // COleDateTimeSpan class
  2258. long COleDateTimeSpan::GetHours() const
  2259. {
  2260.     ASSERT(GetStatus() == valid);
  2261.  
  2262.     double dblTemp;
  2263.  
  2264.     // Truncate days and scale up
  2265.     dblTemp = modf(m_span, &dblTemp);
  2266.  
  2267.     long lReturns = (long)((dblTemp + AFX_OLE_DATETIME_HALFSECOND) * 24);
  2268.     if (lReturns >= 24)
  2269.         lReturns -= 24;
  2270.  
  2271.     return lReturns;
  2272. }
  2273.  
  2274. long COleDateTimeSpan::GetMinutes() const
  2275. {
  2276.     ASSERT(GetStatus() == valid);
  2277.  
  2278.     double dblTemp;
  2279.  
  2280.     // Truncate hours and scale up
  2281.     dblTemp = modf(m_span * 24, &dblTemp);
  2282.  
  2283.     long lReturns = (long) ((dblTemp + AFX_OLE_DATETIME_HALFSECOND) * 60);
  2284.     if (lReturns >= 60)
  2285.         lReturns -= 60;
  2286.  
  2287.     return lReturns;
  2288. }
  2289.  
  2290. long COleDateTimeSpan::GetSeconds() const
  2291. {
  2292.     ASSERT(GetStatus() == valid);
  2293.  
  2294.     double dblTemp;
  2295.  
  2296.     // Truncate minutes and scale up
  2297.     dblTemp = modf(m_span * 24 * 60, &dblTemp);
  2298.  
  2299.     long lReturns = (long) ((dblTemp + AFX_OLE_DATETIME_HALFSECOND) * 60);
  2300.     if (lReturns >= 60)
  2301.         lReturns -= 60;
  2302.  
  2303.     return lReturns;
  2304. }
  2305.  
  2306. const COleDateTimeSpan& COleDateTimeSpan::operator=(double dblSpanSrc)
  2307. {
  2308.     m_span = dblSpanSrc;
  2309.     SetStatus(valid);
  2310.     return *this;
  2311. }
  2312.  
  2313. const COleDateTimeSpan& COleDateTimeSpan::operator=(const COleDateTimeSpan& dateSpanSrc)
  2314. {
  2315.     m_span = dateSpanSrc.m_span;
  2316.     m_status = dateSpanSrc.m_status;
  2317.     return *this;
  2318. }
  2319.  
  2320. COleDateTimeSpan COleDateTimeSpan::operator+(const COleDateTimeSpan& dateSpan) const
  2321. {
  2322.     COleDateTimeSpan dateSpanTemp;
  2323.  
  2324.     // If either operand Null, result Null
  2325.     if (GetStatus() == null || dateSpan.GetStatus() == null)
  2326.     {
  2327.         dateSpanTemp.SetStatus(null);
  2328.         return dateSpanTemp;
  2329.     }
  2330.  
  2331.     // If either operand Invalid, result Invalid
  2332.     if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
  2333.     {
  2334.         dateSpanTemp.SetStatus(invalid);
  2335.         return dateSpanTemp;
  2336.     }
  2337.  
  2338.     // Add spans and validate within legal range
  2339.     dateSpanTemp.m_span = m_span + dateSpan.m_span;
  2340.     dateSpanTemp.CheckRange();
  2341.  
  2342.     return dateSpanTemp;
  2343. }
  2344.  
  2345. COleDateTimeSpan COleDateTimeSpan::operator-(const COleDateTimeSpan& dateSpan) const
  2346. {
  2347.     COleDateTimeSpan dateSpanTemp;
  2348.  
  2349.     // If either operand Null, result Null
  2350.     if (GetStatus() == null || dateSpan.GetStatus() == null)
  2351.     {
  2352.         dateSpanTemp.SetStatus(null);
  2353.         return dateSpanTemp;
  2354.     }
  2355.  
  2356.     // If either operand Invalid, result Invalid
  2357.     if (GetStatus() == invalid || dateSpan.GetStatus() == invalid)
  2358.     {
  2359.         dateSpanTemp.SetStatus(invalid);
  2360.         return dateSpanTemp;
  2361.     }
  2362.  
  2363.     // Subtract spans and validate within legal range
  2364.     dateSpanTemp.m_span = m_span - dateSpan.m_span;
  2365.     dateSpanTemp.CheckRange();
  2366.  
  2367.     return dateSpanTemp;
  2368. }
  2369.  
  2370. void COleDateTimeSpan::SetDateTimeSpan(
  2371.     long lDays, int nHours, int nMins, int nSecs)
  2372. {
  2373.     // Set date span by breaking into fractional days (all input ranges valid)
  2374.     m_span = lDays + ((double)nHours)/24 + ((double)nMins)/(24*60) +
  2375.         ((double)nSecs)/(24*60*60);
  2376.  
  2377.     SetStatus(valid);
  2378. }
  2379.  
  2380. CString COleDateTimeSpan::Format(LPCTSTR pFormat) const
  2381. {
  2382.     CString strSpan;
  2383.     struct tm tmTemp;
  2384.  
  2385.     // If null, return empty string
  2386.     if (GetStatus() == null)
  2387.         return strSpan;
  2388.  
  2389.     // If invalid, return DateTimeSpan resource string
  2390.     if (GetStatus() == invalid || !_AfxTmFromOleDate(m_span, tmTemp))
  2391.     {
  2392.         VERIFY(strSpan.LoadString(AFX_IDS_INVALID_DATETIMESPAN));
  2393.         return strSpan;
  2394.     }
  2395.  
  2396.     // Convert tm from afx internal format to standard format
  2397.     _AfxTmConvertToStandardFormat(tmTemp);
  2398.  
  2399.     // _tcsftime() doesn't handle %D, so do it here
  2400.  
  2401.     CString strPreParsed;
  2402.     LPCTSTR pstrSource = pFormat;
  2403.     int nTargetChar = 0;
  2404.     int nAccumulatedLength = lstrlen(pFormat);
  2405.     LPTSTR pstrTarget = strPreParsed.GetBuffer(nAccumulatedLength);
  2406.  
  2407.     while (*pstrSource)
  2408.     {
  2409.         if (*pstrSource == '%' && pstrSource[1] == 'D')
  2410.         {
  2411.             TCHAR szDay[12];
  2412.             _itot(GetDays(), szDay, 10);
  2413.             strPreParsed.ReleaseBuffer(nTargetChar);
  2414.             strPreParsed += szDay;
  2415.             int nTemp = lstrlen(szDay);
  2416.             nAccumulatedLength += nTemp;
  2417.             nTargetChar += nTemp;
  2418.             pstrTarget = strPreParsed.GetBuffer(nAccumulatedLength)
  2419.                 + nTargetChar;
  2420.             pstrSource = _tcsinc(pstrSource);
  2421.             pstrSource = _tcsinc(pstrSource);
  2422.         }
  2423.         *pstrTarget = *pstrSource;
  2424.         nTargetChar++;
  2425.         pstrSource = _tcsinc(pstrSource);
  2426.         pstrTarget = _tcsinc(pstrTarget);
  2427.     }
  2428.     strPreParsed.ReleaseBuffer(nTargetChar);
  2429.  
  2430.     // Fill in the buffer, disregard return value as it's not necessary
  2431.     LPTSTR lpszTemp = strSpan.GetBufferSetLength(MAX_TIME_BUFFER_SIZE);
  2432.     _tcsftime(lpszTemp, strSpan.GetLength(), (LPCTSTR) strPreParsed, &tmTemp);
  2433.     strSpan.ReleaseBuffer();
  2434.  
  2435.     return strSpan;
  2436. }
  2437.  
  2438. CString COleDateTimeSpan::Format(UINT nFormatID) const
  2439. {
  2440.     CString strFormat;
  2441.     VERIFY(strFormat.LoadString(nFormatID) != 0);
  2442.     return Format(strFormat);
  2443. }
  2444.  
  2445. void COleDateTimeSpan::CheckRange()
  2446. {
  2447.     if(m_span < -MAX_DAYS_IN_SPAN || m_span > MAX_DAYS_IN_SPAN)
  2448.         SetStatus(invalid);
  2449. }
  2450.  
  2451. // serialization
  2452. #ifdef _DEBUG
  2453. CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleDateTimeSpan dateSpanSrc)
  2454. {
  2455.     dc << "\nCOleDateTimeSpan Object:";
  2456.     dc << "\n\tm_status = " << (long)dateSpanSrc.m_status;
  2457.  
  2458.     COleVariant var(dateSpanSrc.m_span);
  2459.     var.ChangeType(VT_BSTR);
  2460.  
  2461.     return dc << "\n\tdateSpan = " << var.bstrVal;
  2462. }
  2463. #endif // _DEBUG
  2464.  
  2465. CArchive& AFXAPI operator<<(CArchive& ar, COleDateTimeSpan dateSpanSrc)
  2466. {
  2467.     ar << (long)dateSpanSrc.m_status;
  2468.     return ar << dateSpanSrc.m_span;
  2469. }
  2470.  
  2471. CArchive& AFXAPI operator>>(CArchive& ar, COleDateTimeSpan& dateSpanSrc)
  2472. {
  2473.     ar >> (long&)dateSpanSrc.m_status;
  2474.     return ar >> dateSpanSrc.m_span;
  2475. }
  2476.  
  2477. /////////////////////////////////////////////////////////////////////////////
  2478. // COleSafeArray class
  2479.  
  2480. COleSafeArray::COleSafeArray(const SAFEARRAY& saSrc, VARTYPE vtSrc)
  2481. {
  2482.     AfxSafeArrayInit(this);
  2483.     vt = (VARTYPE)(vtSrc | VT_ARRAY);
  2484.     AfxCheckError(::SafeArrayCopy((LPSAFEARRAY)&saSrc, &parray));
  2485.     m_dwDims = GetDim();
  2486.     m_dwElementSize = GetElemSize();
  2487. }
  2488.  
  2489. COleSafeArray::COleSafeArray(LPCSAFEARRAY pSrc, VARTYPE vtSrc)
  2490. {
  2491.     AfxSafeArrayInit(this);
  2492.     vt = (VARTYPE)(vtSrc | VT_ARRAY);
  2493.     AfxCheckError(::SafeArrayCopy((LPSAFEARRAY)pSrc, &parray));
  2494.     m_dwDims = GetDim();
  2495.     m_dwElementSize = GetElemSize();
  2496. }
  2497.  
  2498. COleSafeArray::COleSafeArray(const COleSafeArray& saSrc)
  2499. {
  2500.     AfxSafeArrayInit(this);
  2501.     *this = saSrc;
  2502.     m_dwDims = GetDim();
  2503.     m_dwElementSize = GetElemSize();
  2504. }
  2505.  
  2506. COleSafeArray::COleSafeArray(const VARIANT& varSrc)
  2507. {
  2508.     AfxSafeArrayInit(this);
  2509.     *this = varSrc;
  2510.     m_dwDims = GetDim();
  2511.     m_dwElementSize = GetElemSize();
  2512. }
  2513.  
  2514. COleSafeArray::COleSafeArray(LPCVARIANT pSrc)
  2515. {
  2516.     AfxSafeArrayInit(this);
  2517.     *this = pSrc;
  2518.     m_dwDims = GetDim();
  2519.     m_dwElementSize = GetElemSize();
  2520. }
  2521.  
  2522. // Operations
  2523. void COleSafeArray::Attach(VARIANT& varSrc)
  2524. {
  2525.     ASSERT(varSrc.vt & VT_ARRAY);
  2526.  
  2527.     // Free up previous safe array if necessary
  2528.     Clear();
  2529.  
  2530.     // give control of data to COleSafeArray
  2531.     memcpy(this, &varSrc, sizeof(varSrc));
  2532.     varSrc.vt = VT_EMPTY;
  2533. }
  2534.  
  2535. VARIANT COleSafeArray::Detach()
  2536. {
  2537.     VARIANT varResult = *this;
  2538.     vt = VT_EMPTY;
  2539.     return varResult;
  2540. }
  2541.  
  2542. // Assignment operators
  2543. COleSafeArray& COleSafeArray::operator=(const COleSafeArray& saSrc)
  2544. {
  2545.     ASSERT(saSrc.vt & VT_ARRAY);
  2546.  
  2547.     AfxCheckError(::VariantCopy(this, (LPVARIANT)&saSrc));
  2548.     return *this;
  2549. }
  2550.  
  2551. COleSafeArray& COleSafeArray::operator=(const VARIANT& varSrc)
  2552. {
  2553.     ASSERT(varSrc.vt & VT_ARRAY);
  2554.  
  2555.     AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
  2556.     return *this;
  2557. }
  2558.  
  2559. COleSafeArray& COleSafeArray::operator=(LPCVARIANT pSrc)
  2560. {
  2561.     ASSERT(pSrc->vt & VT_ARRAY);
  2562.  
  2563.     AfxCheckError(::VariantCopy(this, (LPVARIANT)pSrc));
  2564.     return *this;
  2565. }
  2566.  
  2567. COleSafeArray& COleSafeArray::operator=(const COleVariant& varSrc)
  2568. {
  2569.     ASSERT(varSrc.vt & VT_ARRAY);
  2570.  
  2571.     AfxCheckError(::VariantCopy(this, (LPVARIANT)&varSrc));
  2572.     return *this;
  2573. }
  2574.  
  2575. // Comparison operators
  2576. BOOL COleSafeArray::operator==(const SAFEARRAY& saSrc) const
  2577. {
  2578.     return _AfxCompareSafeArrays(parray, (LPSAFEARRAY)&saSrc);
  2579. }
  2580.  
  2581. BOOL COleSafeArray::operator==(LPCSAFEARRAY pSrc) const
  2582. {
  2583.     return _AfxCompareSafeArrays(parray, (LPSAFEARRAY)pSrc);
  2584. }
  2585.  
  2586. BOOL COleSafeArray::operator==(const COleSafeArray& saSrc) const
  2587. {
  2588.     if (vt != saSrc.vt)
  2589.         return FALSE;
  2590.  
  2591.     return _AfxCompareSafeArrays(parray, saSrc.parray);
  2592. }
  2593.  
  2594. BOOL COleSafeArray::operator==(const VARIANT& varSrc) const
  2595. {
  2596.     if (vt != varSrc.vt)
  2597.         return FALSE;
  2598.  
  2599.     return _AfxCompareSafeArrays(parray, varSrc.parray);
  2600. }
  2601.  
  2602. BOOL COleSafeArray::operator==(LPCVARIANT pSrc) const
  2603. {
  2604.     if (vt != pSrc->vt)
  2605.         return FALSE;
  2606.  
  2607.     return _AfxCompareSafeArrays(parray, pSrc->parray);
  2608. }
  2609.  
  2610. BOOL COleSafeArray::operator==(const COleVariant& varSrc) const
  2611. {
  2612.     if (vt != varSrc.vt)
  2613.         return FALSE;
  2614.  
  2615.     return _AfxCompareSafeArrays(parray, varSrc.parray);
  2616. }
  2617.  
  2618. #ifdef _DEBUG
  2619. void _AfxDumpSafeArrayElement(CDumpContext& dc, COleSafeArray& saSrc,
  2620.    long* piIndices)
  2621. {
  2622.    BYTE* pbData;
  2623.  
  2624.    pbData = (BYTE*)_alloca( saSrc.GetElemSize() );
  2625.  
  2626.    saSrc.GetElement(piIndices, pbData);
  2627.    switch(saSrc.vt&(~VT_ARRAY))
  2628.    {
  2629.    case VT_BOOL:
  2630.       dc << *((VARIANT_BOOL*)pbData);
  2631.       break;
  2632.  
  2633.    case VT_I1:
  2634.       dc << *((char*)pbData);
  2635.       break;
  2636.  
  2637.    case VT_I2:
  2638.       dc << *((short*)pbData);
  2639.       break;
  2640.  
  2641.    case VT_I4:
  2642.       dc << *((long*)pbData);
  2643.       break;
  2644.  
  2645.    case VT_UI1:
  2646.       dc << *((BYTE*)pbData);
  2647.       break;
  2648.  
  2649.    case VT_UI2:
  2650.       dc << *((WORD*)pbData);
  2651.       break;
  2652.  
  2653.    case VT_UI4:
  2654.       dc << *((DWORD*)pbData);
  2655.       break;
  2656.  
  2657.    case VT_R4:
  2658.       dc << *((float*)pbData);
  2659.       break;
  2660.  
  2661.    case VT_R8:
  2662.       dc << *((double*)pbData);
  2663.       break;
  2664.  
  2665.    case VT_CY:
  2666.       {
  2667.          COleVariant var;
  2668.  
  2669.          var.vt = VT_CY;
  2670.          var.cyVal = *((CY*)pbData);
  2671.          var.ChangeType(VT_BSTR);
  2672.          dc << var.bstrVal;
  2673.       }
  2674.       break;
  2675.  
  2676.    case VT_DATE:
  2677.       {
  2678.          COleVariant var;
  2679.          var.vt = VT_DATE;
  2680.          var.date = *((DATE*)pbData);
  2681.          var.ChangeType(VT_BSTR);
  2682.          dc << var.bstrVal;
  2683.       }
  2684.  
  2685.    case VT_BSTR:
  2686.       dc << *((BSTR*)pbData);
  2687.       break;
  2688.  
  2689.    case VT_ERROR:
  2690.       dc << *((SCODE*)pbData);
  2691.       break;
  2692.  
  2693.    case VT_DISPATCH:
  2694.    case VT_UNKNOWN:
  2695.       dc << *((IUnknown**)pbData);
  2696.       break;
  2697.  
  2698.    case VT_VARIANT:
  2699.       {
  2700.          COleVariant var;
  2701.  
  2702.          var = *((VARIANT*)pbData);
  2703.          dc << var;
  2704.       }
  2705.       break;
  2706.  
  2707.    default:
  2708.       ASSERT(FALSE);
  2709.       break;
  2710.    }
  2711. }
  2712.  
  2713. CDumpContext& AFXAPI operator<<(CDumpContext& dc, COleSafeArray& saSrc)
  2714. {
  2715.    long iDimension;
  2716.    long nDimensions;
  2717.    long* piLBounds;
  2718.    long* piUBounds;
  2719.    long* piIndices;
  2720.    BOOL bWrapped;
  2721.  
  2722.    dc << "\nCOleSafeArray Object:";
  2723.    dc << "\n\tvt = " << saSrc.vt;
  2724.    dc << "\n\tbounds:";
  2725.  
  2726.    nDimensions = saSrc.GetDim();
  2727.    piLBounds = (long*)_alloca( nDimensions*sizeof( long ) );
  2728.    piUBounds = (long*)_alloca( nDimensions*sizeof( long ) );
  2729.    piIndices = (long*)_alloca( nDimensions*sizeof( long ) );
  2730.    // Dump the bounds
  2731.    for( iDimension = 0; iDimension < nDimensions; iDimension++ )
  2732.    {
  2733.       saSrc.GetLBound( iDimension+1, &piLBounds[iDimension] );
  2734.       saSrc.GetUBound( iDimension+1, &piUBounds[iDimension] );
  2735.       dc << "\n\t(" << piLBounds[iDimension] << ", " <<
  2736.          piUBounds[iDimension] << ")";
  2737.    }
  2738.  
  2739.    if( dc.GetDepth() > 0 )
  2740.    {
  2741.       // Dump the contents of the array.
  2742.       for( iDimension = 0; iDimension < nDimensions; iDimension++ )
  2743.       {
  2744.          piIndices[iDimension] = piLBounds[iDimension];
  2745.       }
  2746.  
  2747.       while( piIndices[0] <= piUBounds[0] )
  2748.       {
  2749.          dc << "\n\t";
  2750.          for( iDimension = 0; iDimension < nDimensions; iDimension++ )
  2751.          {
  2752.             dc << "[" << piIndices[iDimension] << "]";
  2753.          }
  2754.          dc << " = ";
  2755.  
  2756.          // Dump the value of the element
  2757.          _AfxDumpSafeArrayElement(dc, saSrc, piIndices);
  2758.  
  2759.          // Increment the rightmost index, with wraparound and carry logic
  2760.          iDimension = nDimensions-1;
  2761.          bWrapped = TRUE;
  2762.          do
  2763.          {
  2764.             bWrapped = FALSE;
  2765.             piIndices[iDimension]++;
  2766.             if( piIndices[iDimension] > piUBounds[iDimension] )
  2767.             {
  2768.                if( iDimension > 0 )
  2769.                {
  2770.                   // We've overstepped the bounds of this dimension, so wrap
  2771.                   // around to the lower bound and make sure to increment the
  2772.                   // next dimension to the left.
  2773.                   bWrapped = TRUE;
  2774.                   piIndices[iDimension] = piLBounds[iDimension];
  2775.                   iDimension--;
  2776.                }
  2777.             }
  2778.          } while( bWrapped && (iDimension >= 0) );
  2779.       }
  2780.    }
  2781.  
  2782.    return dc;
  2783. }
  2784. #endif // _DEBUG
  2785.  
  2786. void COleSafeArray::CreateOneDim(VARTYPE vtSrc, DWORD dwElements,
  2787.     const void* pvSrcData, long nLBound)
  2788. {
  2789.     ASSERT(dwElements > 0);
  2790.  
  2791.     // Setup the bounds and create the array
  2792.     SAFEARRAYBOUND rgsabound;
  2793.     rgsabound.cElements = dwElements;
  2794.     rgsabound.lLbound = nLBound;
  2795.     Create(vtSrc, 1, &rgsabound);
  2796.  
  2797.     // Copy over the data if neccessary
  2798.     if (pvSrcData != NULL)
  2799.     {
  2800.         void* pvDestData;
  2801.         AccessData(&pvDestData);
  2802.         memcpy(pvDestData, pvSrcData, GetElemSize() * dwElements);
  2803.         UnaccessData();
  2804.     }
  2805. }
  2806.  
  2807. DWORD COleSafeArray::GetOneDimSize()
  2808. {
  2809.     ASSERT(GetDim() == 1);
  2810.  
  2811.     long nUBound, nLBound;
  2812.  
  2813.     GetUBound(1, &nUBound);
  2814.     GetLBound(1, &nLBound);
  2815.  
  2816.     return nUBound + 1 - nLBound;
  2817. }
  2818.  
  2819. void COleSafeArray::ResizeOneDim(DWORD dwElements)
  2820. {
  2821.     ASSERT(GetDim() == 1);
  2822.  
  2823.     SAFEARRAYBOUND rgsabound;
  2824.  
  2825.     rgsabound.cElements = dwElements;
  2826.     rgsabound.lLbound = 0;
  2827.  
  2828.     Redim(&rgsabound);
  2829. }
  2830.  
  2831. void COleSafeArray::Create(VARTYPE vtSrc, DWORD dwDims, DWORD* rgElements)
  2832. {
  2833.     ASSERT(rgElements != NULL);
  2834.  
  2835.     // Allocate and fill proxy array of bounds (with lower bound of zero)
  2836.     SAFEARRAYBOUND* rgsaBounds = new SAFEARRAYBOUND[dwDims];
  2837.  
  2838.     for (DWORD dwIndex = 0; dwIndex < dwDims; dwIndex++)
  2839.     {
  2840.         // Assume lower bound is 0 and fill in element count
  2841.         rgsaBounds[dwIndex].lLbound = 0;
  2842.         rgsaBounds[dwIndex].cElements = rgElements[dwIndex];
  2843.     }
  2844.  
  2845.     TRY
  2846.     {
  2847.         Create(vtSrc, dwDims, rgsaBounds);
  2848.     }
  2849.     CATCH_ALL(e)
  2850.     {
  2851.         // Must free up memory
  2852.         delete[] rgsaBounds;
  2853.         THROW_LAST();
  2854.     }
  2855.     END_CATCH_ALL
  2856.  
  2857.     delete[] rgsaBounds;
  2858. }
  2859.  
  2860. void COleSafeArray::Create(VARTYPE vtSrc, DWORD dwDims, SAFEARRAYBOUND* rgsabound)
  2861. {
  2862.     ASSERT(dwDims > 0);
  2863.     ASSERT(rgsabound != NULL);
  2864.  
  2865.     // Validate the VARTYPE for SafeArrayCreate call
  2866.     ASSERT(!(vtSrc & VT_ARRAY));
  2867.     ASSERT(!(vtSrc & VT_BYREF));
  2868.     ASSERT(!(vtSrc & VT_VECTOR));
  2869.     ASSERT(vtSrc != VT_EMPTY);
  2870.     ASSERT(vtSrc != VT_NULL);
  2871.  
  2872.     // Free up old safe array if necessary
  2873.     Clear();
  2874.  
  2875.     parray = ::SafeArrayCreate(vtSrc, dwDims, rgsabound);
  2876.  
  2877.     if (parray == NULL)
  2878.         AfxThrowMemoryException();
  2879.  
  2880.     vt = unsigned short(vtSrc | VT_ARRAY);
  2881.     m_dwDims = dwDims;
  2882.     m_dwElementSize = GetElemSize();
  2883. }
  2884.  
  2885. void COleSafeArray::AccessData(void** ppvData)
  2886. {
  2887.     AfxCheckError(::SafeArrayAccessData(parray, ppvData));
  2888. }
  2889.  
  2890. void COleSafeArray::UnaccessData()
  2891. {
  2892.     AfxCheckError(::SafeArrayUnaccessData(parray));
  2893. }
  2894.  
  2895. void COleSafeArray::AllocData()
  2896. {
  2897.     AfxCheckError(::SafeArrayAllocData(parray));
  2898. }
  2899.  
  2900. void COleSafeArray::AllocDescriptor(DWORD dwDims)
  2901. {
  2902.     AfxCheckError(::SafeArrayAllocDescriptor(dwDims, &parray));
  2903. }
  2904.  
  2905. void COleSafeArray::Copy(LPSAFEARRAY* ppsa)
  2906. {
  2907.     AfxCheckError(::SafeArrayCopy(parray, ppsa));
  2908. }
  2909.  
  2910. void COleSafeArray::GetLBound(DWORD dwDim, long* pLbound)
  2911. {
  2912.     AfxCheckError(::SafeArrayGetLBound(parray, dwDim, pLbound));
  2913. }
  2914.  
  2915. void COleSafeArray::GetUBound(DWORD dwDim, long* pUbound)
  2916. {
  2917.     AfxCheckError(::SafeArrayGetUBound(parray, dwDim, pUbound));
  2918. }
  2919.  
  2920. void COleSafeArray::GetElement(long* rgIndices, void* pvData)
  2921. {
  2922.     AfxCheckError(::SafeArrayGetElement(parray, rgIndices, pvData));
  2923. }
  2924.  
  2925. void COleSafeArray::PtrOfIndex(long* rgIndices, void** ppvData)
  2926. {
  2927.     AfxCheckError(::SafeArrayPtrOfIndex(parray, rgIndices, ppvData));
  2928. }
  2929.  
  2930. void COleSafeArray::PutElement(long* rgIndices, void* pvData)
  2931. {
  2932.     AfxCheckError(::SafeArrayPutElement(parray, rgIndices, pvData));
  2933. }
  2934.  
  2935. void COleSafeArray::Redim(SAFEARRAYBOUND* psaboundNew)
  2936. {
  2937.     AfxCheckError(::SafeArrayRedim(parray, psaboundNew));
  2938. }
  2939.  
  2940. void COleSafeArray::Lock()
  2941. {
  2942.     AfxCheckError(::SafeArrayLock(parray));
  2943. }
  2944.  
  2945. void COleSafeArray::Unlock()
  2946. {
  2947.     AfxCheckError(::SafeArrayUnlock(parray));
  2948. }
  2949.  
  2950. void COleSafeArray::Destroy()
  2951. {
  2952.     AfxCheckError(::SafeArrayDestroy(parray));
  2953. }
  2954.  
  2955. void COleSafeArray::DestroyData()
  2956. {
  2957.     AfxCheckError(::SafeArrayDestroyData(parray));
  2958. }
  2959.  
  2960. void COleSafeArray::DestroyDescriptor()
  2961. {
  2962.     AfxCheckError(::SafeArrayDestroyDescriptor(parray));
  2963. }
  2964.  
  2965. ///////////////////////////////////////////////////////////////////////////////
  2966. // COleSafeArray Helpers
  2967. void AFXAPI AfxSafeArrayInit(COleSafeArray* psa)
  2968. {
  2969.     memset(psa, 0, sizeof(*psa));
  2970. }
  2971.  
  2972. /////////////////////////////////////////////////////////////////////////////
  2973. // Simple field formatting to text item - see dlgdata.cpp for base types
  2974.  
  2975. void AFXAPI DDX_Text(CDataExchange* pDX, int nIDC, COleDateTime& value)
  2976. {
  2977.     HWND hWndCtrl = pDX->PrepareEditCtrl(nIDC);
  2978.     if (pDX->m_bSaveAndValidate)
  2979.     {
  2980.         int nLen = ::GetWindowTextLength(hWndCtrl);
  2981.         CString strTemp;
  2982.  
  2983.         ::GetWindowText(hWndCtrl, strTemp.GetBufferSetLength(nLen), nLen+1);
  2984.         strTemp.ReleaseBuffer();
  2985.  
  2986.         if (!value.ParseDateTime(strTemp))  // throws exception
  2987.         {
  2988.             // Can't convert string to datetime
  2989.             AfxMessageBox(AFX_IDP_PARSE_DATETIME);
  2990.             pDX->Fail();    // throws exception
  2991.         }
  2992.     }
  2993.     else
  2994.     {
  2995.         CString strTemp = value.Format();
  2996.         AfxSetWindowText(hWndCtrl, strTemp);
  2997.     }
  2998. }
  2999.  
  3000. void AFXAPI DDX_Text(CDataExchange* pDX, int nIDC, COleCurrency& value)
  3001. {
  3002.     HWND hWndCtrl = pDX->PrepareEditCtrl(nIDC);
  3003.     if (pDX->m_bSaveAndValidate)
  3004.     {
  3005.         int nLen = ::GetWindowTextLength(hWndCtrl);
  3006.         CString strTemp;
  3007.  
  3008.         ::GetWindowText(hWndCtrl, strTemp.GetBufferSetLength(nLen), nLen+1);
  3009.         strTemp.ReleaseBuffer();
  3010.  
  3011.         if (!value.ParseCurrency(strTemp))  // throws exception
  3012.         {
  3013.             // Can't convert string to currency
  3014.             AfxMessageBox(AFX_IDP_PARSE_CURRENCY);
  3015.             pDX->Fail();    // throws exception
  3016.         }
  3017.     }
  3018.     else
  3019.     {
  3020.         CString strTemp = value.Format();
  3021.         AfxSetWindowText(hWndCtrl, strTemp);
  3022.     }
  3023. }
  3024.  
  3025. /////////////////////////////////////////////////////////////////////////////
  3026.