当前位置: 首页 > news >正文

C++ MFC/BCG编程:文件对话框(CFileDialog、CFolderPickerDialog)

文章目录

  • CFileDialog 简介
  • CFileDialog 常用方法
  • CFileDialog 使用示例
  • CFolderPickerDialog 简介
  • CFolderPickerDialog 常用方法
  • CFolderPickerDialog 使用示例
  • CFolderPickerDialog 现代项目替换方案
  • CBCGPEdit启用“浏览”功能按钮


CFileDialog 简介

CFileDialog 是一个非常重要的类,用于显示标准的“打开文件”和“保存文件”对话框。它封装了Windows API中的文件对话框功能,使得开发者可以方便地实现文件的打开与保存操作。它支持两种模式:

  • 打开文件对话框(OPENFILENAME 结构):用于选择一个或多个要打开的文件。
  • 保存文件对话框:用于指定一个文件名用于保存数据。

CFileDialog 常用方法

构造函数详解

CFileDialog::CFileDialog(BOOL bOpenFileDialog,           // TRUE表示打开对话框,FALSE表示保存对话框LPCTSTR lpszDefExt = NULL,      // 默认扩展名LPCTSTR lpszFileName = NULL,    // 初始文件名DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT, // 标志位LPCTSTR lpszFilter = NULL,      // 文件类型过滤器CWnd* pParentWnd = NULL         // 父窗口指针
);
  • bOpenFileDialog TRUE 创建“打开”对话框,FALSE 创建“保存”对话框

  • lpszDefExt 默认文件扩展名,如 “txt”

  • lpszFileName 对话框中初始显示的文件名

  • dwFlags 控制对话框行为的标志,常用如

    • OFN_HIDEREADONLY 隐藏“只读”复选框
    • OFN_OVERWRITEPROMPT 保存时若文件已存在,提示是否覆盖
    • OFN_FILEMUSTEXIST 打开时要求文件必须存在
    • OFN_PATHMUSTEXIST 路径必须存在
    • OFN_ALLOWMULTISELECT 允许选择多个文件
    • OFN_NOCHANGEDIR 不改变当前工作目录
    • OFN_ENABLESIZING 允许调整对话框大小(Vista风格)
  • lpszFilter 文件过滤器

    CString filter =_T("图像文件 (*.jpg, *.png, *.dr)|*.jpg;*.jpeg;*.png;*.dr|")_T("JPEG 文件 (*.jpg)|*.jpg;*.jpeg|")_T("PNG 文件 (*.png)|*.png|")_T("DR 文件 (*.dr)|*.dr|");
    
  • pParentWnd 父窗口,通常为 this


  • DoModal() 显示对话框,返回 IDOK 或 IDCANCEL
  • GetPathName() 获取完整路径(含文件名)
  • GetFileName() 获取文件名(不含路径)
  • GetFileExt() 获取文件扩展名
  • GetFolderPath() 获取文件夹路径(多选时使用)
  • GetStartPosition() / GetNextPathName() 多文件选择时遍历文件列表
  • GetReadOnlyPref() 是否选择了“只读”选项

CFileDialog 使用示例

示例1:打开单个文件

void CMyDialog::OnOpenFile()
{CFileDialog fileDlg(TRUE, _T("txt"), NULL,OFN_FILEMUSTEXIST | OFN_HIDEREADONLY,_T("文本文件 (*.txt)|*.txt|所有文件 (*.*)|*.*||"));if (fileDlg.DoModal() == IDOK){CString filePath = fileDlg.GetPathName();CString fileName = fileDlg.GetFileName();AfxMessageBox(_T("选择的文件:") + filePath);}
}

示例2:保存文件(自动提示覆盖)

void CMyDialog::OnSaveFile()
{CFileDialog fileDlg(FALSE,_T("dat"),_T("mydata.dat"),OFN_OVERWRITEPROMPT,_T("数据文件 (*.dat)|*.dat|文本文件 (*.txt)|*.txt||"));if (fileDlg.DoModal() == IDOK){CString savePath = fileDlg.GetPathName();// 执行保存操作...}
}

示例3:选择多个文件

void CMyDialog::OnOpenMultipleFiles()
{CFileDialog fileDlg(TRUE,NULL,NULL,OFN_ALLOWMULTISELECT | OFN_FILEMUSTEXIST,_T("图像文件 (*.bmp;*.jpg)|*.bmp;*.jpg|所有文件 (*.*)|*.*||"));// 设置缓冲区大小以支持多选TCHAR szFileName[4096] = {0};fileDlg.m_ofn.lpstrFile = szFileName;fileDlg.m_ofn.nMaxFile = 4096;if (fileDlg.DoModal() == IDOK){POSITION pos = fileDlg.GetStartPosition();while (pos != NULL){CString path = fileDlg.GetNextPathName(pos);AfxMessageBox(_T("选中文件:") + path);}}
}

CFolderPickerDialog 简介

CFolderPickerDialog 是 MFC 中用于显示标准文件夹选择对话框的类。它封装了 Windows Shell API 中的 SHBrowseForFolder 函数,提供了一个树形结构的目录浏览界面,用户可以方便地选择一个文件夹。

CFolderPickerDialog 常用方法

构造函数详解

CFolderPickerDialog::CFolderPickerDialog(LPCTSTR lpszTitle = NULL,           // 对话框标题DWORD dwFlags = BIF_RETURNONLYFSDIRS | BIF_USENEWUI | BIF_EDITBOX,       // 浏览标志CWnd* pParentWnd = NULL,            // 父窗口int iImage = -1                     // 图标索引(可选)
);
  • lpszTitle 对话框顶部显示的提示文本,如 “请选择安装目录”
  • dwFlags 控制对话框行为和外观的标志,来自 SHBrowseForFolder
    这些标志定义在 CommCtrl.h 中,常用于 BROWSEINFO 结构:
    • BIF_RETURNONLYFSDIRS 只返回文件系统目录(推荐)
    • BIF_DONTGOBELOWDOMAIN 不展开域(网络环境)
    • BIF_STATUSTEXT 显示状态栏
    • BIF_USENEWUI 使用新版UI(带“新建文件夹”按钮等)
    • BIF_EDITBOX 允许用户手动输入路径(Vista及以上)
    • BIF_VALIDATE 验证用户输入的路径有效性
  • pParentWnd 父窗口指针,通常为 this
  • iImage 可选,指定在树节点前显示的图标索引

  • DoModal() 显示对话框,返回 IDOK 或 IDCANCEL
  • GetFolderPath() 获取用户选择的文件夹路径(CString)
  • GetBi() 获取内部 BROWSEINFO 结构(高级用法)

CFolderPickerDialog 使用示例

示例:选择一个文件夹用于保存数据

void CMyDialog::OnSelectFolder()
{CFolderPickerDialog folderDlg(_T("请选择备份目录"),BIF_RETURNONLYFSDIRS | BIF_USENEWUI | BIF_EDITBOX,this);if (folderDlg.DoModal() == IDOK){CString selectedPath = folderDlg.GetFolderPath();AfxMessageBox(_T("选择的文件夹:") + selectedPath);// 可在此进行后续操作,如设置为输出路径m_outputPath = selectedPath;UpdateData(FALSE);}
}

示例:限制选择“我的文档”或“桌面”
虽然 CFolderPickerDialog 本身不直接支持根目录限制,但可通过设置 BROWSEINFO::pidlRoot 实现(需调用 Shell API):

// 高级用法:限制根目录为“我的文档”
LPITEMIDLIST pidl;
if (SUCCEEDED(SHGetSpecialFolderLocation(NULL, CSIDL_MYDOCUMENTS, &pidl)))
{CFolderPickerDialog dlg(_T("选择文档目录"), BIF_RETURNONLYFSDIRS);dlg.GetBi().pidlRoot = pidl;if (dlg.DoModal() == IDOK){AfxMessageBox(dlg.GetFolderPath());}// 释放PIDLLPMALLOC pMalloc;if (SUCCEEDED(SHGetMalloc(&pMalloc))){pMalloc->Free(pidl);pMalloc->Release();}
}

CFolderPickerDialog 现代项目替换方案

在较新的 MFC 项目中,推荐使用 IFileDialog 接口 结合 FOS_PICKFOLDERS 标志来实现更现代化的文件夹选择器:

#include <shobjidl.h>HRESULT ShowFolderPicker(CString& outPath)
{IFileOpenDialog* pDialog = nullptr;HRESULT hr = CoCreateInstance(CLSID_FileOpenDialog, nullptr, CLSCTX_ALL, IID_IFileOpenDialog, (void**)&pDialog);if (SUCCEEDED(hr)){// 设置为仅选择文件夹pDialog->SetOptions(FOS_PICKFOLDERS);hr = pDialog->Show(NULL);if (SUCCEEDED(hr)){IShellItem* pItem = nullptr;hr = pDialog->GetResult(&pItem);if (SUCCEEDED(hr)){PWSTR pszPath = nullptr;hr = pItem->GetDisplayName(SIGDN_FILESYSPATH, &pszPath);if (SUCCEEDED(hr)){outPath = pszPath;CoTaskMemFree(pszPath);}pItem->Release();}}pDialog->Release();}return hr;
}

优点:支持缩略图、搜索、库、快速访问等现代功能。

CBCGPEdit启用“浏览”功能按钮

CBCGPEdit 提供了三种方式来启用“浏览”功能:

EnableBrowseButton():通用浏览按钮(可自定义行为)
EnableFileBrowseButton():专用于文件选择
EnableFolderBrowseButton():专用于文件夹选择
这三种方法都会在编辑框右侧添加一个按钮,点击后自动弹出对应的选择对话框,并将结果填入编辑框。

void EnableBrowseButton (BOOL bEnable = TRUE, LPCTSTR szLabel = _T("..."));void EnableFileBrowseButton (LPCTSTR lpszDefExt = NULL, LPCTSTR lpszFilter = NULL, LPCTSTR lpszInitialFolder = NULL, DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT);void EnableFolderBrowseButton (LPCTSTR lpszTitle = NULL, UINT ulFlags = BIF_RETURNONLYFSDIRS, LPCTSTR lpszInitialFolder = NULL);
  • lpszDefExt 默认文件扩展名,如 “txt”
  • lpszTitle 对话框顶部的提示文本,如 “请选择项目目录”
  • lpszFilter 文件类型过滤器,格式同 CFileDialog,如 `"文本文件 (*.txt)
  • lpszInitialFolder 初始打开的文件夹路径
http://www.lryc.cn/news/626671.html

相关文章:

  • 力扣48:旋转矩阵
  • 数据结构之排序大全(1)
  • 2.Shell脚本修炼手册之---创建第一个 Shell 脚本
  • 大模型入门实战 | 单卡 3090 十分钟完成 Qwen2.5-7B 首次微调
  • 电脑驱动免费更新? 这款驱动管理工具:一键扫更新,还能备份恢复,小白也会用~
  • c语言多任务处理(并发程序设计)
  • iOS App 混淆工具实战 医疗健康类 App 的安全与合规保护
  • Elasticsearch 写入全链路:从单机到集群
  • [系统架构设计师]面向服务架构设计理论与实践(十五)
  • [element-plus] el-tree 拖拽到其他地方,不拖拽到树上
  • Vue3 element ui 给表格的列设置背景颜色
  • 晨控EtherCAT设备分配IP操作手册
  • LWIP的TCP协议
  • 在 Golang 中复用 HTTP 连接
  • 26_基于深度学习的茶叶等级检测识别系统(yolo11、yolov8、yolov5+UI界面+Python项目源码+模型+标注好的数据集)
  • CTFshow系列——命令执行web38-40
  • Qt音乐播放器项目实践:本地持久化与边角问题处理
  • 小红书账号隔离:解决IP关联问题方案
  • 网络工程师考试重点:OSI七层模型TCP/IP四层模型解析
  • 【北京迅为】iTOP-4412精英版使用手册-第三十二章 网络通信-TCP套字节
  • yolo_RK3588系列(三)
  • 5.4 4pnpm 使用介绍
  • FreeRTOS---进阶知识1---列表的创建
  • SQL 中大于小于号的表示方法总结
  • Claude Code NPM 包发布命令
  • 内网安全——出网协议端口探测
  • Java开源工具Apache PDFBox(强大的处理 PDF文档工具:创建、读取、修改、解析和提取 PDF)
  • Apache ShenYu和Nacos之间的通信原理
  • 【Tech Arch】Apache Pig大数据处理的高效利器
  • Spring Boot 日志体系详解:配置与实战