CFileDialog設(shè)置多選的問題解決
更新時(shí)間:2013年02月19日 09:30:30 作者:
前幾天同事問我在CFileDialog中多選時(shí)按確定按鈕后DoModal函數(shù)的返回值是IDCANCEL的問題解決
他的代碼大致如下:
static TCHAR BASED_CODE szFilter[] = _T(&TXT(*.txt)|*.txt||&);
CFileDialog dlg(TRUE,_T(&txt&), NULL,OFN_HIDEREADONLY|OFN_ALLOWMULTISELECT,
szFilter, NULL );
INT_PTR nResult = dlg.DoModal();
我測試了一下,選擇比較多txt文件時(shí),確實(shí)如他所言,會(huì)出現(xiàn)返回值是IDCANCEL的情況,但是有時(shí)如果少幾個(gè)文件,就會(huì)返回IDOK。這說明多選文件對(duì)話框所選擇的文件有一個(gè)臨界值。選擇文件的多少里面體現(xiàn)的應(yīng)該是一個(gè)字符串緩沖區(qū)。因此我猜想CFileDialog里面應(yīng)該有一個(gè)字符串緩沖區(qū)用于存貯用戶所選的文件名,這個(gè)緩沖區(qū)有一個(gè)默認(rèn)長度,假如所選的全部文件長度超出了默認(rèn)長度,DoModal函數(shù)的返回值是IDCANCEL。如果是這樣,那么就有以下一些問題:
1.如果存在這個(gè)緩沖區(qū),CFileDialog類中有哪些數(shù)據(jù)成員負(fù)責(zé)控制這個(gè)緩沖區(qū),這個(gè)緩沖區(qū)的默認(rèn)長度又是多少?
2.如何增加這個(gè)緩沖區(qū)的長度以增加用戶選擇更多文件的需要?
為此我搜索了一些資料。設(shè)置CFileDialog類的初始化值主要集中在m_ofn這個(gè)數(shù)據(jù)成員。
m_ofn
The Windows OPENFILENAME structure. Provides access to basic file dialog box parameters.
其中m_ofn有兩個(gè)成員負(fù)責(zé)文件名緩沖區(qū):lpstrFile和nMaxFile。
lpstrFile
指向包含初始化文件名編輯控件使用的文件名的緩沖。如果不需要初始值,這個(gè)緩沖的第一個(gè)字符必須是NULL。當(dāng)GetOpenFileName或GetSaveFileName函數(shù)返回成功時(shí),這個(gè)緩沖包含驅(qū)動(dòng)器,路徑,文件名,及所選擇的文件的擴(kuò)展名。
如果OFN_ALLOWMULTISELECT標(biāo)記被設(shè)置并且用戶選擇了多個(gè)文件,緩沖包含了當(dāng)前目錄下被選擇文件的文件名。對(duì)于Explorer 風(fēng)格對(duì)話框,目錄和文件名字符串是被NULL分開的,在文件名之后有一個(gè)額外的NULL。對(duì)于舊風(fēng)格對(duì)話框,字符串是被空格分開的并且函數(shù)為帶有空格的文件名使用短文件名。你可以使用FindFirstFile函數(shù)在長短文件名之間轉(zhuǎn)換。如果用戶只選擇了一個(gè)文件,lpstrFile字符串在路徑和文件名之間沒有分隔。
如果緩沖太小,函數(shù)返回FALSE并且CommDlgExtendedError函數(shù)返回FNERR_BUFFERTOOSMALL.。既然這樣,lpstrFile緩沖的首先兩個(gè)字節(jié)包含必需的大?。ㄗ止?jié)或字符)。
nMaxFile
指定lpstrFile緩沖的大小,以TCHARs為單位。對(duì)于ANSI版本,是字節(jié)的個(gè)數(shù);對(duì)于 Unicode版本,是字符的個(gè)數(shù)。這個(gè)緩沖必須足夠存儲(chǔ)路徑和文件名字符串,包含結(jié)尾的null字符。如果緩沖太小,GetOpenFileName和GetSaveFileName函數(shù)返回假(FALSE)緩沖最小應(yīng)該在256個(gè)字符長。
經(jīng)過調(diào)試觀察,我發(fā)現(xiàn)nMaxFile的初始值為260。但是我寫程序測試這個(gè)緩沖區(qū)的默認(rèn)大小時(shí),卻和這個(gè)初始值有矛盾。
我的測試辦法是這樣的。首先在E盤建一個(gè)Txt Data的文件夾,然后創(chuàng)建40個(gè)空的txt文件。創(chuàng)建代碼如下:
for (int i = 0;i<40;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
CreateFile(strName, // file to open
GENERIC_READ, // open for reading
FILE_SHARE_READ, // share for reading
NULL, // default security
CREATE_NEW, // existing file only
FILE_ATTRIBUTE_NORMAL, // normal file
NULL); // no attr. template
}
然后我經(jīng)過多次嘗試,發(fā)現(xiàn)在選擇0..txt,1.txt,2.xtxt,27.txt(共28個(gè)文件)時(shí)DoModal函數(shù)的返回值是IDOK,但是在選擇0..txt,1.txt,2.xtxt,27.txt,28.txt(共29個(gè)文件)時(shí)DoModal函數(shù)的返回值是IDCANCEL。接著我計(jì)算了一下所選中的文件的總長度(在unicode字符集下編譯):
CString strAllFiles = _T(&&);
for (int i = 0;i<28;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
strAllFiles = strAllFiles + strName;
}
int nStrLen = strAllFiles.GetLength();
nStrLen的返回值是494,如果增加一個(gè)28.txt,即:
CString strAllFiles = _T(&&);
for (int i = 0;i<29;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
strAllFiles = strAllFiles + strName;
}
int nStrLen = strAllFiles.GetLength();
nStrLen的返回值是512.在多字節(jié)字符集下也是這個(gè)數(shù)值。這里需要注意的是CString::GetLength() 對(duì)于ASCII,返回字符串所占字節(jié)的數(shù)目,但如果是Unicode則實(shí)際上返回的是字符數(shù)而不是字節(jié)數(shù)
那么我初步斷定那個(gè)緩沖區(qū)的默認(rèn)大小不是我調(diào)試觀察到的260,而是512。至于開頭如何解決那個(gè)問題,只需要定義一個(gè)更大的緩沖區(qū),將lpstrFile指向這個(gè)緩沖區(qū),重設(shè)nMaxFile的值即可,具體是:
TCHAR szLargeBuf[4096]; // 定義一個(gè)臨時(shí)緩沖區(qū)
memset(szLargeBuf,'0',4096);
static TCHAR BASED_CODE szFilter[] = _T(&TXT(*.txt)|*.txt||&);
CFileDialog dlg(TRUE,_T(&txt&), NULL,OFN_HIDEREADONLY|OFN_ALLOWMULTISELECT,
szFilter, NULL );
dlg.m_ofn.lpstrFile = szLargeBuf;
#ifdef UNICODE
dlg.m_ofn.nMaxFile = 4096;
#else
dlg.m_ofn.nMaxFile = sizeof (szLargeBuf);
#endif
想讀入多文件,但總是最多讀入8個(gè)文件,超過8個(gè)讀不進(jìn)來,設(shè)斷點(diǎn)檢查發(fā)現(xiàn),如果想讀入8個(gè)文件,程序運(yùn)行到 if (dlgOpen->DoModal()==IDOK)就跳出if語句,不執(zhí)行下面的語句。難怪!但是究竟怎么才能讀入多個(gè)文件那,我搜索DoModal函數(shù)源代碼,在文件DLGFILE.CPP中找到。發(fā)現(xiàn)函數(shù)中有個(gè)判斷語句 DWORD nOffset = lstrlen(m_ofn.lpstrFile)+1; ASSERT(nOffset <= m_ofn.nMaxFile);而nMaxFile最大文件數(shù)在構(gòu)造函數(shù)中為空,如果不指定nMaxFile的值,判斷語句必然從DoModal函數(shù)中跳出。所以我在if (dlgOpen->DoModal()==IDOK)前面寫入下面語句CString str; dlgOpen->m_ofn.lpstrFile=str.GetBuffer(100000); str.ReleaseBuffer(); dlgOpen->m_ofn.nMaxFile = 5000;一切搞定! 但是要記住,m_ofn是不可見的,但是寫上去不會(huì)報(bào)錯(cuò)。
CFileDialog如何實(shí)現(xiàn)文件多選
CFileDialog類設(shè)置OFN_ALLOWMULTISELECT 標(biāo)志可以實(shí)現(xiàn)文件多選功能,但是文件的數(shù)量是有限制的,如果要突破這個(gè)限制,就必須自己提供緩沖區(qū)。例子如下:
CString fileExtensions = "jpg文件(*.jpg) |*.jpg||";
CFileDialog fileDlg(TRUE,
NULL,
NULL,
OFN_ALLOWMULTISELECT | OFN_ENABLESIZING | OFN_HIDEREADONLY,
fileExtensions);
const int MIN_FILE_NUMBER = 10; //至少允許選擇10個(gè)文件
fileDlg.m_ofn.lpstrFile = new TCHAR[_MAX_PATH * MIN_FILE_NUMBER]; //重新定義緩沖區(qū)大小
memset(fileDlg.m_ofn.lpstrFile, 0, _MAX_PATH * MIN_FILE_NUMBER); //初始化定義的緩沖區(qū)
fileDlg.m_ofn.nMaxFile = _MAX_PATH * MIN_FILE_NUMBER;
if (IDOK == fileDlg.DoModal())
{
POSITION pos = fileDlg.GetStartPosition();
while (NULL != pos)
{
TRACE(fileDlg.GetNextPathName(pos)); //獲取文件名
//使用文件...
}
}
delete[] fileDlg.m_ofn.lpstrFile; //最后別忘了釋放內(nèi)存
static TCHAR BASED_CODE szFilter[] = _T(&TXT(*.txt)|*.txt||&);
CFileDialog dlg(TRUE,_T(&txt&), NULL,OFN_HIDEREADONLY|OFN_ALLOWMULTISELECT,
szFilter, NULL );
INT_PTR nResult = dlg.DoModal();
我測試了一下,選擇比較多txt文件時(shí),確實(shí)如他所言,會(huì)出現(xiàn)返回值是IDCANCEL的情況,但是有時(shí)如果少幾個(gè)文件,就會(huì)返回IDOK。這說明多選文件對(duì)話框所選擇的文件有一個(gè)臨界值。選擇文件的多少里面體現(xiàn)的應(yīng)該是一個(gè)字符串緩沖區(qū)。因此我猜想CFileDialog里面應(yīng)該有一個(gè)字符串緩沖區(qū)用于存貯用戶所選的文件名,這個(gè)緩沖區(qū)有一個(gè)默認(rèn)長度,假如所選的全部文件長度超出了默認(rèn)長度,DoModal函數(shù)的返回值是IDCANCEL。如果是這樣,那么就有以下一些問題:
1.如果存在這個(gè)緩沖區(qū),CFileDialog類中有哪些數(shù)據(jù)成員負(fù)責(zé)控制這個(gè)緩沖區(qū),這個(gè)緩沖區(qū)的默認(rèn)長度又是多少?
2.如何增加這個(gè)緩沖區(qū)的長度以增加用戶選擇更多文件的需要?
為此我搜索了一些資料。設(shè)置CFileDialog類的初始化值主要集中在m_ofn這個(gè)數(shù)據(jù)成員。
m_ofn
The Windows OPENFILENAME structure. Provides access to basic file dialog box parameters.
其中m_ofn有兩個(gè)成員負(fù)責(zé)文件名緩沖區(qū):lpstrFile和nMaxFile。
lpstrFile
指向包含初始化文件名編輯控件使用的文件名的緩沖。如果不需要初始值,這個(gè)緩沖的第一個(gè)字符必須是NULL。當(dāng)GetOpenFileName或GetSaveFileName函數(shù)返回成功時(shí),這個(gè)緩沖包含驅(qū)動(dòng)器,路徑,文件名,及所選擇的文件的擴(kuò)展名。
如果OFN_ALLOWMULTISELECT標(biāo)記被設(shè)置并且用戶選擇了多個(gè)文件,緩沖包含了當(dāng)前目錄下被選擇文件的文件名。對(duì)于Explorer 風(fēng)格對(duì)話框,目錄和文件名字符串是被NULL分開的,在文件名之后有一個(gè)額外的NULL。對(duì)于舊風(fēng)格對(duì)話框,字符串是被空格分開的并且函數(shù)為帶有空格的文件名使用短文件名。你可以使用FindFirstFile函數(shù)在長短文件名之間轉(zhuǎn)換。如果用戶只選擇了一個(gè)文件,lpstrFile字符串在路徑和文件名之間沒有分隔。
如果緩沖太小,函數(shù)返回FALSE并且CommDlgExtendedError函數(shù)返回FNERR_BUFFERTOOSMALL.。既然這樣,lpstrFile緩沖的首先兩個(gè)字節(jié)包含必需的大?。ㄗ止?jié)或字符)。
nMaxFile
指定lpstrFile緩沖的大小,以TCHARs為單位。對(duì)于ANSI版本,是字節(jié)的個(gè)數(shù);對(duì)于 Unicode版本,是字符的個(gè)數(shù)。這個(gè)緩沖必須足夠存儲(chǔ)路徑和文件名字符串,包含結(jié)尾的null字符。如果緩沖太小,GetOpenFileName和GetSaveFileName函數(shù)返回假(FALSE)緩沖最小應(yīng)該在256個(gè)字符長。
經(jīng)過調(diào)試觀察,我發(fā)現(xiàn)nMaxFile的初始值為260。但是我寫程序測試這個(gè)緩沖區(qū)的默認(rèn)大小時(shí),卻和這個(gè)初始值有矛盾。
我的測試辦法是這樣的。首先在E盤建一個(gè)Txt Data的文件夾,然后創(chuàng)建40個(gè)空的txt文件。創(chuàng)建代碼如下:
for (int i = 0;i<40;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
CreateFile(strName, // file to open
GENERIC_READ, // open for reading
FILE_SHARE_READ, // share for reading
NULL, // default security
CREATE_NEW, // existing file only
FILE_ATTRIBUTE_NORMAL, // normal file
NULL); // no attr. template
}
然后我經(jīng)過多次嘗試,發(fā)現(xiàn)在選擇0..txt,1.txt,2.xtxt,27.txt(共28個(gè)文件)時(shí)DoModal函數(shù)的返回值是IDOK,但是在選擇0..txt,1.txt,2.xtxt,27.txt,28.txt(共29個(gè)文件)時(shí)DoModal函數(shù)的返回值是IDCANCEL。接著我計(jì)算了一下所選中的文件的總長度(在unicode字符集下編譯):
CString strAllFiles = _T(&&);
for (int i = 0;i<28;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
strAllFiles = strAllFiles + strName;
}
int nStrLen = strAllFiles.GetLength();
nStrLen的返回值是494,如果增加一個(gè)28.txt,即:
CString strAllFiles = _T(&&);
for (int i = 0;i<29;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
strAllFiles = strAllFiles + strName;
}
int nStrLen = strAllFiles.GetLength();
nStrLen的返回值是512.在多字節(jié)字符集下也是這個(gè)數(shù)值。這里需要注意的是CString::GetLength() 對(duì)于ASCII,返回字符串所占字節(jié)的數(shù)目,但如果是Unicode則實(shí)際上返回的是字符數(shù)而不是字節(jié)數(shù)
那么我初步斷定那個(gè)緩沖區(qū)的默認(rèn)大小不是我調(diào)試觀察到的260,而是512。至于開頭如何解決那個(gè)問題,只需要定義一個(gè)更大的緩沖區(qū),將lpstrFile指向這個(gè)緩沖區(qū),重設(shè)nMaxFile的值即可,具體是:
TCHAR szLargeBuf[4096]; // 定義一個(gè)臨時(shí)緩沖區(qū)
memset(szLargeBuf,'0',4096);
static TCHAR BASED_CODE szFilter[] = _T(&TXT(*.txt)|*.txt||&);
CFileDialog dlg(TRUE,_T(&txt&), NULL,OFN_HIDEREADONLY|OFN_ALLOWMULTISELECT,
szFilter, NULL );
dlg.m_ofn.lpstrFile = szLargeBuf;
#ifdef UNICODE
dlg.m_ofn.nMaxFile = 4096;
#else
dlg.m_ofn.nMaxFile = sizeof (szLargeBuf);
#endif
想讀入多文件,但總是最多讀入8個(gè)文件,超過8個(gè)讀不進(jìn)來,設(shè)斷點(diǎn)檢查發(fā)現(xiàn),如果想讀入8個(gè)文件,程序運(yùn)行到 if (dlgOpen->DoModal()==IDOK)就跳出if語句,不執(zhí)行下面的語句。難怪!但是究竟怎么才能讀入多個(gè)文件那,我搜索DoModal函數(shù)源代碼,在文件DLGFILE.CPP中找到。發(fā)現(xiàn)函數(shù)中有個(gè)判斷語句 DWORD nOffset = lstrlen(m_ofn.lpstrFile)+1; ASSERT(nOffset <= m_ofn.nMaxFile);而nMaxFile最大文件數(shù)在構(gòu)造函數(shù)中為空,如果不指定nMaxFile的值,判斷語句必然從DoModal函數(shù)中跳出。所以我在if (dlgOpen->DoModal()==IDOK)前面寫入下面語句CString str; dlgOpen->m_ofn.lpstrFile=str.GetBuffer(100000); str.ReleaseBuffer(); dlgOpen->m_ofn.nMaxFile = 5000;一切搞定! 但是要記住,m_ofn是不可見的,但是寫上去不會(huì)報(bào)錯(cuò)。
CFileDialog如何實(shí)現(xiàn)文件多選
CFileDialog類設(shè)置OFN_ALLOWMULTISELECT 標(biāo)志可以實(shí)現(xiàn)文件多選功能,但是文件的數(shù)量是有限制的,如果要突破這個(gè)限制,就必須自己提供緩沖區(qū)。例子如下:
CString fileExtensions = "jpg文件(*.jpg) |*.jpg||";
CFileDialog fileDlg(TRUE,
NULL,
NULL,
OFN_ALLOWMULTISELECT | OFN_ENABLESIZING | OFN_HIDEREADONLY,
fileExtensions);
const int MIN_FILE_NUMBER = 10; //至少允許選擇10個(gè)文件
fileDlg.m_ofn.lpstrFile = new TCHAR[_MAX_PATH * MIN_FILE_NUMBER]; //重新定義緩沖區(qū)大小
memset(fileDlg.m_ofn.lpstrFile, 0, _MAX_PATH * MIN_FILE_NUMBER); //初始化定義的緩沖區(qū)
fileDlg.m_ofn.nMaxFile = _MAX_PATH * MIN_FILE_NUMBER;
if (IDOK == fileDlg.DoModal())
{
POSITION pos = fileDlg.GetStartPosition();
while (NULL != pos)
{
TRACE(fileDlg.GetNextPathName(pos)); //獲取文件名
//使用文件...
}
}
delete[] fileDlg.m_ofn.lpstrFile; //最后別忘了釋放內(nèi)存
相關(guān)文章
C語言 fscanf 和 fprintf函數(shù)示例詳解
這篇文章主要介紹了 C語言 fscanf 和 fprintf函數(shù)示例詳解,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧2024-12-12
Windows下ncnn環(huán)境配置教程詳解(VS2019)
這篇文章主要介紹了Windows下ncnn環(huán)境配置(VS2019),本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
C語言中關(guān)于scanf函數(shù)的一些問題詳解
這篇文章主要為大家介紹了C語言中關(guān)于scanf函數(shù)的一些問題,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2021-12-12
C++實(shí)操之內(nèi)聯(lián)成員函數(shù)介紹
大家好,本篇文章主要講的是C++實(shí)操之內(nèi)聯(lián)成員函數(shù)介紹,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12

