如何使用OpenFileDialog选择文件夹?

Yun*_*Yun 254 .net c# dialog openfiledialog

如何用来OpenFileDialog选择文件夹?

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

但是,我遇到了一个问题.它使用GetOpenFileName功能和OPENFILENAME结构.并OPENFILENAME有成员的名字templateID.它是对话框模板的标识符.该项目也包含res1.rc文件和模板化对话框init.但我无法弄清楚如何将此文件附加到我的C#项目.

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

Ion*_*zău 372

基本上你需要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");
    }
}
Run Code Online (Sandbox Code Playgroud)

如果您在WPF中工作,则必须添加引用System.Windows.Forms.

你还必须using System.IODirectory课堂添加

  • FolderBrowserDialog缺乏可用性.主要缺点是它不允许您从Windows资源管理器中复制文件夹路径以进行快速导航,这使您在需要向下钻取三个以上级别时无用.不希望钻入每个文件夹,尤其是当存储速度很慢或者其中一个级别有很多文件夹时. (162认同)
  • 请,**永远不要使用它**!我记得作为一个用户我责怪这些可怜的程序员用这个糟糕的*树视图对话框**(这只是FolderBrowserDialog)*制作了另一个应用程序.它是完全无法使用的:一堆根目录,一个缺少的收藏夹面板,以及最可怕的 - 你甚至无法在那里粘贴一条路径!现在作为程序员,我看到了使用它的建议......请不要这样做. (91认同)
  • 对于具有损坏UI的此对话框,请使用**CommonOpenFileDialog**:`new CommonOpenFileDialog {IsFolderPicker = true}`. (20认同)
  • 问题特别是关于使用 OpenFileDialog (OFD) 选择文件夹,而不是 FolderBrowserDialog (FBD)。我同意 FBD 从用户的角度来看很糟糕。 (11认同)
  • 请注意,`FolderBrowserDialog` 在幕后使用 [`SHBrowseForFolder`](https://docs.microsoft.com/en-us/windows/win32/api/shlobj_core/nf-shlobj_core-shbrowseforfoldera)。此函数的文档特别指出,*“对于 Windows Vista 或更高版本,建议您使用带有“FOS_PICKFOLDERS”选项的“IFileDialog”,而不是“SHBrowseForFolder”函数。这在选择文件夹模式下使用“打开文件”对话框,并且是首选实现。”* 因此,除了所有可用性问题之外,自 Vista(2006 年推出)以来,它一直不是推荐的解决方案! (5认同)
  • 在 .NET Core 3 中,“FolderBrowserDialog”将具有与 WindowsAPICodePack 中的 UI 相同的 UI。https://docs.microsoft.com/en-us/dotnet/core/porting/winforms-writing-changes (4认同)
  • 除了其他用户所说的其他一切之外,FolderBrowserDialog还有一个很大的缺陷.它不记得最后选择的路径! (2认同)

小智 200

作为希望避免使用的未来用户的注释FolderBrowserDialog,Microsoft曾经发布了一个名为WindowsAPICodePack的API,该API具有一个有用的对话框CommonOpenFileDialog,可以设置为一种IsFolderPicker模式.

如果您可以访问nuget包,它仍然可以在公共列表中使用 CommonOpenFileDialog

对于包含行:

Install-Package WindowsAPICodePack-Shell
Run Code Online (Sandbox Code Playgroud)

用法:

using Microsoft.WindowsAPICodePack.Dialogs;
Run Code Online (Sandbox Code Playgroud)

注意:官方Microsoft Nuget Package现已上市

  • Microsoft.WindowsAPICodePack核心
  • Microsoft.WindowsAPICodePack壳牌

每个软件包都由Microsoft提供,并将同时安装.
项目网址:http://code.msdn.microsoft.com/WindowsAPICodePack.

NuGet用户aybe上传了一系列非官方软件包.大多数人似乎都相信他是一个可靠的来源.

这就是我需要安装才能使用的FolderBrowserDialog.(nuget处理了依赖项)

CommonOpenFileDialog dialog = new CommonOpenFileDialog();
dialog.InitialDirectory = "C:\\Users";
dialog.IsFolderPicker = true;
if (dialog.ShowDialog() == CommonFileDialogResult.Ok)
{
    MessageBox.Show("You selected: " + dialog.FileName);
}
Run Code Online (Sandbox Code Playgroud)

  • 这应该是答案.请注意,在使用此`CommonOpenFileDialog`之前,需要通过NuGet安装`Microsoft.WindowsAPICodePack.Shell`包. (34认同)
  • 我认为这是"如何使用OpenFileDialog选择文件夹?"的最接近的答案.FolderBrowseDialog非常不可用.谢谢你的回答. (24认同)
  • 微软似乎已将其重新发布为“WindowsAPICodePack-Shell” (16认同)
  • 如果你在 VS2017 中使用它,它会重新设计你的主窗口。 (2认同)
  • 链接的 NuGet 包在 .NET 5 中对我不起作用,因为它无法加载程序集。使用 [Microsoft-WindowsAPICodePack-Shell](https://www.nuget.org/packages/Microsoft-WindowsAPICodePack-Shell/1.1.4?_src=template) 确实有效。 (2认同)

Dan*_*ger 22

有一个hackish解决方案使用OpenFileDialogwhere ValidateNamesCheckFileExists都设置为false并FileName给出一个模拟值来指示选择了一个目录.

我说黑客是因为它让用户对如何选择文件夹感到困惑.它们需要位于所需的文件夹中,然后只需按下"打开",文件名称为"文件夹选择".

C#文件夹选择对话框

这是基于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);
    // ...
}
Run Code Online (Sandbox Code Playgroud)

  • 这对我不起作用。它不允许我选择文件夹。它只是打开它们。 (2认同)
  • @Lokiare 这就是我说这是黑客攻击时的意思。请参阅第二段中的说明。 (2认同)
  • @ComradeJoecool 您不必手动手动删除它。这就是代码中的最后一行: string folderPath = Path.GetDirectoryName(folderBrowser.FileName); (2认同)
  • 哦!此方法存在一个问题:如果用户在浏览文件夹时按 **向上** 或 **后退** 按钮,对话框的主“打开”按钮将无法按预期工作!它会导致您跳回上一个文件夹!但当您只需双击文件夹来选择它们或选择每个文件夹中的一些文件(如果有任何文件可以选择)时,它就可以工作 (2认同)

And*_*rew 20

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

using (var fldrDlg = new FolderBrowserDialog())
{ 
    //fldrDlg.Filter = "Png Files (*.png)|*.png";
    //fldrDlg.Filter = "Excel Files (*.xls, *.xlsx)|*.xls;*.xlsx|CSV Files (*.csv)|*.csv"

    if (fldrDlg.ShowDialog() == DialogResult.OK)
    {
        //fldrDlg.SelectedPath -- your result
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 正如已经提到的,这种方法存在一些大问题:这是一个糟糕的树视图对话框!您无法将路径复制粘贴到其中,您必须从根文件夹中逐一钻取,并且没有收藏夹面板! (7认同)
  • 你的答案和@IonicăBizău之间有什么区别[答案](/sf/ask/813700891/#11624322)? (5认同)
  • @Andrew:Re.your Comment 回复“S.Serpooshan”:是的,实际上它*确实*重要。根据“Michael Paulukonis”对“2015 年 4 月 28 日 15:09”(大约在您回答之前 1.5 年)对“Ionică Bizău”的回答的评论:“问题是*特别*关于使用 *OpenFileDialog ( OFD)* 选择文件夹,而不是FolderBrowserDialog (FBD)。我同意从用户的角度来看 FBD 很糟糕。” (强调已添加)。另外,“Joe”在“2017 年 1 月 6 日 18:03”上的回答(在您发表评论之前约 2.5 年)已经*准确地*提供了 OP 所要求的内容(即文件夹选择,具有 OFD 的所有功能,而不是在 FBD 中)。 (4认同)
  • 逻辑是相同的,但我的答案要短得多,您无需为 DialogResult 创建额外的变量。 (2认同)

Sim*_*ead 11

听起来就像你刚刚在FolderBrowserDialog之后.

  • 我猜这是越来越低的b/c(正如mistika已经指出的那样)FolderBrowserDialog具有可怕的可用性,OP显然想要使用OpenFileDialog. (3认同)
  • @mbx也许吧.公平地说,OP并没有说"除了OpenFileDialog之外我不能使用任何东西".当我最初回答这个问题时(超过4年半前......),假设OP只是不知道如何让用户打开文件夹.在发布这个答案之后我实际上没有回到这个问题所以我没有看到任何关于可用性的讨论 - 在回答时我也没有考虑过. (2认同)

Sim*_*ier 11

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

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

用法:

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

代码:

    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
    }
Run Code Online (Sandbox Code Playgroud)

结果:

在此处输入图片说明

  • 显然是最佳答案! (10认同)
  • 这非常出色!谢谢你! (5认同)
  • @LesFerch,它应该类似于“if (dlg.ShowDialog(yourForm.Handle) == true)”或“if (dlg.ShowDialog(someControl.Handle) == true)”,否则用户可能会将注意力集中在以下之一表单/窗口(显示浏览对话框的其他窗口) (3认同)
  • 简直难以置信!您是否已经准备好这段代码,或者您是否真的在 10 分钟内完成了这些增强?如果是后者,你的功夫最令人印象深刻。我花了更长的时间来集成和测试它。顺便说一句,它工作完美!谢谢你! (2认同)

Ben*_*ene 8

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

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

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

Archive.org 链接:


Cai*_*ete 5

看一下Ookii Dialogs库,它具有分别针对Windows Forms和WPF的文件夹浏览器对话框的实现。

在此处输入图片说明

Ookii.Dialogs.WinForms

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


Ookii.Dialogs.Wpf

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

  • @S.Serpooshan——我猜它也不能在我的 Windows 3.1 PC 上运行,对吧?但说实话,在 2018 年,无论如何都不应该有人考虑 Windows XP——它早已消亡了。 (5认同)

归档时间:

查看次数:

473410 次

最近记录:

5 年,11 月 前