跟我一起玩Win32开发(20):浏览文件夹
2013-03-17 11:24
375 查看
最近忙于一些相当无聊的事情,还没忙完,不过,博客还是要写的,不然我头顶上会多了几块砖头。
在上一篇博文中,我们浏览了文件,今天我们也浏览一下目录,如何?
浏览目录我们同样有两个规矩,用托管类库的我就不说了,相信大家两分钟就能弄好。重点还是说说第二个规矩,用API方式。我们用到的是Shell函数,一提到Shell,可能有朋友会感到痛苦,因为它很多函数涉及到COM,不过,也没什么了,反正我的哲学理念是立足于实战。所以,我给大家总结了思路。
就拿今天我们要用到的SHBrowseForFolder函数来说吧,先说说这个函数的用途吧,一句话概括:显示一个对话框,让你可以选择一个目录。
COM对象需要先初始化,再用,用完了释放,根据这个思路,大的步骤如下:
1、CoInitialize(NULL); //初始化。
2、调用函数。
3、CoUninitialize(); //释放。
这个动作就好比你去租用商家的自行车去游玩,别人只是把资源(自行车)借给你用,你给押金(引用计数),等到你耍完了,再去把资源还给人家。
要调用SHBrowseForFolder函数,先初始化它的参数,也就是BROWSEINFO结构体,这个家伙初始化后,将地址传给SHBrowseForFolder的lpbi参数。BROWSEINFO结构你得设置以下成员。
之后就可以调用函数了。
由于PIDLIST也是COM对象资源,通过调用CoTaskMemFree函数将其释放。
现在,运行一下应用程序。
聪明的你会发现,虽然我们得到了选择的目录名字,但不是完整路径,通常我们是需要完整的路径的。这个也不难,我们再请出另一个函数——SHGetPathFromIDList,它可以帮我们得出完整的路径。
所以,我们把上面的代码修改一下。
现在再看看结果,这回你肯定满意了。
-------------- 【C++ Code】 -------------------------
在上一篇博文中,我们浏览了文件,今天我们也浏览一下目录,如何?
浏览目录我们同样有两个规矩,用托管类库的我就不说了,相信大家两分钟就能弄好。重点还是说说第二个规矩,用API方式。我们用到的是Shell函数,一提到Shell,可能有朋友会感到痛苦,因为它很多函数涉及到COM,不过,也没什么了,反正我的哲学理念是立足于实战。所以,我给大家总结了思路。
就拿今天我们要用到的SHBrowseForFolder函数来说吧,先说说这个函数的用途吧,一句话概括:显示一个对话框,让你可以选择一个目录。
COM对象需要先初始化,再用,用完了释放,根据这个思路,大的步骤如下:
1、CoInitialize(NULL); //初始化。
2、调用函数。
3、CoUninitialize(); //释放。
这个动作就好比你去租用商家的自行车去游玩,别人只是把资源(自行车)借给你用,你给押金(引用计数),等到你耍完了,再去把资源还给人家。
要调用SHBrowseForFolder函数,先初始化它的参数,也就是BROWSEINFO结构体,这个家伙初始化后,将地址传给SHBrowseForFolder的lpbi参数。BROWSEINFO结构你得设置以下成员。
PIDLIST_ABSOLUTE pIDLIST;//函数返回的值存在这里 BROWSEINFO brsInfo; WCHAR fdpath[MAX_PATH]; //接收选择目录的字符数组 。。。。。。。。。。。。。。。。。。。。 brsInfo.hwndOwner = hWnd; //拥有者窗口 brsInfo.pidlRoot = NULL; //根目录为桌面 brsInfo.pszDisplayName = fdpath; //返回的被选择的路径 brsInfo.lpszTitle = L"请选择目录"; //对话框标题 brsInfo.ulFlags= BIF_RETURNONLYFSDIRS | BIF_DONTGOBELOWDOMAIN | BIF_NEWDIALOGSTYLE; //不显示网络路径 brsInfo.lpfn = NULL; //没有回调 brsInfo.lParam = NULL; //没有额外参数
之后就可以调用函数了。
// 初始化COM组件,用完后记得释放 CoInitialize(NULL); pIDLIST = SHBrowseForFolder(&brsInfo); if(pIDLIST) { //取出对话框中用来显示结果的控件 HWND hStatic = GetDlgItem(hfDlg, IDC_DISPLAY); SendMessage(hStatic,WM_SETTEXT, NULL, (LPARAM)fdpath); } CoUninitialize(); // 释放IDLIST CoTaskMemFree((LPVOID)pIDLIST);
由于PIDLIST也是COM对象资源,通过调用CoTaskMemFree函数将其释放。
现在,运行一下应用程序。
聪明的你会发现,虽然我们得到了选择的目录名字,但不是完整路径,通常我们是需要完整的路径的。这个也不难,我们再请出另一个函数——SHGetPathFromIDList,它可以帮我们得出完整的路径。
所以,我们把上面的代码修改一下。
if(pIDLIST) { //得到完整路径 SHGetPathFromIDList(pIDLIST, fullPath); //取出对话框中用来显示结果的控件 HWND hStatic = GetDlgItem(hfDlg, IDC_DISPLAY); SendMessage(hStatic, WM_SETTEXT, NULL, (LPARAM)fullPath); //SendMessage(hStatic,WM_SETTEXT, NULL, (LPARAM)fdpath); }
现在再看看结果,这回你肯定满意了。
-------------- 【C++ Code】 -------------------------
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { int wmId, wmEvent; PAINTSTRUCT ps; HDC hdc; PIDLIST_ABSOLUTE pIDLIST;//函数返回的值存在这里 BROWSEINFO brsInfo; WCHAR fdpath[MAX_PATH]; //接收选择目录的字符数组 WCHAR fullPath[MAX_PATH]; //完整路径 static HWND hfDlg; switch (message) { case WM_CREATE: hfDlg = CreateDialog(hInst, MAKEINTRESOURCE(IDD_FORMVIEW), hWnd, NULL); ShowWindow(hfDlg,SW_SHOW); break; case WM_COMMAND: wmId = LOWORD(wParam); wmEvent = HIWORD(wParam); // 分析菜单选择: switch (wmId) { case IDM_ABOUT: DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About); break; case IDM_EXIT: DestroyWindow(hWnd); break; case IDM_BROWSFOLDER: brsInfo.hwndOwner = hWnd; //拥有者窗口 brsInfo.pidlRoot = NULL; //根目录为桌面 brsInfo.pszDisplayName = fdpath; //返回的被选择的路径 brsInfo.lpszTitle = L"请选择目录"; //对话框标题 brsInfo.ulFlags= BIF_RETURNONLYFSDIRS | BIF_DONTGOBELOWDOMAIN | BIF_NEWDIALOGSTYLE; //不显示网络路径 brsInfo.lpfn = NULL; //没有回调 brsInfo.lParam = NULL; //没有额外参数 // 初始化COM组件,用完后记得释放 CoInitialize(NULL); pIDLIST = SHBrowseForFolder(&brsInfo); if(pIDLIST) { //得到完整路径 SHGetPathFromIDList(pIDLIST, fullPath); //取出对话框中用来显示结果的控件 HWND hStatic = GetDlgItem(hfDlg, IDC_DISPLAY); SendMessage(hStatic, WM_SETTEXT, NULL, (LPARAM)fullPath); //SendMessage(hStatic,WM_SETTEXT, NULL, (LPARAM)fdpath); } CoUninitialize(); // 释放IDLIST CoTaskMemFree((LPVOID)pIDLIST); break; default: return DefWindowProc(hWnd, message, wParam, lParam); } break; case WM_PAINT: hdc = BeginPaint(hWnd, &ps); // TODO: 在此添加任意绘图代码... EndPaint(hWnd, &ps); break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hWnd, message, wParam, lParam); } return 0; }
相关文章推荐
- 跟我一起玩Win32开发(19):浏览和打开文件
- win32 api 调用浏览文件 保存文件对话框 以及浏览文件夹对话框
- 跟我一起玩Win32开发(2):完整的开发流程
- 跟我一起玩Win32开发(1):关于C++的几个要点
- 跟我一起玩Win32开发(25):监视剪贴板
- 跟我一起玩Win32开发(11):使用控件——先来耍一下按钮
- win32 api 调用浏览文件 保存文件对话框 以及浏览文件夹对话框
- 跟我一起玩Win32开发(24):计时器
- 跟我一起玩Win32开发(18):使用对话框的两个技巧
- 在win32中实现浏览文件夹功能与删除文件功能
- 跟我一起玩Win32开发(25):监视剪贴板
- 跟我一起玩Win32开发(25):监视剪贴板
- 跟我一起玩Win32开发(5):具有单选标记的菜单
- 跟我一起玩Win32开发(3):窗口的重绘
- 跟我一起玩Win32开发(8):绘图(A)
- 跟我一起玩Win32开发(14):用对话框作为主窗口
- 跟我一起玩Win32开发(23):渐变颜色填充
- 跟我一起玩Win32开发(15):ListView控件
- 跟我一起玩Win32开发(4):创建菜单
- 跟我一起玩Win32开发(9):绘图(B)