MFC浅析(1) 文档视图结构中,缺省的命令处理

来源:互联网 发布:什么是seo和sem 编辑:程序博客网 时间:2024/04/28 08:24

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

FMD(http://www.fmdstudio.net)

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

在文档视图结构所构建的框架中,很多命令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相同


转帖:http://blog.csdn.net/FMD/article/details/5522

原创粉丝点击