您的位置:首页 > 其它

路径(文件夹)选择对话框

2012-01-04 22:49 507 查看
一.首先要为SHBrowseForFolder准备一个结构体BROWSEINFO

typedef struct _browseinfoW {

HWND hwndOwner;

PCIDLIST_ABSOLUTE pidlRoot;

LPWSTR pszDisplayName; // Return display name of item selected.

LPCWSTR lpszTitle; // text to go in the banner over the tree.

UINT ulFlags; // Flags that control the return stuff

BFFCALLBACK lpfn;

LPARAM lParam; // extra info that's passed back in callbacks

int iImage; // output var: where to return the Image index.

} BROWSEINFOW, *PBROWSEINFOW, *LPBROWSEINFOW;

二.获取PIDL

在BROWSEINFOW结构体中,pidlRoot是最关键的,需要调用一些额外的函数来获取PIDL,然后给pidlRoot赋值

1.获取特殊的目录地址

在Windows中有很多特殊的的目录地址,如我的电脑,控件面板,我的图片等,如下定义

#define CSIDL_DESKTOP 0x0000 // <desktop>

#define CSIDL_INTERNET 0x0001 // Internet Explorer (icon on desktop)

#define CSIDL_PROGRAMS 0x0002 // Start Menu\Programs

#define CSIDL_CONTROLS 0x0003 // My Computer\Control Panel

#define CSIDL_PRINTERS 0x0004 // My Computer\Printers

#define CSIDL_PERSONAL 0x0005 // My Documents

#define CSIDL_FAVORITES 0x0006 // <user name>\Favorites

#define CSIDL_STARTUP 0x0007 // Start Menu\Programs\Startup

#define CSIDL_RECENT 0x0008 // <user name>\Recent

我们可以通过SHGetSpecialFolderLocation来填充一个LPITEMIDLIST,

BROWSEINFO bi;

bi.hwndOwner = hDlg;

TCHAR szTitle[MAX_PATH] = {0};

TCHAR szPath[MAX_PATH] = {0};

TCHAR szDisplay[MAX_PATH] = {0};

LPITEMIDLIST pidl = NULL;

SHGetSpecialFolderLocation(NULL, CSIDL_DESKTOP, &pidl);

bi.pidlRoot = pidl;

然后调用SHBrowseForFolder(&bi);就可以出现以下对话框

2.使用IShellFolder获取目录地址

如果不是特殊目录的,可以使用IShellFolder的ParseDisplayName方法,来获取PIDL

首先要调用SHGetDesktopFolder函数来获取IShellFolder,ShellFolder代表桌面,表示是根目录,根目录总是有办法找到任意一个子目录的
然后调用ParseDisplayName方法来解析目录,最终得到LPITEMIDLIST。
下面的步骤还是一样把LPITEMIDLIST填充到BROWSEINFO的pidlRoot 字段,调用SHBrowseForFolder方法

SHParseDisplayName方法提供了一个便利

HRESULT SHPathToPIDL(LPCTSTR szPath, LPITEMIDLIST* ppidl)

{

LPSHELLFOLDER pShellFolder = NULL;

OLECHAR wszPath[MAX_PATH] = {0};

ULONG nCharsParsed = 0;

// Get an IShellFolder interface pointer

HRESULT hr = SHGetDesktopFolder(&pShellFolder);

if(FAILED(hr))

return hr;

// Convert the path name to Unicode

MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, szPath, -1, wszPath, MAX_PATH);

// Call ParseDisplayName() to do the job

hr = pShellFolder->ParseDisplayName(NULL, NULL, wszPath, &nCharsParsed, ppidl, NULL);

// Clean up

pShellFolder->Release();

return hr;

}

3.获取选择的目录

当选中目录点击确定之后,就需要获取一个文件系统路径,可以使用SHGetPathFromIDList函数

SHGetPathFromIDList(pidlFolder, szPath);

szPath就是我们想要的结果

4.BROWSEINFO的回调方法

在.net中会声明很多的事件,若Initialized,SelectedChanged等,c++是采用消息的机制,根据消息的不同进行不同处理,当然这些消息都是预先定义好的.

#define BFFM_INITIALIZED 1

#define BFFM_SELCHANGED 2

#define BFFM_VALIDATEFAILEDA 3 // lParam:szPath ret:1(cont),0(EndDialog)

#define BFFM_VALIDATEFAILEDW 4 // lParam:wzPath ret:1(cont),0(EndDialog)

#define BFFM_IUNKNOWN 5 // provides IUnknown to client. lParam: IUnknown*

示例如下:

bi.lpfn = BrowseCallbackProc;

int CALLBACK BrowseCallbackProc(HWND hwnd, UINT uMsg, LPARAM lParam, LPARAM dwData)

{

switch(uMsg)

{

case BFFM_INITIALIZED:

{

}

break;

case BFFM_SELCHANGED:

{

}

break;

case BFFM_VALIDATEFAILED:

return 1;

}

return 0;

}

三.文件夹选项

如下图,应该是很熟悉

SHGetSettings函数可以获取这些选项的信息,存在结构体SHELLFLAGSTATE当中

typedef struct {

BOOL fShowAllObjects : 1;

BOOL fShowExtensions : 1;

BOOL fNoConfirmRecycle : 1;

BOOL fShowSysFiles : 1;

BOOL fShowCompColor : 1;

BOOL fDoubleClickInWebView : 1;

BOOL fDesktopHTML : 1;

BOOL fWin95Classic : 1;

BOOL fDontPrettyPath : 1;

BOOL fShowAttribCol : 1;

BOOL fMapNetDrvBtn : 1;

BOOL fShowInfoTip : 1;

BOOL fHideIcons : 1;

#if (NTDDI_VERSION >= NTDDI_VISTA)

BOOL fAutoCheckSelect: 1;

BOOL fIconsOnly: 1;

UINT fRestFlags : 1; // when adding additional flags keep SHELLSTATE and SHGetSettings in sync.

#else

UINT fRestFlags : 3; // when adding additional flags keep SHELLSTATE and SHGetSettings in sync.

#endif

} SHELLFLAGSTATE, *LPSHELLFLAGSTATE;

必须输入想要获取标识才行,不然就会是默认值(难道是为了性能,为何如此,太麻烦)

SHELLFLAGSTATE sfs;

SHGetSettings(&sfs, SSF_DESKTOPHTML | SSF_SHOWALLOBJECTS |

SSF_MAPNETDRVBUTTON | SSF_SHOWATTRIBCOL | SSF_SHOWEXTENSIONS);

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:与选中目录相关的图像。该图像将被指定为系统图像列表中的索引值。

  调用例子如下:

  BROWSEINFO bi;

TCHAR buffer[MAX_PATH];

ZeroMemory(buffer, MAX_PATH);

bi.hwndOwner = GetSafeHwnd();

bi.pidlRoot = NULL;

bi.pszDisplayName = buffer;

bi.lpszTitle = _T("另存为:");

bi.ulFlags = BIF_EDITBOX | BIF_NEWDIALOGSTYLE /*| OFN_EXPLORER*/;

bi.lpfn = NULL;

bi.lParam = 0;

bi.iImage = 0;

LPITEMIDLIST pList = NULL;

if((pList=SHBrowseForFolder(&bi)) == NULL)

{

return;

}

TCHAR path[MAX_PATH];

ZeroMemory(path, MAX_PATH);

SHGetPathFromIDList(pList, path);

CString csFilePath(path);

if (csFilePath.IsEmpty())

{

return;

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: