the order of function and event is vatal:
******************
If you add Ocx function/event, MFC will NOT arrange the right order for you!!!!!!!!!!
you should separate the function group and event group, or the dispid will mixed up and
InvokeHelper will not work
******************
BEGIN_DISPATCH_MAP(CMyOcxCtrl1Ctrl, COleControl)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "AboutBox", DISPID_ABOUTBOX, AboutBox, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "MyIntMtd", dispidMyIntMtd, MyIntMtd, VT_UI4, VTS_R8 VTS_UI4)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "MyVoidFunc", dispidMyVoidFunc, MyVoidFunc, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "MyTestOcxFunc", dispidMyTestOcxFunc, MyTestOcxFunc, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "OnMyEvent1", dispidOnMyEvent1, OnMyEvent1, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "OnMyEvent2", dispidOnMyEvent2, OnMyEvent2, VT_EMPTY, VTS_NONE)
END_DISPATCH_MAP()
-------------------------------------------------
reference:
-------------------------------------------------
COleDispatchImpl::Invoke()
{
...
const AFX_DISPMAP_ENTRY* pEntry = pThis->GetDispEntry(dispid);
...
}
const AFX_DISPMAP_ENTRY* PASCAL CCmdTarget::GetDispEntry(MEMBERID memid)
{
const AFX_DISPMAP* pDerivMap = GetDispatchMap();
const AFX_DISPMAP* pDispMap;
const AFX_DISPMAP_ENTRY* pEntry;
.....
pDispMap = pDerivMap;
#ifdef _AFXDLL
for (;;)
#else
while (pDispMap != NULL)
#endif
{
// find AFX_DISPMAP_ENTRY where (pEntry->lDispID == memid)
pEntry = pDispMap->lpEntries;
while (pEntry->nPropOffset != -1)
{
if (pEntry->lDispID == memid)
return pEntry;
++pEntry;
}
// check base class
#ifdef _AFXDLL
if (pDispMap->pfnGetBaseMap == NULL)
break;
pDispMap = (*pDispMap->pfnGetBaseMap)();
#else
pDispMap = pDispMap->pBaseMap;
#endif
}
}
const AFX_DISPMAP* CCmdTarget::GetDispatchMap() const
{
return &CCmdTarget::dispatchMap; //dispatchMap point to the struct defined by the BEGIN_DISPATCH_MAP
}
******************
If you add Ocx function/event, MFC will NOT arrange the right order for you!!!!!!!!!!
you should separate the function group and event group, or the dispid will mixed up and
InvokeHelper will not work
******************
BEGIN_DISPATCH_MAP(CMyOcxCtrl1Ctrl, COleControl)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "AboutBox", DISPID_ABOUTBOX, AboutBox, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "MyIntMtd", dispidMyIntMtd, MyIntMtd, VT_UI4, VTS_R8 VTS_UI4)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "MyVoidFunc", dispidMyVoidFunc, MyVoidFunc, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "MyTestOcxFunc", dispidMyTestOcxFunc, MyTestOcxFunc, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "OnMyEvent1", dispidOnMyEvent1, OnMyEvent1, VT_EMPTY, VTS_NONE)
DISP_FUNCTION_ID(CMyOcxCtrl1Ctrl, "OnMyEvent2", dispidOnMyEvent2, OnMyEvent2, VT_EMPTY, VTS_NONE)
END_DISPATCH_MAP()
-------------------------------------------------
reference:
-------------------------------------------------
COleDispatchImpl::Invoke()
{
...
const AFX_DISPMAP_ENTRY* pEntry = pThis->GetDispEntry(dispid);
...
}
const AFX_DISPMAP_ENTRY* PASCAL CCmdTarget::GetDispEntry(MEMBERID memid)
{
const AFX_DISPMAP* pDerivMap = GetDispatchMap();
const AFX_DISPMAP* pDispMap;
const AFX_DISPMAP_ENTRY* pEntry;
.....
pDispMap = pDerivMap;
#ifdef _AFXDLL
for (;;)
#else
while (pDispMap != NULL)
#endif
{
// find AFX_DISPMAP_ENTRY where (pEntry->lDispID == memid)
pEntry = pDispMap->lpEntries;
while (pEntry->nPropOffset != -1)
{
if (pEntry->lDispID == memid)
return pEntry;
++pEntry;
}
// check base class
#ifdef _AFXDLL
if (pDispMap->pfnGetBaseMap == NULL)
break;
pDispMap = (*pDispMap->pfnGetBaseMap)();
#else
pDispMap = pDispMap->pBaseMap;
#endif
}
}
const AFX_DISPMAP* CCmdTarget::GetDispatchMap() const
{
return &CCmdTarget::dispatchMap; //dispatchMap point to the struct defined by the BEGIN_DISPATCH_MAP
}
本文强调了在MFC中实现OCX控件时,正确组织函数和事件的重要性,避免 DISP_ID 混淆导致 InvokeHelper 失效。通过具体的 CMyOcxCtrl1Ctrl 示例展示了如何分离函数和事件组。

246

被折叠的 条评论
为什么被折叠?



