程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C >> C語言基礎知識 >> CFileDialog設置多選的問題解決

CFileDialog設置多選的問題解決

編輯:C語言基礎知識
他的代碼大致如下:
  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文件時,確實如他所言,會出現返回值是IDCANCEL的情況,但是有時如果少幾個文件,就會返回IDOK。這說明多選文件對話框所選擇的文件有一個臨界值。選擇文件的多少裡面體現的應該是一個字符串緩沖區。因此我猜想CFileDialog裡面應該有一個字符串緩沖區用於存貯用戶所選的文件名,這個緩沖區有一個默認長度,假如所選的全部文件長度超出了默認長度,DoModal函數的返回值是IDCANCEL。如果是這樣,那麼就有以下一些問題:
  1.如果存在這個緩沖區,CFileDialog類中有哪些數據成員負責控制這個緩沖區,這個緩沖區的默認長度又是多少?
  2.如何增加這個緩沖區的長度以增加用戶選擇更多文件的需要?
  為此我搜索了一些資料。設置CFileDialog類的初始化值主要集中在m_ofn這個數據成員。
  m_ofn
  The Windows OPENFILENAME structure. Provides access to basic file dialog box parameters.
  其中m_ofn有兩個成員負責文件名緩沖區:lpstrFile和nMaxFile。
  lpstrFile
  指向包含初始化文件名編輯控件使用的文件名的緩沖。如果不需要初始值,這個緩沖的第一個字符必須是NULL。當GetOpenFileName或GetSaveFileName函數返回成功時,這個緩沖包含驅動器,路徑,文件名,及所選擇的文件的擴展名。
  如果OFN_ALLOWMULTISELECT標記被設置並且用戶選擇了多個文件,緩沖包含了當前目錄下被選擇文件的文件名。對於Explorer 風格對話框,目錄和文件名字符串是被NULL分開的,在文件名之後有一個額外的NULL。對於舊風格對話框,字符串是被空格分開的並且函數為帶有空格的文件名使用短文件名。你可以使用FindFirstFile函數在長短文件名之間轉換。如果用戶只選擇了一個文件,lpstrFile字符串在路徑和文件名之間沒有分隔。
  如果緩沖太小,函數返回FALSE並且CommDlgExtendedError函數返回FNERR_BUFFERTOOSMALL.。既然這樣,lpstrFile緩沖的首先兩個字節包含必需的大小(字節或字符)。
  nMaxFile
  指定lpstrFile緩沖的大小,以TCHARs為單位。對於ANSI版本,是字節的個數;對於 Unicode版本,是字符的個數。這個緩沖必須足夠存儲路徑和文件名字符串,包含結尾的null字符。如果緩沖太小,GetOpenFileName和GetSaveFileName函數返回假(FALSE)緩沖最小應該在256個字符長。
  經過調試觀察,我發現nMaxFile的初始值為260。但是我寫程序測試這個緩沖區的默認大小時,卻和這個初始值有矛盾。
  我的測試辦法是這樣的。首先在E盤建一個Txt Data的文件夾,然後創建40個空的txt文件。創建代碼如下:
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
  }
  然後我經過多次嘗試,發現在選擇0..txt,1.txt,2.xtxt,27.txt(共28個文件)時DoModal函數的返回值是IDOK,但是在選擇0..txt,1.txt,2.xtxt,27.txt,28.txt(共29個文件)時DoModal函數的返回值是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.在多字節字符集下也是這個數值。這裡需要注意的是CString::GetLength() 對於ASCII,返回字符串所占字節的數目,但如果是Unicode則實際上返回的是字符數而不是字節數
  那麼我初步斷定那個緩沖區的默認大小不是我調試觀察到的260,而是512。至於開頭如何解決那個問題,只需要定義一個更大的緩沖區,將lpstrFile指向這個緩沖區,重設nMaxFile的值即可,具體是:
  TCHAR szLargeBuf[4096]; // 定義一個臨時緩沖區
  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個讀不進來,設斷點檢查發現,如果想讀入8個文件,程序運行到 if (dlgOpen->DoModal()==IDOK)就跳出if語句,不執行下面的語句。難怪!但是究竟怎麼才能讀入多個文件那,我搜索DoModal函數源代碼,在文件DLGFILE.CPP中找到。發現函數中有個判斷語句 DWORD nOffset = lstrlen(m_ofn.lpstrFile)+1; ASSERT(nOffset <= m_ofn.nMaxFile);而nMaxFile最大文件數在構造函數中為空,如果不指定nMaxFile的值,判斷語句必然從DoModal函數中跳出。所以我在if (dlgOpen->DoModal()==IDOK)前面寫入下面語句CString str; dlgOpen->m_ofn.lpstrFile=str.GetBuffer(100000); str.ReleaseBuffer(); dlgOpen->m_ofn.nMaxFile = 5000;一切搞定! 但是要記住,m_ofn是不可見的,但是寫上去不會報錯。
CFileDialog如何實現文件多選
      CFileDialog類設置OFN_ALLOWMULTISELECT 標志可以實現文件多選功能,但是文件的數量是有限制的,如果要突破這個限制,就必須自己提供緩沖區。例子如下:
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]; //重新定義緩沖區大小         
memset(fileDlg.m_ofn.lpstrFile, 0, _MAX_PATH * MIN_FILE_NUMBER);  //初始化定義的緩沖區
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;     //最後別忘了釋放內存
  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved