【问题标题】:How do I use OpenFileDialog to select a folder?如何使用 OpenFileDialog 选择文件夹?
【发布时间】:2012-07-22 09:12:08
【问题描述】:

如何使用OpenFileDialog选择文件夹?

我打算使用以下项目:https://github.com/scottwis/OpenFileOrFolderDialog

但是,我遇到了一个问题。它使用GetOpenFileName 函数和OPENFILENAME 结构。而OPENFILENAME 有一个名为templateID 的成员。它是对话框模板的标识符。该项目还包含res1.rc 文件和模板化对话框初始化。但我不知道如何将此文件附加到我的 C# 项目中。

有没有更好的方法来使用OpenFileDialog 来选择文件夹?

【问题讨论】:

  • 如果你用编辑器打开项目文件,你会注意到底部的一些附加属性:.您将看到它运行 rc.exe 来编译资源文件 res1.rc(请务必将“resource.h”也复制到您的项目中)。确保你已经安装了 VisualC 并且 VCIncludePath 指向正确的位置(github 的指向 VC9.0 版本,你可能需要更改它)。编译 .rc 文件后,生成的 .res 文件将添加为使用 Win32Resource 指令的可执行文件的资源。
  • 有一个使用 OpenFileDialog 的 hackish 解决方案,其中 ValidateNamesCheckFileExists 都设置为 false,FileName 被赋予一个模拟值以指示选择了一个目录。我说 hack 是因为它让用户对如何选择文件夹感到困惑。见Select file or folder from the same dialog
  • 感谢 Dan 指向 OpenFileDialog-Hack!这比 FolderBrowserDialog 好得多,因为 OF​​D 显示书签文件夹等,所以每个人 - 特别是在大公司中 - 都会发现他们的废话。 FBD 在这些地方不会有多大好处。
  • @ComradeJoecool 我已将我的评论转换为answer。我试了几次,没有出现“找不到文件”的问题。您是否在重复使用同一个 OpenFileDialog 实例?
  • @DanielBallinger 啊,我发现了我的问题,因为我使用 Powershell 创建对话框,将 ValidateNamesCheckFileExists 设置为 false 不起作用,我需要将它们设置为 @ 987654337@(或者更好地学习powershell)

标签: c# .net winforms dialog openfiledialog


【解决方案1】:

这是一个纯 C# 版本,无 nuget,应该适用于所有版本的 .NET(包括 .NET Core、.NET 5、WPF、Winforms 等)并使用 Windows Vista(及更高版本)@987654321 @ 带有FOS_PICKFOLDERS 选项的接口,因此它具有漂亮的文件夹选择器 Windows 标准 UI。

我还添加了 WPF 的 Window 类型支持,但这是可选的,可以删除 WPF 标记的行。

用法:

var dlg = new FolderPicker();
dlg.InputPath = @"c:\windows\system32";
if (dlg.ShowDialog() == true)
{
    MessageBox.Show(dlg.ResultPath);
}

代码:

    using System;
    using System.Diagnostics;
    using System.Runtime.InteropServices;
    using System.Runtime.InteropServices.ComTypes;
    using System.Windows; // for WPF support
    using System.Windows.Interop;  // for WPF support

    public class FolderPicker
    {
        public virtual string ResultPath { get; protected set; }
        public virtual string ResultName { get; protected set; }
        public virtual string InputPath { get; set; }
        public virtual bool ForceFileSystem { 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;
            }
            return options;
        }

        // for WPF support
        public bool? ShowDialog(Window owner = null, bool throwOnError = false)
        {
            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.GetResult(out var result), throwOnError) != 0)
                return null;

            if (CheckHr(result.GetDisplayName(SIGDN.SIGDN_DESKTOPABSOLUTEPARSING, out var path), throwOnError) != 0)
                return null;

            ResultPath = path;

            if (CheckHr(result.GetDisplayName(SIGDN.SIGDN_DESKTOPABSOLUTEEDITING, out path), false) == 0)
            {
                ResultName = path;
            }
            return true;
        }

        private static int CheckHr(int hr, bool throwOnError)
        {
            if (hr != 0)
            {
                if (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("42f85136-db7e-439c-85f1-e4075d135fc8"), 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([MarshalAs(UnmanagedType.IUnknown)] out object 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
        }

        #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
    }

结果:

【讨论】:

  • 嗨,我目前正在尝试使上述内容在 Windows 10 Pro Build 21H1 OS Build 19043.1055 下的 Visual Studio、C# 2015、.Net v4.8.04084 中工作。这一行: if (CheckHr(SHCreateItemFromParsingName(InputPath, null, typeof(IShellItem).GUID, out var item), throwOnError) != 0) return null;例如返回 CS1003 C# 语法错误,',' 预期,任何帮助将不胜感激,在 c# 中变得很好,但是 - 还没有这么好..
  • @FalloutBoy - 如果您使用的是旧 C# 版本,请将 var 替换为 out 变量类型(例如在这种情况下为 IShellItem),甚至在调用之外声明它,而不是内联docs.microsoft.com/en-us/dotnet/csharp/whats-new/…
  • 谢谢西蒙,在阅读了您的 cmets 后,我最终得到了这个工作,而如果您取出 WPF 部分,您的代码确实在此版本下运行,它确实需要对子例程进行一些变量本地化 - 我想一想,后来的 c# 语法让我感到困惑,但现在一切都很好。
  • 显然是最佳答案!
  • 这太棒了!谢谢!
【解决方案2】:

奇怪这么多答案/投票,但没有人添加以下代码作为答案:

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
    }
}

【讨论】:

  • 你的回答和@Ionică Bizău Answer 有什么不同吗?
  • 逻辑是一样的,但我的答案要短得多,你不需要为 DialogResult 创建额外的变量。
  • 如前所述,这种方法存在一些大问题:这是一个糟糕的树形视图对话框!您不能复制粘贴路径,您必须从根文件夹中逐个钻取,并且没有收藏夹面板!
  • @S.Serpooshan 没关系。这只是一个使用 ANY 对话框的示例。如果需要,您可以使用任何其他对话框。 :)
  • @Andrew:Re.your 对“S.Serpooshan”的评论回复:是的,实际上它确实很重要。根据“Michael Paulukonis”对“2015 年 4 月 28 日 15:09”的评论(比您的回答早 1.5 年)对“Ionică Bizău”的回答:“这个问题具体地 “ (强调补充)。此外,“Joe”在“2017 年 1 月 6 日 18:03”(在您发表评论之前约 2.5 年)的回答已经确切地提供了 OP 的要求(即文件夹选择 w/all OFD 的功能不在 FBD 中)。
【解决方案3】:

作为对希望避免使用FolderBrowserDialog 的未来用户的说明,Microsoft 曾经发布了一个名为 WindowsAPICodePack 的 API,该 API 有一个名为 CommonOpenFileDialog 的有用对话框,可以设置为 IsFolderPicker 模式。该 API 可从 Microsoft 以NuGet package 的形式获得。

这就是我安装和使用CommonOpenFileDialog 所需的全部内容。 (NuGet 处理依赖关系)

Install-Package Microsoft.WindowsAPICodePack-Shell

对于包含行:

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);
}

【讨论】:

  • 我认为这是“如何使用 OpenFileDialog 选择文件夹?”的最接近的答案。 FolderBrowseDialog 非常不可用。谢谢你的回答。
  • 这应该是答案。并注意,您需要通过 NuGet 安装 Microsoft.WindowsAPICodePack.Shell 包,然后才能使用此 CommonOpenFileDialog
  • 如果你在 VS2017 中使用它,它会重新设置你的主窗口的样式。
  • 微软似乎已将其重新发布为WindowsAPICodePack-Shell
  • 链接的 NuGet 包在 .NET 5 中对我不起作用,因为它无法加载程序集。不过使用Microsoft-WindowsAPICodePack-Shell 确实有效。
【解决方案4】:

看看 Ookii Dialogs 库,它分别为 Windows 窗体和 WPF 实现了文件夹浏览器对话框。

Ookii.Dialogs.WinForms

https://github.com/augustoproiete/ookii-dialogs-winforms


Ookii.Dialogs.Wpf

https://github.com/augustoproiete/ookii-dialogs-wpf

【讨论】:

  • 好。注意:Ookii.Dialogs 需要 Microsoft .NET Framework 4.5 或更高版本。 (不能在WinXP中使用)
  • @S.Serpooshan -- 好吧,我想它也不能在我的 Windows 3.1 PC 上运行,对吧?但说真的,在 2018 年,无论如何都不应该考虑 Windows XP——它早已死去。
  • @rory.ap 实际上,这个解决方案对我来说的主要问题是它在浏览文件夹时不显示文件。当用户想要选择文件夹时,能够看到文件(例如要处理的图像)有时非常有用!
  • 不幸的是,如果Application.VisualStyleState 被禁用,这将不起作用:Application.VisualStyleState = System.Windows.Forms.VisualStyles.VisualStyleState.NoneEnabled;。你会遇到this...
  • @AugustoProiete 当然,禁用VisualStyleState 的原因是因为它使例如带有很多控件的 winforms 表单实际上“更快”。请参阅我的 SO 帖子here。恐怕这个问题很深,并且很长一段时间都无法解决(如果有的话)......我的解决方法是FolderBrowserDialogEx - 我希望尽快切换到 WPF。
【解决方案5】:

这应该是最明显和最直接的方式

using (var dialog = new System.Windows.Forms.FolderBrowserDialog())
{

   System.Windows.Forms.DialogResult result = dialog.ShowDialog();

   if(result == System.Windows.Forms.DialogResult.OK)
   {
      selectedFolder = dialog.SelectedPath;
   }

}

【讨论】:

  • FolderBrowserDialog 已在此处多次提供,这是错误的答案。这是一个过时且非常不友好的界面。它在下面使用SHBrowseForFolder,甚至文档状态“对于Windows Vista或更高版本,建议您使用IFileDialogFOS_PICKFOLDERS选项而不是SHBrowseForFolder函数。这使用打开文件选择文件夹模式下的对话框,是首选实现。”
【解决方案6】:

这是另一种解决方案,它在一个简单的 ZIP 文件中包含所有可用的源代码。

它为 OpenFileDialog 提供了额外的窗口标志,使其像 Windows 7+ 文件夹选择对话框一样工作。

根据网站,它是公共领域:“没有许可证,您可以随意获取和使用代码。”

Archive.org 链接:

【讨论】:

  • 完美运行!您也可以通过在“FolderSelectDialog.cs”中添加此行来使其选择多个文件夹: public string[] FileNames { get { return ofd.FileNames; } } 并更改 ofd.Multiselect = true;在构造函数中
  • 不幸的是,如果Application.VisualStyleState 被禁用,这将不起作用:Application.VisualStyleState = System.Windows.Forms.VisualStyles.VisualStyleState.NoneEnabled;。你会遇到this...
  • 不幸的是,我没有机会研究这个(并且不会有一段时间)但是从这里开始:medium.com/lextm/… 他们建议将 FileDialog.AutoUpgradeEnabled 设置为 false
【解决方案7】:

有一个使用OpenFileDialog 的hackish 解决方案,其中ValidateNamesCheckFileExists 都设置为false,FileName 被赋予一个模拟值以指示选择了一个目录。

我之所以说 hack,是因为它让用户对如何选择文件夹感到困惑。它们需要位于所需的文件夹中,然后在文件名显示“文件夹选择”时按打开。

这是基于 Denis Stankovski 的 Select file or folder from the same dialog

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);
    // ...
}

【讨论】:

  • 我明白了,是的,我已经让它工作了。一件令人讨厌的事情是Folder Selection. 将被添加到文件名的末尾,如下所示:C:\Folder Selection. 我猜你总是可以从字符串中删除这些字符。看起来仍然比 FolderBrowserDialog 好
  • 这对我不起作用。它不允许我选择文件夹。它只是打开它们。
  • @Lokiare 这就是我所说的黑客攻击的意思。请参阅第二段中的说明。
  • @ComradeJoecool 您不必手动删除它。这就是代码中最后一行的用途:string folderPath = Path.GetDirectoryName(folderBrowser.FileName);
  • 哦!此方法存在一个问题:如果用户在浏览文件夹时按下 UpBack 按钮,对话框的主 Open 按钮将无法按预期工作!它会导致您跳回上一个文件夹!但是当您只需双击文件夹以选择它们或在每个文件夹中选择一些文件时它就可以工作(如果有任何文件可以选择)
【解决方案8】:

基本上你需要 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 的引用。

您还必须为Directory 类添加using System.IO

【讨论】:

  • FolderBrowserDialog 缺乏可用性。主要缺点是它不允许您从 Windows 资源管理器复制文件夹路径以进行快速导航,当您需要向下钻取三个以上级别时,它就毫无用处。不希望深入每个文件夹,尤其是当存储速度较慢或其中一个级别有很多文件夹时。
  • 问题具体是关于使用 OpenFileDialog (OFD) 选择文件夹,而不是 FolderBrowserDialog (FBD)。从用户的角度来看,我同意 FBD 很糟糕。
  • 除了这个 UI 损坏的对话框之外,还可以使用 CommonOpenFileDialognew CommonOpenFileDialog { IsFolderPicker = true }
  • 永远不要使用它!我记得作为用户,我曾指责这些可怜的程序员用这个糟糕的树视图对话框 (它只是FolderBrowserDialog)制作了另一个应用程序。它完全无法使用:一堆根目录,缺少收藏夹面板,还有最可怕的——你甚至不能在那里粘贴路径!现在,作为一名程序员,我看到了使用它的建议……请不要这样做。
  • FolderBrowserDialog 除了其他用户所说的一切之外,还有一个很大的缺陷。它不记得最后选择的路径!
【解决方案9】:

在我看来,你只是在关注 FolderBrowserDialog

【讨论】:

  • 我猜这在 b/c 中被否决了(正如 mistika 已经指出的那样)FolderBrowserDialog 的可用性很差,OP 明确想要使用 OpenFileDialog。
  • @mbx 也许吧。公平地说,OP 并没有说“我不能使用除了 OpenFileDialog 之外的任何东西”。当我最初回答这个问题时(超过 4 年半前......),假设是 OP 只是不知道如何让用户打开文件夹。在发布此答案后,我实际上并没有回到这个问题,所以我没有看到任何关于可用性的讨论——我在回答时也没有考虑到它。
猜你喜欢
  • 2013-09-21
  • 1970-01-01
  • 1970-01-01
  • 2017-10-19
  • 1970-01-01
  • 2012-07-22
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多