2020-09-30 16:53:55 +02:00

562 lines
16 KiB
C++

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1998 - 1999
//
// File: dataobj_.cpp
//
//--------------------------------------------------------------------------
#include <strsafe.h>
///////////////////////////////////////////////////////////////////////////////
// Sample code to show how to Create DataObjects
// Minimal error checking for clarity
///////////////////////////////////////////////////////////////////////////////
// Snap-in NodeType in both GUID format and string format
// Note - Typically there is a node type for each different object, sample
// only uses one node type.
const wchar_t* CCF_DNS_SNAPIN_INTERNAL = L"DNS_SNAPIN_INTERNAL";
CLIPFORMAT CDataObject::m_cfNodeType = (CLIPFORMAT)RegisterClipboardFormat(CCF_NODETYPE);
CLIPFORMAT CDataObject::m_cfNodeTypeString = (CLIPFORMAT)RegisterClipboardFormat(CCF_SZNODETYPE);
CLIPFORMAT CDataObject::m_cfDisplayName = (CLIPFORMAT)RegisterClipboardFormat(CCF_DISPLAY_NAME);
CLIPFORMAT CDataObject::m_cfCoClass = (CLIPFORMAT)RegisterClipboardFormat(CCF_SNAPIN_CLASSID);
CLIPFORMAT CDataObject::m_cfColumnID = (CLIPFORMAT)RegisterClipboardFormat(CCF_COLUMN_SET_ID);
CLIPFORMAT CDataObject::m_cfInternal = (CLIPFORMAT)RegisterClipboardFormat(CCF_DNS_SNAPIN_INTERNAL);
CLIPFORMAT CDataObject::m_cfMultiSel = (CLIPFORMAT)RegisterClipboardFormat(CCF_MULTI_SELECT_SNAPINS);
CLIPFORMAT CDataObject::m_cfMultiObjTypes = (CLIPFORMAT)RegisterClipboardFormat(CCF_OBJECT_TYPES_IN_MULTI_SELECT);
#ifdef _DEBUG_REFCOUNT
unsigned int CDataObject::m_nOustandingObjects = 0;
#endif // _DEBUG_REFCOUNT
/////////////////////////////////////////////////////////////////////////////
// CInternalFormatCracker
HRESULT CInternalFormatCracker::Extract(LPDATAOBJECT lpDataObject)
{
if (DOBJ_CUSTOMOCX == lpDataObject ||
DOBJ_CUSTOMWEB == lpDataObject ||
DOBJ_NULL == lpDataObject)
{
return DV_E_CLIPFORMAT;
}
if (m_pInternal != NULL)
_Free();
SMMCDataObjects * pDO = NULL;
STGMEDIUM stgmedium = { TYMED_HGLOBAL, NULL };
FORMATETC formatetc = { CDataObject::m_cfInternal, NULL,
DVASPECT_CONTENT, -1, TYMED_HGLOBAL
};
FORMATETC formatetc2 = { CDataObject::m_cfMultiSel, NULL,
DVASPECT_CONTENT, -1, TYMED_HGLOBAL
};
HRESULT hr = lpDataObject->GetData(&formatetc2, &stgmedium);
if (FAILED(hr))
{
hr = lpDataObject->GetDataHere(&formatetc, &stgmedium);
if (FAILED(hr))
return hr;
m_pInternal = reinterpret_cast<INTERNAL*>(stgmedium.hGlobal);
}
else
{
pDO = reinterpret_cast<SMMCDataObjects*>(stgmedium.hGlobal);
for (UINT i = 0; i < pDO->count; i++)
{
hr = pDO->lpDataObject[i]->GetDataHere(&formatetc, &stgmedium);
if (FAILED(hr))
break;
m_pInternal = reinterpret_cast<INTERNAL*>(stgmedium.hGlobal);
if (m_pInternal != NULL)
break;
}
}
return hr;
}
void CInternalFormatCracker::GetCookieList(CNodeList& list)
{
for (DWORD dwCount = 0; dwCount < m_pInternal->m_cookie_count; dwCount++)
{
list.AddTail(m_pInternal->m_p_cookies[dwCount]);
}
}
/////////////////////////////////////////////////////////////////////////////
// CDataObject implementations
STDMETHODIMP CDataObject::GetDataHere(LPFORMATETC lpFormatetc, LPSTGMEDIUM lpMedium)
{
HRESULT hr = DV_E_CLIPFORMAT;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
// Based on the CLIPFORMAT write data to the stream
const CLIPFORMAT cf = lpFormatetc->cfFormat;
if(cf == m_cfNodeType)
{
hr = CreateNodeTypeData(lpMedium);
}
else if(cf == m_cfNodeTypeString)
{
hr = CreateNodeTypeStringData(lpMedium);
}
else if (cf == m_cfDisplayName)
{
hr = CreateDisplayName(lpMedium);
}
else if (cf == m_cfCoClass)
{
hr = CreateCoClassID(lpMedium);
}
else if (cf == m_cfInternal)
{
hr = CreateInternal(lpMedium);
}
else if (cf == m_cfMultiObjTypes)
{
hr = CreateMultiSelectObject(lpMedium);
}
else
{
// if not successful, maybe there is a node specific clipboard format,
// so ask the node itself to provide
CTreeNode* pNode = GetTreeNodeFromCookie();
ASSERT(pNode != NULL);
if (pNode != NULL)
{
hr = pNode->GetDataHere(cf, lpMedium, this);
}
}
return hr;
}
// Note - Sample does not implement these
STDMETHODIMP CDataObject::GetData(LPFORMATETC lpFormatetcIn, LPSTGMEDIUM lpMedium)
{
HRESULT hr = DV_E_CLIPFORMAT;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
// Based on the CLIPFORMAT write data to the stream
const CLIPFORMAT cf = lpFormatetcIn->cfFormat;
if (cf == m_cfColumnID)
{
hr = CreateColumnID(lpMedium);
}
else if (cf == m_cfMultiObjTypes)
{
hr = CreateMultiSelectObject(lpMedium);
}
else
{
// if not successful, maybe there is a node specific clipboard format,
// so ask the node itself to provide
CTreeNode* pNode = GetTreeNodeFromCookie();
if (pNode != NULL)
{
hr = pNode->GetData(cf, lpMedium, this);
}
}
return hr;
}
STDMETHODIMP CDataObject::EnumFormatEtc(DWORD, LPENUMFORMATETC*)
{
return E_NOTIMPL;
}
/////////////////////////////////////////////////////////////////////////////
// CDataObject creation members
HRESULT CDataObject::Create(const void* pBuffer, size_t len, LPSTGMEDIUM lpMedium)
{
HRESULT hr = DV_E_TYMED;
// Do some simple validation
if (pBuffer == NULL || lpMedium == NULL)
return E_POINTER;
// Make sure the type medium is HGLOBAL
if (lpMedium->tymed == TYMED_HGLOBAL)
{
// Create the stream on the hGlobal passed in
LPSTREAM lpStream;
hr = CreateStreamOnHGlobal(lpMedium->hGlobal, FALSE, &lpStream);
if (SUCCEEDED(hr))
{
// Write to the stream the number of bytes
ULONG bytesToWrite = static_cast<ULONG>(min(len, ::GlobalSize(lpMedium->hGlobal)));
unsigned long written;
hr = lpStream->Write(pBuffer, bytesToWrite, &written);
// Because we told CreateStreamOnHGlobal with 'FALSE',
// only the stream is released here.
// Note - the caller (i.e. snap-in, object) will free the HGLOBAL
// at the correct time. This is according to the IDataObject specification.
lpStream->Release();
}
}
return hr;
}
HRESULT CDataObject::CreateColumnID(LPSTGMEDIUM lpMedium)
{
CTreeNode* pTreeNode = GetTreeNodeFromCookie();
if (pTreeNode == NULL)
{
return E_FAIL;
}
ASSERT(pTreeNode->IsContainer());
CContainerNode* pContainerNode = (CContainerNode*)pTreeNode;
// build the column id
LPCWSTR lpszColumnID = pContainerNode->GetColumnID();
// We are assuming the column ID is NULL terminated. Since this is usually
// hardcoded and there is no good way to verify that it is NULL terminated
// this usage should be fine.
size_t iLen = wcslen(lpszColumnID);
iLen += 1; // Include space for null.
// allocate enough memory for the struct and the string for the column id
size_t arraySizeInBytes = sizeof(SColumnSetID) + (iLen * sizeof(WCHAR));
SColumnSetID* pColumnID = (SColumnSetID*)malloc(arraySizeInBytes);
if (pColumnID != NULL)
{
memset(pColumnID, 0, arraySizeInBytes);
pColumnID->cBytes = static_cast<DWORD>(iLen * sizeof(WCHAR));
// NOTICE-2002/04/18-artm Part of fix for ntraid#ntbug9-540061.
// Unlike wcscpy(), StringCchCopy() will ensure that the destination
// buffer is null terminated and report an error code if there was
// a truncation (won't overrun the destination buffer).
//
// Since we needed to use strsafe.h elsewhere in this file, I decided
// to replace these dangerous wcscpy() uses that were deprecated by
// strsafe.h.
HRESULT err;
err = StringCchCopyW(
reinterpret_cast<LPWSTR>(pColumnID->id), // destination string
iLen, // size of destination string (including null)
lpszColumnID); // source string
if (FAILED(err))
{
ASSERT(false); // This should never happen.
free(pColumnID);
return err;
}
// copy the column id to global memory
size_t cb = sizeof(SColumnSetID) + (iLen * sizeof(WCHAR));
lpMedium->tymed = TYMED_HGLOBAL;
lpMedium->hGlobal = ::GlobalAlloc(GMEM_SHARE|GMEM_MOVEABLE, cb);
if (lpMedium->hGlobal == NULL)
return STG_E_MEDIUMFULL;
BYTE* pb = reinterpret_cast<BYTE*>(::GlobalLock(lpMedium->hGlobal));
if (pb)
{
// REVIEWED-2002/03/0-JeffJon-The count of bytes is equal to the number
// of bytes allocated
memcpy(pb, pColumnID, cb);
::GlobalUnlock(lpMedium->hGlobal);
}
free(pColumnID);
}
return S_OK;
}
HRESULT CDataObject::CreateNodeTypeData(LPSTGMEDIUM lpMedium)
{
// Create the node type object in GUID format
// First ask the related node, if failed, get the default GUID
// from the root node
CTreeNode* pNode = GetTreeNodeFromCookie();
if (pNode == NULL)
{
return E_FAIL;
}
const GUID* pNodeType = pNode->GetNodeType();
if (pNodeType == NULL)
{
pNodeType = GetDataFromComponentDataObject()->GetNodeType();
}
HRESULT hr = Create(pNodeType, sizeof(GUID), lpMedium);
return hr;
}
HRESULT CDataObject::CreateNodeTypeStringData(LPSTGMEDIUM lpMedium)
{
// Create the node type object in GUID string format
OLECHAR szNodeType[128] = {0};
// First ask the related node, if failed, get the default GUID
// from the root node
CTreeNode* pNode = GetTreeNodeFromCookie();
if (pNode == NULL)
{
return E_FAIL;
}
const GUID* pNodeType = pNode->GetNodeType();
if (pNodeType == NULL)
{
pNodeType = GetDataFromComponentDataObject()->GetNodeType();
}
::StringFromGUID2(*pNodeType,szNodeType,128);
return Create(szNodeType, BYTE_MEM_LEN_W(szNodeType), lpMedium);
}
HRESULT CDataObject::CreateDisplayName(LPSTGMEDIUM lpMedium)
{
// This is the display named used in the scope pane and snap-in manager
// We get it from the root node.
CString szDispName;
szDispName = GetDataFromComponentDataObject()->GetDisplayName();
return Create(szDispName, (szDispName.GetLength()+1) * sizeof(wchar_t), lpMedium);
}
HRESULT CDataObject::CreateCoClassID(LPSTGMEDIUM lpMedium)
{
// TODO
ASSERT(m_pUnkComponentData != NULL);
IPersistStream* pIPersistStream = NULL;
HRESULT hr = m_pUnkComponentData->QueryInterface(IID_IPersistStream, (void**)&pIPersistStream);
if (FAILED(hr))
return hr;
ASSERT(pIPersistStream != NULL);
// Create the CoClass information
CLSID clsid;
VERIFY(SUCCEEDED(pIPersistStream->GetClassID(&clsid)));
hr = Create(reinterpret_cast<const void*>(&clsid), sizeof(CLSID), lpMedium);
ASSERT(SUCCEEDED(hr));
pIPersistStream->Release();
return hr;
}
HRESULT CDataObject::CreateInternal(LPSTGMEDIUM lpMedium)
{
HRESULT hr = S_OK;
INTERNAL * pInt = NULL;
void * pBuf = NULL;
UINT size = sizeof(INTERNAL);
size += sizeof(CTreeNode*) * (m_internal.m_cookie_count);
pBuf = GlobalAlloc (GPTR, size);
if (pBuf != NULL)
{
pInt = (INTERNAL *) pBuf;
lpMedium->hGlobal = pBuf;
// copy the data
pInt->m_type = m_internal.m_type;
pInt->m_cookie_count = m_internal.m_cookie_count;
pInt->m_p_cookies = (CTreeNode**) ((BYTE *)pInt + sizeof(INTERNAL));
// REVIEWED-2002/03/08-JeffJon-The number of bytes being copied
// will fit in the supplied buffer
memcpy (pInt->m_p_cookies, m_internal.m_p_cookies,
sizeof(CTreeNode*) * (m_internal.m_cookie_count));
hr = Create(pBuf, size, lpMedium);
}
else
{
hr = E_OUTOFMEMORY;
}
return hr;
}
//+----------------------------------------------------------------------------
//
// Method: CDSDataObject::CreateMultiSelectObject
//
// Synopsis: this is to create the list of types selected
//
//-----------------------------------------------------------------------------
HRESULT CDataObject::CreateMultiSelectObject(LPSTGMEDIUM lpMedium)
{
CTreeNode** cookieArray = NULL;
cookieArray = (CTreeNode**) GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
m_internal.m_cookie_count*sizeof(CTreeNode*));
if (!cookieArray)
{
return E_OUTOFMEMORY;
}
for (UINT k=0; k<m_internal.m_cookie_count; k++)
{
cookieArray[k] = m_internal.m_p_cookies[k];
}
BOOL* bDuplicateArr = NULL;
bDuplicateArr = (BOOL*)GlobalAlloc(GMEM_FIXED | GMEM_ZEROINIT,
m_internal.m_cookie_count*sizeof(BOOL));
if (!bDuplicateArr)
{
if (cookieArray)
{
GlobalFree (cookieArray);
}
return E_OUTOFMEMORY;
}
UINT cCount = 0;
for (UINT index = 0; index < m_internal.m_cookie_count; index++)
{
for (UINT j = 0; j < index; j++)
{
GUID Guid1 = *(cookieArray[index]->GetNodeType());
GUID Guid2 = *(cookieArray[j]->GetNodeType());
if (IsEqualGUID (Guid1, Guid2))
{
bDuplicateArr[index] = TRUE;
break; //repeated GUID
}
}
if (!bDuplicateArr[index])
{
cCount++;
}
}
UINT size = sizeof(SMMCObjectTypes) + (cCount) * sizeof(GUID);
void * pTmp = ::GlobalAlloc(GPTR, size);
if (!pTmp)
{
if (cookieArray)
{
GlobalFree (cookieArray);
}
if (bDuplicateArr)
{
GlobalFree (bDuplicateArr);
}
return E_OUTOFMEMORY;
}
SMMCObjectTypes* pdata = reinterpret_cast<SMMCObjectTypes*>(pTmp);
pdata->count = cCount;
UINT i = 0;
for (index=0; index<m_internal.m_cookie_count; index++)
{
if (!bDuplicateArr[index])
{
pdata->guid[i++] = *(cookieArray[index]->GetNodeType());
}
}
ASSERT(i == cCount);
lpMedium->hGlobal = pTmp;
GlobalFree (cookieArray);
GlobalFree (bDuplicateArr);
return S_OK;
}
CRootData* CDataObject::GetDataFromComponentDataObject()
{
CComponentDataObject* pObject =
reinterpret_cast<CComponentDataObject*>(m_pUnkComponentData);
CRootData* pRootData = pObject->GetRootData();
ASSERT(pRootData != NULL);
return pRootData;
}
CTreeNode* CDataObject::GetTreeNodeFromCookie()
{
CComponentDataObject* pObject =
reinterpret_cast<CComponentDataObject*>(m_pUnkComponentData);
CTreeNode* pNode = NULL;
if (m_internal.m_cookie_count > 0)
{
pNode = m_internal.m_p_cookies[0];
if (pNode == NULL)
{
return pObject->GetRootData();
}
}
return pNode;
}
void CDataObject::AddCookie(CTreeNode* cookie)
{
const UINT MEM_CHUNK_SIZE = 10;
void * pTMP = NULL;
if ((m_internal.m_cookie_count) % MEM_CHUNK_SIZE == 0)
{
if (m_internal.m_p_cookies != NULL)
{
pTMP = realloc (m_internal.m_p_cookies,
(m_internal.m_cookie_count +
MEM_CHUNK_SIZE) * sizeof (CTreeNode*));
}
else
{
pTMP = malloc (MEM_CHUNK_SIZE * sizeof (CTreeNode*));
}
if (pTMP == NULL)
{
TRACE(_T("CDataObject::AddCookie - malloc/realloc failed.."));
ASSERT (pTMP != NULL);
if (m_internal.m_p_cookies)
{
free(m_internal.m_p_cookies);
m_internal.m_p_cookies = 0;
m_internal.m_cookie_count = 0;
return;
}
}
else
{
m_internal.m_p_cookies = (CTreeNode**)pTMP;
}
}
m_internal.m_p_cookies[m_internal.m_cookie_count] = cookie;
m_internal.m_cookie_count++;
}