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

            yehao's Blog

            NM_CUSTOMDRAW消息解釋

            NM_CUSTOMDRAW消息解釋
            common control 4.7版本介紹了一個(gè)新的特性叫做Custom Draw,這個(gè)名字顯得模糊不清,讓人有點(diǎn)摸不著頭腦,而且MSDN里也只給出了一些如風(fēng)的解釋和例子,沒有誰告訴你你想知道的,和究竟這個(gè)特性有什么好處。
            Custom draw可以被想象成一個(gè)輕量級(jí)的,容易使用的重繪方法(重繪方法還有幾種,例如Owner Draw等)。這種容易來自于我們只需要處理一個(gè)消息(NM_CUSTOMDRAW),就可以讓W(xué)indows為你干活了,你就不用被逼去處理"重繪過程"中所有的臟活了。
            這篇文章的焦點(diǎn)是如何在一個(gè)LISTCTRL控件上使用Custom Draw消息。究其原因,一部分是因?yàn)槲乙呀?jīng)在我的工作上使用了Custom Draw有一段時(shí)間了,我很熟悉它。另一個(gè)原因是這個(gè)機(jī)制確實(shí)是非常好用,你只需要寫很少量的代碼就可以達(dá)到很好的效果。使用 Custom draw 來對(duì)控件外觀編程甚至可以代替很多的古老方法。
            以下代碼是在WIN98 和VC6 SP2的環(huán)境下寫的,common controls DLL的版本是5.0。我已經(jīng)對(duì)其在WinNT 4上進(jìn)行了測試。系統(tǒng)要運(yùn)行這些代碼,它的common controls DLL的版本必須至少是4.71。但隨著IE4 的發(fā)布,這已經(jīng)不是問題了。(IE會(huì)夾帶著這個(gè)DLL一起發(fā)布)
             
            Custom Draw 基礎(chǔ)     
            我將會(huì)盡我所能把Custom Draw的處理描述清楚,而不是簡單的引用MSDN的文檔。這些例子都需要你的程序有一個(gè)ListCtrl在對(duì)話框上,并且這個(gè)ListCtrl處于Report和多列模式。
             
            Custom Draw 的消息映射入口    
            Custom draw 是一個(gè)類似于回調(diào)的處理過程,Windows在繪制List Ctrl的某個(gè)時(shí)間點(diǎn)上通過 Notification 消息通知你的程序,你可以選擇忽略所有的通知(這樣你就會(huì)看到標(biāo)準(zhǔn)的ListCtrl),或者處理某部分的繪制(實(shí)現(xiàn)簡單的效果),甚至整個(gè)的控件都由你來繪制(就象使用Owner-Drawing一樣)。這個(gè)機(jī)制的真正賣點(diǎn)是:你只需要實(shí)現(xiàn)一些你需要的,其余的可以讓W(xué)indows為你代勞。 
            好了,現(xiàn)在你可以開始為你的ListCtrl添加Custom Draw去做一些個(gè)性化的事情了。你首先要有正確的Comm Ctrl Dll版本,然后Windows會(huì)為你發(fā)送NM_CUSTOMDRAW消息,你只需要添加一個(gè)處理函數(shù)以便開始使用Custom draw。首先添加一個(gè)消息映射,象下面一樣:

            ON_NOTIFY ( NM_CUSTOMDRAW, IDC_MY_LIST, OnCustomdrawMyList )

            處理函數(shù)的原形如下:

            afx_msg void OnCustomdrawMyList ( NMHDR* pNMHDR, LRESULT* pResult );

            這就告訴MFC你要處理從你的ListCtrl控件發(fā)出的WM_NOTIFY消息,ID為IDC_MY_LIST,通知碼為NM_CUSTOMDRAWOnCustomdrawMyList就是你的處理函數(shù)。
             
            如果你有一個(gè)從ClistCtr派生的類,你想為它添加custom draw,你就可以使用ON_NOTIFY_REFLECT來代替。如下:

            ON_NOTIFY_REFLECT ( NM_CUSTOMDRAW, OnCustomdraw )OnCustomdraw的原形和上面的函數(shù)一致,但它是聲明在你的派生類里的。

            Custom draw將控件的繪制分為兩部分:擦除和繪畫。Windows在每部分的開始和結(jié)束都會(huì)發(fā)送NM_CUSTOMDRAW消息。所以總共就有4個(gè)消息。但是實(shí)際上你的程序所收到消息可能就只有1個(gè)或者多于四個(gè),這取決于你想要讓W(xué)INDOWS怎么做。每次發(fā)送消息的時(shí)段被稱作為一個(gè)“繪畫段”。你必須緊緊抓住這個(gè)概念,因?yàn)樗灤┯谡麄€(gè)“重繪”的過程。
             
            所以,你將會(huì)在以下的時(shí)間點(diǎn)收到通知:
             

            l         一個(gè)item被畫之前——“繪畫前”段l         一個(gè)item被畫之后——“繪畫后”段l         一個(gè)item被擦除之前——“擦除前”段l         一個(gè)item被擦除之后——“擦除后”段

             
            并不是所有的消息都是一樣有用的,實(shí)際上,我不需要處理所有的消息,直到這篇文章完成之前,我還沒使用過擦除前和擦除后的消息。所以,不要被這些消息嚇到你。 

            NM_CUSTOMDRAW Messages提供給你的信息:  

            l         NM_CUSTOMDRAW消息將會(huì)給你提供以下的信息:l         ListCtrl的句柄l         ListCtrl的IDl         當(dāng)前的“繪畫段”l         繪畫的DC,讓你可以用它來畫畫l         正在被繪制的控件、item、subitem的RECT值l         正在被繪制的Item的Index值l         正在被繪制的SubItem的Index值l        正被繪制的Item的狀態(tài)值(selected, grayed,等等l         Item的LPARAM值,就是你使用CListCtrl::SetItemData所設(shè)的那個(gè)值

             
            上述所有的信息對(duì)你來說可能都很重要,這取決于你想實(shí)現(xiàn)什么效果,但最經(jīng)常用到的就是“繪畫段”、“繪畫DC”、“Item Index”、“LPARAM”這幾個(gè)值。 

            一個(gè)簡單的例子:    

            好了,經(jīng)過上面的無聊的細(xì)節(jié)之后,我們是時(shí)候來看一些簡單的代碼了。第一個(gè)例子非常的簡單,它只是改變了一下控件中文字的顏色。
             
            處理的代碼如下:
            void CPanel1::OnCustomdrawList ( NMHDR* pNMHDR, LRESULT* pResult )

            {

             NMLVCUSTOMDRAW* pLVCD = reinterpret_cast<NMLVCUSTOMDRAW*>( pNMHDR );

                // Take the default processing unless we set this to something else below.
                *pResult = 0;
                // First thing - check the draw stage. If it's the control's prepaint
                // stage, then tell Windows we want messages for every item.
                if ( CDDS_PREPAINT == pLVCD->nmcd.dwDrawStage )
                    {
                    *pResult = CDRF_NOTIFYITEMDRAW;
                    }
                else if ( CDDS_ITEMPREPAINT == pLVCD->nmcd.dwDrawStage )
                    {
                    // This is the prepaint stage for an item. Here's where we set the
                    // item's text color. Our return value will tell Windows to draw the
                    // item itself, but it will use the new color we set here.
                    // We'll cycle the colors through red, green, and light blue.
                    COLORREF crText;
                    if ( (pLVCD->nmcd.dwItemSpec % 3) == 0 )
                        crText = #ff0000;
                    else if ( (pLVCD->nmcd.dwItemSpec % 3) == 1 )
                        crText = #00ff00;
                    else
                        crText = #8080ff;
                    // Store the color back in the NMLVCUSTOMDRAW struct.
                    pLVCD->clrText = crText;
                    // Tell Windows to paint the control itself.
                    *pResult = CDRF_DODEFAULT;
                    }
            }
            結(jié)果如下,你可以看到行和行間的顏色的交錯(cuò)顯示,多酷,而這只需要兩個(gè)if的判斷就可以做到了。

            有一件事情必須記住,在做任何的繪畫之前,你都要檢查正處身的“繪畫段”,因?yàn)槟愕奶幚砗瘮?shù)會(huì)接收到非常多的消息,而“繪畫段”將決定你代碼的行為。 

            一個(gè)更小的簡單例子:    下面的例子將演示怎么去處理subitem的繪畫(其實(shí)subitem也就是列)ListCtrl控件繪畫前處理NM_CUSTOMDRAW消息。告訴Windows我們想對(duì)每個(gè)Item處理NM_CUSTOMDRAW消息。當(dāng)這些消息中的一個(gè)到來,告訴Windows我們想在每個(gè)SubItem的繪制前處理這個(gè)消息當(dāng)這些消息到達(dá),我們就為每個(gè)SubItem設(shè)置文字和背景的顏色。

            這里需要注意兩件事: 

            l        clrTextBk的顏色只是針對(duì)每一列,在最后一列的右邊那個(gè)區(qū)域顏色也還是和ListCtrl控件的背景顏色一致。l         當(dāng)我重新看文檔的時(shí)候,我注意到有一篇題目是“NM_CUSTOMDRAW(list view)”的文章,它說你可以在最開始的custom draw消息中返回CDRF_NOTIFYSUBITEMDRAW就可以處理SubItem了,而不需要在CDDS_ITEMPREPAINT繪畫段中去指定CDRF_NOTIFYSUBITEMDRAW。但是我試了一下,發(fā)現(xiàn)這種方法并不起作用,你還是需要處理CDDS_ITEMPREPAINT段。

            posted on 2014-03-07 14:27 厚積薄發(fā) 閱讀(7327) 評(píng)論(0)  編輯 收藏 引用 所屬分類: Windows編程

            導(dǎo)航

            <2025年8月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            統(tǒng)計(jì)

            常用鏈接

            留言簿

            隨筆分類

            文章分類

            文章檔案

            搜索

            最新評(píng)論

            久久久久亚洲av综合波多野结衣| 亚洲欧美伊人久久综合一区二区| 久久亚洲精精品中文字幕| 亚洲国产小视频精品久久久三级| 久久99精品免费一区二区| 国产精品久久久久久久人人看| 中文字幕日本人妻久久久免费 | 狠狠色噜噜色狠狠狠综合久久| 久久精品国产精品亚洲精品| 久久精品国产99久久久| 久久久久亚洲爆乳少妇无 | 伊人 久久 精品| 国产午夜福利精品久久2021| 久久国产精品国语对白| 色诱久久久久综合网ywww| 免费精品久久久久久中文字幕| 久久久久AV综合网成人| 色综合久久中文字幕综合网| 青青青国产成人久久111网站| 午夜精品久久久久久中宇| 国产免费久久精品99久久| segui久久国产精品| 国内精品久久久久久久coent| 久久一本综合| 亚洲人成精品久久久久| 三级片免费观看久久| 国产呻吟久久久久久久92| 国产精品视频久久| 性欧美大战久久久久久久久| 久久久久久久免费视频| 国产偷久久久精品专区| | 女人高潮久久久叫人喷水| 亚洲婷婷国产精品电影人久久| 热久久最新网站获取| 国产综合久久久久久鬼色| 久久久久久狠狠丁香| 一本一本久久A久久综合精品| 日韩一区二区久久久久久 | 国产L精品国产亚洲区久久| 少妇无套内谢久久久久|