Chinaunix首页 | 论坛 | 博客
  • 博客访问: 9495087
  • 博文数量: 1227
  • 博客积分: 10026
  • 博客等级: 上将
  • 技术积分: 20273
  • 用 户 组: 普通用户
  • 注册时间: 2008-01-16 12:40
文章分类

全部博文(1227)

文章存档

2010年(1)

2008年(1226)

我的朋友

分类: C/C++

2008-04-23 21:39:55

实现类似Excel和Visual C 里文件夹式样的标签控制(三)
——实例应用之功能扩充

编译/



在本文的前面第一部分第二部分中,我们描述了CFolderTabCtrl标签控制的设计思想、创建过程以及工作原理,通过一个实用程序分析了将CFolderTabCtrl与MFC框架结构融于一体的思路以及关键技巧。CFolderTabCtrl的主要目的是仿真Excel和Visual C 应用程序中标签控制页的UI功能。在这一部分我们将进一步增强和完善CFolderTabCtrl标签控制的仿真效果以及可重用性。内容包括创建多个标签页、并增加让标签页左右滚动的箭头按钮,这两个按钮上分别是示意左右的小三角形。
我们将以里一篇关于图像显示的文章中所附带的源代码作为例子,将CFolderTabCtrl实现的标签控制功能应用到图像显示程序中。原来的程序是一个MFC程序,它通过一个C 类(CPicture)封装Windows系统提供的IPicture 低级COM接口,使我们能轻松地显示各种格式图像文件,包括*.gif、*.jpg、*.bmp和DIBs文件。有关图像显示的具体细节不是本文要讨论的内容,具体细节请参考文章——“在MFC程序中显示JPG/GIF图像”。
首先,我们来看看如何实现箭头按钮?我的基本思路是将按钮作为CFolderTabCtrl的自绘窗口来创建,将标签放置在按钮的右边。如图一所示:


图一 程序中有13个标签页

为实现这个按钮,我创建了一个新类,CFolderButton。它个类是一个自绘按钮类,它有一个DrawItem函数负责绘制按钮,而不是用位图按钮。我选择用GDI类绘制表示左右的三角形,这样的话就不用担心由于缩放而导致的边缘显示问题。CFolderButton::DrawItem自己能绘制三角形来适应按钮的大小。按钮在置灰状态时用3D阴影颜色表示,但按钮被按下时,用象素替换的方法表示按钮状态。CFolderButton还处理鼠标消息以实现两个专用的特性,通常,按钮时不响应双击操作的,但这个按钮可以处理双击鼠标事件,它使得标签滚动两页。也就是说双击相当于两次单击一样。下面是实现代码:

void CFolderButton::OnLButtonDblClk  (UINT nFlags, CPoint pt)

{

  SendMessage(WM_LBUTTONDOWN,   

  nFlags,   

  MAKELONG(pt.x,pt.y));

}      
另一个特性是用户按着按钮不放,则标签会一直滚动,直到标签的端口。实现这个特性的方法是用一个定时器,当定时器被激活时,CFolderButton向它的父窗口发送一个WM_COMMAND消息,就好像按钮已经被压下一样。
void CFolderButton::OnTimer(UINT nIDEvent)

{

  GetParent()->SendMessage(WM_COMMAND, GetDlgCtrlID());

}      
详细的实现细节请参考本文的源代码。实际上,CFolderButton::OnTimer的实现是有一个启动延时的,所以持续滚动特性犹如键盘操作一样:在重复按下之前有轻微的延时。
CFolderButton并不知道有标签页以及如何滚动它们。它只知道如何画出按钮并响应鼠标行为。当用户按下按钮,Windows将WM_COMMAND/BN_CLICKED消息发送到父窗口:也就是CFolderTabCtrl。CFolderTabCtrl才能使标签页滚动。CFolderTabCtrl是按钮和标签的操纵者,就有点象组合框(ComboBox)操纵其编辑框、下拉按钮和列表框一样。
在CFolderTabCtrl中添加滚动按钮需要对几个地方进行修改。首先,你必须创建按钮。在哪里创建呢?记住!无论你什么时候创建有子窗口的复合控制,都应该在OnCreate中进行。
// 在 CFolderTabCtrl::OnCreate 中

if (m_dwFtabStyle & FTS_BUTTONS) {

  CRect rc;

  for (int id=FTBPREV; id<=FTBNEXT; id  ) {

    VERIFY(m_wndButton[id-1].Create(

      WS_VISIBLE|WS_CHILD, this, rc, id));

  }

  m_cxButtons = 2*CXBUTTON;

}      
FTS_BUTTONS是CFolderTabCtrl显示按钮的新式样。FTBPREV和FTBNEXT是枚举类型,其值分别为1和2,用于标示按钮的IDs。
在Windows系统里,你只要有子窗口,就必须管理它们的大小。这个工作由CFolderTabCtrl::OnSize专门负责。现在标签控制中加入了按钮,你就必须修改CFolderTabCtrl::OnPaint函数,将标签画在按钮的右边。为此不用修改原来的绘制代码,只要改一下视图窗口就可以了:
// x origin = (按钮的宽度) - (第一个标签页的x 坐标);

int xOrigin = m_cxButtons - 

  GetTab(m_iFirstTab)->GetRect().left;

dc.SetViewportOrg(xOrigin,0);      
设置完这个视图窗口后,你不用修改原来的代码便能正确切换标签。其中的转换发生在GDI内部的底层。
最后一个难题是处理按钮的单击。这个任务落在了CFolderTabCtrl对WM_COMMAND/BN_CLICKED消息的处理上。下面是关键代码:
BEGIN_MESSAGE_MAP(CFolderTabCtrl, CWnd)

  ……

  ON_BN_CLICKED(FTBNEXT,OnNextTab)

END_MESSAGE_MAP()

   

void CFolderTabCtrl::OnNextTab()

{

  if (m_iFirstTab < m_lsTabs.GetCount()-1) {

    m_iFirstTab  ;

    Invalidate();

    UpdateButtons();

  }

}
CFolderTabCtrl递增m_iFirstTab并重画。同时调用UpdateButtons函数更新按钮的状态(Enabled)。如果第一个标签可见,也就是左边不会再有标签,UpdateButtons便置灰左边按钮;如果最后一个标签完全可见,也就是右边不会再有标签,UpdateButtons便置灰右边按钮。其实,标签除了占据屏幕空间外不做任何事情。实现细节请参考源代码。(完)
阅读(291) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~