https://blog.csdn.net/shanshangyouzhiyangm/article/details/51996497
BROWSEINFO结构
typedef struct_browseinfo
{
HWND hwndOwner;
LPCITEMIDLIST pidlRoot;
LPSTR pszDisplayName;
LPCSTR lpszTitle;
UINT ulFlags;
BFFCALLBACK lpfn;
LPARAM lParam;
int iImage;
} BROWSEINFO, *PBROWSEINFO, *LPBROWSEINFO;
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
hwndOwner: 浏览文件夹对话框的父窗体句柄
pidlRoot:ITEMIDLIST结构的地址,包含浏览时的初始根目录,而且只有被指定的目录和其子目录才显示在浏览文件夹对话框中。该成员变量可以是NULL,在此时桌面目录将被使用。
pszDisplayName:用来保存用户选中的目录字符串的内存地址。该缓冲区的大小缺省是定义的MAX_PATH(260)常量宏。
lpszTitle:该浏览文件夹对话框的显示文本,用来提示该浏览文件夹对话框的功能、作用和目的。
ulFlags:该标志位描述了对话框的选项。它可以为0,也可以是以下常量的任意组合。
lpfn:应用程序定义的浏览对话框回调函数的地址。当对话框中的事件发生时,该对话框将调用回调函数。该参数可用为NULL。
lParam:对话框传递给回调函数的一个参数指针。
iImage:与选中目录相关的图像。该图像将被指定为系统图像列表中的索引值。
ulFlags使用说明:
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消息给回调函数。
注意:所有标志可以配合使用例如:
BIF_EDITBOX | BIF_RETURNFSANCESTORS
栗子:
// 回调函数
static int CALLBACK BrowseCallbackProc(HWND hWnd, UINT uMsg, LPARAM lParam, LPARAM lpData)
{
switch(uMsg)
{
case BFFM_INITIALIZED:// 初始化消息BFFM_INITIALIZED
CString tempPath = AfxGetApp()->GetProfileString(_T("Recently"), _T("WaveSavePath"), _T(""));
::SendMessage(hWnd, BFFM_SETSELECTION, TRUE, (LPARAM)tempPath.GetBuffer(tempPath.GetLength()));
break;
case BFFM_SELCHANGED:// 选择路径变化,BFFM_SELCHANGED
char curr[MAX_PATH];
CString curr;
SHGetPathFromIDList((LPCITEMIDLIST)lParam,(LPWSTR)(LPCTSTR)curr);
::SendMessage(hwnd,BFFM_SETSTATUSTEXT,0,/*(LPARAM)*/(LPARAM)(LPCTSTR)curr);
break;
default:
break;
}
return 0;
}
// 创建一个菜单
CMenu menu;
menu.LoadMenu(IDR_POPUP_SAVE_WAVE);
CMenu *contextMenu = menu.GetSubMenu(0);
ClientToScreen(&point);
int choice = contextMenu->TrackPopupMenu(TPM_RETURNCMD, point.x, point.y, this);
// 若选中“保存波形”这个选项
if (choice == ID_SAVE_WAVE)
{
BROWSEINFO bi;// 选择路径的对话框
bi.hwndOwner = NULL;
bi.pidlRoot = NULL;
bi.pszDisplayName = NULL;
bi.lpszTitle = _T("请选择用于保存的文件夹");
bi.ulFlags = 0;
bi.lpfn = BrowseCallbackProc;
bi.iImage = 0;
//调用显示选择对话框,并返回选定的路径(格式为LPCITEMIDLIST)
//注意下 这个函数会分配内存 但不会释放 需要手动释放
LPCITEMIDLIST pidl = SHBrowseForFolder(&bi);
if (pidl)// 若为空,则说明上面的函数调用错误
{
TCHAR folderName[MAX_PATH];
//将文件夹路径放到folderName中
SHGetPathFromIDList(pidl, folderName);
//保存选定的路径到注册表中
AfxGetApp()->WriteProfileString(_T("Recently"), _T("WaveSavePath"), folderName);
CoTaskMemFree((LPVOID)pidl); //释放pIDL所指向内存空间
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
说明:
以上程序实现的功能为:
右键弹出菜单 -> 保存波形,点击之后弹出一个路径选择的对话框。对话框的初始默认路径从注册表 [Recently] [WaveSavePath] 读取,并且点击确认后再将选定的路径保存到注册表对应的位置中。
另外:
系统默认的是如果该路径下有子路径,则默认展开子路径,但是其实大部分时间我们是不需要的。下面是在回调函数中实现的不展开子路径的代码:
static int CALLBACK BrowseCallbackProc(HWND hWnd, UINT uMsg, LPARAM , LPARAM lpData)
{
if(uMsg == BFFM_INITIALIZED)
{
CTreeCtrl treePath;
HTREEITEM hItemSel;
::SendMessage(hWnd, BFFM_SETSELECTION, TRUE, lpData);
treePath.SubclassWindow(::GetDlgItem(hWnd, 0x3741));
hItemSel = treePath.GetSelectedItem();
treePath.Expand(hItemSel, TVE_COLLAPSE);
treePath.UnsubclassWindow();
}
return 0;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
<link rel="stylesheet" href="https://csdnimg.cn/release/phoenix/template/css/markdown_views-ea0013b516.css">
</div>