Programing

폴더를 선택하도록 OpenFileDialog를 어떻게 구성합니까?

lottogame 2020. 4. 7. 08:16
반응형

폴더를 선택하도록 OpenFileDialog를 어떻게 구성합니까?


VS .NET에서는 프로젝트 폴더를 선택할 때 OpenFileDialog 또는 SaveFileDialog처럼 보이는 대화 상자가 표시되지만 폴더 만 허용하도록 설정되어 있습니다. 이걸 본 이후로 어떻게되는지 알고 싶었습니다. FolderBrowserDialog에 대해 알고 있지만 실제로는이 대화 상자를 좋아하지 않았습니다. 너무 작게 시작하여 경로를 입력 할 수 있다는 이점을 얻지 못했습니다.

지금까지는 .NET 에서이 작업을 수행 할 수있는 방법이 거의 없다고 확신하지만 관리되지 않는 코드에서도 어떻게하는지 궁금합니다. 대화 상자를 처음부터 완전히 다시 구현하지 못하는 경우이 동작을 갖도록 대화 상자를 어떻게 수정합니까?

또한 FolderBrowserDialog에 대해 알고 있지만 때로는이 방법으로 대화 상자를 구성하는 방법에 대해 정말로 궁금한 것 외에도 사용하고 싶지 않습니다. FolderBrowserDialog 만 사용하면 일관된 UI 환경을 유지하는 데 도움이되지만 호기심을 만족시키지 않으므로 답변으로 계산되지 않습니다.

비스타 고유의 것도 아닙니다. VS .NET 2003 이후 로이 대화 상자를 보았으므로 Win2k 및 WinXP에서 가능합니다. 이것은 "이 작업을 수행하는 올바른 방법을 알고 싶습니다"라는 질문이 아니라 "VS 2003에서 처음 사용하고 싶었 기 때문에 이것에 대해 궁금했습니다"라는 질문에 더 가깝습니다. Vista의 파일 대화 상자 에이 작업을 수행 할 수있는 옵션이 있음을 이해하지만 XP에서 작동하므로 작동하도록 하기 위해 무언가했다는 것을 알고 있습니다. Vista는 질문과 관련하여 존재하지 않기 때문에 Vista 관련 답변은 답변이 아닙니다.

업데이트 :이 작업 샘플을 제공하기 때문에 나는 스콧 위스 니 스키의 답변을 받아들이는거야,하지만 난 방탄복 대화 문화를 가리키는에 대한 크레딧을받을 권리가 생각 (.NET에서 인정 하듯이 불쾌한 인 있지만 않습니다 MS를 알아내는 그것을 위해 일을)와 마크 랜섬 아마도이 작업에 대한 사용자 정의 대화 상자를 굴 렀을 것입니다.


폴더 나 파일을 열 수있는 OpenFileOrFolder 대화 상자라는 대화 상자가 있습니다.

AcceptFiles 값을 false로 설정하면 수락 폴더 모드에서만 작동합니다.

GitHub에서 소스를 다운로드 할 수 있습니다.


Windows API 코드 팩이 있습니다. 네임 스페이스 CommonOpenFileDialog클래스를 포함하여 많은 셸 관련 항목이 Microsoft.WindowsAPICodePack.Dialogs있습니다. 이것은 완벽한 솔루션입니다-폴더 만 표시되는 일반적인 열린 대화 상자.

사용 방법의 예는 다음과 같습니다.

CommonOpenFileDialog cofd = new CommonOpenFileDialog();
cofd.IsFolderPicker = true;
cofd.ShowDialog();

불행히도 Microsoft는 더 이상이 패키지를 제공하지 않지만 몇몇 사람들은 바이너리를 공식적으로 NuGet에 업로드했습니다. 여기 에서 하나의 예를 찾을 수 있습니다 . 이 패키지는 단지 쉘 전용입니다. 필요한 경우 동일한 사용자에게 원래 패키지에 더 많은 기능을 제공하는 다른 여러 패키지가 있습니다.


내장 된 FolderBrowserDialog의 재사용 가능한 파생물 인 FolderBrowserDialogEx 를 사용할 수 있습니다 . 이를 통해 경로, 심지어 UNC 경로를 입력 할 수 있습니다. 컴퓨터 나 프린터를 찾아 볼 수도 있습니다. 내장 FBD와 동일하게 작동하지만 ... 더 좋습니다.

(편집 :이 대화 상자에서 파일이나 폴더를 선택하도록 설정할 수 있음을 지적해야합니다.)

전체 소스 코드 (한 개의 짧은 C # 모듈) 비어 있는. MS- 공개 라이센스.

그것을 사용하는 코드 :

var dlg1 = new Ionic.Utils.FolderBrowserDialogEx();
dlg1.Description = "Select a folder to extract to:";
dlg1.ShowNewFolderButton = true;
dlg1.ShowEditBox = true;
//dlg1.NewStyle = false;
dlg1.SelectedPath = txtExtractDirectory.Text;
dlg1.ShowFullPathInEditBox = true;
dlg1.RootFolder = System.Environment.SpecialFolder.MyComputer;

// Show the FolderBrowserDialog.
DialogResult result = dlg1.ShowDialog();
if (result == DialogResult.OK)
{
    txtExtractDirectory.Text = dlg1.SelectedPath;
}

Ookii.Dialogs의 패키지는 브라우저 대화 폴더에 새로운 (비스타 스타일) 주위에 관리되는 래퍼가 포함되어 있습니다. 또한 오래된 운영 체제에서도 정상적으로 저하됩니다.


이를 위해 FolderBrowserDialog를 사용하는 것이 좋습니다.

using (FolderBrowserDialog dlg = new FolderBrowserDialog())
{
    dlg.Description = "Select a folder";
    if (dlg.ShowDialog() == DialogResult.OK)
    {
        MessageBox.Show("You selected: " + dlg.SelectedPath);
    }
}

몇 시간 동안 검색 한 후 leetNightShade작동하는 솔루션에 대한 답변찾았 습니다 .

이 솔루션을 다른 솔루션보다 훨씬 우수하게 생각하는 세 가지가 있습니다.

  1. 사용하기 간단합니다. 프로젝트에 두 개의 파일 만 포함하면됩니다 (어쨌든 하나로 결합 할 수 있음).
  2. XP 또는 이전 시스템에서 사용되는 경우 표준 FolderBrowserDialog로 돌아갑니다 .
  3. 저자는 귀하가 적합하다고 생각하는 목적으로 코드를 사용할 권한을 부여합니다.

    코드를 자유롭게 사용하고 수행 할 수있는 라이센스는 없습니다.

여기 에서 코드를 다운로드 하십시오 .


정확한 오디오 복사 는 Windows XP에서이 방식으로 작동합니다. 표준 파일 열기 대화 상자가 표시되지만 파일 이름 필드에 "파일 이름은 무시됩니다"라는 텍스트가 포함됩니다.

여기서 추측하지만, 대화 상자가 크게 변경 될 때마다 문자열이 콤보 상자 편집 컨트롤에 주입되는 것 같습니다. 필드가 비어 있지 않고 대화 상자 플래그가 파일의 존재를 확인하지 않도록 설정되어 있으면 대화 상자를 정상적으로 닫을 수 있습니다.

편집 : 이것은 생각보다 훨씬 쉽습니다. C ++ / MFC의 코드는 다음과 같습니다. 원하는 환경으로 변환 할 수 있습니다.

CFileDialog dlg(true, NULL, "Filename will be ignored", OFN_HIDEREADONLY | OFN_NOVALIDATE | OFN_PATHMUSTEXIST | OFN_READONLY, NULL, this);
dlg.DoModal();

편집 2 : 이것은 C #으로의 번역이어야하지만 C #에 능숙하지 않으므로 작동하지 않으면 나를 쏘지 마십시오.

OpenFileDialog openFileDialog1 = new OpenFileDialog();

openFileDialog1.FileName = "Filename will be ignored";
openFileDialog1.CheckPathExists = true;
openFileDialog1.ShowReadOnly = false;
openFileDialog1.ReadOnlyChecked = true;
openFileDialog1.CheckFileExists = false;
openFileDialog1.ValidateNames = false;

if(openFileDialog1.ShowDialog() == DialogResult.OK)
{
    // openFileDialog1.FileName should contain the folder and a dummy filename
}

편집 3 : 마지막으로 Visual Studio 2005에서 실제 대화 상자를 살펴 보았습니다 (이전에 액세스 할 수 없었습니다). 표준 파일 열기 대화 상자가 아닙니다! Spy ++에서 창을 검사하여 열려있는 표준 파일과 비교하면 구조와 클래스 이름이 일치하지 않는 것을 볼 수 있습니다. 자세히 살펴보면 대화 상자 내용 사이에 약간의 차이가 있음을 알 수 있습니다. 내 결론은 Microsoft가 Visual Studio의 표준 대화 상자를 완전히 대체 하여이 기능을 제공한다는 것입니다. 내 솔루션이나 이와 비슷한 것이 처음부터 자신의 코드를 기꺼이 코딩하지 않는 한 가능한 한 가깝습니다.


Spy ++ 또는 Winspector로 조금만 재생하면 VS 프로젝트 위치의 폴더 텍스트 상자가 표준 대화 상자의 사용자 정의임을 알 수 있습니다. 메모장과 같은 표준 파일 대화 상자의 파일 이름 텍스트 상자와 동일한 필드가 아닙니다.

거기에서 VS는 파일 이름과 파일 유형 텍스트 상자 / 콤보 상자를 숨기고 사용자 정의 대화 상자 템플릿을 사용하여 대화 상자 아래쪽에 자체 부분을 추가합니다.

편집 : 다음은 이러한 사용자 정의의 예와 수행 방법입니다 (.NET이 아닌 Win32에서).

m_ofn은 파일 대화 상자의 기본이되는 OPENFILENAME 구조체입니다. 이 두 줄을 추가하십시오 :

  m_ofn.lpTemplateName = MAKEINTRESOURCE(IDD_FILEDIALOG_IMPORTXLIFF);
  m_ofn.Flags |= OFN_ENABLETEMPLATE;

여기서 IDD_FILEDIALOG_IMPORTXLIFF는 대화 상자의 맨 아래에 추가 될 사용자 정의 대화 상자 템플리트입니다. 아래의 빨간색 부분을 참조하십시오. (출처 : apptranslator.com )대체 텍스트

이 경우 사용자 정의 된 부분은 레이블 + 하이퍼 링크 일 뿐이며 모든 대화 상자 일 수 있습니다. 폴더 만 선택을 확인할 수있는 확인 버튼이 포함될 수 있습니다.

그러나 대화 상자의 표준 부분에서 일부 컨트롤을 제거하는 방법을 모르겠습니다.

이 MSDN 기사 에서 자세히 설명 합니다.


파일 대화 상자를 서브 클래 싱하고 모든 컨트롤에 액세스 할 수 있습니다. 각각에는 창 핸들을 얻는 데 사용할 수있는 식별자가 있습니다. 그런 다음 표시 및 숨기기, 선택 변경 등에 대한 메시지를 표시 할 수 있습니다. 모두 얼마나 많은 노력을 기울이고 있는지에 달려 있습니다.

우리는 WTL 클래스 지원을 사용하여 사용자 지정 장소 표시 줄 및 플러그인 COM보기를 포함하도록 파일 대화 상자를 사용자 정의했습니다.

MSDN은 이 Win32에서 사용하는 방법에 대한 정보 제공 이 CodeProject의 기사가 예를 포함을 하고, 이 CodeProject의 기사는 .NET 예제를 제공합니다 .


이런 코드를 사용할 수 있습니다

  • 필터는 파일을 숨 깁니다
  • 파일 이름은 첫 번째 텍스트 숨기기

파일 이름에 대한 텍스트 상자를 숨기려면 OpenFileDialogEx 를 봐야합니다.

코드:

{
    openFileDialog2.FileName = "\r";
    openFileDialog1.Filter = "folders|*.neverseenthisfile";
    openFileDialog1.CheckFileExists = false;
    openFileDialog1.CheckPathExists = false;
}

VS2008을 사용하는 Vista를 사용한다고 가정합니까? 이 경우 Vista 파일 대화 상자 IFileDialog를 호출 할 때 FOS_PICKFOLDERS 옵션 이 사용되고 있다고 생각합니다 . .NET 코드에서 작동하려면 많은 P / Invoke interop 코드가 필요합니다.


첫 번째 해결책

나는 이것을 lyquidity.com의 Bill Seddon이 정리 한 .NET Win 7 스타일 폴더 선택 대화 상자 버전으로 개발했습니다 (나는 제휴 관계가 없습니다). ( 이 페이지의 다른 답변에서 그의 코드에 대해 배웠습니다 ). 그의 솔루션에는이 집중된 목적에 필요하지 않은 추가 Reflection 클래스가 필요하고 예외 기반 흐름 제어를 사용하며 리플렉션 호출의 결과를 캐시하지 않기 때문에 직접 작성했습니다. 중첩 정적 클래스는 메서드가 호출되지 않은 경우 정적 리플렉션 변수가 채워지지 않도록합니다 . Windows 버전이 충분하지 않으면 Vista 이전 대화 상자로 돌아갑니다. Windows 7, 8, 9, 10 이상 (이론적으로)에서 작동합니다.VistaDialogShow

using System;
using System.Reflection;
using System.Windows.Forms;

namespace ErikE.Shuriken {
    /// <summary>
    /// Present the Windows Vista-style open file dialog to select a folder. Fall back for older Windows Versions
    /// </summary>
    public class FolderSelectDialog {
        private string _initialDirectory;
        private string _title;
        private string _fileName = "";

        public string InitialDirectory {
            get { return string.IsNullOrEmpty(_initialDirectory) ? Environment.CurrentDirectory : _initialDirectory; }
            set { _initialDirectory = value; }
        }
        public string Title {
            get { return _title ?? "Select a folder"; }
            set { _title = value; }
        }
        public string FileName { get { return _fileName; } }

        public bool Show() { return Show(IntPtr.Zero); }

        /// <param name="hWndOwner">Handle of the control or window to be the parent of the file dialog</param>
        /// <returns>true if the user clicks OK</returns>
        public bool Show(IntPtr hWndOwner) {
            var result = Environment.OSVersion.Version.Major >= 6
                ? VistaDialog.Show(hWndOwner, InitialDirectory, Title)
                : ShowXpDialog(hWndOwner, InitialDirectory, Title);
            _fileName = result.FileName;
            return result.Result;
        }

        private struct ShowDialogResult {
            public bool Result { get; set; }
            public string FileName { get; set; }
        }

        private static ShowDialogResult ShowXpDialog(IntPtr ownerHandle, string initialDirectory, string title) {
            var folderBrowserDialog = new FolderBrowserDialog {
                Description = title,
                SelectedPath = initialDirectory,
                ShowNewFolderButton = false
            };
            var dialogResult = new ShowDialogResult();
            if (folderBrowserDialog.ShowDialog(new WindowWrapper(ownerHandle)) == DialogResult.OK) {
                dialogResult.Result = true;
                dialogResult.FileName = folderBrowserDialog.SelectedPath;
            }
            return dialogResult;
        }

        private static class VistaDialog {
            private const string c_foldersFilter = "Folders|\n";

            private const BindingFlags c_flags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic;
            private readonly static Assembly s_windowsFormsAssembly = typeof(FileDialog).Assembly;
            private readonly static Type s_iFileDialogType = s_windowsFormsAssembly.GetType("System.Windows.Forms.FileDialogNative+IFileDialog");
            private readonly static MethodInfo s_createVistaDialogMethodInfo = typeof(OpenFileDialog).GetMethod("CreateVistaDialog", c_flags);
            private readonly static MethodInfo s_onBeforeVistaDialogMethodInfo = typeof(OpenFileDialog).GetMethod("OnBeforeVistaDialog", c_flags);
            private readonly static MethodInfo s_getOptionsMethodInfo = typeof(FileDialog).GetMethod("GetOptions", c_flags);
            private readonly static MethodInfo s_setOptionsMethodInfo = s_iFileDialogType.GetMethod("SetOptions", c_flags);
            private readonly static uint s_fosPickFoldersBitFlag = (uint) s_windowsFormsAssembly
                .GetType("System.Windows.Forms.FileDialogNative+FOS")
                .GetField("FOS_PICKFOLDERS")
                .GetValue(null);
            private readonly static ConstructorInfo s_vistaDialogEventsConstructorInfo = s_windowsFormsAssembly
                .GetType("System.Windows.Forms.FileDialog+VistaDialogEvents")
                .GetConstructor(c_flags, null, new[] { typeof(FileDialog) }, null);
            private readonly static MethodInfo s_adviseMethodInfo = s_iFileDialogType.GetMethod("Advise");
            private readonly static MethodInfo s_unAdviseMethodInfo = s_iFileDialogType.GetMethod("Unadvise");
            private readonly static MethodInfo s_showMethodInfo = s_iFileDialogType.GetMethod("Show");

            public static ShowDialogResult Show(IntPtr ownerHandle, string initialDirectory, string title) {
                var openFileDialog = new OpenFileDialog {
                    AddExtension = false,
                    CheckFileExists = false,
                    DereferenceLinks = true,
                    Filter = c_foldersFilter,
                    InitialDirectory = initialDirectory,
                    Multiselect = false,
                    Title = title
                };

                var iFileDialog = s_createVistaDialogMethodInfo.Invoke(openFileDialog, new object[] { });
                s_onBeforeVistaDialogMethodInfo.Invoke(openFileDialog, new[] { iFileDialog });
                s_setOptionsMethodInfo.Invoke(iFileDialog, new object[] { (uint) s_getOptionsMethodInfo.Invoke(openFileDialog, new object[] { }) | s_fosPickFoldersBitFlag });
                var adviseParametersWithOutputConnectionToken = new[] { s_vistaDialogEventsConstructorInfo.Invoke(new object[] { openFileDialog }), 0U };
                s_adviseMethodInfo.Invoke(iFileDialog, adviseParametersWithOutputConnectionToken);

                try {
                    int retVal = (int) s_showMethodInfo.Invoke(iFileDialog, new object[] { ownerHandle });
                    return new ShowDialogResult {
                        Result = retVal == 0,
                        FileName = openFileDialog.FileName
                    };
                }
                finally {
                    s_unAdviseMethodInfo.Invoke(iFileDialog, new[] { adviseParametersWithOutputConnectionToken[1] });
                }
            }
        }

        // Wrap an IWin32Window around an IntPtr
        private class WindowWrapper : IWin32Window {
            private readonly IntPtr _handle;
            public WindowWrapper(IntPtr handle) { _handle = handle; }
            public IntPtr Handle { get { return _handle; } }
        }
    }
}

Windows Form에서 다음과 같이 사용됩니다.

var dialog = new FolderSelectDialog {
    InitialDirectory = musicFolderTextBox.Text,
    Title = "Select a folder to import music from"
};
if (dialog.Show(Handle)) {
    musicFolderTextBox.Text = dialog.FileName;
}

물론 옵션과 노출되는 속성을 가지고 놀 수 있습니다. 예를 들어 Vista 스타일 대화 상자에서 다중 선택을 허용합니다.

두번째 해결책

Simon Mourier는 Windows API에 대해 interop을 사용하여 동일한 작업을 직접 수행하는 방법을 보여주는 답변 을 제공했지만 이전 버전의 Windows에서는 이전 스타일 대화 상자를 사용하려면 해당 버전을 보완해야합니다. 불행히도 솔루션을 완성했을 때 아직 그의 게시물을 찾지 못했습니다. 독의 이름을 지정하십시오!


Codeproject (신용에서 Nitron으로) 에서 이것을 사용해보십시오 .

대화와 동일한 대화라고 생각합니다. 스크린 샷을 추가하면 도움이 될까요?

bool GetFolder(std::string& folderpath, const char* szCaption=NULL, HWND hOwner=NULL)
{
    bool retVal = false;

    // The BROWSEINFO struct tells the shell how it should display the dialog.
    BROWSEINFO bi;
    memset(&bi, 0, sizeof(bi));

    bi.ulFlags   = BIF_USENEWUI;
    bi.hwndOwner = hOwner;
    bi.lpszTitle = szCaption;

    // must call this if using BIF_USENEWUI
    ::OleInitialize(NULL);

    // Show the dialog and get the itemIDList for the selected folder.
    LPITEMIDLIST pIDL = ::SHBrowseForFolder(&bi);

    if(pIDL != NULL)
    {
        // Create a buffer to store the path, then get the path.
        char buffer[_MAX_PATH] = {'\0'};
        if(::SHGetPathFromIDList(pIDL, buffer) != 0)
        {
            // Set the string value.
            folderpath = buffer;
            retVal = true;
        }       

        // free the item id list
        CoTaskMemFree(pIDL);
    }

    ::OleUninitialize();

    return retVal;
}

비스타에서 당신은 사용할 수 있습니다 IFileDialog을 FOS_PICKFOLDERS 옵션 세트. 그러면 폴더를 선택할 수있는 OpenFileDialog와 같은 창이 표시됩니다.

var frm = (IFileDialog)(new FileOpenDialogRCW());
uint options;
frm.GetOptions(out options);
options |= FOS_PICKFOLDERS;
frm.SetOptions(options);

if (frm.Show(owner.Handle) == S_OK) {
    IShellItem shellItem;
    frm.GetResult(out shellItem);
    IntPtr pszString;
    shellItem.GetDisplayName(SIGDN_FILESYSPATH, out pszString);
    this.Folder = Marshal.PtrToStringAuto(pszString);
}

이전 Windows의 경우 항상 폴더에서 파일을 선택하여 속일 수 있습니다.

.NET Framework 2.0 이상에서 작동하는 실제 예제는 여기 에서 찾을 수 있습니다 .


이런 코드를 사용할 수 있습니다

필터가 빈 문자열입니다. 파일 이름은 AnyName이지만 비어 있지는 않습니다.

        openFileDialog.FileName = "AnyFile";
        openFileDialog.Filter = string.Empty;
        openFileDialog.CheckFileExists = false;
        openFileDialog.CheckPathExists = false;

WPF에 대한 Ookii 대화 상자 라이브러리는 WPF의 폴더 브라우저 대화의 구현을 제공하는 클래스가 있습니다.

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

여기에 이미지 설명을 입력하십시오

Windows Forms 와 호환되는 버전도 있습니다 .


나는 질문이 구성되어 있다는 것을 알고 OpenFileDialog있지만 Google이 나를 여기로 데려 갔다는 것을 알 수 있습니다. 폴더 만 찾고 FolderBrowserDialog있다면 아래의 또 다른 SO 질문에 대한 답변 대신 대신 을 사용해야한다고 지적 할 수 있습니다

vb.net에서 열린 파일 대화 상자를 사용하여 경로를 지정하는 방법은 무엇입니까?

참고 URL : https://stackoverflow.com/questions/31059/how-do-you-configure-an-openfiledialog-to-select-folders

반응형