CFileDialog的鉤子函數(shù)解決對(duì)話框的多選之DoModal問題
前幾天領(lǐng)導(dǎo)問我一個(gè)問題:就是使用CFileDialog類在設(shè)置多選時(shí)選中的文件所放的文件緩沖區(qū)不知設(shè)置多大合適,設(shè)置小了DoModal返回為失敗, 通過CommDlgExtendedError函數(shù)獲取錯(cuò)誤碼為FNERR_BUFFERTOOSMALL(即緩沖區(qū)太?。?,設(shè)置大了又浪費(fèi)內(nèi)存。(我們 一次要選幾百個(gè)文件,實(shí)在不知設(shè)置多大合適)。
我談了我的思路:CFileDialog的數(shù)據(jù)成員m_ofn有一個(gè)數(shù)據(jù)成員為鉤子函數(shù)指針,通過設(shè)置這個(gè)函數(shù),可以勾取CFileDialog的相關(guān)消 息,比如用戶改變路徑的消息,然后獲取當(dāng)前路徑的文件個(gè)數(shù),以此為依據(jù)來設(shè)置緩沖區(qū)的大小。領(lǐng)導(dǎo)不是很明白我的思路,他上網(wǎng)搜了搜,找到一種方法,就是通過派生CFileDialog類的方法來做,具體如下:
Multiple Selection in a File Dialog
上面的鏈接提到的方法確實(shí)可行。但是我也相信我的方法是可行的。下班后我上網(wǎng)搜索了一下,發(fā)現(xiàn)微軟官網(wǎng)上有一個(gè)對(duì)此問題的解決辦法,鏈接如下:
如何處理在 Windows 中 FNERR_BUFFERTOOSMALL
該鏈接提供的代碼適合的是Win 32的程序,并不適合MFC的程序,而且我建了一個(gè)Win32的程序測(cè)試該例子的代碼時(shí),發(fā)現(xiàn)一個(gè)問題,就是當(dāng)選擇的文件過多時(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è)試過,我比較難以理解的是為何在這一點(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ì)話框問題
- C++基于對(duì)話框的程序的框架實(shí)例
相關(guān)文章
C++實(shí)現(xiàn)動(dòng)態(tài)煙花效果
這篇文章主要介紹了利用C++實(shí)現(xiàn)的放煙花程序,用到了EGE圖形庫,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C++有一定幫助,需要的可以參考一下2022-01-01使用UART與PC通信實(shí)現(xiàn)msp430g2553單片機(jī)超聲波測(cè)距示例
這篇文章主要介紹了使用UART與PC通信實(shí)現(xiàn)msp430g2553單片機(jī)超聲波測(cè)距示例,需要的朋友可以參考下2014-05-05Clion-MinGW編譯后的exe文件添加ico圖標(biāo)的操作方法
這篇文章主要介紹了Clion-MinGW編譯后的exe文件添加ico圖標(biāo)的操作方法,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07C語言壓縮文件和用MD5算法校驗(yàn)文件完整性的實(shí)例教程
這篇文章主要介紹了C語言壓縮文件和用MD5算法校驗(yàn)文件完整性的實(shí)例教程,這里演示了Windows下將文件壓縮為7z格式以及MD5檢驗(yàn)文件和密碼的方法,需要的朋友可以參考下2016-04-04Qt數(shù)據(jù)庫應(yīng)用之實(shí)現(xiàn)文件編碼格式識(shí)別
在做數(shù)據(jù)導(dǎo)入導(dǎo)出的過程中,如果應(yīng)用場(chǎng)景多了,相信各位都會(huì)遇到一個(gè)問題就是文件編碼的問題。本文將用Qt實(shí)現(xiàn)文件編碼格式識(shí)別,感興趣的可以了解一下2022-06-06