实现类似Excel和Visual C++里文件夹式样的标签控制(二)
2007-11-01 20:23:07 来源:WEB开发网综上所述,你必须在主框架的OnCreateClient中创建CFolderFrame,此外还必需从CFolderView派生自己的视类,并按照前面所讲的方法进行必要的修改。这样不用费太多的周折就可以将标签控制应用到MFC程序框架中。
下面就让我们到幕后看一看CFolderFrame和CFolderView时如何运作的。CFolderFrame 中有保存滚动条和标签控制宽度的成员数据。当你调用CFolderFrame::Create的时候,它首先创建CFolderFrame框架,然后创建视图。它象MFC所做的那样用上下文信息来创建视图。但是要注意CFolderFrame创建的是自身的子视图,不是主框架(或者MDI子框架)。对于框架中的标签控制和水平/垂直滚动条以及调整大小的机关,都由CFolderFrame在OnCreate来创建,实现起来也不难,细节请参见源代码。
在代码中有一个小地方虽然不起眼,但是它很重要,就是将CFolderFrame框架的式样设置成WS_EX_CLIENTEDGE,以便使你的窗口保持与Windows 9x及Windows NT一致的凹陷效果。CFolderFrame在PreCreateWindow中做这件事。
一旦CFolderFrame创建了属于自己的窗口,你就必须管理它们的大小。通常这是OnSize的事情。具体细节纯粹是一些算法问题,我就不再这里赘言了,请自己参考源代码。
CFolderFrame从m_cxFolderTabCtrl获取标签的宽度。用SetFolderTabWidth函数来设置宽度,但一般情况下你不必使用这个函数,因为CFolderFrame总是试图用CFolderTabCtrl::GetDesiredWidth返回理想的宽度。只要OnSize工作正常,你就能看到CFolderFrame是如何仿真文件夹式样标签控制的,就像图四所示的那样,效果很好,选中的标签上边缘白色边缘与其上视图的背景无缝连接,尽管它们是两个不同的窗口,但看不出痕迹。
这一切好象挺简单,不过我还有最精彩的一部分没有讲到,那就是滚动条如何工作?派生的视图如何让MFC在CFolderFrame中操作滚动条,而不是象MFC所想象的那样使用滚动条?当我刚开始实现单独的滚动条时——我预料到会碰上很头疼的问题,因为原来的位图显示程序中CDIBView2视类是从CScrollView派生的,它是以子窗口方式隐藏或显示滚动条。但我此时想让视图使用自己的滚动条,一个在CFolderFrame中的滚动条——至少水平滚动条是这样。如何改变它的大小,使它与标签控制共处一室呢?于是我开始钻研CScrollView的内部运作机制,看看能否找到解决问题的办法。最后我发现只需要改写一下虚函数CWnd::GetScrollBarCtrl就能搞掂。这个函数有一个参数,其取值要么是SB_HORZ,要么是SB_VERT,最后返回相应的滚动条窗口。缺省实现返回Null。MFC走了一条很长的逻辑链来创建它需要的滚动条(通过调用::ShowScrollBar)。而CFolderView对GetScrollBarCtrl的改写很简单:
CScrollBar* CFolderView::GetScrollBarCtrl(int nBar) const
{return GetFolderFrame()->GetScrollBar(nBar);}
它将控制传递到CFolderFrame:
CScrollBar* CFolderFrame::GetScrollBar (int nBar)
{
return nBar==SB_HORZ ? &m_wndSBHorz
: nBar==SB_VERT ? &m_wndSBVert : NULL;
}
这比预料的要简单多了!MFC设计的灵活性真是令人吃惊。要想使用自己创建的与CScrollView一致的滚动条,仅仅改写GetScrollBarCtrl就可以了。例如,你可以发明有一个有迷幻色彩的超级滚动条……哈哈,帅呆了。
标签控制还有一个地方要说明:因为滚动条是CFolderFrame框架的一个子窗口,而不是视图的子窗口,CFolderFrame传递滚动消息(WM_HSCROLL 和 WM_VSCROLL),所以还必须自己编写代码将它们转发到视图:
void CFolderFrame::OnHScroll(...)
{
GetView()->SendMessage(WM_HSCROLL, ...);
}
对于WM_VSCROLL消息也同样如法炮制。一旦你解决的这些鸡毛蒜皮似的问题,滚动条的运行便OK了,它工作起来就像CScrollView一样。
其实,GetScrollBarCtrl是CFolderView存在一个主要理由,虽然CFolderView也将通知消息FTN_TABCHANGED从标签控制转换成轻松的OnChangedFolder虚函数调用,这使得通知消息的处理也变成了仅仅在视图中改写虚函数这么简单。
好了,看了这篇文章,你肯定觉得CFolderFrame和CFolderView并没有实现每一个想要的有关标签的UI特性,例如,没有提供左右以及开始/结束按钮,就像Excel界面那样(如图一所示)——说得没错,但是CFolderFrame和CFolderView已经提供了基本的框架,你可以在此基础上添加想要的特性。把按钮作为标签控制的附加控制添加进去,并在上下文的逻辑链中传递BN_CLICKED即可。总之,没有做不到的,只有想不到的,你可以对自己的程序进行所随心所欲的控制。下一部分我们将对标签页的数量进行扩展,并加上移动控制。(待续)
更多精彩
赞助商链接