• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            萬(wàn)星星@豌豆莢 歡迎加入我們
            一個(gè)吃軟飯的男人!!!!!我只想寫(xiě)程序####
            微博:http://weibo.com/wanlianwen
            posts - 172,  comments - 1253,  trackbacks - 0

            背景知識(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

            FeedBack:
            # re: MFC的容器站點(diǎn)控件模型
            2006-09-18 12:30 | 絕緣電阻測(cè)試儀
            樓主說(shuō)的確實(shí)對(duì)我很有用,多謝了!@_@~~  回復(fù)  更多評(píng)論
              
            # re: MFC的容器站點(diǎn)控件模型
            2007-12-26 09:07 | xuliang
            請(qǐng)教一下,我包裝了ie后,瀏覽一般網(wǎng)頁(yè)沒(méi)問(wèn)題,但是所有網(wǎng)頁(yè)上javascript的按鍵事件都不響應(yīng)了(document.onkeydown),你碰到過(guò)嗎?  回復(fù)  更多評(píng)論
              
            # re: MFC的容器站點(diǎn)控件模型
            2007-12-26 09:49 | 萬(wàn)連文
            我想應(yīng)該不會(huì),這種情況我沒(méi)有遇到過(guò),如果有疑問(wèn),可以將source給我,我有時(shí)間可以看看能不能解決。  回復(fù)  更多評(píng)論
              
            # re: MFC的容器站點(diǎn)控件模型
            2009-10-30 12:56 | 學(xué)習(xí)
            請(qǐng)問(wèn)在 Windows mobile 里面能實(shí)現(xiàn)上述功能嗎???  回復(fù)  更多評(píng)論
              
            簡(jiǎn)歷下載
            聯(lián)系我

            <2007年3月>
            25262728123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            常用鏈接

            留言簿(66)

            隨筆分類

            隨筆檔案

            相冊(cè)

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            91精品国产91久久久久久蜜臀| 国产精品伦理久久久久久| 国产精品久久网| 精品久久久久久中文字幕人妻最新| 国内精品伊人久久久久AV影院| 无码AV中文字幕久久专区| 午夜精品久久久久久久久| 久久99国产精品尤物| 久久经典免费视频| 人妻无码久久精品| 日韩美女18网站久久精品| 久久久精品视频免费观看| 99久久精品无码一区二区毛片| 午夜精品久久影院蜜桃| 亚洲成色www久久网站夜月| 欧美成人免费观看久久| 久久久久久精品免费看SSS| 久久青青草原精品国产| 久久大香香蕉国产| 久久免费精品视频| 国产日韩久久久精品影院首页| 偷窥少妇久久久久久久久| 久久精品中文字幕大胸| 久久影院综合精品| 国产精品一区二区久久精品| a高清免费毛片久久| 91精品国产高清久久久久久91| 18禁黄久久久AAA片| 国产午夜精品久久久久免费视| 久久国产免费直播| 久久久久久亚洲精品成人| 精品久久香蕉国产线看观看亚洲| 色综合久久天天综线观看| 精品乱码久久久久久久| 亚洲AV无码久久精品狠狠爱浪潮| 久久996热精品xxxx| 亚洲精品综合久久| 欧洲精品久久久av无码电影 | 久久久久亚洲av无码专区导航| 国产精品日韩深夜福利久久 | 99热热久久这里只有精品68|