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

VC++
宏的妙用
泛型编程与设计新思维
C++中的虚函数(一)
C++模板元编程
C++多态技术
通用结构复制函数
<C++实践系列>C++中的虚函数(virtual function)
<C++实践系列>C++中的引用(reference)
<C++实践系列>C++中的异常(exception)
<C++实践系列>C++中的模板(template)
构造函数中的this指针
串行化(Serialization)
二进制浏览、编辑的实现
介绍一个模板动态数组
VC++界面一揽子解决方案(第三版) 介绍
VC++通用GIS功能开发解决方案 2.0v 介绍
确定有穷自动机分析内核
委托、信号和消息反馈的模板实现技术
按照类型名称动态创建对象
Boost中应用的泛型编程技术

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


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