该部分主要介绍一些基本概念和创建拆分视图的一般过程。
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,如需转载请自行联系原作者