背景知識
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();
我們看下它的原型(AFXDISP.H ):
void?AFX_CDECL?AfxEnableControlContainer(COccManager*?pOccManager=NULL);它的實現如下(OCCMGR.CPP ):
void?AFX_CDECL?AfxEnableControlContainer(COccManager*?pOccManager)


{
????if?(pOccManager?==?NULL)
????????afxOccManager?=?_afxOccManager.GetData();
????else
????????afxOccManager?=?pOccManager;
}解釋一下:如果傳入NULL,MFC自動創建默認管理器,否則接管用戶定義的管理器,這個地方是有意思的,我們可以在這個根部替換管理器從而替換容器對象的創建,進而定制站點。默認管理器依靠一個宏定義(OCCMGR.CPP )::
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);
}看上面實現,由于已經有afxOccManager?了,繼續向下看有
InitControlContainer()函數,控件父窗口初始化容器(OCCCONT.CPP ):
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);
}這里看出加入沒有容器對象,afxOccManager?則負責創建容器(每個窗體存在唯一容器對象),并且修改所有祖先窗口支持OLE風格。
接著父窗口的容器對象調用
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;
}afxOccManager負責為COleControlContainer創建COleControlSite對象,有COleControlSite對象創建具體ActiveX控件,由于COleControlSite::
CreateControl代碼較長,此處不贅述,有興趣可以自己看看,是ControlSite將ActiveX控件定位的過程。
為了便于大家理解,我根據自己理解繪制一個創建過程:

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