VC弹出选择文件夹对话框和弹出本地指定文件夹窗口


VC编程中经常需要弹出选择文件夹对话框供用户选择目标文件夹

-----------------------------------------------------------------------------------------------------------------------------------

一种是用MFC提供的CFileDiglog类

(vs2010环境)

#include "stdafx.h"
#include <Windows.h>
#include "BaseFunc.h"

using namespace std;
using namespace BaseFunc;

unsigned BaseFunc::selFile( string &strFile,const string &strExt,bool bOpen )
{
    string strDir = "D:\\Downloads";//这里通过strFile解析目录,CFileDialog会自动记住
    string filename = "hi.txt";  //通过strFile解析文件名
    string filter = strExt + "文件 (*." + strExt + ")|*." + strExt + "||";
    string ext = "." + strExt;

    CFileDialog dlg(bOpen,ext.c_str(),filename.c_str(),OFN_READONLY|OFN_OVERWRITEPROMPT,filter.c_str(),NULL);
    dlg.GetOFN().lpstrInitialDir = strFile.c_str();// 默认目录

    if (dlg.DoModal())
    {
        strFile = dlg.GetPathName();
        return IDOK;
    }

    return IDCANCEL;
}


-----------------------------------------------------------------------------------------------------------------------------------

另一种是通过SHBrowseForFolder函数,可以通过回调函数设置标题及路径

 

基本实现的源码如下

 

  1. void CTestDlg::OnBtnTest()   
  2. {  
  3.     // TODO: Add your control notification handler code here  
  4.     TCHAR pszPath[MAX_PATH];  
  5.     BROWSEINFO bi;   
  6.     bi.hwndOwner      = this->GetSafeHwnd();  
  7.     bi.pidlRoot       = NULL;  
  8.     bi.pszDisplayName = NULL;   
  9.     bi.lpszTitle      = TEXT("请选择文件夹");   
  10.     bi.ulFlags        = BIF_RETURNONLYFSDIRS | BIF_STATUSTEXT;  
  11.     bi.lpfn           = NULL;   
  12.     bi.lParam         = 0;  
  13.     bi.iImage         = 0;   
  14.   
  15.     LPITEMIDLIST pidl = SHBrowseForFolder(&bi);  
  16.     if (pidl == NULL)  
  17.     {  
  18.         return;  
  19.     }  
  20.   
  21.     if (SHGetPathFromIDList(pidl, pszPath))  
  22.     {  
  23.         AfxMessageBox(pszPath);  
  24.     }  
  25. }  

 

这一般均能够满足要求,但有时还是需要在此基础上增强一些功能。

比如在弹出选择文件夹对话框时选中默认的文件夹,或在STATUSTEXT区域显示一些信息等等。这需要在BrowseCallbackProc回调函数中实现。具体实现的源码如下:

(具体BrowseCallBackFun回调函数的用法请参照MSDN)

 

  1. //选择文件夹对话框回调函数   
  2. int CALLBACK BrowseCallBackFun(HWND hwnd, UINT uMsg, LPARAM lParam, LPARAM lpData)  
  3. {  
  4.     switch(uMsg)  
  5.     {  
  6.     case BFFM_INITIALIZED:  //选择文件夹对话框初始化  
  7.         //设置默认路径为lpData即'D:\'   
  8.         ::SendMessage(hwnd, BFFM_SETSELECTION, TRUE, lpData);  
  9.         //在STATUSTEXT区域显示当前路径   
  10.         ::SendMessage(hwnd, BFFM_SETSTATUSTEXT, 0, lpData);  
  11.         //设置选择文件夹对话框的标题   
  12.         ::SetWindowText(hwnd, TEXT("请先设置个工作目录"));   
  13.         break;  
  14.     case BFFM_SELCHANGED:   //选择文件夹变更时  
  15.         {  
  16.             TCHAR pszPath[MAX_PATH];  
  17.             //获取当前选择路径   
  18.             SHGetPathFromIDList((LPCITEMIDLIST)lParam, pszPath);  
  19.             //在STATUSTEXT区域显示当前路径  
  20.             ::SendMessage(hwnd, BFFM_SETSTATUSTEXT, TRUE, (LPARAM)pszPath);  
  21.         }  
  22.         break;  
  23.     }  
  24.     return 0;  
  25. }  
  26.   
  27.   
  28. void CTestDlg::OnBtnTest()   
  29. {  
  30.     // TODO: Add your control notification handler code here  
  31.     TCHAR pszPath[MAX_PATH];  
  32.     BROWSEINFO bi;   
  33.     bi.hwndOwner      = this->GetSafeHwnd();  
  34.     bi.pidlRoot       = NULL;  
  35.     bi.pszDisplayName = NULL;   
  36.     bi.lpszTitle      = TEXT("请选择文件夹");   
  37.     bi.ulFlags        = BIF_RETURNONLYFSDIRS | BIF_STATUSTEXT;  
  38.     bi.lpfn           = BrowseCallBackFun;     //回调函数  
  39.     bi.lParam         = (LPARAM)TEXT("D:\\");  //传给回调函数的参数,设置默认路径  
  40.     bi.iImage         = 0;   
  41.       
  42.     LPITEMIDLIST pidl = SHBrowseForFolder(&bi);  
  43.     if (pidl == NULL)  
  44.     {  
  45.         return;  
  46.     }  
  47.       
  48.     if (SHGetPathFromIDList(pidl, pszPath))  
  49.     {  
  50.         AfxMessageBox(pszPath);  
  51.     }  
  52. }  

附图片:

 

当然也可以设置选择文件对话框的其他样式,比如使其具有新增文件夹的功能,可如下实现

bi.ulFlags        =        BIF_USENEWUI        

附图:(改图来源于www.VCKBASE.com,本人电脑上安装的是VC6,不支持BIF_USENEWUI,请在VC2003+上尝试)

具体请参照MSDN

 

定义

WINSHELLAPI LPITEMIDLIST WINAPI SHBrowseForFolder( LPBROWSEINFO lpbi); 

结构

BROWSEINFO的结构如下:
typedef struct _browseinfo {
HWND hwndOwner; // 父 窗口句柄
LPCITEMIDLIST pidlRoot; // 要显示的文件夹的根(Root)
LPTSTR pszDisplayName; // 保存被选取的文件夹路径的缓冲区
LPCTSTR lpszTitle; // 显示位于对话框左上部的标题
UINT ulFlags; // 指定对话框的外观和功能的标志
BFFCALLBACK lpfn; // 处理事件的回调函数
LPARAM lParam; // 应用程序传给 回调函数的参数
int iImage; // 保存被选取的文件夹的图片索引
BROWSEINFO, *PBROWSEINFO, *LPBROWSEINFO

成员变量

hwndOwner:浏览文件夹对话框的父 窗体句柄。
pidlRoot:ITEMIDLIST结构的地址,包含浏览时的初始根目录,而且只有被指定的目录和其子目录才显示在浏览文件夹对话框中。该 成员变量可以是NULL,在此时桌面目录将被使用。
pszDisplayName:用来保存用户选中的目录字符串的内存地址。该缓冲区的大小缺省是定义的MAX_PATH 常量宏。
lpszTitle:该浏览文件夹对话框对话框的显示文本,用来提示该浏览文件夹对话框的功能、作用和目的。
ulFlags:该标志位描述了对话框的选项。它可以为0,也可以是以下 常量的任意组合:
BIF_BROWSEFORCOMPUTER:返回计算机名。除非用户选中浏览器中的一个计算机名,否则该对话框中的“OK”按钮为灰色。
BIF_BROWSEFORPRINTER:返回打印机名。除非选中一个打印机名,否则“OK”按钮为灰色。
BIF_BROWSEINCLUDEFILES:浏览器将显示目录,同时也显示文件。
BIF_DONTGOBELOWDOMAIN:在树形视窗中,不包含域名底下的网络目录结构。
BIF_EDITBOX:浏览对话框中包含一个编辑框,在该编辑框中用户可以输入选中项的名字。
BIF_RETURNFSANCESTORS:返回文件系统的一个节点。仅仅当选中的是有意义的节点时,“OK”按钮才可以使用。
BIF_RETURNONLYFSDIRS:仅仅返回文件系统的目录。例如:在浏览文件夹对话框中,当选中任意一个目录时,该“OK”按钮可用,而当选中“ 我的电脑”或“ 网上邻居”等非有意义的节点时,“OK”按钮为灰色。
BIF_STATUSTEXT:在对话框中包含一个状态区域。通过给对话框发送消息使 回调函数设置状态文本。
BIF_VALIDATE:当没有BIF_EDITBOX标志位时,该标志位被忽略。如果用户在编辑框中输入的名字非法,浏览对话框将发送BFFM_VALIDATEFAILED消息给回调函数。
lpfn:应用程序定义的浏览对话框 回调函数的地址。当对话框中的事件发生时,该对话框将调用 回调函数。该参数可用为NULL。
lParam:对话框传递给 回调函数的一个参数 指针
iImage:与选中目录相关的图像。该图像将被指定为系统图像列表中的索引值。

 

------------------------------------------------------

若要建立可以新建文件夹的对话框:

在CPP开头加上:

#define BIF_NEWDIALOGSTYLE   0x40

#define BIF_USENEWUI (BIF_NEWDIALOGSTYLE|BIF_EDITBOX)

在风格中多加上 BIF_USENEWUI

bi.ulFlags        = BIF_RETURNONLYFSDIRS | BIF_STATUSTEXT |BIF_USENEWUI;

 

 

弹出指本地特殊指定的文件夹窗口,类似我们双击某个文件夹打开

文件目录转成LPITEMIDLIST

当用目录选择对话框时,可以用如下的结构打开。
typedef struct _browseinfo {

HWND hwndOwner;       // 父窗口句柄

LPCITEMIDLIST pidlRoot;      // 要显示的文件夾的根(Root)         

LPTSTR pszDisplayName;    // 保存被选取的文件夾路径的缓冲区

LPCTSTR lpszTitle;       // 显示位于对话框左上部的标题     

UINT ulFlags;           // 指定对话框的外观和功能的標志   

BFFCALLBACK lpfn;     // 处理事件的回调函数                

LPARAM lParam;       // 应用程序传给回调函数的参数        

int iImage;         // 保存被选取的文件夾的图片索引       

} BROWSEINFO, *PBROWSEINFO, *LPBROWSEINFO;

LPCITEMIDLIST pidlRoot; 这个就是 默认为桌面。

还有可以通过这种方式实现,如下:
LPMALLOC pMalloc;     
     if ( SUCCEEDED( SHGetSpecialFolderLocation (NULL, CSIDL_DRIVES, &pidl)))
     {
         SHELLEXECUTEINFO sei;
         ZeroMemory(&sei, sizeof(sei));
         sei.cbSize = sizeof(sei);
         sei.fMask = SEE_MASK_IDLIST;
         sei.lpIDList = pidl;
         sei.lpVerb = "open";
         sei.hwnd = AfxGetMainWnd()->GetSafeHwnd();
         sei.nShow = BIF_RETURNONLYFSDIRS;
         ShellExecuteEx(&sei);
         if (SUCCEEDED( SHGetMalloc (&pMalloc)))
         {
             pMalloc->Free ( pidl );
             pMalloc->Release();
         }
     }

其中://CSIDL_DRIVES是我的电脑
参数有好些,
CSIDL_BITBUCKET 回收站
CSIDL_CONTROLS 控制面板
CSIDL_DESKTOP Windows 桌面Desktop
CSIDL_DESKTOPDIRECTORY Desktop的目录
CSIDL_DRIVES 我的电脑
CSIDL_FONTS 字体目录
CSIDL_NETHOOD 网上邻居
CSIDL_NETWORK 网上邻居虚拟目录
CSIDL_PERSONAL 我的文档
CSIDL_PRINTERS 打印机
CSIDL_PROGRAMS 程序组
CSIDL_RECENT 最近打开的文档
CSIDL_SENDTO “发送到”菜单项
CSIDL_STARTMENU 任务条启动菜单项
CSIDL_STARTUP 启动目录
CSIDL_TEMPLATES 文档模板
其他的参数可以查阅MSDN。

现在像论坛上的朋友,他要实现的是要打开 如:f:,f:\\site等这样的目录。
很明显这样的字符串是不被支持的,所以必须要转成 LPCITEMIDLIST 这种结构的才支持。
但微软好像没有类似的函数实现这样的功能。于是就写了一个函数如下:

//文件目录转成LPITEMIDLIST
LPITEMIDLIST CTestBrowseDlg::ParsePidlFromPath(LPCSTR path)
{    
     OLECHAR szOleChar[MAX_PATH];    
     LPSHELLFOLDER IpsfDeskTop;    
     LPITEMIDLIST lpifq;    
     ULONG ulEaten, ulAttribs;    
     HRESULT hres;    
     SHGetDesktopFolder(&IpsfDeskTop);    
     MultiByteToWideChar(CP_ACP,MB_PRECOMPOSED,path,-1,szOleChar,sizeof(szOleChar));    
     hres = IpsfDeskTop ->ParseDisplayName(NULL, NULL, szOleChar, &ulEaten, &lpifq, &ulAttribs);    
     hres=IpsfDeskTop->Release( );        
     if(FAILED(hres))
         return NULL;
     return lpifq;
}

如这样写:    
方法1:
LPITEMIDLIST pidl;
    
     LPMALLOC pMalloc;
     pidl = ParsePidlFromPath("F:\\site");//请先确定f:下有这个文件夹
    
     SHELLEXECUTEINFO sei;
     ZeroMemory(&sei, sizeof(sei));
     sei.cbSize = sizeof(sei);
     sei.fMask = SEE_MASK_IDLIST;
     sei.lpIDList = pidl;
     sei.lpVerb = "open";
     sei.hwnd = AfxGetMainWnd()->GetSafeHwnd();
     sei.nShow = BIF_RETURNONLYFSDIRS;
     ShellExecuteEx(&sei);
     if (SUCCEEDED( SHGetMalloc (&pMalloc)))
     {
         pMalloc->Free ( pidl );
         pMalloc->Release();
     }
就可以打开F:\site

方法2:
     CString str;
     BROWSEINFO bi;
     char name[MAX_PATH];
     ZeroMemory(&bi,sizeof(BROWSEINFO));    
     bi.pidlRoot = ParsePidlFromPath("F:\");
     bi.hwndOwner=GetSafeHwnd();
     bi.pszDisplayName=name;
     bi.lpszTitle="S浏览文件夹";
     bi.ulFlags=BIF_RETURNONLYFSDIRS;
    
     LPITEMIDLIST idl=SHBrowseForFolder(&bi);
     if(idl==NULL)
         return;
就能打开f:

 

下面的例子中返回路径,如果没有选,返回"",选择了路径,则返回选择的路径。

char *GetPath(HWND   hWnd,char   *pBuffer) 

    BROWSEINFO   bf; 

    LPITEMIDLIST   lpitem; 

    memset(&bf,0,sizeof   BROWSEINFO); 

    bf.hwndOwner=hWnd; 

    bf.lpszTitle="选择路径"; 

    bf.ulFlags=BIF_RETURNONLYFSDIRS;     //属性你可自己选择 

    lpitem=SHBrowseForFolder(&bf); 

    if(lpitem==NULL)     //如果没有选择路径则返回   0 

        return   ""; 

  

    //如果选择了路径则复制路径,返回路径长度 

  

    SHGetPathFromIDList(lpitem,pBuffer); 

    return    pBuffer;

}  


免责声明!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系本站邮箱yoyou2525@163.com删除。



 
粤ICP备18138465号  © 2018-2025 CODEPRJ.COM