我打算使用如下项目:https://github.com/scottwis/OpenFileOrFolderDialog
然而,存在一个问题:它使用了GetOpenFileName
函数和OPENFILENAME
结构体。OPENFILENAME
有一个名为templateID
的成员变量,表示对话框模板的标识符。该项目还包含res1.rc
文件和模板化的对话框初始化,但我无法弄清如何将此文件附加到我的C#项目中。
是否有更好的方式来使用OpenFileDialog
选择文件夹?
我打算使用如下项目:https://github.com/scottwis/OpenFileOrFolderDialog
然而,存在一个问题:它使用了GetOpenFileName
函数和OPENFILENAME
结构体。OPENFILENAME
有一个名为templateID
的成员变量,表示对话框模板的标识符。该项目还包含res1.rc
文件和模板化的对话框初始化,但我无法弄清如何将此文件附加到我的C#项目中。
是否有更好的方式来使用OpenFileDialog
选择文件夹?
基本上,您需要使用FolderBrowserDialog
类:
提示用户选择文件夹。 该类不能被继承。
示例:
using(var fbd = new FolderBrowserDialog())
{
DialogResult result = fbd.ShowDialog();
if (result == DialogResult.OK && !string.IsNullOrWhiteSpace(fbd.SelectedPath))
{
string[] files = Directory.GetFiles(fbd.SelectedPath);
System.Windows.Forms.MessageBox.Show("Files found: " + files.Length.ToString(), "Message");
}
}
如果您在使用WPF,则需要添加对System.Windows.Forms
的引用。
您还需要添加using System.IO
以使用Directory
类。
new CommonOpenFileDialog { IsFolderPicker = true }
。请注意,我已经尽力使翻译通俗易懂,并保持了原文的意思,没有提供额外的解释。 - ANevesFolderBrowserDialog
在幕后使用 SHBrowseForFolder
。该函数的文档明确指出:“对于 Windows Vista 或更高版本,建议您使用带有 FOS_PICKFOLDERS
选项的 IFileDialog
而不是 SHBrowseForFolder
函数。这将在选择文件夹模式下使用“打开文件”对话框,并且是首选实现。”因此,除了所有可用性问题之外,自从2006年发布Windows Vista以来,它已经不是推荐的解决方案了! - HerohtarFolderBrowserDialog
的用户的提示,Microsoft曾经发布了一个名为WindowsAPICodePack的API,其中包含了一个称为CommonOpenFileDialog
的有用对话框,可以设置为IsFolderPicker
模式。该API可在Microsoft的NuGet包中获得。CommonOpenFileDialog
的全部内容。(NuGet会处理依赖项)Install-Package Microsoft.WindowsAPICodePack-Shell
对于包含(include)语句:
using Microsoft.WindowsAPICodePack.Dialogs;
使用方法:
CommonOpenFileDialog dialog = new CommonOpenFileDialog();
dialog.InitialDirectory = "C:\\Users";
dialog.IsFolderPicker = true;
if (dialog.ShowDialog() == CommonFileDialogResult.Ok)
{
MessageBox.Show("You selected: " + dialog.FileName);
}
CommonOpenFileDialog
之前,您需要通过NuGet安装Microsoft.WindowsAPICodePack.Shell
包。 - smwikipediaWindowsAPICodePack-Shell
。 - NucS这是一个纯C#版本,不需要NuGet,适用于所有版本的.NET(包括.NET Core、.NET 5、WPF、Winforms等),并使用Windows Vista(及更高版本)的IFileDialog接口和FOS_PICKFOLDERS选项,因此具有良好的文件夹选择器 Windows 标准 UI。
更新2023/3/17: 现在该类支持多选。
我还添加了WPF的 Window
类型支持,但这是可选的,可以删除带有WPF标记的行。
用法:
var dlg = new FolderPicker();
dlg.InputPath = @"c:\windows\system32";
if (dlg.ShowDialog() == true)
{
MessageBox.Show(dlg.ResultPath);
}
代码:
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Runtime.InteropServices;
using System.Runtime.InteropServices.ComTypes;
using System.Windows;
using System.Windows.Interop;
public class FolderPicker
{
private readonly List<string> _resultPaths = new List<string>();
private readonly List<string> _resultNames = new List<string>();
public IReadOnlyList<string> ResultPaths => _resultPaths;
public IReadOnlyList<string> ResultNames => _resultNames;
public string ResultPath => ResultPaths.FirstOrDefault();
public string ResultName => ResultNames.FirstOrDefault();
public virtual string InputPath { get; set; }
public virtual bool ForceFileSystem { get; set; }
public virtual bool Multiselect { get; set; }
public virtual string Title { get; set; }
public virtual string OkButtonLabel { get; set; }
public virtual string FileNameLabel { get; set; }
protected virtual int SetOptions(int options)
{
if (ForceFileSystem)
{
options |= (int)FOS.FOS_FORCEFILESYSTEM;
}
if (Multiselect)
{
options |= (int)FOS.FOS_ALLOWMULTISELECT;
}
return options;
}
// for WPF support
public bool? ShowDialog(Window owner = null, bool throwOnError = false)
{
owner = owner ?? Application.Current?.MainWindow;
return ShowDialog(owner != null ? new WindowInteropHelper(owner).Handle : IntPtr.Zero, throwOnError);
}
// for all .NET
public virtual bool? ShowDialog(IntPtr owner, bool throwOnError = false)
{
var dialog = (IFileOpenDialog)new FileOpenDialog();
if (!string.IsNullOrEmpty(InputPath))
{
if (CheckHr(SHCreateItemFromParsingName(InputPath, null, typeof(IShellItem).GUID, out var item), throwOnError) != 0)
return null;
dialog.SetFolder(item);
}
var options = FOS.FOS_PICKFOLDERS;
options = (FOS)SetOptions((int)options);
dialog.SetOptions(options);
if (Title != null)
{
dialog.SetTitle(Title);
}
if (OkButtonLabel != null)
{
dialog.SetOkButtonLabel(OkButtonLabel);
}
if (FileNameLabel != null)
{
dialog.SetFileName(FileNameLabel);
}
if (owner == IntPtr.Zero)
{
owner = Process.GetCurrentProcess().MainWindowHandle;
if (owner == IntPtr.Zero)
{
owner = GetDesktopWindow();
}
}
var hr = dialog.Show(owner);
if (hr == ERROR_CANCELLED)
return null;
if (CheckHr(hr, throwOnError) != 0)
return null;
if (CheckHr(dialog.GetResults(out var items), throwOnError) != 0)
return null;
items.GetCount(out var count);
for (var i = 0; i < count; i++)
{
items.GetItemAt(i, out var item);
CheckHr(item.GetDisplayName(SIGDN.SIGDN_DESKTOPABSOLUTEPARSING, out var path), throwOnError);
CheckHr(item.GetDisplayName(SIGDN.SIGDN_DESKTOPABSOLUTEEDITING, out var name), throwOnError);
if (path != null || name != null)
{
_resultPaths.Add(path);
_resultNames.Add(name);
}
}
return true;
}
private static int CheckHr(int hr, bool throwOnError)
{
if (hr != 0 && throwOnError) Marshal.ThrowExceptionForHR(hr);
return hr;
}
[DllImport("shell32")]
private static extern int SHCreateItemFromParsingName([MarshalAs(UnmanagedType.LPWStr)] string pszPath, IBindCtx pbc, [MarshalAs(UnmanagedType.LPStruct)] Guid riid, out IShellItem ppv);
[DllImport("user32")]
private static extern IntPtr GetDesktopWindow();
#pragma warning disable IDE1006 // Naming Styles
private const int ERROR_CANCELLED = unchecked((int)0x800704C7);
#pragma warning restore IDE1006 // Naming Styles
[ComImport, Guid("DC1C5A9C-E88A-4dde-A5A1-60F82A20AEF7")] // CLSID_FileOpenDialog
private class FileOpenDialog { }
[ComImport, Guid("d57c7288-d4ad-4768-be02-9d969532d960"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
private interface IFileOpenDialog
{
[PreserveSig] int Show(IntPtr parent); // IModalWindow
[PreserveSig] int SetFileTypes(); // not fully defined
[PreserveSig] int SetFileTypeIndex(int iFileType);
[PreserveSig] int GetFileTypeIndex(out int piFileType);
[PreserveSig] int Advise(); // not fully defined
[PreserveSig] int Unadvise();
[PreserveSig] int SetOptions(FOS fos);
[PreserveSig] int GetOptions(out FOS pfos);
[PreserveSig] int SetDefaultFolder(IShellItem psi);
[PreserveSig] int SetFolder(IShellItem psi);
[PreserveSig] int GetFolder(out IShellItem ppsi);
[PreserveSig] int GetCurrentSelection(out IShellItem ppsi);
[PreserveSig] int SetFileName([MarshalAs(UnmanagedType.LPWStr)] string pszName);
[PreserveSig] int GetFileName([MarshalAs(UnmanagedType.LPWStr)] out string pszName);
[PreserveSig] int SetTitle([MarshalAs(UnmanagedType.LPWStr)] string pszTitle);
[PreserveSig] int SetOkButtonLabel([MarshalAs(UnmanagedType.LPWStr)] string pszText);
[PreserveSig] int SetFileNameLabel([MarshalAs(UnmanagedType.LPWStr)] string pszLabel);
[PreserveSig] int GetResult(out IShellItem ppsi);
[PreserveSig] int AddPlace(IShellItem psi, int alignment);
[PreserveSig] int SetDefaultExtension([MarshalAs(UnmanagedType.LPWStr)] string pszDefaultExtension);
[PreserveSig] int Close(int hr);
[PreserveSig] int SetClientGuid(); // not fully defined
[PreserveSig] int ClearClientData();
[PreserveSig] int SetFilter([MarshalAs(UnmanagedType.IUnknown)] object pFilter);
[PreserveSig] int GetResults(out IShellItemArray ppenum);
[PreserveSig] int GetSelectedItems([MarshalAs(UnmanagedType.IUnknown)] out object ppsai);
}
[ComImport, Guid("43826D1E-E718-42EE-BC55-A1E261C37BFE"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
private interface IShellItem
{
[PreserveSig] int BindToHandler(); // not fully defined
[PreserveSig] int GetParent(); // not fully defined
[PreserveSig] int GetDisplayName(SIGDN sigdnName, [MarshalAs(UnmanagedType.LPWStr)] out string ppszName);
[PreserveSig] int GetAttributes(); // not fully defined
[PreserveSig] int Compare(); // not fully defined
}
[ComImport, Guid("b63ea76d-1f85-456f-a19c-48159efa858b"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
private interface IShellItemArray
{
[PreserveSig] int BindToHandler(); // not fully defined
[PreserveSig] int GetPropertyStore(); // not fully defined
[PreserveSig] int GetPropertyDescriptionList(); // not fully defined
[PreserveSig] int GetAttributes(); // not fully defined
[PreserveSig] int GetCount(out int pdwNumItems);
[PreserveSig] int GetItemAt(int dwIndex, out IShellItem ppsi);
[PreserveSig] int EnumItems(); // not fully defined
}
#pragma warning disable CA1712 // Do not prefix enum values with type name
private enum SIGDN : uint
{
SIGDN_DESKTOPABSOLUTEEDITING = 0x8004c000,
SIGDN_DESKTOPABSOLUTEPARSING = 0x80028000,
SIGDN_FILESYSPATH = 0x80058000,
SIGDN_NORMALDISPLAY = 0,
SIGDN_PARENTRELATIVE = 0x80080001,
SIGDN_PARENTRELATIVEEDITING = 0x80031001,
SIGDN_PARENTRELATIVEFORADDRESSBAR = 0x8007c001,
SIGDN_PARENTRELATIVEPARSING = 0x80018001,
SIGDN_URL = 0x80068000
}
[Flags]
private enum FOS
{
FOS_OVERWRITEPROMPT = 0x2,
FOS_STRICTFILETYPES = 0x4,
FOS_NOCHANGEDIR = 0x8,
FOS_PICKFOLDERS = 0x20,
FOS_FORCEFILESYSTEM = 0x40,
FOS_ALLNONSTORAGEITEMS = 0x80,
FOS_NOVALIDATE = 0x100,
FOS_ALLOWMULTISELECT = 0x200,
FOS_PATHMUSTEXIST = 0x800,
FOS_FILEMUSTEXIST = 0x1000,
FOS_CREATEPROMPT = 0x2000,
FOS_SHAREAWARE = 0x4000,
FOS_NOREADONLYRETURN = 0x8000,
FOS_NOTESTFILECREATE = 0x10000,
FOS_HIDEMRUPLACES = 0x20000,
FOS_HIDEPINNEDPLACES = 0x40000,
FOS_NODEREFERENCELINKS = 0x100000,
FOS_OKBUTTONNEEDSINTERACTION = 0x200000,
FOS_DONTADDTORECENT = 0x2000000,
FOS_FORCESHOWHIDDEN = 0x10000000,
FOS_DEFAULTNOMINIMODE = 0x20000000,
FOS_FORCEPREVIEWPANEON = 0x40000000,
FOS_SUPPORTSTREAMABLEITEMS = unchecked((int)0x80000000)
}
#pragma warning restore CA1712 // Do not prefix enum values with type name
}
结果:
if (dlg.ShowDialog(yourForm.Handle) == true)
或者if (dlg.ShowDialog(someControl.Handle) == true)
,否则用户可能会将焦点放在其中一个窗体/窗口上(即弹出浏览对话框的窗体之外的另一个窗口)。 - Maris B.使用OpenFileDialog
的一个巧妙解决方案是将ValidateNames
和CheckFileExists
都设置为false,然后给FileName
赋予一个模拟值以指示选择了目录。
我说这是巧妙的,因为它让用户对如何选择文件夹感到困惑。他们需要在所需文件夹中,然后只需按“打开”按钮,尽管文件名显示“选择文件夹”。
这是基于Denis Stankovski的同一对话框中选择文件或文件夹的文章。
OpenFileDialog folderBrowser = new OpenFileDialog();
// Set validate names and check file exists to false otherwise windows will
// not let you select "Folder Selection."
folderBrowser.ValidateNames = false;
folderBrowser.CheckFileExists = false;
folderBrowser.CheckPathExists = true;
// Always default to Folder Selection.
folderBrowser.FileName = "Folder Selection.";
if (folderBrowser.ShowDialog() == DialogResult.OK)
{
string folderPath = Path.GetDirectoryName(folderBrowser.FileName);
// ...
}
奇怪的是有这么多的回答/投票,但没有人将以下代码作为答案添加:
using (var opnDlg = new OpenFileDialog()) //ANY dialog
{
//opnDlg.Filter = "Png Files (*.png)|*.png";
//opnDlg.Filter = "Excel Files (*.xls, *.xlsx)|*.xls;*.xlsx|CSV Files (*.csv)|*.csv"
if (opnDlg.ShowDialog() == DialogResult.OK)
{
//opnDlg.SelectedPath -- your result
}
}
Application.VisualStyleState
,这种方法就行不通了:Application.VisualStyleState = System.Windows.Forms.VisualStyles.VisualStyleState.NoneEnabled;
。你会遇到 这个问题... - RotaxVisualStyleState
,是否存在有效的用例,也许可以调查是否有解决方法。您是否想在存储库中打开一个问题?https://github.com/augustoproiete/ookii-dialogs-winforms/issues - C. Augusto ProieteVisualStyleState
的原因是它使得在拥有大量控件的 WinForms 表单上的实际速度更快。请参阅我的 Stack Overflow 帖子这里。我很担心这个问题很深且长时间无法解决(如果有的话)... 我的解决方法是 FolderBrowserDialogEx,而且我希望尽快转换到 WPF。 - Rotax这里有另一种解决方案,所有源代码都在一个简单的ZIP文件中。
它使用附加的窗口标志来呈现OpenFileDialog,使其像Windows 7+文件夹选择对话框一样工作。
根据该网站,它是公共领域:“没有许可证,您可以自由地获取并使用代码。”
Archive.org链接:
Application.VisualStyleState
,这种方法就行不通了:Application.VisualStyleState = System.Windows.Forms.VisualStyles.VisualStyleState.NoneEnabled;
。你会遇到这个问题... - Rotax考虑到 OP 的问题,Simon Mourier 给出的答案应该是最好的答案。它不涉及 NuGET 包,因此在选择文件夹方法时将来不会有任何依赖性问题。
如果遇到与 "...is not available in C# 7.3" 相关的错误,请在您的 .csproj 中添加 <LangVersion>8.0</LangVersion>
(在使用 Visual Studio 进行构建和运行时不会产生任何错误)
如果您无法更改项目语言,则只需将 owner
??= Application.Current.MainWindow
替换为
owner = owner ?? Application.Current.MainWindow
ValidateNames
和CheckFileExists
均设置为 false,而FileName
则被赋予一个模拟值以表示选择了一个目录。我称其为“hack”,因为用户可能会对如何选择文件夹感到困惑。参见从同一对话框中选择文件或文件夹。 - Daniel Ballinger