还在苦苦敲代码开发APP?你out啦! 试试积木搭建APP吧~

VC菜单命令详解(文件打开、保存与关闭)

来源:     2016-12-06 08:30:05    人气:     我有话说( 0 人参与)

第一部分:五个命令ID: 处理函数ID_FILE_NEW CWinApp::OnFileNewID_FILE_OPEN CWinApp::OnFileOpenID_FILE_SAVE CDocument::OnFileSav...

第一部分:

五个命令ID: 处理函数   
ID_FILE_NEW CWinApp::OnFileNew    
ID_FILE_OPEN    CWinApp::OnFileOpen   
ID_FILE_SAVE    CDocument::OnFileSave   
ID_FILE_SAVEAS  CDocument::OnFileSaveAs   
ID_FILE_CLOSE   CDocument::OnFileClose   

  
1.ID_FILE_NEW   
CWinApp::OnFileNew调用CDocManager::OnFileNew。   
|   
CDocManager::OnFileNew判断文档模板是否多于一个,是则显示文档类型对话框(AFX_IDD_NEWTYPEDLG)   
让用户选择要创建的文档类型。然后调用CDocTemplate::OpenDocumentFile(NULL)。   
|   
CDocTemplate::OpenDocumentFile(LPCTSTR lpszPathName,BOOL bMake_Visible=TRUE)对于SDI和MDI的   
处理不一样。   
  
对于SDI,1,若已有文档打开,对其重新初始化,调用CDocument::SaveModified()保存当前文档;若没   
有文档存在,则调用CreateNewDocument()创建文档对象,再调用CreateNewFrame(pDoucment,NULL)创建   
文档的框架窗口。2,若lpszPathName为NULL,则调用CDocument::OnNewDocument()处理新文档,用   
pDocument->SetPathName(lpszPathName)设置文档的路径。3,判断当前线程主框架窗口是否存在,不存   
在则将1中创建的新框架作为主框架。4,调用InitialUpdateFrame显示框架窗口。   
  
对于MDI,与SDI基本相同,但是没有第3步。   
|   
CDocument::OnNewDocument()首先调用DeleteContents()删除原文档内容,使用m_strPathName.Empty()   
清除当前文档路径,SetModifiedFlag(FALSE)。   
  
2.ID_FILE_OPEN   
CWinApp::OnFileOpen调用CDocManager::OnFileOpen。   
|   
CDocManager::OnFileOpen首先显示文件打开对话框(AFX_IDS_OPENFILE),然后调用CWinApp::   
OpenDocumentFile(FileName)。   
|   
CWinApp::OpenDocumentFile(FileName)调用CDocManager::OpenDocumentFile。   
|   
CDocManager::OpenDocumentFile(LPCTSTR lpszFileName)遍历文档模板,对每个模板用   
MatchDocType(szPath,pOpenDocument)匹配文档类型。匹配时主要根据文件扩展名判断。若文件已经在   
某个文档中打开,则激活文档的第一个视图,否则用匹配的文档模板pBestTemplate->OpenDocumentFile   
(szPath)。   
|   
CDocTemplate::OpenDocumentFile调用CDocument::OnOpenDocument打开文件。   
|   
CDocument::OnOpenDocument打开文件,用DeleteContents删除现有文档内容,创建文件对应的CArchive   
对象loadArchive,Serialize(loadArchive)读文件内容,SetModifiedFlage(FALSE)。   
MDI: New Frame, New View?   
SDI: Update View?   
  
3.ID_FILE_SAVE   
CDocument::OnFileSave调用CDocument::DoFileSave()。   
|   
CDocument::OnFileSave判断文件是否只读,是则DoSave(NULL),否则DoSave(m_strPathName)。   
|   
CDocument::DoSave(LPCTSTR lpszPathName,BOOL bReplace=TRUE)的流程为:1,判断lpszPathName不空   
则跳第二步,为空则获取文档的路径名,加上扩展符。bReplace为TRUE则显示保存文件对话框(AFX_IDS_   
SAVEFILE),否则显示拷贝保存对话框(AFX_IDS+SAVEFILECOPY)。2,调用CDocument::OnSaveDocument保   
存文档。3,若bReplace==TRUE则SetPathName(newName)覆盖当前路径名。   
|   
CDocument::OnSaveDocument打开文件,创建CArchive对象saveArchive,Serialize(saveArchive)读写   
文件,SetModifiedFlag(FALSE)。   
  
4.ID_FILE_SAVEAS   
CDocument::OnFileSave调用DoSave(NULL)。   
  
5.ID_FILE_CLOSE   
CDocument::OnFileClose调用SaveModified()保存文件,再用OnCloseDocument处理文档关闭。   
|   
CDocument::SaveModified()用IsModified()判断是否修改,是则显示文件保存对话框(AFX_IDP_ASK_TO_   
SAVE),若用户选择“是”,则调用DoFileSave()保存文件。   
|   
CDocument::OnCloseDocument()首先销毁文档框架,再用DeleteContents()删除文档内容,若m_   
bAutoDelete则delete this。   
  
第二部分:

 

MDI打开文件的过程   
CWinApp::OnFileOpen()   
{   
    CDocManager::OnFileOpen()   
    {   
        CDocManager::DoPromptFileName()   
        {   
            CFileDialog::DoModal();   
        }   
        CWinApp::OpenDocumentFile()   
        {   
            ...选择文档模板;   
            ...调整视图和框架;   
            CDocTemplate::OpenDocumentFile();   
            {   
                判断有无现存文档,是否需要保存   
                新建框架   
                判断文件是否存在,调用CMyDoc::OnOpenDocunment/OnNewDocument   
            }   
        }   
    }   
}   
----------------------------------------   
Document/View命令处理流程   
ID_FILE_NEW   
CWinApp::OnFileNew()   
{   
    CDocManager::OnFileNew()   
    {   
        if(没有文档模板)   
            return;   
        if(有多个文档模板)   
        {   
            弹出对话框让用户选择;   
            取得模板指针;   
        }   
        CMultiDocTemplate::OpenDocumentFile()   
        {   
            new一个文档;   
            创建子框架;   
            构建frame,doc,view之间的关系;   
            CDocument::OnNewDocument()   
            {   
                DeleteContents();   
            }   
            InitialUpdateFrame();   
            return pDoc;   
        }   
    }   
}   
  
ID_FILE_OPEN   
CWinApp::OnFileOpen()   
{   
    CDocManager::OnFileOpen()   
    {   
        CDocManager::DoPromptFileName()   
        {   
            CFileDialog::DoModal();   
        }   
        CWinApp::OpenDocumentFile()   
        {   
            ...选择文档模板;   
            ...调整视图和框架;   
            CDocTemplate::OpenDocumentFile();   
            {   
                判断有无现存文档,是否需要保存   
                新建框架   
                判断文件是否存在   
                CMyDoc::OnOpenDocunment()   
                {   
                    打开文件;   
                    DeleteContents();   
                    建立CArchive;   
                    Serialize;   
                    关闭CArchive;   
                }   
            }   
        }   
    }   
}   
  
ID_FILE_SAVE   
CDocument::OnFileSave()   
{   
    DoFileSave(NULL)   
    {   
        CFileDialog::DoModal();   
        OnSaveDocument()   
        {   
            CArchive;   
            Serialize;   
        }   
    }   

 

 

***************************************************************

在这里将讲述SDI程序中application object、the main frame window、the document、the view、the document template object以及the associate string and menu resources之间的关系。
 
The Windows Application Object
在CWinApp派生类的Implement文件中会有类似CMyApp theApp的语句。theApp是一个全局变量,它就是启动MFC应用程序的机制所在。
以下是MFC应用程序启动的摘要:
1、Windows把应用程序加载到内存;
2、构造全局变量theApp(所有全局变量在程序被载入内存时被构造);
3、Windows调用全局函数WinMain,它是MFC的一部分,等同于无窗口应用程序的main函数---主程序的入口;
4、WinMain寻找到CWinApp派生类的对象theApp;
5、WinMain为theApp调用InitInstance函数,InitInstance函数可以被重载;
6、重载的InitInstance函数启动加载document、显示the main frame and view windows;
7、WinMain为theApp启动Run函数,Run函数负责分派window messages和command messages。
 
The Document Template Class
在CWinApp派生类的InitInstance中会有如下代码:
CSingleDocTemplate* pDocTemplate;
pDocTemplate = new CSingleDocTemplate(
    IDR_MAINFRAME,
    RUNTIME_CLASS(CStudentDoc),
    RUNTIME_CLASS(CMainFrame),       // main SDI frame window
    RUNTIME_CLASS(CStudentView));
AddDocTemplate(pDocTemplate);
这段代码建立了the application class、the document class、the view window class、the main frame window四个类之间的联系。这时,application class的对象已经存在(theApp),但其他四个类的对象还没有构造,MFC的动态创建机制就起到了作用。
一下两个图分别表示了以上四个类之间的关系、四个类的对象之间的关系:


 
Creating an Empty Document—The CWinApp::OnFileNew Function
当application class的InitInstance调用完AddDocTemplate以后,它将调用OnFileNew。
OnFileNew做如下事情:
1、构造Document对象,但不从硬盘中读取数据;
2、构造the main frame对象和the main frame window,但不显示它,它包括IDR_MAINFRAME、the toolbar、the status bar;
3、构造view对象和view窗口,但不显示它;
4、建立the document、main frame、view对象之间的关系,这里要借助AddDocTemplate建立的类关系;
5、为document对象调用虚函数CDocument::OnNewDocument,CDocument::OnNewDocument函数将调用虚函数DeleteContent函数;
6、为view对象调用虚函数CView::OnInitialUpdate;
7、为frame对象调用CFrameWnd::ActivateFrame以显示the main frame window(有the menus,view window,control bars)。
 
The Document Class's OnNewDocument Function
构造函数所完成的工作越少越好,这样构造函数出错的几率就会越来越小。而CDocument::OnNewDocument 和 CView::OnInitialUpdate 是做初始化的好地方。在这里你可以弹出一个MessageBox。如果出错,OnNewDocument还可以返回FALSE。值得注意的是,OnNewDocument可以被调用很多次,所以有些指令需只执行一次的话,就需要做一个“first time” flag成员变量。
 
Connecting File Open to Your Serialization Code—The OnFileOpen Function
文件打开菜单映射到CWinApp::OnFileOpen函数,它将执行以下几个步骤:
1、提示用户选择一个文件;
2、调用虚函数CDocument::OnOpenDocument,CDocument::OnOpenDocument打开文件,调用CDocument::DeleteContents,并且构造一个CArchive对象。随后调用document的Serialize函数从archive中载入数据;
3、调用view的OnInitialUpdate函数。
 
Connecting File Save and File Save As to Your Serialization Code
文件保存和另存为菜单映射到CDocument的OnFileSave函数,OnFileSave函数依次调用Serialize函数,它使用一个archive对象进行存储。
 
The Document's "Dirty" Flag
CDocument有一个protected成员函数m_bModified。可以通过CDocument的函数SetModifiedFlag和IsModified访问m_bModified。当新建一个document或者读取完硬盘上的数据后,m_bModified被Application Framework设为FALSE。当document的内容被改写后要确保把m_bModified设为TRUE。
 
小结:OnFileNew和OnFileOpen都在Application类中,Application类里面有DocumentTemplate,可以同时调动Document和View(OnInitialUpdate)。
OnFileSave和OnFileSaveAs都在Document类中,它只需做Serialization,不牵扯View的操作,故不需要把它们放在Application类中。

VC 菜单 命令

本文源自互联网,采用知识共享署名-非商业性使用 4.0 国际许可协议进行许可,
版权归原作者,如有问题请联系service@tsingfun.com (编辑:admin)
分享到: