用VC++在单文档界面中创建多个视图(上)

发表于:2007-07-04来源:作者:点击数: 标签:
用VC++在单文档界面中创建多个视图(上) [返回] 电脑报2000年第46期 用VC++在单文档界面中创建多个视图(上) 北京 周伟 一个单文档界面中存在多个视图,并且可以根据需要进行视图的动态切换,这是当前比较流行的界面风格,它可以满足许多用户在操作和显
用VC++在单文档界面中创建多个视图(上)

[返回]
电脑报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;}

原文转自:http://www.ltesting.net