以前常认为自己mfc学得不错,因为有CSDN+MSDN+GOOGLE,可是今天收到一个网友(女的哦!)的消息,说他有个程序有点问题,她把程序发到我的邮箱里去了,我down了下来.研究了一吓,这个程序给我补了一些基础知识,并认识到其实自己还很菜.然后就写了这篇文章了.
那个程序(网友给我发的那个程序)是一个SDI程序,错误是在点击文件(自动生成的那个)->保存(打开\另存为)时会出错,是一个断言的错误提示形式,这时我就去看了看她的CXXXDoc里,OnSaveDocument()是怎样写的,去看了看,发现并没有重载这个函数,这时就郁闷啦?那到底错在哪里呢?我马上想到不如debug一吓吧,在debug下,我再点文件->保存,可是并没有断下来?它直接就弹出个错误框.不知道是为什么,苦思了一会,在没办法的情况下,又试了一吓,,看能不能断下来,可是还是断不下来,又在没办法+失望的情况下,我点了重试(ASSERT的提示框不是有三个按钮的吗,分别是终止\重试\忽略),这下就可以断下来了?哈,我看看是断到哪里,噢,断到了CDocument::DoSave(...)里去了,我看了看,这个函数被CDocument::DoFileSave()所调用,看看CDocument::DoFileSave()的实现:
BOOL CDocument::DoFileSave()
{
DWORD dwAttrib = GetFileAttributes(m_strPathName);
if (dwAttrib & FILE_ATTRIBUTE_READONLY)
{
// we do not have read-write access or the file does not (now) exist
if (!DoSave(NULL))
{
TRACE0("Warning: File save with new name failed.\n");
return FALSE;
}
}
else
{
if (!DoSave(m_strPathName))
{
TRACE0("Warning: File save failed.\n");
return FALSE;
}
}
return TRUE;
}
明显,第一个if是另存为时调用的,下面的则为保存时的.
然后我又看了看CDocument::DoSave(LPCTSTR lpszPathName, BOOL bReplace)
BOOL CDocument::DoSave(LPCTSTR lpszPathName, BOOL bReplace)
// Save the document data to a file
// lpszPathName = path name where to save document file
// if lpszPathName is NULL then the user will be prompted (SaveAs)
// note: lpszPathName can be different than 'm_strPathName'
// if 'bReplace' is TRUE will change file name if successful (SaveAs)
// if 'bReplace' is FALSE will not change path name (SaveCopyAs)
{
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;
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 after failed SaveAs.\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(newName)没?我们平时写程序重载的OnSaveDocument就是运行到这里被调用的.
回到正题来,看看上面代码的粗体字那句话,程序就断到该句话上,这时我通过Watch1框(在上面输入变量
名,能看到该变量的值)上看到strExt的值为Progra.Document.则strExt[0]是'p',不是'.',我再往上面看一吓,
看到这句if (pTemplate->GetDocString(strExt, CDocTemplate::filterExt) ....,知道strExt的值是CDocTemplate::filterExt的值,我想,有GetDocString(...)就应该有SetDocString(...)吧?我去试了一吓,并没有这个函数.我在想,文件的后缀名应该是可以定制的,突然想到字符串资源,以前在看深入浅出MFC时,讲到过后缀是可以
定制的,我翻一吓深入浅出MFC,找到了,在321页,我复习了一吓,也说一吓(呵呵),是这样的,其实在创建工程的时候,你就可以定制7个字符串,分别是CDocTemplate::windowTitle
CDocTemplate::docName
CDocTemplate::fileNewName
CDocTemplate::filterName
CDocTemplate::filterExt
CDocTemplate::regFileTypeld
CDocTemplate::regFileTypeName
我解释一吓每个字符串的意义,
CDocTemplate::windowTitle很简单,就是窗口标题栏上的字符串,如你指定为BM,那运行时在标题栏上显示的就是"你的文档名 - BM".
CDocTemplate::docName这个是在MDI程序上才有用,如你指定为DL(众人:怎样又是剑圣,又是恐惧,是打魔兽还是谈VC啊!!!),那在MDI程序时,这个DL将做为默认的名字,如第一个文件将叫做DL1,再新建一个文件的话,则叫DL2,如此类推.
CDocTemplate::filterName这个是在文件->打开后,弹出打开对话框后,这个字符串将显示在文件类型里,并不是作为打开对话框的过滤字符串
CDocTemplate::filterExt这个作为打开对话框的过滤字符串,如设为".dl",则刚打开"打开"对话框后,将显示*.dl类型的文件,其他的都不显示.
CDocTemplate::regFileTypeld和CDocTemplate::regFileTypeName是在自定义文件类型时有用,详细的就不说了,说下去将是一篇教程.:)`
回到正题来,我们刚才发现在的问题是strExt为Progra.Document,我们要把他改成*.某个后缀名,我们去到工程文件夹下,找到xxx.rc,xxx是你的工程名,用记事本打开他,然后往下拉,找到String Table,其实可以搜索String Table,就可以直接到那里,找到后往下看一吓(也就是String Table后的第一段),就能看到
BEGIN
IDR_MAINFRAME "通讯录\n\nProgra\n通讯录(*.adr)\nProgra.Document\nProgra Document"
END(我这里是这样,你那里应该不一样,反正是这个格式),每个字符串之间用\n分隔
我们要修改的是Progra.Document这个,把他改成你喜欢的一个格式,如".mk",".xy"等等,
修改完后,保存,关闭,回到VC6.0里,会弹出一个对话框,问你是否重新加载资源文件, 点"是"就好了.这时再点文件->保存就不会出错了
呼,写完了,感觉不错,又进步了,呵呵!