简单的例子:
CString GetDirtoryPath()
{
BROWSEINFO bInfo = {0};
ITEMIDLIST *pItem;
CString file_path = _T("D:\\");
CString path_flat_char = _T("\\");
TCHAR dirtory[MAX_PATH];
//bInfo.hwndOwner = GetForegroundWindow(); // 父窗口;
//bInfo.hwndOwner = ::GetWindow(GetDesktopWindow(), GW_CHILD);
bInfo.pidlRoot = nullptr;
bInfo.pszDisplayName = dirtory;
bInfo.lpszTitle = _T("请选择目录 ");//strDlgTitle;
bInfo.ulFlags = BIF_RETURNONLYFSDIRS;
bInfo.lpfn = nullptr;
bInfo.lParam = 0;
bInfo.iImage = 0;
pItem = SHBrowseForFolder(&bInfo);
if (pItem == nullptr)
return file_path;
if (!SHGetPathFromIDList(pItem, dirtory)) // 将BROWSEINFO转换为文件系统路径。
return file_path;
file_path = dirtory;
/* the last character must be '\' */
CString lstr_Temp1 = path_flat_char;
CString lstr_Temp2 = path_flat_char;
lstr_Temp2 = file_path.Mid(file_path.GetLength() - 1, 1);
if (lstr_Temp1 != lstr_Temp2)
{
file_path += path_flat_char;
}
return file_path;
}
解释说明:
Visual C++(VC)中,BROWSEINFO结构中包含有用户选中目录的重要信息。
(1)BROWSEINFO结构
typedef struct_browseinfo
{
HWND hwndOwner; // 浏览文件夹对话框的父窗体句柄。
LPCITEMIDLIST pidlRoot; // ITEMIDLIST结构的地址,包含浏览时的初始根目录,而且只有被指定的目录和其子目录才显示在浏览文件夹对话框中。该成员变量可以是NULL,在此时桌面目录将被使用。
LPSTR pszDisplayName; // 用来保存用户选中的目录字符串的内存地址。该缓冲区的大小缺省是定义的MAX_PATH常量宏。
LPCSTR lpszTitle; // 该浏览文件夹对话框的显示文本,用来提示该浏览文件夹对话框的功能、作用和目的。
UINT ulFlags; //
BFFCALLBACK lpfn; // 应用程序定义的浏览对话框回调函数的地址。当对话框中的事件发生时,该对话框将调用回调函数。该参数可用为NULL。
LPARAM lParam; // 对话框传递给回调函数的一个参数指针。
int iImage; // 与选中目录相关的图像。该图像将被指定为系统图像列表中的索引值。
}BROWSEINFO,*PBROWSEINFO,*LPBROWSEINFO;
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消息给回调函数。
PIDLIST_ABSOLUTE SHBrowseForFolderA(
LPBROWSEINFOA lpbi
);
msdn:
https://docs.microsoft.com/en-us/windows/win32/api/shlobj_core/nf-shlobj_core-shbrowseforfoldera
BOOL SHGetPathFromIDListA(
PCIDLIST_ABSOLUTE pidl,
LPSTR pszPath
);
pidl
项目标识符列表的地址,该列表指定相对于名称空间的根目录(桌面)的文件或目录位置。
pszPath
接收文件系统路径的缓冲区的地址。此缓冲区的大小必须至少为MAX_PATH个字符。
返回值
如果成功,则返回TRUE;否则为FALSE。
备注
如果pidl参数指定的位置不是文件系统的一部分,则此功能将失败。
如果pidl参数指定了快捷方式,则pszPath将包含该快捷方式的路径,而不是该快捷方式的目标的路径。