• <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>

            旅途

            如果想飛得高,就該把地平線忘掉

            全面解讀WM_NOTIFY

            摘要:控件通知消息有很多種,但是有一種是很常用,但是又不是很容易掌握的,那就是WM_NOTIFY,我試著對此做一下比較全面的論述,有不對的地方,還希望各路大蝦批評指正。

                控件通知消息
                在《深度解析VC中的消息(上)》中,我們提到了消息的分類有3種:窗口消息、命令消息和控件通知消息,我們這里要談的是最后一種:控件通知消息。
                控件通知消息,是指這樣一種消息,一個窗口內(nèi)的子控件發(fā)生了一些事情,需要通知父窗口。通知消息只適用于標(biāo)準(zhǔn)的窗口控件如按鈕、列表框、組合框、編輯框,以及Windows公共控件如樹狀視圖、列表視圖等。例如,單擊或雙擊一個控件、在控件中選擇部分文本、操作控件的滾動條都會產(chǎn)生通知消息。 她類似于命令消息,當(dāng)用戶與控件窗口交互時,那么控件通知消息就會從控件窗口發(fā)送到它的主窗口。但是這種消息的存在并不是為了處理用戶命令,而是為了讓主窗口能夠改變控件,例如加載、顯示數(shù)據(jù)。例如按下一個按鈕,他向父窗口發(fā)送的消息也可以看作是一個控件通知消息;單擊鼠標(biāo)所產(chǎn)生的消息可以由主窗口直接處理,然后交給控件窗口處理。
                控件通知消息主要由窗口類即直接或間接由CWND類派生類處理。

                控件通知格式
                控件通知經(jīng)歷了一個演變過程,因而SendMessage( )的變量Message、wParam和lParam有三種格式。
                第一控件通知格式
                第一控件通知格式只是窗口消息的子集。它的特征格式如下:WM_XXXX。它主要來自下面的3種消息類型:
                (1)表示一個控件窗口要么已經(jīng)被創(chuàng)建或銷毀,要么已經(jīng)被鼠標(biāo)單擊的消息:WM_PARENTNOTIFY;
                (2)發(fā)送到父窗口,用來繪制自身窗口的消息,例如: WM_CTLCOLOR、WM_DRAWITEM、WM_MEASUREITEM、WM_DELETEITEM、WM_CHARTOITEM、WM_VKTOITEM、WM_COMMAND和WM_COMPAREITEM
                (3)有滾動調(diào)控件發(fā)送,通知父窗口滾動窗口的消息:WM_VSCROLL和WM_HSCROLL
                第二控件通知格式
                第二控件通知格式與命令消息共享,它的特征格式如下:WM_COMMAND。
                在WM_COMMAND中,lParam用來區(qū)分是命令消息還是控件通知消息:如果lParam為NULL,則這是個命令消息,否則lParam里面放的必然就是控件的句柄,是一個控件通知消息。對于wParam則是低位放的是控件ID,高位放的是相應(yīng)的消息事件。
                第三控件通知格式
                這個才真正涉及到我們要講的內(nèi)容,同時他也是最為靈活的一種格式。它的特征格式如下:WM_NOTIFY。
                在WM_NOTIFY中,lParam中放的是一個稱為NMHDR結(jié)構(gòu)的指針。在wParam中放的則是控件的ID。

                NMHDR結(jié)構(gòu)的由來
                NMHDR結(jié)構(gòu)是很值得一提的,該結(jié)構(gòu)包括有關(guān)制作該通知的控件的任何內(nèi)容,而不受空間和類型的限制,他的來歷也是很有意思的。
                在最初的windows3.x中,根本就不存在什么WM_NOTIFY,控件通知它們父窗口,如鼠標(biāo)點(diǎn)擊,控件背景繪制事件,通過發(fā)送一個消息到父窗口。簡單的通知僅發(fā)送一個WM_COMMAND消息,包含一個通知碼和一個在wParam中的控件ID及一個在lPraram中的控件句柄。這樣一來,wParam和lParam就都被填充了,沒有額外的空間來傳遞一些其它的消息,例如鼠標(biāo)按下的位置和時間。
                為了克服這個困難,windows3.x就提出了一個比較低級的解決策略,那就是給一些消息添加一些附加消息,最為明顯的就是控件自畫用到的DRAWITEMSTRUCT。不知道大家對這個結(jié)構(gòu)熟悉不,不過,如果你是老手,你應(yīng)該非常清楚這個結(jié)構(gòu),這個結(jié)構(gòu)包含了9個內(nèi)容,幾乎你需要控制的信息都給你提供了。為什么說它比較低級呢?因?yàn)椴煌南⒏郊拥膬?nèi)容不同,結(jié)果就是一盤散沙,非常混亂。
                在win32中,MS又提出了一個更好的解決方案:引進(jìn)NMHDR結(jié)構(gòu)。這個結(jié)構(gòu)的引進(jìn)就是消息統(tǒng)一起來,利用它可以傳遞復(fù)雜的信息。這個結(jié)構(gòu)的布局如下:
                NMHDR
                {
                    HWnd hWndFrom ; 相當(dāng)于原WM_COMMAND傳遞方式的lParam
                    UINT idFrom ;   相當(dāng)于原WM_COMMAND傳遞方式的wParam(low-order)
                    UINT code ;     相當(dāng)于原WM_COMMAND傳遞方式的Notify Code(wParam"s high-order)
                };
                對于這個結(jié)構(gòu)的應(yīng)用于WM_NOTIFY信息結(jié)構(gòu),結(jié)果WM_NOTIFY就變成了:
                A、無附加信息。結(jié)構(gòu)變得很簡單,就是一個NMHDR結(jié)構(gòu)。
                B、有附加信息。定義一個大的結(jié)構(gòu),它的第一個元素就是NMHDR結(jié)構(gòu),它的后面放置附加信息。
               
                WM_NOTIFY結(jié)構(gòu)的好處
                除了上面我們所說的好處外,WN_NOTIFY還有自己的獨(dú)特的好處:
                由于在大結(jié)構(gòu)中,第一個成員為NMHDR,這樣一來,我們就可以利用指向NMHDR的指針來傳遞結(jié)構(gòu)地址,根據(jù)指針的特性,無論消息有沒有附加信息,這個指針都適用,也能夠很方便的進(jìn)行強(qiáng)制轉(zhuǎn)換。

                分析ON_NOTIFY
              
            類向?qū)Э梢詣?chuàng)建ON_NOTIFY消息映射入口并提供一個處理函數(shù)的框架,來處理 WM_NOTIFY類型的消息。ON_NOTIFY消息映射宏有如下語法.
               ON_NOTIFY(wNotifyCode,id,memberFxn)
              其中:wNotifyCode:要處理的通知消息通知碼。比如上面我們提到的LVN_KEYDOWN;Id:控件標(biāo)識ID;MemberFxn:處理此消息的成員函數(shù)。
              此成員函數(shù)有如下的原型聲明:
               afx_msg void memberFxn( NMHDR * pNotifyStruct, LRESULT * result);
              比如:假設(shè)你想成員函數(shù)OnKeydownList1處理ClistCtrl(標(biāo)識ID=IDC_LIST1)的 LVN_KEYDOWN消息,你可以使用類向?qū)砑尤缦碌南⒂成?
               ON_NOTIFY( LVN_KEYDOWN, IDC_LIST1, OnKeydownList1 )
              在上面的例子中,類向?qū)峁┤缦潞瘮?shù):
               void CMessageReflectionDlg::OnKeydownList1(NMHDR* pNMHDR, LRESULT* pResult)
                {
                 LV_KEYDOWN* pLVKey= (LV_KEYDOWN*)pNMHDR;
                 *pResult = 0;
                 }
              這時類向?qū)峁┝艘粋€適當(dāng)類型的指針,你既可以通過pNMHDR,也可以通過 pLVKey來訪問這個通知結(jié)構(gòu)。

                ON_NOTIFY_RANGE
              
            有時我們可能需要為一組控件處理相同的WM_NOTIFY消息。這時需要使用ON_NOTIFY_RANGE而不是ON_NOTIFY。不過,很不幸的是,VC6的ClassWizard并不支持這個消息,所以我們必須手工添加。方法和一般的手工添加的消息一樣,不過需要注意的是:
                (1)當(dāng)你使用 ON_NOTIFY_RANGE時,你需要指定控件的ID范圍.其消息映射入口及函數(shù)原型如下:
                 ON_NOTIFY_RANGE( wNotifyCode, id, idLast, memberFxn )
                其中:wNotifyCode:消息通知碼.比如:LVN_KEYDOWN。id: 第一控件的標(biāo)識ID。
                 idLast:最后一個控件的標(biāo)識ID。(標(biāo)識值一定要連續(xù))memberFxn: 消息處理函數(shù)。
              (2)成員函數(shù)必須有如下原型申明:afx_msg void memberFxn( UINT id, NMHDR * pNotifyStruct, LRESULT * result );

                結(jié)束語:
                WM_NOTIFY是一個很值得思考的結(jié)構(gòu),另外他和Reflect Message聯(lián)系非常緊密,我們接下來就要討論一下反射消息。

            posted on 2007-07-18 15:36 旅途 閱讀(204) 評論(0)  編輯 收藏 引用 所屬分類: 深入windows

            四虎国产精品成人免费久久| 人人妻久久人人澡人人爽人人精品| 伊色综合久久之综合久久| 精品99久久aaa一级毛片| 欧美激情精品久久久久| 久久亚洲高清观看| 国产成人无码精品久久久久免费| 久久国产精品99精品国产987| 久久国产高潮流白浆免费观看| 久久久久亚洲av无码专区喷水| 国产综合久久久久久鬼色| 精品午夜久久福利大片| 成人精品一区二区久久久| 久久久亚洲精品蜜桃臀| 怡红院日本一道日本久久 | 狠狠色狠狠色综合久久| 久久亚洲AV成人无码软件| 色综合久久综合中文综合网| 免费观看成人久久网免费观看| 精品免费久久久久国产一区| 伊人久久大香线蕉综合5g| av无码久久久久久不卡网站| 精品一久久香蕉国产线看播放| 亚洲另类欧美综合久久图片区| 亚洲国产精品无码久久久秋霞2| 青青青国产成人久久111网站| 欧美久久天天综合香蕉伊| 午夜欧美精品久久久久久久| 伊人久久免费视频| 久久亚洲精品无码VA大香大香| 久久天堂电影网| 97精品伊人久久大香线蕉| 青青草原综合久久大伊人精品| 久久久久免费精品国产| 亚洲午夜久久影院| 一本色道久久综合亚洲精品| 久久精品国产亚洲Aⅴ蜜臀色欲| 色婷婷综合久久久久中文 | 亚洲午夜无码久久久久| 久久精品国产一区| 久久天天躁狠狠躁夜夜avapp|