实战静态拆分视图

简介:
该部分主要介绍一些基本概念和创建拆分视图的一般过程。
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,如需转载请自行联系原作者
目录
相关文章
|
6月前
|
SQL 数据格式
视图有哪些特点?哪些使用场景?
视图有哪些特点?哪些使用场景?
|
编解码 数据可视化 Java
3D模型拆分与合并展示,IVX真的可以简单实现
iVX 平台的优势和特点,包括逻辑完备性、操作流畅性、面向对象设计方法、可独立作为编程语言等方面的优势,下面来详细的介绍介绍。
120 0
|
5月前
|
安全 关系型数据库 MySQL
MySQL数据库——视图的更新、视图作用以及案例
MySQL数据库——视图的更新、视图作用以及案例
191 0
|
6月前
|
前端开发
实战:思考如何拆分组件
实战:思考如何拆分组件
|
存储 XML SQL
浅谈扩展字段设计
浅谈扩展字段设计
419 0
|
编解码 Android开发
视图 总结 基础
DragViewTaskRelease 使用FrameLayout布局(可以用其他的),内嵌imageView实现悬浮拖动和点击事件,继承View的ontouchEvent进行动作监听和移动 getX(),getY()等方法的区别(图解) BarPercentView 条形进度条(可设置 线性渐变-背景色-进度条颜色-进度条高度) 游戏下载进度条显示 zidingyiView矩形进度框,在构造函数里定义基础属性,初始化布局,定义俩个矩形,onMeasure获取宽高和onDraw方法显示
|
Rust Java 编译器
use关键字妙用|模块内容拆分
use关键字妙用|模块内容拆分
136 0
use关键字妙用|模块内容拆分
|
关系型数据库 测试技术 流计算