实战静态拆分视图

简介:
该部分主要介绍一些基本概念和创建拆分视图的一般过程。
MFC 支持两种类型的拆分窗口:静态的和动态的。这里只探讨静态拆分,不过首先还是要熟悉一下这些概念。
静态拆分窗口的行列数在拆分窗口被创建时就设置好了,用户不能更改。但是用户可以缩放各行各列。一个静态拆分窗口最多可以包含 16 16 列。要找一个使用了静态拆分窗口的应用程序,只要看一下 windows 管理器即可。
动态拆分窗口最多可以有两行两列,但它们可以相互拆分和合并。 Vc 就使用了动态拆分窗口使得可以同时编辑源程序文件的两个以上不同的部分。
选择静态或动态拆分的一个准则是是否希望用户能够交互地修改拆分窗口的行列配置。另一个决定因素是计划在拆分窗口中使用的视图种类。在静态拆分窗口中很容易使用两个以上不同种类的视图,因为您可以在每个窗格中指定所用的视图类型。但是在动态拆分窗口中, MFC 管理着视图,除非从 CsplitterWnd 派生一个新类并修改拆分窗口的默认操作性能,否则拆分窗口中的所有视图使用的都是相同的视图类。
静态拆分窗口是用 CsplitterWnd::CreateStatic 而不是 CsplitterWnd::Create 创建,并且由于 MFC 不会自动创建静态拆分窗口中显示的视图,所以您要亲自在 CreateStatic 返回之后创建视图。 CsplitterWnd 为此提供了名为 CreateView 的函数。给框架窗口添加静态拆分视图的过程如下:
<!--[if !supportLists]--> 1.  <!--[endif]--> 给框架窗口类添加一个 CsplitterWnd 数据成员。
<!--[if !supportLists]--> 2.  <!--[endif]--> 覆盖框架窗口的 OnCreateClient 函数,并调用 CsplitterWnd::CreateStatic 来创建静态拆分视图。
<!--[if !supportLists]--> 3.  <!--[endif]--> 使用 CsplitterWnd:: CreateView 在每个静态拆分窗口的窗格中创建视图。
使用静态拆分窗口的一个优点是由于您自己给窗格添加视图,所以可以控制放入视图的种类。
下列中创建的静态拆分窗口包含了两种不同的视图:
BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CcreateContext* pContext)
{
if(!m_wndSplitter.CreateStatic(this, 1, 2) ||
!m_wndSplitter.CreateView(0, 0, RUNTIME_CLASS(CtextView), Csize(128, 0), pContext) ||
!m_wndSplitter.CreateView(0, 1, RUNTIME_CLASS(CpictureView), Csize(0, 0),pContext) )

return FALSE;
<!--[if !supportEmptyParas]--> <!--[endif]-->

return TRUE;
}
传递给 CreateStatic 的参数指定了拆分窗口的父亲以及拆分窗口包含的行列数。对每个窗格调用一次 CreateView 。用从 0 开始的行列编号来标示窗格。在上面的代码中,第一次调用 CreateView 在左窗格( 0 0 列)中加入类型为 CtextView 的视图,第二次调用在右窗格( 0 1 列)加入类型为 CpictureView 的视图。传递给 CreateView Csize 对象指定了窗格的初始尺寸。在上面的代码中, CtextView 窗格的初始宽度为 128 象素, CpictureView 窗格将占据剩余的窗口宽度。指定右窗格宽度的值和指定两个窗格高度的值都是 0 ,这是因为主结构会忽略它们。
 
下面以一个单文档程序为例,说明静态拆分视图的实现过程:
<!--[if !supportLists]--> 1.  <!--[endif]--> 首先建立一个单文档应用程序 SplitWnd ,视图 CSplitWndView 类型为列表视图。利用 CSplitWndView ::OnInitialUpdate 初始化列表视图。
<!--[if !supportLists]--> 2.  <!--[endif]--> 为该工程新增一个树型视图类 CMyTreeView ,并在该类中添加 HTREEITEM 类型的成员变量 m_hSubTree[2] ,该成员变量用来保存树型视图的子树句柄。利用 CMyTreeView ::OnInitialUpdate 初始化树型视图,为该树型视图添加一个树根,两个子树,参考代码如下:
HTREEITEM hRoot = GetTreeCtrl().InsertItem(_T(" 树根 "),  。。。 。。。 , TVI_ROOT);
m_hSubTree[0] = GetTreeCtrl().InsertItem(_T(" 子树 1"),  。。。 。。。 , hRoot);  
m_hSubTree [1] = GetTreeCtrl().InsertItem(_T(" 子树 2"),  。。。 。。。 , hRoot);
<!--[if !supportLists]--> 3.  <!--[endif]--> 在框架类中添加一个 CSplitterWnd  类型的成员变量 m_wndSplitter1 ,并重载 OnCreateClient 函数来拆分视图,代码如下:
BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext)
{
if(!(m_wndSplitter1.CreateStatic(this, 1, 2) ) ||
              !(m_wndSplitter1.CreateView(0, 1, RUNTIME_CLASS(CSplitWndView), CSize(0,0), pContext) ) ||
              !(m_wndSplitter1.CreateView(0, 0, RUNTIME_CLASS(CMyTreeView), CSize(180 ,0), pContext) ) 
       {
              return FALSE;
       }
       return TRUE;
}
如果你设计的程序需要更多的拆分视图,可以再在框架类中添加 CSplitterWnd  类型的成员变量 m_wndSplitter2 ,再次利用 CreateStatic 拆分视图,代码如下:
BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext)
{if(!(m_wndSplitter1.CreateStatic(this, 2, 1) ) ||
              !(m_wndSplitter1.CreateView(1, 0, RUNTIME_CLASS(CMyListView), CSize(0,0), pContext) ) ||
              !(m_wndSplitter2.CreateStatic(&m_wndSplitter1, 1, 2, WS_CHILD|WS_VISIBLE, m_wndSplitter1.IdFromRowCol(0,0)) ) ||
              !(m_wndSplitter2.CreateView(0, 0, RUNTIME_CLASS(CMyTreeView), CSize(180 ,0), pContext) ) ||
              !(m_wndSplitter2.CreateView(0, 1, RUNTIME_CLASS(CSplitWndView), CSize(0,0), pContext) )
       )
{
              return FALSE;
       }
       m_wndSplitter1.SetRowInfo(0, 350, 0);  // 重新设置行宽
       m_wndSplitter1.RecalcLayout();
<!--[if !supportEmptyParas]--> <!--[endif]-->

return TRUE;
}
现在基本的界面就建立好了,有关初始化列表视图的代码要根据具体情况来添加。另外,别忘记在框架 .cpp 中包含视图类的头文件。
为了视类之间的通讯,我们让这些视类共用一个文档类,现在来编写视图之间通讯的代码:
1 .在文档类中添加 int  m_nViewType ,表示要显示的类型。当用户单击“子树 1 ”或“子树 2 ”时改变其值。
<!--[if !supportLists]--> 2.  <!--[endif]--> 建立树型视图通知 TVN_SELCHANGED 响应函数 OnSelchanged ,添加如下代码:
CSplitWndDoc* m_pDoc = (CSplitWndDoc*)GetDocument();
HTREEITEM hTmp = GetTreeCtrl().GetSelectedItem();  // 得到当前选中的子树句柄
<!--[if !supportEmptyParas]--> <!--[endif]-->

// m_pDoc->m_nViewType 复位
m_pDoc->m_nViewType = 1000;
<!--[if !supportEmptyParas]--> <!--[endif]-->

if(hTmp == m_hSubTree [0] && m_pDoc)
{
       m_pDoc->m_nViewType = 0;       // 将显示类型设置为子树 1
}
if(hTmp == m_hSubTree [1] && m_pDoc)   
{
m_pDoc->m_nViewType = 1;  // 将显示类型设置为子树 2
}
// 在改变子树的情况下才刷新 CSplitWndView 视图
if(hTmp == m_hSubTree [0] || hTmp == m_hSubTree [1] && m_pDoc)
{
       // 通知视图更新 CSplitWndView
       m_pDoc->UpdateAllViews(this);
}
3 .在 CSplitWndView 中重载 OnUpdate 函数,响应 UpdateAllViews
CSplitWndDoc* pDoc = GetDocument();
GetListCtrl().DeleteAllItems();            // 删除列表视图中的项
       switch(pDoc->m_nViewType)        // 查看视图类型
       {
       case 0:
              // 将与子树 1 相关的项添加到列表视图中
              break;
       case 1:
              // 将与子树 2 相关的项添加到列表视图中
              break;
       }




本文转自 张宇 51CTO博客,原文链接:http://blog.51cto.com/zhangyu/33861,如需转载请自行联系原作者
目录
相关文章
|
编解码 搜索推荐 算法
Celero:一个 C++ 的基准测试管理库
对代码进行持续性开发和有意义的基准测试是一个复杂的任务。虽然测试工具本身(Intel® VTune™ Amplifier, SmartBear AQTime, Valgrind)与应用程序没有相关性,但是它们在某些时候对一些小团队,或者说是一些繁琐的工作来说还是很重要的。这个Celero项目,主要是要建仓一个小型的程序库,使它可以在加入 C++ 工程和对代码进行基准测试时能够非常容易地去重建,分享,并允许在独立的运行进程、开发者或者是工程间进行比较。Celero 使用一个与 GoogleTest 相似的构架,使得他的 API 很容易地使用,并融入一个工程中。当你在开发过程中进行自动测试时,自动
744 0
Celero:一个 C++ 的基准测试管理库
|
算法 JavaScript 前端开发
开源项目推荐:CNC+CRC/SoftPLC/OpenCASCADE/CAD/CAM(三)
开源项目推荐:CNC+CRC/SoftPLC/OpenCASCADE/CAD/CAM
3733 1
开源项目推荐:CNC+CRC/SoftPLC/OpenCASCADE/CAD/CAM(三)
|
C# 前端开发
WPF - 图形设计器(Diagram Designer)
原文:WPF - 图形设计器(Diagram Designer)   OpenExpressApp计划中包括建模工具,计划是采用MetaEdit+模型来作为元模型,使用codeproject的《WPF Diagram Designer》一系列文章来做为设计器实现参考,本篇介绍一下codeprojcet的这四个文章,推荐给对图形设计器感兴趣的人去看看,通过WPF的模板功能和其他功能可以很方便的设计出图形编辑器。
3798 0
|
uml
UML 类图几种关系(依赖、关联、泛化、实现、聚合、组合)及其对应代码
UML 类图几种关系(依赖、关联、泛化、实现、聚合、组合)及其对应代码
2729 0
|
存储 网络协议 算法
[计算机网络(谢希仁 第八版)]第一章 概述(学习复习笔记)
[计算机网络(谢希仁 第八版)]第一章 概述(学习复习笔记)
|
存储 安全 数据库连接
【C++智能指针】深入探究C++智能指针:自定义删除器的设计与选择
【C++智能指针】深入探究C++智能指针:自定义删除器的设计与选择
766 0
|
编译器 C++ 容器
STL常用之vector,list,stack,queue,deque总结与对比
STL常用之vector,list,stack,queue,deque总结与对比
|
算法 C++ 芯片
RANSAC算法拟合平面实现(附代码c++)
RANSAC算法拟合平面实现(附代码c++)
917 0
|
安全 容器
超经典单例模板类详解
超经典单例模板类详解
|
编译器 C++
C/C++ - enum 与 int 相互转换
C/C++ - enum 与 int 相互转换
1989 0