1912 lines
44 KiB
C
1912 lines
44 KiB
C
|
// This is a part of the Active Template Library.
|
||
|
// Copyright (C) 1996-1998 Microsoft Corporation
|
||
|
// All rights reserved.
|
||
|
//
|
||
|
// This source code is only intended as a supplement to the
|
||
|
// Active Template Library Reference and related
|
||
|
// electronic documentation provided with the library.
|
||
|
// See these sources for detailed information regarding the
|
||
|
// Active Template Library product.
|
||
|
|
||
|
#ifndef __ATL_SNAPIN_H__
|
||
|
#define __ATL_SNAPIN_H__
|
||
|
|
||
|
#ifndef UNICODE
|
||
|
#error "Only Unicode builds supported"
|
||
|
#endif
|
||
|
|
||
|
#include <mmc.h>
|
||
|
#include <commctrl.h>
|
||
|
#pragma comment(lib, "mmc.lib")
|
||
|
|
||
|
|
||
|
// Wrappers for propertypage
|
||
|
#pragma comment(lib, "comctl32.lib")
|
||
|
|
||
|
template <class T>
|
||
|
class ATL_NO_VTABLE CSnapInPropertyPageImpl : public CDialogImplBase
|
||
|
{
|
||
|
public:
|
||
|
PROPSHEETPAGE m_psp;
|
||
|
|
||
|
operator PROPSHEETPAGE*() { return &m_psp; }
|
||
|
|
||
|
// Construction
|
||
|
CSnapInPropertyPageImpl(LPCTSTR lpszTitle = NULL)
|
||
|
{
|
||
|
// initialize PROPSHEETPAGE struct
|
||
|
memset(&m_psp, 0, sizeof(PROPSHEETPAGE));
|
||
|
m_psp.dwSize = sizeof(PROPSHEETPAGE);
|
||
|
m_psp.dwFlags = PSP_USECALLBACK;
|
||
|
m_psp.hInstance = _Module.GetResourceInstance();
|
||
|
m_psp.pszTemplate = MAKEINTRESOURCE(T::IDD);
|
||
|
m_psp.pfnDlgProc = (DLGPROC)T::StartDialogProc;
|
||
|
m_psp.pfnCallback = T::PropPageCallback;
|
||
|
m_psp.lParam = (LPARAM)this;
|
||
|
|
||
|
if(lpszTitle != NULL)
|
||
|
{
|
||
|
m_psp.pszTitle = lpszTitle;
|
||
|
m_psp.dwFlags |= PSP_USETITLE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static UINT CALLBACK PropPageCallback(HWND hWnd, UINT uMsg, LPPROPSHEETPAGE ppsp)
|
||
|
{
|
||
|
if(uMsg == PSPCB_CREATE)
|
||
|
{
|
||
|
ATLASSERT(hWnd == NULL);
|
||
|
CDialogImplBase* pPage = (CDialogImplBase*)ppsp->lParam;
|
||
|
_Module.AddCreateWndData(&pPage->m_thunk.cd, pPage);
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
HPROPSHEETPAGE Create()
|
||
|
{
|
||
|
return ::CreatePropertySheetPage(&m_psp);
|
||
|
}
|
||
|
|
||
|
BOOL EndDialog(int)
|
||
|
{
|
||
|
// do nothing here, calling ::EndDialog will close the whole sheet
|
||
|
ATLASSERT(FALSE);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
// Operations
|
||
|
void CancelToClose()
|
||
|
{
|
||
|
ATLASSERT(::IsWindow(m_hWnd));
|
||
|
ATLASSERT(GetParent() != NULL);
|
||
|
|
||
|
::SendMessage(GetParent(), PSM_CANCELTOCLOSE, 0, 0L);
|
||
|
}
|
||
|
void SetModified(BOOL bChanged = TRUE)
|
||
|
{
|
||
|
ATLASSERT(::IsWindow(m_hWnd));
|
||
|
ATLASSERT(GetParent() != NULL);
|
||
|
|
||
|
if(bChanged)
|
||
|
::SendMessage(GetParent(), PSM_CHANGED, (WPARAM)m_hWnd, 0L);
|
||
|
else
|
||
|
::SendMessage(GetParent(), PSM_UNCHANGED, (WPARAM)m_hWnd, 0L);
|
||
|
}
|
||
|
LRESULT QuerySiblings(WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
ATLASSERT(::IsWindow(m_hWnd));
|
||
|
ATLASSERT(GetParent() != NULL);
|
||
|
|
||
|
return ::SendMessage(GetParent(), PSM_QUERYSIBLINGS, wParam, lParam);
|
||
|
}
|
||
|
|
||
|
BEGIN_MSG_MAP(CSnapInPropertyPageImpl< T >)
|
||
|
MESSAGE_HANDLER(WM_NOTIFY, OnNotify)
|
||
|
END_MSG_MAP()
|
||
|
|
||
|
// Message handler
|
||
|
LRESULT OnNotify(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
||
|
{
|
||
|
ATLASSERT(::IsWindow(m_hWnd));
|
||
|
NMHDR* pNMHDR = (NMHDR*)lParam;
|
||
|
|
||
|
// don't handle messages not from the page/sheet itself
|
||
|
if(pNMHDR->hwndFrom != m_hWnd && pNMHDR->hwndFrom != ::GetParent(m_hWnd))
|
||
|
{
|
||
|
bHandled = FALSE;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
T* pT = (T*)this;
|
||
|
LRESULT lResult = 0;
|
||
|
// handle default
|
||
|
switch(pNMHDR->code)
|
||
|
{
|
||
|
case PSN_SETACTIVE:
|
||
|
lResult = pT->OnSetActive() ? 0 : -1;
|
||
|
break;
|
||
|
case PSN_KILLACTIVE:
|
||
|
lResult = !pT->OnKillActive();
|
||
|
break;
|
||
|
case PSN_APPLY:
|
||
|
lResult = pT->OnApply() ? PSNRET_NOERROR : PSNRET_INVALID_NOCHANGEPAGE;
|
||
|
break;
|
||
|
case PSN_RESET:
|
||
|
pT->OnReset();
|
||
|
break;
|
||
|
case PSN_QUERYCANCEL:
|
||
|
lResult = !pT->OnQueryCancel();
|
||
|
break;
|
||
|
case PSN_WIZNEXT:
|
||
|
lResult = !pT->OnWizardNext();
|
||
|
break;
|
||
|
case PSN_WIZBACK:
|
||
|
lResult = !pT->OnWizardBack();
|
||
|
break;
|
||
|
case PSN_WIZFINISH:
|
||
|
lResult = !pT->OnWizardFinish();
|
||
|
break;
|
||
|
case PSN_HELP:
|
||
|
lResult = pT->OnHelp();
|
||
|
break;
|
||
|
default:
|
||
|
bHandled = FALSE; // not handled
|
||
|
}
|
||
|
|
||
|
return lResult;
|
||
|
}
|
||
|
|
||
|
// Overridables
|
||
|
BOOL OnSetActive()
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
BOOL OnKillActive()
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
BOOL OnApply()
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
void OnReset()
|
||
|
{
|
||
|
}
|
||
|
BOOL OnQueryCancel()
|
||
|
{
|
||
|
return TRUE; // ok to cancel
|
||
|
}
|
||
|
BOOL OnWizardBack()
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
BOOL OnWizardNext()
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
BOOL OnWizardFinish()
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
BOOL OnHelp()
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
#if _ATL_VER < 0x0300
|
||
|
//Overridden to reference overridden DialogProc
|
||
|
static LRESULT CALLBACK StartDialogProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
CSnapInPropertyPageImpl< T >* pThis = (CSnapInPropertyPageImpl< T >*)_Module.ExtractCreateWndData();
|
||
|
ATLASSERT(pThis != NULL);
|
||
|
pThis->m_hWnd = hWnd;
|
||
|
pThis->m_thunk.Init(DialogProc, pThis);
|
||
|
WNDPROC pProc = (WNDPROC)&(pThis->m_thunk.thunk);
|
||
|
WNDPROC pOldProc;
|
||
|
pOldProc = (WNDPROC)::SetWindowLongPtr(hWnd, DWLP_DLGPROC, (LONG_PTR)pProc);
|
||
|
#ifdef _DEBUG
|
||
|
// check if somebody has subclassed us already since we discard it
|
||
|
if(pOldProc != StartDialogProc)
|
||
|
ATLTRACE(_T("ATL: Subclassing through a hook discarded.\n"));
|
||
|
#endif
|
||
|
return pProc(hWnd, uMsg, wParam, lParam);
|
||
|
}
|
||
|
|
||
|
// Overriden for handling WM_NCDESTROY correctly
|
||
|
static LRESULT CALLBACK DialogProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
#ifdef _M_IX86
|
||
|
__asm mov dword ptr[hWnd], ecx
|
||
|
#endif
|
||
|
CSnapInPropertyPageImpl< T >* pThis = (CSnapInPropertyPageImpl< T >*)hWnd;
|
||
|
LRESULT lRes;
|
||
|
if(pThis->ProcessWindowMessage(pThis->m_hWnd, uMsg, wParam, lParam, lRes, 0))
|
||
|
{
|
||
|
switch (uMsg)
|
||
|
{
|
||
|
case WM_COMPAREITEM:
|
||
|
case WM_VKEYTOITEM:
|
||
|
case WM_CHARTOITEM:
|
||
|
case WM_INITDIALOG:
|
||
|
case WM_QUERYDRAGICON:
|
||
|
case WM_CTLCOLORMSGBOX:
|
||
|
case WM_CTLCOLOREDIT:
|
||
|
case WM_CTLCOLORLISTBOX:
|
||
|
case WM_CTLCOLORBTN:
|
||
|
case WM_CTLCOLORDLG:
|
||
|
case WM_CTLCOLORSCROLLBAR:
|
||
|
case WM_CTLCOLORSTATIC:
|
||
|
return lRes;
|
||
|
break;
|
||
|
}
|
||
|
::SetWindowLongPtr(pThis->m_hWnd, DWLP_MSGRESULT, lRes);
|
||
|
return TRUE;
|
||
|
}
|
||
|
if(uMsg == WM_NCDESTROY)
|
||
|
{
|
||
|
// clear out window handle
|
||
|
HWND hWnd = pThis->m_hWnd;
|
||
|
pThis->m_hWnd = NULL;
|
||
|
// clean up after dialog is destroyed
|
||
|
pThis->OnFinalMessage(hWnd);
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
virtual void OnFinalMessage(HWND hWnd)
|
||
|
{
|
||
|
};
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
#if _ATL_VER < 0x0300
|
||
|
// intended for small number of simple types or pointers
|
||
|
template <class TKey, class TVal>
|
||
|
class CSnapInSimpleMap
|
||
|
{
|
||
|
public:
|
||
|
TKey* m_aKey;
|
||
|
TVal* m_aVal;
|
||
|
int m_nSize;
|
||
|
|
||
|
// Construction/destruction
|
||
|
CSnapInSimpleMap() : m_aKey(NULL), m_aVal(NULL), m_nSize(0)
|
||
|
{ }
|
||
|
|
||
|
~CSnapInSimpleMap()
|
||
|
{
|
||
|
RemoveAll();
|
||
|
}
|
||
|
|
||
|
// Operations
|
||
|
int GetSize() const
|
||
|
{
|
||
|
return m_nSize;
|
||
|
}
|
||
|
BOOL Add(TKey key, TVal val)
|
||
|
{
|
||
|
TKey* pKey;
|
||
|
pKey = (TKey*)realloc(m_aKey, (m_nSize + 1) * sizeof(TKey));
|
||
|
if(pKey == NULL)
|
||
|
return FALSE;
|
||
|
m_aKey = pKey;
|
||
|
TVal* pVal;
|
||
|
pVal = (TVal*)realloc(m_aVal, (m_nSize + 1) * sizeof(TVal));
|
||
|
if(pVal == NULL)
|
||
|
return FALSE;
|
||
|
m_aVal = pVal;
|
||
|
m_nSize++;
|
||
|
SetAtIndex(m_nSize - 1, key, val);
|
||
|
return TRUE;
|
||
|
}
|
||
|
BOOL Remove(TKey key)
|
||
|
{
|
||
|
int nIndex = FindKey(key);
|
||
|
if(nIndex == -1)
|
||
|
return FALSE;
|
||
|
if(nIndex != (m_nSize - 1))
|
||
|
{
|
||
|
memmove((void*)&m_aKey[nIndex], (void*)&m_aKey[nIndex + 1], (m_nSize - (nIndex + 1)) * sizeof(TKey));
|
||
|
memmove((void*)&m_aVal[nIndex], (void*)&m_aVal[nIndex + 1], (m_nSize - (nIndex + 1)) * sizeof(TVal));
|
||
|
}
|
||
|
TKey* pKey;
|
||
|
pKey = (TKey*)realloc(m_aKey, (m_nSize - 1) * sizeof(TKey));
|
||
|
if(pKey != NULL || m_nSize == 1)
|
||
|
m_aKey = pKey;
|
||
|
TVal* pVal;
|
||
|
pVal = (TVal*)realloc(m_aVal, (m_nSize - 1) * sizeof(TVal));
|
||
|
if(pVal != NULL || m_nSize == 1)
|
||
|
m_aVal = pVal;
|
||
|
m_nSize--;
|
||
|
return TRUE;
|
||
|
}
|
||
|
void RemoveAll()
|
||
|
{
|
||
|
if(m_nSize > 0)
|
||
|
{
|
||
|
free(m_aKey);
|
||
|
free(m_aVal);
|
||
|
m_aKey = NULL;
|
||
|
m_aVal = NULL;
|
||
|
m_nSize = 0;
|
||
|
}
|
||
|
}
|
||
|
BOOL SetAt(TKey key, TVal val)
|
||
|
{
|
||
|
int nIndex = FindKey(key);
|
||
|
if(nIndex == -1)
|
||
|
return FALSE;
|
||
|
SetAtIndex(nIndex, key, val);
|
||
|
return TRUE;
|
||
|
}
|
||
|
TVal Lookup(TKey key) const
|
||
|
{
|
||
|
int nIndex = FindKey(key);
|
||
|
if(nIndex == -1)
|
||
|
return NULL; // must be able to convert
|
||
|
return GetValueAt(nIndex);
|
||
|
}
|
||
|
TKey ReverseLookup(TVal val) const
|
||
|
{
|
||
|
int nIndex = FindVal(val);
|
||
|
if(nIndex == -1)
|
||
|
return NULL; // must be able to convert
|
||
|
return GetKeyAt(nIndex);
|
||
|
}
|
||
|
TKey& GetKeyAt(int nIndex) const
|
||
|
{
|
||
|
ATLASSERT(nIndex >= 0 && nIndex < m_nSize);
|
||
|
return m_aKey[nIndex];
|
||
|
}
|
||
|
TVal& GetValueAt(int nIndex) const
|
||
|
{
|
||
|
ATLASSERT(nIndex >= 0 && nIndex < m_nSize);
|
||
|
return m_aVal[nIndex];
|
||
|
}
|
||
|
|
||
|
// Implementation
|
||
|
void SetAtIndex(int nIndex, TKey& key, TVal& val)
|
||
|
{
|
||
|
ATLASSERT(nIndex >= 0 && nIndex < m_nSize);
|
||
|
m_aKey[nIndex] = key;
|
||
|
m_aVal[nIndex] = val;
|
||
|
}
|
||
|
int FindKey(TKey& key) const
|
||
|
{
|
||
|
for(int i = 0; i < m_nSize; i++)
|
||
|
{
|
||
|
if(m_aKey[i] == key)
|
||
|
return i;
|
||
|
}
|
||
|
return -1; // not found
|
||
|
}
|
||
|
int FindVal(TVal& val) const
|
||
|
{
|
||
|
for(int i = 0; i < m_nSize; i++)
|
||
|
{
|
||
|
if(m_aVal[i] == val)
|
||
|
return i;
|
||
|
}
|
||
|
return -1; // not found
|
||
|
}
|
||
|
};
|
||
|
#endif
|
||
|
|
||
|
/*class CSnapInBitmap
|
||
|
{
|
||
|
public:
|
||
|
HBITMAP m_hBitmap;
|
||
|
|
||
|
CSnapInBitmap(HBITMAP hBitmap = NULL) : m_hBitmap(hBitmap)
|
||
|
{ }
|
||
|
~CSnapInBitmap()
|
||
|
{
|
||
|
if(m_hBitmap != NULL)
|
||
|
DeleteObject();
|
||
|
}
|
||
|
|
||
|
CSnapInBitmap& operator=(HBITMAP hBitmap)
|
||
|
{
|
||
|
m_hBitmap = hBitmap;
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
void Attach(HBITMAP hBitmap)
|
||
|
{
|
||
|
m_hBitmap = hBitmap;
|
||
|
}
|
||
|
HBITMAP Detach()
|
||
|
{
|
||
|
HBITMAP hBitmap = m_hBitmap;
|
||
|
m_hBitmap = NULL;
|
||
|
return hBitmap;
|
||
|
}
|
||
|
|
||
|
operator HBITMAP() const { return m_hBitmap; }
|
||
|
|
||
|
HBITMAP LoadBitmap(LPCTSTR lpszResourceName)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::LoadBitmap(_Module.GetResourceInstance(), lpszResourceName);
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
HBITMAP LoadBitmap(UINT nIDResource)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::LoadBitmap(_Module.GetResourceInstance(), MAKEINTRESOURCE(nIDResource));
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
HBITMAP LoadOEMBitmap(UINT nIDBitmap) // for OBM_/OCR_/OIC_
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::LoadBitmap(NULL, MAKEINTRESOURCE(nIDBitmap));
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
HBITMAP LoadMappedBitmap(UINT nIDBitmap, UINT nFlags = 0, LPCOLORMAP lpColorMap = NULL, int nMapSize = 0)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::CreateMappedBitmap(_Module.GetResourceInstance(), nIDBitmap, (WORD)nFlags, lpColorMap, nMapSize);
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
HBITMAP CreateBitmap(int nWidth, int nHeight, UINT nPlanes, UINT nBitcount, const void* lpBits)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::CreateBitmap(nWidth, nHeight, nPlanes, nBitcount, lpBits);
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
HBITMAP CreateBitmapIndirect(LPBITMAP lpBitmap)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::CreateBitmapIndirect(lpBitmap);
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
HBITMAP CreateCompatibleBitmap(HDC hDC, int nWidth, int nHeight)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::CreateCompatibleBitmap(hDC, nWidth, nHeight);
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
HBITMAP CreateDiscardableBitmap(HDC hDC, int nWidth, int nHeight)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap == NULL);
|
||
|
m_hBitmap = ::CreateDiscardableBitmap(hDC, nWidth, nHeight);
|
||
|
return m_hBitmap;
|
||
|
}
|
||
|
|
||
|
BOOL DeleteObject()
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap != NULL);
|
||
|
BOOL bRet = ::DeleteObject(m_hBitmap);
|
||
|
if(bRet)
|
||
|
m_hBitmap = NULL;
|
||
|
return bRet;
|
||
|
}
|
||
|
|
||
|
// Attributes
|
||
|
int GetBitmap(BITMAP* pBitMap)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap != NULL);
|
||
|
return ::GetObject(m_hBitmap, sizeof(BITMAP), pBitMap);
|
||
|
}
|
||
|
// Operations
|
||
|
DWORD SetBitmapBits(DWORD dwCount, const void* lpBits)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap != NULL);
|
||
|
return ::SetBitmapBits(m_hBitmap, dwCount, lpBits);
|
||
|
}
|
||
|
DWORD GetBitmapBits(DWORD dwCount, LPVOID lpBits) const
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap != NULL);
|
||
|
return ::GetBitmapBits(m_hBitmap, dwCount, lpBits);
|
||
|
}
|
||
|
BOOL SetBitmapDimension(int nWidth, int nHeight, LPSIZE lpSize = NULL)
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap != NULL);
|
||
|
return ::SetBitmapDimensionEx(m_hBitmap, nWidth, nHeight, lpSize);
|
||
|
}
|
||
|
BOOL GetBitmapDimension(LPSIZE lpSize) const
|
||
|
{
|
||
|
ATLASSERT(m_hBitmap != NULL);
|
||
|
return ::GetBitmapDimensionEx(m_hBitmap, lpSize);
|
||
|
}
|
||
|
};
|
||
|
*/
|
||
|
class CSnapInItem;
|
||
|
|
||
|
class CObjectData
|
||
|
{
|
||
|
public:
|
||
|
CSnapInItem* m_pItem;
|
||
|
DATA_OBJECT_TYPES m_type;
|
||
|
};
|
||
|
|
||
|
class ATL_NO_VTABLE CSnapInItem
|
||
|
{
|
||
|
public:
|
||
|
virtual ~CSnapInItem()
|
||
|
{
|
||
|
}
|
||
|
STDMETHOD(Notify)(MMC_NOTIFY_TYPE event,
|
||
|
LPARAM arg,
|
||
|
LPARAM param,
|
||
|
IComponentData* pComponentData,
|
||
|
IComponent* pComponent,
|
||
|
DATA_OBJECT_TYPES type) = 0;
|
||
|
|
||
|
STDMETHOD(GetScopePaneInfo)(SCOPEDATAITEM *pScopeDataItem) = 0;
|
||
|
|
||
|
STDMETHOD(GetResultViewType)(LPOLESTR *ppViewType,
|
||
|
long *pViewOptions) = 0;
|
||
|
|
||
|
STDMETHOD(GetResultPaneInfo)(RESULTDATAITEM *pResultDataItem) = 0;
|
||
|
|
||
|
STDMETHOD(AddMenuItems)(LPCONTEXTMENUCALLBACK piCallback,
|
||
|
long *pInsertionAllowed,
|
||
|
DATA_OBJECT_TYPES type) = 0;
|
||
|
|
||
|
STDMETHOD(Command)(long lCommandID,
|
||
|
DATA_OBJECT_TYPES type) = 0;
|
||
|
|
||
|
STDMETHOD(CreatePropertyPages)(LPPROPERTYSHEETCALLBACK lpProvider,
|
||
|
LONG_PTR handle,
|
||
|
IUnknown* pUnk,
|
||
|
DATA_OBJECT_TYPES type) = 0;
|
||
|
|
||
|
STDMETHOD(QueryPagesFor)(DATA_OBJECT_TYPES type) = 0;
|
||
|
|
||
|
STDMETHOD(SetControlbar)(IControlbar *pControlbar,
|
||
|
IExtendControlbar *pExtendControlbar,
|
||
|
CSnapInSimpleMap<UINT, IUnknown*>* pToolbarMap) = 0;
|
||
|
|
||
|
STDMETHOD(ControlbarNotify)(IControlbar *pControlbar,
|
||
|
IExtendControlbar *pExtendControlbar,
|
||
|
CSnapInSimpleMap<UINT, IUnknown*>* pToolbarMap,
|
||
|
MMC_NOTIFY_TYPE event,
|
||
|
LPARAM arg,
|
||
|
LPARAM param,
|
||
|
DATA_OBJECT_TYPES type) = 0;
|
||
|
|
||
|
STDMETHOD(GetScopeData)(SCOPEDATAITEM * *pScopeDataItem) = 0;
|
||
|
|
||
|
STDMETHOD(GetResultData)(RESULTDATAITEM * *pResultDataItem) = 0;
|
||
|
|
||
|
STDMETHOD(FillData)(CLIPFORMAT cf,
|
||
|
LPSTREAM pStream) = 0;
|
||
|
|
||
|
virtual void InitDataClass(IDataObject* pDataObject, CSnapInItem* pDefault)
|
||
|
{
|
||
|
_ASSERTE(0 && "Override this function in derived class");
|
||
|
}
|
||
|
|
||
|
static HRESULT GetDataClass(IDataObject* pDataObj, CSnapInItem** ppItem, DATA_OBJECT_TYPES* pType)
|
||
|
{
|
||
|
if (ppItem == NULL)
|
||
|
return E_POINTER;
|
||
|
if (pType == NULL)
|
||
|
return E_POINTER;
|
||
|
|
||
|
*ppItem = NULL;
|
||
|
*pType = CCT_UNINITIALIZED;
|
||
|
|
||
|
STGMEDIUM stgmedium = { TYMED_HGLOBAL, NULL };
|
||
|
FORMATETC formatetc = { m_CCF_SNAPIN_GETOBJECTDATA,
|
||
|
NULL,
|
||
|
DVASPECT_CONTENT,
|
||
|
-1,
|
||
|
TYMED_HGLOBAL
|
||
|
};
|
||
|
|
||
|
stgmedium.hGlobal = GlobalAlloc(0, sizeof(CObjectData));
|
||
|
if (stgmedium.hGlobal == NULL)
|
||
|
return E_OUTOFMEMORY;
|
||
|
|
||
|
HRESULT hr = pDataObj->GetDataHere(&formatetc, &stgmedium);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
CObjectData* pData = (CObjectData*)stgmedium.hGlobal;
|
||
|
*ppItem = pData->m_pItem;
|
||
|
*pType = pData->m_type;
|
||
|
}
|
||
|
|
||
|
GlobalFree(stgmedium.hGlobal);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
virtual HRESULT STDMETHODCALLTYPE GetDataObject(IDataObject** pDataObj, DATA_OBJECT_TYPES type) = 0;
|
||
|
|
||
|
static void Init()
|
||
|
{
|
||
|
m_CCF_NODETYPE = (CLIPFORMAT) RegisterClipboardFormat(_T("CCF_NODETYPE"));
|
||
|
m_CCF_SZNODETYPE = (CLIPFORMAT) RegisterClipboardFormat(_T("CCF_SZNODETYPE"));
|
||
|
m_CCF_DISPLAY_NAME = (CLIPFORMAT) RegisterClipboardFormat(_T("CCF_DISPLAY_NAME"));
|
||
|
m_CCF_SNAPIN_CLASSID = (CLIPFORMAT) RegisterClipboardFormat(_T("CCF_SNAPIN_CLASSID"));
|
||
|
m_CCF_SNAPIN_GETOBJECTDATA = (CLIPFORMAT) RegisterClipboardFormat(_T("CCF_GETOBJECTDATA"));
|
||
|
}
|
||
|
public:
|
||
|
static CLIPFORMAT m_CCF_NODETYPE;
|
||
|
static CLIPFORMAT m_CCF_SZNODETYPE;
|
||
|
static CLIPFORMAT m_CCF_DISPLAY_NAME;
|
||
|
static CLIPFORMAT m_CCF_SNAPIN_CLASSID;
|
||
|
static CLIPFORMAT m_CCF_SNAPIN_GETOBJECTDATA;
|
||
|
};
|
||
|
|
||
|
class CSnapInObjectRoot
|
||
|
{
|
||
|
public:
|
||
|
CComPtr <IControlbar> m_spControlbar;
|
||
|
CSnapInSimpleMap <UINT, IUnknown*> m_toolbarMap;
|
||
|
|
||
|
HRESULT GetDataClass(IDataObject* pDataObject, CSnapInItem** ppItem, DATA_OBJECT_TYPES* pType)
|
||
|
{
|
||
|
return CSnapInItem::GetDataClass(pDataObject, ppItem, pType);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
#define EXTENSION_SNAPIN_DATACLASS(dataClass) dataClass m_##dataClass;
|
||
|
|
||
|
#define BEGIN_EXTENSION_SNAPIN_NODEINFO_MAP(classname) \
|
||
|
HRESULT GetDataClass(IDataObject* pDataObject, CSnapInItem** ppItem, DATA_OBJECT_TYPES* pType) \
|
||
|
{ \
|
||
|
if (ppItem == NULL) \
|
||
|
return E_POINTER; \
|
||
|
if (pType == NULL) \
|
||
|
return E_POINTER; \
|
||
|
\
|
||
|
*ppItem = NULL; \
|
||
|
\
|
||
|
*pType = CCT_UNINITIALIZED; \
|
||
|
\
|
||
|
STGMEDIUM stgmedium = { TYMED_HGLOBAL, NULL }; \
|
||
|
FORMATETC formatetc = { CSnapInItem::m_CCF_NODETYPE, \
|
||
|
NULL, \
|
||
|
DVASPECT_CONTENT, \
|
||
|
-1, \
|
||
|
TYMED_HGLOBAL \
|
||
|
}; \
|
||
|
\
|
||
|
stgmedium.hGlobal = GlobalAlloc(0, sizeof(GUID)); \
|
||
|
if (stgmedium.hGlobal == NULL) \
|
||
|
return E_OUTOFMEMORY; \
|
||
|
\
|
||
|
HRESULT hr = pDataObject->GetDataHere(&formatetc, &stgmedium); \
|
||
|
if (FAILED(hr)) \
|
||
|
{ \
|
||
|
GlobalFree(stgmedium.hGlobal); \
|
||
|
return hr; \
|
||
|
} \
|
||
|
\
|
||
|
GUID guid; \
|
||
|
memcpy(&guid, stgmedium.hGlobal, sizeof(GUID)); \
|
||
|
\
|
||
|
GlobalFree(stgmedium.hGlobal); \
|
||
|
hr = S_OK;
|
||
|
|
||
|
#define EXTENSION_SNAPIN_NODEINFO_ENTRY(dataClass) \
|
||
|
if (IsEqualGUID(guid, *(GUID*)m_##dataClass.GetNodeType())) \
|
||
|
{ \
|
||
|
*ppItem = m_##dataClass.GetExtNodeObject(pDataObject, &m_##dataClass); \
|
||
|
_ASSERTE(*ppItem != NULL); \
|
||
|
(*ppItem)->InitDataClass(pDataObject, &m_##dataClass); \
|
||
|
return hr; \
|
||
|
}
|
||
|
|
||
|
#define END_EXTENSION_SNAPIN_NODEINFO_MAP() \
|
||
|
return CSnapInItem::GetDataClass(pDataObject, ppItem, pType); \
|
||
|
};
|
||
|
|
||
|
class ATL_NO_VTABLE CSnapInDataObjectImpl : public IDataObject,
|
||
|
public CComObjectRoot
|
||
|
{
|
||
|
public:
|
||
|
BEGIN_COM_MAP(CSnapInDataObjectImpl)
|
||
|
COM_INTERFACE_ENTRY(IDataObject)
|
||
|
END_COM_MAP()
|
||
|
STDMETHOD(GetData)(FORMATETC *pformatetcIn, STGMEDIUM *pmedium)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::GetData\n"));
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetDataHere)(FORMATETC* pformatetc, STGMEDIUM* pmedium)
|
||
|
{
|
||
|
ATLTRACE(_T("SnapInDataObjectImpl::GetDataHere\n"));
|
||
|
if (pmedium == NULL)
|
||
|
return E_POINTER;
|
||
|
|
||
|
HRESULT hr = DV_E_TYMED;
|
||
|
// Make sure the type medium is HGLOBAL
|
||
|
if (pmedium->tymed == TYMED_HGLOBAL)
|
||
|
{
|
||
|
// Create the stream on the hGlobal passed in
|
||
|
CComPtr<IStream> spStream;
|
||
|
hr = CreateStreamOnHGlobal(pmedium->hGlobal, FALSE, &spStream);
|
||
|
if (SUCCEEDED(hr))
|
||
|
if (pformatetc->cfFormat == CSnapInItem::m_CCF_SNAPIN_GETOBJECTDATA)
|
||
|
{
|
||
|
hr = DV_E_CLIPFORMAT;
|
||
|
ULONG uWritten;
|
||
|
hr = spStream->Write(&m_objectData, sizeof(CObjectData), &uWritten);
|
||
|
}
|
||
|
else
|
||
|
hr = m_objectData.m_pItem->FillData(pformatetc->cfFormat, spStream);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(QueryGetData)(FORMATETC* /* pformatetc */)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::QueryGetData\n"));
|
||
|
}
|
||
|
STDMETHOD(GetCanonicalFormatEtc)(FORMATETC* /* pformatectIn */,FORMATETC* /* pformatetcOut */)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::GetCanonicalFormatEtc\n"));
|
||
|
}
|
||
|
STDMETHOD(SetData)(FORMATETC* /* pformatetc */, STGMEDIUM* /* pmedium */, BOOL /* fRelease */)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::SetData\n"));
|
||
|
}
|
||
|
STDMETHOD(EnumFormatEtc)(DWORD /* dwDirection */, IEnumFORMATETC** /* ppenumFormatEtc */)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::EnumFormatEtc\n"));
|
||
|
}
|
||
|
STDMETHOD(DAdvise)(FORMATETC *pformatetc, DWORD advf, IAdviseSink *pAdvSink,
|
||
|
DWORD *pdwConnection)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::SetData\n"));
|
||
|
}
|
||
|
STDMETHOD(DUnadvise)(DWORD dwConnection)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::SetDatan\n"));
|
||
|
}
|
||
|
STDMETHOD(EnumDAdvise)(IEnumSTATDATA **ppenumAdvise)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("SnapInDataObjectImpl::SetData\n"));
|
||
|
}
|
||
|
|
||
|
CObjectData m_objectData;
|
||
|
};
|
||
|
|
||
|
|
||
|
template <class T, class Component>
|
||
|
class ATL_NO_VTABLE IComponentDataImpl : public IComponentData
|
||
|
{
|
||
|
public :
|
||
|
IComponentDataImpl()
|
||
|
{
|
||
|
m_pNode = NULL;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(Initialize)(LPUNKNOWN pUnknown)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentDataImpl::Initialize\n"));
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(pUnknown != NULL);
|
||
|
if (pUnknown == NULL)
|
||
|
ATLTRACE(_T("IComponentData::Initialize called with pUnknown == NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
// review change to QI
|
||
|
hr = S_OK;
|
||
|
m_spConsole = pUnknown;
|
||
|
if (m_spConsole == NULL)
|
||
|
{
|
||
|
ATLTRACE(_T("QI for IConsole failed\n"));
|
||
|
hr = E_UNEXPECTED;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(CreateComponent)(LPCOMPONENT *ppComponent)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentDataImpl::CreateComponent\n"));
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(ppComponent != NULL);
|
||
|
if (ppComponent == NULL)
|
||
|
ATLTRACE(_T("IComponentData::CreateComponent called with ppComponent == NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
*ppComponent = NULL;
|
||
|
|
||
|
CComObject< Component >* pComponent;
|
||
|
hr = CComObject< Component >::CreateInstance(&pComponent);
|
||
|
ATLASSERT(SUCCEEDED(hr));
|
||
|
if (FAILED(hr))
|
||
|
ATLTRACE(_T("IComponentData::CreateComponent : Could not create IComponent object\n"));
|
||
|
else
|
||
|
hr = pComponent->QueryInterface(IID_IComponent, (void**)ppComponent);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
STDMETHOD(Notify)(
|
||
|
LPDATAOBJECT lpDataObject,
|
||
|
MMC_NOTIFY_TYPE event,
|
||
|
LPARAM arg,
|
||
|
LPARAM param)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentDataImpl::Notify\n"));
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(lpDataObject != NULL);
|
||
|
if (lpDataObject == NULL)
|
||
|
ATLTRACE(_T("IComponentData::Notify called with lpDataObject == NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
CSnapInItem* pItem;
|
||
|
DATA_OBJECT_TYPES type;
|
||
|
hr = pT->GetDataClass(lpDataObject, &pItem, &type);
|
||
|
ATLASSERT(SUCCEEDED(hr));
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pItem->Notify(event, arg, param, pT, NULL, type);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(Destroy)(void)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentDataImpl::Destroy\n"));
|
||
|
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
if (pT->m_spControlbar != NULL)
|
||
|
{
|
||
|
int n = pT->m_toolbarMap.GetSize();
|
||
|
for (int i = 0; i < n; i++)
|
||
|
{
|
||
|
IToolbar* pToolbar = (IToolbar*)pT->m_toolbarMap.GetValueAt(i);
|
||
|
if (pToolbar != NULL)
|
||
|
{
|
||
|
pT->m_spControlbar->Detach(pToolbar);
|
||
|
pToolbar->Release();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
pT->m_toolbarMap.RemoveAll();
|
||
|
|
||
|
m_spConsole.Release();
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(QueryDataObject)(LONG_PTR cookie,
|
||
|
DATA_OBJECT_TYPES type,
|
||
|
LPDATAOBJECT *ppDataObject)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentDataImpl::QueryDataObject\n"));
|
||
|
ATLASSERT(m_pNode != NULL);
|
||
|
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(ppDataObject != NULL);
|
||
|
if (ppDataObject == NULL)
|
||
|
ATLTRACE(_T("IComponentData::QueryDataObject called with ppDataObject == NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
*ppDataObject = NULL;
|
||
|
|
||
|
CSnapInItem* pItem = (CSnapInItem*) cookie;
|
||
|
if (cookie == NULL)
|
||
|
pItem = m_pNode;
|
||
|
|
||
|
hr = pItem->GetDataObject(ppDataObject, type);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetDisplayInfo)(SCOPEDATAITEM *pScopeDataItem)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentDataImpl::GetDisplayInfo\n"));
|
||
|
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(pScopeDataItem != NULL);
|
||
|
if (pScopeDataItem == NULL)
|
||
|
ATLTRACE(_T("IComponentData::GetDisplayInfo called with pScopeDataItem == NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
CSnapInItem* pItem= (CSnapInItem*) pScopeDataItem->lParam;
|
||
|
if (pItem == NULL)
|
||
|
pItem = m_pNode;
|
||
|
|
||
|
hr = E_UNEXPECTED;
|
||
|
if (pItem != NULL)
|
||
|
hr = pItem->GetScopePaneInfo(pScopeDataItem);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(CompareObjects)(LPDATAOBJECT lpDataObjectA,
|
||
|
LPDATAOBJECT lpDataObjectB)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("IComponentDataImpl::CompareObjects\n"));
|
||
|
}
|
||
|
|
||
|
CComQIPtr<IConsole, &IID_IConsole> m_spConsole;
|
||
|
|
||
|
protected:
|
||
|
CSnapInItem* m_pNode;
|
||
|
};
|
||
|
|
||
|
|
||
|
template <class T>
|
||
|
class ATL_NO_VTABLE IComponentImpl : public IComponent
|
||
|
{
|
||
|
public:
|
||
|
STDMETHOD(Initialize)(LPCONSOLE lpConsole)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentImpl::Initialize\n"));
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(lpConsole != NULL);
|
||
|
if (lpConsole == NULL)
|
||
|
ATLTRACE(_T("lpConsole is NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
m_spConsole = lpConsole;
|
||
|
|
||
|
CComQIPtr<IHeaderCtrl, &IID_IHeaderCtrl> spHeaderCtrl;
|
||
|
// review : change to QI
|
||
|
spHeaderCtrl = lpConsole;
|
||
|
if (spHeaderCtrl == NULL)
|
||
|
ATLTRACE(_T("QI for IHeaderCtrl failed\n"));
|
||
|
else
|
||
|
{
|
||
|
hr = m_spConsole->SetHeader(spHeaderCtrl);
|
||
|
if (FAILED(hr))
|
||
|
ATLTRACE(_T("IConsole::SetHeader failed (HRESULT = %x)\n"), hr);
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(Notify)(LPDATAOBJECT lpDataObject,
|
||
|
MMC_NOTIFY_TYPE event,
|
||
|
LPARAM arg,
|
||
|
LPARAM param)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentImpl::Notify\n"));
|
||
|
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(lpDataObject != NULL);
|
||
|
if (lpDataObject == NULL)
|
||
|
ATLTRACE(_T("IComponent::Notify called with lpDataObject==NULL \n"));
|
||
|
else
|
||
|
{
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
CSnapInItem* pItem;
|
||
|
DATA_OBJECT_TYPES type;
|
||
|
hr = pT->GetDataClass(lpDataObject, &pItem, &type);
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pItem->Notify(event, arg, param, NULL, pT, type);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(Destroy)(LONG_PTR cookie)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentImpl::Destroy\n"));
|
||
|
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
|
||
|
if (pT->m_spControlbar != NULL)
|
||
|
{
|
||
|
int n = pT->m_toolbarMap.GetSize();
|
||
|
for (int i = 0; i < n; i++)
|
||
|
{
|
||
|
IToolbar* pToolbar = (IToolbar*)pT->m_toolbarMap.GetValueAt(i);
|
||
|
if (pToolbar != NULL)
|
||
|
{
|
||
|
pT->m_spControlbar->Detach(pToolbar);
|
||
|
pToolbar->Release();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
pT->m_toolbarMap.RemoveAll();
|
||
|
|
||
|
m_spConsole->SetHeader(NULL);
|
||
|
m_spConsole.Release();
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(QueryDataObject)(LONG_PTR cookie,
|
||
|
DATA_OBJECT_TYPES type,
|
||
|
LPDATAOBJECT *ppDataObject)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentImpl::QueryDataObject\n"));
|
||
|
|
||
|
ATLASSERT(ppDataObject != NULL);
|
||
|
if (ppDataObject == NULL)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponent::QueryDataObject called with ppDataObject==NULL \n"));
|
||
|
return E_POINTER;
|
||
|
}
|
||
|
|
||
|
if (cookie == NULL)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponent::QueryDataObject called with cookie==NULL \n"));
|
||
|
return E_UNEXPECTED;
|
||
|
}
|
||
|
|
||
|
*ppDataObject = NULL;
|
||
|
|
||
|
CSnapInItem* pItem = (CSnapInItem*) cookie;
|
||
|
return pItem->GetDataObject(ppDataObject, type);
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetResultViewType)(LONG_PTR cookie,
|
||
|
LPOLESTR *ppViewType,
|
||
|
long *pViewOptions)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentImpl::GetResultViewType\n"));
|
||
|
|
||
|
if (cookie == NULL)
|
||
|
{
|
||
|
*ppViewType = NULL;
|
||
|
*pViewOptions = MMC_VIEW_OPTIONS_NONE;
|
||
|
return S_FALSE;
|
||
|
}
|
||
|
|
||
|
CSnapInItem* pItem = (CSnapInItem*)cookie;
|
||
|
return pItem->GetResultViewType(ppViewType, pViewOptions);
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetDisplayInfo)(RESULTDATAITEM *pResultDataItem)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponentImpl::GetDisplayInfo\n"));
|
||
|
|
||
|
ATLASSERT(pResultDataItem != NULL);
|
||
|
if (pResultDataItem == NULL)
|
||
|
{
|
||
|
ATLTRACE(_T("IComponent::GetDisplayInfo called with pResultDataItem==NULL\n"));
|
||
|
return E_POINTER;
|
||
|
}
|
||
|
|
||
|
CSnapInItem* pItem = (CSnapInItem*) pResultDataItem->lParam;
|
||
|
|
||
|
if (pItem == NULL)
|
||
|
{
|
||
|
ATLTRACE(_T("Invalid Item\n"));
|
||
|
return E_UNEXPECTED;
|
||
|
}
|
||
|
return pItem->GetResultPaneInfo(pResultDataItem);
|
||
|
}
|
||
|
|
||
|
STDMETHOD(CompareObjects)( LPDATAOBJECT lpDataObjectA,
|
||
|
LPDATAOBJECT lpDataObjectB)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("IComponentImpl::CompareObjects\n"));
|
||
|
}
|
||
|
|
||
|
CComPtr<IConsole> m_spConsole;
|
||
|
};
|
||
|
|
||
|
template <class T, class D>
|
||
|
class ATL_NO_VTABLE IResultDataCompareImpl : public IResultDataCompare
|
||
|
{
|
||
|
public:
|
||
|
STDMETHOD(Compare)(long lUserParam,
|
||
|
LONG_PTR cookieA,
|
||
|
LONG_PTR cookieB,
|
||
|
int *pnResult)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("IResultDataCompareImpl::Compare"));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
template <class T>
|
||
|
class ATL_NO_VTABLE IExtendContextMenuImpl : public IExtendContextMenu
|
||
|
{
|
||
|
public:
|
||
|
STDMETHOD(AddMenuItems)(LPDATAOBJECT pDataObject,
|
||
|
LPCONTEXTMENUCALLBACK piCallback,
|
||
|
long *pInsertionAllowed)
|
||
|
{
|
||
|
ATLTRACE(_T("IExtendContextMenuImpl::AddMenuItems\n"));
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(pDataObject != NULL);
|
||
|
if (pDataObject == NULL)
|
||
|
ATLTRACE(_T("IExtendContextMenu::AddMenuItems called with pDataObject==NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
CSnapInItem* pItem;
|
||
|
DATA_OBJECT_TYPES type;
|
||
|
hr = pT->GetDataClass(pDataObject, &pItem, &type);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pItem->AddMenuItems(piCallback, pInsertionAllowed, type);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(Command)(long lCommandID,
|
||
|
LPDATAOBJECT pDataObject)
|
||
|
{
|
||
|
ATLTRACE(_T("IExtendContextMenuImpl::Command\n"));
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(pDataObject != NULL);
|
||
|
if (pDataObject == NULL)
|
||
|
ATLTRACE(_T("IExtendContextMenu::Command called with pDataObject==NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
CSnapInItem* pItem;
|
||
|
DATA_OBJECT_TYPES type;
|
||
|
hr = pT->GetDataClass(pDataObject, &pItem, &type);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pItem->Command(lCommandID, type);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
template<class T>
|
||
|
class ATL_NO_VTABLE IExtendPropertySheetImpl : public IExtendPropertySheet
|
||
|
{
|
||
|
public:
|
||
|
STDMETHOD(CreatePropertyPages)(LPPROPERTYSHEETCALLBACK lpProvider,
|
||
|
LONG_PTR handle,
|
||
|
LPDATAOBJECT pDataObject)
|
||
|
{
|
||
|
ATLTRACE(_T("IExtendPropertySheetImpl::CreatePropertyPages\n"));
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(pDataObject != NULL);
|
||
|
if (pDataObject == NULL)
|
||
|
ATLTRACE(_T("IExtendPropertySheetImpl::CreatePropertyPages called with pDataObject==NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
CSnapInItem* pItem;
|
||
|
DATA_OBJECT_TYPES type;
|
||
|
hr = pT->GetDataClass(pDataObject, &pItem, &type);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pItem->CreatePropertyPages(lpProvider, handle, this, type);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(QueryPagesFor)(LPDATAOBJECT pDataObject)
|
||
|
{
|
||
|
ATLTRACE(_T("IExtendPropertySheetImpl::QueryPagesFor\n"));
|
||
|
|
||
|
|
||
|
HRESULT hr = E_POINTER;
|
||
|
|
||
|
ATLASSERT(pDataObject != NULL);
|
||
|
if (pDataObject == NULL)
|
||
|
ATLTRACE(_T("IExtendPropertySheetImpl::QueryPagesFor called with pDataObject==NULL\n"));
|
||
|
else
|
||
|
{
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
CSnapInItem* pItem;
|
||
|
DATA_OBJECT_TYPES type;
|
||
|
hr = pT->GetDataClass(pDataObject, &pItem, &type);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = pItem->QueryPagesFor(type);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
template <class T>
|
||
|
class ATL_NO_VTABLE IExtendControlbarImpl : public IExtendControlbar
|
||
|
{
|
||
|
public:
|
||
|
STDMETHOD(SetControlbar)(LPCONTROLBAR pControlbar)
|
||
|
{
|
||
|
ATLTRACE(_T("IExtendControlbarImpl::SetControlbar\n"));
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
|
||
|
if (pT->m_spControlbar != NULL)
|
||
|
{
|
||
|
int n = pT->m_toolbarMap.GetSize();
|
||
|
for (int i = 0; i < n; i++)
|
||
|
{
|
||
|
IToolbar* pToolbar = (IToolbar*)pT->m_toolbarMap.GetValueAt(i);
|
||
|
if (pToolbar != NULL)
|
||
|
{
|
||
|
pT->m_spControlbar->Detach(pToolbar);
|
||
|
pToolbar->Release();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
pT->m_toolbarMap.RemoveAll();
|
||
|
|
||
|
pT->m_spControlbar = pControlbar;
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(ControlbarNotify)(MMC_NOTIFY_TYPE event,
|
||
|
LPARAM arg,
|
||
|
LPARAM param)
|
||
|
{
|
||
|
ATLTRACE(_T("IExtendControlbarImpl::ControlbarNotify\n"));
|
||
|
|
||
|
CSnapInItem* pItem = NULL;
|
||
|
DATA_OBJECT_TYPES type;
|
||
|
HRESULT hr = S_OK;
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
|
||
|
if (event == MMCN_BTN_CLICK)
|
||
|
hr = pT->GetDataClass((IDataObject*) arg, &pItem, &type);
|
||
|
else if (event == MMCN_SELECT)
|
||
|
hr = pT->GetDataClass((IDataObject*) param, &pItem, &type);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = pItem->ControlbarNotify(pT->m_spControlbar, this, &(pT->m_toolbarMap), event, arg, param, type);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
#define SNAPINMENUID(id) \
|
||
|
public: \
|
||
|
static const UINT GetMenuID() \
|
||
|
{ \
|
||
|
static const UINT IDMENU = id; \
|
||
|
return id; \
|
||
|
}
|
||
|
|
||
|
#define EXT_SNAPINMENUID(id) \
|
||
|
public: \
|
||
|
static const UINT GetMenuID() \
|
||
|
{ \
|
||
|
static const UINT IDMENU = id; \
|
||
|
return id; \
|
||
|
}
|
||
|
|
||
|
#define BEGIN_SNAPINCOMMAND_MAP(theClass, bIsExtension) \
|
||
|
HRESULT ProcessCommand(UINT nID, \
|
||
|
bool& bHandled, \
|
||
|
CSnapInObjectRoot* pObj, \
|
||
|
DATA_OBJECT_TYPES type) \
|
||
|
{ \
|
||
|
bHandled = true; \
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
#define SNAPINCOMMAND_ENTRY(id, func) \
|
||
|
if (id == nID) \
|
||
|
{ \
|
||
|
hr = func(bHandled, pObj); \
|
||
|
if (bHandled) \
|
||
|
return hr; \
|
||
|
}
|
||
|
|
||
|
#define SNAPINCOMMAND_RANGE_ENTRY(id1, id2, func) \
|
||
|
if (id1 >= nID && nID <= id2) \
|
||
|
{ \
|
||
|
hr = func(nID, bHandled, pObj); \
|
||
|
if (bHandled) \
|
||
|
return hr; \
|
||
|
}
|
||
|
|
||
|
#define CHAIN_SNAPINCOMMAND_MAP(theChainClass) \
|
||
|
{ \
|
||
|
hr = theChainClass.ProcessCommand(nID, bHandled, pObj, type); \
|
||
|
if (bHandled) \
|
||
|
return hr; \
|
||
|
}
|
||
|
|
||
|
#define END_SNAPINCOMMAND_MAP() \
|
||
|
return hr; \
|
||
|
}
|
||
|
|
||
|
struct CSnapInToolBarData
|
||
|
{
|
||
|
WORD wVersion;
|
||
|
WORD wWidth;
|
||
|
WORD wHeight;
|
||
|
WORD wItemCount;
|
||
|
//WORD aItems[wItemCount]
|
||
|
|
||
|
WORD* items()
|
||
|
{ return (WORD*)(this+1); }
|
||
|
};
|
||
|
|
||
|
#define RT_TOOLBAR MAKEINTRESOURCE(241)
|
||
|
|
||
|
class CSnapInToolbarInfo
|
||
|
{
|
||
|
public:
|
||
|
~CSnapInToolbarInfo()
|
||
|
{
|
||
|
CleanUp();
|
||
|
}
|
||
|
|
||
|
HRESULT CleanUp()
|
||
|
{
|
||
|
if (m_pStrToolTip)
|
||
|
{
|
||
|
for (UINT i = 0; i < m_nButtonCount; i++)
|
||
|
{
|
||
|
delete m_pStrToolTip[i];
|
||
|
m_pStrToolTip[i] = NULL;
|
||
|
}
|
||
|
delete [] m_pStrToolTip;
|
||
|
m_pStrToolTip = NULL;
|
||
|
}
|
||
|
|
||
|
if (m_pStrButtonText)
|
||
|
{
|
||
|
for (UINT i = 0; i < m_nButtonCount; i++)
|
||
|
{
|
||
|
delete m_pStrButtonText[i];
|
||
|
m_pStrButtonText[i] = NULL;
|
||
|
}
|
||
|
|
||
|
delete [] m_pStrButtonText;
|
||
|
m_pStrButtonText = NULL;
|
||
|
}
|
||
|
if (m_pnButtonID)
|
||
|
{
|
||
|
delete m_pnButtonID;
|
||
|
m_pnButtonID = NULL;
|
||
|
}
|
||
|
|
||
|
m_nButtonCount = 0;
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
TCHAR** m_pStrToolTip;
|
||
|
TCHAR** m_pStrButtonText;
|
||
|
UINT* m_pnButtonID;
|
||
|
UINT m_idToolbar;
|
||
|
UINT m_nButtonCount;
|
||
|
};
|
||
|
|
||
|
#define BEGIN_SNAPINTOOLBARID_MAP(theClass) \
|
||
|
public: \
|
||
|
static CSnapInToolbarInfo* GetToolbarInfo() \
|
||
|
{ \
|
||
|
static CSnapInToolbarInfo m_toolbarInfo[] = \
|
||
|
{
|
||
|
|
||
|
#define SNAPINTOOLBARID_ENTRY(id) \
|
||
|
{ NULL, NULL, NULL, id, 0},
|
||
|
|
||
|
#define END_SNAPINTOOLBARID_MAP() \
|
||
|
{ NULL, NULL, NULL, 0, 0} \
|
||
|
}; \
|
||
|
return m_toolbarInfo; \
|
||
|
}
|
||
|
|
||
|
template <class T, BOOL bIsExtension = FALSE>
|
||
|
class ATL_NO_VTABLE CSnapInItemImpl : public CSnapInItem
|
||
|
{
|
||
|
public:
|
||
|
CSnapInItemImpl()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
virtual ~CSnapInItemImpl()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
|
||
|
STDMETHOD(Notify)( MMC_NOTIFY_TYPE event,
|
||
|
LPARAM arg,
|
||
|
LPARAM param,
|
||
|
IComponentData* pComponentData,
|
||
|
IComponent* pComponent,
|
||
|
DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
ATLASSERT("Override Function in Derived Class");
|
||
|
ATLTRACENOTIMPL(_T("CSnapInItemImpl::Notify"));
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetScopePaneInfo)(SCOPEDATAITEM *pScopeDataItem)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("CSnapInItemImpl::GetScopePaneInfo"));
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetResultViewType)(LPOLESTR *ppViewType,
|
||
|
long *pViewOptions)
|
||
|
{
|
||
|
ATLTRACE(_T("CSnapInItemImpl::GetResultViewType\n"));
|
||
|
*ppViewType = NULL;
|
||
|
*pViewOptions = MMC_VIEW_OPTIONS_NONE;
|
||
|
return S_FALSE;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetResultPaneInfo)(RESULTDATAITEM *pResultDataItem)
|
||
|
{
|
||
|
ATLTRACENOTIMPL(_T("CSnapInItemImpl::GetResultPaneInfo"));
|
||
|
}
|
||
|
|
||
|
STDMETHOD(AddMenuItems)(LPCONTEXTMENUCALLBACK piCallback,
|
||
|
long *pInsertionAllowed,
|
||
|
DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
ATLTRACE(_T("CSnapInItemImpl::AddMenuItems\n"));
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
|
||
|
if (!bIsExtension)
|
||
|
pT->SetMenuInsertionFlags(true, pInsertionAllowed);
|
||
|
|
||
|
UINT menuID = pT->GetMenuID();
|
||
|
if (menuID == 0)
|
||
|
return S_OK;
|
||
|
|
||
|
HMENU hMenu = LoadMenu(_Module.GetResourceInstance(), MAKEINTRESOURCE(menuID));
|
||
|
long insertionID;
|
||
|
if (hMenu)
|
||
|
{
|
||
|
for (int i = 0; 1; i++)
|
||
|
{
|
||
|
HMENU hSubMenu = GetSubMenu(hMenu, i);
|
||
|
if (hSubMenu == NULL)
|
||
|
break;
|
||
|
|
||
|
MENUITEMINFO menuItemInfo;
|
||
|
memset(&menuItemInfo, 0, sizeof(menuItemInfo));
|
||
|
menuItemInfo.cbSize = sizeof(menuItemInfo);
|
||
|
|
||
|
switch (i)
|
||
|
{
|
||
|
case 0:
|
||
|
if (! (*pInsertionAllowed & CCM_INSERTIONALLOWED_TOP) )
|
||
|
continue;
|
||
|
insertionID = CCM_INSERTIONPOINTID_PRIMARY_TOP;
|
||
|
break;
|
||
|
|
||
|
case 1:
|
||
|
if (! (*pInsertionAllowed & CCM_INSERTIONALLOWED_NEW) )
|
||
|
continue;
|
||
|
if (bIsExtension)
|
||
|
insertionID = CCM_INSERTIONPOINTID_3RDPARTY_NEW;
|
||
|
else
|
||
|
insertionID = CCM_INSERTIONPOINTID_PRIMARY_NEW;
|
||
|
break;
|
||
|
|
||
|
case 2:;
|
||
|
if (! (*pInsertionAllowed & CCM_INSERTIONALLOWED_TASK) )
|
||
|
continue;
|
||
|
if (bIsExtension)
|
||
|
insertionID = CCM_INSERTIONPOINTID_3RDPARTY_TASK;
|
||
|
else
|
||
|
insertionID = CCM_INSERTIONPOINTID_PRIMARY_TASK;
|
||
|
break;
|
||
|
case 3:;
|
||
|
if (! (*pInsertionAllowed & CCM_INSERTIONALLOWED_VIEW) )
|
||
|
continue;
|
||
|
insertionID = CCM_INSERTIONPOINTID_PRIMARY_VIEW;
|
||
|
break;
|
||
|
default:
|
||
|
{
|
||
|
insertionID = 0;
|
||
|
continue;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
menuItemInfo.fMask = MIIM_TYPE | MIIM_STATE | MIIM_ID;
|
||
|
menuItemInfo.fType = MFT_STRING;
|
||
|
TCHAR buf[128];
|
||
|
|
||
|
for (int j = 0; 1; j++)
|
||
|
{
|
||
|
menuItemInfo.fMask = MIIM_TYPE | MIIM_STATE | MIIM_ID;
|
||
|
menuItemInfo.fType = MFT_STRING;
|
||
|
menuItemInfo.cch = 128;
|
||
|
menuItemInfo.dwTypeData = buf;
|
||
|
TCHAR strStatusBar[256];
|
||
|
|
||
|
if (!GetMenuItemInfo(hSubMenu, j, TRUE, &menuItemInfo))
|
||
|
break;
|
||
|
if (menuItemInfo.fType != MFT_STRING)
|
||
|
continue;
|
||
|
|
||
|
pT->UpdateMenuState(menuItemInfo.wID, buf, &menuItemInfo.fState);
|
||
|
LoadString(_Module.GetResourceInstance(), menuItemInfo.wID, strStatusBar, 256);
|
||
|
|
||
|
CONTEXTMENUITEM contextMenuItem;
|
||
|
memset(&contextMenuItem, 0, sizeof(contextMenuItem));
|
||
|
contextMenuItem.strName = buf;
|
||
|
contextMenuItem.strStatusBarText = strStatusBar;
|
||
|
contextMenuItem.lCommandID = menuItemInfo.wID;
|
||
|
contextMenuItem.lInsertionPointID = insertionID;
|
||
|
contextMenuItem.fFlags = menuItemInfo.fState;
|
||
|
|
||
|
HRESULT hr;
|
||
|
hr = piCallback->AddItem(&contextMenuItem);
|
||
|
ATLASSERT(SUCCEEDED(hr));
|
||
|
}
|
||
|
}
|
||
|
DestroyMenu(hMenu);
|
||
|
}
|
||
|
|
||
|
if (!bIsExtension)
|
||
|
pT->SetMenuInsertionFlags(true, pInsertionAllowed);
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(Command)(long lCommandID,
|
||
|
DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
ATLTRACE(_T("CSnapInItemImpl::Command\n"));
|
||
|
bool bHandled;
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
return pT->ProcessCommand(lCommandID, bHandled, (CSnapInObjectRoot*) this, type);
|
||
|
}
|
||
|
|
||
|
STDMETHOD(CreatePropertyPages)(LPPROPERTYSHEETCALLBACK lpProvider,
|
||
|
LONG_PTR handle,
|
||
|
IUnknown* pUnk,
|
||
|
DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
ATLASSERT("Override Function in Derived Class");
|
||
|
ATLTRACENOTIMPL(_T("CSnapInItemImpl::CreatePropertyPages"));
|
||
|
}
|
||
|
|
||
|
STDMETHOD(QueryPagesFor)(DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
ATLASSERT("Override Function in Derived Class");
|
||
|
ATLTRACENOTIMPL(_T("CSnapInItemImpl::QueryPagesFor"));
|
||
|
}
|
||
|
|
||
|
STDMETHOD(SetControlbar)(IControlbar *pControlbar,
|
||
|
IExtendControlbar* pExtendControlBar,
|
||
|
CSnapInSimpleMap<UINT, IUnknown*>* pToolbarMap)
|
||
|
{
|
||
|
ATLTRACE(_T("CSnapInItemImpl::SetControlbar\n"));
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
|
||
|
CSnapInToolbarInfo* pInfo = pT->GetToolbarInfo();
|
||
|
if (pInfo == NULL)
|
||
|
return S_OK;
|
||
|
|
||
|
for( ; pInfo->m_idToolbar; pInfo++)
|
||
|
{
|
||
|
IToolbar* p = (IToolbar*) pToolbarMap->Lookup(pInfo->m_idToolbar);
|
||
|
if (p != NULL)
|
||
|
continue;
|
||
|
|
||
|
HBITMAP hBitmap = LoadBitmap(_Module.GetResourceInstance(), MAKEINTRESOURCE(pInfo->m_idToolbar));
|
||
|
if (hBitmap == NULL)
|
||
|
return S_OK;
|
||
|
|
||
|
HRSRC hRsrc = ::FindResource(_Module.GetResourceInstance(), MAKEINTRESOURCE(pInfo->m_idToolbar), RT_TOOLBAR);
|
||
|
if (hRsrc == NULL)
|
||
|
return S_OK;
|
||
|
|
||
|
HGLOBAL hGlobal = LoadResource(_Module.GetResourceInstance(), hRsrc);
|
||
|
if (hGlobal == NULL)
|
||
|
return S_OK;
|
||
|
|
||
|
CSnapInToolBarData* pData = (CSnapInToolBarData*)LockResource(hGlobal);
|
||
|
if (pData == NULL)
|
||
|
return S_OK;
|
||
|
ATLASSERT(pData->wVersion == 1);
|
||
|
ATLASSERT(pData->wWidth == 16);
|
||
|
ATLASSERT(pData->wHeight == 16);
|
||
|
|
||
|
pInfo->m_nButtonCount = pData->wItemCount;
|
||
|
if (pInfo->m_pnButtonID == NULL)
|
||
|
pInfo->m_pnButtonID = new UINT[pInfo->m_nButtonCount];
|
||
|
|
||
|
if (pInfo->m_pnButtonID == NULL)
|
||
|
continue;
|
||
|
|
||
|
MMCBUTTON *pButtons = new MMCBUTTON[pData->wItemCount];
|
||
|
if (pButtons == NULL)
|
||
|
{
|
||
|
delete []pInfo->m_pnButtonID;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (pInfo->m_pStrToolTip == NULL)
|
||
|
{
|
||
|
pInfo->m_pStrToolTip = new TCHAR* [pData->wItemCount];
|
||
|
if (pInfo->m_pStrToolTip)
|
||
|
memset(pInfo->m_pStrToolTip, 0, sizeof(TCHAR*) * pData->wItemCount);
|
||
|
}
|
||
|
|
||
|
if (pInfo->m_pStrToolTip == NULL)
|
||
|
{
|
||
|
delete []pInfo->m_pnButtonID;
|
||
|
delete []pButtons;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
|
||
|
for (int i = 0, j = 0; i < pData->wItemCount; i++)
|
||
|
{
|
||
|
// pInfo->m_pStrToolTip[i] = NULL;
|
||
|
memset(&pButtons[i], 0, sizeof(MMCBUTTON));
|
||
|
pInfo->m_pnButtonID[i] = pButtons[i].idCommand = pData->items()[i];
|
||
|
if (pButtons[i].idCommand)
|
||
|
{
|
||
|
pButtons[i].nBitmap = j++;
|
||
|
// get the statusbar string and allow modification of the button state
|
||
|
TCHAR strStatusBar[512];
|
||
|
LoadString(_Module.GetResourceInstance(), pButtons[i].idCommand, strStatusBar, 512);
|
||
|
|
||
|
if (pInfo->m_pStrToolTip[i] == NULL)
|
||
|
pInfo->m_pStrToolTip[i] = new TCHAR[lstrlen(strStatusBar) + 1];
|
||
|
if (pInfo->m_pStrToolTip[i] == NULL)
|
||
|
continue;
|
||
|
lstrcpy(pInfo->m_pStrToolTip[i], strStatusBar);
|
||
|
pButtons[i].lpTooltipText = pInfo->m_pStrToolTip[i];
|
||
|
pButtons[i].lpButtonText = _T("");
|
||
|
pT->SetToolbarButtonInfo(pButtons[i].idCommand, &pButtons[i].fsState, &pButtons[i].fsType);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pButtons[i].lpTooltipText = _T("");
|
||
|
pButtons[i].lpButtonText = _T("");
|
||
|
pButtons[i].fsType = TBSTYLE_SEP;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
IToolbar* pToolbar;
|
||
|
HRESULT hr = pControlbar->Create(TOOLBAR, pExtendControlBar, reinterpret_cast<LPUNKNOWN*>(&pToolbar));
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
|
||
|
hr = pToolbar->AddBitmap(pData->wItemCount, hBitmap, pData->wWidth, pData->wHeight, RGB(192, 192, 192));
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = pToolbar->AddButtons(pData->wItemCount, pButtons);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
pToolbar->AddRef();
|
||
|
pToolbarMap->Add(pInfo->m_idToolbar, (IUnknown*)pToolbar);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
pToolbar->Release();
|
||
|
delete [] pButtons;
|
||
|
}
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(ControlbarNotify)(IControlbar *pControlbar,
|
||
|
IExtendControlbar *pExtendControlbar,
|
||
|
CSnapInSimpleMap<UINT, IUnknown*>* pToolbarMap,
|
||
|
MMC_NOTIFY_TYPE event,
|
||
|
LPARAM arg,
|
||
|
LPARAM param,
|
||
|
DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
ATLTRACE(_T("CSnapInItemImpl::ControlbarNotify\n"));
|
||
|
T* pT = static_cast<T*>(this);
|
||
|
|
||
|
SetControlbar(pControlbar, pExtendControlbar, pToolbarMap);
|
||
|
|
||
|
CComPtr<IUnknown> spUnknown;
|
||
|
HRESULT hr = pControlbar->QueryInterface(IID_IUnknown, (void**)&spUnknown);
|
||
|
if (FAILED(hr))
|
||
|
return hr;
|
||
|
|
||
|
if(event == MMCN_SELECT)
|
||
|
{
|
||
|
if (pControlbar == NULL)
|
||
|
return S_OK;
|
||
|
|
||
|
BOOL bSelect = (BOOL) HIWORD (arg);
|
||
|
BOOL bScope;
|
||
|
bScope = (BOOL) LOWORD(arg);
|
||
|
|
||
|
CSnapInToolbarInfo* pInfo = pT->GetToolbarInfo();
|
||
|
if (pInfo == NULL)
|
||
|
return S_OK;
|
||
|
|
||
|
for(; pInfo->m_idToolbar; pInfo++)
|
||
|
{
|
||
|
IToolbar* pToolbar = (IToolbar*)pToolbarMap->Lookup(pInfo->m_idToolbar);
|
||
|
if (pToolbar == NULL)
|
||
|
continue;
|
||
|
|
||
|
if (!bSelect)
|
||
|
pControlbar->Detach(pToolbar);
|
||
|
else
|
||
|
{
|
||
|
pControlbar->Attach(TOOLBAR, pToolbar);
|
||
|
for (UINT i = 0; i < pInfo->m_nButtonCount; i++)
|
||
|
{
|
||
|
if (pInfo->m_pnButtonID[i])
|
||
|
{
|
||
|
pToolbar->SetButtonState(pInfo->m_pnButtonID[i],
|
||
|
ENABLED,
|
||
|
pT->UpdateToolbarButton(pInfo->m_pnButtonID[i],
|
||
|
ENABLED));
|
||
|
pToolbar->SetButtonState(pInfo->m_pnButtonID[i],
|
||
|
CHECKED,
|
||
|
pT->UpdateToolbarButton(pInfo->m_pnButtonID[i],
|
||
|
CHECKED));
|
||
|
pToolbar->SetButtonState(pInfo->m_pnButtonID[i],
|
||
|
HIDDEN,
|
||
|
pT->UpdateToolbarButton(pInfo->m_pnButtonID[i],
|
||
|
HIDDEN));
|
||
|
pToolbar->SetButtonState(pInfo->m_pnButtonID[i],
|
||
|
INDETERMINATE,
|
||
|
pT->UpdateToolbarButton(pInfo->m_pnButtonID[i],
|
||
|
INDETERMINATE));
|
||
|
pToolbar->SetButtonState(pInfo->m_pnButtonID[i],
|
||
|
BUTTONPRESSED,
|
||
|
pT->UpdateToolbarButton(pInfo->m_pnButtonID[i],
|
||
|
BUTTONPRESSED));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return S_OK;
|
||
|
}
|
||
|
else if (event == MMCN_BTN_CLICK)
|
||
|
{
|
||
|
bool bHandled;
|
||
|
return pT->ProcessCommand((UINT) param, bHandled, (CSnapInObjectRoot*) this, type);
|
||
|
}
|
||
|
|
||
|
return E_UNEXPECTED;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetScopeData)(SCOPEDATAITEM **pScopeDataItem)
|
||
|
{
|
||
|
if (pScopeDataItem == NULL)
|
||
|
return E_FAIL;
|
||
|
|
||
|
*pScopeDataItem = &m_scopeDataItem;
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetResultData)(RESULTDATAITEM **pResultDataItem)
|
||
|
{
|
||
|
if (pResultDataItem == NULL)
|
||
|
return E_FAIL;
|
||
|
|
||
|
*pResultDataItem = &m_resultDataItem;
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD(GetDataObject)(IDataObject** pDataObj, DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
CComObject<CSnapInDataObjectImpl>* pData;
|
||
|
HRESULT hr = CComObject<CSnapInDataObjectImpl>::CreateInstance(&pData);
|
||
|
if (FAILED(hr))
|
||
|
return hr;
|
||
|
|
||
|
T* pT = static_cast<T*> (this);
|
||
|
pData->m_objectData.m_pItem = pT;
|
||
|
pData->m_objectData.m_type = type;
|
||
|
|
||
|
hr = pData->QueryInterface(IID_IDataObject, (void**)(pDataObj));
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
void UpdateMenuState(UINT id, LPTSTR pBuf, UINT *flags)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
void SetToolbarButtonInfo(UINT id, BYTE *fsState, BYTE *fsType)
|
||
|
{
|
||
|
*fsState = TBSTATE_ENABLED;
|
||
|
*fsType = TBSTYLE_BUTTON;
|
||
|
}
|
||
|
|
||
|
BOOL UpdateToolbarButton(UINT id, BYTE fsState)
|
||
|
{
|
||
|
if (fsState == ENABLED)
|
||
|
return TRUE;
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
HRESULT ProcessCommand(UINT nID,
|
||
|
bool& bHandled,
|
||
|
CSnapInObjectRoot* pObj,
|
||
|
DATA_OBJECT_TYPES type)
|
||
|
{
|
||
|
ATLTRACE(_T("No handler for item with ID %d\n"), nID);
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHOD (FillData)(CLIPFORMAT cf, LPSTREAM pStream)
|
||
|
{
|
||
|
HRESULT hr = DV_E_CLIPFORMAT;
|
||
|
ULONG uWritten;
|
||
|
|
||
|
T* pT = static_cast<T*> (this);
|
||
|
|
||
|
if (cf == m_CCF_NODETYPE)
|
||
|
{
|
||
|
hr = pStream->Write(pT->GetNodeType(), sizeof(GUID), &uWritten);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
if (cf == m_CCF_SZNODETYPE)
|
||
|
{
|
||
|
hr = pStream->Write(pT->GetSZNodeType(), (lstrlen((LPCTSTR)pT->GetSZNodeType()) + 1 )* sizeof(TCHAR), &uWritten);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
if (cf == m_CCF_DISPLAY_NAME)
|
||
|
{
|
||
|
hr = pStream->Write(pT->GetDisplayName(), (lstrlen((LPCTSTR)pT->GetDisplayName()) + 1) * sizeof(TCHAR), &uWritten);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
if (cf == m_CCF_SNAPIN_CLASSID)
|
||
|
{
|
||
|
hr = pStream->Write(pT->GetSnapInCLSID(), sizeof(GUID), &uWritten);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
static CSnapInToolbarInfo* GetToolbarInfo()
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
static const UINT GetMenuID()
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void SetMenuInsertionFlags(bool bBeforeInsertion, long* pInsertionAllowed)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void* GetNodeType()
|
||
|
{
|
||
|
return (void*)T::m_NODETYPE;
|
||
|
}
|
||
|
void* GetSZNodeType()
|
||
|
{
|
||
|
return (void*)T::m_SZNODETYPE;
|
||
|
}
|
||
|
|
||
|
void* GetDisplayName()
|
||
|
{
|
||
|
return (void*)T::m_SZDISPLAY_NAME;
|
||
|
}
|
||
|
|
||
|
void* GetSnapInCLSID()
|
||
|
{
|
||
|
return (void*)T::m_SNAPIN_CLASSID;
|
||
|
}
|
||
|
|
||
|
CComBSTR m_bstrDisplayName;
|
||
|
SCOPEDATAITEM m_scopeDataItem;
|
||
|
RESULTDATAITEM m_resultDataItem;
|
||
|
};
|
||
|
|
||
|
|
||
|
_declspec( selectany ) CLIPFORMAT CSnapInItem::m_CCF_NODETYPE = 0;
|
||
|
_declspec( selectany ) CLIPFORMAT CSnapInItem::m_CCF_SZNODETYPE = 0;
|
||
|
_declspec( selectany ) CLIPFORMAT CSnapInItem::m_CCF_DISPLAY_NAME = 0;
|
||
|
_declspec( selectany ) CLIPFORMAT CSnapInItem::m_CCF_SNAPIN_CLASSID = 0;
|
||
|
_declspec( selectany ) CLIPFORMAT CSnapInItem::m_CCF_SNAPIN_GETOBJECTDATA = 0;
|
||
|
|
||
|
#endif //__ATL_SNAPIN_H__
|