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

VC++
用VC实现支持多语言的程序
用鼠标手势动作来发送命令
在VC中使用MATLAB C/C++函数库
水晶报表在VC++6.0中的简单使用方法
关于限次软件的制作
可定制大小和数量的随机数函数
用vc.net实现记录开关机时间的程序
隐藏你的鼠标
一个黑客程序开发实例 -- IE终结者
修改QQ的程序
ACE自适配通信环境
超频软件原理探析
OutlookGrid:以Outlook样式分组和排列数据项(C#)
遍历和查找外部程序 Tree-View 中的项目
用 Win API 实现自绘按钮类
VC++ 6.0使用复选框和单选钮
对C#中的TreeView加背景图
在MFC中应用CTreeCtrl控件的技巧
CSortListCtrl 加强版
使用 VC.net 轻松实现按钮控件自绘

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


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-10-30   浏览: 66 ::
收藏到网摘: 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