369 lines
7.8 KiB
C++
369 lines
7.8 KiB
C++
|
|
// StandardEnum.Cpp
|
|
|
|
// Copyright 1995 Microsoft Corporation. All Rights Reserved.
|
|
|
|
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF
|
|
// ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO
|
|
// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
|
|
// PARTICULAR PURPOSE.
|
|
|
|
|
|
// implementation of a generic enumerator object.
|
|
|
|
|
|
#include "stdafx.h"
|
|
#pragma hdrstop
|
|
|
|
#include "stdenum.h"
|
|
|
|
// Used by creators of CStandardEnum
|
|
|
|
void WINAPI CopyAndAddRefObject
|
|
(
|
|
void *pDest, // dest
|
|
const void *pSource, // src
|
|
DWORD dwSize // size, ignored, since it's always 4
|
|
)
|
|
{
|
|
IUnknown *pUnk = *((IUnknown **)pSource);
|
|
*((IUnknown **)pDest) = pUnk;
|
|
pUnk->AddRef();
|
|
}
|
|
|
|
void* CStandardEnum_CreateInstance(REFIID riid, BOOL fMembersAreInterfaces, int cElement, int cbElement, void *rgElements,
|
|
void (WINAPI * pfnCopyElement)(void *, const void *, DWORD))
|
|
{
|
|
return (LPVOID)new CStandardEnum(riid, fMembersAreInterfaces, cElement, cbElement, rgElements, pfnCopyElement);
|
|
}
|
|
|
|
|
|
// CStandardEnum::CStandardEnum
|
|
|
|
// create the object and initialize the refcount
|
|
|
|
// Parameters:
|
|
// REFCLSID - [in] type of enumerator that we are
|
|
// int - [in] number of elements in the enumeration
|
|
// int - [in] size of each element
|
|
// void * - [in] pointer to element data
|
|
// void (WINAPI *pfnCopyElement)(void *, const void *, DWORD)
|
|
// - [in] copying function
|
|
|
|
// Notes:
|
|
|
|
#pragma warning(disable:4355) // using 'this' in constructor
|
|
CStandardEnum::CStandardEnum
|
|
(
|
|
REFCLSID rclsid,
|
|
BOOL fMembersAreInterfaces,
|
|
int cElements,
|
|
int cbElementSize,
|
|
void *rgElements,
|
|
void (WINAPI *pfnCopyElement)(void *, const void *, DWORD)
|
|
)
|
|
: m_cRef(1),
|
|
m_iid(rclsid),
|
|
m_cElements(cElements),
|
|
m_cbElementSize(cbElementSize),
|
|
m_iCurrent(0),
|
|
m_rgElements(rgElements),
|
|
m_pfnCopyElement(pfnCopyElement),
|
|
m_fMembersAreInterfaces(fMembersAreInterfaces)
|
|
{
|
|
m_pEnumClonedFrom = NULL;
|
|
if(m_fMembersAreInterfaces)
|
|
{
|
|
if(m_rgElements)
|
|
{
|
|
int i;
|
|
for(i=0; i<m_cElements; i++)
|
|
{
|
|
LPUNKNOWN *ppunk = (IUnknown **)GetNthElement(i);
|
|
(*ppunk)->AddRef();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#pragma warning(default:4355) // using 'this' in constructor
|
|
|
|
|
|
|
|
// CStandardEnum::CStandardEnum
|
|
|
|
// "it is not death, but dying, which is terrible."
|
|
// - Henry Fielding (1707-54)
|
|
|
|
// Notes:
|
|
|
|
CStandardEnum::~CStandardEnum ()
|
|
{
|
|
// if we're a cloned object, then just release our parent object and
|
|
// we're done. otherwise, free up the allocated memory we were given
|
|
|
|
if (m_pEnumClonedFrom)
|
|
{
|
|
m_pEnumClonedFrom->Release();
|
|
}
|
|
else
|
|
{
|
|
if (m_rgElements)
|
|
{
|
|
if(m_fMembersAreInterfaces)
|
|
{
|
|
int i;
|
|
|
|
for(i=0; i<m_cElements; i++)
|
|
{
|
|
LPUNKNOWN *ppunk = (IUnknown **)GetNthElement(i);
|
|
(*ppunk)->Release();
|
|
}
|
|
}
|
|
GlobalFree(m_rgElements);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// CStandardEnum::InternalQueryInterface
|
|
|
|
// we support our internal iid, and that's all
|
|
|
|
// Parameters:
|
|
// REFIID - [in] interface they want
|
|
// void ** - [out] where they want to put the resulting object ptr.
|
|
|
|
// Output:
|
|
// HRESULT - S_OK, E_NOINTERFACE
|
|
|
|
// Notes:
|
|
|
|
HRESULT CStandardEnum::QueryInterface
|
|
(
|
|
REFIID riid,
|
|
void **ppvObjOut
|
|
)
|
|
{
|
|
*ppvObjOut = NULL;
|
|
if (IsEqualIID(riid, m_iid))
|
|
{
|
|
*ppvObjOut = (IEnumGeneric *)this;
|
|
}
|
|
else if (IsEqualIID(riid, IID_IUnknown))
|
|
{
|
|
*ppvObjOut = (IUnknown *)this;
|
|
}
|
|
|
|
if (*ppvObjOut)
|
|
{
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
else
|
|
{
|
|
return E_NOINTERFACE;
|
|
}
|
|
}
|
|
|
|
ULONG CStandardEnum::AddRef(void)
|
|
{
|
|
return ++m_cRef;
|
|
}
|
|
|
|
ULONG CStandardEnum::Release(void)
|
|
{
|
|
int n = --m_cRef;
|
|
|
|
if (n == 0)
|
|
delete this;
|
|
|
|
return(n);
|
|
}
|
|
|
|
|
|
|
|
// CStandardEnum::Next
|
|
|
|
// returns the next dude in our iteration
|
|
|
|
// Parameters:
|
|
// unsigned long - [in] count of elements requested
|
|
// void * - [out] array of slots to put values in.
|
|
// unsigned long * - [out] actual number fetched
|
|
|
|
// Output:
|
|
// HRESULT - S_OK, E_INVALIDARG, S_FALSE
|
|
|
|
// Notes:
|
|
|
|
STDMETHODIMP CStandardEnum::Next
|
|
(
|
|
unsigned long cElm,
|
|
void *rgDest,
|
|
unsigned long *pcElmOut
|
|
)
|
|
{
|
|
unsigned long cElementsFetched = 0;
|
|
void *pElementDest = rgDest;
|
|
const void *pElementSrc = GetNthElement(m_iCurrent);
|
|
|
|
while (cElementsFetched < cElm) {
|
|
|
|
// if we hit EOF, break out
|
|
|
|
if (m_iCurrent >= m_cElements)
|
|
break;
|
|
|
|
// copy the element out for them
|
|
|
|
m_pfnCopyElement(pElementDest, pElementSrc, m_cbElementSize);
|
|
|
|
// increase the counters
|
|
|
|
pElementDest = (LPBYTE)pElementDest + m_cbElementSize;
|
|
pElementSrc = (const BYTE *)pElementSrc + m_cbElementSize;
|
|
m_iCurrent++;
|
|
cElementsFetched++;
|
|
}
|
|
|
|
if (pcElmOut)
|
|
*pcElmOut = cElementsFetched;
|
|
|
|
return (cElementsFetched < cElm)? S_FALSE : S_OK;
|
|
}
|
|
|
|
|
|
// CStandardEnum::Skip
|
|
|
|
// skips the requested number of rows.
|
|
|
|
// Parameters:
|
|
// unsigned long - [in] number to skip
|
|
|
|
// Output:
|
|
// HRESULT - S_OK, S_FALSE
|
|
|
|
// Notes:
|
|
|
|
STDMETHODIMP CStandardEnum::Skip
|
|
(
|
|
unsigned long cSkip
|
|
)
|
|
{
|
|
// handle running off the end
|
|
|
|
if (m_iCurrent + (int)cSkip > m_cElements) {
|
|
m_iCurrent = m_cElements;
|
|
return S_FALSE;
|
|
}
|
|
|
|
m_iCurrent += cSkip;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
// CStandardEnum::Reset
|
|
|
|
// reset the counter.
|
|
|
|
// Output:
|
|
// HRESULT - S_OK
|
|
|
|
// Notes:
|
|
|
|
STDMETHODIMP CStandardEnum::Reset
|
|
(
|
|
void
|
|
)
|
|
{
|
|
m_iCurrent = 0;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
|
|
// CStandardEnum::Clone
|
|
|
|
// clones the object and gives the new one the same position
|
|
|
|
// Parameters:
|
|
// IEnumVARIANT ** - [out] where to put the new object.
|
|
|
|
// Output;
|
|
// HRESULT - S_OK, E_OUTOFMEMORY
|
|
|
|
// Notes:
|
|
|
|
STDMETHODIMP CStandardEnum::Clone
|
|
(
|
|
IEnumGeneric **ppEnumClone
|
|
)
|
|
{
|
|
CStandardEnum *pNewEnum = new CStandardEnum(m_iid, m_fMembersAreInterfaces, m_cElements,
|
|
m_cbElementSize, m_rgElements, m_pfnCopyElement);
|
|
RETURN_ON_NULLALLOC(pNewEnum);
|
|
|
|
// The clone has the same current position as we do
|
|
pNewEnum->m_iCurrent = m_iCurrent;
|
|
|
|
// hold on to who we were cloned from so m_rgElements stays alive, and we don't
|
|
// have to copy it.
|
|
|
|
pNewEnum->m_pEnumClonedFrom = this;
|
|
|
|
// AddRef() ourselves on their behalf.
|
|
|
|
AddRef();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
// Helper function for creating IConnectionPoint enumerators
|
|
|
|
HRESULT CreateInstance_IEnumConnectionPoints(LPENUMCONNECTIONPOINTS * ppEnum, DWORD count, ...)
|
|
{
|
|
DWORD countTemp;
|
|
IConnectionPoint **rgCPs;
|
|
|
|
if (NULL == ppEnum)
|
|
return E_POINTER;
|
|
|
|
ASSERT(count > 0);
|
|
|
|
// GlobalAlloc an array of connection points [since our standard enum
|
|
// assumes this and GlobalFree's it later]
|
|
|
|
rgCPs = (LPCONNECTIONPOINT*)GlobalAlloc(GPTR, SIZEOF(LPCONNECTIONPOINT) * count);
|
|
if (NULL == rgCPs)
|
|
return E_OUTOFMEMORY;
|
|
|
|
va_list ArgList;
|
|
va_start(ArgList, count);
|
|
|
|
IConnectionPoint **prgCPs = rgCPs;
|
|
countTemp = count;
|
|
while (countTemp)
|
|
{
|
|
IConnectionPoint *pArg = va_arg(ArgList, IConnectionPoint*);
|
|
|
|
*prgCPs = pArg;
|
|
|
|
prgCPs++;
|
|
countTemp--;
|
|
}
|
|
|
|
va_end(ArgList);
|
|
|
|
*ppEnum = (IEnumConnectionPoints *)(IEnumGeneric *) new CStandardEnum(IID_IEnumConnectionPoints,
|
|
TRUE, count, SIZEOF(LPCONNECTIONPOINT), (LPVOID)rgCPs,
|
|
CopyAndAddRefObject);
|
|
if (!*ppEnum)
|
|
{
|
|
GlobalFree(rgCPs);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|