当前位置: 首页 > 图文教程 > 开发语言 > VC++ > 智能ABC窗口的实现

VC++
用VC++创建自定义向导程序
自定义 CRichEditCtrl 控件
Office 2000 风格的停泊、智能型菜单
黑客攻击手段之偷梁换柱
完善 CPopupText 类
让工具条显示256色图像
定制编辑框的上下文菜单
列表控件排序功能的实现
一个优秀的网格控件CGridCtrl
如何用代码动态添加控件
如何在编辑框中使用IAutoComplete接口
应用程序中添加“Coolbars”的简单方法
介绍一个有Toolbar功能的可重用类 CPopupText
在工具栏按钮上添加文本标签
类似Dreamweaver的颜色选择器
如何设置ListView控件的完全行(Full Row)选项
如何enable/disable菜单项
关于CEdit控件的透明 --作者:monsoon
动态菜单项、状态条提示、工具条提示问题
CAnimateCtrl::Open的使用问题

VC++ 中的 智能ABC窗口的实现


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-10-30   浏览: 55 ::
收藏到网摘: n/a

智能ABC窗口的实现

作者:王海军

下载源代码

  高级顶层窗口是一种无标题栏、菜单、工具条的一种常居顶层的窗口,最常见的例子就是输入法窗口,比如智能ABC窗口。这类窗口的特点除了上边所说的,最重要的就是窗口移动。普通窗口移动是通过标题栏由系统自动实现的,对于这类窗口就无法依靠系统了,需要手工实现。实现的方法有两种:

  1. 手工处理WM_MOVE、WM_LBUTTONUP消息,并根据鼠标的移动自己实现窗口移动过程中的绘制;
  2. 在WM_MOVE消息处理函数中通过向自己发送参数wParam为HTCAPTION lParam为鼠标位置的WM_NCLBUTTONDOWN消息实现窗口移动,此时窗口移动过程的绘制由系统实现。

显然,后者较前者简单。本文采用后者。
  为了实现移动,还要对窗口定义有效移动区域,就是当鼠标移动到窗口中哪部分时鼠标变为十字箭头,暗示现在可以按下左键开始拖动。本文实现的窗口中有四个紧密排列的位图“按钮”,它们之间没有空隙,所以我定义位图以外窗口以内为有效移动区域。鼠标进入这片区域时将变为十字箭头形状,提示用户现在可以进行拖动。
  在定义窗口类时,如果窗口有边框,那么当鼠标在边框上时,接收到的是WM_NCMOUSEMOVE消息,所以为了在WM_MOUSEMOVE消息处理函数中统一处理,本文窗口没有使用边框,只要鼠标进入窗口,就开始接收到WM_MOUSEMOVE消息了。但这又引入另一个问题:如何绘制3D边框?本文使用了函数 :

void Draw3dRect( LPCRECT lpRect, COLORREF clrTopLeft, COLORREF clrBottomRight)

实现了3D边框的绘制。

本文实现的基本原理:

  1. 从CWnd直接派生一个窗口类CMainWnd,并添加一个共有函数bool CreateMainWnd()用来创建窗口;
  2. 在相应WM_PAINT时,在客户区中根据四个标记变量绘制相应的四个位图;
  3. 在WM_LBUTTONDOWN 消息处理中,检测鼠标是否落入某个“按钮”位图,设置相应标志,并强制系统重绘客户区;
  4. 在WM_MOUSEMOVE消息处理中,检测鼠标是否落入有效移动区域,并设置相应光标。如果按下了左键,就开始发送WM_NCLBUTTONDOWN消息,实现窗口移动;
  5. 为了关闭窗口,本程序设置了加速键F12,按F12可关闭程序;
代码如下:

1、创建窗口:
  bool CMainWnd::CreateMainWnd(void)  {   if(!CreateEx(0, ::AfxRegisterWndClass(0,                     ::LoadCursor(NULL,MAKEINTRESOURCE(IDC_ARROW)),                     GetSysColorBrush(COLOR_ACTIVEBORDER)                     ),           NULL,           WS_POPUP,           0,0,300,50,           NULL,           NULL))      return false;   return true;  }	
2、窗口绘制:根据四个位图“按钮”标志变量绘制相应的位图。
  void CMainWnd::OnPaint()  {   CPaintDC dc(this); // device context for painting   // TODO: 在此处添加消息处理程序代码   // 不为绘图消息调用 CWnd::OnPaint()   CBitmap* pOldBmp;   CDC CompatibleDC;   CompatibleDC.CreateCompatibleDC(&dc);   //绘制第一个按钮   if(m_bBTState[0])   pOldBmp=CompatibleDC.SelectObject(&m_BT1_1);   else   pOldBmp=CompatibleDC.SelectObject(&m_BT1_2);   dc.BitBlt(m_rcBT1.left,m_rcBT1.top,m_rcBT1.Width(),m_rcBT1.Height(),&CompatibleDC,0,0,SRCCOPY);   //绘制第二个按钮   if(m_bBTState[1])    CompatibleDC.SelectObject(&m_BT2_1);   else    CompatibleDC.SelectObject(&m_BT2_2);   dc.BitBlt(m_rcBT2.left,m_rcBT2.top,m_rcBT2.Width(),m_rcBT2.Height(),&CompatibleDC,0,0,SRCCOPY);   //绘制第三个按钮   if(m_bBTState[2])    CompatibleDC.SelectObject(&m_BT3_1);   else    CompatibleDC.SelectObject(&m_BT3_2);   dc.BitBlt(m_rcBT3.left,m_rcBT3.top,m_rcBT3.Width(),m_rcBT3.Height(),&CompatibleDC,0,0,SRCCOPY);   //绘制第四个按钮   if(m_bBTState[3])    CompatibleDC.SelectObject(&m_BT4_1);   else    CompatibleDC.SelectObject(&m_BT4_2);   dc.BitBlt(m_rcBT4.left,m_rcBT4.top,m_rcBT4.Width(),m_rcBT4.Height(),&CompatibleDC,0,0,SRCCOPY);   //绘制第五个按钮   if(m_bBTState[4])    CompatibleDC.SelectObject(&m_BT5_1);   else    CompatibleDC.SelectObject(&m_BT5_2);   dc.BitBlt(m_rcBT5.left,m_rcBT5.top,m_rcBT5.Width