• <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>
            萬星星@豌豆莢 歡迎加入我們
            一個吃軟飯的男人!!!!!我只想寫程序####
            微博:http://weibo.com/wanlianwen
            posts - 172,  comments - 1253,  trackbacks - 0

            一個軟件專業(yè)與否,在細微之處體現(xiàn)的淋漓精致。目前軟件開發(fā)基于組件思想,使得軟件開發(fā)像搭積木一樣。軟件模塊的封裝分兩種,一種是業(yè)務功能的封裝(我稱它為組件),一種是表現(xiàn)界面的封裝(我稱它為控件)。

            組件的封裝因為與界面無關,所以問題大多在接口數(shù)據(jù)類型上。控件與界面有關,很多朋友在開發(fā)控件的時候很有激情,看著自己的東東在什么環(huán)境下面都可以使用,很是有成就感。然而稍有專業(yè)水準的人應該會發(fā)現(xiàn)這不是一件完美的事情,因為microsoft沒有為activex控件處理加速鍵消息處理,以至于導致很多問題:up、down、left、right、home、end、tab、backspace等在自己的控件里面不起作用,編輯框無法刪除字符,tab鍵無法跳轉(zhuǎn)到下一個控件上。我是深受其害,以至于有一年多不敢染指控件開發(fā)。這個問題很多人遇見過,但是沒見到誰給出個好的方法來。

            對于上面的問題,微軟也意識到,但是一直沒有給出完美的一套方案,只是在msdn里面零星提到一下,給出了一個基于mfc activex的解決辦法。由于我一般使用atl開發(fā)控件,所以那個辦法我一直未曾感到滿意過。上網(wǎng)搜索不少,一般都不能解決實質(zhì)性問題,有甚者把鉤子都搬出來,個人覺得沒有那么大必要,畢竟我一般不敢將hook技術應用在大軟件里面。我依據(jù)個人摸索,基本解決了大部分問題,當然任何事情都是不完美的,拿出來為了一個csdn朋友需要,也為了大家提出更好的解決辦法。

            本篇以vs2005環(huán)境,實做出atl標準控件、復合控件,測試環(huán)境分別為VB6? VC6(dialog) .net(C#) IE。分別展示出問題以及解決辦法,記憶里vb控件在mfc里面也有問題(好像tab鍵出不去,需要一個隱藏控件),對于此本文不做探討,如有需要,下次在說。

            這里先談談做控件的選擇:標準控件分無窗口和有窗口。
            如果只設計圖形操作,不需要窗口,可以通過IOleObject來解決位置以及大小,這樣可以減少不少控件的尺寸。
            對于有窗口的控件,一般情況是一個做好的窗口類,為了封裝以使用于各種開發(fā)環(huán)境而依附到atl控件窗口上面(至于不創(chuàng)建窗口可否?當然可以,這里不想講,牽扯太遠)。對于這種控件個人認為最好只依附一個窗口,如果想依附幾個窗口,請使用復合控件。
            復合控件類似一個對話框面板,你可以托放控件到面板上,這種控件問題比較少,我推薦使用窗口控件的人選擇此類型,盡管dll尺寸可能大點。

            標準帶窗口控件問題:
            我制作的控件是一個edit創(chuàng)建在atl窗口上面,不作任何處理在各種環(huán)境里面使用情況如下:
            VB6:Tab正常,但是在atl控件的時候,edit無Caret(不知道怎么翻譯)閃爍,并且鼠標點擊在里面后有Caret,然而按下左鍵跑到上一個tab窗口上面去了。
            VC6:比VB6好一點,左右鍵是好的,也是tab后無Caret閃爍
            .net:和VB6情況一樣,無語,難怪.net和VB6一樣好用(^_^,沒有瞧不起.net,我深受VC之苦,好想用.net開發(fā)界面)
            IE:和VB6大致一樣就是左右鍵全部不起作用。

            處理方法:
            1、添加SetFocuse消息處理:

            LRESULT?CATLFullCtrl_Step2::OnSetFocus(UINT?uMsg,?WPARAM?wParam,?LPARAM?lParam,?BOOL & ?bHandled)
            {
            ????
            // ?TODO:?在此添加消息處理程序代碼和/或調(diào)用默認值
            ????LRESULT?lRes? = ?CComControl < CATLFullCtrl_Step2 > ::OnSetFocus(uMsg,?wParam,?lParam,?bHandled);
            ????
            if ?(m_bInPlaceActive)
            ????
            {
            ????????DoVerbUIActivate(
            & m_rcPos,??NULL);
            ????????
            if ( ! IsChild(::GetFocus()))
            ????????
            {
            ????????????m_wndEdit.SetFocus();
            ????????}

            ????}

            ????
            return ?lRes;
            }

            當atl控件接到focuse消息后,把焦點給edit窗口。還不夠,接著是主要的:
            2、重載加速鍵需函數(shù):
            讓我們先看看微軟做了什么:
            BOOL?PreTranslateAccelerator(LPMSG?/*pMsg*/,?HRESULT&?/*hRet*/)
            ????{
            ????????return?FALSE;
            ????}
            靠,明顯不作為嘛!什么都不處理,怎么可能正確啊......
            我來試一試:
            BOOL?CATLFullCtrl_Step2::PreTranslateAccelerator(LPMSG?pMsg,?HRESULT&?hRet)
            {
            ????//?TODO:?在此添加專用代碼和/或調(diào)用基類
            ????if(pMsg->message?==?WM_KEYDOWN?&&
            ????????(pMsg->wParam?==?VK_LEFT?||
            ????????pMsg->wParam?==?VK_RIGHT?||
            ????????pMsg->wParam?==?VK_UP?||
            ????????pMsg->wParam?==?VK_DOWN))
            ????{
            ????????hRet?=?S_FALSE;
            ????????return?TRUE;
            ????}
            ????return?__super::PreTranslateAccelerator(pMsg,?hRet);
            }
            為什么??還記得前面的問題嘛??Tab鍵好像是好的,就是左右鍵不正確啊,^_^,這里我假設處理拉,你就別跳別處了好嘛???
            下面來看看各種環(huán)境里面使用情況:
            VB6:正常
            VC6:正常
            .net:正常
            IE:第一此tab到控件焦點控件外框,點擊編輯框之后下次就是好的了。不曉得是不是我沒有處理好。

            復合控件的問題:
            我制作的復合控件是一個按鈕、一個復選框、一個編輯框、一個單選框。不作任何處理在各種環(huán)境里面使用情況如下:
            VB6:沒大毛病,就是默認焦點在復合控件第一個tab窗口上(假象,你切換一下窗口就恢復到正常情況下,但是復合控件button里面按鈕的默認黑色外框還有,我實在處理不好),且編輯框輸入漢語是亂碼(好像與Unicode編碼有關)。
            VC6:很完美,還是自產(chǎn)自消比較對路。
            .net:和VB6情況差不多,但是沒有亂碼問題。
            IE:問題不大,你自己看看,實在不好描述,我保證不影響使用。

            處理方法:
            1、亂碼問題,不要問我為什么,我也是瞎蒙的,^_^。不過你要想知道為什么復合控件問題這么少,我建議你看看基類CComCompositeControl的PreTranslateAccelerator實現(xiàn),暈,那么多判斷代碼,還有問題,咳,不說了。
            STDMETHOD(TranslateAccelerator)(LPMSG?pMsg)
            ????{
            ????????if(pMsg->message?==?WM_CHAR)
            ????????{
            ????????????return?S_FALSE;
            ????????}
            ????????HRESULT????????hr????=?NO_ERROR;
            ????????hr?=?IOleInPlaceActiveObjectImpl
            <CATLCompoundCtrl_Step2>::TranslateAccelerator(pMsg);
            ????????if(hr?!=?S_OK)
            ????????{
            ????????????CComQIPtr
            <IOleControlSite,&IID_IOleControlSite>????spCtrlSite(m_spClientSite);
            ????????????if(spCtrlSite)
            ????????????{
            ????????????????hr?=?spCtrlSite->TranslateAccelerator(pMsg,?0);
            ????????????????if(hr?!=?S_OK)
            ????????????????{
            ????????????????????IsDialogMessage(pMsg);
            ????????????????}
            ????????????}
            ????????}
            ????????return?S_OK;
            ????}
            主要解決亂碼的就是那段判斷WM_CHAR消息返回S_FALSE的代碼。下面那段代碼借用microsoft的,呵呵。
            2、VB6那個假象focus處理,這個我是在VB6里面做的,就是強制focus到form第一個tab窗口上,焦點倒是對了,但是復合控件上面那個按鈕的默認黑色外框在tab一個輪回后才正常。處理代碼:
            Private?Sub?Form_Activate()
            ????Command1.SetFocus
            End?Sub
            3、.net以及IE的我沒有解決掉那個假focus,望高人支招。
            下面來看看各種環(huán)境里面使用情況:
            VB6:那個默認按鈕黑色外框沒解決掉,其他ok
            VC6:完美
            .net:假focuse沒有解決掉
            IE:假focuse沒有解決掉

            下面來總結(jié)一下:
            好像沒有一種萬能辦法可以解決掉所有情況下的問題,我嘗試很久總結(jié)出這些東西不知道算不算好的辦法,但是還算是消除了一些關鍵問題,不知道是否使用你遇到的問題。.net控件好像很方便也沒有那么多資源切換啊,加速鍵消息處理問題啊。沒辦法,我們這些it“馬崽”很多事情是不由自己的。

            遺留問題:
            假focuse、默認按鈕黑色外框沒有處理好。

            代碼下載

            說明:包括activex控件、VB6測試、VC6測試、.net測試。
            posted on 2006-11-17 21:40 萬連文 閱讀(2375) 評論(1)  編輯 收藏 引用 所屬分類: ATL

            FeedBack:
            # re: activex控件加速鍵消息處理不完全方案
            2007-01-08 19:45 | Philip
            同感,我做一個看似簡單的按紐控件的加速鍵,也是頭大,最后用hook處理的,為避免窗體中有多個按紐時相互轉(zhuǎn)發(fā)重復消息,就采用指定第一個按紐實例負責轉(zhuǎn)發(fā)消息的古怪方式,而且按紐在vba中還處理不了加速鍵.  回復  更多評論
              
            簡歷下載
            聯(lián)系我

            <2006年7月>
            2526272829301
            2345678
            9101112131415
            16171819202122
            23242526272829
            303112345

            常用鏈接

            留言簿(66)

            隨筆分類

            隨筆檔案

            相冊

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            97久久精品国产精品青草| 久久99精品久久久久久不卡| 亚洲国产综合久久天堂| 日韩精品久久久久久久电影| 久久久久久国产a免费观看不卡| 97久久综合精品久久久综合| 久久99国产乱子伦精品免费| 69国产成人综合久久精品| 国内精品久久久久影院免费| 国产视频久久| 一本色综合网久久| 久久婷婷成人综合色综合| 久久久精品国产sm调教网站| 久久99精品久久久久久不卡| 无码国内精品久久综合88| 久久精品视频网| 亚洲AV乱码久久精品蜜桃| 久久久国产精品| 久久精品国产亚洲av麻豆小说 | 国产2021久久精品| 无码伊人66久久大杳蕉网站谷歌| 亚洲伊人久久大香线蕉苏妲己| 国内精品久久久久久不卡影院| 久久精品国产99国产精品导航 | 久久亚洲私人国产精品| 久久精品国产99久久香蕉| 久久久久亚洲av无码专区喷水| 亚州日韩精品专区久久久| 精品久久久久久亚洲| 日韩人妻无码一区二区三区久久99 | 亚洲国产精品无码久久久蜜芽| 国内精品久久久久久麻豆| 97久久久精品综合88久久| 99久久精品免费看国产一区二区三区 | 久久精品国产免费| 久久99精品国产自在现线小黄鸭 | 亚洲国产精品一区二区三区久久| 久久免费国产精品一区二区| 精品免费久久久久久久| 久久精品夜夜夜夜夜久久| 99国产精品久久|