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


{
????if?(pOccManager?==?NULL)
????????afxOccManager?=?_afxOccManager.GetData();
????else
????????afxOccManager?=?pOccManager;
}解釋一下:如果傳入NULL,MFC自動(dòng)創(chuàng)建默認(rèn)管理器,否則接管用戶定義的管理器,這個(gè)地方是有意思的,我們可以在這個(gè)根部替換管理器從而替換容器對(duì)象的創(chuàng)建,進(jìn)而定制站點(diǎn)。默認(rèn)管理器依靠一個(gè)宏定義(OCCMGR.CPP )::
PROCESS_LOCAL(COccManager,?_afxOccManager)在此不深入,有興趣可以看看。
到這里有了控件管理器,這個(gè)應(yīng)用就算支持ActiveX控件。下面來(lái)看看當(dāng)ActiveX控件創(chuàng)建的時(shí)候發(fā)生了什么。
假設(shè)一個(gè)ActiveX控件創(chuàng)建采用如下形式(歸根到底也應(yīng)該如此)(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);
}看上面實(shí)現(xiàn),由于已經(jīng)有afxOccManager?了,繼續(xù)向下看有
InitControlContainer()函數(shù),控件父窗口初始化容器(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);
}這里看出加入沒(méi)有容器對(duì)象,afxOccManager?則負(fù)責(zé)創(chuàng)建容器(每個(gè)窗體存在唯一容器對(duì)象),并且修改所有祖先窗口支持OLE風(fēng)格。
接著父窗口的容器對(duì)象調(diào)用
CreateControl函數(shù),看看它的實(shí)現(xiàn)過(guò)程(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負(fù)責(zé)為COleControlContainer創(chuàng)建COleControlSite對(duì)象,有COleControlSite對(duì)象創(chuàng)建具體ActiveX控件,由于COleControlSite::
CreateControl代碼較長(zhǎng),此處不贅述,有興趣可以自己看看,是ControlSite將ActiveX控件定位的過(guò)程。
為了便于大家理解,我根據(jù)自己理解繪制一個(gè)創(chuàng)建過(guò)程:

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