背景知識
VC向導里面有一個MFC ActiveX項,我們可以使用它來創建ActiveX控件,ActiveX技術是OLE技術的延伸,微軟早期推出OLE技術不是非常成功,于是修改了名字以抹去人們對OLE的陰影。
ActiveX技術在現在應用非常廣泛,它以COM思想為基礎,以MFC技術實現,使得開發人員可以快速創建組件功能模塊應用于Windows平臺任何語言。
我們在使用ActiveX控件的時候(本文限于VC討論)很簡單,直接拖放控件或者通過CWnd::CreateControl創建。如果僅限于此,可以說能夠很好的使用ActiveX控件,然而無法對控件實現定制。為什么要定制?我不想解釋太多,因為有些東西意會而不能言傳。用Spy看一下TT瀏覽器、遨游瀏覽器就可以知道它的內核就是IE的WebBrowser,它們的實現其實就是對WebBrowser的定制。如何才能進行定制?當然必須了解控件創建機制,了解應用與控件之間的模型,了解MFC背后為我們做了什么。
我以前一向很少使用ActiveX控件,原因很簡單,我不了解內部技術,不是不會用,而是不能隨心所欲的用。我一方面希望軟件組件化,一方面又對ActiveX控件如此畏懼,自己內心其實都很矛盾。隨著知識的積累,慢慢終于有能力去了解這些東西,我希望自己可以講明白,也希望有人渴望了解,盡管是老技術。
牽扯到的類匯總
牽扯到ActiveX控件創建的有這些MFC類:COccManager、COleControlContainer、COleControlSite、COleControl。下面分別解釋一下這些類:
COccManager:control container manager控件容器管理器,任何應用程序若要支持AxtiveX控件必須創建該對象,創建位置一般在×××App: InitInstance()函數里面AfxEnableControlContainer()。為容納控件窗體創建COleControlContainer。
COleControlContainer:控件容器類,容納ActiveX控件的窗體都創建該對象,用來創建COleControlSite對象以登錄控件信息。
COleControlSite:COleControlContainer為每個ActiveX控件創建一個COleControlSite對象以登錄控件信息。
COleControl:所有ActiveX控件從其派生。
對象創建流程分析
下面介紹這些對象的創建流程,通過分析可以清楚一個ActiveX控件創建細節:
控件容器管理對象位于應用程序級,如果應用支持ActiveX控件,那么會在應用初始化的時候創建一個管理器,MFC缺省實現:
AfxEnableControlContainer();
void?AFX_CDECL?AfxEnableControlContainer(COccManager*?pOccManager=NULL);
void?AFX_CDECL?AfxEnableControlContainer(COccManager*?pOccManager)

{
????if?(pOccManager?==?NULL)
????????afxOccManager?=?_afxOccManager.GetData();
????else
????????afxOccManager?=?pOccManager;
}
PROCESS_LOCAL(COccManager,?_afxOccManager)到這里有了控件管理器,這個應用就算支持ActiveX控件。下面來看看當ActiveX控件創建的時候發生了什么。
假設一個ActiveX控件創建采用如下形式(歸根到底也應該如此)(OCCCONT.CPP ):
BOOL?CWnd::CreateControl(REFCLSID?clsid,?LPCTSTR?lpszWindowName,?DWORD?dwStyle,
????const?POINT*?ppt,?const?SIZE*?psize,?CWnd*?pParentWnd,?UINT?nID,
???CFile*?pPersist,?BOOL?bStorage,?BSTR?bstrLicKey)

{
????ASSERT(pParentWnd?!=?NULL);
#ifdef?_DEBUG
????if?(afxOccManager?==?NULL)
????
{
????????TRACE0("Warning:?AfxEnableControlContainer?has?not?been?called?yet.\n");
????????TRACE0(">>>?You?should?call?it?in?your?app's?InitInstance?function.\n");
????}
#endif
????if?((pParentWnd?==?NULL)?||?!pParentWnd->InitControlContainer())
????????return?FALSE;
????return?pParentWnd->m_pCtrlCont->CreateControl(this,?clsid,?lpszWindowName,
????????dwStyle,?ppt,?psize,?nID,?pPersist,?bStorage,?bstrLicKey);
}
BOOL?CWnd::InitControlContainer()

{
????TRY
????
{
????????if?(m_pCtrlCont?==?NULL)
????????????m_pCtrlCont?=?afxOccManager->CreateContainer(this);
????}
????END_TRY
????//?Mark?all?ancestor?windows?as?containing?OLE?controls.
????if?(m_pCtrlCont?!=?NULL)
????
{
????????CWnd*?pWnd?=?this;
????????while?((pWnd?!=?NULL)?&&?!(pWnd->m_nFlags?&?WF_OLECTLCONTAINER))
????????
{
????????????pWnd->m_nFlags?|=?WF_OLECTLCONTAINER;
????????????pWnd?=?pWnd->GetParent();
????????????if?(!?(GetWindowLong(pWnd->GetSafeHwnd(),?GWL_STYLE)?&?WS_CHILD))
????????????????break;
????????}
????}
????return?(m_pCtrlCont?!=?NULL);
}接著父窗口的容器對象調用CreateControl函數,看看它的實現過程(OCCSITE.CPP ):
BOOL?COleControlContainer::CreateControl(CWnd*?pWndCtrl,?REFCLSID?clsid,
????LPCTSTR?lpszWindowName,?DWORD?dwStyle,?const?POINT*?ppt,?const?SIZE*?psize,
???UINT?nID,?CFile*?pPersist,?BOOL?bStorage,?BSTR?bstrLicKey,
???COleControlSite**?ppNewSite)

{
????COleControlSite*?pSite?=?NULL;
????TRY
????
{
????????pSite?=?afxOccManager->CreateSite(this);
????}
????END_TRY
????if?(pSite?==?NULL)
????????return?FALSE;
????BOOL?bCreated?=?SUCCEEDED(?pSite->CreateControl(pWndCtrl,?clsid,
????????lpszWindowName,?dwStyle,?ppt,?psize,?nID,?pPersist,?bStorage,
??????bstrLicKey?)?);
????if?(bCreated)
????
{
????????ASSERT(pSite->m_hWnd?!=?NULL);
????????m_siteMap.SetAt(pSite->m_hWnd,?pSite);
????????if?(ppNewSite?!=?NULL)
????????????*ppNewSite?=?pSite;
????}
????else
????
{
????????delete?pSite;
????}
????return?bCreated;
}為了便于大家理解,我根據自己理解繪制一個創建過程:

應用示例
上面介紹了創建流程,只談這些你可能不明白到底有什么好處。這里示例還是我以前的一個例子:
使MFC變漂亮二:MFC與HTML交互示例
WebBrowser控件通過IDocHostUIHandler接口處理UI以及一些交互事件,我們可以定制自己的COleControlSite實現自定義行為,要創建自定義的COleControlSite對象就必須實現自定義的COccManager。因此例子中派生了兩個類,分別實現定制COccManager、COleControlSite,當然你也可以創建自定義的COleControlContainer對象以在創建COleControlSite對象時為其提供某種服務。
不知道講清楚沒有,反正我是又糊涂了,^_^。難得糊涂!如果你想對Office有深入了解、希望應用集成VBA開發,這些知識都是必不可少。OLE技術還是ActiveX技術,我分不清,需要了解的太多,慢慢來。


