深入浅出MFC文档视图架构之框架(2)

深入浅出MFC文档视图架构之框架(2),第1张

概述  (2)CMDIFrameWnd类用于MDI应用程序的主框架窗口,主框架窗口是所有MDI文档子窗口的容器,并与子窗口共享菜单;CMDIFrameWnd类相较CFrameWnd类增加的重要函数有:MDIActivate(激活另一个MDI子窗口)、MDIGetActive(得到目前的活动子窗口)、MDIMaximize(最大化一个子窗口)、MDINext  
 (2)CMDiframeWnd类用于MDI应用程序的主框架窗口,主框架窗口是所有MDI文档子窗口的容器,并与子窗口共享菜单;CMDiframeWnd类相较CFrameWnd类增加的重要函数有:MDIActivate(激活另一个MDI子窗口)、MDIGetActive(得到目前的活动子窗口)、MDIMaximize(最大化一个子窗口)、MDINext(激活目前活动子窗口的下一子窗口并将当前活动子窗口排入所有子窗口末尾)、MDIRestore(还原MDI子窗口)、MdisetMenu(设置MDI子窗口对应的菜单)、MDITile(平铺子窗口)、MDICascade(重叠子窗口)。

  Visual C++开发环境是典型的MDI程序,其执行MDI Cascade的效果如图5.3。


图5.3 MDI Cascade的效果

  而执行MDI Tile的效果则如图5.4。


图5.4 MDI Tile的效果

  MdisetMenu函数的重要意义体现在一个MDI程序可以为不同类型的文档(与文档模板关联)显示不同的菜单,例如下面的这个函数Load一个菜单,并将目前主窗口的菜单替换为该菜单:

voID CMdivIEw::OnReplaceMenu()
{
 // Load a new menu resource named IDR_SHORT_MENU
 CMdIDoc* pdoc = Getdocument();
 pdoc->m_DefaultMenu = ::LoadMenu(AfxGetResourceHandle(),MAKEINTRESOURCE(IDR_SHORT_MENU));
 if (pdoc->m_DefaultMenu == NulL)
  return;

 // Get the parent window of this vIEw window. The parent window is
 // a CMDIChilDWnd-derived class. We can then obtain the MDI parent
 // frame window using the CMDIChilDWnd*. Then,replace the current
 // menu bar with the new loaded menu resource.
 CMDiframeWnd* frame = ((CMDIChilDWnd *) GetParent())->GetMDiframe();
 frame->MdisetMenu(cmenu::FromHandle(pdoc->m_DefaultMenu),NulL);
 frame->DrawMenubar();
}

  CMDiframeWnd类另一个不讲"不足以服众"的函数是OnCreateClIEnt,它是子框架窗口的创造者,其实现如下:

BOol CMDiframeWnd::OnCreateClIEnt(LPCREATESTRUCT lpcs,CCreateContext*)
{
 cmenu* pMenu = NulL;
 if (m_hMenuDefault == NulL)
 {
  // default implementation for MFC V1 backward compatibility
  pMenu = GetMenu();
  ASSERT(pMenu != NulL);
  // This is attempting to guess which sub-menu is the Window menu.
  // The windows user interface guIDelines say that the right-most
  // menu on the menu bar should be Help and Window should be one
  // to the left of that.
  int iMenu = pMenu->GetMenuItemCount() - 2;

  // If this assertion fails,your menu bar does not follow the guIDelines
  // so you will have to overrIDe this function and call CreateClIEnt
  // appropriately or use the MFC V2 MDI functionality.
  ASSERT(iMenu >= 0);
  pMenu = pMenu->GetSubMenu(iMenu);
  ASSERT(pMenu != NulL);
 }
 return CreateClIEnt(lpcs,pMenu);
}

  从CMDiframeWnd::OnCreateClIEnt的源代码可以看出,其中真正起核心作用的是对函数CreateClIEnt的调用:

BOol CMDiframeWnd::CreateClIEnt(LPCREATESTRUCT lpCreateStruct,
cmenu* pWindowMenu)
{
 ASSERT(m_hWnd != NulL);
 ASSERT(m_hWndMDIClIEnt == NulL);
 DWORD DWStyle = WS_VISIBLE | WS_CHILD | WS_border | WS_CliPCHILDREN | WS_CliPSIBliNGS |
Mdis_ALLCHILDSTYLES; // allow children to be created invisible
 DWORD DWExStyle = 0;
 // will be inset by the frame

 if (afxData.bWin4)
 {
  // special styles for 3d effect on Win4
  DWStyle &= ~WS_border;
  DWExStyle = WS_EX_CLIENTEDGE;
 }

 CLIENTCREATESTRUCT ccs;
 ccs.hWindowMenu = pWindowMenu->GetSafeHmenu();
 // set hWindowMenu for MFC V1 backward compatibility
 // for MFC V2,window menu will be set in OnMDIActivate
 ccs.IDFirstChild = AFX_IDM_FirsT_MDICHILD;

 if (lpCreateStruct->style & (WS_HSCRolL|WS_VSCRolL))
 {
  // parent MDiframe's scroll styles move to the MDICLIENT
  DWStyle |= (lpCreateStruct->style & (WS_HSCRolL|WS_VSCRolL));

  // fast way to turn off the scrollbar bits (without a resize)
  ModifyStyle(WS_HSCRolL|WS_VSCRolL,SWP_norEDRAW|SWP_FRAMECHANGED);
 }

 // Create MDICLIENT control with special IDC
 if ((m_hWndMDIClIEnt = ::CreateWindowEx(DWExStyle,_T("mdiclIEnt"),NulL,
DWStyle,m_hWnd,(HMENU)AFX_IDW_PANE_FirsT,
AfxGetInstanceHandle(),(LPVOID)&ccs)) == NulL)
 {
  TRACE(_T("Warning: CMDiframeWnd::OnCreateClIEnt: Failed to create MDICLIENT.")
_T(" GetLastError returns 0x%8.8X/n"),::GetLastError());
  return FALSE;
 }
 // Move it to the top of z-order
 ::BringWindowTotop(m_hWndMDIClIEnt);

 return TRUE;
}

  (3)CMDIChilDWnd类用于在MDI主框架窗口中显示打开的文档。每个视图都有一个对应的子框架窗口,子框架窗口包含在主框架窗口中,并使用主框架窗口的菜单。

  CMDIChilDWnd类的一个重要函数GetMDiframe()返回目前MDI客户窗口的父窗口,其实现如下:

CMDiframeWnd *CMDIChilDWnd::GetMDiframe()
{
 HWND hWndMDIClIEnt = ::GetParent(m_hWnd);
 CMDiframeWnd *pMDiframe;
 pMDiframe = (CMDiframeWnd*)CWnd::FromHandle(::GetParent(hWndMDIClIEnt));
 return pMDiframe;
}

  利用AppWizard生成的名为"example"的MDI工程包含如图5.5所示的类。


图5.5 一个MDI工程包含的类

  其中的CMainFrame继承自CMDiframeWnd,CChildFrame类继承自CMDIChilDWnd类,CExampleVIEw视图类则负责在CMDIChilDWnd类对应的子框架窗口中显示文档的数据。

  文中只是对CMDiframeWnd的CreateClIEnt成员函数进行了介绍,实际上,CFrameWnd、CMDIChilDWnd均包含CreateClIEnt成员函数。我们经常通过重载CFrameWnd:: CreateClIEnt、CMDIChilDWnd:: CreateClIEnt函数的方法来实现"窗口分割",例如:

BOol CChildFrame::OnCreateClIEnt(LPCREATESTRUCT lpcs,
CCreateContext *pContext)
{
 …
 if (!m_wndSplitter.Create(this,2,// 分割的行、列数
   CSize(10,10),// 最小化尺寸
   pContext))
 {
  TRACE0("创建分割失败");
  return FALSE;
 }
 …
 return TRUE;
}
总结

以上是内存溢出为你收集整理的深入浅出MFC文档/视图架构之框架(2)全部内容,希望文章能够帮你解决深入浅出MFC文档/视图架构之框架(2)所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1156376.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-01
下一篇 2022-06-01

发表评论

登录后才能评论

评论列表(0条)

保存