欢迎访问:常州市武进区嘉泽中心小学网站 !今天是:
栏目列表
您现在的位置是:首页>>教师>>计算机技术>>程序设计>>杂项>>文章内容
MFC浅析(1) 文档视图结构中,缺省的命令处理
发布时间:2008-11-20   点击:   来源:本站原创   录入者:佚名
  文档视图结构中,缺省的命令处理

文档视图结构中,缺省的命令处理

FMD()

文档视图结构中,缺省的命令处理

在文档视图结构所构建的框架中,很多命令ID都有缺省的命令处理。很多功能都由他们完成,但这些功能在程序中不“可见”,不便于对程序的理解。

以下归纳了常见的缺省处理及其流程

在需要时候,可以重载这些函数以实现特定功能

1.ID_FILE_NEW

2.ID_FILE_OPEN

3.ID_FILE_SAVE

4.ID_FILE_SAVE_AS

5.ID_FILE_SAVE_COPY_AS

6.ID_FILE_CLOSE

7.ID_FILE_UPDATE

8.ID_FILE_PRINT_SETUP

9.ID_FILE_PRINT

10.ID_FILE_PRINT_PREVIEW

11.缺省编辑控制ID

12.ID_WINDOW_NEW

13.ID_WINDOW_ARRANGE

14.ID_WINDOW_CASCADE

15.ID_WINDOW_TILE_HORZ

16.ID_WINDOW_TILE_VERT

17.ID_WINDOW_SPLIT

18.ID_APP_ABOUT

19.ID_APP_EXIT

20.ID_HELP_INDEX

21.ID_HELP_USING

22.ID_CONTEXT_HELP

23.ID_HELP

24.ID_DEFAULT_HELP

25.ID_NEXT_PANE

26.ID_PREV_PANE

27.ID_OLE_INSERT_NEW

28.ID_OLE_EDIT_LINKS

29.ID_VIEW_TOOLBAR

30.ID_VIEW_STATUS_BAR

1.ID_FILE_NEW

CWinApp::OnFileNew

调用m_pDocManager->OnFileNew()


void CDocManager::OnFileNew()

{

 if (m_templateList.IsEmpty())

 {

  TRACE0("Error: no document templates registered with CWinApp.\n");

  AfxMessageBox(AFX_IDP_FAILED_TO_CREATE_DOC);

  return;

 }

 CDocTemplate* pTemplate = (CDocTemplate*)m_templateList.GetHead();

 //如果含有多个文档模板,显示一个新建文档类型对话框

 if (m_templateList.GetCount() > 1)

 {

  // more than one document template to choose from

  // bring up dialog prompting user

  CNewTypeDlg dlg(&m_templateList);

  int nID = dlg.DoModal();

  if (nID == IDOK)

   pTemplate = dlg.m_pSelectedTemplate;

  else

   return;     // none - cancel operation

 }

 ASSERT(pTemplate != NULL);

 ASSERT_KINDOF(CDocTemplate, pTemplate);

 

 //建立空文件

 pTemplate->OpenDocumentFile(NULL);

  // if returns NULL, the user has already been alerted

}

2.ID_FILE_OPEN

CWinApp::OnFileOpen

调用m_pDocManager->OnFileOpen()


void CDocManager::OnFileOpen()

{

 //出现打开文件对话框文件取得文件名

 CString newName;

 if (!DoPromptFileName(newName, AFX_IDS_OPENFILE,

   OFN_HIDEREADONLY ¦ OFN_FILEMUSTEXIST, TRUE, NULL))

  return; 

 

 //使用OpenDocumentFile

 AfxGetApp()->OpenDocumentFile(newName);

  // if returns NULL, the user has already been alerted

}

3.ID_FILE_SAVE

CDocument::OnFileSave()

调用DoFileSave()


void CDocument::OnFileSave()

{

 DoFileSave();

}

DoFileSave()又将调用DoSave()


BOOL CDocument::DoFileSave()

{

 DWORD dwAttrib = GetFileAttributes(m_strPathName);

 //如果文件是只读,或已经不存在了

 if (dwAttrib & FILE_ATTRIBUTE_READONLY)

 {

  //使用带NULL参数的DoSave

  if (!DoSave(NULL))

  {

   TRACE0("Warning: File save with new name failed.\n");

   return FALSE;

  }

 }

 else

 {

  //使用DoSave(....)

  if (!DoSave(m_strPathName))

  {

   TRACE0("Warning: File save failed.\n");

   return FALSE;

  }

 }

 return TRUE;

}

DoSave()的实现


BOOL CDocument::DoSave(LPCTSTR lpszPathName, BOOL bReplace)

//如果文件名参数为空,则让用户取名

// note: lpszPathName can be different than 'm_strPathName'

{

 CString newName = lpszPathName;

 if (newName.IsEmpty())

 {

  CDocTemplate* pTemplate = GetDocTemplate();

  ASSERT(pTemplate != NULL);

  newName = m_strPathName;

  if (bReplace && newName.IsEmpty())

  {

   newName = m_strTitle;

   // check for dubious filename

   int iBad = newName.FindOneOf(_T(" #%;/\\"));

   if (iBad != -1)

    newName.ReleaseBuffer(iBad);

   // append the default suffix if there is one

   CString strExt;

   if (pTemplate->GetDocString(strExt, CDocTemplate::filterExt) &&

     !strExt.IsEmpty())

   {

    ASSERT(strExt[0] == '.');

    newName += strExt;

   }

  }

  //"保存为"对话框

  if (!AfxGetApp()->DoPromptFileName(newName,

    bReplace ? AFX_IDS_SAVEFILE : AFX_IDS_SAVEFILECOPY,

    OFN_HIDEREADONLY ¦ OFN_PATHMUSTEXIST, FALSE, pTemplate))

   return FALSE;       // don't even attempt to save

 }

 CWaitCursor wait;

 

 //保存,由OnSaveDocument完成

 if (!OnSaveDocument(newName))

 {

  if (lpszPathName == NULL)

  {

   // be sure to delete the file

   TRY

   {

    CFile::Remove(newName);

   }

   CATCH_ALL(e)

   {

    TRACE0("Warning: failed to delete file \n");

    DELETE_EXCEPTION(e);

   }

   END_CATCH_ALL

  }

  return FALSE;

 }

 // reset the title and change the document name

 if (bReplace)

  SetPathName(newName);

 return TRUE;        // success

}

使用了文档类的OnSaveDocument完成保存动作


BOOL CDocument::OnSaveDocument(LPCTSTR lpszPathName)

{

 CFileException fe;

 CFile* pFile = NULL;

 pFile = GetFile(lpszPathName, CFile::modeCreate ¦

  CFile::modeReadWrite ¦ CFile::shareExclusive, &fe);

 if (pFile == NULL)

 {

  ReportSaveLoadException(lpszPathName, &fe,

   TRUE, AFX_IDP_INVALID_FILENAME);

  return FALSE;

 }

 //建立保存用的CArchive 

 CArchive saveArchive(pFile, CArchive::store ¦ CArchive::bNoFlushOnDelete);

 saveArchive.m_pDocument = this;

 saveArchive.m_bForceFlat = FALSE;

 TRY

 {

  CWaitCursor wait;

  //使用文档类的序列化操作完成实际存档行为

  Serialize(saveArchive);     // save me

  saveArchive.Close();

  ReleaseFile(pFile, FALSE);

 }

 CATCH_ALL(e)

 {

  ReleaseFile(pFile, TRUE);

  TRY

  {

   ReportSaveLoadException(lpszPathName, e,

    TRUE, AFX_IDP_FAILED_TO_SAVE_DOC);

  }

  END_TRY

  DELETE_EXCEPTION(e);

  return FALSE;

 }

 END_CATCH_ALL

 SetModifiedFlag(FALSE);     // back to unmodified

 return TRUE;        // success

}

4.ID_FILE_SAVE_AS

CDocument::OnFileSaveAs()

具体实现:调用NULL参数的DoSave(..)


void CDocument::OnFileSaveAs()

{

 if (!DoSave(NULL))

  TRACE0("Warning: File save-as failed.\n");

}

5.ID_FILE_SAVE_COPY_AS

The COleServerDoc::OnFileSaveCopyAs

6.ID_FILE_CLOSE

CDocument::OnFileClose


void CDocument::OnFileClose()

{

 //保存内容

 if (!SaveModified())

  return;

 //关闭文档

 OnCloseDocument();

}

如果需要保存,调用DoFileSave(..)


BOOL CDocument::SaveModified()

{

 //没有改动

 if (!IsModified())

  return TRUE;        

 //得到文件名

 CString name;

 if (m_strPathName.IsEmpty())

 {

  // get name based on caption

  name = m_strTitle;

  if (name.IsEmpty())

   VERIFY(name.LoadString(AFX_IDS_UNTITLED));

 }

 else

 {

  // get name based on file title of path name

  name = m_strPathName;

  if (afxData.bMarked4)

  {

   AfxGetFileTitle(m_strPathName, 

    name.GetBuffer(_MAX_PATH), _MAX_PATH);

   name.ReleaseBuffer();

  }

 }

 CString prompt;

 AfxFormatString1(prompt, AFX_IDP_ASK_TO_SAVE, name);

 //对话框:文件已被改动,是否保存

 switch (AfxMessageBox(prompt, MB_YESNOCANCEL, AFX_IDP_ASK_TO_SAVE))

 {

 case IDCANCEL:

  return FALSE;       // don't continue

 case IDYES:

  //保存文件

  if (!DoFileSave())

   return FALSE;       // don't continue

  break;

 case IDNO:

  // If not saving changes, revert the document

  break;

 default:

  ASSERT(FALSE);

  break;

 }

 return TRUE;

}

7.ID_FILE_UPDATE

COleServerDoc::OnUpdateDocument

8.ID_FILE_PRINT_SETUP

CWinApp::OnFilePrintSetup

9.ID_FILE_PRINT

CView::OnFilePrint

10.ID_FILE_PRINT_PREVIEW

CView::OnFilePrintPreview

11.缺省编辑控制ID

ID_EDIT_CLEAR

ID_EDIT_CLEAR_ALL

ID_EDIT_COPY

ID_EDIT_CUT

ID_EDIT_FIND

ID_EDIT_PASTE

ID_EDIT_PASTE_LINK

ID_EDIT_PASTE_SPECIAL

ID_EDIT_REPEAT

ID_EDIT_REPLACE

ID_EDIT_SELECT_ALL

ID_EDIT_UNDO

ID_EDIT_REDO

这些ID在对应的编辑视中有相应处理函数

12.ID_WINDOW_NEW

CMDIFrameWnd::OnWindowNew


void CMDIFrameWnd::OnWindowNew()

{

 //找到当前窗口,得到对应文档模板,使用文档模板的成员CreateNewFrame建立新窗口

 CMDIChildWnd* pActiveChild = MDIGetActive();

 CDocument* pDocument;

 if (pActiveChild == NULL ¦¦

   (pDocument = pActiveChild->GetActiveDocument()) == NULL)

 {

  TRACE0("Warning: No active document for WindowNew command.\n");

  AfxMessageBox(AFX_IDP_COMMAND_FAILURE);

  return;     // command failed

 }

 // otherwise we have a new frame !

 CDocTemplate* pTemplate = pDocument->GetDocTemplate();

 ASSERT_VALID(pTemplate);

 CFrameWnd* pFrame = pTemplate->CreateNewFrame(pDocument, pActiveChild);

 if (pFrame == NULL)

 {

  TRACE0("Warning: failed to create new frame.\n");

  return;     // command failed

 }

 pTemplate->InitialUpdateFrame(pFrame, pDocument);

}

13.ID_WINDOW_ARRANGE

ID_WINDOW_ARRANGE,以及ID_WINDOW_CASCADE、ID_WINDOW_TILE_HORZ、ID_WINDOW_TILE_VERT都由OnMDIWindowCmd()处理

OnMDIWindowCmd将相应消息发送给m_hWndMDIClient


BOOL CMDIFrameWnd::OnMDIWindowCmd(UINT nID)

{

 ASSERT(m_hWndMDIClient != NULL);

 UINT msg;

 UINT wParam = 0;

 switch (nID)

 {

 default:

  return FALSE;       // not for us

 case ID_WINDOW_ARRANGE:

  msg = WM_MDIICONARRANGE;

  break;

 case ID_WINDOW_CASCADE:

  msg = WM_MDICASCADE;

  break;

 case ID_WINDOW_TILE_HORZ:

  wParam = MDITILE_HORIZONTAL;

  // fall through

 case ID_WINDOW_TILE_VERT:

  ASSERT(MDITILE_VERTICAL == 0);

  msg = WM_MDITILE;

  break;

 }

 ::SendMessage(m_hWndMDIClient, msg, wParam, 0);

 return TRUE;

}

14.ID_WINDOW_CASCADE

参见ID_WINDOW_ARRANGE

15.ID_WINDOW_TILE_HORZ

参见ID_WINDOW_ARRANGE

16.ID_WINDOW_TILE_VERT

参见ID_WINDOW_ARRANGE

17.ID_WINDOW_SPLIT

CSplitterWnd::DoKeyboardSplit

用键盘控制分割窗口


BOOL CSplitterWnd::DoKeyboardSplit()

{

 ASSERT_VALID(this);

 int ht;

 if (m_nRows > 1 && m_nCols > 1)

  ht = splitterIntersection1; // split existing row+col

 else if (m_nRows > 1)

  ht = vSplitterBar1;         // split existing row

 else if (m_nCols > 1)

  ht = hSplitterBar1;         // split existing col

 else if (m_nMaxRows > 1 && m_nMaxCols > 1)

  ht = bothSplitterBox;       // we can split both

 else if (m_nMaxRows > 1)

  ht = vSplitterBox;          // we can split rows

 else if (m_nMaxCols > 1)

  ht = hSplitterBox;          // we can split columns

 else

  return FALSE;               // can't split

 // start tracking

 StartTracking(ht);

 CRect rect;

 rect.left = m_rectTracker.Width() / 2;

 rect.top = m_rectTracker.Height() / 2;

 if (m_ptTrackOffset.y != 0)

  rect.top = m_rectTracker.top;

 if (m_ptTrackOffset.x != 0)

  rect.left = m_bTracking2 ? m_rectTracker2.left :m_rectTracker.left;

 rect.OffsetRect(-m_ptTrackOffset.x, -m_ptTrackOffset.y);

 ClientToScreen(&rect);

 SetCursorPos(rect.left, rect.top);

 return TRUE;

}

18.ID_APP_ABOUT

建立CWinApp::OnAppAbout();

19.ID_APP_EXIT

CWinApp::OnAppExit

直接向主窗口发送WM_CLOSE消息


void CWinApp::OnAppExit()

{

 // same as double-clicking on main window close box

 ASSERT(m_pMainWnd != NULL);

 m_pMainWnd->SendMessage(WM_CLOSE);

}

20.ID_HELP_INDEX

CWinApp::OnHelpIndex

21.ID_HELP_USING

CWinApp::OnHelpUsing

22.ID_CONTEXT_HELP

CWinApp::OnContextHelp

23.ID_HELP

CWinApp::OnHelp

24.ID_DEFAULT_HELP

CWinApp::OnHelpIndex

25.ID_NEXT_PANE

CSplitterWnd::OnNextPaneCmd

26.ID_PREV_PANE

CSplitterWnd::OnNextPaneCmd

27.ID_OLE_INSERT_NEW

28.ID_OLE_EDIT_LINKS

29.ID_VIEW_TOOLBAR

切换AFX_IDW_TOOLBAR工具条

消息映射

ON_UPDATE_COMMAND_UI(ID_VIEW_TOOLBAR, OnUpdateControlBarMenu)

ON_COMMAND_EX(ID_VIEW_TOOLBAR, OnBarCheck)


BOOL CFrameWnd::OnBarCheck(UINT nID)

{

 ASSERT(ID_VIEW_STATUS_BAR == AFX_IDW_STATUS_BAR);

 ASSERT(ID_VIEW_TOOLBAR == AFX_IDW_TOOLBAR);

 ASSERT(ID_VIEW_REBAR == AFX_IDW_REBAR);

 CControlBar* pBar = GetControlBar(nID);

 if (pBar != NULL)

 {

  //设置该工具条状态

  ShowControlBar(pBar, (pBar->GetStyle() & WS_VISIBLE) == 0, FALSE);

  return TRUE;

 }

 return FALSE;

}


void CFrameWnd::OnUpdateControlBarMenu(CCmdUI* pCmdUI)

{

 ASSERT(ID_VIEW_STATUS_BAR == AFX_IDW_STATUS_BAR);

 ASSERT(ID_VIEW_TOOLBAR == AFX_IDW_TOOLBAR);

 ASSERT(ID_VIEW_REBAR == AFX_IDW_REBAR);

 CControlBar* pBar = GetControlBar(pCmdUI->m_nID);

 if (pBar != NULL)

 {

  pCmdUI->SetCheck((pBar->GetStyle() & WS_VISIBLE) != 0);

  return;

 }

 pCmdUI->ContinueRouting();

}

30.ID_VIEW_STATUS_BAR

实现方法与ID_VIEW_TOOLBAR相同


附件:
    关闭窗口
    打印文档
    账号登录
    保持登录 忘记密码?
    账号与武进教师培训平台同步