实战静态拆分视图

该部分主要介绍一些基本概念和创建拆分视图的一般过程。
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]--><?XML:NAMESPACE PREFIX = O />

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;
       }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值