1、问题的提出 使用MFC自动创建的应用程序中,对文档的新建和打开提供较好的支持。但是当我们的应用程序要求支持多种文档类型(注意不是多文档)时问题就出现了——当我们要创建一个新的文档时只能从MFC提供的默认的(也是十分简单的)CNewTypeDlg对话框中选择我们想要创建的文档类型。默认的CNewTypeDlg中文档模板名称的显示最长只有6个字符,给的信息是十分的少,对一般用户来说很难选择(如果你的文档类有前6位字符相同的,那么你很可能连自己都不知道应该选择哪一项)。 2、问题原理 应用程序类与文档、图文框和文档视图的关系如下图所示: 创建新文档的过程为: CWinAPP::OnFileNew()→CDocManager::OnFileNew()→CDocManager::OpenDocumentFile()→CDocTemplate::OpenDocumentFile()(这是一个纯的虚拟函数,在CMultiDocTemplate/CSingleDocTemplate中实现,其中调用了CreateNewDocument()和CreateNewFrame()以及CDocument::OnNewDocument())。
CWinAPP有个成员变量m_pDocManager调用
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 }
由上可见,CDocManager有个成员变量m_templateList,若有多个模板则弹出选择框,用户选择之后再调用CDocTemplate的OpenDocumentFile()。
3、解决问题 //不用说了,消息映射
在新建菜单右边添加文件类型菜单如图像文件(ID_FILE_NEW_IMAGE),文本文件(ID_FILE_NEW_TEXT)。工程名为MDI BEGIN_MESSAGE_MAP(CMDIApp, CWinApp) //{{AFX_MSG_MAP(CMDIApp) ON_COMMAND(ID_FILE_NEW_IMAGE, OnFileNewIMAGE)
ON_COMMAND(ID_FILE_NEW_TEXT, OnFileNewTEXT) //}}AFX_MSG_MAP END_MESSAGE_MAP() //应该没什么问题,添加文档模板 {//add second document template CMultiDocTemplate* pDocTemplate; pDocTemplate = new CMultiDocTemplate( IDR_OtherTYPE,//在字符串资源和菜单资源中添加相应的资源 RUNTIME_CLASS(COtherDoc),//添加相应的文档类 RUNTIME_CLASS(CChildFrame),// custom MDI child frame RUNTIME_CLASS(COtherView));//视图类 AddDocTemplate(pDocTemplate); } //这就是创建第一个文档模板的文档了 void CMy_MultiDocApp::OnFileNewIMAGE() { // TODO: Add your command handler code here //获取文档模板列表 POSITION pos=this->GetFirstDocTemplatePosition(); //取得一个文档模板,我取的是第一个,你可以去任何一个模板 CDocTemplate* pDocTemplate=this->GetNextDocTemplate(pos); //利用文档模板创建新文档 pDocTemplate->OpenDocumentFile(NULL); } 当然这并不完全,如果只是如此的话,我们运行程序的时候,主框架还没有显示,而我们想要避免的CNewTypeDlg对话框就会“很及时地”出现。因此我们得在BOOL C****App::InitInstance()函数中加入: if(cmdInfo.m_nShellCommand==CCommandLineInfo::FileNew) cmdInfo.m_nShellCommand=CCommandLineInfo::FileNothing; 用于告诉程序,开始时不要出现FileNew对话框。