2022年消息映射机制 .pdf
一,消息映射机制1,消息响应函数:(例:在CDrawView类响应鼠标左键按下消息)1)在头文件 (DrawView.h) 中声明消息响应函数原型。/AFX_MSG(CDrawView) / 注释宏afx_msg void OnLButtonDown(UINT nFlags, CPoint point); /AFX_MSG / 注释宏说明:在注释宏之间的声明在VC 中灰色显示。afx_msg 宏表示声明的是一个消息响应函数。2)在源文件( DrawView.cpp)中进行消息映射。BEGIN_MESSAGE_MAP(CDrawView, CView) /AFX_MSG_MAP(CDrawView) ON_WM_LBUTTONDOWN() /AFX_MSG_MAP / Standard printing commands ON_COMMAND(ID_FILE_PRINT, CView:OnFilePrint) ON_COMMAND(ID_FILE_PRINT_DIRECT, CView:OnFilePrint) ON_COMMAND(ID_FILE_PRINT_PREVIEW, CView:OnFilePrintPreview) END_MESSAGE_MAP() 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 1 页,共 8 页 - - - - - - - - - 说明:在宏 BEGIN_MESSAGE_MAP() 与 END_MESSAGE_MAP() 之间进行消息映射。宏 ON_WM_LBUTTONDOWN()把消息 WM_LBUTTONDOWN与它的响应函数OnLButtonDown()相关联。这样一旦有消息的产生,就会自动调用相关联的消息响应函数去处理。宏 ON_WM_LBUTTONDOWN()定义如下:#define ON_WM_LBUTTONDOWN() WM_LBUTTONDOWN, 0, 0, 0, AfxSig_vwp, (AFX_PMSG)(AFX_PMSGW)(void (AFX_MSG_CALL CWnd:*)(UINT, CPoint)&OnLButtonDown , 3)源文件中进行消息响应函数处理。(DrawView.cpp中自动生成 OnLButtonDown函数轮廓,如下)void CDrawView:OnLButtonDown(UINT nFlags, CPoint point) / TODO: Add your message handler code here and/or call default CView:OnLButtonDown(nFlags, point); 说明:可见当增加一个消息响应处理,在以上三处进行了修改。可在消息响应函数里添加消息处理代码完成对消息的响应、处理。名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 2 页,共 8 页 - - - - - - - - - 2,消息响应的方式:1)在基类中针对每种消息做一个虚函数,当子类对消息响应时候,只要在子类中重写这个虚函数即可。缺点: MFC 类派生层次很多,如果在基类对每个消息进行虚函数处理,那么从基类派生的每个子类都将背负一个庞大的虚表,这样浪费内存,故MFC 没有采取这中方式而采取消息映射方式。2)消息映射方式: MFC 在后台维护了一个句柄和C+ 对象指针对照表,当收到一个消息后,通过消息结构里资源句柄(查对照表)就可找到与它对应的一个C+ 对象指针,然后把这个指针传给基类,基类利用这个指针调用WindowProc()函数对消息进行处理,WindowProc()函数中调用 OnWndMsg()函数,真正的消息路由及处理是由OnWndMsg()函数完成的。由于WindowProc()和 OnWndMsg()都是虚函数,而且是用派生类对象指针调用的,由多态性知最总终调用子类的。在OnWndMsg()函数处理的时候,根据消息种类去查找消息映射,判断所发的消息有没有响应函数,具体方式是到相关的头文件和源文件中寻找消息响应函数声明(从注释宏/AFX_MSG(CDrawView)./AFX_MSG之间寻找),消息映射(从宏 BEGIN_MESSAGE_MAP(.).END_MESSAGE_MAP() 之间寻找),最终找到对应的消息处理函数。当然,如果子类中没有对消息进行处理,则消息交由基类处理。说明:virtual LRESULT WindowProc(UINT message, WPARAM wParam, LPARAM lParam); 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 3 页,共 8 页 - - - - - - - - - virtual BOOL OnWndMsg(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pResult); 二,有关绘图1,使用 SDK 获取 DC 句柄:HDC hdc; hdc=:GetDc(m_hWnd);/获取 DC 句柄MoveToEx(hdc,m_ptOrigin.x,m_ptOrigin.y,NULL); LineTo(hdc,point.x,point.y); :ReleaseDC(m_hWnd,hdc);/释放 DC 2,利用 CDC 类指针和 CWin 类成员函数获取DC。CDC *pDC=GetDC(); pDC-MoveTo(m_ptOrigin); pDC-LineTo(point); ReleaseDC(pDC); 3,利用 CClientDC 对象。( CClientDC 类从 CDC 类派生来的)CClientDC dc(this); dc.MoveTo(m_ptOrigin); 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 4 页,共 8 页 - - - - - - - - - dc.LineTo(point); 说明:The CClientDC class is derived from CDC and takes care of calling the Windows functions GetDC at construction time and ReleaseDC at destruction time. This means that the device context associated with a CClientDC object is the client area of a window. 4,利用 CWindowDC对象。( CWindowDC类从 CDC 类派生来的)CWindowDC dc(this);/ dc.MoveTo(m_ptOrigin); dc.LineTo(point); 说明:The CWindowDC class is derived from CDC. It calls the Windows functionsGetWindowDC at construction time andReleaseDC at destruction time. This means that a CWindowDC object accesses the entire screen area of a CWnd (both client and nonclient areas). 5,GetParent() 得到父窗口指针; GetDesktopWindow()得到屏幕窗口指针。6,利用画笔改变线条颜色和类型:CPen pen(PS_DOT,1,RGB(0,255,0);/构造画笔对象名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 5 页,共 8 页 - - - - - - - - - CClientDC dc(this);CPen *pOldPen=dc.SelectObject(&pen);/将画笔选入 DC dc.MoveTo(m_ptOrigin); dc.LineTo(point); dc.SelectObject(pOldPen);/恢复先前的画笔7,使用画刷(通常利用画刷去填充矩形区域):使用单色画刷CBrush brush(RGB(255,0,0);/构造画刷对象CClientDC dc(this); dc.FillRect(CRect(m_ptOrigin,point),&brush);/用指定的画刷去填充矩形区域使用位图画刷CBitmap bitmap;/构造位图对象(使用前需要初试化)bitmap.LoadBitmap(IDB_BITMAP1);/初试化位图对象CBrush brush(&bitmap);/构造位图画刷CClientDC dc(this); dc.FillRect(CRect(m_ptOrigin,point),&brush);/用指定的位图画刷去填充矩形区域使用透明画刷CBrush *pBrush=CBrush:FromHandle(HBRUSH)GetStockObject(NULL_BRUSH);/获取透明画刷对象指针名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 6 页,共 8 页 - - - - - - - - - CClientDC dc(this); CBrush *pOldBrush=dc.SelectObject(pBrush);/将透明画刷选入 DC dc.Rectangle(CRect(m_ptOrigin,point); dc.SelectObject(pOldBrush);/释放透明画刷说明:The GetStockObject function retrieves a handle to one of the predefined stock pens, brushes, fonts, or palettes. HGDIOBJ GetStockObject( int fnObject / type of stock object ); Returns a pointer to a CBrush object when given a handle to a Windows HBRUSH object. static CBrush* PASCAL FromHandle( HBRUSH hBrush );/FromHandle是一个静态方法,故可用 CBrush:FromHandle()形式调用。注意点:1)静态方法不属于某一个具体对象,而属于类本身,在类加载的时候就已经为类静态方法分配了代码去,故可用CBrush:FromHandle()形式调用。2)静态方法中,不能引用非静态的数据成员和方法。名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 7 页,共 8 页 - - - - - - - - - 3)静态数据成员需要在类外单独做初始化,形式如:变量类型类名:变量名 = 初始值; 8,CDC:SetROP2 方法:int SetROP2( int nDrawMode ); Sets the current drawing mode. 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 8 页,共 8 页 - - - - - - - - -