CFileDialog設(shè)置多選的問題解決
更新時間:2013年02月19日 09:30:30 作者:
前幾天同事問我在CFileDialog中多選時按確定按鈕后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í)如他所言,會出現(xiàn)返回值是IDCANCEL的情況,但是有時如果少幾個文件,就會返回IDOK。這說明多選文件對話框所選擇的文件有一個臨界值。選擇文件的多少里面體現(xiàn)的應(yīng)該是一個字符串緩沖區(qū)。因此我猜想CFileDialog里面應(yīng)該有一個字符串緩沖區(qū)用于存貯用戶所選的文件名,這個緩沖區(qū)有一個默認(rèn)長度,假如所選的全部文件長度超出了默認(rèn)長度,DoModal函數(shù)的返回值是IDCANCEL。如果是這樣,那么就有以下一些問題:
1.如果存在這個緩沖區(qū),CFileDialog類中有哪些數(shù)據(jù)成員負(fù)責(zé)控制這個緩沖區(qū),這個緩沖區(qū)的默認(rèn)長度又是多少?
2.如何增加這個緩沖區(qū)的長度以增加用戶選擇更多文件的需要?
為此我搜索了一些資料。設(shè)置CFileDialog類的初始化值主要集中在m_ofn這個數(shù)據(jù)成員。
m_ofn
The Windows OPENFILENAME structure. Provides access to basic file dialog box parameters.
其中m_ofn有兩個成員負(fù)責(zé)文件名緩沖區(qū):lpstrFile和nMaxFile。
lpstrFile
指向包含初始化文件名編輯控件使用的文件名的緩沖。如果不需要初始值,這個緩沖的第一個字符必須是NULL。當(dāng)GetOpenFileName或GetSaveFileName函數(shù)返回成功時,這個緩沖包含驅(qū)動器,路徑,文件名,及所選擇的文件的擴(kuò)展名。
如果OFN_ALLOWMULTISELECT標(biāo)記被設(shè)置并且用戶選擇了多個文件,緩沖包含了當(dāng)前目錄下被選擇文件的文件名。對于Explorer 風(fēng)格對話框,目錄和文件名字符串是被NULL分開的,在文件名之后有一個額外的NULL。對于舊風(fēng)格對話框,字符串是被空格分開的并且函數(shù)為帶有空格的文件名使用短文件名。你可以使用FindFirstFile函數(shù)在長短文件名之間轉(zhuǎn)換。如果用戶只選擇了一個文件,lpstrFile字符串在路徑和文件名之間沒有分隔。
如果緩沖太小,函數(shù)返回FALSE并且CommDlgExtendedError函數(shù)返回FNERR_BUFFERTOOSMALL.。既然這樣,lpstrFile緩沖的首先兩個字節(jié)包含必需的大?。ㄗ止?jié)或字符)。
nMaxFile
指定lpstrFile緩沖的大小,以TCHARs為單位。對于ANSI版本,是字節(jié)的個數(shù);對于 Unicode版本,是字符的個數(shù)。這個緩沖必須足夠存儲路徑和文件名字符串,包含結(jié)尾的null字符。如果緩沖太小,GetOpenFileName和GetSaveFileName函數(shù)返回假(FALSE)緩沖最小應(yīng)該在256個字符長。
經(jīng)過調(diào)試觀察,我發(fā)現(xiàn)nMaxFile的初始值為260。但是我寫程序測試這個緩沖區(qū)的默認(rèn)大小時,卻和這個初始值有矛盾。
我的測試辦法是這樣的。首先在E盤建一個Txt Data的文件夾,然后創(chuàng)建40個空的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個文件)時DoModal函數(shù)的返回值是IDOK,但是在選擇0..txt,1.txt,2.xtxt,27.txt,28.txt(共29個文件)時DoModal函數(shù)的返回值是IDCANCEL。接著我計算了一下所選中的文件的總長度(在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,如果增加一個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é)字符集下也是這個數(shù)值。這里需要注意的是CString::GetLength() 對于ASCII,返回字符串所占字節(jié)的數(shù)目,但如果是Unicode則實(shí)際上返回的是字符數(shù)而不是字節(jié)數(shù)
那么我初步斷定那個緩沖區(qū)的默認(rèn)大小不是我調(diào)試觀察到的260,而是512。至于開頭如何解決那個問題,只需要定義一個更大的緩沖區(qū),將lpstrFile指向這個緩沖區(qū),重設(shè)nMaxFile的值即可,具體是:
TCHAR szLargeBuf[4096]; // 定義一個臨時緩沖區(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個文件,超過8個讀不進(jìn)來,設(shè)斷點(diǎn)檢查發(fā)現(xiàn),如果想讀入8個文件,程序運(yùn)行到 if (dlgOpen->DoModal()==IDOK)就跳出if語句,不執(zhí)行下面的語句。難怪!但是究竟怎么才能讀入多個文件那,我搜索DoModal函數(shù)源代碼,在文件DLGFILE.CPP中找到。發(fā)現(xiàn)函數(shù)中有個判斷語句 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是不可見的,但是寫上去不會報錯。
CFileDialog如何實(shí)現(xiàn)文件多選
CFileDialog類設(shè)置OFN_ALLOWMULTISELECT 標(biāo)志可以實(shí)現(xiàn)文件多選功能,但是文件的數(shù)量是有限制的,如果要突破這個限制,就必須自己提供緩沖區(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個文件
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í)如他所言,會出現(xiàn)返回值是IDCANCEL的情況,但是有時如果少幾個文件,就會返回IDOK。這說明多選文件對話框所選擇的文件有一個臨界值。選擇文件的多少里面體現(xiàn)的應(yīng)該是一個字符串緩沖區(qū)。因此我猜想CFileDialog里面應(yīng)該有一個字符串緩沖區(qū)用于存貯用戶所選的文件名,這個緩沖區(qū)有一個默認(rèn)長度,假如所選的全部文件長度超出了默認(rèn)長度,DoModal函數(shù)的返回值是IDCANCEL。如果是這樣,那么就有以下一些問題:
1.如果存在這個緩沖區(qū),CFileDialog類中有哪些數(shù)據(jù)成員負(fù)責(zé)控制這個緩沖區(qū),這個緩沖區(qū)的默認(rèn)長度又是多少?
2.如何增加這個緩沖區(qū)的長度以增加用戶選擇更多文件的需要?
為此我搜索了一些資料。設(shè)置CFileDialog類的初始化值主要集中在m_ofn這個數(shù)據(jù)成員。
m_ofn
The Windows OPENFILENAME structure. Provides access to basic file dialog box parameters.
其中m_ofn有兩個成員負(fù)責(zé)文件名緩沖區(qū):lpstrFile和nMaxFile。
lpstrFile
指向包含初始化文件名編輯控件使用的文件名的緩沖。如果不需要初始值,這個緩沖的第一個字符必須是NULL。當(dāng)GetOpenFileName或GetSaveFileName函數(shù)返回成功時,這個緩沖包含驅(qū)動器,路徑,文件名,及所選擇的文件的擴(kuò)展名。
如果OFN_ALLOWMULTISELECT標(biāo)記被設(shè)置并且用戶選擇了多個文件,緩沖包含了當(dāng)前目錄下被選擇文件的文件名。對于Explorer 風(fēng)格對話框,目錄和文件名字符串是被NULL分開的,在文件名之后有一個額外的NULL。對于舊風(fēng)格對話框,字符串是被空格分開的并且函數(shù)為帶有空格的文件名使用短文件名。你可以使用FindFirstFile函數(shù)在長短文件名之間轉(zhuǎn)換。如果用戶只選擇了一個文件,lpstrFile字符串在路徑和文件名之間沒有分隔。
如果緩沖太小,函數(shù)返回FALSE并且CommDlgExtendedError函數(shù)返回FNERR_BUFFERTOOSMALL.。既然這樣,lpstrFile緩沖的首先兩個字節(jié)包含必需的大?。ㄗ止?jié)或字符)。
nMaxFile
指定lpstrFile緩沖的大小,以TCHARs為單位。對于ANSI版本,是字節(jié)的個數(shù);對于 Unicode版本,是字符的個數(shù)。這個緩沖必須足夠存儲路徑和文件名字符串,包含結(jié)尾的null字符。如果緩沖太小,GetOpenFileName和GetSaveFileName函數(shù)返回假(FALSE)緩沖最小應(yīng)該在256個字符長。
經(jīng)過調(diào)試觀察,我發(fā)現(xiàn)nMaxFile的初始值為260。但是我寫程序測試這個緩沖區(qū)的默認(rèn)大小時,卻和這個初始值有矛盾。
我的測試辦法是這樣的。首先在E盤建一個Txt Data的文件夾,然后創(chuàng)建40個空的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個文件)時DoModal函數(shù)的返回值是IDOK,但是在選擇0..txt,1.txt,2.xtxt,27.txt,28.txt(共29個文件)時DoModal函數(shù)的返回值是IDCANCEL。接著我計算了一下所選中的文件的總長度(在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,如果增加一個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é)字符集下也是這個數(shù)值。這里需要注意的是CString::GetLength() 對于ASCII,返回字符串所占字節(jié)的數(shù)目,但如果是Unicode則實(shí)際上返回的是字符數(shù)而不是字節(jié)數(shù)
那么我初步斷定那個緩沖區(qū)的默認(rèn)大小不是我調(diào)試觀察到的260,而是512。至于開頭如何解決那個問題,只需要定義一個更大的緩沖區(qū),將lpstrFile指向這個緩沖區(qū),重設(shè)nMaxFile的值即可,具體是:
TCHAR szLargeBuf[4096]; // 定義一個臨時緩沖區(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個文件,超過8個讀不進(jìn)來,設(shè)斷點(diǎn)檢查發(fā)現(xiàn),如果想讀入8個文件,程序運(yùn)行到 if (dlgOpen->DoModal()==IDOK)就跳出if語句,不執(zhí)行下面的語句。難怪!但是究竟怎么才能讀入多個文件那,我搜索DoModal函數(shù)源代碼,在文件DLGFILE.CPP中找到。發(fā)現(xiàn)函數(shù)中有個判斷語句 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是不可見的,但是寫上去不會報錯。
CFileDialog如何實(shí)現(xiàn)文件多選
CFileDialog類設(shè)置OFN_ALLOWMULTISELECT 標(biāo)志可以實(shí)現(xiàn)文件多選功能,但是文件的數(shù)量是有限制的,如果要突破這個限制,就必須自己提供緩沖區(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個文件
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語言字符串函數(shù)與內(nèi)存函數(shù)精講
這篇文章主要介紹一些c語言中常用字符串函數(shù)和內(nèi)存函數(shù)的使用,并且為了幫助讀者理解和使用,也都模擬實(shí)現(xiàn)了他們的代碼,需要的朋友可以參考一下2022-04-04C語言靜態(tài)與動態(tài)通訊錄的實(shí)現(xiàn)流程詳解
這篇文章主要為大家介紹了C語言分別實(shí)現(xiàn)靜態(tài)與動態(tài)的通訊錄示例代碼教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2021-11-11