青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

我住包子山

this->blog.MoveTo("blog.baozishan.in")

翻譯習(xí)作:Create your own controls - the art of subclassing 子類化,繼承公共控件

Create your own controls - the art of subclassing

By Chris Maunder

An introduction to subclassing the Windows common controls using MFC

Introduction

程序員們可以用許多windows通用控件提供的功能方便的編程,這些控件從listbutton甚至是進(jìn)度條都是可以直接拿來(lái)用的.即便如此,仍然有時(shí)候你所選擇的標(biāo)準(zhǔn)控件并不夠用.歡迎學(xué)習(xí)子類化控件這個(gè)經(jīng)典技法.

子類化一個(gè)窗體控件同子類化一個(gè)C++類并不一樣.子類化一個(gè)控件意味著你要替換這個(gè)窗口的某些默認(rèn)消息處理方法(message handlers),子類化能讓你可以高效的劫持這個(gè)控件讓它按照你要求的行為工作,而不是默認(rèn)Windows的默認(rèn)的行為.子類化幾乎允許你把控件做成你想的那么完美.有兩種子類化類型.實(shí)例對(duì)象實(shí)例化(instance subclassing)和全局實(shí)例化(global subclassing).實(shí)例對(duì)象子類化是當(dāng)你將一個(gè)單個(gè)的窗口實(shí)例作為子類.全局子類化是將某一種某一窗口類型的(CLASSWND,我吃不準(zhǔn))窗口(控件)做成自定義版本.這里只講下前者,實(shí)例對(duì)象實(shí)例化..

很重要的一點(diǎn)要了解繼承自CWnd的對(duì)象同窗口本身(一個(gè)HWND)的區(qū)別.你的CWnd繼承類對(duì)象包含一個(gè)成員變量指向HWND,并且包含HWND處理消息(eg WM_PAINT, WM_MOUSEMOVE)用到的處理調(diào)用函數(shù)?(message pump calls吃不準(zhǔn)這個(gè))

子類化是簡(jiǎn)單的.首先你建立一個(gè)能夠處理你關(guān)注消息的窗口消息類,接著將你想要子類化的存在的窗體的行為用你指定的窗口消息類替換.之后這個(gè)窗口就很神奇了..下面就是一個(gè)對(duì)Button的子類化演示.

A New Class

為了子類化控件,我們要新建一個(gè)消息處理類,分別處理我們有興趣的所有消息,我們都很懶,所以就盡量少處理幾個(gè)消息,就處理我們常用的幾個(gè)就好,最好的建立消息處理類的方法是直接繼承存在的類 CButton ,這個(gè)類是Button 的默認(rèn)消息處理類.

讓我們假定我們要做寫古怪的事,比如在每次鼠標(biāo)落在按鈕時(shí)后讓按鈕變成亮黃色.第一件事用ClassWizard建立CButton繼承類CMyButton.

clip_image001

MFC里繼承CButton有很多好處,最大好處就是不用再寫多一行來(lái)實(shí)現(xiàn)原有的默認(rèn)消息處理.如果我們?cè)敢馕覀兛梢灾苯舆M(jìn)行下一步,用這個(gè)CMyButton子類化一個(gè)按鈕實(shí)例,這個(gè)按鈕實(shí)例已經(jīng)是一個(gè)功能完善的控件了,只是有點(diǎn)無(wú)趣(廢話),button control 而已.因?yàn)?/span>MFC實(shí)現(xiàn)了所有的默認(rèn)消息處理,所以我們可以僅僅重載幾個(gè)我們有興趣的消息處理,忽略其余的.

然而在這個(gè)例子中我們還是將其做成自己喜歡的怪異按鈕吧.

判斷鼠標(biāo)是否在按鈕上需要一個(gè)bool類型變量m_bOverControl. TRUE代表鼠標(biāo)在按鈕上,這個(gè)檢查是定期的(使用定時(shí)器timer).不幸的是對(duì)于我們來(lái)說(shuō)沒有OnMouseEnter OnMouseLeave 函數(shù)能夠跨越平臺(tái)使用(?不解?can be used across platforms),所以我們使用OnMouseMove做這件事. 當(dāng)計(jì)時(shí)器激發(fā)時(shí),我們發(fā)現(xiàn)鼠標(biāo)不在按鈕上時(shí)我們讓計(jì)時(shí)器失效,重繪空間.

使用ClassWizard添加WM_MOUSEMOVE WM_TIMER消息處理,他們分別對(duì)應(yīng) OnMouseMove OnTimer.

clip_image002

ClassWizard會(huì)添加如下代碼:

BEGIN_MESSAGE_MAP(CMyButton, CButton)

    //{{AFX_MSG_MAP(CMyButton)

    ON_WM_MOUSEMOVE()

    ON_WM_TIMER()

    //}}AFX_MSG_MAP

END_MESSAGE_MAP()

/////////////////////////////////////////////////////////////////////////////

// CMyButton message handlers

void CMyButton::OnMouseMove(UINT nFlags, CPoint point)

{

    // TODO: Add your message handler code here and/or call default

    CButton::OnMouseMove(nFlags, point);

}

void CMyButton::OnTimer(UINT nIDEvent)

{

    // TODO: Add your message handler code here and/or call default

    CButton::OnTimer(nIDEvent);

}

消息表內(nèi)容( BEGIN_MESSAGE_MAP 段中) 將消息與函數(shù)一一映射. ON_WM_MOUSEMOVE 映射了 WM_MOUSEMOVE消息 => OnMouseMove(..),  ON_WM_TIMER 映射 WM_TIMER消息 => OnTimer(..). 這兩個(gè)宏在MFC里很常用,有興趣可以看看.

假設(shè)我們定義了兩個(gè)變BOOL m_bOverControl , UINT m_nTimerID, 在構(gòu)造函數(shù)初始化好,我們的消息處理如下:

void CMyButton::OnMouseMove(UINT nFlags, CPoint point)

{

    if (!m_bOverControl)                    // Cursor has just moved over control

    {

        TRACE0("Entering control\n");

        m_bOverControl = TRUE;              // Set flag telling us the mouse is in

        Invalidate();                       // Force a redraw

        SetTimer(m_nTimerID, 100, NULL);    // Keep checking back every 1/10 sec

    }

    CButton::OnMouseMove(nFlags, point);    // drop through to default handler

}

void CMyButton::OnTimer(UINT nIDEvent)

{

    // Where is the mouse?

    CPoint p(GetMessagePos());

    ScreenToClient(&p);

    // Get the bounds of the control (just the client area)

    CRect rect;

    GetClientRect(rect);

    // Check the mouse is inside the control

    if (!rect.PtInRect(p))

    {

        TRACE0("Leaving control\n");

        // if not then stop looking...

        m_bOverControl = FALSE;

        KillTimer(m_nTimerID);

        // ...and redraw the control

        Invalidate();

    }

    // drop through to default handler

    CButton::OnTimer(nIDEvent);

}

我們的工作最后一塊就是繪制,繪制不需處理消息,而是重載CWnd::DrawItem這個(gè)虛函數(shù).只有自繪控件才會(huì)調(diào)用這個(gè)函數(shù),它沒有默認(rèn)的實(shí)現(xiàn)(it ASSERT's if you try).這個(gè)函數(shù)只用來(lái)對(duì)于控件類的重載.

clip_image003

使用ClassWizard添加DrawItem重載,修改函數(shù)如下:

void CMyButton::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)

{

    CDC* pDC   = CDC::FromHandle(lpDrawItemStruct->hDC);

    CRect rect = lpDrawItemStruct->rcItem;

    UINT state = lpDrawItemStruct->itemState;

    CString strText;

    GetWindowText(strText);

    // draw the control edges (DrawFrameControl is handy!)

    if (state & ODS_SELECTED)

        pDC->DrawFrameControl(rect, DFC_BUTTON, DFCS_BUTTONPUSH | DFCS_PUSHED);

    else

        pDC->DrawFrameControl(rect, DFC_BUTTON, DFCS_BUTTONPUSH);

    // Deflate the drawing rect by the size of the button's edges

    rect.DeflateRect( CSize(GetSystemMetrics(SM_CXEDGE), GetSystemMetrics(SM_CYEDGE)));

    // Fill the interior color if necessary

    if (m_bOverControl)

        pDC->FillSolidRect(rect, RGB(255, 255, 0)); // yellow

    // Draw the text

    if (!strText.IsEmpty())

    {

        CSize Extent = pDC->GetTextExtent(strText);

        CPoint pt( rect.CenterPoint().x - Extent.cx/2,

        rect.CenterPoint().y - Extent.cy/2 );

        if (state & ODS_SELECTED)

            pt.Offset(1,1);

        int nMode = pDC->SetBkMode(TRANSPARENT);

 

        if (state & ODS_DISABLED)

            pDC->DrawState(pt, Extent, strText, DSS_DISABLED, TRUE, 0, (HBRUSH)NULL);

        else

            pDC->TextOut(pt.x, pt.y, strText);

        pDC->SetBkMode(nMode);

    }

}

這個(gè)類的各部分就差不多講完了,但是這里要注意一點(diǎn)點(diǎn). DrawItem 函數(shù)需要控件設(shè)置自繪屬性,在資源編輯界面為控件填上自繪屬性.不過(guò)還有一種更好的方法讓這個(gè)子類自動(dòng)設(shè)置它的窗體風(fēng)格,打開自繪屬性.要這樣做我們需要重載這最后的函數(shù)PreSubclassWindow.

這個(gè)函數(shù)叫做SubclassWindow, 當(dāng)調(diào)用CWnd::Create DDX_Control后都會(huì)被調(diào)用,也就是說(shuō)如果你建立一個(gè)類的實(shí)例,無(wú)論是動(dòng)態(tài)建立還是用對(duì)話框模板,PreSubclassWindow都會(huì)被調(diào)用. PreSubclassWindow 將會(huì)在你要子類化的窗口建立后,在這個(gè)窗口顯示之前調(diào)用.也就是說(shuō)它是需要對(duì)窗體表現(xiàn)進(jìn)行初始化的一個(gè)最適合的位置.

非常重要的一點(diǎn):如果你使用對(duì)話框資源新建了一個(gè)控件,那么你子類化的控件將不會(huì)得到WM_CREATE 消息, 因此我們不能使用 OnCreate 來(lái)初始化控件, 因?yàn)樗袝r(shí)候不會(huì)被調(diào)用(譯注:但是SubclassWindow都可以).

使用ClassWizard重載PreSubclassWindow 添加如下代碼:

void CMyButton::PreSubclassWindow()

{

    CButton::PreSubclassWindow();

    ModifyStyle(0, BS_OWNERDRAW);    // make the button owner drawn

}

恭喜你你現(xiàn)在有了CMybutton 這個(gè)繼承類!

子類化

使用DDX 在創(chuàng)建過(guò)程中子類化一個(gè)窗體

在這個(gè)例子中,我在Demo對(duì)話框放上了一個(gè)按鈕控件

clip_image004

我讓默認(rèn)的對(duì)話框創(chuàng)建過(guò)程創(chuàng)建這個(gè)控件,然后使用DDX_...將控件關(guān)聯(lián)于自定義的類. 這種方法可以用ClassWizard簡(jiǎn)單的實(shí)現(xiàn):添加一個(gè)成員變量,連接對(duì)應(yīng)的控件ID(這里是IDC_BUTTON),設(shè)定為控件類型變量(Control type),class nameCMyButton我們自定義的類.

clip_image005

ClassWizard添加了一個(gè)DDX_Control調(diào)用在對(duì)話框的DoDataExchange 函數(shù)里. DDX_Control 調(diào)用SubclassWindow函數(shù)將CMyButton與控件關(guān)聯(lián),于是有了自定義消息處理的效果.這個(gè)按鈕算是已經(jīng)被劫持了,現(xiàn)在它的行為就跟我們想要的一樣了.

使用ClassWizard子類化窗體使用一個(gè)不識(shí)別的類

如果你在工程中添加了一個(gè)窗體類并且想要子類化一個(gè)窗體,但是ClassWizard添加變量時(shí)里面沒有顯示新的類型,你需要重新build Classwizard 文件. (這個(gè)是VC6BUG)

備份.clw,刪除原來(lái)的文件,回去原來(lái)的界面Ctrl+W.添加工程需要包括的文件,這樣應(yīng)該就行了 不行的話只有將控件關(guān)聯(lián)的基類名稱替換成自定義類名稱了..

子類化一個(gè)存在的窗體

使用DDX 是簡(jiǎn)單的,但是如果我們需要子類化一個(gè)已經(jīng)存在的空間就沒有用了. 比如,如果你想要子類化一個(gè)在combobox中的Edit控件.你需要在你能子類化Edit窗口之前先擁有combobox(combobox包含一個(gè)Edit控件)

在這種情況下你需要用到 SubclassDlgItem SubclassWindow 函數(shù).這兩個(gè)函數(shù)允許你動(dòng)態(tài)子類化窗體,換句話說(shuō),將已存在的某窗體關(guān)聯(lián)于你的自定義窗體類.

舉例來(lái)說(shuō),假設(shè)我們有一個(gè)對(duì)話框包括一個(gè)按鈕IDC_BUTTON1. 這個(gè)按鈕已經(jīng)創(chuàng)建了.我們想把它關(guān)聯(lián)于CMyButton ,讓它表現(xiàn)出我們想要的行為.

要這樣做,我們需要有一個(gè)新類型的對(duì)象,最好有一個(gè)對(duì)話框或視圖的成員函數(shù).

CMyButton m_btnMyButton;

接著在你的 OnInitDialog(或其它適合的初始化函數(shù)) 中調(diào)用

m_btnMyButton.SubclassDlgItem(IDC_BUTTON1, this);

如果你已經(jīng)有了你想要子類化的窗體的指針,或者你在動(dòng)態(tài)創(chuàng)建控件的CView或其他CWnd子類下使用子類化,或者你不想用SubclassDlgItem,你可以簡(jiǎn)單的調(diào)用:

CWnd* pWnd = GetDlgItem(IDC_BUTTON1); // or use some other method to get

                                      // a pointer to the window you wish

                                      // to subclass

ASSERT( pWnd && pWnd->GetSafeHwnd() );

m_btnMyButton.SubclassWindow(pWnd->GetSafeHwnd());

 

這個(gè)按鈕繪制很簡(jiǎn)單, 但是也很經(jīng)典了..是個(gè)基礎(chǔ)吧.程序編譯運(yùn)行后就是如圖:

clip_image006

注意我們只是重載了繪制函數(shù),使用獲取鼠標(biāo)狀態(tài)函數(shù).這意味著你的控件仍然是一個(gè)按鈕..為你的對(duì)話框類添加一個(gè)click的消息處理,你能看到它一樣會(huì)被調(diào)用.

gohan 2008.1.24

23:59

Conclusion

Subclassing is not hard - you just need to choose the class you wish to subclass carefully, and be aware of what messages you need to handle. Read up on the control you are subclassing - learn about the messages it handles and also the virtual member functions of its implementation class. Once you've hooked into a control and taken over it's inner workings the sky's the limit.

History

26 Oct 2001 - added info in SubclassWindow and SubclassDlgItem

License

This article is licensed under The Code Project Open License (CPOL)

About the Author

Chris Maunder

clip_image007


Sitebuilder, Editor, Staff, Admin

Chris is the Co-founder, Administrator, Architect, Chief Editor and Shameless Hack who wrote and runs CodeProject. He's been programming since 1988 while pretending to be, in various guises, an astrophysicist, mathematician, physicist, hydrologist, geomorphologist, defence intelligence researcher and then, when all that got a bit rough on the nerves, a web developer. He is a Microsoft Visual C++ MVP both globally and for Canada locally.

His programming experience includes C/C++, C#, MFC, ASP, ASP.NET, and far, far too much FORTRAN. He has worked on PocketPCs, AIX mainframes, Sun workstations, and a CRAY YMP C90 behemoth but finds notebooks take up less desk space.

He dodges, he weaves, and he never gets enough sleep. He is kind to small animals.

Chris was born and bred in Australia but splits his time between Toronto and Melbourne, depending on the weather. For relaxation he is into road cycling, snowboarding, rock climbing, and storm chasing.

Occupation:

Founder

Company:

The Code Project

Location:

clip_image008Canada

 

 

posted on 2008-01-25 00:02 Gohan 閱讀(1503) 評(píng)論(0)  編輯 收藏 引用 所屬分類: MFC/SDK

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美一区二区三区视频在线| 亚洲一区二三| 在线色欧美三级视频| 伊人久久婷婷色综合98网| 黄色在线一区| 91久久国产综合久久| 亚洲精品在线免费| 亚洲视频视频在线| 午夜精品免费| 久久精品成人| 免费成人黄色片| 亚洲黑丝在线| 一本一本久久a久久精品综合妖精 一本一本久久a久久精品综合麻豆 | 在线成人小视频| 亚洲二区在线视频| 亚洲伦理精品| 亚洲欧美区自拍先锋| 久久精品首页| 欧美刺激性大交免费视频| 亚洲品质自拍| 亚洲免费在线视频| 久久久久久色| 欧美日韩大片一区二区三区| 国产精品www.| 国产亚洲欧美色| 亚洲国语精品自产拍在线观看| 亚洲另类自拍| 欧美在线黄色| 欧美电影在线观看完整版| 亚洲美女少妇无套啪啪呻吟| 亚洲在线观看视频| 久久综合伊人77777尤物| 欧美日本中文字幕| 国产综合视频| 一区二区三区欧美亚洲| 久久久久九九九九| 亚洲精品一区二区在线| 欧美一级欧美一级在线播放| 欧美 日韩 国产精品免费观看| 欧美四级电影网站| 国产主播一区二区三区| 99国产精品久久久久老师| 欧美一区二区黄色| 亚洲激情自拍| 性久久久久久| 欧美日韩少妇| 影音先锋日韩有码| 亚洲欧美成人综合| 欧美国产综合视频| 性18欧美另类| 欧美视频在线观看免费| 在线精品视频一区二区| 午夜精品一区二区在线观看| 欧美黄污视频| 欧美专区亚洲专区| 国产精品白丝jk黑袜喷水| 亚洲国产欧美日韩| 久久精品女人的天堂av| 日韩午夜精品视频| 蜜臀久久99精品久久久画质超高清 | 中文网丁香综合网| 欧美国产欧美亚洲国产日韩mv天天看完整 | 欧美一区三区三区高中清蜜桃| 亚洲国产成人精品久久久国产成人一区 | 亚洲午夜精品福利| 欧美高清免费| 欧美在线视频在线播放完整版免费观看 | 性欧美1819性猛交| 91久久国产自产拍夜夜嗨| 午夜在线视频观看日韩17c| 欧美日韩一区二区在线| 亚洲黄色成人久久久| 久久久久久国产精品mv| 亚洲专区欧美专区| 欧美日韩一区二区三区免费看| 亚洲国产高清一区二区三区| 久久久国产精品亚洲一区| 国产精品99久久久久久宅男 | 亚洲精品免费一区二区三区| 蜜臀av国产精品久久久久| 亚洲免费视频一区二区| 欧美午夜视频在线观看| 在线视频欧美日韩| 亚洲精品日韩在线观看| 欧美国产日本高清在线| 最新国产成人在线观看| 免费高清在线一区| 久久精品亚洲一区| 韩国成人精品a∨在线观看| 久久精品国产欧美激情| 欧美一区二区啪啪| 国产亚洲精品bt天堂精选| 欧美在线首页| 性高湖久久久久久久久| 国产一区二区三区在线观看免费| 欧美一区二区三区免费大片| 亚洲宅男天堂在线观看无病毒| 欧美性猛交99久久久久99按摩 | 一区二区三区国产精品| 欧美少妇一区二区| 亚洲小视频在线观看| 亚洲最新在线视频| 国产精品福利在线观看网址| 亚洲欧美日韩精品久久久久| 亚洲一区二区三区在线看| 国产精品嫩草影院av蜜臀| 欧美一区日韩一区| 欧美在线一二三区| 亚洲国产精品成人精品| 欧美国产国产综合| 欧美精品啪啪| 亚洲自拍偷拍网址| 欧美亚洲一区二区三区| 黄色在线一区| 亚洲国产欧美一区二区三区同亚洲 | 亚洲视频精品| 亚洲尤物在线| 黄色成人精品网站| 亚洲国产高清在线| 欧美日韩一区二区三区| 欧美在线影院| 久久一区二区精品| 99精品视频免费观看视频| 中国成人黄色视屏| 韩国三级在线一区| 欧美激情一区二区三区在线| 欧美日韩一级黄| 久久av免费一区| 鲁大师成人一区二区三区| 一区二区三区www| 午夜欧美大片免费观看| 1024成人| 一区二区日韩| 精品成人免费| 99国产精品久久久久久久成人热 | 亚洲一区一卡| 在线看一区二区| 亚洲三级毛片| 国内视频一区| 日韩视频第一页| 韩国成人理伦片免费播放| 91久久久久久久久| 国产午夜精品视频| 亚洲高清在线播放| 国产日韩在线播放| 亚洲精品乱码久久久久久黑人| 国产精品资源| 亚洲人成网站在线观看播放| 国产农村妇女毛片精品久久麻豆| 欧美激情一级片一区二区| 国产精品成人在线| 欧美国产一区二区三区激情无套| 国产精品av免费在线观看 | 久久精品国产v日韩v亚洲| 夜夜嗨av一区二区三区中文字幕| 午夜伦理片一区| 这里只有视频精品| 麻豆成人综合网| 欧美在线不卡视频| 欧美日韩精品一区视频| 免费国产自线拍一欧美视频| 国产精品国产自产拍高清av王其 | 日韩亚洲欧美成人| 精品动漫3d一区二区三区| 亚洲视频免费看| 亚洲每日在线| 久久婷婷国产综合精品青草| 午夜久久电影网| 欧美日韩亚洲视频| 亚洲国产91精品在线观看| 国产午夜精品一区理论片飘花| 99精品国产在热久久| 亚洲国产欧美一区二区三区同亚洲| 亚洲欧美日韩在线播放| 宅男在线国产精品| 欧美电影资源| 欧美激情一区二区三区在线| 国内精品伊人久久久久av一坑| 亚洲性感激情| 亚洲一区二区三区涩| 欧美久久久久久| 亚洲大片精品永久免费| 亚洲成人影音| 久久久久一区二区| 久久精品日产第一区二区| 国产精品日韩在线| 在线综合亚洲欧美在线视频| 9久草视频在线视频精品| 欧美v日韩v国产v| 欧美国产免费| 亚洲激情校园春色| 久久综合给合| 欧美国产大片| 91久久精品一区| 欧美大片网址| 亚洲肉体裸体xxxx137| 99re热这里只有精品视频| 欧美精品18| 日韩网站在线看片你懂的| 在线亚洲一区|