[返回]
电脑报2000年第46期
用VC++在单文档界面中创建多个视图(上)
北京 周伟
一个单文档界面中存在多个视图,并且可以根据需要进行视图的动态切换,这是当前比较流行的界面风格,它可以满足许多用户在操作和显示方面的需要。这种界面风格的主要代表软件是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函数,函数定义如下:
BOOL EnumChildWindows(
HWND hWndParent, // 父窗口的句柄
WNDENUMPROC lpEnumFunc, // 用户自定义回调函数
LPARAM lParam // 传给回调函数的自定义参数
);
回调函数的定义如下:
BOOL CALLBACK EnumChildProc(
HWND hwnd, // 字窗口的句柄
LPARAM lParam // 自定义参数
);
EnumChildWindows函数遍历父窗口的所有子窗口,递归调用用户定义的回调函数,当回调函数返回FALSE时,停止遍历,
至于何时返回FALSE,这根据用户自己需要编写的回调函数来决定。
删除视图使用DeleteWindow()这个函数,用delete也可以删除,但还要其他底层的操作,这里就不详细介绍了,因为删除视图使用DeleteWindow()最合适、方便了。在删除视图的时候还要注意不能将文档同时自动删除。
删除视图的代码如下:
{ ......
CWnd* pWnd;
CWnd* pWndToDelete;
// 使用EnumChildWindows查找从CView继承的子窗口
::EnumChildWindows(m_hWnd, MyWndEnumProc, (LPARAM)&(pWnd));
if(pWnd == NULL)
{// 没有发现子窗口
return FALSE;}
// 发现子窗口,找到级别最高的子窗口,即父窗口为CMainFrame的窗口
while( lstrcmp(pWnd->GetRuntimeClass()->m_lpszClassName, ″CMainFrame″)
)
{
pWndToDelete = pWnd;
pWnd = pWnd->GetParent();
}
// 确保视图被删除时文档不被删除
pDoc->m_bAutoDelete = FALSE;}