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

VC++
在类VC的界面实现中加入目录树
软件换肤技术在 BCB 中的实现
利用非模窗口生成MDI介面
报表输出轻松搞定
Windows 中不规则窗体的编程实现
解说Win32的窗口子类化
使用测试优先方法开发用户界面
一个简单的登录对话框的实现
一个简单的日记本程序
从资源中加载皮肤
一个在RichEdit中添加表情图象的类
ActiveSkin 4.3 软件换肤在VC中的实现
一种另类“关于(About)”对话框的动态显示方法
对话框打印预览及打印
关于如何换肤、子类化的解决方案
制作 MSN、QQ 的消息提示窗口
如何对 BCGControlBarPro 进行换肤
定制个性化的对话框窗口类
改变窗口中的光标形状
更新MFC中的视图,跟踪.NET Framework中的事件

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


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