CFileDialog的鉤子函數(shù)解決對(duì)話框的多選之DoModal問(wèn)題
前幾天領(lǐng)導(dǎo)問(wèn)我一個(gè)問(wèn)題:就是使用CFileDialog類在設(shè)置多選時(shí)選中的文件所放的文件緩沖區(qū)不知設(shè)置多大合適,設(shè)置小了DoModal返回為失敗, 通過(guò)CommDlgExtendedError函數(shù)獲取錯(cuò)誤碼為FNERR_BUFFERTOOSMALL(即緩沖區(qū)太?。?,設(shè)置大了又浪費(fèi)內(nèi)存。(我們 一次要選幾百個(gè)文件,實(shí)在不知設(shè)置多大合適)。
我談了我的思路:CFileDialog的數(shù)據(jù)成員m_ofn有一個(gè)數(shù)據(jù)成員為鉤子函數(shù)指針,通過(guò)設(shè)置這個(gè)函數(shù),可以勾取CFileDialog的相關(guān)消 息,比如用戶改變路徑的消息,然后獲取當(dāng)前路徑的文件個(gè)數(shù),以此為依據(jù)來(lái)設(shè)置緩沖區(qū)的大小。領(lǐng)導(dǎo)不是很明白我的思路,他上網(wǎng)搜了搜,找到一種方法,就是通過(guò)派生CFileDialog類的方法來(lái)做,具體如下:
Multiple Selection in a File Dialog
上面的鏈接提到的方法確實(shí)可行。但是我也相信我的方法是可行的。下班后我上網(wǎng)搜索了一下,發(fā)現(xiàn)微軟官網(wǎng)上有一個(gè)對(duì)此問(wèn)題的解決辦法,鏈接如下:
如何處理在 Windows 中 FNERR_BUFFERTOOSMALL
該鏈接提供的代碼適合的是Win 32的程序,并不適合MFC的程序,而且我建了一個(gè)Win32的程序測(cè)試該例子的代碼時(shí),發(fā)現(xiàn)一個(gè)問(wèn)題,就是當(dāng)選擇的文件過(guò)多時(shí),就是需要分配的緩沖區(qū)比較多時(shí),使用鏈接中的HeapAlloc函數(shù)會(huì)出現(xiàn)錯(cuò)誤,錯(cuò)誤提示如下:

因此要將鏈接中分配內(nèi)存和釋放的內(nèi)存的HeapAlloc和HeapFree函數(shù)分別用C++的new和delete操作符替換。
在微軟官網(wǎng)提供的做法的基礎(chǔ)上我摸索出用在MFC程序的做法,具體代碼如下:
// 鉤子函數(shù)
UINT_PTR CALLBACK MyOFNHookProc( HWND hdlg, // handle to child dialog box
UINT uiMsg, // message identifier
WPARAM wParam, // message parameter
LPARAM lParam // message parameter
)
{
int nResult = FALSE;
if (hdlg == NULL)
return 0;
#ifdef _DEBUG
// from "_AfxCommDlgProc()" of the file "dlgcomm.cpp"
_AFX_THREAD_STATE* pThreadState = AfxGetThreadState();
if (pThreadState->m_pAlternateWndInit != NULL)
pThreadState->m_pAlternateWndInit = NULL;
#endif
switch(uiMsg)
{
case WM_NOTIFY:
{
LPOFNOTIFY pOfn = (LPOFNOTIFY)lParam;
switch(pOfn->hdr.code)
{
case CDN_SELCHANGE:
{
TCHAR dummy_buffer;
// Get the required size for the 'files' buffer
HWND hOwner = GetParent(hdlg);
HWND hParent = GetParent(hOwner);
UINT nfiles = CommDlg_OpenSave_GetSpec(hOwner, &dummy_buffer, 1);
// Get the required size for the 'folder' buffer
int cbLength = CommDlg_OpenSave_GetSpec(GetParent(hdlg), NULL, 0);
cbLength += _MAX_PATH;
if(cbLength>(pOfn->lpOFN)->nMaxFile)
{
delete (pOfn->lpOFN)->lpstrFile;
(pOfn->lpOFN)->lpstrFile = new TCHAR[cbLength];
ZeroMemory((pOfn->lpOFN)->lpstrFile,cbLength);
(pOfn->lpOFN)->nMaxFile = cbLength;
}
nResult = TRUE;
break;
}
default:
break;
}
break;
}
default:
break;
}
return nResult;
}
#define NAMEBUF 1024
// 調(diào)用函數(shù)
void CMultiSelectDlg::OnButton1()
{
m_listbox.ResetContent();
m_static.SetWindowText(_T("0 files selected"));
TCHAR szFilters[]= _T("MyType Files (*.doc)|*.doc||");
// Create an Open dialog; the default file name extension is ".doc".
CFileDialog fileDlg(TRUE, _T("doc"), _T("*.doc"),
OFN_FILEMUSTEXIST | OFN_HIDEREADONLY|OFN_ALLOWMULTISELECT, szFilters);
fileDlg.m_ofn.lpstrFile=new TCHAR[NAMEBUF]; // 重新定義lpstrFile 緩沖大小
memset(fileDlg.m_ofn.lpstrFile,0,NAMEBUF); // 初始化定義的緩沖
fileDlg.m_ofn.nMaxFile = NAMEBUF; // 重定義nMaxFile
fileDlg.m_ofn.lpfnHook = (LPOFNHOOKPROC)MyOFNHookProc;
INT_PTR ret = fileDlg.DoModal();
if (ret == IDOK)
{
int width = 0;
CString str;
CDC *pDC = m_listbox.GetDC();
int saved = pDC->SaveDC();
pDC->SelectObject(GetFont());
UINT count = 0;
POSITION pos = fileDlg.GetStartPosition();
while (pos)
{
str = fileDlg.GetNextPathName(pos);
m_listbox.AddString(str);
CSize size(0, 0);
size = pDC->GetTextExtent(str);
width = width > size.cx ? width : size.cx;
++count;
}
pDC->RestoreDC(saved);
ReleaseDC(pDC);
m_listbox.SetHorizontalExtent(width + 5);
str.Format(_T("%u files selected"), count);
m_static.SetWindowText(str);
}
DWORD dwCode = CommDlgExtendedError();
if (FNERR_BUFFERTOOSMALL==dwCode)
{
AfxMessageBox(_T("獲取文件路徑失敗!"));
}
delete []fileDlg.m_ofn.lpstrFile;
fileDlg.m_ofn.lpstrFile = NULL;
}
另外使用鉤子函數(shù)的一個(gè)嚴(yán)重缺點(diǎn)是程序必須使用Unicode字符集進(jìn)行編譯,使用多字節(jié)字符集編譯程序執(zhí)行后FNERR_BUFFERTOOSMALL的錯(cuò)誤(這一點(diǎn)已經(jīng)測(cè)試過(guò),我比較難以理解的是為何在這一點(diǎn)上微軟不予支持多字節(jié)程序)。我的測(cè)試環(huán)境為: VS C++ 2005 + sp1,Win XP + sp3,unicode字符集。
- C++采用openfilename打開文件對(duì)話框用法實(shí)例
- VC++中HTControl控制類使用之CHTDlgBase對(duì)話框基類實(shí)例
- MFC創(chuàng)建模態(tài)對(duì)話框和非模態(tài)對(duì)話框的方法
- VC MFC非模態(tài)對(duì)話框的實(shí)現(xiàn)方法
- VC小技巧匯總之對(duì)話框技巧
- C#使用Word中的內(nèi)置對(duì)話框?qū)嵗?/a>
- C# 調(diào)用API函數(shù)彈出映射網(wǎng)絡(luò)驅(qū)動(dòng)器對(duì)話框問(wèn)題
- C++基于對(duì)話框的程序的框架實(shí)例
相關(guān)文章
C語(yǔ)言開發(fā)實(shí)現(xiàn)貪吃蛇小游戲
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言開發(fā)實(shí)現(xiàn)貪吃蛇小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-10-10
位運(yùn)算實(shí)現(xiàn)十進(jìn)制轉(zhuǎn)換為二進(jìn)制
這篇文章主要介紹了位運(yùn)算實(shí)現(xiàn)十進(jìn)制轉(zhuǎn)換為二進(jìn)制的相關(guān)資料,需要的朋友可以參考下2015-03-03
C++算法實(shí)現(xiàn)leetcode 1252奇數(shù)值單元格數(shù)目
這篇文章為大家主要介紹了C++實(shí)現(xiàn)leetcode 1252奇數(shù)值單元格的數(shù)目題解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08
c++類的隱式轉(zhuǎn)換與強(qiáng)制轉(zhuǎn)換重載詳解
轉(zhuǎn)換函數(shù)的名稱是類型轉(zhuǎn)換的目標(biāo)類型,因此,不必再為它指定返回值類型;轉(zhuǎn)換函數(shù)是被用于本類型的數(shù)值或變量轉(zhuǎn)換為其他的類型,也不必帶參數(shù)2013-09-09
C++面試題之結(jié)構(gòu)體內(nèi)存對(duì)齊計(jì)算問(wèn)題總結(jié)大全
這篇文章主要給大家總結(jié)了關(guān)于C++面試題中結(jié)構(gòu)體內(nèi)存對(duì)齊計(jì)算問(wèn)題的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),通過(guò)這些介紹的內(nèi)容對(duì)大家在面試C++工作的時(shí)候,會(huì)有一定的參考幫助,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-08-08
C語(yǔ)言中對(duì)字母進(jìn)行大小寫轉(zhuǎn)換的簡(jiǎn)單方法
這篇文章主要介紹了C語(yǔ)言中對(duì)字母進(jìn)行大小寫轉(zhuǎn)換的簡(jiǎn)單方法,是C語(yǔ)言入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-08-08

