一个单文档界面中存在多个视图,并且可以根据需要进行视图的动态切换,这是当前比较流行的界面风格,它可以满足许多用户在操作和显示方面的需要。这种界面风格的主要代表软件是Outlook EXPress。而用VC++实现这种风格的界面有一定难度,笔者就这个问题进行了研究,并归纳总结出两种实现方法(这些代码都在VC++ 6.0下调试通过),使用时关键注意步骤和实现思路,不必拘泥于代码的形式。
方法一:静态创建切换法
步骤描述:
1.在窗口显示之前先将需要切换的所有的视图对象创建好,除首先显示的视图以外,其他在创建时都设置为不可见属性。
CMyWinApp::InitInstance()
{ ......
m_pViews[0] = pView1;
m_pViews[1] = (CView*) new CView2;
CDocument* pCurrentDoc = ((CFrameWnd*) m_pMainWnd)->GetActiveDocument();
// 初始化创建上下文相关指针
CCreateContext newContext;
newContext.m_pNewViewClass = NULL;
newContext.m_pNewDocTemplate = NULL;
newContext.m_pLastView = NULL;
newContext.m_pCurrentFrame = NULL;
newContext.m_pCurrentDoc = pCurrentDoc;
// 最初激活视的ID为AFX_IDW_PANE_FIRST,
//对新创建的视图增加这个值,注意对CSplitterWnd不能这样使用
UINT viewID[2];
viewID[1] = AFX_IDW_PANE_FIRST + 1;
CRect rect(0, 0, 0, 0);
for ( int nView=1; nView<NUMVIEWS; nView++ ) {
// 创建新的视图,创建的视图在应用中永久存在,直到应用程序退出,
//应用程序会自动删除新创建的视图
m_pViews[nView]->Create(NULL, NULL,
(AFX_WS_DEFAULT_VIEW & ~WS_VISIBLE),
// AFX_WS_DEFAULT_VIEW代表(WS_BORDER WS_VISIBLE WS_CHILD)
rect, m_pMainWnd, viewID[nView], &newContext);
}
// 当文档模板创建视图的时候,会自动发送WM_INITIALUPDATE消息,
//因此对于我们自己创建的视图,需要人工发送这条消息
((CForm2*)m_pViews[1])->OnInitialUpdate();
((CVswapView*)m_pViews[2])->OnInitialUpdate();
......
}
2.视图的切换
CView* CMyWinApp::SwitchView( UINT nIndex )
{
ASSERT( nIndex >=0 && nIndex < NUMVIEWS );
CView* pNewView = m_pViews[nIndex];
CView* pActiveView =((CFrameWnd*) m_pMainWnd)->GetActiveView();
if ( !pActiveView ) // 当前没有激活的视图
return NULL;
if ( pNewView == pActiveView ) // 当前视图和需要切换的视图相同
return pActiveView;
// 交换视图的窗口ID,使RecalcLayout()可以工作
UINT temp = ::GetWindowLong(pActiveView->m_hWnd, GWL_ID);
::SetWindowLong(pActiveView->m_hWnd, GWL_ID, ::GetWindowLong(pNewView->m_hWnd, GWL_ID));
::SetWindowLong(pNewView->m_hWnd, GWL_ID, temp);
// 显示新的视图,隐藏前一个视图
pActiveView->ShowWindow(SW_HIDE);
pNewView->ShowWindow(SW_SHOW);
((CFrameWnd*) m_pMainWnd)->SetActiveView(pNewView);
((CFrameWnd*) m_pMainWnd)->RecalcLayout();
pNewView->Invalidate();
return pActiveView;
} 方法二:动态创建切换法
步骤描述:
1.删除当前的视图
首先需要获得当前视图的指针,不能使用GetActiveView()和GetActiveDocument()这两个函数,当前视图有可能处在未激活状态,
所以应该使用EnumChildWindows这个Win32API函数,函数定义如下: