分类:
2006-09-13 10:44:39
介绍:细处着手,巧处用功。高手和菜鸟之间的差别就是:高手什么都知道,菜鸟知道一些。电脑小技巧收集最新奇招高招,让你轻松踏上高手之路。
一个单文档界面中存在多个视图,并且可以根据需要进行视图的动态切换,这是当前比较流行的界面风格,它可以满足许多用户在操作和显示方面的需要。这种界面风格的主要代表软件是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 //应用程序会自动删除新创建的视图 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(); ...... } |
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; } |
BOOL EnumChildWindows( HWND hWndParent, // 父窗口的句柄 WNDENUMPROC lpEnumFunc, // 用户自定义回调函数 LPARAM lParam // 传给回调函数的自定义参数 ); 回调函数的定义如下: BOOL CALLBACK EnumChildProc( HWND hwnd, // 字窗口的句柄 LPARAM lParam // 自定义参数 ); |
{ ...... 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;} // 删除视图 |
用户定义的回调函数:
BOOL CALLBACK MyWndEnumProc(HWND hWnd, LPARAM ppWndLPARAM) { CWnd* pWndChild = CWnd::FromHandlePermanent(hWnd); CWnd** ppWndTemp = (CWnd**)ppWndLPARAM; if( pWndChild && pWndChild->IsKindOf(RUNTIME_CLASS(CView)) ) { // 发现任何从CView继承的子窗口,将子窗口指针传递出去 *ppWndTemp = pWndChild; // 停止继续搜索 return FALSE; } else { *ppWndTemp = NULL; // 继续搜索 return TRUE; }} |
2.创建新的视图
CDocument* pCurrentDoc = ((CFrameWnd*) m_pMainWnd)->GetActiveDocument(); // 初始化创建上下文相关指针 CCreateContext newContext; newContext.m_pNewViewClass = RUNTIME_CLASS(CView1); newContext.m_pCurrentDoc = pCurrentDoc; newContext.m_pNewDocTemplate = NULL; newContext.m_pLastView = NULL; newContext.m_pCurrentFrame = NULL; CView* pNewView = STATIC_DOWNCAST(CView, CreateView(&newContext)); if( pNewView == NULL ) { return FALSE; } // 使用CreateView创建的视图不能自动调用OnInitialUpdate函数,需要人工调用OnInitialUpdate函数或者发送WM_INITIALUPDATE消息 pNewView->OnInitialUpdate(); // 使用CreateView创建的视图不会自动显示并且激活,需要人工操作 pNewView->ShowWindow(SW_SHOW); SetActiveView(pNewView); RecalcLayout(); |
注:RUNTIME_CLASS宏含义
每一个从CObject类继承的类,在定义DECLARE_DYNAMIC、DECLARE_DYNCREATE、DECLARE_SERIAL三个中任意一个宏时都会产生一个CRuntimeClass结构的静态对象,RUNTIME_CLASS返回的就是这个对象的指针,这个对象包含了其基类和本身在运行时刻的信息。以上是笔者在编写多个视应用在单文档的程序时整理出来的两种方法,因为VC++的强大和灵活,笔者相信还有更加巧妙的方法实现这个功能,希望这篇文章能够起到抛砖引玉的作用,也希望广大VC++编程爱好者对本文能够给予批评和指正。